diff options
| author | Dave Martin <dave.martin@linaro.org> | 2011-05-23 07:22:10 -0400 |
|---|---|---|
| committer | Russell King <rmk+kernel@arm.linux.org.uk> | 2011-05-26 05:31:06 -0400 |
| commit | dc2eb928a1bcf6a48f40c1f2ff21b66bdbf91a3c (patch) | |
| tree | 3f877569ca481ad79378ed0e5932006434a1cc1e | |
| parent | 4db70f73e56961b9bcdfd0c36c62847a18b7dbb5 (diff) | |
ARM: 6938/1: fiq: Refactor {get,set}_fiq_regs() for Thumb-2
* To remove the risk of inconvenient register allocation decisions
by the compiler, these functions are separated out as pure
assembler.
* The apcs frame manipulation code is not applicable for Thumb-2
(and also not easily compatible). Since it's not essential to
have a full frame on these leaf assembler functions, the frame
manipulation is removed, in the interests of simplicity.
* Split up ldm/stm instructions to be compatible with Thumb-2,
as well as avoiding instruction forms deprecated on >= ARMv7.
Signed-off-by: Dave Martin <dave.martin@linaro.org>
Reviewed-by: Nicolas Pitre <nicolas.pitre@linaro.org>
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
| -rw-r--r-- | arch/arm/include/asm/fiq.h | 16 | ||||
| -rw-r--r-- | arch/arm/kernel/Makefile | 2 | ||||
| -rw-r--r-- | arch/arm/kernel/fiq.c | 45 | ||||
| -rw-r--r-- | arch/arm/kernel/fiqasm.S | 49 |
4 files changed, 66 insertions, 46 deletions
diff --git a/arch/arm/include/asm/fiq.h b/arch/arm/include/asm/fiq.h index 2242ce22ec6c..45bb3eeeab7c 100644 --- a/arch/arm/include/asm/fiq.h +++ b/arch/arm/include/asm/fiq.h | |||
| @@ -29,9 +29,21 @@ struct fiq_handler { | |||
| 29 | extern int claim_fiq(struct fiq_handler *f); | 29 | extern int claim_fiq(struct fiq_handler *f); |
| 30 | extern void release_fiq(struct fiq_handler *f); | 30 | extern void release_fiq(struct fiq_handler *f); |
| 31 | extern void set_fiq_handler(void *start, unsigned int length); | 31 | extern void set_fiq_handler(void *start, unsigned int length); |
| 32 | extern void set_fiq_regs(struct pt_regs *regs); | ||
| 33 | extern void get_fiq_regs(struct pt_regs *regs); | ||
| 34 | extern void enable_fiq(int fiq); | 32 | extern void enable_fiq(int fiq); |
| 35 | extern void disable_fiq(int fiq); | 33 | extern void disable_fiq(int fiq); |
| 36 | 34 | ||
| 35 | /* helpers defined in fiqasm.S: */ | ||
| 36 | extern void __set_fiq_regs(unsigned long const *regs); | ||
| 37 | extern void __get_fiq_regs(unsigned long *regs); | ||
| 38 | |||
| 39 | static inline void set_fiq_regs(struct pt_regs const *regs) | ||
| 40 | { | ||
| 41 | __set_fiq_regs(®s->ARM_r8); | ||
| 42 | } | ||
| 43 | |||
| 44 | static inline void get_fiq_regs(struct pt_regs *regs) | ||
| 45 | { | ||
| 46 | __get_fiq_regs(®s->ARM_r8); | ||
| 47 | } | ||
| 48 | |||
| 37 | #endif | 49 | #endif |
diff --git a/arch/arm/kernel/Makefile b/arch/arm/kernel/Makefile index 8d95446150a3..01c0292eea8b 100644 --- a/arch/arm/kernel/Makefile +++ b/arch/arm/kernel/Makefile | |||
| @@ -24,7 +24,7 @@ obj-$(CONFIG_OC_ETM) += etm.o | |||
| 24 | 24 | ||
| 25 | obj-$(CONFIG_ISA_DMA_API) += dma.o | 25 | obj-$(CONFIG_ISA_DMA_API) += dma.o |
| 26 | obj-$(CONFIG_ARCH_ACORN) += ecard.o | 26 | obj-$(CONFIG_ARCH_ACORN) += ecard.o |
| 27 | obj-$(CONFIG_FIQ) += fiq.o | 27 | obj-$(CONFIG_FIQ) += fiq.o fiqasm.o |
| 28 | obj-$(CONFIG_MODULES) += armksyms.o module.o | 28 | obj-$(CONFIG_MODULES) += armksyms.o module.o |
| 29 | obj-$(CONFIG_ARTHUR) += arthur.o | 29 | obj-$(CONFIG_ARTHUR) += arthur.o |
| 30 | obj-$(CONFIG_ISA_DMA) += dma-isa.o | 30 | obj-$(CONFIG_ISA_DMA) += dma-isa.o |
diff --git a/arch/arm/kernel/fiq.c b/arch/arm/kernel/fiq.c index e72dc34eea1c..4c164ece5891 100644 --- a/arch/arm/kernel/fiq.c +++ b/arch/arm/kernel/fiq.c | |||
| @@ -89,47 +89,6 @@ void set_fiq_handler(void *start, unsigned int length) | |||
| 89 | flush_icache_range(0x1c, 0x1c + length); | 89 | flush_icache_range(0x1c, 0x1c + length); |
| 90 | } | 90 | } |
| 91 | 91 | ||
| 92 | /* | ||
| 93 | * Taking an interrupt in FIQ mode is death, so both these functions | ||
| 94 | * disable irqs for the duration. Note - these functions are almost | ||
| 95 | * entirely coded in assembly. | ||
| 96 | */ | ||
| 97 | void __naked set_fiq_regs(struct pt_regs *regs) | ||
| 98 | { | ||
| 99 | register unsigned long tmp; | ||
| 100 | asm volatile ( | ||
| 101 | "mov ip, sp\n\ | ||
| 102 | stmfd sp!, {fp, ip, lr, pc}\n\ | ||
| 103 | sub fp, ip, #4\n\ | ||
| 104 | mrs %0, cpsr\n\ | ||
| 105 | msr cpsr_c, %2 @ select FIQ mode\n\ | ||
| 106 | mov r0, r0\n\ | ||
| 107 | ldmia %1, {r8 - r14}\n\ | ||
| 108 | msr cpsr_c, %0 @ return to SVC mode\n\ | ||
| 109 | mov r0, r0\n\ | ||
| 110 | ldmfd sp, {fp, sp, pc}" | ||
| 111 | : "=&r" (tmp) | ||
| 112 | : "r" (®s->ARM_r8), "I" (PSR_I_BIT | PSR_F_BIT | FIQ_MODE)); | ||
| 113 | } | ||
| 114 | |||
| 115 | void __naked get_fiq_regs(struct pt_regs *regs) | ||
| 116 | { | ||
| 117 | register unsigned long tmp; | ||
| 118 | asm volatile ( | ||
| 119 | "mov ip, sp\n\ | ||
| 120 | stmfd sp!, {fp, ip, lr, pc}\n\ | ||
| 121 | sub fp, ip, #4\n\ | ||
| 122 | mrs %0, cpsr\n\ | ||
| 123 | msr cpsr_c, %2 @ select FIQ mode\n\ | ||
| 124 | mov r0, r0\n\ | ||
| 125 | stmia %1, {r8 - r14}\n\ | ||
| 126 | msr cpsr_c, %0 @ return to SVC mode\n\ | ||
| 127 | mov r0, r0\n\ | ||
| 128 | ldmfd sp, {fp, sp, pc}" | ||
| 129 | : "=&r" (tmp) | ||
| 130 | : "r" (®s->ARM_r8), "I" (PSR_I_BIT | PSR_F_BIT | FIQ_MODE)); | ||
| 131 | } | ||
| 132 | |||
| 133 | int claim_fiq(struct fiq_handler *f) | 92 | int claim_fiq(struct fiq_handler *f) |
| 134 | { | 93 | { |
| 135 | int ret = 0; | 94 | int ret = 0; |
| @@ -174,8 +133,8 @@ void disable_fiq(int fiq) | |||
| 174 | } | 133 | } |
| 175 | 134 | ||
| 176 | EXPORT_SYMBOL(set_fiq_handler); | 135 | EXPORT_SYMBOL(set_fiq_handler); |
| 177 | EXPORT_SYMBOL(set_fiq_regs); | 136 | EXPORT_SYMBOL(__set_fiq_regs); /* defined in fiqasm.S */ |
| 178 | EXPORT_SYMBOL(get_fiq_regs); | 137 | EXPORT_SYMBOL(__get_fiq_regs); /* defined in fiqasm.S */ |
| 179 | EXPORT_SYMBOL(claim_fiq); | 138 | EXPORT_SYMBOL(claim_fiq); |
| 180 | EXPORT_SYMBOL(release_fiq); | 139 | EXPORT_SYMBOL(release_fiq); |
| 181 | EXPORT_SYMBOL(enable_fiq); | 140 | EXPORT_SYMBOL(enable_fiq); |
diff --git a/arch/arm/kernel/fiqasm.S b/arch/arm/kernel/fiqasm.S new file mode 100644 index 000000000000..207f9d652010 --- /dev/null +++ b/arch/arm/kernel/fiqasm.S | |||
| @@ -0,0 +1,49 @@ | |||
| 1 | /* | ||
| 2 | * linux/arch/arm/kernel/fiqasm.S | ||
| 3 | * | ||
| 4 | * Derived from code originally in linux/arch/arm/kernel/fiq.c: | ||
| 5 | * | ||
| 6 | * Copyright (C) 1998 Russell King | ||
| 7 | * Copyright (C) 1998, 1999 Phil Blundell | ||
| 8 | * Copyright (C) 2011, Linaro Limited | ||
| 9 | * | ||
| 10 | * FIQ support written by Philip Blundell <philb@gnu.org>, 1998. | ||
| 11 | * | ||
| 12 | * FIQ support re-written by Russell King to be more generic | ||
| 13 | * | ||
| 14 | * v7/Thumb-2 compatibility modifications by Linaro Limited, 2011. | ||
| 15 | */ | ||
| 16 | |||
| 17 | #include <linux/linkage.h> | ||
| 18 | #include <asm/assembler.h> | ||
| 19 | |||
| 20 | /* | ||
| 21 | * Taking an interrupt in FIQ mode is death, so both these functions | ||
| 22 | * disable irqs for the duration. | ||
| 23 | */ | ||
| 24 | |||
| 25 | ENTRY(__set_fiq_regs) | ||
| 26 | mov r2, #PSR_I_BIT | PSR_F_BIT | FIQ_MODE | ||
| 27 | mrs r1, cpsr | ||
| 28 | msr cpsr_c, r2 @ select FIQ mode | ||
| 29 | mov r0, r0 @ avoid hazard prior to ARMv4 | ||
| 30 | ldmia r0!, {r8 - r12} | ||
| 31 | ldr sp, [r0], #4 | ||
| 32 | ldr lr, [r0] | ||
| 33 | msr cpsr_c, r1 @ return to SVC mode | ||
| 34 | mov r0, r0 @ avoid hazard prior to ARMv4 | ||
| 35 | mov pc, lr | ||
| 36 | ENDPROC(__set_fiq_regs) | ||
| 37 | |||
| 38 | ENTRY(__get_fiq_regs) | ||
| 39 | mov r2, #PSR_I_BIT | PSR_F_BIT | FIQ_MODE | ||
| 40 | mrs r1, cpsr | ||
| 41 | msr cpsr_c, r2 @ select FIQ mode | ||
| 42 | mov r0, r0 @ avoid hazard prior to ARMv4 | ||
| 43 | stmia r0!, {r8 - r12} | ||
| 44 | str sp, [r0], #4 | ||
| 45 | str lr, [r0] | ||
| 46 | msr cpsr_c, r1 @ return to SVC mode | ||
| 47 | mov r0, r0 @ avoid hazard prior to ARMv4 | ||
| 48 | mov pc, lr | ||
| 49 | ENDPROC(__get_fiq_regs) | ||
