Linux 2.6.21
[linux/fpc-iii.git] / arch / arm26 / kernel / entry.S
blob91496cc687b1d6831acbbd7bb82a861da12783da
1 /* arch/arm26/kernel/entry.S
2  * 
3  * Assembled from chunks of code in arch/arm
4  *
5  * Copyright (C) 2003 Ian Molton
6  * Based on the work of RMK.
7  *
8  */
10 #include <linux/linkage.h>
12 #include <asm/assembler.h>
13 #include <asm/asm-offsets.h>
14 #include <asm/errno.h>
15 #include <asm/hardware.h>
16 #include <asm/sysirq.h>
17 #include <asm/thread_info.h>
18 #include <asm/page.h>
19 #include <asm/ptrace.h>
21         .macro  zero_fp
22 #ifndef CONFIG_NO_FRAME_POINTER
23         mov     fp, #0
24 #endif
25         .endm
27         .text
29 @ Bad Abort numbers
30 @ -----------------
32 #define BAD_PREFETCH    0
33 #define BAD_DATA        1
34 #define BAD_ADDREXCPTN  2
35 #define BAD_IRQ         3
36 #define BAD_UNDEFINSTR  4
38 @ OS version number used in SWIs
39 @  RISC OS is 0
40 @  RISC iX is 8
42 #define OS_NUMBER       9
43 #define ARMSWI_OFFSET   0x000f0000
46 @ Stack format (ensured by USER_* and SVC_*)
47 @ PSR and PC are comined on arm26
50 #define S_OFF           8
52 #define S_OLD_R0        64
53 #define S_PC            60
54 #define S_LR            56
55 #define S_SP            52
56 #define S_IP            48
57 #define S_FP            44
58 #define S_R10           40
59 #define S_R9            36
60 #define S_R8            32
61 #define S_R7            28
62 #define S_R6            24
63 #define S_R5            20
64 #define S_R4            16
65 #define S_R3            12
66 #define S_R2            8
67 #define S_R1            4
68 #define S_R0            0
70         .macro  save_user_regs
71         str     r0, [sp, #-4]!   @ Store SVC r0
72         str     lr, [sp, #-4]!   @ Store user mode PC
73         sub     sp, sp, #15*4
74         stmia   sp, {r0 - lr}^   @ Store the other user-mode regs
75         mov     r0, r0
76         .endm
78         .macro  slow_restore_user_regs
79         ldmia   sp, {r0 - lr}^   @ restore the user regs not including PC
80         mov     r0, r0
81         ldr     lr, [sp, #15*4]  @ get user PC
82         add     sp, sp, #15*4+8  @ free stack
83         movs    pc, lr           @ return
84         .endm
86         .macro  fast_restore_user_regs
87         add     sp, sp, #S_OFF
88         ldmib   sp, {r1 - lr}^
89         mov     r0, r0
90         ldr     lr, [sp, #15*4]
91         add     sp, sp, #15*4+8
92         movs    pc, lr
93         .endm
95         .macro  save_svc_regs
96         str     sp, [sp, #-16]!
97         str     lr, [sp, #8]
98         str     lr, [sp, #4]
99         stmfd   sp!, {r0 - r12}
100         mov     r0, #-1
101         str     r0, [sp, #S_OLD_R0]
102         zero_fp
103         .endm
105         .macro  save_svc_regs_irq
106         str     sp, [sp, #-16]!
107         str     lr, [sp, #4]
108         ldr     lr, .LCirq
109         ldr     lr, [lr]
110         str     lr, [sp, #8]
111         stmfd   sp!, {r0 - r12}
112         mov     r0, #-1
113         str     r0, [sp, #S_OLD_R0]
114         zero_fp
115         .endm
117         .macro  restore_svc_regs
118                 ldmfd   sp, {r0 - pc}^
119         .endm
121         .macro  mask_pc, rd, rm
122         bic     \rd, \rm, #PCMASK
123         .endm
125         .macro  disable_irqs, temp
126         mov     \temp, pc
127         orr     \temp, \temp, #PSR_I_BIT
128         teqp    \temp, #0
129         .endm
131         .macro  enable_irqs, temp
132         mov     \temp, pc
133         and     \temp, \temp, #~PSR_I_BIT
134         teqp    \temp, #0
135         .endm
137         .macro  initialise_traps_extra
138         .endm
140         .macro  get_thread_info, rd
141         mov     \rd, sp, lsr #13
142         mov     \rd, \rd, lsl #13
143         .endm
146  * These are the registers used in the syscall handler, and allow us to
147  * have in theory up to 7 arguments to a function - r0 to r6.
149  * Note that tbl == why is intentional.
151  * We must set at least "tsk" and "why" when calling ret_with_reschedule.
152  */
153 scno    .req    r7              @ syscall number
154 tbl     .req    r8              @ syscall table pointer
155 why     .req    r8              @ Linux syscall (!= 0)
156 tsk     .req    r9              @ current thread_info
159  * Get the system call number.
160  */
161         .macro  get_scno
162         mask_pc lr, lr
163         ldr     scno, [lr, #-4]         @ get SWI instruction
164         .endm
166  *  -----------------------------------------------------------------------
167  */
169 /* 
170  * We rely on the fact that R0 is at the bottom of the stack (due to
171  * slow/fast restore user regs).
172  */
173 #if S_R0 != 0
174 #error "Please fix"
175 #endif
178  * This is the fast syscall return path.  We do as little as
179  * possible here, and this includes saving r0 back into the SVC
180  * stack.
181  */
182 ret_fast_syscall:
183         disable_irqs r1                         @ disable interrupts
184         ldr     r1, [tsk, #TI_FLAGS]
185         tst     r1, #_TIF_WORK_MASK
186         bne     fast_work_pending
187         fast_restore_user_regs
190  * Ok, we need to do extra processing, enter the slow path.
191  */
192 fast_work_pending:
193         str     r0, [sp, #S_R0+S_OFF]!          @ returned r0
194 work_pending:
195         tst     r1, #_TIF_NEED_RESCHED
196         bne     work_resched
197         tst     r1, #_TIF_NOTIFY_RESUME | _TIF_SIGPENDING
198         beq     no_work_pending
199         mov     r0, sp                          @ 'regs'
200         mov     r2, why                         @ 'syscall'
201         bl      do_notify_resume
202         disable_irqs r1                         @ disable interrupts
203         b       no_work_pending
205 work_resched:
206         bl      schedule
208  * "slow" syscall return path.  "why" tells us if this was a real syscall.
209  */
210 ENTRY(ret_to_user)
211 ret_slow_syscall:
212         disable_irqs r1                         @ disable interrupts
213         ldr     r1, [tsk, #TI_FLAGS]
214         tst     r1, #_TIF_WORK_MASK
215         bne     work_pending
216 no_work_pending:
217         slow_restore_user_regs
220  * This is how we return from a fork.
221  */
222 ENTRY(ret_from_fork)
223         bl      schedule_tail
224         get_thread_info tsk
225         ldr     r1, [tsk, #TI_FLAGS]            @ check for syscall tracing
226         mov     why, #1
227         tst     r1, #_TIF_SYSCALL_TRACE         @ are we tracing syscalls?
228         beq     ret_slow_syscall
229         mov     r1, sp
230         mov     r0, #1                          @ trace exit [IP = 1]
231         bl      syscall_trace
232         b       ret_slow_syscall
233         
234 // FIXME - is this strictly necessary?
235 #include "calls.S"
237 /*=============================================================================
238  * SWI handler
239  *-----------------------------------------------------------------------------
240  */
242         .align  5
243 ENTRY(vector_swi)
244         save_user_regs
245         zero_fp
246         get_scno
248         enable_irqs ip
250         str     r4, [sp, #-S_OFF]!              @ push fifth arg
252         get_thread_info tsk
253         ldr     ip, [tsk, #TI_FLAGS]            @ check for syscall tracing
254         bic     scno, scno, #0xff000000         @ mask off SWI op-code
255         eor     scno, scno, #OS_NUMBER << 20    @ check OS number
256         adr     tbl, sys_call_table             @ load syscall table pointer
257         tst     ip, #_TIF_SYSCALL_TRACE         @ are we tracing syscalls?
258         bne     __sys_trace
260         adral   lr, ret_fast_syscall            @ set return address
261         orral   lr, lr, #PSR_I_BIT | MODE_SVC26 @ Force SVC mode on return
262         cmp     scno, #NR_syscalls              @ check upper syscall limit
263         ldrcc   pc, [tbl, scno, lsl #2]         @ call sys_* routine
265         add     r1, sp, #S_OFF
266 2:      mov     why, #0                         @ no longer a real syscall
267         cmp     scno, #ARMSWI_OFFSET
268         eor     r0, scno, #OS_NUMBER << 20      @ put OS number back
269         bcs     arm_syscall     
270         b       sys_ni_syscall                  @ not private func
272         /*
273          * This is the really slow path.  We're going to be doing
274          * context switches, and waiting for our parent to respond.
275          */
276 __sys_trace:
277         add     r1, sp, #S_OFF
278         mov     r0, #0                          @ trace entry [IP = 0]
279         bl      syscall_trace
281         adral   lr, __sys_trace_return          @ set return address
282         orral   lr, lr, #PSR_I_BIT | MODE_SVC26 @ Force SVC mode on return
283         add     r1, sp, #S_R0 + S_OFF           @ pointer to regs
284         cmp     scno, #NR_syscalls              @ check upper syscall limit
285         ldmccia r1, {r0 - r3}                   @ have to reload r0 - r3
286         ldrcc   pc, [tbl, scno, lsl #2]         @ call sys_* routine
287         b       2b
289 __sys_trace_return:
290         str     r0, [sp, #S_R0 + S_OFF]!        @ save returned r0
291         mov     r1, sp
292         mov     r0, #1                          @ trace exit [IP = 1]
293         bl      syscall_trace
294         b       ret_slow_syscall
296         .align  5
298         .type   sys_call_table, #object
299 ENTRY(sys_call_table)
300 #include "calls.S"
302 /*============================================================================
303  * Special system call wrappers
304  */
305 @ r0 = syscall number
306 @ r5 = syscall table
307                 .type   sys_syscall, #function
308 sys_syscall:
309                 eor     scno, r0, #OS_NUMBER << 20
310                 cmp     scno, #NR_syscalls      @ check range
311                 stmleia sp, {r5, r6}            @ shuffle args
312                 movle   r0, r1
313                 movle   r1, r2
314                 movle   r2, r3
315                 movle   r3, r4
316                 ldrle   pc, [tbl, scno, lsl #2]
317                 b       sys_ni_syscall
319 sys_fork_wrapper:
320                 add     r0, sp, #S_OFF
321                 b       sys_fork
323 sys_vfork_wrapper:
324                 add     r0, sp, #S_OFF
325                 b       sys_vfork
327 sys_execve_wrapper:
328                 add     r3, sp, #S_OFF
329                 b       sys_execve
331 sys_clone_wapper:
332                 add     r2, sp, #S_OFF
333                 b       sys_clone
335 sys_sigsuspend_wrapper:
336                 add     r3, sp, #S_OFF
337                 b       sys_sigsuspend
339 sys_rt_sigsuspend_wrapper:
340                 add     r2, sp, #S_OFF
341                 b       sys_rt_sigsuspend
343 sys_sigreturn_wrapper:
344                 add     r0, sp, #S_OFF
345                 b       sys_sigreturn
347 sys_rt_sigreturn_wrapper:
348                 add     r0, sp, #S_OFF
349                 b       sys_rt_sigreturn
351 sys_sigaltstack_wrapper:
352                 ldr     r2, [sp, #S_OFF + S_SP]
353                 b       do_sigaltstack
356  * Note: off_4k (r5) is always units of 4K.  If we can't do the requested
357  * offset, we return EINVAL.  FIXME - this lost some stuff from arm32 to
358  * ifdefs. check it out.
359  */
360 sys_mmap2:
361                 tst     r5, #((1 << (PAGE_SHIFT - 12)) - 1)
362                 moveq   r5, r5, lsr #PAGE_SHIFT - 12
363                 streq   r5, [sp, #4]
364                 beq     do_mmap2
365                 mov     r0, #-EINVAL
366                 RETINSTR(mov,pc, lr)
369  *  Design issues:
370  *   - We have several modes that each vector can be called from,
371  *     each with its own set of registers.  On entry to any vector,
372  *     we *must* save the registers used in *that* mode.
374  *   - This code must be as fast as possible.
376  *  There are a few restrictions on the vectors:
377  *   - the SWI vector cannot be called from *any* non-user mode
379  *   - the FP emulator is *never* called from *any* non-user mode undefined
380  *     instruction.
382  */
384                 .text
386                 .macro handle_irq
387 1:              mov     r4, #IOC_BASE
388                 ldrb    r6, [r4, #0x24]            @ get high priority first
389                 adr     r5, irq_prio_h
390                 teq     r6, #0
391                 ldreqb  r6, [r4, #0x14]            @ get low priority
392                 adreq   r5, irq_prio_l
394                 teq     r6, #0                     @ If an IRQ happened...
395                 ldrneb  r0, [r5, r6]               @ get IRQ number
396                 movne   r1, sp                     @ get struct pt_regs
397                 adrne   lr, 1b                     @ Set return address to 1b
398                 orrne   lr, lr, #PSR_I_BIT | MODE_SVC26  @ (and force SVC mode)
399                 bne     asm_do_IRQ                 @ process IRQ (if asserted)
400                 .endm
404  * Interrupt table (incorporates priority)
405  */
406                 .macro  irq_prio_table
407 irq_prio_l:     .byte    0, 0, 1, 0, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3
408                 .byte    4, 0, 1, 0, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3
409                 .byte    5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5
410                 .byte    5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5
411                 .byte    6, 6, 6, 6, 6, 6, 6, 6, 3, 3, 3, 3, 3, 3, 3, 3
412                 .byte    6, 6, 6, 6, 6, 6, 6, 6, 3, 3, 3, 3, 3, 3, 3, 3
413                 .byte    5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5
414                 .byte    5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5
415                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
416                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
417                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
418                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
419                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
420                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
421                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
422                 .byte    7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7
423 irq_prio_h:     .byte    0, 8, 9, 8,10,10,10,10,11,11,11,11,10,10,10,10
424                 .byte   12, 8, 9, 8,10,10,10,10,11,11,11,11,10,10,10,10
425                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
426                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
427                 .byte   14,14,14,14,10,10,10,10,11,11,11,11,10,10,10,10
428                 .byte   14,14,14,14,10,10,10,10,11,11,11,11,10,10,10,10
429                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
430                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
431                 .byte   15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10
432                 .byte   15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10
433                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
434                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
435                 .byte   15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10
436                 .byte   15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10
437                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
438                 .byte   13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10
439                 .endm
441 #if 1
443  * Uncomment these if you wish to get more debugging into about data aborts.
444  * FIXME - I bet we can find a way to encode these and keep performance.
445  */
446 #define FAULT_CODE_LDRSTRPOST   0x80
447 #define FAULT_CODE_LDRSTRPRE    0x40
448 #define FAULT_CODE_LDRSTRREG    0x20
449 #define FAULT_CODE_LDMSTM       0x10
450 #define FAULT_CODE_LDCSTC       0x08
451 #endif
452 #define FAULT_CODE_PREFETCH     0x04
453 #define FAULT_CODE_WRITE        0x02
454 #define FAULT_CODE_FORCECOW     0x01
456 /*=============================================================================
457  * Undefined FIQs
458  *-----------------------------------------------------------------------------
459  */
460 _unexp_fiq:     ldr     sp, .LCfiq
461                 mov     r12, #IOC_BASE
462                 strb    r12, [r12, #0x38]       @ Disable FIQ register
463                 teqp    pc, #PSR_I_BIT | PSR_F_BIT | MODE_SVC26
464                 mov     r0, r0
465                 stmfd   sp!, {r0 - r3, ip, lr}
466                 adr     r0, Lfiqmsg
467                 bl      printk
468                 ldmfd   sp!, {r0 - r3, ip, lr}
469                 teqp    pc, #PSR_I_BIT | PSR_F_BIT | MODE_FIQ26
470                 mov     r0, r0
471                 movs    pc, lr
473 Lfiqmsg:        .ascii  "*** Unexpected FIQ\n\0"
474                 .align
476 .LCfiq:         .word   __temp_fiq
477 .LCirq:         .word   __temp_irq
479 /*=============================================================================
480  * Undefined instruction handler
481  *-----------------------------------------------------------------------------
482  * Handles floating point instructions
483  */
484 vector_undefinstr:
485                 tst     lr, #MODE_SVC26          @ did we come from a non-user mode?
486                 bne     __und_svc                @ yes - deal with it.
487 /* Otherwise, fall through for the user-space (common) case. */
488                 save_user_regs
489                 zero_fp                                 @ zero frame pointer
490                 teqp    pc, #PSR_I_BIT | MODE_SVC26     @ disable IRQs
491 .Lbug_undef:
492                 ldr     r4, .LC2
493                 ldr     pc, [r4]         @ Call FP module entry point
494 /* FIXME - should we trap for a null pointer here? */
496 /* The SVC mode case */
497 __und_svc:      save_svc_regs                           @ Non-user mode
498                 mask_pc r0, lr
499                 and     r2, lr, #3
500                 sub     r0, r0, #4
501                 mov     r1, sp
502                 bl      do_undefinstr
503                 restore_svc_regs
505 /* We get here if the FP emulator doesnt handle the undef instr.
506  * If the insn WAS handled, the emulator jumps to ret_from_exception by itself/
507  */
508                 .globl  fpundefinstr 
509 fpundefinstr:
510                 mov     r0, lr
511                 mov     r1, sp
512                 teqp    pc, #MODE_SVC26
513                 bl      do_undefinstr
514                 b       ret_from_exception              @ Normal FP exit
516 #if defined CONFIG_FPE_NWFPE || defined CONFIG_FPE_FASTFPE
517                 /* The FPE is always present */
518                 .equ    fpe_not_present, 0
519 #else
520 /* We get here if an undefined instruction happens and the floating
521  * point emulator is not present.  If the offending instruction was
522  * a WFS, we just perform a normal return as if we had emulated the
523  * operation.  This is a hack to allow some basic userland binaries
524  * to run so that the emulator module proper can be loaded. --philb
525  * FIXME - probably a broken useless hack...
526  */
527 fpe_not_present:
528                 adr     r10, wfs_mask_data
529                 ldmia   r10, {r4, r5, r6, r7, r8}
530                 ldr     r10, [sp, #S_PC]                @ Load PC
531                 sub     r10, r10, #4
532                 mask_pc r10, r10
533                 ldrt    r10, [r10]                      @ get instruction
534                 and     r5, r10, r5
535                 teq     r5, r4                          @ Is it WFS?
536                 beq     ret_from_exception
537                 and     r5, r10, r8
538                 teq     r5, r6                          @ Is it LDF/STF on sp or fp?
539                 teqne   r5, r7
540                 bne     fpundefinstr
541                 tst     r10, #0x00200000                @ Does it have WB
542                 beq     ret_from_exception
543                 and     r4, r10, #255                   @ get offset
544                 and     r6, r10, #0x000f0000
545                 tst     r10, #0x00800000                @ +/-
546                 ldr     r5, [sp, r6, lsr #14]           @ Load reg
547                 rsbeq   r4, r4, #0
548                 add     r5, r5, r4, lsl #2
549                 str     r5, [sp, r6, lsr #14]           @ Save reg
550                 b       ret_from_exception
552 wfs_mask_data:  .word   0x0e200110                      @ WFS/RFS
553                 .word   0x0fef0fff
554                 .word   0x0d0d0100                      @ LDF [sp]/STF [sp]
555                 .word   0x0d0b0100                      @ LDF [fp]/STF [fp]
556                 .word   0x0f0f0f00
557 #endif
559 .LC2:           .word   fp_enter
561 /*=============================================================================
562  * Prefetch abort handler
563  *-----------------------------------------------------------------------------
564  */
565 #define DEBUG_UNDEF
566 /* remember: lr = USR pc */
567 vector_prefetch:
568                 sub     lr, lr, #4
569                 tst     lr, #MODE_SVC26
570                 bne     __pabt_invalid
571                 save_user_regs
572                 teqp    pc, #MODE_SVC26         @ Enable IRQs...
573                 mask_pc r0, lr                  @ Address of abort
574                 mov     r1, sp                  @ Tasks registers
575                 bl      do_PrefetchAbort
576                 teq     r0, #0                  @ If non-zero, we believe this abort..
577                 bne     ret_from_exception
578 #ifdef DEBUG_UNDEF
579                 adr     r0, t
580                 bl      printk
581 #endif
582                 ldr     lr, [sp,#S_PC]          @ FIXME program to test this on.  I think its
583                 b       .Lbug_undef             @ broken at the moment though!)
585 __pabt_invalid: save_svc_regs
586                 mov     r0, sp                  @ Prefetch aborts are definitely *not*
587                 mov     r1, #BAD_PREFETCH       @ allowed in non-user modes.  We cant
588                 and     r2, lr, #3              @ recover from this problem.
589                 b       bad_mode
591 #ifdef DEBUG_UNDEF
592 t:              .ascii "*** undef ***\r\n\0"
593                 .align
594 #endif
596 /*=============================================================================
597  * Address exception handler
598  *-----------------------------------------------------------------------------
599  * These aren't too critical.
600  * (they're not supposed to happen).
601  * In order to debug the reason for address exceptions in non-user modes,
602  * we have to obtain all the registers so that we can see what's going on.
603  */
605 vector_addrexcptn:
606                 sub     lr, lr, #8
607                 tst     lr, #3
608                 bne     Laddrexcptn_not_user
609                 save_user_regs
610                 teq     pc, #MODE_SVC26
611                 mask_pc r0, lr                  @ Point to instruction
612                 mov     r1, sp                  @ Point to registers
613                 mov     r2, #0x400
614                 mov     lr, pc
615                 bl      do_excpt
616                 b       ret_from_exception
618 Laddrexcptn_not_user:
619                 save_svc_regs
620                 and     r2, lr, #3
621                 teq     r2, #3
622                 bne     Laddrexcptn_illegal_mode
623                 teqp    pc, #MODE_SVC26
624                 mask_pc r0, lr
625                 mov     r1, sp
626                 orr     r2, r2, #0x400
627                 bl      do_excpt
628                 ldmia   sp, {r0 - lr}           @ I cant remember the reason I changed this...
629                 add     sp, sp, #15*4
630                 movs    pc, lr
632 Laddrexcptn_illegal_mode:
633                 mov     r0, sp
634                 str     lr, [sp, #-4]!
635                 orr     r1, r2, #PSR_I_BIT | PSR_F_BIT
636                 teqp    r1, #0                  @ change into mode (wont be user mode)
637                 mov     r0, r0
638                 mov     r1, r8                  @ Any register from r8 - r14 can be banked
639                 mov     r2, r9
640                 mov     r3, r10
641                 mov     r4, r11
642                 mov     r5, r12
643                 mov     r6, r13
644                 mov     r7, r14
645                 teqp    pc, #PSR_F_BIT | MODE_SVC26 @ back to svc
646                 mov     r0, r0
647                 stmfd   sp!, {r1-r7}
648                 ldmia   r0, {r0-r7}
649                 stmfd   sp!, {r0-r7}
650                 mov     r0, sp
651                 mov     r1, #BAD_ADDREXCPTN
652                 b       bad_mode
654 /*=============================================================================
655  * Interrupt (IRQ) handler
656  *-----------------------------------------------------------------------------
657  * Note: if the IRQ was taken whilst in user mode, then *no* kernel routine
658  * is running, so do not have to save svc lr.
660  * Entered in IRQ mode.
661  */
663 vector_IRQ:     ldr     sp, .LCirq         @ Setup some temporary stack
664                 sub     lr, lr, #4
665                 str     lr, [sp]           @ push return address
667                 tst     lr, #3
668                 bne     __irq_non_usr
670 __irq_usr:      teqp    pc, #PSR_I_BIT | MODE_SVC26     @ Enter SVC mode
671                 mov     r0, r0
673                 ldr     lr, .LCirq
674                 ldr     lr, [lr]           @ Restore lr for jump back to USR
676                 save_user_regs
678                 handle_irq
680                 mov     why, #0
681                 get_thread_info tsk
682                 b       ret_to_user
684 @ Place the IRQ priority table here so that the handle_irq macros above
685 @ and below here can access it.
687                 irq_prio_table
689 __irq_non_usr:  teqp    pc, #PSR_I_BIT | MODE_SVC26     @ Enter SVC mode
690                 mov     r0, r0
692                 save_svc_regs_irq
694                 and     r2, lr, #3
695                 teq     r2, #3
696                 bne     __irq_invalid                @ IRQ not from SVC mode
698                 handle_irq
700                 restore_svc_regs
702 __irq_invalid:  mov     r0, sp
703                 mov     r1, #BAD_IRQ
704                 b       bad_mode
706 /*=============================================================================
707  * Data abort handler code
708  *-----------------------------------------------------------------------------
710  * This handles both exceptions from user and SVC modes, computes the address
711  *  range of the problem, and does any correction that is required.  It then
712  *  calls the kernel data abort routine.
714  * This is where I wish that the ARM would tell you which address aborted.
715  */
717 vector_data:    sub     lr, lr, #8              @ Correct lr
718                 tst     lr, #3
719                 bne     Ldata_not_user
720                 save_user_regs
721                 teqp    pc, #MODE_SVC26
722                 mask_pc r0, lr
723                 bl      Ldata_do
724                 b       ret_from_exception
726 Ldata_not_user:
727                 save_svc_regs
728                 and     r2, lr, #3
729                 teq     r2, #3
730                 bne     Ldata_illegal_mode
731                 tst     lr, #PSR_I_BIT
732                 teqeqp  pc, #MODE_SVC26
733                 mask_pc r0, lr
734                 bl      Ldata_do
735                 restore_svc_regs
737 Ldata_illegal_mode:
738                 mov     r0, sp
739                 mov     r1, #BAD_DATA
740                 b       bad_mode
742 Ldata_do:       mov     r3, sp
743                 ldr     r4, [r0]                @ Get instruction
744                 mov     r2, #0
745                 tst     r4, #1 << 20            @ Check to see if it is a write instruction
746                 orreq   r2, r2, #FAULT_CODE_WRITE @ Indicate write instruction
747                 mov     r1, r4, lsr #22         @ Now branch to the relevent processing routine
748                 and     r1, r1, #15 << 2
749                 add     pc, pc, r1
750                 movs    pc, lr
751                 b       Ldata_unknown
752                 b       Ldata_unknown
753                 b       Ldata_unknown
754                 b       Ldata_unknown
755                 b       Ldata_ldrstr_post       @ ldr   rd, [rn], #m
756                 b       Ldata_ldrstr_numindex   @ ldr   rd, [rn, #m]    @ RegVal
757                 b       Ldata_ldrstr_post       @ ldr   rd, [rn], rm
758                 b       Ldata_ldrstr_regindex   @ ldr   rd, [rn, rm]
759                 b       Ldata_ldmstm            @ ldm*a rn, <rlist>
760                 b       Ldata_ldmstm            @ ldm*b rn, <rlist>
761                 b       Ldata_unknown
762                 b       Ldata_unknown
763                 b       Ldata_ldrstr_post       @ ldc   rd, [rn], #m    @ Same as ldr   rd, [rn], #m
764                 b       Ldata_ldcstc_pre        @ ldc   rd, [rn, #m]
765                 b       Ldata_unknown
766 Ldata_unknown:  @ Part of jumptable
767                 mov     r0, r1
768                 mov     r1, r4
769                 mov     r2, r3
770                 b       baddataabort
772 Ldata_ldrstr_post:
773                 mov     r0, r4, lsr #14         @ Get Rn
774                 and     r0, r0, #15 << 2        @ Mask out reg.
775                 teq     r0, #15 << 2
776                 ldr     r0, [r3, r0]            @ Get register
777                 biceq   r0, r0, #PCMASK
778                 mov     r1, r0
779 #ifdef FAULT_CODE_LDRSTRPOST
780                 orr     r2, r2, #FAULT_CODE_LDRSTRPOST
781 #endif
782                 b       do_DataAbort
784 Ldata_ldrstr_numindex:
785                 mov     r0, r4, lsr #14         @ Get Rn
786                 and     r0, r0, #15 << 2        @ Mask out reg.
787                 teq     r0, #15 << 2
788                 ldr     r0, [r3, r0]            @ Get register
789                 mov     r1, r4, lsl #20
790                 biceq   r0, r0, #PCMASK
791                 tst     r4, #1 << 23
792                 addne   r0, r0, r1, lsr #20
793                 subeq   r0, r0, r1, lsr #20
794                 mov     r1, r0
795 #ifdef FAULT_CODE_LDRSTRPRE
796                 orr     r2, r2, #FAULT_CODE_LDRSTRPRE
797 #endif
798                 b       do_DataAbort
800 Ldata_ldrstr_regindex:
801                 mov     r0, r4, lsr #14         @ Get Rn
802                 and     r0, r0, #15 << 2        @ Mask out reg.
803                 teq     r0, #15 << 2
804                 ldr     r0, [r3, r0]            @ Get register
805                 and     r7, r4, #15
806                 biceq   r0, r0, #PCMASK
807                 teq     r7, #15                 @ Check for PC
808                 ldr     r7, [r3, r7, lsl #2]    @ Get Rm
809                 and     r8, r4, #0x60           @ Get shift types
810                 biceq   r7, r7, #PCMASK
811                 mov     r9, r4, lsr #7          @ Get shift amount
812                 and     r9, r9, #31
813                 teq     r8, #0
814                 moveq   r7, r7, lsl r9
815                 teq     r8, #0x20               @ LSR shift
816                 moveq   r7, r7, lsr r9
817                 teq     r8, #0x40               @ ASR shift
818                 moveq   r7, r7, asr r9
819                 teq     r8, #0x60               @ ROR shift
820                 moveq   r7, r7, ror r9
821                 tst     r4, #1 << 23
822                 addne   r0, r0, r7
823                 subeq   r0, r0, r7              @ Apply correction
824                 mov     r1, r0
825 #ifdef FAULT_CODE_LDRSTRREG
826                 orr     r2, r2, #FAULT_CODE_LDRSTRREG
827 #endif
828                 b       do_DataAbort
830 Ldata_ldmstm:
831                 mov     r7, #0x11
832                 orr     r7, r7, r7, lsl #8
833                 and     r0, r4, r7
834                 and     r1, r4, r7, lsl #1
835                 add     r0, r0, r1, lsr #1
836                 and     r1, r4, r7, lsl #2
837                 add     r0, r0, r1, lsr #2
838                 and     r1, r4, r7, lsl #3
839                 add     r0, r0, r1, lsr #3
840                 add     r0, r0, r0, lsr #8
841                 add     r0, r0, r0, lsr #4
842                 and     r7, r0, #15             @ r7 = no. of registers to transfer.
843                 mov     r5, r4, lsr #14         @ Get Rn
844                 and     r5, r5, #15 << 2
845                 ldr     r0, [r3, r5]            @ Get reg
846                 eor     r6, r4, r4, lsl #2
847                 tst     r6, #1 << 23            @ Check inc/dec ^ writeback
848                 rsbeq   r7, r7, #0
849                 add     r7, r0, r7, lsl #2      @ Do correction (signed)
850                 subne   r1, r7, #1
851                 subeq   r1, r0, #1
852                 moveq   r0, r7
853                 tst     r4, #1 << 21            @ Check writeback
854                 strne   r7, [r3, r5]
855                 eor     r6, r4, r4, lsl #1
856                 tst     r6, #1 << 24            @ Check Pre/Post ^ inc/dec
857                 addeq   r0, r0, #4
858                 addeq   r1, r1, #4
859                 teq     r5, #15*4               @ CHECK FOR PC
860                 biceq   r1, r1, #PCMASK
861                 biceq   r0, r0, #PCMASK
862 #ifdef FAULT_CODE_LDMSTM
863                 orr     r2, r2, #FAULT_CODE_LDMSTM
864 #endif
865                 b       do_DataAbort
867 Ldata_ldcstc_pre:
868                 mov     r0, r4, lsr #14         @ Get Rn
869                 and     r0, r0, #15 << 2        @ Mask out reg.
870                 teq     r0, #15 << 2
871                 ldr     r0, [r3, r0]            @ Get register
872                 mov     r1, r4, lsl #24         @ Get offset
873                 biceq   r0, r0, #PCMASK
874                 tst     r4, #1 << 23
875                 addne   r0, r0, r1, lsr #24
876                 subeq   r0, r0, r1, lsr #24
877                 mov     r1, r0
878 #ifdef FAULT_CODE_LDCSTC
879                 orr     r2, r2, #FAULT_CODE_LDCSTC
880 #endif
881                 b       do_DataAbort
885  * This is the return code to user mode for abort handlers
886  */
887 ENTRY(ret_from_exception)
888                 get_thread_info tsk
889                 mov     why, #0
890                 b       ret_to_user
892                 .data
893 ENTRY(fp_enter)
894                 .word   fpe_not_present
895                 .text
897  * Register switch for older 26-bit only ARMs
898  */
899 ENTRY(__switch_to)
900                 add     r0, r0, #TI_CPU_SAVE
901                 stmia   r0, {r4 - sl, fp, sp, lr}
902                 add     r1, r1, #TI_CPU_SAVE
903                 ldmia   r1, {r4 - sl, fp, sp, pc}^
906  *=============================================================================
907  *              Low-level interface code
908  *-----------------------------------------------------------------------------
909  *              Trap initialisation
910  *-----------------------------------------------------------------------------
912  * Note - FIQ code has changed.  The default is a couple of words in 0x1c, 0x20
913  * that call _unexp_fiq.  Nowever, we now copy the FIQ routine to 0x1c (removes
914  * some excess cycles).
916  * What we need to put into 0-0x1c are branches to branch to the kernel.
917  */
919                 .section ".init.text",#alloc,#execinstr
921 .Ljump_addresses:
922                 swi     SYS_ERROR0
923                 .word   vector_undefinstr       - 12
924                 .word   vector_swi              - 16
925                 .word   vector_prefetch         - 20
926                 .word   vector_data             - 24
927                 .word   vector_addrexcptn       - 28
928                 .word   vector_IRQ              - 32
929                 .word   _unexp_fiq              - 36
930                 b       . + 8
932  * initialise the trap system
933  */
934 ENTRY(__trap_init)
935                 stmfd   sp!, {r4 - r7, lr}
936                 adr     r1, .Ljump_addresses
937                 ldmia   r1, {r1 - r7, ip, lr}
938                 orr     r2, lr, r2, lsr #2
939                 orr     r3, lr, r3, lsr #2
940                 orr     r4, lr, r4, lsr #2
941                 orr     r5, lr, r5, lsr #2
942                 orr     r6, lr, r6, lsr #2
943                 orr     r7, lr, r7, lsr #2
944                 orr     ip, lr, ip, lsr #2
945                 mov     r0, #0
946                 stmia   r0, {r1 - r7, ip}
947                 ldmfd   sp!, {r4 - r7, pc}^
949                 .bss
950 __temp_irq:     .space  4                               @ saved lr_irq
951 __temp_fiq:     .space  128