coredump: custom core dump stack is now supported on Xtensa targets

This commit is contained in:
Omar Chebib 2022-11-30 11:12:30 +08:00
parent 490216a2ac
commit eb66430793
4 changed files with 92 additions and 23 deletions

View File

@ -23,23 +23,39 @@
extern "C" { extern "C" {
#endif #endif
/**
* @brief Define the type that will be used to describe the current context when
* doing a backup of the current stack. This same structure is used to restore the stack.
*/
typedef struct {
uint32_t sp;
} core_dump_stack_context_t;
/** /**
* @brief Set the stack pointer to the address passed as a parameter. * @brief Set the stack pointer to the address passed as a parameter.
* @note This function must be inlined. * @note This function must be inlined.
* *
* @param new_sp New stack pointer to set in sp register. * @param new_sp New stack pointer to set in sp register.
* * @param old_ctx CPU context, related to SP, to fill. It will be given back when restoring SP.
* @return Former stack pointer address (sp register value).
*/ */
FORCE_INLINE_ATTR void* esp_core_dump_replace_sp(void* new_sp) FORCE_INLINE_ATTR void esp_core_dump_replace_sp(void* new_sp, core_dump_stack_context_t* old_ctx)
{ {
void* current_sp = NULL;
asm volatile ("mv %0, sp \n\t\ asm volatile ("mv %0, sp \n\t\
mv sp, %1 \n\t\ mv sp, %1 \n\t\
" "
: "=&r"(current_sp) : "=&r"(old_ctx->sp)
: "r"(new_sp)); : "r"(new_sp));
return current_sp; }
/**
* @brief Restore the stack pointer that was returned when calling `esp_core_dump_replace_sp()` function.
*
* @param ctx CPU context, related to SP, to restore.
*/
FORCE_INLINE_ATTR void esp_core_dump_restore_sp(core_dump_stack_context_t* old_ctx)
{
asm volatile ("mv sp, %0 \n\t" :: "r"(old_ctx->sp));
} }
#ifdef __cplusplus #ifdef __cplusplus

View File

@ -33,23 +33,75 @@
extern "C" { extern "C" {
#endif #endif
/**
* @brief Define the type that will be used to describe the current context when
* doing a backup of the current stack. This same structure is used to restore the stack.
*/
typedef struct {
uint32_t sp;
uint32_t a0;
uint32_t ps;
uint32_t windowbase;
uint32_t windowstart;
} core_dump_stack_context_t;
/** /**
* @brief Set the stack pointer to the address passed as a parameter. * @brief Set the stack pointer to the address passed as a parameter.
* @note This function must be inlined. * @note This function must be inlined.
* *
* @param new_sp New stack pointer to set in sp register. * @param new_sp New stack pointer to set in sp register.
* * @param old_ctx CPU context, related to SP, to fill. It will be given back when restoring SP.
* @return Former stack pointer address (sp register value).
*/ */
FORCE_INLINE_ATTR void* esp_core_dump_replace_sp(void* new_sp) FORCE_INLINE_ATTR void esp_core_dump_replace_sp(void* new_sp, core_dump_stack_context_t* old_ctx)
{ {
void* current_sp = NULL; /* We have to spill all the windows to the stack first as the new stack pointer
asm volatile ("mov %0, sp \n\t\ * represents a clean new environment. */
" xthal_window_spill();
: "=&r"(current_sp)
:); /* Backup the special registers PS, WindowBase and WindowStart. We will need to restore them later */
asm volatile ("mov %0, sp \n" \
"mov %1, a0 \n" \
"rsr.ps %2 \n"\
"rsr.windowbase %3 \n"\
"rsr.windowstart %4 \n"\
: "=r"(old_ctx->sp),
"=r"(old_ctx->a0),
"=r"(old_ctx->ps),
"=r"(old_ctx->windowbase),
"=r"(old_ctx->windowstart) :);
/* Set the new stack */
SET_STACK(new_sp); SET_STACK(new_sp);
return current_sp; }
/**
* @brief Restore the stack pointer that was returned when calling `esp_core_dump_replace_sp()` function.
*
* @param ctx CPU context, related to SP, to restore.
*/
FORCE_INLINE_ATTR void esp_core_dump_restore_sp(core_dump_stack_context_t* old_ctx)
{
/* Start by disabling WindowOverflowEnable bit from PS to make sure we won't get a Window Overflow exception
* restoring WindowBase and WindowStart registers */
const uint32_t ps_woe = old_ctx->ps & ~(PS_WOE_MASK);
asm volatile ( \
"wsr.ps %0 \n"\
"rsync \n"\
"wsr.windowbase %1 \n"\
"rsync \n"\
"wsr.windowstart %2 \n"\
"rsync \n"\
"mov sp, %3 \n" \
"mov a0, %4 \n" \
"wsr.ps %5 \n"\
"rsync \n"\
:: "r"(ps_woe),
"r"(old_ctx->windowbase),
"r"(old_ctx->windowstart),
"r"(old_ctx->sp),
"r"(old_ctx->a0),
"r"(old_ctx->ps));
} }
#ifdef __cplusplus #ifdef __cplusplus

View File

@ -57,7 +57,7 @@ extern int _coredump_rtc_fast_end;
static uint8_t s_coredump_stack[ESP_COREDUMP_STACK_SIZE]; static uint8_t s_coredump_stack[ESP_COREDUMP_STACK_SIZE];
static uint8_t* s_core_dump_sp = NULL; static uint8_t* s_core_dump_sp = NULL;
static uint8_t* s_core_dump_backup = NULL; static core_dump_stack_context_t s_stack_context;
/** /**
* @brief Function setting up the core dump stack. * @brief Function setting up the core dump stack.
@ -77,9 +77,9 @@ FORCE_INLINE_ATTR void esp_core_dump_setup_stack(void)
/* Replace the stack pointer depending on the architecture, but save the /* Replace the stack pointer depending on the architecture, but save the
* current stack pointer, in order to be able too restore it later. * current stack pointer, in order to be able too restore it later.
* This function must be inlined. */ * This function must be inlined. */
s_core_dump_backup = esp_core_dump_replace_sp(s_core_dump_sp); esp_core_dump_replace_sp(s_core_dump_sp, &s_stack_context);
ESP_COREDUMP_LOGI("Backing up stack @ %p and use core dump stack @ %p", ESP_COREDUMP_LOGI("Backing up stack @ %p and use core dump stack @ %p",
s_core_dump_backup, esp_cpu_get_sp()); s_stack_context.sp, esp_cpu_get_sp());
} }
/** /**
@ -111,8 +111,8 @@ FORCE_INLINE_ATTR void esp_core_dump_report_stack_usage(void)
s_core_dump_sp - s_coredump_stack - bytes_free, bytes_free); s_core_dump_sp - s_coredump_stack - bytes_free, bytes_free);
/* Restore the stack pointer. */ /* Restore the stack pointer. */
ESP_COREDUMP_LOGI("Restoring stack @ %p", s_core_dump_backup); ESP_COREDUMP_LOGI("Restoring stack @ %p", s_stack_context.sp);
esp_core_dump_replace_sp(s_core_dump_backup); esp_core_dump_restore_sp(&s_stack_context);
} }
#else #else

