diff --git a/slc/component/ot_crash_handler.slcc b/slc/component/ot_crash_handler.slcc
new file mode 100644
index 00000000..0b1743ee
--- /dev/null
+++ b/slc/component/ot_crash_handler.slcc
@@ -0,0 +1,63 @@
+id: ot_crash_handler
+label: Crash Handler
+package: OpenThread
+category: OpenThread
+quality: production
+description: |-
+ This component provides a set of APIs for printing crash info.
+ In the case of a crash, this component captures the details.
+ The provided `efr32PrintResetInfo()` API prints the crash details.
+provides:
+ - name: ot_crash_handler
+conflicts:
+ - name: legacy_hal_soc
+requires:
+ - name: component_catalog
+ - name: device
+ - name: emlib_rmu
+ - name: event_handler
+include:
+ - path: third_party/silabs/gecko_sdk/protocol/openthread/src/legacy_hal/include
+ file_list:
+ - path: crash_handler.h
+ - path: third_party/silabs/gecko_sdk/platform/service/legacy_hal/inc
+ file_list:
+ - path: asm.h
+ - path: efm32_micro.h
+ condition:
+ - device_cortexm
+ - path: micro-common.h
+ - path: micro-types.h
+ - path: micro.h
+ - path: reset-def.h
+ - path: platform-header.h
+ - path: cortexm3/diagnostic.h
+ condition:
+ - device_cortexm
+source:
+ - path: third_party/silabs/gecko_sdk/platform/service/legacy_hal/src/faults.s
+ - path: third_party/silabs/gecko_sdk/protocol/openthread/src/legacy_hal/crash_handler.c
+ - path: src/legacy_hal/diagnostic.c
+define:
+ - name: "PLATFORM_HEADER"
+ value: "\"platform-header.h\""
+ - name: CORTEXM3_EFM32_MICRO
+ condition:
+ - device_cortexm
+ - name: CORTEXM3
+ condition:
+ - device_cortexm
+ - name: CORTEXM3_EFR32
+ condition:
+ - device_cortexm
+ - name: PHY_RAIL
+ condition:
+ - device_cortexm
+template_contribution:
+ - name: component_catalog
+ value: ot_crash_handler
+ - name: event_handler
+ value:
+ event: platform_init
+ include: crash_handler.h
+ handler: sl_ot_crash_handler_init
diff --git a/slc/platform_projects/openthread-efr32-rcp-spi.slcp b/slc/platform_projects/openthread-efr32-rcp-spi.slcp
index 26ca9341..763fb5e7 100644
--- a/slc/platform_projects/openthread-efr32-rcp-spi.slcp
+++ b/slc/platform_projects/openthread-efr32-rcp-spi.slcp
@@ -7,6 +7,7 @@ quality: production
component:
- id: ot_crash_handler
+ from: ot-efr32
- id: ot_platform_abstraction_core
- id: ot_mbedtls
- id: ot_stack_features_config
diff --git a/slc/platform_projects/openthread-efr32-rcp-uart.slcp b/slc/platform_projects/openthread-efr32-rcp-uart.slcp
index 0807469d..7ca5d4a2 100644
--- a/slc/platform_projects/openthread-efr32-rcp-uart.slcp
+++ b/slc/platform_projects/openthread-efr32-rcp-uart.slcp
@@ -7,6 +7,7 @@ quality: production
component:
- id: ot_crash_handler
+ from: ot-efr32
- id: ot_platform_abstraction_core
- id: ot_mbedtls
- id: ot_stack_features_config
diff --git a/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager-csl.slcp b/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager-csl.slcp
index bc975502..83e87381 100644
--- a/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager-csl.slcp
+++ b/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager-csl.slcp
@@ -8,6 +8,7 @@ quality: production
component:
- id: ot_crash_handler
+ from: ot-efr32
- id: ot_platform_abstraction_core
- id: ot_psa_crypto
- id: ot_mbedtls
diff --git a/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager.slcp b/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager.slcp
index c0ee1363..23fb1adc 100644
--- a/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager.slcp
+++ b/slc/platform_projects/openthread-efr32-soc-with-buttons-power-manager.slcp
@@ -8,6 +8,7 @@ quality: production
component:
- id: ot_crash_handler
+ from: ot-efr32
- id: ot_platform_abstraction_core
- id: ot_psa_crypto
- id: ot_mbedtls
diff --git a/slc/platform_projects/openthread-efr32-soc-with-buttons.slcp b/slc/platform_projects/openthread-efr32-soc-with-buttons.slcp
index ce811797..d8685f1a 100644
--- a/slc/platform_projects/openthread-efr32-soc-with-buttons.slcp
+++ b/slc/platform_projects/openthread-efr32-soc-with-buttons.slcp
@@ -8,6 +8,7 @@ quality: production
component:
- id: ot_crash_handler
+ from: ot-efr32
- id: ot_platform_abstraction_core
- id: ot_psa_crypto
- id: ot_mbedtls
diff --git a/slc/platform_projects/openthread-efr32-soc.slcp b/slc/platform_projects/openthread-efr32-soc.slcp
index 404dda29..c87ac5aa 100644
--- a/slc/platform_projects/openthread-efr32-soc.slcp
+++ b/slc/platform_projects/openthread-efr32-soc.slcp
@@ -7,6 +7,7 @@ quality: production
component:
- id: ot_crash_handler
+ from: ot-efr32
- id: ot_platform_abstraction_core
- id: ot_psa_crypto
- id: ot_mbedtls
diff --git a/src/legacy_hal/diagnostic.c b/src/legacy_hal/diagnostic.c
new file mode 100644
index 00000000..2b5d480c
--- /dev/null
+++ b/src/legacy_hal/diagnostic.c
@@ -0,0 +1,1422 @@
+/***************************************************************************/
+/**
+ * @file
+ *
+ * @brief Diagnostic functions for the crash handler
+ *******************************************************************************
+ * # License
+ * Copyright 2023 Silicon Laboratories Inc. www.silabs.com
+ *******************************************************************************
+ *
+ * The licensor of this software is Silicon Laboratories Inc. Your use of this
+ * software is governed by the terms of Silicon Labs Master Software License
+ * Agreement (MSLA) available at
+ * www.silabs.com/about-us/legal/master-software-license-agreement. This
+ * software is distributed to you in Source Code format and is governed by the
+ * sections of the MSLA applicable to Source Code.
+ *
+ ******************************************************************************/
+/* clang-format off */
+
+#include PLATFORM_HEADER
+
+#include
+#include
+
+#include "crash_handler.h"
+
+#include "em_emu.h"
+#include "em_rmu.h"
+
+#include
+
+// Crash info live in noinit RAM segment that is not modified during startup.
+NO_INIT(HalCrashInfoType halCrashInfo);
+
+//------------------------------------------------------------------------------
+// Preprocessor definitions
+
+// Reserved instruction executed after a failed assert to cause a usage fault
+#define ASSERT_USAGE_OPCODE 0xDE42
+
+#if !defined(WDOG0)
+ #define WDOG0 WDOG
+ #define WDOG0_IRQn WDOG_IRQn
+#endif
+
+#ifdef RTOS
+ #include "rtos/rtos.h"
+ #define freeRTOS 1
+#else
+ #define freeRTOS 0
+#endif
+
+// Forward Declarations
+extern void sli_802154phy_radio_sleep(void);
+
+//------------------------------------------------------------------------------
+// Local Variables
+
+static const char * const cfsrBits[] =
+{
+ // Memory management (MPU) faults
+ "IACCVIOL: attempted instruction fetch from a no-execute address", // B0
+ "DACCVIOL: attempted load or store at an illegal address", // B1
+ "", // B2
+ "MUNSTKERR: unstack from exception return caused access violation", // B3
+ "MSTKERR: stacking from exception caused access violation", // B4
+ "", // B5
+ "", // B6
+ "MMARVALID: MMAR contains valid fault address", // B7
+
+ // Bus faults
+ "IBUSERR: instruction prefetch caused bus fault", // B8
+ "PRECISERR: precise data bus fault", // B9
+ "IMPRECISERR: imprecise data bus fault", // B10
+ "UNSTKERR: unstacking on exception return caused data bus fault", // B11
+ "STKERR: stacking on exception entry caused data bus fault", // B12
+ "", // B13
+ "", // B14
+ "BFARVALID: BFAR contains valid fault address", // B15
+
+ // Usage faults
+ "UNDEFINSTR: tried to execute an undefined instruction", // B16
+ "INVSTATE: invalid EPSR - e.g., tried to switch to ARM mode", // B17
+ "INVPC: exception return integrity checks failed", // B18
+ "NOCP: attempted to execute a coprocessor instruction", // B19
+ "", // B20
+ "", // B21
+ "", // B22
+ "", // B23
+ "UNALIGNED: attempted an unaligned memory access", // B24
+ "DIVBYZERO: attempted to execute SDIV or UDIV with divisor of 0" // B25
+};
+
+static const char * const intActiveBits[] =
+{
+#if defined (_EFR_DEVICE)
+ #if defined (_SILICON_LABS_32B_SERIES_1_CONFIG_1)
+ "EMU_IRQn", // B0
+ "FRC_PRI_IRQn", // B1
+ "WDOG0_IRQn", // B2
+ "FRC_IRQn", // B3
+ "MODEM_IRQn", // B4
+ "RAC_SEQ_IRQn", // B5
+ "RAC_RSM_IRQn", // B6
+ "BUFC_IRQn", // B7
+ "LDMA_IRQn", // B8
+ "GPIO_EVEN_IRQn", // B9
+ "TIMER0_IRQn", // B10
+ "USART0_RX_IRQn", // B11
+ "USART0_TX_IRQn", // B12
+ "ACMP0_IRQn", // B13
+ "ADC0_IRQn", // B14
+ "IDAC0_IRQn", // B15
+ "I2C0_IRQn", // B16
+ "GPIO_ODD_IRQn", // B17
+ "TIMER1_IRQn", // B18
+ "USART1_RX_IRQn", // B19
+ "USART1_TX_IRQn", // B20
+ "LEUART0_IRQn", // B21
+ "PCNT0_IRQn", // B22
+ "CMU_IRQn", // B23
+ "MSC_IRQn", // B24
+ "CRYPTO_IRQn", // B25
+ "LETIMER0_IRQn", // B26
+ "AGC_IRQn", // B27
+ "PROTIMER_IRQn", // B28
+ "RTCC_IRQn", // B29
+ "SYNTH_IRQn", // B30
+ "CRYOTIMER_IRQn", // B31
+ "RFSENSE_IRQn", // B32
+ "FPUEH_IRQn", // B33
+ #elif defined (_SILICON_LABS_32B_SERIES_1_CONFIG_2)
+ "EMU_IRQn", // B0
+ "FRC_PRI_IRQn", // B1
+ "WDOG0_IRQn", // B2
+ "WDOG1_IRQn", // B3
+ "FRC_IRQn", // B4
+ "MODEM_IRQn", // B5
+ "RAC_SEQ_IRQn", // B6
+ "RAC_RSM_IRQn", // B7
+ "BUFC_IRQn", // B8
+ "LDMA_IRQn", // B9
+ "GPIO_EVEN_IRQn", // B10
+ "TIMER0_IRQn", // B11
+ "USART0_RX_IRQn", // B12
+ "USART0_TX_IRQn", // B13
+ "ACMP0_IRQn", // B14
+ "ADC0_IRQn", // B15
+ "IDAC0_IRQn", // B16
+ "I2C0_IRQn", // B17
+ "GPIO_ODD_IRQn", // B18
+ "TIMER1_IRQn", // B19
+ "USART1_RX_IRQn", // B20
+ "USART1_TX_IRQn", // B21
+ "LEUART0_IRQn", // B22
+ "PCNT0_IRQn", // B23
+ "CMU_IRQn", // B24
+ "MSC_IRQn", // B25
+ "CRYPTO0_IRQn", // B26
+ "LETIMER0_IRQn", // B27
+ "AGC_IRQn", // B28
+ "PROTIMER_IRQn", // B29
+ "RTCC_IRQn", // B30
+ "SYNTH_IRQn", // B31
+ "CRYOTIMER_IRQn", // B32
+ "RFSENSE_IRQn", // B33
+ "FPUEH_IRQn", // B34
+ "SMU_IRQn", // B35
+ "WTIMER0_IRQn", // B36
+ "WTIMER1_IRQn", // B37
+ "PCNT1_IRQn", // B38
+ "PCNT2_IRQn", // B39
+ "USART2_RX_IRQn", // B40
+ "USART2_TX_IRQn", // B41
+ "I2C1_IRQn", // B42
+ "USART3_RX_IRQn", // B43
+ "USART3_TX_IRQn", // B44
+ "VDAC0_IRQn", // B45
+ "CSEN_IRQn", // B46
+ "LESENSE_IRQn", // B47
+ "CRYPTO1_IRQn", // B48
+ "TRNG0_IRQn", // B49
+ #elif defined (_SILICON_LABS_32B_SERIES_1_CONFIG_3)
+ "EMU_IRQn", // B0
+ "FRC_PRI_IRQn", // B1
+ "WDOG0_IRQn", // B2
+ "WDOG1_IRQn", // B3
+ "FRC_IRQn", // B4
+ "MODEM_IRQn", // B5
+ "RAC_SEQ_IRQn", // B6
+ "RAC_RSM_IRQn", // B7
+ "BUFC_IRQn", // B8
+ "LDMA_IRQn", // B9
+ "GPIO_EVEN_IRQn", // B10
+ "TIMER0_IRQn", // B11
+ "USART0_RX_IRQn", // B12
+ "USART0_TX_IRQn", // B13
+ "ACMP0_IRQn", // B14
+ "ADC0_IRQn", // B15
+ "IDAC0_IRQn", // B16
+ "I2C0_IRQn", // B17
+ "GPIO_ODD_IRQn", // B18
+ "TIMER1_IRQn", // B19
+ "USART1_RX_IRQn", // B20
+ "USART1_TX_IRQn", // B21
+ "LEUART0_IRQn", // B22
+ "PCNT0_IRQn", // B23
+ "CMU_IRQn", // B24
+ "MSC_IRQn", // B25
+ "CRYPTO0_IRQn", // B26
+ "LETIMER0_IRQn", // B27
+ "AGC_IRQn", // B28
+ "PROTIMER_IRQn", // B29
+ "PRORTC_IRQn", // B30
+ "RTCC_IRQn", // B31
+ "SYNTH_IRQn", // B32
+ "CRYOTIMER_IRQn", // B33
+ "RFSENSE_IRQn", // B34
+ "FPUEH_IRQn", // B35
+ "SMU_IRQn", // B36
+ "WTIMER0_IRQn", // B37
+ "USART2_RX_IRQn", // B38
+ "USART2_TX_IRQn", // B39
+ "I2C1_IRQn", // B40
+ "VDAC0_IRQn", // B41
+ "CSEN_IRQn", // B42
+ "LESENSE_IRQn", // B43
+ "CRYPTO1_IRQn", // B44
+ "TRNG0_IRQn" // B45
+ #elif defined (_SILICON_LABS_32B_SERIES_1_CONFIG_4)
+ "EMU_IRQn", // B0
+ "FRC_PRI_IRQn", // B1
+ "WDOG0_IRQn", // B2
+ "WDOG1_IRQn", // B3
+ "FRC_IRQn", // B4
+ "MODEM_IRQn", // B5
+ "RAC_SEQ_IRQn", // B6
+ "RAC_RSM_IRQn", // B7
+ "BUFC_IRQn", // B8
+ "LDMA_IRQn", // B9
+ "GPIO_EVEN_IRQn", // B10
+ "TIMER0_IRQn", // B11
+ "USART0_RX_IRQn", // B12
+ "USART0_TX_IRQn", // B13
+ "ACMP0_IRQn", // B14
+ "ADC0_IRQn", // B15
+ "IDAC0_IRQn", // B16
+ "I2C0_IRQn", // B17
+ "GPIO_ODD_IRQn", // B18
+ "TIMER1_IRQn", // B19
+ "USART1_RX_IRQn", // B20
+ "USART1_TX_IRQn", // B21
+ "LEUART0_IRQn", // B22
+ "PCNT0_IRQn", // B23
+ "CMU_IRQn", // B24
+ "MSC_IRQn", // B25
+ "CRYPTO0_IRQn", // B26
+ "LETIMER0_IRQn", // B27
+ "AGC_IRQn", // B28
+ "PROTIMER_IRQn", // B29
+ "PRORTC_IRQn", // B30
+ "RTCC_IRQn", // B31
+ "SYNTH_IRQn", // B32
+ "CRYOTIMER_IRQn", // B33
+ "RFSENSE_IRQn", // B34
+ "FPUEH_IRQn", // B35
+ "SMU_IRQn", // B36
+ "WTIMER0_IRQn", // B37
+ "VDAC0_IRQn", // B38
+ "LESENSE_IRQn", // B39
+ "TRNG0_IRQn", // B40
+ "SYSCFG_IRQn", // B41
+ #elif defined (_SILICON_LABS_32B_SERIES_2_CONFIG_1)
+ "SETAMPERHOST_IRQn", // B0
+ "SEMBRX_IRQn", // B1
+ "SEMBTX_IRQn", // B2
+ "SMU_SECURE_IRQn", // B3
+ "SMU_PRIVILEGED_IRQn", // B4
+ "EMU_IRQn", // B5
+ "TIMER0_IRQn", // B6
+ "TIMER1_IRQn", // B7
+ "TIMER2_IRQn", // B8
+ "TIMER3_IRQn", // B9
+ "RTCC_IRQn", // B10
+ "USART0_RX_IRQn", // B11
+ "USART0_TX_IRQn", // B12
+ "USART1_RX_IRQn", // B13
+ "USART1_TX_IRQn", // B14
+ "USART2_RX_IRQn", // B15
+ "USART2_TX_IRQn", // B16
+ "ICACHE0_IRQn", // B17
+ "BURTC_IRQn", // B18
+ "LETIMER0_IRQn", // B19
+ "SYSCFG_IRQn", // B20
+ "LDMA_IRQn", // B21
+ "LFXO_IRQn", // B22
+ "LFRCO_IRQn", // B23
+ "ULFRCO_IRQn", // B24
+ "GPIO_ODD_IRQn", // B25
+ "GPIO_EVEN_IRQn", // B26
+ "I2C0_IRQn", // B27
+ "I2C1_IRQn", // B28
+ "EMUDG_IRQn", // B29
+ "EMUSE_IRQn", // B30
+ "AGC_IRQn", // B31
+ "BUFC_IRQn", // B32
+ "FRC_PRI_IRQn", // B33
+ "FRC_IRQn", // B34
+ "MODEM_IRQn", // B35
+ "PROTIMER_IRQn", // B36
+ "RAC_RSM_IRQn", // B37
+ "RAC_SEQ_IRQn", // B38
+ "PRORTC_IRQn", // B39
+ "SYNTH_IRQn", // B40
+ "ACMP0_IRQn", // B41
+ "ACMP1_IRQn", // B42
+ "WDOG0_IRQn", // B43
+ "WDOG1_IRQn", // B44
+ "HFXO00_IRQn", // B45
+ "HFRCO0_IRQn", // B46
+ "HFRCOEM23_IRQn", // B47
+ "CMU_IRQn", // B48
+ "AES_IRQn", // B49
+ "IADC_IRQn", // B50
+ "MSC_IRQn", // B51
+ "DPLL0_IRQn", // B52
+ "SW0_IRQn", // B53
+ "SW1_IRQn", // B54
+ "SW2_IRQn", // B55
+ "SW3_IRQn", // B56
+ "KERNEL0_IRQn", // B57
+ "KERNEL1_IRQn", // B58
+ "M33CTI0_IRQn", // B59
+ "M33CTI1_IRQn", // B60
+ #elif defined (_SILICON_LABS_32B_SERIES_2_CONFIG_2)
+ "CRYPTOACC_IRQn", // B0
+ "TRNG_IRQn", // B1
+ "PKE_IRQn", // B2
+ "SMU_SECURE_IRQn", // B3
+ "SMU_S_PRIVILEGED_IRQn", // B4
+ "SMU_NS_PRIVILEGED_IRQn", // B5
+ "EMU_IRQn", // B6
+ "TIMER0_IRQn", // B7
+ "TIMER1_IRQn", // B8
+ "TIMER2_IRQn", // B9
+ "TIMER3_IRQn", // B10
+ "TIMER4_IRQn", // B11
+ "RTCC_IRQn", // B12
+ "USART0_RX_IRQn", // B13
+ "USART0_TX_IRQn", // B14
+ "USART1_RX_IRQn", // B15
+ "USART1_TX_IRQn", // B16
+ "ICACHE0_IRQn", // B17
+ "BURTC_IRQn", // B18
+ "LETIMER0_IRQn", // B19
+ "SYSCFG_IRQn", // B20
+ "LDMA_IRQn", // B21
+ "LFXO_IRQn", // B22
+ "LFRCO_IRQn", // B23
+ "ULFRCO_IRQn", // B24
+ "GPIO_ODD_IRQn", // B25
+ "GPIO_EVEN_IRQn", // B26
+ "I2C0_IRQn", // B27
+ "I2C1_IRQn", // B28
+ "EMUDG_IRQn", // B29
+ "EMUSE_IRQn", // B30
+ "AGC_IRQn", // B31
+ "BUFC_IRQn", // B32
+ "FRC_PRI_IRQn", // B33
+ "FRC_IRQn", // B34
+ "MODEM_IRQn", // B35
+ "PROTIMER_IRQn", // B36
+ "RAC_RSM_IRQn", // B37
+ "RAC_SEQ_IRQn", // B38
+ "RDMAILBOX_IRQn", // B39
+ "RFSENSE_IRQn", // B40
+ "PRORTC_IRQn", // B41
+ "SYNTH_IRQn", // B42
+ "WDOG0_IRQn", // B43
+ "HFXO0_IRQn", // B44
+ "HFRCO0_IRQn", // B45
+ "CMU_IRQn", // B46
+ "AES_IRQn", // B47
+ "IADC_IRQn", // B48
+ "MSC_IRQn", // B49
+ "DPLL0_IRQn", // B50
+ "PDM_IRQn", // B51
+ "SW0_IRQn", // B52
+ "SW1_IRQn", // B53
+ "SW2_IRQn", // B54
+ "SW3_IRQn", // B55
+ "KERNEL0_IRQn", // B56
+ "KERNEL1_IRQn", // B57
+ "M33CTI0_IRQn", // B58
+ "M33CTI1_IRQn", // B59
+ "EMUEFP_IRQn", // B60
+ "DCDC_IRQn", // B61
+ "EUART0_RX_IRQn", // B62
+ "EUART0_TX_IRQn", // B63
+#elif defined(_SILICON_LABS_32B_SERIES_2_CONFIG_3)
+ "SMU_SECURE_IRQn", // B0
+ "SMU_S_PRIVILEGED_IRQn", // B1
+ "SMU_NS_PRIVILEGED_IRQn", // B2
+ "EMU_IRQn", // B3
+ "TIMER0_IRQn", // B4
+ "TIMER1_IRQn", // B5
+ "TIMER2_IRQn", // B6
+ "TIMER3_IRQn", // B7
+ "TIMER4_IRQn", // B8
+ "USART0_RX_IRQn", // B9
+ "USART0_TX_IRQn", // B10
+ "EUSART0_RX_IRQn", // B11
+ "EUSART0_TX_IRQn", // B12
+ "EUSART1_RX_IRQn", // B13
+ "EUSART1_TX_IRQn", // B14
+ "EUSART2_RX_IRQn", // B15
+ "EUSART2_TX_IRQn", // B16
+ "ICACHE0_IRQn", // B17
+ "BURTC_IRQn", // B18
+ "LETIMER0_IRQn", // B19
+ "SYSCFG_IRQn", // B20
+ "MPAHBRAM_IRQn", // B21
+ "LDMA_IRQn", // B22
+ "LFXO_IRQn", // B23
+ "LFRCO_IRQn", // B24
+ "ULFRCO_IRQn", // B25
+ "GPIO_ODD_IRQn", // B26
+ "GPIO_EVEN_IRQn", // B27
+ "I2C0_IRQn", // B28
+ "I2C1_IRQn", // B29
+ "EMUDG_IRQn", // B30
+ "AGC_IRQn", // B31
+ "BUFC_IRQn", // B32
+ "FRC_PRI_IRQn", // B33
+ "FRC_IRQn", // B34
+ "MODEM_IRQn", // B35
+ "PROTIMER_IRQn", // B36
+ "RAC_RSM_IRQn", // B37
+ "RAC_SEQ_IRQn", // B38
+ "HOSTMAILBOX_IRQn", // B39
+ "SYNTH_IRQn", // B40
+ "ACMP0_IRQn", // B41
+ "ACMP1_IRQn", // B42
+ "WDOG0_IRQn", // B43
+ "WDOG1_IRQn", // B44
+ "HFXO0_IRQn", // B45
+ "HFRCO0_IRQn", // B46
+ "HFRCOEM23_IRQn", // B47
+ "CMU_IRQn", // B48
+ "AES_IRQn", // B49
+ "IADC_IRQn", // B50
+ "MSC_IRQn", // B51
+ "DPLL0_IRQn", // B52
+ "EMUEFP_IRQn", // B53
+ "DCDC_IRQn", // B54
+ "VDAC_IRQn", // B55
+ "PCNT0_IRQn", // B56
+ "SW0_IRQn", // B57
+ "SW1_IRQn", // B58
+ "SW2_IRQn", // B59
+ "SW3_IRQn", // B60
+ "KERNEL0_IRQn", // B61
+ "KERNEL1_IRQn", // B62
+ "M33CTI0_IRQn", // B63
+ "M33CTI1_IRQn", // B64
+ "FPUEXH_IRQn", // B65
+ "SEMBRX_IRQn", // B67
+ "SEMBTX_IRQn", // B68
+ "LESENSE_IRQn", // B69
+ "SYSRTC_APP_IRQn", // B70
+ "SYSRTC_SEQ_IRQn", // B71
+ "LCD_IRQn", // B72
+ "KEYSCAN_IRQn", // B73
+ "RFECA0_IRQn", // B74
+ "RFECA1_IRQn", // B75
+#elif defined(_SILICON_LABS_32B_SERIES_2_CONFIG_4)
+ "SMU_SECURE_IRQn", // B0
+ "SMU_S_PRIVILEGED_IRQn", // B1
+ "SMU_NS_PRIVILEGED_IRQn", // B2
+ "EMU_IRQn", // B3
+ "TIMER0_IRQn", // B4
+ "TIMER1_IRQn", // B5
+ "TIMER2_IRQn", // B6
+ "TIMER3_IRQn", // B7
+ "TIMER4_IRQn", // B8
+ "USART0_RX_IRQn", // B9
+ "USART0_TX_IRQn", // B10
+ "EUSART0_RX_IRQn", // B11
+ "EUSART0_TX_IRQn", // B12
+ "EUSART1_RX_IRQn", // B13
+ "EUSART1_TX_IRQn", // B14
+ "MVP_IRQn", // B15
+ "ICACHE0_IRQn", // B16
+ "BURTC_IRQn", // B17
+ "LETIMER0_IRQn", // B18
+ "SYSCFG_IRQn", // B19
+ "MPAHBRAM_IRQn", // B20
+ "LDMA_IRQn", // B21
+ "LFXO_IRQn", // B22
+ "LFRCO_IRQn", // B23
+ "ULFRCO_IRQn", // B24
+ "GPIO_ODD_IRQn", // B25
+ "GPIO_EVEN_IRQn", // B26
+ "I2C0_IRQn", // B27
+ "I2C1_IRQn", // B28
+ "EMUDG_IRQn", // B29
+ "AGC_IRQn", // B30
+ "BUFC_IRQn", // B31
+ "FRC_PRI_IRQn", // B32
+ "FRC_IRQn", // B33
+ "MODEM_IRQn", // B34
+ "PROTIMER_IRQn", // B35
+ "RAC_RSM_IRQn", // B36
+ "RAC_SEQ_IRQn", // B37
+ "HOSTMAILBOX_IRQn", // B38
+ "SYNTH_IRQn", // B39
+ "ACMP0_IRQn", // B40
+ "ACMP1_IRQn", // B41
+ "WDOG0_IRQn", // B42
+ "WDOG1_IRQn", // B43
+ "HFXO0_IRQn", // B44
+ "HFRCO0_IRQn", // B45
+ "HFRCOEM23_IRQn", // B46
+ "CMU_IRQn", // B47
+ "AES_IRQn", // B48
+ "IADC_IRQn", // B49
+ "MSC_IRQn", // B50
+ "DPLL0_IRQn", // B51
+ "EMUEFP_IRQn", // B52
+ "DCDC_IRQn", // B53
+ "PCNT0_IRQn", // B54
+ "SW0_IRQn", // B55
+ "SW1_IRQn", // B56
+ "SW2_IRQn", // B57
+ "SW3_IRQn", // B58
+ "KERNEL0_IRQn", // B59
+ "KERNEL1_IRQn", // B60
+ "FPUEXH_IRQn", // B61
+ "SETAMPERHOST_IRQn", // B62
+ "SEMBRX_IRQn", // B63
+ "SEMBTX_IRQn", // B64
+ "SYSRTC_APP_IRQn", // B65
+ "SYSRTC_SEQ_IRQn", // B66
+ "KEYSCAN_IRQn", // B67
+ "RFECA0_IRQn", // B68
+ "RFECA1_IRQn", // B69
+ "VDAC0_IRQn", // B70
+ "VDAC1_IRQn", // B71
+ "AHB2AHB0_IRQn", // B72
+ "AHB2AHB1_IRQn" // B73
+#elif defined (_SILICON_LABS_32B_SERIES_2_CONFIG_7)
+ "CRYPTOACC_IRQn", // B0
+ "TRNG_IRQn", // B1
+ "PKE_IRQn", // B2
+ "SMU_SECURE_IRQn", // B3
+ "SMU_S_PRIVILEGED_IRQn", // B4
+ "SMU_NS_PRIVILEGED_IRQn", // B5
+ "EMU_IRQn", // B6
+ "EMUEFP_IRQn", // B7
+ "DCDC_IRQn", // B8
+ "ETAMPDET_IRQn", // B9
+ "TIMER0_IRQn", // B10
+ "TIMER1_IRQn", // B11
+ "TIMER2_IRQn", // B12
+ "TIMER3_IRQn", // B13
+ "TIMER4_IRQn", // B14
+ "RTCC_IRQn", // B15
+ "USART0_RX_IRQn", // B16
+ "USART0_TX_IRQn", // B17
+ "USART1_RX_IRQn", // B18
+ "USART1_TX_IRQn", // B19
+ "EUSART0_RX_IRQn", // B20
+ "EUSART0_TX_IRQn", // B21
+ "ICACHE0_IRQn", // B22
+ "BURTC_IRQn", // B23
+ "LETIMER0_IRQn", // B24
+ "SYSCFG_IRQn", // B25
+ "LDMA_IRQn", // B26
+ "LFXO_IRQn", // B27
+ "LFRCO_IRQn", // B28
+ "ULFRCO_IRQn", // B29
+ "GPIO_ODD_IRQn", // B30
+ "GPIO_EVEN_IRQn", // B31
+ "I2C0_IRQn", // B32
+ "I2C1_IRQn", // B33
+ "EMUDG_IRQn", // B34
+ "EMUSE_IRQn", // B35
+ "AGC_IRQn", // B36
+ "BUFC_IRQn", // B37
+ "FRC_PRI_IRQn", // B38
+ "FRC_IRQn", // B39
+ "MODEM_IRQn", // B40
+ "PROTIMER_IRQn", // B41
+ "RAC_RSM_IRQn", // B42
+ "RAC_SEQ_IRQn", // B43
+ "RDMAILBOX_IRQn", // B44
+ "RFSENSE_IRQn", // B45
+ "SYNTH_IRQn", // B46
+ "PRORTC_IRQn", // B47
+ "ACMP0_IRQn", // B48
+ "WDOG0_IRQn", // B49
+ "HFXO0_IRQn", // B50
+ "HFRCO0_IRQn", // B51
+ "CMU_IRQn", // B52
+ "AES_IRQn", // B53
+ "IADC_IRQn", // B54
+ "MSC_IRQn", // B55
+ "DPLL0_IRQn", // B56
+ "PDM_IRQn", // B57
+ "SW0_IRQn", // B58
+ "SW1_IRQn", // B59
+ "SW2_IRQn", // B60
+ "SW3_IRQn", // B61
+ "KERNEL0_IRQn", // B62
+ "KERNEL1_IRQn", // B63
+ "M33CTI0_IRQn", // B64
+ "M33CTI1_IRQn", // B65
+ "FPUEXH_IRQn", // B66
+#elif defined (_SILICON_LABS_32B_SERIES_2_CONFIG_8)
+ "SMU_SECURE_IRQn", // B0
+ "SMU_S_PRIVILEGED_IRQn", // B1
+ "SMU_NS_PRIVILEGED_IRQn", // B2
+ "EMU_IRQn", // B3
+ "TIMER0_IRQn", // B4
+ "TIMER1_IRQn", // B5
+ "TIMER2_IRQn", // B6
+ "TIMER3_IRQn", // B7
+ "TIMER4_IRQn", // B8
+ "USART0_RX_IRQn", // B9
+ "USART0_TX_IRQn", // B10
+ "EUSART0_RX_IRQn", // B11
+ "EUSART0_TX_IRQn", // B12
+ "EUSART1_RX_IRQn", // B13
+ "EUSART1_TX_IRQn", // B14
+ "EUSART2_RX_IRQn", // B15
+ "EUSART2_TX_IRQn", // B16
+ "ICACHE0_IRQn", // B17
+ "BURTC_IRQn", // B18
+ "LETIMER0_IRQn", // B19
+ "SYSCFG_IRQn", // B20
+ "MPAHBRAM_IRQn", // B21
+ "LDMA_IRQn", // B22
+ "LFXO_IRQn", // B23
+ "LFRCO_IRQn", // B24
+ "ULFRCO_IRQn", // B25
+ "GPIO_ODD_IRQn", // B26
+ "GPIO_EVEN_IRQn", // B27
+ "I2C0_IRQn", // B28
+ "I2C1_IRQn", // B29
+ "EMUDG_IRQn", // B30
+ "AGC_IRQn", // B31
+ "BUFC_IRQn", // B32
+ "FRC_PRI_IRQn", // B33
+ "FRC_IRQn", // B34
+ "MODEM_IRQn", // B35
+ "PROTIMER_IRQn", // B36
+ "RAC_RSM_IRQn", // B37
+ "RAC_SEQ_IRQn", // B38
+ "HOSTMAILBOX_IRQn", // B39
+ "SYNTH_IRQn", // B40
+ "ACMP0_IRQn", // B41
+ "ACMP1_IRQn", // B42
+ "WDOG0_IRQn", // B43
+ "WDOG1_IRQn", // B44
+ "HFXO0_IRQn", // B45
+ "HFRCO0_IRQn", // B46
+ "HFRCOEM23_IRQn", // B47
+ "CMU_IRQn", // B48
+ "AES_IRQn", // B49
+ "IADC_IRQn", // B50
+ "MSC_IRQn", // B51
+ "DPLL0_IRQn", // B52
+ "EMUEFP_IRQn", // B53
+ "DCDC_IRQn", // B54
+ "VDAC0_IRQn", // B55
+ "PCNT0_IRQn", // B56
+ "SW0_IRQn", // B57
+ "SW1_IRQn", // B58
+ "SW2_IRQn", // B59
+ "SW3_IRQn", // B60
+ "KERNEL0_IRQn", // B61
+ "KERNEL1_IRQn", // B62
+ "M33CTI0_IRQn", // B63
+ "M33CTI1_IRQn", // B64
+ "FPUEXH_IRQn", // B65
+ "SETAMPERHOST_IRQn", // B66
+ "SEMBRX_IRQn", // B67
+ "SEMBTX_IRQn", // B68
+ "LESENSE_IRQn", // B69
+ "SYSRTC_APP_IRQn", // B70
+ "SYSRTC_SEQ_IRQn", // B71
+ "LCD_IRQn", // B72
+ "KEYSCAN_IRQn", // B73
+ "RFECA0_IRQn", // B74
+ "RFECA1_IRQn", // B75
+ "AHB2AHB0_IRQn", // B76
+ "AHB2AHB1_IRQn", // B77
+ "MVP_IRQn", // B78
+ #endif
+#elif defined (CORTEXM3_EFM32_MICRO)
+ "DMA", // B0
+ "GPIO_EVEN", // B1
+ "TIMER0", // B2
+ "USART0_RX", // B3
+ "USART0_TX", // B4
+ "USB", // B5
+ "ACMP0", // B6
+ "ADC0", // B7
+ "DAC0", // B8
+ "I2C0", // B9
+ "I2C1", // B10
+ "GPIO_ODD", // B11
+ "TIMER1", // B12
+ "TIMER2", // B13
+ "TIMER3", // B14
+ "USART1_RX", // B15
+ "USART1_TX", // B16
+ "LESENSE", // B17
+ "USART2_RX", // B18
+ "USART2_TX", // B19
+ "UART0_RX", // B20
+ "UART0_TX", // B21
+ "UART1_RX", // B22
+ "UART1_TX", // B23
+ "LEUART0", // B24
+ "LEUART1", // B25
+ "LETIMER0", // B26
+ "PCNT0", // B27
+ "PCNT1", // B28
+ "PCNT2", // B29
+ "RTC", // B30
+ "BURTC", // B31
+ "CMU", // B32
+ "VCMP", // B33
+ "LCD", // B34
+ "MSC", // B35
+ "AES", // B36
+ "EBI", // B37
+ "EMU", // B38
+#else
+ "Timer1", // B0
+ "Timer2", // B1
+ "Management", // B2
+ "Baseband", // B3
+ "Sleep_Timer", // B4
+ "SC1", // B5
+ "SC2", // B6
+ "Security", // B7
+ "MAC_Timer", // B8
+ "MAC_TX", // B9
+ "MAC_RX", // B10
+ "ADC", // B11
+ "IRQ_A", // B12
+ "IRQ_B", // B13
+ "IRQ_C", // B14
+ "IRQ_D", // B15
+ "Debug" // B16
+#endif
+};
+
+// Names of raw crash data items - each name is null terminated, and the
+// end of the array is flagged by two null bytes in a row.
+// NOTE: the order of these names must match HalCrashInfoType members.
+static const char nameStrings[] = "R0\0R1\0R2\0R3\0"
+ "R4\0R5\0R6\0R7\0"
+ "R8\0R9\0R10\0R11\0"
+ "R12\0R13(LR)\0MSP\0PSP\0"
+ "PC\0xPSR\0MSP used\0PSP used\0"
+ "CSTACK bottom\0ICSR\0SHCSR\0INT_ACTIVE0\0"
+ "INT_ACTIVE1\0"
+ "CFSR\0HFSR\0DFSR\0MMAR/BFAR\0AFSR\0"
+ "Ret0\0Ret1\0Ret2\0Ret3\0"
+ "Ret4\0Ret5\0Dat0\0Dat1\0";
+
+static uint16_t savedResetCause;
+static HalAssertInfoType savedAssertInfo;
+
+//------------------------------------------------------------------------------
+// Functions
+
+void halPrintCrashData(uint8_t port)
+{
+ (void)port;
+ uint32_t *data = (uint32_t*)&halCrashInfo.R0;
+ char const *name = nameStrings;
+ char const *separator;
+ uint8_t i = 0;
+ char outBuf[256] = {0};
+ uint8_t outBufIdx = 0;
+
+ while (*name != '\0') {
+ /*lint -save -e448 */
+ separator = ((*name != '\0') && ((i & 3) != 3)) ? ", " : "";
+ /*lint -restore */
+
+ outBufIdx += snprintf(outBuf+outBufIdx, sizeof outBuf,"%s = 0x%08lx%s", name, (unsigned long)*data++,separator);
+
+ // increment pointer to end of name
+ while (*name != '\0') {
+ name++;
+ }
+ // increment past null pointer for next name
+ name++;
+
+ i++;
+ if (!(i % 4))
+ {
+ otLogCritPlat(outBuf);
+ outBufIdx = 0;
+ }
+ }
+
+ // Print out remaining contents of outBuf
+ if (outBufIdx != 0)
+ {
+ otLogCritPlat(outBuf);
+ outBufIdx = 0;
+ }
+}
+
+void halPrintCrashDetails(uint8_t port)
+{
+ (void)port;
+
+ HalCrashInfoType *c = &halCrashInfo;
+ uint16_t reason = savedResetCause;
+ uint8_t bit;
+ const uint8_t numFaults = sizeof(cfsrBits) / sizeof(cfsrBits[0]);
+
+ // RESET_* are defined in `reset-def.h`
+ switch (reason) {
+ case RESET_WATCHDOG_EXPIRED:
+ otLogCritPlat("Reset cause: Watchdog expired, no reliable extra information");
+ break;
+ case RESET_WATCHDOG_CAUGHT:
+ otLogCritPlat("Reset cause: Watchdog caught with enhanced info");
+ otLogCritPlat("Instruction address: %4lx", (unsigned long)c->PC);
+ break;
+ case RESET_CRASH_ASSERT:
+ otLogCritPlat("Reset cause: Assert %s:%ld",
+ c->data.assertInfo.file, (long)c->data.assertInfo.line);
+ break;
+ case RESET_FAULT_HARD:
+ otLogCritPlat("Reset cause: Hard Fault");
+ if (c->hfsr.bits.VECTTBL) {
+ otLogCritPlat( "HFSR.VECTTBL: error reading vector table for an exception");
+ }
+ if (c->hfsr.bits.FORCED) {
+ otLogCritPlat( "HFSR.FORCED: configurable fault could not activate");
+ }
+ if (c->hfsr.bits.DEBUGEVT) {
+ otLogCritPlat( "HFSR.DEBUGEVT: fault related to debug - e.g., executed BKPT");
+ }
+ break;
+ case RESET_FAULT_MEM:
+ otLogCritPlat("Reset cause: Memory Management Fault");
+ if (c->cfsr.bits.DACCVIOL || c->cfsr.bits.IACCVIOL) {
+ otLogCritPlat("Instruction address: %4lx", (unsigned long)c->PC);
+ }
+ if (c->cfsr.bits.MMARVALID) {
+ otLogCritPlat("Illegal access address: %4lx", (unsigned long)c->faultAddress);
+ }
+ for (bit = SCB_CFSR_MEMFAULTSR_Pos; bit < (SCB_CFSR_MEMFAULTSR_Pos + 8); bit++) {
+ if ((c->cfsr.word & (1 << bit)) && (*cfsrBits[bit] != '\0')) {
+ otLogCritPlat("CFSR.%s", cfsrBits[bit]);
+ }
+ }
+ break;
+ case RESET_FAULT_BUS:
+ otLogCritPlat("Reset cause: Bus Fault");
+ otLogCritPlat("Instruction address: %4lx", (unsigned long)c->PC);
+ if (c->cfsr.bits.IMPRECISERR) {
+ otLogCritPlat( "Address is of an instruction after bus fault occurred, not the cause.");
+ }
+ if (c->cfsr.bits.BFARVALID) {
+ otLogCritPlat("Illegal access address: %4lx",
+ (unsigned long)c->faultAddress);
+ }
+ for (bit = SCB_CFSR_BUSFAULTSR_Pos; bit < SCB_CFSR_USGFAULTSR_Pos; bit++) {
+ if (((c->cfsr.word >> bit) & 1U) && (*cfsrBits[bit] != '\0')) {
+ otLogCritPlat("CFSR.%s", cfsrBits[bit]);
+ }
+ }
+ if ((c->cfsr.word & 0xFF) == 0) {
+ otLogCritPlat("CFSR.(none) load or store at an illegal address");
+ }
+ break;
+ case RESET_FAULT_USAGE:
+ otLogCritPlat("Reset cause: Usage Fault");
+ otLogCritPlat("Instruction address: %4lx", (unsigned long)c->PC);
+ for (bit = SCB_CFSR_USGFAULTSR_Pos;
+ (bit < numFaults) && (bit < (sizeof(c->cfsr.word) * 8));
+ bit++) {
+ if (((c->cfsr.word >> bit) & 1U) && (*cfsrBits[bit] != '\0')) {
+ otLogCritPlat("CFSR.%s", cfsrBits[bit]);
+ }
+ }
+ break;
+ case RESET_FAULT_DBGMON:
+ otLogCritPlat("Reset cause: Debug Monitor Fault");
+ otLogCritPlat("Instruction address: %4lx", (unsigned long)c->PC);
+ break;
+ default:
+ break;
+ }
+}
+
+void halPrintCrashSummary(uint8_t port)
+{
+ (void)port;
+
+ HalCrashInfoType *c = &halCrashInfo;
+ uint32_t sp, stackBegin, stackEnd, size, used;
+ uint16_t pct;
+ uint8_t *mode;
+ const char *stack;
+ uint8_t bit;
+
+ if (c->LR & 4) {
+ stack = "process";
+ sp = c->processSP;
+ used = c->processSPUsed;
+ stackBegin = 0;
+ stackEnd = 0;
+ } else {
+ stack = "main";
+ sp = c->mainSP;
+ used = c->mainSPUsed;
+ stackBegin = (uint32_t)c->mainStackBottom;
+ stackEnd = (uint32_t)(uint8_t *)_CSTACK_SEGMENT_END;
+ }
+
+ mode = (uint8_t *)((c->LR & 8) ? "Thread" : "Handler");
+ size = stackEnd - stackBegin;
+ pct = size ? (uint16_t)(((100 * used) + (size / 2)) / size) : 0;
+ otLogCritPlat("%s mode using %s stack (%4lx to %4lx), SP = %4lx",
+ mode, stack, (unsigned long)stackBegin, (unsigned long)stackEnd, (unsigned long)sp);
+ otLogCritPlat("%u bytes used (%u%%) in %s stack (out of %u bytes total)",
+ (uint16_t)used, pct, stack, (uint16_t)size);
+
+ // Valid SP range is [stackBegin, stackEnd] inclusive, but contents
+ // of stack only go into [stackBegin, stackend).
+ if ((sp > stackEnd) || (sp < stackBegin)) {
+ otLogCritPlat("SP is outside %s stack range!", stack);
+ }
+
+ if (c->intActive.word[0] || c->intActive.word[1]) {
+ otLogCritPlat("Interrupts active (or pre-empted and stacked):");
+ for (bit = 0; bit < 32; bit++) {
+ if ((c->intActive.word[0] & (1 << bit)) && (*intActiveBits[bit] != '\0')) {
+ otLogCritPlat(" %s", intActiveBits[bit]);
+ }
+ }
+ for (bit = 0; bit < (sizeof(intActiveBits) / sizeof(intActiveBits[0])) - 32; bit++) {
+ if ((c->intActive.word[1] & (1 << bit)) && (*intActiveBits[bit + 32] != '\0')) {
+ otLogCritPlat(" %s", intActiveBits[bit + 32]);
+ }
+ }
+ } else {
+ otLogCritPlat("No interrupts active");
+ }
+}
+
+void halStartPCDiagnostics(void)
+{
+}
+
+void halStopPCDiagnostics(void)
+{
+}
+
+uint16_t halGetPCDiagnostics(void)
+{
+ return 0;
+}
+
+//------------------------------------------------------------------------------
+
+void halInternalClassifyReset(void)
+{
+ // Table used to convert from RESET_EVENT register bits to reset types
+ static const uint16_t resetEventTable[] = {
+ #if defined (_SILICON_LABS_32B_SERIES_2)
+ RESET_POWERON_HV, // bit 0 : POR
+ RESET_EXTERNAL_PIN, // bit 1 : PIN
+ RESET_SOFTWARE_EM4, // bit 2 : EM4
+ RESET_WATCHDOG_EXPIRED, // bit 3 : WDOG0
+ RESET_WATCHDOG_EXPIRED, // bit 4 : WDOG1
+ RESET_FATAL_LOCKUP, // bit 5 : LOCKUP
+ RESET_SOFTWARE, // bit 6 : SYSREQ
+ RESET_BROWNOUT_DVDD, // bit 7 : DVDDBOD
+ RESET_UNKNOWN_UNKNOWN, // bit 8 : DVDDLEBOD // TODO: make new reset cause?
+ RESET_BROWNOUT_DEC, // bit 9 : DECBOD
+ RESET_BROWNOUT_AVDD, // bit 10 : AVDDBOD
+ RESET_UNKNOWN_UNKNOWN, // bit 11 : IOVDD0BOD // TODO: make new reset cause?
+ RESET_UNKNOWN_UNKNOWN, // bit 12 : RESERVED
+ RESET_UNKNOWN_UNKNOWN, // bit 13 : TAMPER // TODO: make new reset cause?
+ RESET_UNKNOWN_UNKNOWN, // bit 14 : M0SYSREQ // TODO: make new reset cause?
+ RESET_UNKNOWN_UNKNOWN, // bit 15 : M0LOCKUP // TODO: make new reset cause?
+ #elif defined (_EFR_DEVICE)
+ RESET_POWERON_HV, // bit 0: PORST
+ RESET_UNKNOWN_UNKNOWN, // bit 1: RESERVED
+ RESET_BROWNOUT_AVDD, // bit 2: AVDDBOD
+ RESET_BROWNOUT_DVDD, // bit 3: DVDDBOD
+ RESET_BROWNOUT_DEC, // bit 4: DECBOD
+ RESET_UNKNOWN_UNKNOWN, // bit 5: RESERVED
+ RESET_UNKNOWN_UNKNOWN, // bit 6: RESERVED
+ RESET_UNKNOWN_UNKNOWN, // bit 7: RESERVED
+ RESET_EXTERNAL_PIN, // bit 8: EXTRST
+ RESET_FATAL_LOCKUP, // bit 9: LOCKUPRST
+ RESET_SOFTWARE, // bit 10: SYSREQRST
+ RESET_WATCHDOG_EXPIRED, // bit 11: WDOGRST
+ RESET_UNKNOWN_UNKNOWN, // bit 12: RESERVED
+ RESET_UNKNOWN_UNKNOWN, // bit 13: RESERVED
+ RESET_UNKNOWN_UNKNOWN, // bit 14: RESERVED
+ RESET_UNKNOWN_UNKNOWN, // bit 15: RESERVED
+ RESET_SOFTWARE_EM4, // bit 16: EM4RST
+ #endif
+ };
+
+ uint32_t resetEvent = RMU_ResetCauseGet();
+ RMU_ResetCauseClear();
+ uint16_t cause = RESET_UNKNOWN;
+ uint16_t i;
+
+ HalResetCauseType *resetCause = (HalResetCauseType*)(RAM_MEM_BASE);
+
+ for (i = 0; i < sizeof(resetEventTable) / sizeof(resetEventTable[0]); i++) {
+ if (resetEvent & (1 << i)) {
+ cause = resetEventTable[i];
+ break;
+ }
+ }
+
+ if (cause == RESET_SOFTWARE) {
+ if ((resetCause->signature == RESET_VALID_SIGNATURE)
+ && (RESET_BASE_TYPE(resetCause->reason) < NUM_RESET_BASE_TYPES)) {
+ // The extended reset cause is recovered from RAM
+ // This can be trusted because the hardware reset event was software
+ // and additionally because the signature is valid
+ savedResetCause = resetCause->reason;
+ } else {
+ savedResetCause = RESET_SOFTWARE_UNKNOWN;
+ }
+ // mark the signature as invalid
+ resetCause->signature = RESET_INVALID_SIGNATURE;
+ } else if ((cause == RESET_BOOTLOADER_DEEPSLEEP)
+ && (resetCause->signature == RESET_VALID_SIGNATURE)
+ && (resetCause->reason == RESET_BOOTLOADER_DEEPSLEEP)) {
+ // Save the crash info for bootloader deep sleep (even though it's not used
+ // yet) and invalidate the reset signature.
+ resetCause->signature = RESET_INVALID_SIGNATURE;
+ savedResetCause = resetCause->reason;
+ } else {
+ savedResetCause = cause;
+ }
+
+ // If the last reset was due to an assert, save the assert info.
+ if (savedResetCause == RESET_CRASH_ASSERT) {
+ savedAssertInfo = halCrashInfo.data.assertInfo;
+ }
+}
+
+uint16_t halGetExtendedResetInfo(void)
+{
+ return savedResetCause;
+}
+
+const HalAssertInfoType *halGetAssertInfo(void)
+{
+ return &savedAssertInfo;
+}
+
+uint8_t halGetResetInfo(void)
+{
+ return RESET_BASE_TYPE(savedResetCause);
+}
+
+// Translate EM3xx reset codes to the codes previously used by the EM2xx.
+// If there is no corresponding code, return the EM3xx base code with bit 7 set.
+uint8_t halGetEm2xxResetInfo(void)
+{
+ uint8_t reset = halGetResetInfo();
+
+ // Any reset with an extended value field of zero is considered an unknown
+ // reset, except for FIB resets.
+ if ((RESET_EXTENDED_FIELD(halGetExtendedResetInfo()) == 0)
+ && (reset != RESET_FIB)) {
+ return EM2XX_RESET_UNKNOWN;
+ }
+
+ switch (reset) {
+ case RESET_UNKNOWN:
+ return EM2XX_RESET_UNKNOWN;
+ case RESET_BOOTLOADER:
+ return EM2XX_RESET_BOOTLOADER;
+ case RESET_EXTERNAL: // map pin resets to poweron for EM2xx compatibility
+// return EM2XX_RESET_EXTERNAL;
+ case RESET_POWERON:
+ return EM2XX_RESET_POWERON;
+ case RESET_WATCHDOG:
+ return EM2XX_RESET_WATCHDOG;
+ case RESET_SOFTWARE:
+ return EM2XX_RESET_SOFTWARE;
+ case RESET_CRASH:
+ return EM2XX_RESET_ASSERT;
+ default:
+ return (reset | 0x80); // set B7 for all other reset codes
+ }
+}
+
+const char * halGetResetString(void)
+{
+ // Table used to convert from reset types to reset strings.
+ #define RESET_BASE_DEF(basename, value, string) string,
+ #define RESET_EXT_DEF(basename, extname, extvalue, string) /*nothing*/
+ static const char resetStringTable[][4] = {
+ #include "reset-def.h"
+ };
+ #undef RESET_BASE_DEF
+ #undef RESET_EXT_DEF
+ uint8_t resetInfo = halGetResetInfo();
+ if (resetInfo >= (sizeof(resetStringTable) / sizeof(resetStringTable[0]))) {
+ return resetStringTable[0x00]; // return unknown
+ } else {
+ return resetStringTable[resetInfo];
+ }
+}
+
+const char * halGetExtendedResetString(void)
+{
+ // Create a table of reset strings for each extended reset type
+ typedef const char ResetStringTableType[][4];
+ #define RESET_BASE_DEF(basename, value, string) \
+ }; static ResetStringTableType basename##ResetStringTable = {
+ #define RESET_EXT_DEF(basename, extname, extvalue, string) string,
+ {
+ #include "reset-def.h"
+ };
+ #undef RESET_BASE_DEF
+ #undef RESET_EXT_DEF
+
+ // Create a table of pointers to each of the above tables
+ #define RESET_BASE_DEF(basename, value, string) (ResetStringTableType *)basename##ResetStringTable,
+ #define RESET_EXT_DEF(basename, extname, extvalue, string) /*nothing*/
+ static ResetStringTableType * const extendedResetStringTablePtrs[] = {
+ #include "reset-def.h"
+ };
+ #undef RESET_BASE_DEF
+ #undef RESET_EXT_DEF
+
+ uint16_t extResetInfo = halGetExtendedResetInfo();
+ // access the particular table of extended strings we are interested in
+ ResetStringTableType *extendedResetStringTable =
+ extendedResetStringTablePtrs[RESET_BASE_TYPE(extResetInfo)];
+
+ // return the string from within the proper table
+ return (*extendedResetStringTable)[((extResetInfo) & 0xFF)];
+}
+
+#if _SILICON_LABS_GECKO_INTERNAL_SDID == 80
+
+// Workaround for brownouts on Dumbo when DCDC is retimed and radio subsystem is reset
+__STATIC_INLINE void disableDcdcRetimingAndRcosync(void)
+{
+ // Ensure access to EMU registers
+ EMU_Unlock();
+ EMU_PowerUnlock();
+
+ // Don't need to disable retiming if DCDC is not powering DVDD
+ if ((EMU->PWRCFG & _EMU_PWRCFG_PWRCFG_MASK) != EMU_PWRCFG_PWRCFG_DCDCTODVDD) {
+ return;
+ }
+
+ // Ensure sequencer is halted
+ uint32_t clockEnable = *(volatile uint32_t *)(0x400E4000 + 0xC8);
+ volatile uint32_t *reg;
+
+ if (clockEnable & 0x4UL) {
+ reg = (volatile uint32_t *)(0x40084000UL + 0x40);
+ *reg = 0x1UL;
+ }
+
+ // If DCDC is in use, ensure retiming and rcosync are disabled
+ uint32_t dcdcMode = EMU->DCDCCTRL & _EMU_DCDCCTRL_DCDCMODE_MASK;
+ if ((dcdcMode == EMU_DCDCCTRL_DCDCMODE_LOWNOISE)
+ || (dcdcMode == EMU_DCDCCTRL_DCDCMODE_LOWPOWER)) {
+ BUS_RegBitWrite(&EMU->DCDCTIMING, 28, 0);
+ // EMU->DCDCRCOSC is internal, _EMU_DCDCRCOSC_RCOSYNC_SHIFT = 0
+ BUS_RegBitWrite((void *)(EMU_BASE + 0x74), 0, 0);
+ }
+}
+
+#else
+
+// Workaround not needed for dies other than Dumbo
+#define disableDcdcRetimingAndRcosync() ((void)0)
+
+#endif
+
+void halInternalSysReset(uint16_t extendedCause)
+{
+ HalResetCauseType *resetCause = (HalResetCauseType*)(RAM_MEM_BASE);
+ INTERRUPTS_OFF();
+ // Ensure DCDC settings are compatible with the upcoming radio subsystem reset
+ disableDcdcRetimingAndRcosync();
+
+ resetCause->reason = extendedCause;
+ resetCause->signature = RESET_VALID_SIGNATURE;
+ // force write to complete before reset
+ asm ("DMB");
+ NVIC_SystemReset();
+}
+
+// Cause a usage fault by executing a special UNDEFINED instruction.
+// The high byte (0xDE) is reserved to be undefined - the low byte (0x42)
+// is arbitrary and distiguishes a failed assert from other usage faults.
+// the fault handler with then decode this, grab the filename and linenumber
+// parameters from R0 and R1 and save the information for display after a reset
+#if defined (__ICCARM__)
+#pragma diag_suppress=Og014
+static void halInternalAssertFault(const char * filename, int linenumber)
+{
+ asm ("DC16 0DE42h");
+}
+#pragma diag_default=Og014
+#elif defined (__GNUC__)
+__attribute__((noinline))
+static void halInternalAssertFault(const char * filename, int linenumber)
+{
+ asm (".short 0xDE42" : : "r" (filename), "r" (linenumber));
+}
+#endif
+
+void halInternalAssertFailed(const char * filename, int linenumber)
+{
+ INTERRUPTS_OFF();
+
+ otLogCritPlat("\r\n[ASSERT:%s:%d]\r", filename, linenumber);
+
+#if defined (__ICCARM__) || defined (__GNUC__)
+ // We can use the special fault mechanism to preserve more assert
+ // information for display after a crash
+ halInternalAssertFault(filename, linenumber);
+#else
+ // Other toolchains don't handle the inline assembly correctly, so
+ // we just call the internal reset
+ halCrashInfo.data.assertInfo.file = filename;
+ halCrashInfo.data.assertInfo.line = linenumber;
+ halInternalSysReset(RESET_CRASH_ASSERT);
+#endif
+}
+
+// Returns the bytes used in the main stack area.
+static uint32_t halInternalGetMainStackBytesUsed(uint32_t *p)
+{
+ for (; p < (uint32_t *)_CSTACK_SEGMENT_END; p++) {
+ if (*p != STACK_FILL_VALUE) {
+ break;
+ }
+ }
+ return (uint32_t)((uint8_t *)_CSTACK_SEGMENT_END - (uint8_t *)p);
+}
+
+// After the low-level fault handler (in faults.s79) has saved the processor
+// registers (R0-R12, LR and both MSP an PSP), it calls halInternalCrashHandler
+// to finish saving additional crash data. This function returns the reason for
+// the crash to the low-level fault handler that then calls
+// halInternalSystsemReset() to reset the processor.
+uint16_t halInternalCrashHandler(void)
+{
+ uint32_t activeException;
+ uint16_t reason = (uint16_t)RESET_FAULT_UNKNOWN;
+ HalCrashInfoType *c = &halCrashInfo;
+ uint8_t i, j;
+ uint32_t *sp, *s, *sEnd, *stackBottom, *stackTop;
+ uint32_t data;
+
+ c->icsr.word = SCB->ICSR;
+ c->shcsr.word = SCB->SHCSR;
+
+ c->intActive.word[0] = NVIC->IABR[0];
+ c->intActive.word[1] = NVIC->IABR[1];
+
+ c->cfsr.word = SCB->CFSR;
+ c->hfsr.word = SCB->HFSR;
+ c->dfsr.word = SCB->DFSR;
+ c->faultAddress = SCB->MMFAR;
+ c->afsr.word = SCB->AFSR;
+
+ // Examine B2 of the saved LR to know the stack in use when the fault occurred
+ sp = (uint32_t *)(((c->LR & 4U) != 0U) ? c->processSP : c->mainSP);
+ sEnd = sp; // Keep a copy around for walking the stack later
+
+ // Get the bottom of the stack since we allow stack resizing
+ c->mainStackBottom = (uint32_t)_CSTACK_SEGMENT_BEGIN;
+
+ // If we're running FreeRTOS and this is a process stack then add
+ // extra diagnostic information
+ if ((freeRTOS != 0) && ((c->LR & 4U) != 0U)) {
+ stackBottom = sp;
+ stackTop = sp + 8;
+ c->processSPUsed = stackTop - sp;
+ } else {
+ stackBottom = (uint32_t*)c->mainStackBottom;
+ stackTop = (uint32_t*)_CSTACK_SEGMENT_END;
+ c->processSPUsed = 0; // process stack not in use
+ }
+
+ // If the stack pointer is valid, read and save the stacked PC and xPSR
+ if ((sp >= stackBottom)
+ && ((sp + 8) <= stackTop)) {
+ sp += 6; // Skip over R0,R1,R2,R3,R12,LR
+ c->PC = *sp++;
+ c->xPSR.word = *sp++;
+
+ // See if fault was due to a failed assert. This is indicated by
+ // a usage fault caused by executing a reserved instruction.
+ if ( c->icsr.bits.VECTACTIVE == USAGE_FAULT_VECTOR_INDEX
+ && ((uint16_t *)c->PC >= (uint16_t *)_TEXT_SEGMENT_BEGIN)
+ && ((uint16_t *)c->PC < (uint16_t *)_TEXT_SEGMENT_END)
+ && *(uint16_t *)(c->PC) == ASSERT_USAGE_OPCODE ) {
+ // Copy halInternalAssertFailed() arguments into data member specific
+ // to asserts.
+ c->data.assertInfo.file = (const char *)c->R0;
+ c->data.assertInfo.line = c->R1;
+#ifdef PUSH_REGS_BEFORE_ASSERT
+ // Just before calling halInternalAssertFailed(), R0, R1, R2 and LR were
+ // pushed onto the stack - copy these values into the crash data struct.
+ c->R0 = *sp++;
+ c->R1 = *sp++;
+ c->R2 = *sp++;
+ c->LR = *sp++;
+#endif
+ reason = (uint16_t)RESET_CRASH_ASSERT;
+ }
+ // If a bad stack pointer, PC and xPSR to 0 to indicate they are not known.
+ } else {
+ c->PC = 0;
+ c->xPSR.word = 0;
+ sEnd = stackBottom;
+ }
+
+ c->mainSPUsed = halInternalGetMainStackBytesUsed((uint32_t*)c->mainStackBottom);
+
+ for (i = 0; i < NUM_RETURNS; i++) {
+ c->returns[i] = 0;
+ }
+
+ // Search the stack downward for probable return addresses. A probable
+ // return address is a value in the CODE segment that also has bit 0 set
+ // (since we're in Thumb mode).
+ i = 0U;
+ s = stackTop;
+ while (s > sEnd) {
+ data = *(--s);
+ if (((uint16_t *)data >= (uint16_t *)_TEXT_SEGMENT_BEGIN)
+ && ((uint16_t *)data < (uint16_t *)_TEXT_SEGMENT_END)
+ && ((data & 1U) != 0U)) {
+ // Only record the first occurrence of a return - other copies could
+ // have been in registers that then were pushed.
+ for (j = 0; j < NUM_RETURNS; j++) {
+ if (c->returns[j] == data) {
+ break;
+ }
+ }
+ if (j != NUM_RETURNS) {
+ continue;
+ }
+ // Save the return in the returns array managed as a circular buffer.
+ // This keeps only the last NUM_RETURNS in the event that there are more.
+ i = (i != 0U) ? i - 1U : NUM_RETURNS - 1U;
+ c->returns[i] = data;
+ }
+ }
+ // Shuffle the returns array so returns[0] has last probable return found.
+ // If there were fewer than NUM_RETURNS, unused entries will contain zero.
+ while ((i--) != 0U) {
+ data = c->returns[0];
+ for (j = 0; j < NUM_RETURNS - 1U; j++ ) {
+ c->returns[j] = c->returns[j + 1U];
+ }
+ c->returns[NUM_RETURNS - 1U] = data;
+ }
+
+ // Read the highest priority active exception to get reason for fault
+ activeException = c->icsr.bits.VECTACTIVE;
+ switch (activeException) {
+ #if defined(WDOG_IF_WARN) && !defined(BOOTLOADER)
+ case IRQ_TO_VECTOR_NUMBER(WDOG0_IRQn):
+ if (WDOG0->IF & WDOG_IF_WARN) {
+ reason = RESET_WATCHDOG_CAUGHT;
+ }
+ break;
+ #endif
+ case HARD_FAULT_VECTOR_INDEX:
+ reason = (uint16_t)RESET_FAULT_HARD;
+ break;
+ case MEMORY_FAULT_VECTOR_INDEX:
+ reason = (uint16_t)RESET_FAULT_MEM;
+ break;
+ case BUS_FAULT_VECTOR_INDEX:
+ reason = (uint16_t)RESET_FAULT_BUS;
+ break;
+ case USAGE_FAULT_VECTOR_INDEX:
+ // make sure we didn't already identify the usage fault as an assert
+ if (reason == (uint16_t)RESET_FAULT_UNKNOWN) {
+ reason = (uint16_t)RESET_FAULT_USAGE;
+ }
+ break;
+ case DEBUG_MONITOR_VECTOR_INDEX:
+ reason = (uint16_t)RESET_FAULT_DBGMON;
+ break;
+ default:
+ if ((activeException != 0U) && (activeException < VECTOR_TABLE_LENGTH)) {
+ reason = (uint16_t)RESET_FAULT_BADVECTOR;
+ }
+ break;
+ }
+ return reason;
+}