| /* arch/arm26/kernel/entry.S |
| * |
| * Assembled from chunks of code in arch/arm |
| * |
| * Copyright (C) 2003 Ian Molton |
| * Based on the work of RMK. |
| * |
| */ |
| |
| #include <linux/linkage.h> |
| |
| #include <asm/assembler.h> |
| #include <asm/asm_offsets.h> |
| #include <asm/errno.h> |
| #include <asm/hardware.h> |
| #include <asm/sysirq.h> |
| #include <asm/thread_info.h> |
| #include <asm/page.h> |
| #include <asm/ptrace.h> |
| |
| .macro zero_fp |
| #ifndef CONFIG_NO_FRAME_POINTER |
| mov fp, #0 |
| #endif |
| .endm |
| |
| .text |
| |
| @ Bad Abort numbers |
| @ ----------------- |
| @ |
| #define BAD_PREFETCH 0 |
| #define BAD_DATA 1 |
| #define BAD_ADDREXCPTN 2 |
| #define BAD_IRQ 3 |
| #define BAD_UNDEFINSTR 4 |
| |
| @ OS version number used in SWIs |
| @ RISC OS is 0 |
| @ RISC iX is 8 |
| @ |
| #define OS_NUMBER 9 |
| #define ARMSWI_OFFSET 0x000f0000 |
| |
| @ |
| @ Stack format (ensured by USER_* and SVC_*) |
| @ PSR and PC are comined on arm26 |
| @ |
| |
| #define S_OFF 8 |
| |
| #define S_OLD_R0 64 |
| #define S_PC 60 |
| #define S_LR 56 |
| #define S_SP 52 |
| #define S_IP 48 |
| #define S_FP 44 |
| #define S_R10 40 |
| #define S_R9 36 |
| #define S_R8 32 |
| #define S_R7 28 |
| #define S_R6 24 |
| #define S_R5 20 |
| #define S_R4 16 |
| #define S_R3 12 |
| #define S_R2 8 |
| #define S_R1 4 |
| #define S_R0 0 |
| |
| .macro save_user_regs |
| str r0, [sp, #-4]! @ Store SVC r0 |
| str lr, [sp, #-4]! @ Store user mode PC |
| sub sp, sp, #15*4 |
| stmia sp, {r0 - lr}^ @ Store the other user-mode regs |
| mov r0, r0 |
| .endm |
| |
| .macro slow_restore_user_regs |
| ldmia sp, {r0 - lr}^ @ restore the user regs not including PC |
| mov r0, r0 |
| ldr lr, [sp, #15*4] @ get user PC |
| add sp, sp, #15*4+8 @ free stack |
| movs pc, lr @ return |
| .endm |
| |
| .macro fast_restore_user_regs |
| add sp, sp, #S_OFF |
| ldmib sp, {r1 - lr}^ |
| mov r0, r0 |
| ldr lr, [sp, #15*4] |
| add sp, sp, #15*4+8 |
| movs pc, lr |
| .endm |
| |
| .macro save_svc_regs |
| str sp, [sp, #-16]! |
| str lr, [sp, #8] |
| str lr, [sp, #4] |
| stmfd sp!, {r0 - r12} |
| mov r0, #-1 |
| str r0, [sp, #S_OLD_R0] |
| zero_fp |
| .endm |
| |
| .macro save_svc_regs_irq |
| str sp, [sp, #-16]! |
| str lr, [sp, #4] |
| ldr lr, .LCirq |
| ldr lr, [lr] |
| str lr, [sp, #8] |
| stmfd sp!, {r0 - r12} |
| mov r0, #-1 |
| str r0, [sp, #S_OLD_R0] |
| zero_fp |
| .endm |
| |
| .macro restore_svc_regs |
| ldmfd sp, {r0 - pc}^ |
| .endm |
| |
| .macro mask_pc, rd, rm |
| bic \rd, \rm, #PCMASK |
| .endm |
| |
| .macro disable_irqs, temp |
| mov \temp, pc |
| orr \temp, \temp, #PSR_I_BIT |
| teqp \temp, #0 |
| .endm |
| |
| .macro enable_irqs, temp |
| mov \temp, pc |
| and \temp, \temp, #~PSR_I_BIT |
| teqp \temp, #0 |
| .endm |
| |
| .macro initialise_traps_extra |
| .endm |
| |
| .macro get_thread_info, rd |
| mov \rd, sp, lsr #13 |
| mov \rd, \rd, lsl #13 |
| .endm |
| |
| /* |
| * These are the registers used in the syscall handler, and allow us to |
| * have in theory up to 7 arguments to a function - r0 to r6. |
| * |
| * Note that tbl == why is intentional. |
| * |
| * We must set at least "tsk" and "why" when calling ret_with_reschedule. |
| */ |
| scno .req r7 @ syscall number |
| tbl .req r8 @ syscall table pointer |
| why .req r8 @ Linux syscall (!= 0) |
| tsk .req r9 @ current thread_info |
| |
| /* |
| * Get the system call number. |
| */ |
| .macro get_scno |
| mask_pc lr, lr |
| ldr scno, [lr, #-4] @ get SWI instruction |
| .endm |
| /* |
| * ----------------------------------------------------------------------- |
| */ |
| |
| /* |
| * We rely on the fact that R0 is at the bottom of the stack (due to |
| * slow/fast restore user regs). |
| */ |
| #if S_R0 != 0 |
| #error "Please fix" |
| #endif |
| |
| /* |
| * This is the fast syscall return path. We do as little as |
| * possible here, and this includes saving r0 back into the SVC |
| * stack. |
| */ |
| ret_fast_syscall: |
| disable_irqs r1 @ disable interrupts |
| ldr r1, [tsk, #TI_FLAGS] |
| tst r1, #_TIF_WORK_MASK |
| bne fast_work_pending |
| fast_restore_user_regs |
| |
| /* |
| * Ok, we need to do extra processing, enter the slow path. |
| */ |
| fast_work_pending: |
| str r0, [sp, #S_R0+S_OFF]! @ returned r0 |
| work_pending: |
| tst r1, #_TIF_NEED_RESCHED |
| bne work_resched |
| tst r1, #_TIF_NOTIFY_RESUME | _TIF_SIGPENDING |
| beq no_work_pending |
| mov r0, sp @ 'regs' |
| mov r2, why @ 'syscall' |
| bl do_notify_resume |
| disable_irqs r1 @ disable interrupts |
| b no_work_pending |
| |
| work_resched: |
| bl schedule |
| /* |
| * "slow" syscall return path. "why" tells us if this was a real syscall. |
| */ |
| ENTRY(ret_to_user) |
| ret_slow_syscall: |
| disable_irqs r1 @ disable interrupts |
| ldr r1, [tsk, #TI_FLAGS] |
| tst r1, #_TIF_WORK_MASK |
| bne work_pending |
| no_work_pending: |
| slow_restore_user_regs |
| |
| /* |
| * This is how we return from a fork. |
| */ |
| ENTRY(ret_from_fork) |
| bl schedule_tail |
| get_thread_info tsk |
| ldr r1, [tsk, #TI_FLAGS] @ check for syscall tracing |
| mov why, #1 |
| tst r1, #_TIF_SYSCALL_TRACE @ are we tracing syscalls? |
| beq ret_slow_syscall |
| mov r1, sp |
| mov r0, #1 @ trace exit [IP = 1] |
| bl syscall_trace |
| b ret_slow_syscall |
| |
| // FIXME - is this strictly necessary? |
| #include "calls.S" |
| |
| /*============================================================================= |
| * SWI handler |
| *----------------------------------------------------------------------------- |
| */ |
| |
| .align 5 |
| ENTRY(vector_swi) |
| save_user_regs |
| zero_fp |
| get_scno |
| |
| #ifdef CONFIG_ALIGNMENT_TRAP |
| ldr ip, __cr_alignment |
| ldr ip, [ip] |
| mcr p15, 0, ip, c1, c0 @ update control register |
| #endif |
| enable_irqs ip |
| |
| str r4, [sp, #-S_OFF]! @ push fifth arg |
| |
| get_thread_info tsk |
| ldr ip, [tsk, #TI_FLAGS] @ check for syscall tracing |
| bic scno, scno, #0xff000000 @ mask off SWI op-code |
| eor scno, scno, #OS_NUMBER << 20 @ check OS number |
| adr tbl, sys_call_table @ load syscall table pointer |
| tst ip, #_TIF_SYSCALL_TRACE @ are we tracing syscalls? |
| bne __sys_trace |
| |
| adral lr, ret_fast_syscall @ set return address |
| orral lr, lr, #PSR_I_BIT | MODE_SVC26 @ Force SVC mode on return |
| cmp scno, #NR_syscalls @ check upper syscall limit |
| ldrcc pc, [tbl, scno, lsl #2] @ call sys_* routine |
| |
| add r1, sp, #S_OFF |
| 2: mov why, #0 @ no longer a real syscall |
| cmp scno, #ARMSWI_OFFSET |
| eor r0, scno, #OS_NUMBER << 20 @ put OS number back |
| bcs arm_syscall |
| b sys_ni_syscall @ not private func |
| |
| /* |
| * This is the really slow path. We're going to be doing |
| * context switches, and waiting for our parent to respond. |
| */ |
| __sys_trace: |
| add r1, sp, #S_OFF |
| mov r0, #0 @ trace entry [IP = 0] |
| bl syscall_trace |
| |
| adral lr, __sys_trace_return @ set return address |
| orral lr, lr, #PSR_I_BIT | MODE_SVC26 @ Force SVC mode on return |
| add r1, sp, #S_R0 + S_OFF @ pointer to regs |
| cmp scno, #NR_syscalls @ check upper syscall limit |
| ldmccia r1, {r0 - r3} @ have to reload r0 - r3 |
| ldrcc pc, [tbl, scno, lsl #2] @ call sys_* routine |
| b 2b |
| |
| __sys_trace_return: |
| str r0, [sp, #S_R0 + S_OFF]! @ save returned r0 |
| mov r1, sp |
| mov r0, #1 @ trace exit [IP = 1] |
| bl syscall_trace |
| b ret_slow_syscall |
| |
| .align 5 |
| #ifdef CONFIG_ALIGNMENT_TRAP |
| .type __cr_alignment, #object |
| __cr_alignment: |
| .word cr_alignment |
| #endif |
| |
| .type sys_call_table, #object |
| ENTRY(sys_call_table) |
| #include "calls.S" |
| |
| /*============================================================================ |
| * Special system call wrappers |
| */ |
| @ r0 = syscall number |
| @ r5 = syscall table |
| .type sys_syscall, #function |
| sys_syscall: |
| eor scno, r0, #OS_NUMBER << 20 |
| cmp scno, #NR_syscalls @ check range |
| stmleia sp, {r5, r6} @ shuffle args |
| movle r0, r1 |
| movle r1, r2 |
| movle r2, r3 |
| movle r3, r4 |
| ldrle pc, [tbl, scno, lsl #2] |
| b sys_ni_syscall |
| |
| sys_fork_wrapper: |
| add r0, sp, #S_OFF |
| b sys_fork |
| |
| sys_vfork_wrapper: |
| add r0, sp, #S_OFF |
| b sys_vfork |
| |
| sys_execve_wrapper: |
| add r3, sp, #S_OFF |
| b sys_execve |
| |
| sys_clone_wapper: |
| add r2, sp, #S_OFF |
| b sys_clone |
| |
| sys_sigsuspend_wrapper: |
| add r3, sp, #S_OFF |
| b sys_sigsuspend |
| |
| sys_rt_sigsuspend_wrapper: |
| add r2, sp, #S_OFF |
| b sys_rt_sigsuspend |
| |
| sys_sigreturn_wrapper: |
| add r0, sp, #S_OFF |
| b sys_sigreturn |
| |
| sys_rt_sigreturn_wrapper: |
| add r0, sp, #S_OFF |
| b sys_rt_sigreturn |
| |
| sys_sigaltstack_wrapper: |
| ldr r2, [sp, #S_OFF + S_SP] |
| b do_sigaltstack |
| |
| /* |
| * Note: off_4k (r5) is always units of 4K. If we can't do the requested |
| * offset, we return EINVAL. FIXME - this lost some stuff from arm32 to |
| * ifdefs. check it out. |
| */ |
| sys_mmap2: |
| tst r5, #((1 << (PAGE_SHIFT - 12)) - 1) |
| moveq r5, r5, lsr #PAGE_SHIFT - 12 |
| streq r5, [sp, #4] |
| beq do_mmap2 |
| mov r0, #-EINVAL |
| RETINSTR(mov,pc, lr) |
| |
| /* |
| * Design issues: |
| * - We have several modes that each vector can be called from, |
| * each with its own set of registers. On entry to any vector, |
| * we *must* save the registers used in *that* mode. |
| * |
| * - This code must be as fast as possible. |
| * |
| * There are a few restrictions on the vectors: |
| * - the SWI vector cannot be called from *any* non-user mode |
| * |
| * - the FP emulator is *never* called from *any* non-user mode undefined |
| * instruction. |
| * |
| */ |
| |
| .text |
| |
| .macro handle_irq |
| 1: mov r4, #IOC_BASE |
| ldrb r6, [r4, #0x24] @ get high priority first |
| adr r5, irq_prio_h |
| teq r6, #0 |
| ldreqb r6, [r4, #0x14] @ get low priority |
| adreq r5, irq_prio_l |
| |
| teq r6, #0 @ If an IRQ happened... |
| ldrneb r0, [r5, r6] @ get IRQ number |
| movne r1, sp @ get struct pt_regs |
| adrne lr, 1b @ Set return address to 1b |
| orrne lr, lr, #PSR_I_BIT | MODE_SVC26 @ (and force SVC mode) |
| bne asm_do_IRQ @ process IRQ (if asserted) |
| .endm |
| |
| |
| /* |
| * Interrupt table (incorporates priority) |
| */ |
| .macro irq_prio_table |
| irq_prio_l: .byte 0, 0, 1, 0, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3 |
| .byte 4, 0, 1, 0, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3 |
| .byte 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5 |
| .byte 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5 |
| .byte 6, 6, 6, 6, 6, 6, 6, 6, 3, 3, 3, 3, 3, 3, 3, 3 |
| .byte 6, 6, 6, 6, 6, 6, 6, 6, 3, 3, 3, 3, 3, 3, 3, 3 |
| .byte 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5 |
| .byte 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| .byte 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7 |
| irq_prio_h: .byte 0, 8, 9, 8,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 12, 8, 9, 8,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 14,14,14,14,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 14,14,14,14,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 15,15,15,15,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .byte 13,13,13,13,10,10,10,10,11,11,11,11,10,10,10,10 |
| .endm |
| |
| #if 1 |
| /* |
| * Uncomment these if you wish to get more debugging into about data aborts. |
| * FIXME - I bet we can find a way to encode these and keep performance. |
| */ |
| #define FAULT_CODE_LDRSTRPOST 0x80 |
| #define FAULT_CODE_LDRSTRPRE 0x40 |
| #define FAULT_CODE_LDRSTRREG 0x20 |
| #define FAULT_CODE_LDMSTM 0x10 |
| #define FAULT_CODE_LDCSTC 0x08 |
| #endif |
| #define FAULT_CODE_PREFETCH 0x04 |
| #define FAULT_CODE_WRITE 0x02 |
| #define FAULT_CODE_FORCECOW 0x01 |
| |
| /*============================================================================= |
| * Undefined FIQs |
| *----------------------------------------------------------------------------- |
| */ |
| _unexp_fiq: ldr sp, .LCfiq |
| mov r12, #IOC_BASE |
| strb r12, [r12, #0x38] @ Disable FIQ register |
| teqp pc, #PSR_I_BIT | PSR_F_BIT | MODE_SVC26 |
| mov r0, r0 |
| stmfd sp!, {r0 - r3, ip, lr} |
| adr r0, Lfiqmsg |
| bl printk |
| ldmfd sp!, {r0 - r3, ip, lr} |
| teqp pc, #PSR_I_BIT | PSR_F_BIT | MODE_FIQ26 |
| mov r0, r0 |
| movs pc, lr |
| |
| Lfiqmsg: .ascii "*** Unexpected FIQ\n\0" |
| .align |
| |
| .LCfiq: .word __temp_fiq |
| .LCirq: .word __temp_irq |
| |
| /*============================================================================= |
| * Undefined instruction handler |
| *----------------------------------------------------------------------------- |
| * Handles floating point instructions |
| */ |
| vector_undefinstr: |
| tst lr, #MODE_SVC26 @ did we come from a non-user mode? |
| bne __und_svc @ yes - deal with it. |
| /* Otherwise, fall through for the user-space (common) case. */ |
| save_user_regs |
| zero_fp @ zero frame pointer |
| teqp pc, #PSR_I_BIT | MODE_SVC26 @ disable IRQs |
| .Lbug_undef: |
| ldr r4, .LC2 |
| ldr pc, [r4] @ Call FP module entry point |
| /* FIXME - should we trap for a null pointer here? */ |
| |
| /* The SVC mode case */ |
| __und_svc: save_svc_regs @ Non-user mode |
| mask_pc r0, lr |
| and r2, lr, #3 |
| sub r0, r0, #4 |
| mov r1, sp |
| bl do_undefinstr |
| restore_svc_regs |
| |
| /* We get here if the FP emulator doesnt handle the undef instr. |
| * If the insn WAS handled, the emulator jumps to ret_from_exception by itself/ |
| */ |
| .globl fpundefinstr |
| fpundefinstr: |
| mov r0, lr |
| mov r1, sp |
| teqp pc, #MODE_SVC26 |
| bl do_undefinstr |
| b ret_from_exception @ Normal FP exit |
| |
| #if defined CONFIG_FPE_NWFPE || defined CONFIG_FPE_FASTFPE |
| /* The FPE is always present */ |
| .equ fpe_not_present, 0 |
| #else |
| /* We get here if an undefined instruction happens and the floating |
| * point emulator is not present. If the offending instruction was |
| * a WFS, we just perform a normal return as if we had emulated the |
| * operation. This is a hack to allow some basic userland binaries |
| * to run so that the emulator module proper can be loaded. --philb |
| * FIXME - probably a broken useless hack... |
| */ |
| fpe_not_present: |
| adr r10, wfs_mask_data |
| ldmia r10, {r4, r5, r6, r7, r8} |
| ldr r10, [sp, #S_PC] @ Load PC |
| sub r10, r10, #4 |
| mask_pc r10, r10 |
| ldrt r10, [r10] @ get instruction |
| and r5, r10, r5 |
| teq r5, r4 @ Is it WFS? |
| beq ret_from_exception |
| and r5, r10, r8 |
| teq r5, r6 @ Is it LDF/STF on sp or fp? |
| teqne r5, r7 |
| bne fpundefinstr |
| tst r10, #0x00200000 @ Does it have WB |
| beq ret_from_exception |
| and r4, r10, #255 @ get offset |
| and r6, r10, #0x000f0000 |
| tst r10, #0x00800000 @ +/- |
| ldr r5, [sp, r6, lsr #14] @ Load reg |
| rsbeq r4, r4, #0 |
| add r5, r5, r4, lsl #2 |
| str r5, [sp, r6, lsr #14] @ Save reg |
| b ret_from_exception |
| |
| wfs_mask_data: .word 0x0e200110 @ WFS/RFS |
| .word 0x0fef0fff |
| .word 0x0d0d0100 @ LDF [sp]/STF [sp] |
| .word 0x0d0b0100 @ LDF [fp]/STF [fp] |
| .word 0x0f0f0f00 |
| #endif |
| |
| .LC2: .word fp_enter |
| |
| /*============================================================================= |
| * Prefetch abort handler |
| *----------------------------------------------------------------------------- |
| */ |
| #define DEBUG_UNDEF |
| /* remember: lr = USR pc */ |
| vector_prefetch: |
| sub lr, lr, #4 |
| tst lr, #MODE_SVC26 |
| bne __pabt_invalid |
| save_user_regs |
| teqp pc, #MODE_SVC26 @ Enable IRQs... |
| mask_pc r0, lr @ Address of abort |
| mov r1, sp @ Tasks registers |
| bl do_PrefetchAbort |
| teq r0, #0 @ If non-zero, we believe this abort.. |
| bne ret_from_exception |
| #ifdef DEBUG_UNDEF |
| adr r0, t |
| bl printk |
| #endif |
| ldr lr, [sp,#S_PC] @ FIXME program to test this on. I think its |
| b .Lbug_undef @ broken at the moment though!) |
| |
| __pabt_invalid: save_svc_regs |
| mov r0, sp @ Prefetch aborts are definitely *not* |
| mov r1, #BAD_PREFETCH @ allowed in non-user modes. We cant |
| and r2, lr, #3 @ recover from this problem. |
| b bad_mode |
| |
| #ifdef DEBUG_UNDEF |
| t: .ascii "*** undef ***\r\n\0" |
| .align |
| #endif |
| |
| /*============================================================================= |
| * Address exception handler |
| *----------------------------------------------------------------------------- |
| * These aren't too critical. |
| * (they're not supposed to happen). |
| * In order to debug the reason for address exceptions in non-user modes, |
| * we have to obtain all the registers so that we can see what's going on. |
| */ |
| |
| vector_addrexcptn: |
| sub lr, lr, #8 |
| tst lr, #3 |
| bne Laddrexcptn_not_user |
| save_user_regs |
| teq pc, #MODE_SVC26 |
| mask_pc r0, lr @ Point to instruction |
| mov r1, sp @ Point to registers |
| mov r2, #0x400 |
| mov lr, pc |
| bl do_excpt |
| b ret_from_exception |
| |
| Laddrexcptn_not_user: |
| save_svc_regs |
| and r2, lr, #3 |
| teq r2, #3 |
| bne Laddrexcptn_illegal_mode |
| teqp pc, #MODE_SVC26 |
| mask_pc r0, lr |
| mov r1, sp |
| orr r2, r2, #0x400 |
| bl do_excpt |
| ldmia sp, {r0 - lr} @ I cant remember the reason I changed this... |
| add sp, sp, #15*4 |
| movs pc, lr |
| |
| Laddrexcptn_illegal_mode: |
| mov r0, sp |
| str lr, [sp, #-4]! |
| orr r1, r2, #PSR_I_BIT | PSR_F_BIT |
| teqp r1, #0 @ change into mode (wont be user mode) |
| mov r0, r0 |
| mov r1, r8 @ Any register from r8 - r14 can be banked |
| mov r2, r9 |
| mov r3, r10 |
| mov r4, r11 |
| mov r5, r12 |
| mov r6, r13 |
| mov r7, r14 |
| teqp pc, #PSR_F_BIT | MODE_SVC26 @ back to svc |
| mov r0, r0 |
| stmfd sp!, {r1-r7} |
| ldmia r0, {r0-r7} |
| stmfd sp!, {r0-r7} |
| mov r0, sp |
| mov r1, #BAD_ADDREXCPTN |
| b bad_mode |
| |
| /*============================================================================= |
| * Interrupt (IRQ) handler |
| *----------------------------------------------------------------------------- |
| * Note: if the IRQ was taken whilst in user mode, then *no* kernel routine |
| * is running, so do not have to save svc lr. |
| * |
| * Entered in IRQ mode. |
| */ |
| |
| vector_IRQ: ldr sp, .LCirq @ Setup some temporary stack |
| sub lr, lr, #4 |
| str lr, [sp] @ push return address |
| |
| tst lr, #3 |
| bne __irq_non_usr |
| |
| __irq_usr: teqp pc, #PSR_I_BIT | MODE_SVC26 @ Enter SVC mode |
| mov r0, r0 |
| |
| ldr lr, .LCirq |
| ldr lr, [lr] @ Restore lr for jump back to USR |
| |
| save_user_regs |
| |
| handle_irq |
| |
| mov why, #0 |
| get_thread_info tsk |
| b ret_to_user |
| |
| @ Place the IRQ priority table here so that the handle_irq macros above |
| @ and below here can access it. |
| |
| irq_prio_table |
| |
| __irq_non_usr: teqp pc, #PSR_I_BIT | MODE_SVC26 @ Enter SVC mode |
| mov r0, r0 |
| |
| save_svc_regs_irq |
| |
| and r2, lr, #3 |
| teq r2, #3 |
| bne __irq_invalid @ IRQ not from SVC mode |
| |
| handle_irq |
| |
| restore_svc_regs |
| |
| __irq_invalid: mov r0, sp |
| mov r1, #BAD_IRQ |
| b bad_mode |
| |
| /*============================================================================= |
| * Data abort handler code |
| *----------------------------------------------------------------------------- |
| * |
| * This handles both exceptions from user and SVC modes, computes the address |
| * range of the problem, and does any correction that is required. It then |
| * calls the kernel data abort routine. |
| * |
| * This is where I wish that the ARM would tell you which address aborted. |
| */ |
| |
| vector_data: sub lr, lr, #8 @ Correct lr |
| tst lr, #3 |
| bne Ldata_not_user |
| save_user_regs |
| teqp pc, #MODE_SVC26 |
| mask_pc r0, lr |
| bl Ldata_do |
| b ret_from_exception |
| |
| Ldata_not_user: |
| save_svc_regs |
| and r2, lr, #3 |
| teq r2, #3 |
| bne Ldata_illegal_mode |
| tst lr, #PSR_I_BIT |
| teqeqp pc, #MODE_SVC26 |
| mask_pc r0, lr |
| bl Ldata_do |
| restore_svc_regs |
| |
| Ldata_illegal_mode: |
| mov r0, sp |
| mov r1, #BAD_DATA |
| b bad_mode |
| |
| Ldata_do: mov r3, sp |
| ldr r4, [r0] @ Get instruction |
| mov r2, #0 |
| tst r4, #1 << 20 @ Check to see if it is a write instruction |
| orreq r2, r2, #FAULT_CODE_WRITE @ Indicate write instruction |
| mov r1, r4, lsr #22 @ Now branch to the relevent processing routine |
| and r1, r1, #15 << 2 |
| add pc, pc, r1 |
| movs pc, lr |
| b Ldata_unknown |
| b Ldata_unknown |
| b Ldata_unknown |
| b Ldata_unknown |
| b Ldata_ldrstr_post @ ldr rd, [rn], #m |
| b Ldata_ldrstr_numindex @ ldr rd, [rn, #m] @ RegVal |
| b Ldata_ldrstr_post @ ldr rd, [rn], rm |
| b Ldata_ldrstr_regindex @ ldr rd, [rn, rm] |
| b Ldata_ldmstm @ ldm*a rn, <rlist> |
| b Ldata_ldmstm @ ldm*b rn, <rlist> |
| b Ldata_unknown |
| b Ldata_unknown |
| b Ldata_ldrstr_post @ ldc rd, [rn], #m @ Same as ldr rd, [rn], #m |
| b Ldata_ldcstc_pre @ ldc rd, [rn, #m] |
| b Ldata_unknown |
| Ldata_unknown: @ Part of jumptable |
| mov r0, r1 |
| mov r1, r4 |
| mov r2, r3 |
| b baddataabort |
| |
| Ldata_ldrstr_post: |
| mov r0, r4, lsr #14 @ Get Rn |
| and r0, r0, #15 << 2 @ Mask out reg. |
| teq r0, #15 << 2 |
| ldr r0, [r3, r0] @ Get register |
| biceq r0, r0, #PCMASK |
| mov r1, r0 |
| #ifdef FAULT_CODE_LDRSTRPOST |
| orr r2, r2, #FAULT_CODE_LDRSTRPOST |
| #endif |
| b do_DataAbort |
| |
| Ldata_ldrstr_numindex: |
| mov r0, r4, lsr #14 @ Get Rn |
| and r0, r0, #15 << 2 @ Mask out reg. |
| teq r0, #15 << 2 |
| ldr r0, [r3, r0] @ Get register |
| mov r1, r4, lsl #20 |
| biceq r0, r0, #PCMASK |
| tst r4, #1 << 23 |
| addne r0, r0, r1, lsr #20 |
| subeq r0, r0, r1, lsr #20 |
| mov r1, r0 |
| #ifdef FAULT_CODE_LDRSTRPRE |
| orr r2, r2, #FAULT_CODE_LDRSTRPRE |
| #endif |
| b do_DataAbort |
| |
| Ldata_ldrstr_regindex: |
| mov r0, r4, lsr #14 @ Get Rn |
| and r0, r0, #15 << 2 @ Mask out reg. |
| teq r0, #15 << 2 |
| ldr r0, [r3, r0] @ Get register |
| and r7, r4, #15 |
| biceq r0, r0, #PCMASK |
| teq r7, #15 @ Check for PC |
| ldr r7, [r3, r7, lsl #2] @ Get Rm |
| and r8, r4, #0x60 @ Get shift types |
| biceq r7, r7, #PCMASK |
| mov r9, r4, lsr #7 @ Get shift amount |
| and r9, r9, #31 |
| teq r8, #0 |
| moveq r7, r7, lsl r9 |
| teq r8, #0x20 @ LSR shift |
| moveq r7, r7, lsr r9 |
| teq r8, #0x40 @ ASR shift |
| moveq r7, r7, asr r9 |
| teq r8, #0x60 @ ROR shift |
| moveq r7, r7, ror r9 |
| tst r4, #1 << 23 |
| addne r0, r0, r7 |
| subeq r0, r0, r7 @ Apply correction |
| mov r1, r0 |
| #ifdef FAULT_CODE_LDRSTRREG |
| orr r2, r2, #FAULT_CODE_LDRSTRREG |
| #endif |
| b do_DataAbort |
| |
| Ldata_ldmstm: |
| mov r7, #0x11 |
| orr r7, r7, r7, lsl #8 |
| and r0, r4, r7 |
| and r1, r4, r7, lsl #1 |
| add r0, r0, r1, lsr #1 |
| and r1, r4, r7, lsl #2 |
| add r0, r0, r1, lsr #2 |
| and r1, r4, r7, lsl #3 |
| add r0, r0, r1, lsr #3 |
| add r0, r0, r0, lsr #8 |
| add r0, r0, r0, lsr #4 |
| and r7, r0, #15 @ r7 = no. of registers to transfer. |
| mov r5, r4, lsr #14 @ Get Rn |
| and r5, r5, #15 << 2 |
| ldr r0, [r3, r5] @ Get reg |
| eor r6, r4, r4, lsl #2 |
| tst r6, #1 << 23 @ Check inc/dec ^ writeback |
| rsbeq r7, r7, #0 |
| add r7, r0, r7, lsl #2 @ Do correction (signed) |
| subne r1, r7, #1 |
| subeq r1, r0, #1 |
| moveq r0, r7 |
| tst r4, #1 << 21 @ Check writeback |
| strne r7, [r3, r5] |
| eor r6, r4, r4, lsl #1 |
| tst r6, #1 << 24 @ Check Pre/Post ^ inc/dec |
| addeq r0, r0, #4 |
| addeq r1, r1, #4 |
| teq r5, #15*4 @ CHECK FOR PC |
| biceq r1, r1, #PCMASK |
| biceq r0, r0, #PCMASK |
| #ifdef FAULT_CODE_LDMSTM |
| orr r2, r2, #FAULT_CODE_LDMSTM |
| #endif |
| b do_DataAbort |
| |
| Ldata_ldcstc_pre: |
| mov r0, r4, lsr #14 @ Get Rn |
| and r0, r0, #15 << 2 @ Mask out reg. |
| teq r0, #15 << 2 |
| ldr r0, [r3, r0] @ Get register |
| mov r1, r4, lsl #24 @ Get offset |
| biceq r0, r0, #PCMASK |
| tst r4, #1 << 23 |
| addne r0, r0, r1, lsr #24 |
| subeq r0, r0, r1, lsr #24 |
| mov r1, r0 |
| #ifdef FAULT_CODE_LDCSTC |
| orr r2, r2, #FAULT_CODE_LDCSTC |
| #endif |
| b do_DataAbort |
| |
| |
| /* |
| * This is the return code to user mode for abort handlers |
| */ |
| ENTRY(ret_from_exception) |
| get_thread_info tsk |
| mov why, #0 |
| b ret_to_user |
| |
| .data |
| ENTRY(fp_enter) |
| .word fpe_not_present |
| .text |
| /* |
| * Register switch for older 26-bit only ARMs |
| */ |
| ENTRY(__switch_to) |
| add r0, r0, #TI_CPU_SAVE |
| stmia r0, {r4 - sl, fp, sp, lr} |
| add r1, r1, #TI_CPU_SAVE |
| ldmia r1, {r4 - sl, fp, sp, pc}^ |
| |
| /* |
| *============================================================================= |
| * Low-level interface code |
| *----------------------------------------------------------------------------- |
| * Trap initialisation |
| *----------------------------------------------------------------------------- |
| * |
| * Note - FIQ code has changed. The default is a couple of words in 0x1c, 0x20 |
| * that call _unexp_fiq. Nowever, we now copy the FIQ routine to 0x1c (removes |
| * some excess cycles). |
| * |
| * What we need to put into 0-0x1c are branches to branch to the kernel. |
| */ |
| |
| .section ".init.text",#alloc,#execinstr |
| |
| .Ljump_addresses: |
| swi SYS_ERROR0 |
| .word vector_undefinstr - 12 |
| .word vector_swi - 16 |
| .word vector_prefetch - 20 |
| .word vector_data - 24 |
| .word vector_addrexcptn - 28 |
| .word vector_IRQ - 32 |
| .word _unexp_fiq - 36 |
| b . + 8 |
| /* |
| * initialise the trap system |
| */ |
| ENTRY(__trap_init) |
| stmfd sp!, {r4 - r7, lr} |
| adr r1, .Ljump_addresses |
| ldmia r1, {r1 - r7, ip, lr} |
| orr r2, lr, r2, lsr #2 |
| orr r3, lr, r3, lsr #2 |
| orr r4, lr, r4, lsr #2 |
| orr r5, lr, r5, lsr #2 |
| orr r6, lr, r6, lsr #2 |
| orr r7, lr, r7, lsr #2 |
| orr ip, lr, ip, lsr #2 |
| mov r0, #0 |
| stmia r0, {r1 - r7, ip} |
| ldmfd sp!, {r4 - r7, pc}^ |
| |
| .bss |
| __temp_irq: .space 4 @ saved lr_irq |
| __temp_fiq: .space 128 |