2019-05-10 11:34:06 +08:00
|
|
|
// Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <xtensa/config/core.h>
|
|
|
|
|
|
|
|
#include "esp32s2beta/rom/rtc.h"
|
|
|
|
#include "esp32s2beta/rom/uart.h"
|
|
|
|
|
|
|
|
#include "freertos/FreeRTOS.h"
|
|
|
|
#include "freertos/task.h"
|
|
|
|
#include "freertos/xtensa_api.h"
|
|
|
|
|
|
|
|
#include "soc/uart_reg.h"
|
|
|
|
#include "soc/io_mux_reg.h"
|
|
|
|
#include "soc/dport_reg.h"
|
|
|
|
#include "soc/rtc_cntl_reg.h"
|
|
|
|
#include "soc/timer_group_struct.h"
|
|
|
|
#include "soc/timer_group_reg.h"
|
|
|
|
#include "soc/cpu.h"
|
2019-08-08 14:00:45 +10:00
|
|
|
#include "soc/soc_memory_layout.h"
|
2019-05-10 11:34:06 +08:00
|
|
|
#include "soc/rtc.h"
|
|
|
|
#include "soc/rtc_wdt.h"
|
|
|
|
|
|
|
|
#include "esp_private/gdbstub.h"
|
|
|
|
#include "esp_debug_helpers.h"
|
|
|
|
#include "esp_private/panic_reason.h"
|
|
|
|
#include "esp_attr.h"
|
|
|
|
#include "esp_err.h"
|
|
|
|
#include "esp_core_dump.h"
|
|
|
|
#include "esp_spi_flash.h"
|
|
|
|
#include "esp32s2beta/cache_err_int.h"
|
|
|
|
#include "esp_app_trace.h"
|
|
|
|
#include "esp_private/system_internal.h"
|
|
|
|
#include "sdkconfig.h"
|
|
|
|
#if CONFIG_SYSVIEW_ENABLE
|
|
|
|
#include "SEGGER_RTT.h"
|
|
|
|
#endif
|
|
|
|
|
2019-09-13 15:49:11 +03:00
|
|
|
#if CONFIG_APPTRACE_ONPANIC_HOST_FLUSH_TMO == -1
|
2019-05-10 11:34:06 +08:00
|
|
|
#define APPTRACE_ONPANIC_HOST_FLUSH_TMO ESP_APPTRACE_TMO_INFINITE
|
|
|
|
#else
|
2019-09-13 15:49:11 +03:00
|
|
|
#define APPTRACE_ONPANIC_HOST_FLUSH_TMO (1000*CONFIG_APPTRACE_ONPANIC_HOST_FLUSH_TMO)
|
2019-05-10 11:34:06 +08:00
|
|
|
#endif
|
|
|
|
/*
|
|
|
|
Panic handlers; these get called when an unhandled exception occurs or the assembly-level
|
|
|
|
task switching / interrupt code runs into an unrecoverable error. The default task stack
|
|
|
|
overflow handler and abort handler are also in here.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
Note: The linker script will put everything in this file in IRAM/DRAM, so it also works with flash cache disabled.
|
|
|
|
*/
|
|
|
|
|
2019-06-04 17:02:01 +10:00
|
|
|
#if !CONFIG_ESP32S2_PANIC_SILENT_REBOOT
|
2019-05-10 11:34:06 +08:00
|
|
|
//printf may be broken, so we fix our own printing fns...
|
|
|
|
static void panicPutChar(char c)
|
|
|
|
{
|
2019-06-10 15:07:12 +08:00
|
|
|
while (((READ_PERI_REG(UART_STATUS_REG(CONFIG_ESP_CONSOLE_UART_NUM)) >> UART_TXFIFO_CNT_S)&UART_TXFIFO_CNT) >= 126) ;
|
|
|
|
WRITE_PERI_REG(UART_FIFO_AHB_REG(CONFIG_ESP_CONSOLE_UART_NUM), c);
|
2019-05-10 11:34:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void panicPutStr(const char *c)
|
|
|
|
{
|
|
|
|
int x = 0;
|
|
|
|
while (c[x] != 0) {
|
|
|
|
panicPutChar(c[x]);
|
|
|
|
x++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void panicPutHex(int a)
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
int c;
|
|
|
|
for (x = 0; x < 8; x++) {
|
|
|
|
c = (a >> 28) & 0xf;
|
|
|
|
if (c < 10) {
|
|
|
|
panicPutChar('0' + c);
|
|
|
|
} else {
|
|
|
|
panicPutChar('a' + c - 10);
|
|
|
|
}
|
|
|
|
a <<= 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void panicPutDec(int a)
|
|
|
|
{
|
|
|
|
int n1, n2;
|
|
|
|
n1 = a % 10;
|
|
|
|
n2 = a / 10;
|
|
|
|
if (n2 == 0) {
|
|
|
|
panicPutChar(' ');
|
|
|
|
} else {
|
|
|
|
panicPutChar(n2 + '0');
|
|
|
|
}
|
|
|
|
panicPutChar(n1 + '0');
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
//No printing wanted. Stub out these functions.
|
|
|
|
static void panicPutChar(char c) { }
|
|
|
|
static void panicPutStr(const char *c) { }
|
|
|
|
static void panicPutHex(int a) { }
|
|
|
|
static void panicPutDec(int a) { }
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void __attribute__((weak)) vApplicationStackOverflowHook( TaskHandle_t xTask, signed char *pcTaskName )
|
|
|
|
{
|
|
|
|
panicPutStr("***ERROR*** A stack overflow in task ");
|
|
|
|
panicPutStr((char *)pcTaskName);
|
|
|
|
panicPutStr(" has been detected.\r\n");
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
2019-11-19 11:19:29 +01:00
|
|
|
/* These two weak stubs for esp_reset_reason_{get,set}_hint are used when
|
|
|
|
* the application does not call esp_reset_reason() function, and
|
|
|
|
* reset_reason.c is not linked into the output file.
|
|
|
|
*/
|
|
|
|
void __attribute__((weak)) esp_reset_reason_set_hint(esp_reset_reason_t hint)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_reset_reason_t __attribute__((weak)) esp_reset_reason_get_hint(void)
|
|
|
|
{
|
|
|
|
return ESP_RST_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
static bool abort_called;
|
|
|
|
|
2019-08-12 12:06:07 +10:00
|
|
|
static __attribute__((noreturn)) inline void invoke_abort(void)
|
2019-05-10 11:34:06 +08:00
|
|
|
{
|
|
|
|
abort_called = true;
|
2019-09-13 15:49:11 +03:00
|
|
|
#if CONFIG_APPTRACE_ENABLE
|
2019-05-10 11:34:06 +08:00
|
|
|
#if CONFIG_SYSVIEW_ENABLE
|
2019-09-13 15:49:11 +03:00
|
|
|
SEGGER_RTT_ESP32_FlushNoLock(CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH, APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
2019-05-10 11:34:06 +08:00
|
|
|
#else
|
2019-09-13 15:49:11 +03:00
|
|
|
esp_apptrace_flush_nolock(ESP_APPTRACE_DEST_TRAX, CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH,
|
2019-05-10 11:34:06 +08:00
|
|
|
APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
while (1) {
|
|
|
|
if (esp_cpu_in_ocd_debug_mode()) {
|
|
|
|
__asm__ ("break 0,0");
|
|
|
|
}
|
|
|
|
*((int *) 0) = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-12 12:06:07 +10:00
|
|
|
void abort(void)
|
2019-05-10 11:34:06 +08:00
|
|
|
{
|
2019-06-04 17:02:01 +10:00
|
|
|
#if !CONFIG_ESP32S2_PANIC_SILENT_REBOOT
|
2019-05-10 11:34:06 +08:00
|
|
|
ets_printf("abort() was called at PC 0x%08x on core %d\r\n", (intptr_t)__builtin_return_address(0) - 3, xPortGetCoreID());
|
|
|
|
#endif
|
2019-11-19 11:19:29 +01:00
|
|
|
/* Calling code might have set other reset reason hint (such as Task WDT),
|
|
|
|
* don't overwrite that.
|
|
|
|
*/
|
|
|
|
if (esp_reset_reason_get_hint() == ESP_RST_UNKNOWN) {
|
|
|
|
esp_reset_reason_set_hint(ESP_RST_PANIC);
|
|
|
|
}
|
2019-05-10 11:34:06 +08:00
|
|
|
invoke_abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const char *edesc[] = {
|
|
|
|
"IllegalInstruction", "Syscall", "InstructionFetchError", "LoadStoreError",
|
|
|
|
"Level1Interrupt", "Alloca", "IntegerDivideByZero", "PCValue",
|
|
|
|
"Privileged", "LoadStoreAlignment", "res", "res",
|
|
|
|
"InstrPDAddrError", "LoadStorePIFDataError", "InstrPIFAddrError", "LoadStorePIFAddrError",
|
|
|
|
"InstTLBMiss", "InstTLBMultiHit", "InstFetchPrivilege", "res",
|
|
|
|
"InstrFetchProhibited", "res", "res", "res",
|
|
|
|
"LoadStoreTLBMiss", "LoadStoreTLBMultihit", "LoadStorePrivilege", "res",
|
|
|
|
"LoadProhibited", "StoreProhibited", "res", "res",
|
|
|
|
"Cp0Dis", "Cp1Dis", "Cp2Dis", "Cp3Dis",
|
|
|
|
"Cp4Dis", "Cp5Dis", "Cp6Dis", "Cp7Dis"
|
|
|
|
};
|
|
|
|
|
|
|
|
#define NUM_EDESCS (sizeof(edesc) / sizeof(char *))
|
|
|
|
|
2020-04-17 19:06:26 +05:30
|
|
|
extern int _invalid_pc_placeholder;
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
static void commonErrorHandler(XtExcFrame *frame);
|
2019-08-12 12:06:07 +10:00
|
|
|
static inline void disableAllWdts(void);
|
2019-05-10 11:34:06 +08:00
|
|
|
|
|
|
|
//The fact that we've panic'ed probably means the other CPU is now running wild, possibly
|
|
|
|
//messing up the serial output, so we stall it here.
|
2019-08-12 12:06:07 +10:00
|
|
|
static void haltOtherCore(void)
|
2019-05-10 11:34:06 +08:00
|
|
|
{
|
|
|
|
esp_cpu_stall( xPortGetCoreID() == 0 ? 1 : 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void setFirstBreakpoint(uint32_t pc)
|
|
|
|
{
|
|
|
|
asm(
|
|
|
|
"wsr.ibreaka0 %0\n" \
|
|
|
|
"rsr.ibreakenable a3\n" \
|
|
|
|
"movi a4,1\n" \
|
|
|
|
"or a4, a4, a3\n" \
|
|
|
|
"wsr.ibreakenable a4\n" \
|
|
|
|
::"r"(pc):"a3", "a4");
|
|
|
|
}
|
|
|
|
|
2019-07-17 10:05:18 +08:00
|
|
|
static inline void printCacheError(void)
|
|
|
|
{
|
|
|
|
uint32_t vaddr = 0, size = 0;
|
|
|
|
uint32_t status[2];
|
|
|
|
status[0] = REG_READ(DPORT_CACHE_DBG_STATUS0_REG);
|
|
|
|
status[1] = REG_READ(DPORT_CACHE_DBG_STATUS1_REG);
|
|
|
|
for (int i = 0; i < 32; i++) {
|
2019-07-31 13:49:21 +08:00
|
|
|
switch (status[0] & BIT(i))
|
2019-07-17 10:05:18 +08:00
|
|
|
{
|
|
|
|
case DPORT_IC_SYNC_SIZE_FAULT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_ICACHE_MEM_SYNC0_REG);
|
|
|
|
size = REG_READ(DPORT_PRO_ICACHE_MEM_SYNC1_REG);
|
|
|
|
panicPutStr("Icache sync parameter configuration error, the error address and size is 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
panicPutStr("(0x");
|
|
|
|
panicPutHex(size);
|
|
|
|
panicPutStr(")\r\n");
|
|
|
|
break;
|
|
|
|
case DPORT_IC_PRELOAD_SIZE_FAULT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_ICACHE_PRELOAD_ADDR_REG);
|
|
|
|
size = REG_READ(DPORT_PRO_ICACHE_PRELOAD_SIZE_REG);
|
|
|
|
panicPutStr("Icache preload parameter configuration error, the error address and size is 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
panicPutStr("(0x");
|
|
|
|
panicPutHex(size);
|
|
|
|
panicPutStr(")\r\n");
|
|
|
|
break;
|
|
|
|
case DPORT_ICACHE_REJECT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_ICACHE_REJECT_VADDR_REG);
|
|
|
|
panicPutStr("Icache reject error occurred while accessing the address 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
|
|
|
|
if (REG_READ(DPORT_PRO_CACHE_MMU_ERROR_CONTENT_REG) & DPORT_MMU_INVALID) {
|
|
|
|
panicPutStr(" (invalid mmu entry)");
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2019-07-31 13:49:21 +08:00
|
|
|
switch (status[1] & BIT(i))
|
2019-07-17 10:05:18 +08:00
|
|
|
{
|
|
|
|
case DPORT_DC_SYNC_SIZE_FAULT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_DCACHE_MEM_SYNC0_REG);
|
|
|
|
size = REG_READ(DPORT_PRO_DCACHE_MEM_SYNC1_REG);
|
|
|
|
panicPutStr("Dcache sync parameter configuration error, the error address and size is 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
panicPutStr("(0x");
|
|
|
|
panicPutHex(size);
|
|
|
|
panicPutStr(")\r\n");
|
|
|
|
break;
|
|
|
|
case DPORT_DC_PRELOAD_SIZE_FAULT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_DCACHE_PRELOAD_ADDR_REG);
|
|
|
|
size = REG_READ(DPORT_PRO_DCACHE_PRELOAD_SIZE_REG);
|
|
|
|
panicPutStr("Dcache preload parameter configuration error, the error address and size is 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
panicPutStr("(0x");
|
|
|
|
panicPutHex(size);
|
|
|
|
panicPutStr(")\r\n");
|
|
|
|
break;
|
|
|
|
case DPORT_DCACHE_WRITE_FLASH_ST:
|
|
|
|
panicPutStr("Write back error occurred while dcache tries to write back to flash\r\n");
|
|
|
|
break;
|
|
|
|
case DPORT_DCACHE_REJECT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_DCACHE_REJECT_VADDR_REG);
|
|
|
|
panicPutStr("Dcache reject error occurred while accessing the address 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
|
|
|
|
if (REG_READ(DPORT_PRO_CACHE_MMU_ERROR_CONTENT_REG) & DPORT_MMU_INVALID) {
|
|
|
|
panicPutStr(" (invalid mmu entry)");
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n");
|
|
|
|
break;
|
|
|
|
case DPORT_MMU_ENTRY_FAULT_ST:
|
|
|
|
vaddr = REG_READ(DPORT_PRO_CACHE_MMU_ERROR_VADDR_REG);
|
|
|
|
panicPutStr("MMU entry fault error occurred while accessing the address 0x");
|
|
|
|
panicPutHex(vaddr);
|
|
|
|
|
|
|
|
if (REG_READ(DPORT_PRO_CACHE_MMU_ERROR_CONTENT_REG) & DPORT_MMU_INVALID) {
|
|
|
|
panicPutStr(" (invalid mmu entry)");
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n");
|
|
|
|
}
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
void panicHandler(XtExcFrame *frame)
|
|
|
|
{
|
|
|
|
int core_id = xPortGetCoreID();
|
|
|
|
//Please keep in sync with PANIC_RSN_* defines
|
|
|
|
const char *reasons[] = {
|
|
|
|
"Unknown reason",
|
|
|
|
"Unhandled debug exception",
|
|
|
|
"Double exception",
|
|
|
|
"Unhandled kernel exception",
|
|
|
|
"Coprocessor exception",
|
|
|
|
"Interrupt wdt timeout on CPU0",
|
|
|
|
"Interrupt wdt timeout on CPU1",
|
2019-07-17 10:05:18 +08:00
|
|
|
"Cache exception",
|
2019-05-10 11:34:06 +08:00
|
|
|
};
|
|
|
|
const char *reason = reasons[0];
|
|
|
|
//The panic reason is stored in the EXCCAUSE register.
|
|
|
|
if (frame->exccause <= PANIC_RSN_MAX) {
|
|
|
|
reason = reasons[frame->exccause];
|
|
|
|
}
|
|
|
|
|
2019-11-19 11:19:29 +01:00
|
|
|
if (frame->exccause == PANIC_RSN_INTWDT_CPU0) {
|
|
|
|
esp_reset_reason_set_hint(ESP_RST_INT_WDT);
|
|
|
|
}
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
haltOtherCore();
|
|
|
|
panicPutStr("Guru Meditation Error: Core ");
|
|
|
|
panicPutDec(core_id);
|
|
|
|
panicPutStr(" panic'ed (");
|
|
|
|
panicPutStr(reason);
|
|
|
|
panicPutStr(")\r\n");
|
|
|
|
if (frame->exccause == PANIC_RSN_DEBUGEXCEPTION) {
|
|
|
|
int debugRsn;
|
|
|
|
asm("rsr.debugcause %0":"=r"(debugRsn));
|
|
|
|
panicPutStr("Debug exception reason: ");
|
|
|
|
if (debugRsn & XCHAL_DEBUGCAUSE_ICOUNT_MASK) {
|
|
|
|
panicPutStr("SingleStep ");
|
|
|
|
}
|
|
|
|
if (debugRsn & XCHAL_DEBUGCAUSE_IBREAK_MASK) {
|
|
|
|
panicPutStr("HwBreakpoint ");
|
|
|
|
}
|
|
|
|
if (debugRsn & XCHAL_DEBUGCAUSE_DBREAK_MASK) {
|
|
|
|
//Unlike what the ISA manual says, this core seemingly distinguishes from a DBREAK
|
|
|
|
//reason caused by watchdog 0 and one caused by watchdog 1 by setting bit 8 of the
|
|
|
|
//debugcause if the cause is watchdog 1 and clearing it if it's watchdog 0.
|
|
|
|
if (debugRsn & (1 << 8)) {
|
|
|
|
#if CONFIG_FREERTOS_WATCHPOINT_END_OF_STACK
|
|
|
|
const char *name = pcTaskGetTaskName(xTaskGetCurrentTaskHandleForCPU(core_id));
|
|
|
|
panicPutStr("Stack canary watchpoint triggered (");
|
|
|
|
panicPutStr(name);
|
|
|
|
panicPutStr(") ");
|
|
|
|
#else
|
|
|
|
panicPutStr("Watchpoint 1 triggered ");
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
panicPutStr("Watchpoint 0 triggered ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (debugRsn & XCHAL_DEBUGCAUSE_BREAK_MASK) {
|
|
|
|
panicPutStr("BREAK instr ");
|
|
|
|
}
|
|
|
|
if (debugRsn & XCHAL_DEBUGCAUSE_BREAKN_MASK) {
|
|
|
|
panicPutStr("BREAKN instr ");
|
|
|
|
}
|
|
|
|
if (debugRsn & XCHAL_DEBUGCAUSE_DEBUGINT_MASK) {
|
|
|
|
panicPutStr("DebugIntr ");
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n");
|
2019-07-31 13:49:21 +08:00
|
|
|
} else if (frame->exccause == PANIC_RSN_CACHEERR) {
|
2019-07-17 10:05:18 +08:00
|
|
|
panicPutStr(" ^~~~~~~~~~~~~~~\r\n");
|
|
|
|
printCacheError();
|
|
|
|
}
|
2019-05-10 11:34:06 +08:00
|
|
|
|
|
|
|
if (esp_cpu_in_ocd_debug_mode()) {
|
|
|
|
disableAllWdts();
|
2020-04-17 19:06:26 +05:30
|
|
|
if (!(esp_ptr_executable((void *)((frame->pc & 0x3fffffffU) | 0x40000000U)) && (frame->pc & 0xC0000000U))) {
|
|
|
|
/* Xtensa ABI sets the 2 MSBs of the PC according to the windowed call size
|
|
|
|
* Incase the PC is invalid, GDB will fail to translate addresses to function names
|
|
|
|
* Hence replacing the PC to a placeholder address in case of invalid PC
|
|
|
|
*/
|
|
|
|
frame->pc = (uint32_t)&_invalid_pc_placeholder;
|
|
|
|
}
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
if (frame->exccause == PANIC_RSN_INTWDT_CPU0 ||
|
|
|
|
frame->exccause == PANIC_RSN_INTWDT_CPU1) {
|
|
|
|
TIMERG1.int_clr.wdt = 1;
|
|
|
|
}
|
2019-09-13 15:49:11 +03:00
|
|
|
#if CONFIG_APPTRACE_ENABLE
|
2019-05-10 11:34:06 +08:00
|
|
|
#if CONFIG_SYSVIEW_ENABLE
|
2019-09-13 15:49:11 +03:00
|
|
|
SEGGER_RTT_ESP32_FlushNoLock(CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH, APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
2019-05-10 11:34:06 +08:00
|
|
|
#else
|
2019-09-13 15:49:11 +03:00
|
|
|
esp_apptrace_flush_nolock(ESP_APPTRACE_DEST_TRAX, CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH,
|
2019-05-10 11:34:06 +08:00
|
|
|
APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
setFirstBreakpoint(frame->pc);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
commonErrorHandler(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
void xt_unhandled_exception(XtExcFrame *frame)
|
|
|
|
{
|
|
|
|
haltOtherCore();
|
|
|
|
if (!abort_called) {
|
|
|
|
panicPutStr("Guru Meditation Error: Core ");
|
|
|
|
panicPutDec(xPortGetCoreID());
|
|
|
|
panicPutStr(" panic'ed (");
|
|
|
|
int exccause = frame->exccause;
|
|
|
|
if (exccause < NUM_EDESCS) {
|
|
|
|
panicPutStr(edesc[exccause]);
|
|
|
|
} else {
|
|
|
|
panicPutStr("Unknown");
|
|
|
|
}
|
|
|
|
panicPutStr(")");
|
|
|
|
if (esp_cpu_in_ocd_debug_mode()) {
|
|
|
|
panicPutStr(" at pc=");
|
|
|
|
panicPutHex(frame->pc);
|
|
|
|
panicPutStr(". Setting bp and returning..\r\n");
|
2019-09-13 15:49:11 +03:00
|
|
|
#if CONFIG_APPTRACE_ENABLE
|
2019-05-10 11:34:06 +08:00
|
|
|
#if CONFIG_SYSVIEW_ENABLE
|
2019-09-13 15:49:11 +03:00
|
|
|
SEGGER_RTT_ESP32_FlushNoLock(CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH, APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
2019-05-10 11:34:06 +08:00
|
|
|
#else
|
2019-09-13 15:49:11 +03:00
|
|
|
esp_apptrace_flush_nolock(ESP_APPTRACE_DEST_TRAX, CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH,
|
2019-05-10 11:34:06 +08:00
|
|
|
APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
|
|
|
#endif
|
|
|
|
#endif
|
2020-04-17 19:06:26 +05:30
|
|
|
if (!(esp_ptr_executable((void *)((frame->pc & 0x3fffffffU) | 0x40000000U)) && (frame->pc & 0xC0000000U))) {
|
|
|
|
/* Xtensa ABI sets the 2 MSBs of the PC according to the windowed call size
|
|
|
|
* Incase the PC is invalid, GDB will fail to translate addresses to function names
|
|
|
|
* Hence replacing the PC to a placeholder address in case of invalid PC
|
|
|
|
*/
|
|
|
|
frame->pc = (uint32_t)&_invalid_pc_placeholder;
|
|
|
|
}
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
//Stick a hardware breakpoint on the address the handler returns to. This way, the OCD debugger
|
|
|
|
//will kick in exactly at the context the error happened.
|
|
|
|
setFirstBreakpoint(frame->pc);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
panicPutStr(". Exception was unhandled.\r\n");
|
2019-11-19 11:19:29 +01:00
|
|
|
esp_reset_reason_set_hint(ESP_RST_PANIC);
|
2019-05-10 11:34:06 +08:00
|
|
|
}
|
|
|
|
commonErrorHandler(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
If watchdogs are enabled, the panic handler runs the risk of getting aborted pre-emptively because
|
|
|
|
an overzealous watchdog decides to reset it. On the other hand, if we disable all watchdogs, we run
|
|
|
|
the risk of somehow halting in the panic handler and not resetting. That is why this routine kills
|
|
|
|
all watchdogs except the timer group 0 watchdog, and it reconfigures that to reset the chip after
|
|
|
|
one second.
|
|
|
|
*/
|
2019-08-12 12:06:07 +10:00
|
|
|
static void reconfigureAllWdts(void)
|
2019-05-10 11:34:06 +08:00
|
|
|
{
|
|
|
|
TIMERG0.wdt_wprotect = TIMG_WDT_WKEY_VALUE;
|
|
|
|
TIMERG0.wdt_feed = 1;
|
|
|
|
TIMERG0.wdt_config0.sys_reset_length = 7; //3.2uS
|
|
|
|
TIMERG0.wdt_config0.cpu_reset_length = 7; //3.2uS
|
|
|
|
TIMERG0.wdt_config0.stg0 = TIMG_WDT_STG_SEL_RESET_SYSTEM; //1st stage timeout: reset system
|
|
|
|
TIMERG0.wdt_config1.clk_prescale = 80 * 500; //Prescaler: wdt counts in ticks of 0.5mS
|
|
|
|
TIMERG0.wdt_config2 = 2000; //1 second before reset
|
|
|
|
TIMERG0.wdt_config0.en = 1;
|
|
|
|
TIMERG0.wdt_wprotect = 0;
|
|
|
|
//Disable wdt 1
|
|
|
|
TIMERG1.wdt_wprotect = TIMG_WDT_WKEY_VALUE;
|
|
|
|
TIMERG1.wdt_config0.en = 0;
|
|
|
|
TIMERG1.wdt_wprotect = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
This disables all the watchdogs for when we call the gdbstub.
|
|
|
|
*/
|
2019-08-12 12:06:07 +10:00
|
|
|
static inline void disableAllWdts(void)
|
2019-05-10 11:34:06 +08:00
|
|
|
{
|
|
|
|
TIMERG0.wdt_wprotect = TIMG_WDT_WKEY_VALUE;
|
|
|
|
TIMERG0.wdt_config0.en = 0;
|
|
|
|
TIMERG0.wdt_wprotect = 0;
|
|
|
|
TIMERG1.wdt_wprotect = TIMG_WDT_WKEY_VALUE;
|
|
|
|
TIMERG1.wdt_config0.en = 0;
|
|
|
|
TIMERG1.wdt_wprotect = 0;
|
|
|
|
}
|
|
|
|
|
2019-07-17 10:05:18 +08:00
|
|
|
#if CONFIG_ESP32S2_PANIC_PRINT_REBOOT || CONFIG_ESP32S2_PANIC_SILENT_REBOOT
|
|
|
|
|
2019-08-12 12:06:07 +10:00
|
|
|
static void esp_panic_dig_reset(void) __attribute__((noreturn));
|
2019-05-10 11:34:06 +08:00
|
|
|
|
2019-08-12 12:06:07 +10:00
|
|
|
static void esp_panic_dig_reset(void)
|
2019-05-10 11:34:06 +08:00
|
|
|
{
|
|
|
|
// make sure all the panic handler output is sent from UART FIFO
|
2019-06-10 15:07:12 +08:00
|
|
|
uart_tx_wait_idle(CONFIG_ESP_CONSOLE_UART_NUM);
|
2019-05-10 11:34:06 +08:00
|
|
|
// switch to XTAL (otherwise we will keep running from the PLL)
|
|
|
|
rtc_clk_cpu_freq_set(RTC_CPU_FREQ_XTAL);
|
|
|
|
// reset the digital part
|
|
|
|
esp_cpu_unstall(PRO_CPU_NUM);
|
|
|
|
SET_PERI_REG_MASK(RTC_CNTL_OPTIONS0_REG, RTC_CNTL_SW_SYS_RST);
|
|
|
|
while (true) {
|
|
|
|
;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-17 10:05:18 +08:00
|
|
|
#endif
|
|
|
|
|
2019-05-10 11:34:06 +08:00
|
|
|
static void putEntry(uint32_t pc, uint32_t sp)
|
|
|
|
{
|
|
|
|
if (pc & 0x80000000) {
|
|
|
|
pc = (pc & 0x3fffffff) | 0x40000000;
|
|
|
|
}
|
|
|
|
panicPutStr(" 0x");
|
|
|
|
panicPutHex(pc);
|
|
|
|
panicPutStr(":0x");
|
|
|
|
panicPutHex(sp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void doBacktrace(XtExcFrame *frame)
|
|
|
|
{
|
|
|
|
uint32_t i = 0, pc = frame->pc, sp = frame->a1;
|
|
|
|
panicPutStr("\r\nBacktrace:");
|
|
|
|
/* Do not check sanity on first entry, PC could be smashed. */
|
|
|
|
putEntry(pc, sp);
|
|
|
|
pc = frame->a0;
|
|
|
|
while (i++ < 100) {
|
|
|
|
uint32_t psp = sp;
|
|
|
|
if (!esp_stack_ptr_is_sane(sp) || i++ > 100) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sp = *((uint32_t *) (sp - 0x10 + 4));
|
|
|
|
putEntry(pc - 3, sp); // stack frame addresses are return addresses, so subtract 3 to get the CALL address
|
|
|
|
pc = *((uint32_t *) (psp - 0x10));
|
|
|
|
if (pc < 0x40000000) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n\r\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Dump registers and do backtrace.
|
|
|
|
*/
|
|
|
|
static void commonErrorHandler_dump(XtExcFrame *frame, int core_id)
|
|
|
|
{
|
|
|
|
int *regs = (int *)frame;
|
|
|
|
int x, y;
|
|
|
|
const char *sdesc[] = {
|
|
|
|
"PC ", "PS ", "A0 ", "A1 ", "A2 ", "A3 ", "A4 ", "A5 ",
|
|
|
|
"A6 ", "A7 ", "A8 ", "A9 ", "A10 ", "A11 ", "A12 ", "A13 ",
|
|
|
|
"A14 ", "A15 ", "SAR ", "EXCCAUSE", "EXCVADDR", "LBEG ", "LEND ", "LCOUNT "
|
|
|
|
};
|
|
|
|
|
|
|
|
/* only dump registers for 'real' crashes, if crashing via abort()
|
|
|
|
the register window is no longer useful.
|
|
|
|
*/
|
|
|
|
if (!abort_called) {
|
|
|
|
panicPutStr("Core");
|
|
|
|
panicPutDec(core_id);
|
|
|
|
panicPutStr(" register dump:\r\n");
|
|
|
|
|
|
|
|
for (x = 0; x < 24; x += 4) {
|
|
|
|
for (y = 0; y < 4; y++) {
|
|
|
|
if (sdesc[x + y][0] != 0) {
|
|
|
|
panicPutStr(sdesc[x + y]);
|
|
|
|
panicPutStr(": 0x");
|
|
|
|
panicPutHex(regs[x + y + 1]);
|
|
|
|
panicPutStr(" ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panicPutStr("\r\n");
|
|
|
|
}
|
|
|
|
|
2019-11-28 21:11:49 +08:00
|
|
|
if (xPortInterruptedFromISRContext()) {
|
2019-05-10 11:34:06 +08:00
|
|
|
//If the core which triggers the interrupt watchdog was in ISR context, dump the epc registers.
|
|
|
|
uint32_t __value;
|
|
|
|
panicPutStr("Core");
|
|
|
|
panicPutDec(core_id);
|
|
|
|
panicPutStr(" was running in ISR context:\r\n");
|
|
|
|
|
|
|
|
__asm__("rsr.epc1 %0" : "=a"(__value));
|
|
|
|
panicPutStr("EPC1 : 0x");
|
|
|
|
panicPutHex(__value);
|
|
|
|
|
|
|
|
__asm__("rsr.epc2 %0" : "=a"(__value));
|
|
|
|
panicPutStr(" EPC2 : 0x");
|
|
|
|
panicPutHex(__value);
|
|
|
|
|
|
|
|
__asm__("rsr.epc3 %0" : "=a"(__value));
|
|
|
|
panicPutStr(" EPC3 : 0x");
|
|
|
|
panicPutHex(__value);
|
|
|
|
|
|
|
|
__asm__("rsr.epc4 %0" : "=a"(__value));
|
|
|
|
panicPutStr(" EPC4 : 0x");
|
|
|
|
panicPutHex(__value);
|
|
|
|
|
|
|
|
panicPutStr("\r\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* With windowed ABI backtracing is easy, let's do it. */
|
|
|
|
doBacktrace(frame);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
We arrive here after a panic or unhandled exception, when no OCD is detected. Dump the registers to the
|
|
|
|
serial port and either jump to the gdb stub, halt the CPU or reboot.
|
|
|
|
*/
|
|
|
|
static __attribute__((noreturn)) void commonErrorHandler(XtExcFrame *frame)
|
|
|
|
{
|
|
|
|
|
|
|
|
int core_id = xPortGetCoreID();
|
|
|
|
// start panic WDT to restart system if we hang in this handler
|
2019-08-14 22:54:07 +08:00
|
|
|
if (!rtc_wdt_is_on()) {
|
|
|
|
rtc_wdt_protect_off();
|
|
|
|
rtc_wdt_disable();
|
|
|
|
rtc_wdt_set_length_of_reset_signal(RTC_WDT_SYS_RESET_SIG, RTC_WDT_LENGTH_3_2us);
|
|
|
|
rtc_wdt_set_length_of_reset_signal(RTC_WDT_CPU_RESET_SIG, RTC_WDT_LENGTH_3_2us);
|
|
|
|
rtc_wdt_set_stage(RTC_WDT_STAGE0, RTC_WDT_STAGE_ACTION_RESET_SYSTEM);
|
|
|
|
// 64KB of core dump data (stacks of about 30 tasks) will produce ~85KB base64 data.
|
|
|
|
// @ 115200 UART speed it will take more than 6 sec to print them out.
|
|
|
|
rtc_wdt_set_time(RTC_WDT_STAGE0, 7000);
|
|
|
|
rtc_wdt_enable();
|
|
|
|
rtc_wdt_protect_on();
|
|
|
|
}
|
2019-05-10 11:34:06 +08:00
|
|
|
|
|
|
|
//Feed the watchdogs, so they will give us time to print out debug info
|
|
|
|
reconfigureAllWdts();
|
|
|
|
|
|
|
|
commonErrorHandler_dump(frame, core_id);
|
|
|
|
|
2019-09-13 15:49:11 +03:00
|
|
|
#if CONFIG_APPTRACE_ENABLE
|
2019-05-10 11:34:06 +08:00
|
|
|
disableAllWdts();
|
|
|
|
#if CONFIG_SYSVIEW_ENABLE
|
2019-09-13 15:49:11 +03:00
|
|
|
SEGGER_RTT_ESP32_FlushNoLock(CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH, APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
2019-05-10 11:34:06 +08:00
|
|
|
#else
|
2019-09-13 15:49:11 +03:00
|
|
|
esp_apptrace_flush_nolock(ESP_APPTRACE_DEST_TRAX, CONFIG_APPTRACE_POSTMORTEM_FLUSH_THRESH,
|
2019-05-10 11:34:06 +08:00
|
|
|
APPTRACE_ONPANIC_HOST_FLUSH_TMO);
|
|
|
|
#endif
|
|
|
|
reconfigureAllWdts();
|
|
|
|
#endif
|
|
|
|
|
2019-06-04 17:02:01 +10:00
|
|
|
#if CONFIG_ESP32S2_PANIC_GDBSTUB
|
2019-05-10 11:34:06 +08:00
|
|
|
disableAllWdts();
|
2019-08-14 22:54:07 +08:00
|
|
|
rtc_wdt_disable();
|
2019-05-10 11:34:06 +08:00
|
|
|
panicPutStr("Entering gdb stub now.\r\n");
|
|
|
|
esp_gdbstub_panic_handler(frame);
|
|
|
|
#else
|
|
|
|
#if CONFIG_ESP32_ENABLE_COREDUMP
|
|
|
|
static bool s_dumping_core;
|
|
|
|
if (s_dumping_core) {
|
|
|
|
panicPutStr("Re-entered core dump! Exception happened during core dump!\r\n");
|
|
|
|
} else {
|
|
|
|
disableAllWdts();
|
|
|
|
s_dumping_core = true;
|
|
|
|
#if CONFIG_ESP32_ENABLE_COREDUMP_TO_FLASH
|
|
|
|
esp_core_dump_to_flash(frame);
|
|
|
|
#endif
|
2019-06-10 15:07:12 +08:00
|
|
|
#if CONFIG_ESP32_ENABLE_COREDUMP_TO_UART && !CONFIG_ESP32S2_PANIC_SILENT_REBOOT
|
2019-05-10 11:34:06 +08:00
|
|
|
esp_core_dump_to_uart(frame);
|
|
|
|
#endif
|
|
|
|
s_dumping_core = false;
|
|
|
|
reconfigureAllWdts();
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_ESP32_ENABLE_COREDUMP */
|
2019-08-14 22:54:07 +08:00
|
|
|
rtc_wdt_disable();
|
2019-06-10 15:07:12 +08:00
|
|
|
#if CONFIG_ESP32S2_PANIC_PRINT_REBOOT || CONFIG_ESP32S2_PANIC_SILENT_REBOOT
|
2019-05-10 11:34:06 +08:00
|
|
|
panicPutStr("Rebooting...\r\n");
|
|
|
|
if (frame->exccause != PANIC_RSN_CACHEERR) {
|
|
|
|
esp_restart_noos();
|
|
|
|
} else {
|
|
|
|
// The only way to clear invalid cache access interrupt is to reset the digital part
|
|
|
|
esp_panic_dig_reset();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
disableAllWdts();
|
|
|
|
panicPutStr("CPU halted.\r\n");
|
|
|
|
while (1);
|
2019-06-04 17:02:01 +10:00
|
|
|
#endif /* CONFIG_ESP32S2_PANIC_PRINT_REBOOT || CONFIG_ESP32S2_PANIC_SILENT_REBOOT */
|
|
|
|
#endif /* CONFIG_ESP32S2_PANIC_GDBSTUB */
|
2019-05-10 11:34:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void esp_set_breakpoint_if_jtag(void *fn)
|
|
|
|
{
|
|
|
|
if (esp_cpu_in_ocd_debug_mode()) {
|
|
|
|
setFirstBreakpoint((uint32_t)fn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
esp_err_t esp_set_watchpoint(int no, void *adr, int size, int flags)
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
if (no < 0 || no > 1) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
if (flags & (~0xC0000000)) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
int dbreakc = 0x3F;
|
|
|
|
//We support watching 2^n byte values, from 1 to 64. Calculate the mask for that.
|
|
|
|
for (x = 0; x < 7; x++) {
|
|
|
|
if (size == (1 << x)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
dbreakc <<= 1;
|
|
|
|
}
|
|
|
|
if (x == 7) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
//Mask mask and add in flags.
|
|
|
|
dbreakc = (dbreakc & 0x3f) | flags;
|
|
|
|
|
|
|
|
if (no == 0) {
|
|
|
|
asm volatile(
|
|
|
|
"wsr.dbreaka0 %0\n" \
|
|
|
|
"wsr.dbreakc0 %1\n" \
|
|
|
|
::"r"(adr), "r"(dbreakc));
|
|
|
|
} else {
|
|
|
|
asm volatile(
|
|
|
|
"wsr.dbreaka1 %0\n" \
|
|
|
|
"wsr.dbreakc1 %1\n" \
|
|
|
|
::"r"(adr), "r"(dbreakc));
|
|
|
|
}
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void esp_clear_watchpoint(int no)
|
|
|
|
{
|
|
|
|
//Setting a dbreakc register to 0 makes it trigger on neither load nor store, effectively disabling it.
|
|
|
|
int dbreakc = 0;
|
|
|
|
if (no == 0) {
|
|
|
|
asm volatile(
|
|
|
|
"wsr.dbreakc0 %0\n" \
|
|
|
|
::"r"(dbreakc));
|
|
|
|
} else {
|
|
|
|
asm volatile(
|
|
|
|
"wsr.dbreakc1 %0\n" \
|
|
|
|
::"r"(dbreakc));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _esp_error_check_failed(esp_err_t rc, const char *file, int line, const char *function, const char *expression)
|
|
|
|
{
|
|
|
|
ets_printf("ESP_ERROR_CHECK failed: esp_err_t 0x%x", rc);
|
|
|
|
#ifdef CONFIG_ESP_ERR_TO_NAME_LOOKUP
|
|
|
|
ets_printf(" (%s)", esp_err_to_name(rc));
|
|
|
|
#endif //CONFIG_ESP_ERR_TO_NAME_LOOKUP
|
|
|
|
ets_printf(" at 0x%08x\n", (intptr_t)__builtin_return_address(0) - 3);
|
|
|
|
if (spi_flash_cache_enabled()) { // strings may be in flash cache
|
|
|
|
ets_printf("file: \"%s\" line %d\nfunc: %s\nexpression: %s\n", file, line, function, expression);
|
|
|
|
}
|
|
|
|
invoke_abort();
|
|
|
|
}
|