aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/kernel/entry-armv.S
diff options
context:
space:
mode:
authorRussell King <rmk+kernel@arm.linux.org.uk>2011-06-25 10:44:20 -0400
committerRussell King <rmk+kernel@arm.linux.org.uk>2011-06-30 06:04:59 -0400
commitb059bdc39321696fe8f344acb7117d57fbd7b475 (patch)
tree3a5001287ea0e16339e5cc633f1248470377f967 /arch/arm/kernel/entry-armv.S
parentfbab1c809467efe001194ab8bb17f0f451a17f97 (diff)
ARM: entry: rejig register allocation in exception entry handlers
This allows us to avoid moving registers twice to work around the clobbered registers when we add calls to trace_hardirqs_{on,off}. Ensure that all SVC handlers return with SPSR in r5 for consistency. Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Diffstat (limited to 'arch/arm/kernel/entry-armv.S')
-rw-r--r--arch/arm/kernel/entry-armv.S81
1 files changed, 43 insertions, 38 deletions
diff --git a/arch/arm/kernel/entry-armv.S b/arch/arm/kernel/entry-armv.S
index fd42e667a81b..582bb2310446 100644
--- a/arch/arm/kernel/entry-armv.S
+++ b/arch/arm/kernel/entry-armv.S
@@ -45,7 +45,7 @@
45 .endm 45 .endm
46 46
47 .macro pabt_helper 47 .macro pabt_helper
48 mov r0, r2 @ pass address of aborted instruction. 48 mov r0, r4 @ pass address of aborted instruction.
49#ifdef MULTI_PABORT 49#ifdef MULTI_PABORT
50 ldr ip, .LCprocfns 50 ldr ip, .LCprocfns
51 mov lr, pc 51 mov lr, pc
@@ -56,6 +56,8 @@
56 .endm 56 .endm
57 57
58 .macro dabt_helper 58 .macro dabt_helper
59 mov r2, r4
60 mov r3, r5
59 61
60 @ 62 @
61 @ Call the processor-specific abort handler: 63 @ Call the processor-specific abort handler:
@@ -157,26 +159,26 @@ ENDPROC(__und_invalid)
157 SPFIX( subeq sp, sp, #4 ) 159 SPFIX( subeq sp, sp, #4 )
158 stmia sp, {r1 - r12} 160 stmia sp, {r1 - r12}
159 161
160 ldmia r0, {r1 - r3} 162 ldmia r0, {r3 - r5}
161 add r5, sp, #S_SP - 4 @ here for interlock avoidance 163 add r7, sp, #S_SP - 4 @ here for interlock avoidance
162 mov r4, #-1 @ "" "" "" "" 164 mov r6, #-1 @ "" "" "" ""
163 add r0, sp, #(S_FRAME_SIZE + \stack_hole - 4) 165 add r2, sp, #(S_FRAME_SIZE + \stack_hole - 4)
164 SPFIX( addeq r0, r0, #4 ) 166 SPFIX( addeq r2, r2, #4 )
165 str r1, [sp, #-4]! @ save the "real" r0 copied 167 str r3, [sp, #-4]! @ save the "real" r0 copied
166 @ from the exception stack 168 @ from the exception stack
167 169
168 mov r1, lr 170 mov r3, lr
169 171
170 @ 172 @
171 @ We are now ready to fill in the remaining blanks on the stack: 173 @ We are now ready to fill in the remaining blanks on the stack:
172 @ 174 @
173 @ r0 - sp_svc 175 @ r2 - sp_svc
174 @ r1 - lr_svc 176 @ r3 - lr_svc
175 @ r2 - lr_<exception>, already fixed up for correct return/restart 177 @ r4 - lr_<exception>, already fixed up for correct return/restart
176 @ r3 - spsr_<exception> 178 @ r5 - spsr_<exception>
177 @ r4 - orig_r0 (see pt_regs definition in ptrace.h) 179 @ r6 - orig_r0 (see pt_regs definition in ptrace.h)
178 @ 180 @
179 stmia r5, {r0 - r4} 181 stmia r7, {r2 - r6}
180 .endm 182 .endm
181 183
182 .align 5 184 .align 5
@@ -187,7 +189,7 @@ __dabt_svc:
187 @ get ready to re-enable interrupts if appropriate 189 @ get ready to re-enable interrupts if appropriate
188 @ 190 @
189 mrs r9, cpsr 191 mrs r9, cpsr
190 tst r3, #PSR_I_BIT 192 tst r5, #PSR_I_BIT
191 biceq r9, r9, #PSR_I_BIT 193 biceq r9, r9, #PSR_I_BIT
192 194
193 dabt_helper 195 dabt_helper
@@ -208,8 +210,8 @@ __dabt_svc:
208 @ 210 @
209 @ restore SPSR and restart the instruction 211 @ restore SPSR and restart the instruction
210 @ 212 @
211 ldr r2, [sp, #S_PSR] 213 ldr r5, [sp, #S_PSR]
212 svc_exit r2 @ return from exception 214 svc_exit r5 @ return from exception
213 UNWIND(.fnend ) 215 UNWIND(.fnend )
214ENDPROC(__dabt_svc) 216ENDPROC(__dabt_svc)
215 217
@@ -232,13 +234,13 @@ __irq_svc:
232 tst r0, #_TIF_NEED_RESCHED 234 tst r0, #_TIF_NEED_RESCHED
233 blne svc_preempt 235 blne svc_preempt
234#endif 236#endif
235 ldr r4, [sp, #S_PSR] @ irqs are already disabled 237 ldr r5, [sp, #S_PSR]
236#ifdef CONFIG_TRACE_IRQFLAGS 238#ifdef CONFIG_TRACE_IRQFLAGS
237 @ The parent context IRQs must have been enabled to get here in 239 @ The parent context IRQs must have been enabled to get here in
238 @ the first place, so there's no point checking the PSR I bit. 240 @ the first place, so there's no point checking the PSR I bit.
239 bl trace_hardirqs_on 241 bl trace_hardirqs_on
240#endif 242#endif
241 svc_exit r4 @ return from exception 243 svc_exit r5 @ return from exception
242 UNWIND(.fnend ) 244 UNWIND(.fnend )
243ENDPROC(__irq_svc) 245ENDPROC(__irq_svc)
244 246
@@ -273,15 +275,16 @@ __und_svc:
273 @ r0 - instruction 275 @ r0 - instruction
274 @ 276 @
275#ifndef CONFIG_THUMB2_KERNEL 277#ifndef CONFIG_THUMB2_KERNEL
276 ldr r0, [r2, #-4] 278 ldr r0, [r4, #-4]
277#else 279#else
278 ldrh r0, [r2, #-2] @ Thumb instruction at LR - 2 280 ldrh r0, [r4, #-2] @ Thumb instruction at LR - 2
279 and r9, r0, #0xf800 281 and r9, r0, #0xf800
280 cmp r9, #0xe800 @ 32-bit instruction if xx >= 0 282 cmp r9, #0xe800 @ 32-bit instruction if xx >= 0
281 ldrhhs r9, [r2] @ bottom 16 bits 283 ldrhhs r9, [r4] @ bottom 16 bits
282 orrhs r0, r9, r0, lsl #16 284 orrhs r0, r9, r0, lsl #16
283#endif 285#endif
284 adr r9, BSYM(1f) 286 adr r9, BSYM(1f)
287 mov r2, r4
285 bl call_fpe 288 bl call_fpe
286 289
287 mov r0, sp @ struct pt_regs *regs 290 mov r0, sp @ struct pt_regs *regs
@@ -295,8 +298,8 @@ __und_svc:
295 @ 298 @
296 @ restore SPSR and restart the instruction 299 @ restore SPSR and restart the instruction
297 @ 300 @
298 ldr r2, [sp, #S_PSR] @ Get SVC cpsr 301 ldr r5, [sp, #S_PSR] @ Get SVC cpsr
299 svc_exit r2 @ return from exception 302 svc_exit r5 @ return from exception
300 UNWIND(.fnend ) 303 UNWIND(.fnend )
301ENDPROC(__und_svc) 304ENDPROC(__und_svc)
302 305
@@ -308,7 +311,7 @@ __pabt_svc:
308 @ re-enable interrupts if appropriate 311 @ re-enable interrupts if appropriate
309 @ 312 @
310 mrs r9, cpsr 313 mrs r9, cpsr
311 tst r3, #PSR_I_BIT 314 tst r5, #PSR_I_BIT
312 biceq r9, r9, #PSR_I_BIT 315 biceq r9, r9, #PSR_I_BIT
313 316
314 pabt_helper 317 pabt_helper
@@ -325,8 +328,8 @@ __pabt_svc:
325 @ 328 @
326 @ restore SPSR and restart the instruction 329 @ restore SPSR and restart the instruction
327 @ 330 @
328 ldr r2, [sp, #S_PSR] 331 ldr r5, [sp, #S_PSR]
329 svc_exit r2 @ return from exception 332 svc_exit r5 @ return from exception
330 UNWIND(.fnend ) 333 UNWIND(.fnend )
331ENDPROC(__pabt_svc) 334ENDPROC(__pabt_svc)
332 335
@@ -357,23 +360,23 @@ ENDPROC(__pabt_svc)
357 ARM( stmib sp, {r1 - r12} ) 360 ARM( stmib sp, {r1 - r12} )
358 THUMB( stmia sp, {r0 - r12} ) 361 THUMB( stmia sp, {r0 - r12} )
359 362
360 ldmia r0, {r1 - r3} 363 ldmia r0, {r3 - r5}
361 add r0, sp, #S_PC @ here for interlock avoidance 364 add r0, sp, #S_PC @ here for interlock avoidance
362 mov r4, #-1 @ "" "" "" "" 365 mov r6, #-1 @ "" "" "" ""
363 366
364 str r1, [sp] @ save the "real" r0 copied 367 str r3, [sp] @ save the "real" r0 copied
365 @ from the exception stack 368 @ from the exception stack
366 369
367 @ 370 @
368 @ We are now ready to fill in the remaining blanks on the stack: 371 @ We are now ready to fill in the remaining blanks on the stack:
369 @ 372 @
370 @ r2 - lr_<exception>, already fixed up for correct return/restart 373 @ r4 - lr_<exception>, already fixed up for correct return/restart
371 @ r3 - spsr_<exception> 374 @ r5 - spsr_<exception>
372 @ r4 - orig_r0 (see pt_regs definition in ptrace.h) 375 @ r6 - orig_r0 (see pt_regs definition in ptrace.h)
373 @ 376 @
374 @ Also, separately save sp_usr and lr_usr 377 @ Also, separately save sp_usr and lr_usr
375 @ 378 @
376 stmia r0, {r2 - r4} 379 stmia r0, {r4 - r6}
377 ARM( stmdb r0, {sp, lr}^ ) 380 ARM( stmdb r0, {sp, lr}^ )
378 THUMB( store_user_sp_lr r0, r1, S_SP - S_PC ) 381 THUMB( store_user_sp_lr r0, r1, S_SP - S_PC )
379 382
@@ -397,7 +400,7 @@ ENDPROC(__pabt_svc)
397 @ if it was interrupted in a critical region. Here we 400 @ if it was interrupted in a critical region. Here we
398 @ perform a quick test inline since it should be false 401 @ perform a quick test inline since it should be false
399 @ 99.9999% of the time. The rest is done out of line. 402 @ 99.9999% of the time. The rest is done out of line.
400 cmp r2, #TASK_SIZE 403 cmp r4, #TASK_SIZE
401 blhs kuser_cmpxchg_fixup 404 blhs kuser_cmpxchg_fixup
402#endif 405#endif
403#endif 406#endif
@@ -441,6 +444,8 @@ ENDPROC(__irq_usr)
441 .align 5 444 .align 5
442__und_usr: 445__und_usr:
443 usr_entry 446 usr_entry
447 mov r2, r4
448 mov r3, r5
444 449
445 @ 450 @
446 @ fall through to the emulation code, which returns using r9 if 451 @ fall through to the emulation code, which returns using r9 if
@@ -894,13 +899,13 @@ __kuser_cmpxchg: @ 0xffff0fc0
894 .text 899 .text
895kuser_cmpxchg_fixup: 900kuser_cmpxchg_fixup:
896 @ Called from kuser_cmpxchg_check macro. 901 @ Called from kuser_cmpxchg_check macro.
897 @ r2 = address of interrupted insn (must be preserved). 902 @ r4 = address of interrupted insn (must be preserved).
898 @ sp = saved regs. r7 and r8 are clobbered. 903 @ sp = saved regs. r7 and r8 are clobbered.
899 @ 1b = first critical insn, 2b = last critical insn. 904 @ 1b = first critical insn, 2b = last critical insn.
900 @ If r2 >= 1b and r2 <= 2b then saved pc_usr is set to 1b. 905 @ If r4 >= 1b and r4 <= 2b then saved pc_usr is set to 1b.
901 mov r7, #0xffff0fff 906 mov r7, #0xffff0fff
902 sub r7, r7, #(0xffff0fff - (0xffff0fc0 + (1b - __kuser_cmpxchg))) 907 sub r7, r7, #(0xffff0fff - (0xffff0fc0 + (1b - __kuser_cmpxchg)))
903 subs r8, r2, r7 908 subs r8, r4, r7
904 rsbcss r8, r8, #(2b - 1b) 909 rsbcss r8, r8, #(2b - 1b)
905 strcs r7, [sp, #S_PC] 910 strcs r7, [sp, #S_PC]
906 mov pc, lr 911 mov pc, lr