| /* |
| * arch/xtensa/include/asm/traps.h |
| * |
| * This file is subject to the terms and conditions of the GNU General Public |
| * License. See the file "COPYING" in the main directory of this archive |
| * for more details. |
| * |
| * Copyright (C) 2012 Tensilica Inc. |
| */ |
| #ifndef _XTENSA_TRAPS_H |
| #define _XTENSA_TRAPS_H |
| |
| #include <asm/ptrace.h> |
| |
| typedef void xtensa_exception_handler(struct pt_regs *regs); |
| |
| /* |
| * Per-CPU exception handling data structure. |
| * EXCSAVE1 points to it. |
| */ |
| struct exc_table { |
| /* Kernel Stack */ |
| void *kstk; |
| /* Double exception save area for a0 */ |
| unsigned long double_save; |
| /* Fixup handler */ |
| void *fixup; |
| /* For passing a parameter to fixup */ |
| void *fixup_param; |
| #if XTENSA_HAVE_COPROCESSORS |
| /* Pointers to owner struct thread_info */ |
| struct thread_info *coprocessor_owner[XCHAL_CP_MAX]; |
| #endif |
| /* Fast user exception handlers */ |
| void *fast_user_handler[EXCCAUSE_N]; |
| /* Fast kernel exception handlers */ |
| void *fast_kernel_handler[EXCCAUSE_N]; |
| /* Default C-Handlers */ |
| xtensa_exception_handler *default_handler[EXCCAUSE_N]; |
| }; |
| |
| DECLARE_PER_CPU(struct exc_table, exc_table); |
| |
| xtensa_exception_handler * |
| __init trap_set_handler(int cause, xtensa_exception_handler *handler); |
| |
| asmlinkage void fast_illegal_instruction_user(void); |
| asmlinkage void fast_syscall_user(void); |
| asmlinkage void fast_alloca(void); |
| asmlinkage void fast_load_store(void); |
| asmlinkage void fast_unaligned(void); |
| asmlinkage void fast_second_level_miss(void); |
| asmlinkage void fast_store_prohibited(void); |
| asmlinkage void fast_coprocessor(void); |
| |
| asmlinkage void kernel_exception(void); |
| asmlinkage void user_exception(void); |
| asmlinkage void system_call(struct pt_regs *regs); |
| |
| void do_IRQ(int hwirq, struct pt_regs *regs); |
| void do_page_fault(struct pt_regs *regs); |
| void do_unhandled(struct pt_regs *regs); |
| |
| /* Initialize minimal exc_table structure sufficient for basic paging */ |
| static inline void __init early_trap_init(void) |
| { |
| static struct exc_table init_exc_table __initdata = { |
| #ifdef CONFIG_XTENSA_LOAD_STORE |
| .fast_kernel_handler[EXCCAUSE_LOAD_STORE_ERROR] = |
| fast_load_store, |
| #endif |
| #ifdef CONFIG_MMU |
| .fast_kernel_handler[EXCCAUSE_DTLB_MISS] = |
| fast_second_level_miss, |
| #endif |
| }; |
| xtensa_set_sr(&init_exc_table, excsave1); |
| } |
| |
| void secondary_trap_init(void); |
| |
| static inline void spill_registers(void) |
| { |
| #if defined(__XTENSA_WINDOWED_ABI__) |
| #if XCHAL_NUM_AREGS > 16 |
| __asm__ __volatile__ ( |
| " call8 1f\n" |
| " _j 2f\n" |
| " retw\n" |
| " .align 4\n" |
| "1:\n" |
| #if XCHAL_NUM_AREGS == 32 |
| " _entry a1, 32\n" |
| " addi a8, a0, 3\n" |
| " _entry a1, 16\n" |
| " mov a12, a12\n" |
| " retw\n" |
| #else |
| " _entry a1, 48\n" |
| " call12 1f\n" |
| " retw\n" |
| " .align 4\n" |
| "1:\n" |
| " .rept (" __stringify(XCHAL_NUM_AREGS) " - 16) / 12\n" |
| " _entry a1, 48\n" |
| " mov a12, a0\n" |
| " .endr\n" |
| " _entry a1, 16\n" |
| #if XCHAL_NUM_AREGS % 12 == 0 |
| " mov a12, a12\n" |
| #elif XCHAL_NUM_AREGS % 12 == 4 |
| " mov a4, a4\n" |
| #elif XCHAL_NUM_AREGS % 12 == 8 |
| " mov a8, a8\n" |
| #endif |
| " retw\n" |
| #endif |
| "2:\n" |
| : : : "a8", "a9", "memory"); |
| #else |
| __asm__ __volatile__ ( |
| " mov a12, a12\n" |
| : : : "memory"); |
| #endif |
| #endif |
| } |
| |
| struct debug_table { |
| /* Pointer to debug exception handler */ |
| void (*debug_exception)(void); |
| /* Temporary register save area */ |
| unsigned long debug_save[1]; |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| /* Save area for DBREAKC registers */ |
| unsigned long dbreakc_save[XCHAL_NUM_DBREAK]; |
| /* Saved ICOUNT register */ |
| unsigned long icount_save; |
| /* Saved ICOUNTLEVEL register */ |
| unsigned long icount_level_save; |
| #endif |
| }; |
| |
| void debug_exception(void); |
| |
| #endif /* _XTENSA_TRAPS_H */ |