aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/kernel
diff options
context:
space:
mode:
authorCatalin Marinas <catalin.marinas@arm.com>2009-07-24 07:32:54 -0400
committerCatalin Marinas <catalin.marinas@arm.com>2009-07-24 07:32:54 -0400
commitb86040a59feb255a8193173caa4d5199464433d5 (patch)
tree89c07450eabc2abb88bb4d6e32d61fd3855f000e /arch/arm/kernel
parent0becb088501886f37ade38762c8eaaf4263572cc (diff)
Thumb-2: Implementation of the unified start-up and exceptions code
This patch implements the ARM/Thumb-2 unified kernel start-up and exception handling code. Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
Diffstat (limited to 'arch/arm/kernel')
-rw-r--r--arch/arm/kernel/entry-armv.S165
-rw-r--r--arch/arm/kernel/entry-common.S28
-rw-r--r--arch/arm/kernel/entry-header.S92
-rw-r--r--arch/arm/kernel/head-common.S13
-rw-r--r--arch/arm/kernel/head-nommu.S11
-rw-r--r--arch/arm/kernel/head.S28
-rw-r--r--arch/arm/kernel/process.c2
-rw-r--r--arch/arm/kernel/setup.c28
-rw-r--r--arch/arm/kernel/unwind.c4
9 files changed, 251 insertions, 120 deletions
diff --git a/arch/arm/kernel/entry-armv.S b/arch/arm/kernel/entry-armv.S
index 0befd1cabf45..468425f937dd 100644
--- a/arch/arm/kernel/entry-armv.S
+++ b/arch/arm/kernel/entry-armv.S
@@ -34,7 +34,7 @@
34 @ 34 @
35 @ routine called with r0 = irq number, r1 = struct pt_regs * 35 @ routine called with r0 = irq number, r1 = struct pt_regs *
36 @ 36 @
37 adrne lr, 1b 37 adrne lr, BSYM(1b)
38 bne asm_do_IRQ 38 bne asm_do_IRQ
39 39
40#ifdef CONFIG_SMP 40#ifdef CONFIG_SMP
@@ -46,13 +46,13 @@
46 */ 46 */
47 test_for_ipi r0, r6, r5, lr 47 test_for_ipi r0, r6, r5, lr
48 movne r0, sp 48 movne r0, sp
49 adrne lr, 1b 49 adrne lr, BSYM(1b)
50 bne do_IPI 50 bne do_IPI
51 51
52#ifdef CONFIG_LOCAL_TIMERS 52#ifdef CONFIG_LOCAL_TIMERS
53 test_for_ltirq r0, r6, r5, lr 53 test_for_ltirq r0, r6, r5, lr
54 movne r0, sp 54 movne r0, sp
55 adrne lr, 1b 55 adrne lr, BSYM(1b)
56 bne do_local_timer 56 bne do_local_timer
57#endif 57#endif
58#endif 58#endif
@@ -70,7 +70,10 @@
70 */ 70 */
71 .macro inv_entry, reason 71 .macro inv_entry, reason
72 sub sp, sp, #S_FRAME_SIZE 72 sub sp, sp, #S_FRAME_SIZE
73 stmib sp, {r1 - lr} 73 ARM( stmib sp, {r1 - lr} )
74 THUMB( stmia sp, {r0 - r12} )
75 THUMB( str sp, [sp, #S_SP] )
76 THUMB( str lr, [sp, #S_LR] )
74 mov r1, #\reason 77 mov r1, #\reason
75 .endm 78 .endm
76 79
@@ -126,17 +129,24 @@ ENDPROC(__und_invalid)
126 .macro svc_entry, stack_hole=0 129 .macro svc_entry, stack_hole=0
127 UNWIND(.fnstart ) 130 UNWIND(.fnstart )
128 UNWIND(.save {r0 - pc} ) 131 UNWIND(.save {r0 - pc} )
129 sub sp, sp, #(S_FRAME_SIZE + \stack_hole) 132 sub sp, sp, #(S_FRAME_SIZE + \stack_hole - 4)
133#ifdef CONFIG_THUMB2_KERNEL
134 SPFIX( str r0, [sp] ) @ temporarily saved
135 SPFIX( mov r0, sp )
136 SPFIX( tst r0, #4 ) @ test original stack alignment
137 SPFIX( ldr r0, [sp] ) @ restored
138#else
130 SPFIX( tst sp, #4 ) 139 SPFIX( tst sp, #4 )
131 SPFIX( bicne sp, sp, #4 ) 140#endif
132 stmib sp, {r1 - r12} 141 SPFIX( subeq sp, sp, #4 )
142 stmia sp, {r1 - r12}
133 143
134 ldmia r0, {r1 - r3} 144 ldmia r0, {r1 - r3}
135 add r5, sp, #S_SP @ here for interlock avoidance 145 add r5, sp, #S_SP - 4 @ here for interlock avoidance
136 mov r4, #-1 @ "" "" "" "" 146 mov r4, #-1 @ "" "" "" ""
137 add r0, sp, #(S_FRAME_SIZE + \stack_hole) 147 add r0, sp, #(S_FRAME_SIZE + \stack_hole - 4)
138 SPFIX( addne r0, r0, #4 ) 148 SPFIX( addeq r0, r0, #4 )
139 str r1, [sp] @ save the "real" r0 copied 149 str r1, [sp, #-4]! @ save the "real" r0 copied
140 @ from the exception stack 150 @ from the exception stack
141 151
142 mov r1, lr 152 mov r1, lr
@@ -196,9 +206,8 @@ __dabt_svc:
196 @ 206 @
197 @ restore SPSR and restart the instruction 207 @ restore SPSR and restart the instruction
198 @ 208 @
199 ldr r0, [sp, #S_PSR] 209 ldr r2, [sp, #S_PSR]
200 msr spsr_cxsf, r0 210 svc_exit r2 @ return from exception
201 ldmia sp, {r0 - pc}^ @ load r0 - pc, cpsr
202 UNWIND(.fnend ) 211 UNWIND(.fnend )
203ENDPROC(__dabt_svc) 212ENDPROC(__dabt_svc)
204 213
@@ -225,13 +234,12 @@ __irq_svc:
225 tst r0, #_TIF_NEED_RESCHED 234 tst r0, #_TIF_NEED_RESCHED
226 blne svc_preempt 235 blne svc_preempt
227#endif 236#endif
228 ldr r0, [sp, #S_PSR] @ irqs are already disabled 237 ldr r4, [sp, #S_PSR] @ irqs are already disabled
229 msr spsr_cxsf, r0
230#ifdef CONFIG_TRACE_IRQFLAGS 238#ifdef CONFIG_TRACE_IRQFLAGS
231 tst r0, #PSR_I_BIT 239 tst r4, #PSR_I_BIT
232 bleq trace_hardirqs_on 240 bleq trace_hardirqs_on
233#endif 241#endif
234 ldmia sp, {r0 - pc}^ @ load r0 - pc, cpsr 242 svc_exit r4 @ return from exception
235 UNWIND(.fnend ) 243 UNWIND(.fnend )
236ENDPROC(__irq_svc) 244ENDPROC(__irq_svc)
237 245
@@ -266,7 +274,7 @@ __und_svc:
266 @ r0 - instruction 274 @ r0 - instruction
267 @ 275 @
268 ldr r0, [r2, #-4] 276 ldr r0, [r2, #-4]
269 adr r9, 1f 277 adr r9, BSYM(1f)
270 bl call_fpe 278 bl call_fpe
271 279
272 mov r0, sp @ struct pt_regs *regs 280 mov r0, sp @ struct pt_regs *regs
@@ -280,9 +288,8 @@ __und_svc:
280 @ 288 @
281 @ restore SPSR and restart the instruction 289 @ restore SPSR and restart the instruction
282 @ 290 @
283 ldr lr, [sp, #S_PSR] @ Get SVC cpsr 291 ldr r2, [sp, #S_PSR] @ Get SVC cpsr
284 msr spsr_cxsf, lr 292 svc_exit r2 @ return from exception
285 ldmia sp, {r0 - pc}^ @ Restore SVC registers
286 UNWIND(.fnend ) 293 UNWIND(.fnend )
287ENDPROC(__und_svc) 294ENDPROC(__und_svc)
288 295
@@ -323,9 +330,8 @@ __pabt_svc:
323 @ 330 @
324 @ restore SPSR and restart the instruction 331 @ restore SPSR and restart the instruction
325 @ 332 @
326 ldr r0, [sp, #S_PSR] 333 ldr r2, [sp, #S_PSR]
327 msr spsr_cxsf, r0 334 svc_exit r2 @ return from exception
328 ldmia sp, {r0 - pc}^ @ load r0 - pc, cpsr
329 UNWIND(.fnend ) 335 UNWIND(.fnend )
330ENDPROC(__pabt_svc) 336ENDPROC(__pabt_svc)
331 337
@@ -353,7 +359,8 @@ ENDPROC(__pabt_svc)
353 UNWIND(.fnstart ) 359 UNWIND(.fnstart )
354 UNWIND(.cantunwind ) @ don't unwind the user space 360 UNWIND(.cantunwind ) @ don't unwind the user space
355 sub sp, sp, #S_FRAME_SIZE 361 sub sp, sp, #S_FRAME_SIZE
356 stmib sp, {r1 - r12} 362 ARM( stmib sp, {r1 - r12} )
363 THUMB( stmia sp, {r0 - r12} )
357 364
358 ldmia r0, {r1 - r3} 365 ldmia r0, {r1 - r3}
359 add r0, sp, #S_PC @ here for interlock avoidance 366 add r0, sp, #S_PC @ here for interlock avoidance
@@ -372,7 +379,8 @@ ENDPROC(__pabt_svc)
372 @ Also, separately save sp_usr and lr_usr 379 @ Also, separately save sp_usr and lr_usr
373 @ 380 @
374 stmia r0, {r2 - r4} 381 stmia r0, {r2 - r4}
375 stmdb r0, {sp, lr}^ 382 ARM( stmdb r0, {sp, lr}^ )
383 THUMB( store_user_sp_lr r0, r1, S_SP - S_PC )
376 384
377 @ 385 @
378 @ Enable the alignment trap while in kernel mode 386 @ Enable the alignment trap while in kernel mode
@@ -427,7 +435,7 @@ __dabt_usr:
427 @ 435 @
428 enable_irq 436 enable_irq
429 mov r2, sp 437 mov r2, sp
430 adr lr, ret_from_exception 438 adr lr, BSYM(ret_from_exception)
431 b do_DataAbort 439 b do_DataAbort
432 UNWIND(.fnend ) 440 UNWIND(.fnend )
433ENDPROC(__dabt_usr) 441ENDPROC(__dabt_usr)
@@ -452,7 +460,9 @@ __irq_usr:
452 ldr r0, [tsk, #TI_PREEMPT] 460 ldr r0, [tsk, #TI_PREEMPT]
453 str r8, [tsk, #TI_PREEMPT] 461 str r8, [tsk, #TI_PREEMPT]
454 teq r0, r7 462 teq r0, r7
455 strne r0, [r0, -r0] 463 ARM( strne r0, [r0, -r0] )
464 THUMB( movne r0, #0 )
465 THUMB( strne r0, [r0] )
456#endif 466#endif
457#ifdef CONFIG_TRACE_IRQFLAGS 467#ifdef CONFIG_TRACE_IRQFLAGS
458 bl trace_hardirqs_on 468 bl trace_hardirqs_on
@@ -476,9 +486,10 @@ __und_usr:
476 @ 486 @
477 @ r0 - instruction 487 @ r0 - instruction
478 @ 488 @
479 adr r9, ret_from_exception 489 adr r9, BSYM(ret_from_exception)
480 adr lr, __und_usr_unknown 490 adr lr, BSYM(__und_usr_unknown)
481 tst r3, #PSR_T_BIT @ Thumb mode? 491 tst r3, #PSR_T_BIT @ Thumb mode?
492 itet eq @ explicit IT needed for the 1f label
482 subeq r4, r2, #4 @ ARM instr at LR - 4 493 subeq r4, r2, #4 @ ARM instr at LR - 4
483 subne r4, r2, #2 @ Thumb instr at LR - 2 494 subne r4, r2, #2 @ Thumb instr at LR - 2
4841: ldreqt r0, [r4] 4951: ldreqt r0, [r4]
@@ -488,7 +499,10 @@ __und_usr:
488 beq call_fpe 499 beq call_fpe
489 @ Thumb instruction 500 @ Thumb instruction
490#if __LINUX_ARM_ARCH__ >= 7 501#if __LINUX_ARM_ARCH__ >= 7
4912: ldrht r5, [r4], #2 5022:
503 ARM( ldrht r5, [r4], #2 )
504 THUMB( ldrht r5, [r4] )
505 THUMB( add r4, r4, #2 )
492 and r0, r5, #0xf800 @ mask bits 111x x... .... .... 506 and r0, r5, #0xf800 @ mask bits 111x x... .... ....
493 cmp r0, #0xe800 @ 32bit instruction if xx != 0 507 cmp r0, #0xe800 @ 32bit instruction if xx != 0
494 blo __und_usr_unknown 508 blo __und_usr_unknown
@@ -577,9 +591,11 @@ call_fpe:
577 moveq pc, lr 591 moveq pc, lr
578 get_thread_info r10 @ get current thread 592 get_thread_info r10 @ get current thread
579 and r8, r0, #0x00000f00 @ mask out CP number 593 and r8, r0, #0x00000f00 @ mask out CP number
594 THUMB( lsr r8, r8, #8 )
580 mov r7, #1 595 mov r7, #1
581 add r6, r10, #TI_USED_CP 596 add r6, r10, #TI_USED_CP
582 strb r7, [r6, r8, lsr #8] @ set appropriate used_cp[] 597 ARM( strb r7, [r6, r8, lsr #8] ) @ set appropriate used_cp[]
598 THUMB( strb r7, [r6, r8] ) @ set appropriate used_cp[]
583#ifdef CONFIG_IWMMXT 599#ifdef CONFIG_IWMMXT
584 @ Test if we need to give access to iWMMXt coprocessors 600 @ Test if we need to give access to iWMMXt coprocessors
585 ldr r5, [r10, #TI_FLAGS] 601 ldr r5, [r10, #TI_FLAGS]
@@ -587,36 +603,38 @@ call_fpe:
587 movcss r7, r5, lsr #(TIF_USING_IWMMXT + 1) 603 movcss r7, r5, lsr #(TIF_USING_IWMMXT + 1)
588 bcs iwmmxt_task_enable 604 bcs iwmmxt_task_enable
589#endif 605#endif
590 add pc, pc, r8, lsr #6 606 ARM( add pc, pc, r8, lsr #6 )
591 mov r0, r0 607 THUMB( lsl r8, r8, #2 )
592 608 THUMB( add pc, r8 )
593 mov pc, lr @ CP#0 609 nop
594 b do_fpe @ CP#1 (FPE) 610
595 b do_fpe @ CP#2 (FPE) 611 W(mov) pc, lr @ CP#0
596 mov pc, lr @ CP#3 612 W(b) do_fpe @ CP#1 (FPE)
613 W(b) do_fpe @ CP#2 (FPE)
614 W(mov) pc, lr @ CP#3
597#ifdef CONFIG_CRUNCH 615#ifdef CONFIG_CRUNCH
598 b crunch_task_enable @ CP#4 (MaverickCrunch) 616 b crunch_task_enable @ CP#4 (MaverickCrunch)
599 b crunch_task_enable @ CP#5 (MaverickCrunch) 617 b crunch_task_enable @ CP#5 (MaverickCrunch)
600 b crunch_task_enable @ CP#6 (MaverickCrunch) 618 b crunch_task_enable @ CP#6 (MaverickCrunch)
601#else 619#else
602 mov pc, lr @ CP#4 620 W(mov) pc, lr @ CP#4
603 mov pc, lr @ CP#5 621 W(mov) pc, lr @ CP#5
604 mov pc, lr @ CP#6 622 W(mov) pc, lr @ CP#6
605#endif 623#endif
606 mov pc, lr @ CP#7 624 W(mov) pc, lr @ CP#7
607 mov pc, lr @ CP#8 625 W(mov) pc, lr @ CP#8
608 mov pc, lr @ CP#9 626 W(mov) pc, lr @ CP#9
609#ifdef CONFIG_VFP 627#ifdef CONFIG_VFP
610 b do_vfp @ CP#10 (VFP) 628 W(b) do_vfp @ CP#10 (VFP)
611 b do_vfp @ CP#11 (VFP) 629 W(b) do_vfp @ CP#11 (VFP)
612#else 630#else
613 mov pc, lr @ CP#10 (VFP) 631 W(mov) pc, lr @ CP#10 (VFP)
614 mov pc, lr @ CP#11 (VFP) 632 W(mov) pc, lr @ CP#11 (VFP)
615#endif 633#endif
616 mov pc, lr @ CP#12 634 W(mov) pc, lr @ CP#12
617 mov pc, lr @ CP#13 635 W(mov) pc, lr @ CP#13
618 mov pc, lr @ CP#14 (Debug) 636 W(mov) pc, lr @ CP#14 (Debug)
619 mov pc, lr @ CP#15 (Control) 637 W(mov) pc, lr @ CP#15 (Control)
620 638
621#ifdef CONFIG_NEON 639#ifdef CONFIG_NEON
622 .align 6 640 .align 6
@@ -667,7 +685,7 @@ no_fp: mov pc, lr
667__und_usr_unknown: 685__und_usr_unknown:
668 enable_irq 686 enable_irq
669 mov r0, sp 687 mov r0, sp
670 adr lr, ret_from_exception 688 adr lr, BSYM(ret_from_exception)
671 b do_undefinstr 689 b do_undefinstr
672ENDPROC(__und_usr_unknown) 690ENDPROC(__und_usr_unknown)
673 691
@@ -711,7 +729,10 @@ ENTRY(__switch_to)
711 UNWIND(.cantunwind ) 729 UNWIND(.cantunwind )
712 add ip, r1, #TI_CPU_SAVE 730 add ip, r1, #TI_CPU_SAVE
713 ldr r3, [r2, #TI_TP_VALUE] 731 ldr r3, [r2, #TI_TP_VALUE]
714 stmia ip!, {r4 - sl, fp, sp, lr} @ Store most regs on stack 732 ARM( stmia ip!, {r4 - sl, fp, sp, lr} ) @ Store most regs on stack
733 THUMB( stmia ip!, {r4 - sl, fp} ) @ Store most regs on stack
734 THUMB( str sp, [ip], #4 )
735 THUMB( str lr, [ip], #4 )
715#ifdef CONFIG_MMU 736#ifdef CONFIG_MMU
716 ldr r6, [r2, #TI_CPU_DOMAIN] 737 ldr r6, [r2, #TI_CPU_DOMAIN]
717#endif 738#endif
@@ -736,8 +757,12 @@ ENTRY(__switch_to)
736 ldr r0, =thread_notify_head 757 ldr r0, =thread_notify_head
737 mov r1, #THREAD_NOTIFY_SWITCH 758 mov r1, #THREAD_NOTIFY_SWITCH
738 bl atomic_notifier_call_chain 759 bl atomic_notifier_call_chain
760 THUMB( mov ip, r4 )
739 mov r0, r5 761 mov r0, r5
740 ldmia r4, {r4 - sl, fp, sp, pc} @ Load all regs saved previously 762 ARM( ldmia r4, {r4 - sl, fp, sp, pc} ) @ Load all regs saved previously
763 THUMB( ldmia ip!, {r4 - sl, fp} ) @ Load all regs saved previously
764 THUMB( ldr sp, [ip], #4 )
765 THUMB( ldr pc, [ip] )
741 UNWIND(.fnend ) 766 UNWIND(.fnend )
742ENDPROC(__switch_to) 767ENDPROC(__switch_to)
743 768
@@ -772,6 +797,7 @@ ENDPROC(__switch_to)
772 * if your compiled code is not going to use the new instructions for other 797 * if your compiled code is not going to use the new instructions for other
773 * purpose. 798 * purpose.
774 */ 799 */
800 THUMB( .arm )
775 801
776 .macro usr_ret, reg 802 .macro usr_ret, reg
777#ifdef CONFIG_ARM_THUMB 803#ifdef CONFIG_ARM_THUMB
@@ -1020,6 +1046,7 @@ __kuser_helper_version: @ 0xffff0ffc
1020 .globl __kuser_helper_end 1046 .globl __kuser_helper_end
1021__kuser_helper_end: 1047__kuser_helper_end:
1022 1048
1049 THUMB( .thumb )
1023 1050
1024/* 1051/*
1025 * Vector stubs. 1052 * Vector stubs.
@@ -1054,15 +1081,17 @@ vector_\name:
1054 @ Prepare for SVC32 mode. IRQs remain disabled. 1081 @ Prepare for SVC32 mode. IRQs remain disabled.
1055 @ 1082 @
1056 mrs r0, cpsr 1083 mrs r0, cpsr
1057 eor r0, r0, #(\mode ^ SVC_MODE) 1084 eor r0, r0, #(\mode ^ SVC_MODE | PSR_ISETSTATE)
1058 msr spsr_cxsf, r0 1085 msr spsr_cxsf, r0
1059 1086
1060 @ 1087 @
1061 @ the branch table must immediately follow this code 1088 @ the branch table must immediately follow this code
1062 @ 1089 @
1063 and lr, lr, #0x0f 1090 and lr, lr, #0x0f
1091 THUMB( adr r0, 1f )
1092 THUMB( ldr lr, [r0, lr, lsl #2] )
1064 mov r0, sp 1093 mov r0, sp
1065 ldr lr, [pc, lr, lsl #2] 1094 ARM( ldr lr, [pc, lr, lsl #2] )
1066 movs pc, lr @ branch to handler in SVC mode 1095 movs pc, lr @ branch to handler in SVC mode
1067ENDPROC(vector_\name) 1096ENDPROC(vector_\name)
1068 1097
@@ -1206,14 +1235,16 @@ __stubs_end:
1206 1235
1207 .globl __vectors_start 1236 .globl __vectors_start
1208__vectors_start: 1237__vectors_start:
1209 swi SYS_ERROR0 1238 ARM( swi SYS_ERROR0 )
1210 b vector_und + stubs_offset 1239 THUMB( svc #0 )
1211 ldr pc, .LCvswi + stubs_offset 1240 THUMB( nop )
1212 b vector_pabt + stubs_offset 1241 W(b) vector_und + stubs_offset
1213 b vector_dabt + stubs_offset 1242 W(ldr) pc, .LCvswi + stubs_offset
1214 b vector_addrexcptn + stubs_offset 1243 W(b) vector_pabt + stubs_offset
1215 b vector_irq + stubs_offset 1244 W(b) vector_dabt + stubs_offset
1216 b vector_fiq + stubs_offset 1245 W(b) vector_addrexcptn + stubs_offset
1246 W(b) vector_irq + stubs_offset
1247 W(b) vector_fiq + stubs_offset
1217 1248
1218 .globl __vectors_end 1249 .globl __vectors_end
1219__vectors_end: 1250__vectors_end:
diff --git a/arch/arm/kernel/entry-common.S b/arch/arm/kernel/entry-common.S
index 366e5097a41a..a0540c9f1f0c 100644
--- a/arch/arm/kernel/entry-common.S
+++ b/arch/arm/kernel/entry-common.S
@@ -33,14 +33,7 @@ ret_fast_syscall:
33 /* perform architecture specific actions before user return */ 33 /* perform architecture specific actions before user return */
34 arch_ret_to_user r1, lr 34 arch_ret_to_user r1, lr
35 35
36 @ fast_restore_user_regs 36 restore_user_regs fast = 1, offset = S_OFF
37 ldr r1, [sp, #S_OFF + S_PSR] @ get calling cpsr
38 ldr lr, [sp, #S_OFF + S_PC]! @ get pc
39 msr spsr_cxsf, r1 @ save in spsr_svc
40 ldmdb sp, {r1 - lr}^ @ get calling r1 - lr
41 mov r0, r0
42 add sp, sp, #S_FRAME_SIZE - S_PC
43 movs pc, lr @ return & move spsr_svc into cpsr
44 UNWIND(.fnend ) 37 UNWIND(.fnend )
45 38
46/* 39/*
@@ -73,14 +66,7 @@ no_work_pending:
73 /* perform architecture specific actions before user return */ 66 /* perform architecture specific actions before user return */
74 arch_ret_to_user r1, lr 67 arch_ret_to_user r1, lr
75 68
76 @ slow_restore_user_regs 69 restore_user_regs fast = 0, offset = 0
77 ldr r1, [sp, #S_PSR] @ get calling cpsr
78 ldr lr, [sp, #S_PC]! @ get pc
79 msr spsr_cxsf, r1 @ save in spsr_svc
80 ldmdb sp, {r0 - lr}^ @ get calling r0 - lr
81 mov r0, r0
82 add sp, sp, #S_FRAME_SIZE - S_PC
83 movs pc, lr @ return & move spsr_svc into cpsr
84ENDPROC(ret_to_user) 70ENDPROC(ret_to_user)
85 71
86/* 72/*
@@ -182,8 +168,10 @@ ftrace_stub:
182ENTRY(vector_swi) 168ENTRY(vector_swi)
183 sub sp, sp, #S_FRAME_SIZE 169 sub sp, sp, #S_FRAME_SIZE
184 stmia sp, {r0 - r12} @ Calling r0 - r12 170 stmia sp, {r0 - r12} @ Calling r0 - r12
185 add r8, sp, #S_PC 171 ARM( add r8, sp, #S_PC )
186 stmdb r8, {sp, lr}^ @ Calling sp, lr 172 ARM( stmdb r8, {sp, lr}^ ) @ Calling sp, lr
173 THUMB( mov r8, sp )
174 THUMB( store_user_sp_lr r8, r10, S_SP ) @ calling sp, lr
187 mrs r8, spsr @ called from non-FIQ mode, so ok. 175 mrs r8, spsr @ called from non-FIQ mode, so ok.
188 str lr, [sp, #S_PC] @ Save calling PC 176 str lr, [sp, #S_PC] @ Save calling PC
189 str r8, [sp, #S_PSR] @ Save CPSR 177 str r8, [sp, #S_PSR] @ Save CPSR
@@ -272,7 +260,7 @@ ENTRY(vector_swi)
272 bne __sys_trace 260 bne __sys_trace
273 261
274 cmp scno, #NR_syscalls @ check upper syscall limit 262 cmp scno, #NR_syscalls @ check upper syscall limit
275 adr lr, ret_fast_syscall @ return address 263 adr lr, BSYM(ret_fast_syscall) @ return address
276 ldrcc pc, [tbl, scno, lsl #2] @ call sys_* routine 264 ldrcc pc, [tbl, scno, lsl #2] @ call sys_* routine
277 265
278 add r1, sp, #S_OFF 266 add r1, sp, #S_OFF
@@ -293,7 +281,7 @@ __sys_trace:
293 mov r0, #0 @ trace entry [IP = 0] 281 mov r0, #0 @ trace entry [IP = 0]
294 bl syscall_trace 282 bl syscall_trace
295 283
296 adr lr, __sys_trace_return @ return address 284 adr lr, BSYM(__sys_trace_return) @ return address
297 mov scno, r0 @ syscall number (possibly new) 285 mov scno, r0 @ syscall number (possibly new)
298 add r1, sp, #S_R0 + S_OFF @ pointer to regs 286 add r1, sp, #S_R0 + S_OFF @ pointer to regs
299 cmp scno, #NR_syscalls @ check upper syscall limit 287 cmp scno, #NR_syscalls @ check upper syscall limit
diff --git a/arch/arm/kernel/entry-header.S b/arch/arm/kernel/entry-header.S
index 87ab4e157997..a4eaf4f920c5 100644
--- a/arch/arm/kernel/entry-header.S
+++ b/arch/arm/kernel/entry-header.S
@@ -36,11 +36,6 @@
36#endif 36#endif
37 .endm 37 .endm
38 38
39 .macro get_thread_info, rd
40 mov \rd, sp, lsr #13
41 mov \rd, \rd, lsl #13
42 .endm
43
44 .macro alignment_trap, rtemp 39 .macro alignment_trap, rtemp
45#ifdef CONFIG_ALIGNMENT_TRAP 40#ifdef CONFIG_ALIGNMENT_TRAP
46 ldr \rtemp, .LCcralign 41 ldr \rtemp, .LCcralign
@@ -49,6 +44,93 @@
49#endif 44#endif
50 .endm 45 .endm
51 46
47 @
48 @ Store/load the USER SP and LR registers by switching to the SYS
49 @ mode. Useful in Thumb-2 mode where "stm/ldm rd, {sp, lr}^" is not
50 @ available. Should only be called from SVC mode
51 @
52 .macro store_user_sp_lr, rd, rtemp, offset = 0
53 mrs \rtemp, cpsr
54 eor \rtemp, \rtemp, #(SVC_MODE ^ SYSTEM_MODE)
55 msr cpsr_c, \rtemp @ switch to the SYS mode
56
57 str sp, [\rd, #\offset] @ save sp_usr
58 str lr, [\rd, #\offset + 4] @ save lr_usr
59
60 eor \rtemp, \rtemp, #(SVC_MODE ^ SYSTEM_MODE)
61 msr cpsr_c, \rtemp @ switch back to the SVC mode
62 .endm
63
64 .macro load_user_sp_lr, rd, rtemp, offset = 0
65 mrs \rtemp, cpsr
66 eor \rtemp, \rtemp, #(SVC_MODE ^ SYSTEM_MODE)
67 msr cpsr_c, \rtemp @ switch to the SYS mode
68
69 ldr sp, [\rd, #\offset] @ load sp_usr
70 ldr lr, [\rd, #\offset + 4] @ load lr_usr
71
72 eor \rtemp, \rtemp, #(SVC_MODE ^ SYSTEM_MODE)
73 msr cpsr_c, \rtemp @ switch back to the SVC mode
74 .endm
75
76#ifndef CONFIG_THUMB2_KERNEL
77 .macro svc_exit, rpsr
78 msr spsr_cxsf, \rpsr
79 ldmia sp, {r0 - pc}^ @ load r0 - pc, cpsr
80 .endm
81
82 .macro restore_user_regs, fast = 0, offset = 0
83 ldr r1, [sp, #\offset + S_PSR] @ get calling cpsr
84 ldr lr, [sp, #\offset + S_PC]! @ get pc
85 msr spsr_cxsf, r1 @ save in spsr_svc
86 .if \fast
87 ldmdb sp, {r1 - lr}^ @ get calling r1 - lr
88 .else
89 ldmdb sp, {r0 - lr}^ @ get calling r0 - lr
90 .endif
91 add sp, sp, #S_FRAME_SIZE - S_PC
92 movs pc, lr @ return & move spsr_svc into cpsr
93 .endm
94
95 .macro get_thread_info, rd
96 mov \rd, sp, lsr #13
97 mov \rd, \rd, lsl #13
98 .endm
99#else /* CONFIG_THUMB2_KERNEL */
100 .macro svc_exit, rpsr
101 ldr r0, [sp, #S_SP] @ top of the stack
102 ldr r1, [sp, #S_PC] @ return address
103 tst r0, #4 @ orig stack 8-byte aligned?
104 stmdb r0, {r1, \rpsr} @ rfe context
105 ldmia sp, {r0 - r12}
106 ldr lr, [sp, #S_LR]
107 addeq sp, sp, #S_FRAME_SIZE - 8 @ aligned
108 addne sp, sp, #S_FRAME_SIZE - 4 @ not aligned
109 rfeia sp!
110 .endm
111
112 .macro restore_user_regs, fast = 0, offset = 0
113 mov r2, sp
114 load_user_sp_lr r2, r3, \offset + S_SP @ calling sp, lr
115 ldr r1, [sp, #\offset + S_PSR] @ get calling cpsr
116 ldr lr, [sp, #\offset + S_PC] @ get pc
117 add sp, sp, #\offset + S_SP
118 msr spsr_cxsf, r1 @ save in spsr_svc
119 .if \fast
120 ldmdb sp, {r1 - r12} @ get calling r1 - r12
121 .else
122 ldmdb sp, {r0 - r12} @ get calling r0 - r12
123 .endif
124 add sp, sp, #S_FRAME_SIZE - S_SP
125 movs pc, lr @ return & move spsr_svc into cpsr
126 .endm
127
128 .macro get_thread_info, rd
129 mov \rd, sp
130 lsr \rd, \rd, #13
131 mov \rd, \rd, lsl #13
132 .endm
133#endif /* !CONFIG_THUMB2_KERNEL */
52 134
53/* 135/*
54 * These are the registers used in the syscall handler, and allow us to 136 * These are the registers used in the syscall handler, and allow us to
diff --git a/arch/arm/kernel/head-common.S b/arch/arm/kernel/head-common.S
index dbfaef07a1f4..93ad576b2d74 100644
--- a/arch/arm/kernel/head-common.S
+++ b/arch/arm/kernel/head-common.S
@@ -52,7 +52,9 @@ __mmap_switched:
52 strcc fp, [r6],#4 52 strcc fp, [r6],#4
53 bcc 1b 53 bcc 1b
54 54
55 ldmia r3, {r4, r5, r6, r7, sp} 55 ARM( ldmia r3, {r4, r5, r6, r7, sp})
56 THUMB( ldmia r3, {r4, r5, r6, r7} )
57 THUMB( ldr sp, [r3, #16] )
56 str r9, [r4] @ Save processor ID 58 str r9, [r4] @ Save processor ID
57 str r1, [r5] @ Save machine type 59 str r1, [r5] @ Save machine type
58 str r2, [r6] @ Save atags pointer 60 str r2, [r6] @ Save atags pointer
@@ -156,7 +158,8 @@ ENDPROC(__error)
156 */ 158 */
157__lookup_processor_type: 159__lookup_processor_type:
158 adr r3, 3f 160 adr r3, 3f
159 ldmda r3, {r5 - r7} 161 ldmia r3, {r5 - r7}
162 add r3, r3, #8
160 sub r3, r3, r7 @ get offset between virt&phys 163 sub r3, r3, r7 @ get offset between virt&phys
161 add r5, r5, r3 @ convert virt addresses to 164 add r5, r5, r3 @ convert virt addresses to
162 add r6, r6, r3 @ physical address space 165 add r6, r6, r3 @ physical address space
@@ -187,9 +190,9 @@ ENDPROC(lookup_processor_type)
187 * more information about the __proc_info and __arch_info structures. 190 * more information about the __proc_info and __arch_info structures.
188 */ 191 */
189 .align 2 192 .align 2
190 .long __proc_info_begin 1933: .long __proc_info_begin
191 .long __proc_info_end 194 .long __proc_info_end
1923: .long . 1954: .long .
193 .long __arch_info_begin 196 .long __arch_info_begin
194 .long __arch_info_end 197 .long __arch_info_end
195 198
@@ -205,7 +208,7 @@ ENDPROC(lookup_processor_type)
205 * r5 = mach_info pointer in physical address space 208 * r5 = mach_info pointer in physical address space
206 */ 209 */
207__lookup_machine_type: 210__lookup_machine_type:
208 adr r3, 3b 211 adr r3, 4b
209 ldmia r3, {r4, r5, r6} 212 ldmia r3, {r4, r5, r6}
210 sub r3, r3, r4 @ get offset between virt&phys 213 sub r3, r3, r4 @ get offset between virt&phys
211 add r5, r5, r3 @ convert virt addresses to 214 add r5, r5, r3 @ convert virt addresses to
diff --git a/arch/arm/kernel/head-nommu.S b/arch/arm/kernel/head-nommu.S
index cc87e1765ed2..b16393d2b71e 100644
--- a/arch/arm/kernel/head-nommu.S
+++ b/arch/arm/kernel/head-nommu.S
@@ -34,7 +34,7 @@
34 */ 34 */
35 .section ".text.head", "ax" 35 .section ".text.head", "ax"
36ENTRY(stext) 36ENTRY(stext)
37 msr cpsr_c, #PSR_F_BIT | PSR_I_BIT | SVC_MODE @ ensure svc mode 37 setmode PSR_F_BIT | PSR_I_BIT | SVC_MODE, r9 @ ensure svc mode
38 @ and irqs disabled 38 @ and irqs disabled
39#ifndef CONFIG_CPU_CP15 39#ifndef CONFIG_CPU_CP15
40 ldr r9, =CONFIG_PROCESSOR_ID 40 ldr r9, =CONFIG_PROCESSOR_ID
@@ -50,8 +50,10 @@ ENTRY(stext)
50 50
51 ldr r13, __switch_data @ address to jump to after 51 ldr r13, __switch_data @ address to jump to after
52 @ the initialization is done 52 @ the initialization is done
53 adr lr, __after_proc_init @ return (PIC) address 53 adr lr, BSYM(__after_proc_init) @ return (PIC) address
54 add pc, r10, #PROCINFO_INITFUNC 54 ARM( add pc, r10, #PROCINFO_INITFUNC )
55 THUMB( add r12, r10, #PROCINFO_INITFUNC )
56 THUMB( mov pc, r12 )
55ENDPROC(stext) 57ENDPROC(stext)
56 58
57/* 59/*
@@ -82,7 +84,8 @@ __after_proc_init:
82 mcr p15, 0, r0, c1, c0, 0 @ write control reg 84 mcr p15, 0, r0, c1, c0, 0 @ write control reg
83#endif /* CONFIG_CPU_CP15 */ 85#endif /* CONFIG_CPU_CP15 */
84 86
85 mov pc, r13 @ clear the BSS and jump 87 mov r3, r13
88 mov pc, r3 @ clear the BSS and jump
86 @ to start_kernel 89 @ to start_kernel
87ENDPROC(__after_proc_init) 90ENDPROC(__after_proc_init)
88 .ltorg 91 .ltorg
diff --git a/arch/arm/kernel/head.S b/arch/arm/kernel/head.S
index 21e17dc94cb5..38ccbe1d3b2c 100644
--- a/arch/arm/kernel/head.S
+++ b/arch/arm/kernel/head.S
@@ -76,7 +76,7 @@
76 */ 76 */
77 .section ".text.head", "ax" 77 .section ".text.head", "ax"
78ENTRY(stext) 78ENTRY(stext)
79 msr cpsr_c, #PSR_F_BIT | PSR_I_BIT | SVC_MODE @ ensure svc mode 79 setmode PSR_F_BIT | PSR_I_BIT | SVC_MODE, r9 @ ensure svc mode
80 @ and irqs disabled 80 @ and irqs disabled
81 mrc p15, 0, r9, c0, c0 @ get processor id 81 mrc p15, 0, r9, c0, c0 @ get processor id
82 bl __lookup_processor_type @ r5=procinfo r9=cpuid 82 bl __lookup_processor_type @ r5=procinfo r9=cpuid
@@ -97,8 +97,10 @@ ENTRY(stext)
97 */ 97 */
98 ldr r13, __switch_data @ address to jump to after 98 ldr r13, __switch_data @ address to jump to after
99 @ mmu has been enabled 99 @ mmu has been enabled
100 adr lr, __enable_mmu @ return (PIC) address 100 adr lr, BSYM(__enable_mmu) @ return (PIC) address
101 add pc, r10, #PROCINFO_INITFUNC 101 ARM( add pc, r10, #PROCINFO_INITFUNC )
102 THUMB( add r12, r10, #PROCINFO_INITFUNC )
103 THUMB( mov pc, r12 )
102ENDPROC(stext) 104ENDPROC(stext)
103 105
104#if defined(CONFIG_SMP) 106#if defined(CONFIG_SMP)
@@ -110,7 +112,7 @@ ENTRY(secondary_startup)
110 * the processor type - there is no need to check the machine type 112 * the processor type - there is no need to check the machine type
111 * as it has already been validated by the primary processor. 113 * as it has already been validated by the primary processor.
112 */ 114 */
113 msr cpsr_c, #PSR_F_BIT | PSR_I_BIT | SVC_MODE 115 setmode PSR_F_BIT | PSR_I_BIT | SVC_MODE, r9
114 mrc p15, 0, r9, c0, c0 @ get processor id 116 mrc p15, 0, r9, c0, c0 @ get processor id
115 bl __lookup_processor_type 117 bl __lookup_processor_type
116 movs r10, r5 @ invalid processor? 118 movs r10, r5 @ invalid processor?
@@ -121,12 +123,15 @@ ENTRY(secondary_startup)
121 * Use the page tables supplied from __cpu_up. 123 * Use the page tables supplied from __cpu_up.
122 */ 124 */
123 adr r4, __secondary_data 125 adr r4, __secondary_data
124 ldmia r4, {r5, r7, r13} @ address to jump to after 126 ldmia r4, {r5, r7, r12} @ address to jump to after
125 sub r4, r4, r5 @ mmu has been enabled 127 sub r4, r4, r5 @ mmu has been enabled
126 ldr r4, [r7, r4] @ get secondary_data.pgdir 128 ldr r4, [r7, r4] @ get secondary_data.pgdir
127 adr lr, __enable_mmu @ return address 129 adr lr, BSYM(__enable_mmu) @ return address
128 add pc, r10, #PROCINFO_INITFUNC @ initialise processor 130 mov r13, r12 @ __secondary_switched address
129 @ (return control reg) 131 ARM( add pc, r10, #PROCINFO_INITFUNC ) @ initialise processor
132 @ (return control reg)
133 THUMB( add r12, r10, #PROCINFO_INITFUNC )
134 THUMB( mov pc, r12 )
130ENDPROC(secondary_startup) 135ENDPROC(secondary_startup)
131 136
132 /* 137 /*
@@ -193,8 +198,8 @@ __turn_mmu_on:
193 mcr p15, 0, r0, c1, c0, 0 @ write control reg 198 mcr p15, 0, r0, c1, c0, 0 @ write control reg
194 mrc p15, 0, r3, c0, c0, 0 @ read id reg 199 mrc p15, 0, r3, c0, c0, 0 @ read id reg
195 mov r3, r3 200 mov r3, r3
196 mov r3, r3 201 mov r3, r13
197 mov pc, r13 202 mov pc, r3
198ENDPROC(__turn_mmu_on) 203ENDPROC(__turn_mmu_on)
199 204
200 205
@@ -235,7 +240,8 @@ __create_page_tables:
235 * will be removed by paging_init(). We use our current program 240 * will be removed by paging_init(). We use our current program
236 * counter to determine corresponding section base address. 241 * counter to determine corresponding section base address.
237 */ 242 */
238 mov r6, pc, lsr #20 @ start of kernel section 243 mov r6, pc
244 mov r6, r6, lsr #20 @ start of kernel section
239 orr r3, r7, r6, lsl #20 @ flags + kernel base 245 orr r3, r7, r6, lsl #20 @ flags + kernel base
240 str r3, [r4, r6, lsl #2] @ identity mapping 246 str r3, [r4, r6, lsl #2] @ identity mapping
241 247
diff --git a/arch/arm/kernel/process.c b/arch/arm/kernel/process.c
index 39196dff478c..790fbee92ec5 100644
--- a/arch/arm/kernel/process.c
+++ b/arch/arm/kernel/process.c
@@ -388,7 +388,7 @@ pid_t kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)
388 regs.ARM_r2 = (unsigned long)fn; 388 regs.ARM_r2 = (unsigned long)fn;
389 regs.ARM_r3 = (unsigned long)kernel_thread_exit; 389 regs.ARM_r3 = (unsigned long)kernel_thread_exit;
390 regs.ARM_pc = (unsigned long)kernel_thread_helper; 390 regs.ARM_pc = (unsigned long)kernel_thread_helper;
391 regs.ARM_cpsr = SVC_MODE | PSR_ENDSTATE; 391 regs.ARM_cpsr = SVC_MODE | PSR_ENDSTATE | PSR_ISETSTATE;
392 392
393 return do_fork(flags|CLONE_VM|CLONE_UNTRACED, 0, &regs, 0, NULL, NULL); 393 return do_fork(flags|CLONE_VM|CLONE_UNTRACED, 0, &regs, 0, NULL, NULL);
394} 394}
diff --git a/arch/arm/kernel/setup.c b/arch/arm/kernel/setup.c
index bc5e4128f9f3..d4d4f77c91b2 100644
--- a/arch/arm/kernel/setup.c
+++ b/arch/arm/kernel/setup.c
@@ -25,6 +25,7 @@
25#include <linux/smp.h> 25#include <linux/smp.h>
26#include <linux/fs.h> 26#include <linux/fs.h>
27 27
28#include <asm/unified.h>
28#include <asm/cpu.h> 29#include <asm/cpu.h>
29#include <asm/cputype.h> 30#include <asm/cputype.h>
30#include <asm/elf.h> 31#include <asm/elf.h>
@@ -327,25 +328,38 @@ void cpu_init(void)
327 } 328 }
328 329
329 /* 330 /*
331 * Define the placement constraint for the inline asm directive below.
332 * In Thumb-2, msr with an immediate value is not allowed.
333 */
334#ifdef CONFIG_THUMB2_KERNEL
335#define PLC "r"
336#else
337#define PLC "I"
338#endif
339
340 /*
330 * setup stacks for re-entrant exception handlers 341 * setup stacks for re-entrant exception handlers
331 */ 342 */
332 __asm__ ( 343 __asm__ (
333 "msr cpsr_c, %1\n\t" 344 "msr cpsr_c, %1\n\t"
334 "add sp, %0, %2\n\t" 345 "add r14, %0, %2\n\t"
346 "mov sp, r14\n\t"
335 "msr cpsr_c, %3\n\t" 347 "msr cpsr_c, %3\n\t"
336 "add sp, %0, %4\n\t" 348 "add r14, %0, %4\n\t"
349 "mov sp, r14\n\t"
337 "msr cpsr_c, %5\n\t" 350 "msr cpsr_c, %5\n\t"
338 "add sp, %0, %6\n\t" 351 "add r14, %0, %6\n\t"
352 "mov sp, r14\n\t"
339 "msr cpsr_c, %7" 353 "msr cpsr_c, %7"
340 : 354 :
341 : "r" (stk), 355 : "r" (stk),
342 "I" (PSR_F_BIT | PSR_I_BIT | IRQ_MODE), 356 PLC (PSR_F_BIT | PSR_I_BIT | IRQ_MODE),
343 "I" (offsetof(struct stack, irq[0])), 357 "I" (offsetof(struct stack, irq[0])),
344 "I" (PSR_F_BIT | PSR_I_BIT | ABT_MODE), 358 PLC (PSR_F_BIT | PSR_I_BIT | ABT_MODE),
345 "I" (offsetof(struct stack, abt[0])), 359 "I" (offsetof(struct stack, abt[0])),
346 "I" (PSR_F_BIT | PSR_I_BIT | UND_MODE), 360 PLC (PSR_F_BIT | PSR_I_BIT | UND_MODE),
347 "I" (offsetof(struct stack, und[0])), 361 "I" (offsetof(struct stack, und[0])),
348 "I" (PSR_F_BIT | PSR_I_BIT | SVC_MODE) 362 PLC (PSR_F_BIT | PSR_I_BIT | SVC_MODE)
349 : "r14"); 363 : "r14");
350} 364}
351 365
diff --git a/arch/arm/kernel/unwind.c b/arch/arm/kernel/unwind.c
index dd56e11f339a..39baf1128bfa 100644
--- a/arch/arm/kernel/unwind.c
+++ b/arch/arm/kernel/unwind.c
@@ -62,7 +62,11 @@ struct unwind_ctrl_block {
62}; 62};
63 63
64enum regs { 64enum regs {
65#ifdef CONFIG_THUMB2_KERNEL
66 FP = 7,
67#else
65 FP = 11, 68 FP = 11,
69#endif
66 SP = 13, 70 SP = 13,
67 LR = 14, 71 LR = 14,
68 PC = 15 72 PC = 15