#endif
 }
 
+struct debug_table {
+       /* Pointer to debug exception handler */
+       void (*debug_exception)(void);
+       /* Temporary register save area */
+       unsigned long debug_save[1];
+};
+
+void debug_exception(void);
+
 #endif /* _XTENSA_TRAPS_H */
 
 #include <linux/kbuild.h>
 
 #include <asm/ptrace.h>
+#include <asm/traps.h>
 #include <asm/uaccess.h>
 
 int main(void)
        DEFINE(_CLONE_UNTRACED, CLONE_UNTRACED);
        DEFINE(PG_ARCH_1, PG_arch_1);
 
+       /* struct debug_table */
+       DEFINE(DT_DEBUG_EXCEPTION,
+              offsetof(struct debug_table, debug_exception));
+       DEFINE(DT_DEBUG_SAVE, offsetof(struct debug_table, debug_save));
+
        return 0;
 }
 
 
        movi    a2, 1 << PS_EXCM_BIT
        or      a2, a0, a2
-       movi    a0, debug_exception     # restore a3, debug jump vector
        wsr     a2, ps
-       xsr     a0, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
 
        /* Switch to kernel/user stack, restore jump vector, and save a0 */
 
        bbsi.l  a2, PS_UM_BIT, 2f       # jump if user mode
 
        addi    a2, a1, -16-PT_SIZE     # assume kernel stack
+3:
+       l32i    a0, a3, DT_DEBUG_SAVE
+       s32i    a1, a2, PT_AREG1
        s32i    a0, a2, PT_AREG0
        movi    a0, 0
-       s32i    a1, a2, PT_AREG1
        s32i    a0, a2, PT_DEPC         # mark it as a regular exception
+       xsr     a3, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
        xsr     a0, depc
        s32i    a3, a2, PT_AREG3
        s32i    a0, a2, PT_AREG2
        mov     a1, a2
+
+       rsr     a2, ps
+       bbsi.l  a2, PS_UM_BIT, _user_exception
        j       _kernel_exception
 
 2:     rsr     a2, excsave1
        l32i    a2, a2, EXC_TABLE_KSTK  # load kernel stack pointer
-       s32i    a0, a2, PT_AREG0
-       movi    a0, 0
-       s32i    a1, a2, PT_AREG1
-       s32i    a0, a2, PT_DEPC
-       xsr     a0, depc
-       s32i    a3, a2, PT_AREG3
-       s32i    a0, a2, PT_AREG2
-       mov     a1, a2
-       j       _user_exception
+       j       3b
 
        /* Debug exception while in exception mode. */
 1:     j       1b      // FIXME!!
 
        wsr     a2, ps                  # (enable reg-windows; progmode stack)
        rsync
 
-       /* Set up EXCSAVE[DEBUGLEVEL] to point to the Debug Exception Handler.*/
-
-       movi    a2, debug_exception
-       wsr     a2, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
-
 #ifdef CONFIG_SMP
        /*
         * Notice that we assume with SMP that cores have PRID
 
 
 DEFINE_PER_CPU(unsigned long, exc_table[EXC_TABLE_SIZE/4]);
 
+DEFINE_PER_CPU(struct debug_table, debug_table);
+
 void die(const char*, struct pt_regs*, long);
 
 static inline void
        __asm__ __volatile__("wsr  %0, excsave1\n" : : "a" (excsave1));
 }
 
+static void trap_init_debug(void)
+{
+       unsigned long debugsave = (unsigned long)this_cpu_ptr(&debug_table);
+
+       this_cpu_ptr(&debug_table)->debug_exception = debug_exception;
+       __asm__ __volatile__("wsr %0, excsave" __stringify(XCHAL_DEBUGLEVEL)
+                            :: "a"(debugsave));
+}
+
 /*
  * Initialize dispatch tables.
  *
 
        /* Initialize EXCSAVE_1 to hold the address of the exception table. */
        trap_init_excsave();
+       trap_init_debug();
 }
 
 #ifdef CONFIG_SMP
 void secondary_trap_init(void)
 {
        trap_init_excsave();
+       trap_init_debug();
 }
 #endif
 
 
 
 ENTRY(_DebugInterruptVector)
 
-       xsr     a0, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
+       xsr     a3, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
+       s32i    a0, a3, DT_DEBUG_SAVE
+       l32i    a0, a3, DT_DEBUG_EXCEPTION
        jx      a0
 
 ENDPROC(_DebugInterruptVector)