1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/esp_common/esp-xtensa/exceptions.c

219 lines
6.5 KiB
C
Raw Normal View History

2018-09-05 02:39:50 +02:00
/*
2019-09-05 13:35:58 +02:00
* Copyright (C) 2019 Gunar Schorcht
2018-09-05 02:39:50 +02:00
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
2019-12-11 07:28:45 +01:00
* @ingroup cpu_esp_common
2018-09-05 02:39:50 +02:00
* @{
*
* @file
* @brief Exception handling for Xtensa-based ESP SoCs
2018-09-05 02:39:50 +02:00
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @}
*/
2019-09-05 13:35:58 +02:00
#include <malloc.h>
2018-09-05 02:39:50 +02:00
#include <string.h>
#include <stdlib.h>
#include "log.h"
#include "periph/pm.h"
#include "ps.h"
2019-09-05 13:35:58 +02:00
#include "esp_common.h"
2018-09-05 02:39:50 +02:00
#include "esp/common_macros.h"
#include "esp/xtensa_ops.h"
#include "xtensa/corebits.h"
#include "xtensa/xtensa_api.h"
2020-10-22 11:34:00 +02:00
#define ENABLE_DEBUG 0
2019-12-11 07:28:45 +01:00
#include "debug.h"
2019-09-05 13:35:58 +02:00
extern void heap_stats(void);
2018-09-05 02:39:50 +02:00
static const char* exception_names [] =
{
2019-09-05 13:35:58 +02:00
"IllegalInstructionCause", /* 0 */
"SyscallCause", /* 1 */
"InstructionFetchErrorCause", /* 2 */
"LoadStoreErrorCause", /* 3 */
"Level1InterruptCause", /* 4 */
"AllocaCause", /* 5 */
"IntegerDivideByZeroCause", /* 6 */
"", /* 7 - reserved */
"PrivilegedCause", /* 8 */
"LoadStoreAlignmentCause", /* 9 */
"", /* 10 - reserved */
"", /* 11 - reserved */
"InstrPIFDataErrorCause", /* 12 */
"LoadStorePIFDataErrorCause", /* 13 */
"InstrPIFAddrErrorCause", /* 14 */
"LoadStorePIFAddrErrorCause", /* 15 */
"InstTLBMissCause", /* 16 */
"InstTLBMultiHitCause", /* 17 */
"InstFetchPrivilegeCause", /* 18 */
"", /* 19 - reserved */
"InstFetchProhibitedCause", /* 20 */
"", /* 21 - reserved */
"", /* 22 - reserved */
"", /* 23 - reserved */
"LoadStoreTLBMissCause", /* 24 */
"LoadStoreTLBMultiHitCause", /* 25 */
"LoadStorePrivilegeCause", /* 26 */
"", /* 27 - reserved */
"LoadProhibitedCause", /* 28 */
"StoreProhibitedCause", /* 29 */
"", /* 30 - reserved */
"", /* 31 - reserved */
"Coprocessor0Disabled", /* 32 */
"Coprocessor1Disabled", /* 33 */
"Coprocessor2Disabled", /* 34 */
"Coprocessor3Disabled", /* 35 */
"Coprocessor4Disabled", /* 36 */
"Coprocessor5Disabled", /* 37 */
"Coprocessor6Disabled", /* 38 */
"Coprocessor7Disabled", /* 39 */
2018-09-05 02:39:50 +02:00
};
void IRAM NORETURN exception_handler (XtExcFrame *frame)
{
uint32_t excsave1;
2019-09-05 13:35:58 +02:00
uint32_t epc1;
2018-09-05 02:39:50 +02:00
RSR(excsave1, excsave1);
2019-09-05 13:35:58 +02:00
RSR(epc1, epc1);
2019-12-11 07:28:45 +01:00
#if XCHAL_NMILEVEL >= 2
uint32_t epc2;
2019-09-05 13:35:58 +02:00
RSR(epc2, epc2);
2019-12-11 07:28:45 +01:00
#endif
#if XCHAL_NMILEVEL >= 3
uint32_t epc3;
2019-09-05 13:35:58 +02:00
RSR(epc3, epc3);
2019-12-11 07:28:45 +01:00
#endif
#if XCHAL_NMILEVEL >= 4
2019-09-05 13:35:58 +02:00
uint32_t epc4;
RSR(epc4, epc4);
#endif
2019-12-11 07:28:45 +01:00
#if XCHAL_NMILEVEL >= 5
uint32_t epc5;
RSR(epc5, epc5);
#endif
#if XCHAL_NMILEVEL >= 6
uint32_t epc6;
RSR(epc6, epc6);
#endif
#if XCHAL_NMILEVEL >= 7
uint32_t epc7;
RSR(epc7, epc7);
#endif
2018-09-05 02:39:50 +02:00
2019-09-05 13:35:58 +02:00
ets_printf("EXCEPTION!! exccause=%d (%s) @%08x excvaddr=%08x\n",
2018-09-05 02:39:50 +02:00
frame->exccause, exception_names[frame->exccause],
excsave1, frame->excvaddr);
2019-09-05 13:35:58 +02:00
#if defined(DEVELHELP)
#if defined(MODULE_PS)
ets_printf("processes:\n");
2018-09-05 02:39:50 +02:00
ps();
2019-09-05 13:35:58 +02:00
ets_printf("\n");
#endif /* MODULE_PS */
heap_stats();
ets_printf("\nregister set\n");
ets_printf("pc : %08x\t", frame->pc);
ets_printf("ps : %08x\t", frame->ps);
ets_printf("exccause: %08x\t", frame->exccause);
ets_printf("excvaddr: %08x\n", frame->excvaddr);
ets_printf("epc1 : %08x\t", epc1);
2019-12-11 07:28:45 +01:00
#if XCHAL_NMILEVEL >= 2
2019-09-05 13:35:58 +02:00
ets_printf("epc2 : %08x\t", epc2);
2019-12-11 07:28:45 +01:00
#endif
#if XCHAL_NMILEVEL >= 3
2019-09-05 13:35:58 +02:00
ets_printf("epc3 : %08x\t", epc3);
2019-12-11 07:28:45 +01:00
#endif
#if XCHAL_NMILEVEL >= 4
2019-09-05 13:35:58 +02:00
ets_printf("epc4 : %08x\n", epc4);
2019-12-11 07:28:45 +01:00
#endif
#if XCHAL_NMILEVEL >= 5
ets_printf("epc5 : %08x\t", epc5);
#endif
#if XCHAL_NMILEVEL >= 6
ets_printf("epc6 : %08x\t", epc6);
#endif
#if XCHAL_NMILEVEL >= 7
ets_printf("epc7 : %08x\t", epc7);
#endif
ets_printf("\n");
2019-09-05 13:35:58 +02:00
ets_printf("a0 : %08x\t", frame->a0);
ets_printf("a1 : %08x\t", frame->a1);
ets_printf("a2 : %08x\t", frame->a2);
ets_printf("a3 : %08x\n", frame->a3);
ets_printf("a4 : %08x\t", frame->a4);
ets_printf("a5 : %08x\t", frame->a5);
ets_printf("a6 : %08x\t", frame->a6);
ets_printf("a7 : %08x\n", frame->a7);
ets_printf("a8 : %08x\t", frame->a8);
ets_printf("a9 : %08x\t", frame->a9);
ets_printf("a10 : %08x\t", frame->a10);
ets_printf("a11 : %08x\n", frame->a11);
ets_printf("a12 : %08x\t", frame->a12);
ets_printf("a13 : %08x\t", frame->a13);
ets_printf("a14 : %08x\t", frame->a14);
ets_printf("a15 : %08x\n", frame->a15);
#if XCHAL_HAVE_LOOPS
ets_printf("lbeg : %08x\t", frame->lbeg);
ets_printf("lend : %08x\t", frame->lend);
ets_printf("lcount : %08x\n", frame->lcount);
#endif /* XCHAL_HAVE_LOOPS */
#endif /* DEVELHELP */
2018-09-05 02:39:50 +02:00
2019-09-05 13:35:58 +02:00
/* restart */
/* TODO: Improvement
Normally, we should try to restart the system. However, this
will not work after some exceptions, e.g., the LoadStoreErrorCause.
One option is to break the execution and wait for the WDT reset. Maybe
there is better way. If debugger is active, 'break 0,0' stops the
execution in debugger. */
2019-12-11 07:28:45 +01:00
__asm__ volatile ("break 0,0");
2018-09-05 02:39:50 +02:00
UNREACHABLE();
}
void init_exceptions (void)
{
xt_set_exception_handler(EXCCAUSE_UNALIGNED, exception_handler);
xt_set_exception_handler(EXCCAUSE_ILLEGAL, exception_handler);
xt_set_exception_handler(EXCCAUSE_INSTR_ERROR, exception_handler);
xt_set_exception_handler(EXCCAUSE_LOAD_STORE_ERROR, exception_handler);
xt_set_exception_handler(EXCCAUSE_LOAD_PROHIBITED, exception_handler);
xt_set_exception_handler(EXCCAUSE_STORE_PROHIBITED, exception_handler);
xt_set_exception_handler(EXCCAUSE_PRIVILEGED, exception_handler);
}
void IRAM NORETURN panic_arch(void)
{
2019-12-11 07:28:45 +01:00
#if defined(DEVELHELP)
2019-09-05 13:35:58 +02:00
heap_stats();
2019-12-11 07:28:45 +01:00
/* break in debugger or reboot after WDT */
__asm__ volatile ("break 0,0");
#else /* DEVELHELP */
/* restart */
pm_reboot();
#endif /* DEVELHELP */
2018-09-05 02:39:50 +02:00
UNREACHABLE();
}
2019-09-05 13:35:58 +02:00
void _panic_handler(uint32_t addr)
{
ets_printf("#! _xt_panic called from 0x%08x: powering off\n", addr);
pm_off();
while (1) { };
}