提交 6ec7026a 编写于 作者: M Max Filippov 提交者: Chris Zankel

xtensa: use context structure for debug exceptions

With implementation of data breakpoints debug exceptions raised when
PS.EXCM is set need to be handled, e.g. window overflow code can write
to watched userspace address. Currently debug exception handler uses
EXCSAVE and DEPC SRs to save temporary registers, but DEPC may not be
available when PS.EXCM is set and more space will be needed to save
additional state.
Reorganize debug context: create per-CPU structure debug_table instance
and store its address in the EXCSAVE<debug level> instead of
debug_exception function address. Expand this structure when more save
space is needed.
Signed-off-by: NMax Filippov <jcmvbkbc@gmail.com>
上级 816aa588
...@@ -65,4 +65,13 @@ static inline void spill_registers(void) ...@@ -65,4 +65,13 @@ static inline void spill_registers(void)
#endif #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 */ #endif /* _XTENSA_TRAPS_H */
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include <linux/kbuild.h> #include <linux/kbuild.h>
#include <asm/ptrace.h> #include <asm/ptrace.h>
#include <asm/traps.h>
#include <asm/uaccess.h> #include <asm/uaccess.h>
int main(void) int main(void)
...@@ -117,5 +118,10 @@ int main(void) ...@@ -117,5 +118,10 @@ int main(void)
DEFINE(_CLONE_UNTRACED, CLONE_UNTRACED); DEFINE(_CLONE_UNTRACED, CLONE_UNTRACED);
DEFINE(PG_ARCH_1, PG_arch_1); 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; return 0;
} }
...@@ -789,36 +789,32 @@ ENTRY(debug_exception) ...@@ -789,36 +789,32 @@ ENTRY(debug_exception)
movi a2, 1 << PS_EXCM_BIT movi a2, 1 << PS_EXCM_BIT
or a2, a0, a2 or a2, a0, a2
movi a0, debug_exception # restore a3, debug jump vector
wsr a2, ps wsr a2, ps
xsr a0, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
/* Switch to kernel/user stack, restore jump vector, and save a0 */ /* Switch to kernel/user stack, restore jump vector, and save a0 */
bbsi.l a2, PS_UM_BIT, 2f # jump if user mode bbsi.l a2, PS_UM_BIT, 2f # jump if user mode
addi a2, a1, -16-PT_SIZE # assume kernel stack 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 s32i a0, a2, PT_AREG0
movi a0, 0 movi a0, 0
s32i a1, a2, PT_AREG1
s32i a0, a2, PT_DEPC # mark it as a regular exception s32i a0, a2, PT_DEPC # mark it as a regular exception
xsr a3, SREG_EXCSAVE + XCHAL_DEBUGLEVEL
xsr a0, depc xsr a0, depc
s32i a3, a2, PT_AREG3 s32i a3, a2, PT_AREG3
s32i a0, a2, PT_AREG2 s32i a0, a2, PT_AREG2
mov a1, a2 mov a1, a2
rsr a2, ps
bbsi.l a2, PS_UM_BIT, _user_exception
j _kernel_exception j _kernel_exception
2: rsr a2, excsave1 2: rsr a2, excsave1
l32i a2, a2, EXC_TABLE_KSTK # load kernel stack pointer l32i a2, a2, EXC_TABLE_KSTK # load kernel stack pointer
s32i a0, a2, PT_AREG0 j 3b
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
/* Debug exception while in exception mode. */ /* Debug exception while in exception mode. */
1: j 1b // FIXME!! 1: j 1b // FIXME!!
......
...@@ -197,11 +197,6 @@ ENTRY(_startup) ...@@ -197,11 +197,6 @@ ENTRY(_startup)
wsr a2, ps # (enable reg-windows; progmode stack) wsr a2, ps # (enable reg-windows; progmode stack)
rsync 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 #ifdef CONFIG_SMP
/* /*
* Notice that we assume with SMP that cores have PRID * Notice that we assume with SMP that cores have PRID
......
...@@ -157,6 +157,8 @@ COPROCESSOR(7), ...@@ -157,6 +157,8 @@ COPROCESSOR(7),
DEFINE_PER_CPU(unsigned long, exc_table[EXC_TABLE_SIZE/4]); 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); void die(const char*, struct pt_regs*, long);
static inline void static inline void
...@@ -372,6 +374,15 @@ static void trap_init_excsave(void) ...@@ -372,6 +374,15 @@ static void trap_init_excsave(void)
__asm__ __volatile__("wsr %0, excsave1\n" : : "a" (excsave1)); __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 dispatch tables.
* *
...@@ -415,12 +426,14 @@ void __init trap_init(void) ...@@ -415,12 +426,14 @@ void __init trap_init(void)
/* Initialize EXCSAVE_1 to hold the address of the exception table. */ /* Initialize EXCSAVE_1 to hold the address of the exception table. */
trap_init_excsave(); trap_init_excsave();
trap_init_debug();
} }
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
void secondary_trap_init(void) void secondary_trap_init(void)
{ {
trap_init_excsave(); trap_init_excsave();
trap_init_debug();
} }
#endif #endif
......
...@@ -601,7 +601,9 @@ ENDPROC(window_overflow_restore_a0_fixup) ...@@ -601,7 +601,9 @@ ENDPROC(window_overflow_restore_a0_fixup)
ENTRY(_DebugInterruptVector) 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 jx a0
ENDPROC(_DebugInterruptVector) ENDPROC(_DebugInterruptVector)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册