| 
									
										
										
										
											2016-08-17 23:08:22 +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 "rom/rtc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "freertos/FreeRTOS.h"
 | 
					
						
							|  |  |  | #include "freertos/task.h"
 | 
					
						
							|  |  |  | #include "freertos/xtensa_api.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-13 23:02:03 +08:00
										 |  |  | #include "soc/uart_reg.h"
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | #include "soc/io_mux_reg.h"
 | 
					
						
							|  |  |  | #include "soc/dport_reg.h"
 | 
					
						
							|  |  |  | #include "soc/rtc_cntl_reg.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | #include "soc/timer_group_struct.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-26 12:23:01 +08:00
										 |  |  | #include "soc/timer_group_reg.h"
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-26 12:23:01 +08:00
										 |  |  | #include "esp_gdbstub.h"
 | 
					
						
							|  |  |  | #include "esp_panic.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-28 14:32:11 +08:00
										 |  |  | #include "esp_attr.h"
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | 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 also is in here. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 14:32:11 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  | Note: The linker script will put everything in this file in IRAM/DRAM, so it also works with flash cache disabled. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 11:17:24 +08:00
										 |  |  | #if !CONFIG_ESP32_PANIC_SILENT_REBOOT
 | 
					
						
							| 
									
										
										
										
											2016-08-23 20:14:54 +08:00
										 |  |  | //printf may be broken, so we fix our own printing fns...
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | inline static void panicPutchar(char c) { | 
					
						
							|  |  |  | 	while (((READ_PERI_REG(UART_STATUS_REG(0))>>UART_TXFIFO_CNT_S)&UART_TXFIFO_CNT)>=126) ; | 
					
						
							|  |  |  | 	WRITE_PERI_REG(UART_FIFO_REG(0), c); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline static void panicPutStr(const char *c) { | 
					
						
							|  |  |  | 	int x=0; | 
					
						
							|  |  |  | 	while (c[x]!=0) { | 
					
						
							|  |  |  | 		panicPutchar(c[x]); | 
					
						
							|  |  |  | 		x++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline static void panicPutHex(int a) { | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	int c; | 
					
						
							|  |  |  | 	panicPutchar(' '); | 
					
						
							|  |  |  | 	for (x=0; x<8; x++) { | 
					
						
							|  |  |  | 		c=(a>>28)&0xf; | 
					
						
							|  |  |  | 		if (c<10) panicPutchar('0'+c); else panicPutchar('a'+c-10); | 
					
						
							|  |  |  | 		a<<=4; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline static void panicPutDec(int a) { | 
					
						
							|  |  |  | 	int n1, n2; | 
					
						
							|  |  |  | 	n1=a%10; | 
					
						
							|  |  |  | 	n2=a/10; | 
					
						
							|  |  |  | 	panicPutchar(' '); | 
					
						
							|  |  |  | 	if (n2==0) panicPutchar(' '); else panicPutchar(n2+'0'); | 
					
						
							|  |  |  | 	panicPutchar(n1+'0'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | //No printing wanted. Stub out these functions.
 | 
					
						
							|  |  |  | inline static void panicPutchar(char c) { } | 
					
						
							|  |  |  | inline static void panicPutStr(const char *c) { } | 
					
						
							|  |  |  | inline static void panicPutHex(int a) { } | 
					
						
							|  |  |  | inline static void panicPutDec(int a) { } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void  __attribute__((weak)) vApplicationStackOverflowHook( TaskHandle_t xTask, signed char *pcTaskName ) { | 
					
						
							| 
									
										
										
										
											2016-10-02 02:04:09 +02:00
										 |  |  | 	panicPutStr("***ERROR*** A stack overflow in task "); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	panicPutStr((char*)pcTaskName); | 
					
						
							| 
									
										
										
										
											2016-10-02 02:04:09 +02:00
										 |  |  | 	panicPutStr(" has been detected.\r\n"); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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" | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void commonErrorHandler(XtExcFrame *frame); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //The fact that we've panic'ed probably means the other CPU is now running wild, possibly
 | 
					
						
							|  |  |  | //messing up the serial output, so we kill it here.
 | 
					
						
							|  |  |  | static void haltOtherCore() { | 
					
						
							|  |  |  | 	if (xPortGetCoreID()==0) { | 
					
						
							|  |  |  | 		//Kill app cpu
 | 
					
						
							| 
									
										
										
										
											2016-10-25 18:08:55 +08:00
										 |  |  | 		CLEAR_PERI_REG_MASK(RTC_CNTL_SW_CPU_STALL_REG, RTC_CNTL_SW_STALL_APPCPU_C1_M); | 
					
						
							|  |  |  | 		SET_PERI_REG_MASK(RTC_CNTL_SW_CPU_STALL_REG, 0x21<<RTC_CNTL_SW_STALL_APPCPU_C1_S); | 
					
						
							|  |  |  | 		CLEAR_PERI_REG_MASK(RTC_CNTL_OPTIONS0_REG, RTC_CNTL_SW_STALL_APPCPU_C0_M); | 
					
						
							|  |  |  | 		SET_PERI_REG_MASK(RTC_CNTL_OPTIONS0_REG, 2<<RTC_CNTL_SW_STALL_APPCPU_C0_S); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		//Kill pro cpu
 | 
					
						
							| 
									
										
										
										
											2016-10-25 18:08:55 +08:00
										 |  |  | 		CLEAR_PERI_REG_MASK(RTC_CNTL_SW_CPU_STALL_REG, RTC_CNTL_SW_STALL_PROCPU_C1_M); | 
					
						
							|  |  |  | 		SET_PERI_REG_MASK(RTC_CNTL_SW_CPU_STALL_REG, 0x21<<RTC_CNTL_SW_STALL_PROCPU_C1_S); | 
					
						
							|  |  |  | 		CLEAR_PERI_REG_MASK(RTC_CNTL_OPTIONS0_REG, RTC_CNTL_SW_STALL_PROCPU_C0_M); | 
					
						
							|  |  |  | 		SET_PERI_REG_MASK(RTC_CNTL_OPTIONS0_REG, 2<<RTC_CNTL_SW_STALL_PROCPU_C0_S); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //Returns true when a debugger is attached using JTAG.
 | 
					
						
							|  |  |  | static int inOCDMode() { | 
					
						
							| 
									
										
										
										
											2016-10-27 11:17:24 +08:00
										 |  |  | #if CONFIG_ESP32_DEBUG_OCDAWARE
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	int dcr; | 
					
						
							|  |  |  | 	int reg=0x10200C; //DSRSET register
 | 
					
						
							|  |  |  | 	asm("rer %0,%1":"=r"(dcr):"r"(reg)); | 
					
						
							|  |  |  | 	return (dcr&0x1); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	return 0; //Always return no debugger is attached.
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void panicHandler(XtExcFrame *frame) { | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	int *regs=(int*)frame; | 
					
						
							|  |  |  | 	//Please keep in sync with PANIC_RSN_* defines
 | 
					
						
							|  |  |  | 	const char *reasons[]={ | 
					
						
							|  |  |  | 			"Unknown reason", | 
					
						
							|  |  |  | 			"Unhandled debug exception", | 
					
						
							|  |  |  | 			"Double exception", | 
					
						
							|  |  |  | 			"Unhandled kernel exception", | 
					
						
							|  |  |  | 			"Coprocessor exception", | 
					
						
							| 
									
										
										
										
											2016-10-25 18:08:55 +08:00
										 |  |  | 			"Interrupt wdt timeout on CPU0", | 
					
						
							|  |  |  | 			"Interrupt wdt timeout on CPU1", | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 	const char *reason=reasons[0]; | 
					
						
							|  |  |  | 	//The panic reason is stored in the EXCCAUSE register.
 | 
					
						
							|  |  |  | 	if (regs[20]<=PANIC_RSN_MAX) reason=reasons[regs[20]]; | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	haltOtherCore(); | 
					
						
							|  |  |  | 	panicPutStr("Guru Meditation Error: Core "); | 
					
						
							|  |  |  | 	panicPutDec(xPortGetCoreID()); | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	panicPutStr(" panic'ed ("); | 
					
						
							|  |  |  | 	panicPutStr(reason); | 
					
						
							|  |  |  | 	panicPutStr(")\r\n"); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (inOCDMode()) { | 
					
						
							|  |  |  | 		asm("break.n 1"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	commonErrorHandler(frame); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-24 12:23:58 +08:00
										 |  |  | 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"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | void xt_unhandled_exception(XtExcFrame *frame) { | 
					
						
							|  |  |  | 	int *regs=(int*)frame; | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	haltOtherCore(); | 
					
						
							|  |  |  | 	panicPutStr("Guru Meditation Error of type "); | 
					
						
							|  |  |  | 	x=regs[20]; | 
					
						
							|  |  |  | 	if (x<40) panicPutStr(edesc[x]); else panicPutStr("Unknown"); | 
					
						
							| 
									
										
										
										
											2016-09-26 11:18:43 +08:00
										 |  |  | 	panicPutStr(" occurred on core "); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	panicPutDec(xPortGetCoreID()); | 
					
						
							|  |  |  | 	if (inOCDMode()) { | 
					
						
							|  |  |  | 		panicPutStr(" at pc="); | 
					
						
							|  |  |  | 		panicPutHex(regs[1]); | 
					
						
							|  |  |  | 		panicPutStr(". Setting bp and returning..\r\n"); | 
					
						
							|  |  |  | 		//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.
 | 
					
						
							| 
									
										
										
										
											2016-08-24 12:23:58 +08:00
										 |  |  | 		setFirstBreakpoint(regs[1]); | 
					
						
							| 
									
										
										
										
											2016-08-30 17:55:20 +08:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	panicPutStr(". Exception was unhandled.\r\n"); | 
					
						
							|  |  |  | 	commonErrorHandler(frame); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-26 12:23:01 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  | 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. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | static void reconfigureAllWdts() { | 
					
						
							| 
									
										
										
										
											2016-10-26 14:54:50 +08:00
										 |  |  | 	TIMERG0.wdt_wprotect=TIMG_WDT_WKEY_VALUE; | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	TIMERG0.wdt_feed=1; | 
					
						
							|  |  |  | 	TIMERG0.wdt_config0.sys_reset_length=7;				//3.2uS
 | 
					
						
							|  |  |  | 	TIMERG0.wdt_config0.cpu_reset_length=7;				//3.2uS
 | 
					
						
							| 
									
										
										
										
											2016-10-26 14:54:50 +08:00
										 |  |  | 	TIMERG0.wdt_config0.stg0=TIMG_WDT_STG_SEL_RESET_SYSTEM;	//1st stage timeout: reset system
 | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	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
 | 
					
						
							| 
									
										
										
										
											2016-10-26 14:54:50 +08:00
										 |  |  | 	TIMERG1.wdt_wprotect=TIMG_WDT_WKEY_VALUE; | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	TIMERG1.wdt_config0.en=0; | 
					
						
							|  |  |  | 	TIMERG1.wdt_wprotect=0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 11:17:24 +08:00
										 |  |  | #if CONFIG_ESP32_PANIC_GDBSTUB || CONFIG_ESP32_PANIC_PRINT_HALT
 | 
					
						
							| 
									
										
										
										
											2016-10-26 12:23:01 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  | This disables all the watchdogs for when we call the gdbstub. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | static void disableAllWdts() { | 
					
						
							| 
									
										
										
										
											2016-10-26 14:54:50 +08:00
										 |  |  | 	TIMERG0.wdt_wprotect=TIMG_WDT_WKEY_VALUE; | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	TIMERG0.wdt_config0.en=0; | 
					
						
							|  |  |  | 	TIMERG0.wdt_wprotect=0; | 
					
						
							| 
									
										
										
										
											2016-10-26 14:54:50 +08:00
										 |  |  | 	TIMERG1.wdt_wprotect=TIMG_WDT_WKEY_VALUE; | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	TIMERG1.wdt_config0.en=0; | 
					
						
							|  |  |  | 	TIMERG0.wdt_wprotect=0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 11:17:24 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  | 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. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | void commonErrorHandler(XtExcFrame *frame) { | 
					
						
							|  |  |  | 	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  "}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	//Feed the watchdogs, so they will give us time to print out debug info
 | 
					
						
							|  |  |  | 	reconfigureAllWdts(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	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(": "); | 
					
						
							|  |  |  | 				panicPutHex(regs[x+y+1]); | 
					
						
							|  |  |  | 				panicPutStr("  "); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		panicPutStr("\r\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-27 11:17:24 +08:00
										 |  |  | #if CONFIG_ESP32_PANIC_GDBSTUB
 | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	disableAllWdts(); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	panicPutStr("Entering gdb stub now.\r\n"); | 
					
						
							| 
									
										
										
										
											2016-10-26 12:23:01 +08:00
										 |  |  | 	esp_gdbstub_panic_handler(frame); | 
					
						
							| 
									
										
										
										
											2016-10-27 11:17:24 +08:00
										 |  |  | #elif CONFIG_ESP32_PANIC_PRINT_REBOOT || CONFIG_ESP32_PANIC_SILENT_REBOOT
 | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	panicPutStr("Rebooting...\r\n"); | 
					
						
							|  |  |  | 	for (x=0; x<100; x++) ets_delay_us(1000); | 
					
						
							|  |  |  | 	software_reset(); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-10-25 17:05:13 +08:00
										 |  |  | 	disableAllWdts(); | 
					
						
							| 
									
										
										
										
											2016-08-17 23:08:22 +08:00
										 |  |  | 	panicPutStr("CPU halted.\r\n"); | 
					
						
							|  |  |  | 	while(1); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-08-24 12:23:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-26 12:23:01 +08:00
										 |  |  | void esp_set_breakpoint_if_jtag(void *fn) { | 
					
						
							| 
									
										
										
										
											2016-08-24 12:23:58 +08:00
										 |  |  | 	if (!inOCDMode()) return; | 
					
						
							|  |  |  | 	setFirstBreakpoint((uint32_t)fn); | 
					
						
							|  |  |  | } |