View File

@ -84,11 +84,11 @@
do { \ do { \
uint32_t sp = (uint32_t)new_sp - SAVE_AREA_OFFSET; \ uint32_t sp = (uint32_t)new_sp - SAVE_AREA_OFFSET; \
*(uint32_t*)(sp - BASE_AREA_SP_OFFSET) = (uint32_t)new_sp; \ *(uint32_t*)(sp - BASE_AREA_SP_OFFSET) = (uint32_t)new_sp; \
const uint32_t mask = ~(PS_WOE_MASK | PS_OWB_MASK | PS_CALLINC_MASK); \
uint32_t tmp1 = 0, tmp2 = 0; \ uint32_t tmp1 = 0, tmp2 = 0; \
asm volatile ( \ asm volatile ( \
"rsr.ps %1 \n"\ "rsr.ps %1 \n"\
"movi %2, ~" XTSTR( PS_WOE_MASK | PS_OWB_MASK | PS_CALLINC_MASK ) " \n"\ "and %1, %1, %3 \n"\
"and %1, %1, %2 \n"\
"wsr.ps %1 \n"\ "wsr.ps %1 \n"\
"rsync \n"\ "rsync \n"\
" \n"\ " \n"\
@ -99,6 +99,7 @@
"wsr.windowstart %1 \n"\ "wsr.windowstart %1 \n"\
"rsync \n"\ "rsync \n"\
" \n"\ " \n"\
"movi a0, 0\n" \
"mov sp, %0 \n"\ "mov sp, %0 \n"\
"rsr.ps %1 \n"\ "rsr.ps %1 \n"\
" \n"\ " \n"\
@ -107,6 +108,6 @@
"or %1, %1, %2 \n"\ "or %1, %1, %2 \n"\
"wsr.ps %1 \n"\ "wsr.ps %1 \n"\
"rsync \n"\ "rsync \n"\
: "+r"(sp), "+r"(tmp1), "+r"(tmp2)); \ : "+r"(sp), "+r"(tmp1), "+r"(tmp2) : "r"(mask)); \
} while (0); } while (0);
#endif // __ASSEMBLER__ #endif // __ASSEMBLER__