5#include <zephyr/drivers/hwinfo.h>
6#include <hal/nrf_power.h>
8#include <zephyr/storage/flash_map.h>
10#define BOOTLOADER_VERSION_REGISTER NRF_TIMER2->CC[0]
14static const char *
const TAG =
"debug";
18static void show_reset_reason(std::string &reset_reason,
bool set,
const char *reason) {
22 if (!reset_reason.empty()) {
25 reset_reason += reason;
28static inline uint32_t read_mem_u32(uintptr_t addr) {
29 return *
reinterpret_cast<volatile uint32_t *
>(addr);
32static inline uint8_t read_mem_u8(uintptr_t addr) {
33 return *
reinterpret_cast<volatile uint8_t *
>(addr);
43static inline bool is_sd_present() {
46static inline uint32_t sd_id_get() {
52static inline uint32_t sd_version_get() {
61 auto ret = hwinfo_get_reset_cause(&cause);
63 ESP_LOGE(TAG,
"Unable to get reset cause: %d", ret);
66 std::string reset_reason;
68 show_reset_reason(reset_reason, cause & RESET_PIN,
"External pin");
69 show_reset_reason(reset_reason, cause & RESET_SOFTWARE,
"Software reset");
70 show_reset_reason(reset_reason, cause & RESET_BROWNOUT,
"Brownout (drop in voltage)");
71 show_reset_reason(reset_reason, cause & RESET_POR,
"Power-on reset (POR)");
72 show_reset_reason(reset_reason, cause & RESET_WATCHDOG,
"Watchdog timer expiration");
73 show_reset_reason(reset_reason, cause & RESET_DEBUG,
"Debug event");
74 show_reset_reason(reset_reason, cause & RESET_SECURITY,
"Security violation");
75 show_reset_reason(reset_reason, cause & RESET_LOW_POWER_WAKE,
"Waking up from low power mode");
76 show_reset_reason(reset_reason, cause & RESET_CPU_LOCKUP,
"CPU lock-up detected");
77 show_reset_reason(reset_reason, cause & RESET_PARITY,
"Parity error");
78 show_reset_reason(reset_reason, cause & RESET_PLL,
"PLL error");
79 show_reset_reason(reset_reason, cause & RESET_CLOCK,
"Clock error");
80 show_reset_reason(reset_reason, cause & RESET_HARDWARE,
"Hardware reset");
81 show_reset_reason(reset_reason, cause & RESET_USER,
"User reset");
82 show_reset_reason(reset_reason, cause & RESET_TEMPERATURE,
"Temperature reset");
84 ESP_LOGD(TAG,
"Reset Reason: %s", reset_reason.c_str());
90static void fa_cb(
const struct flash_area *fa,
void *user_data) {
91#if CONFIG_FLASH_MAP_LABELS
92 const char *fa_label = flash_area_label(fa);
94 if (fa_label ==
nullptr) {
97 ESP_LOGCONFIG(TAG,
"%2d 0x%0*" PRIxPTR
" %-26s %-24.24s 0x%-10x 0x%-12x", (
int) fa->fa_id,
98 sizeof(uintptr_t) * 2, (uintptr_t) fa->fa_dev, fa->fa_dev->name, fa_label, (uint32_t) fa->fa_off,
101 ESP_LOGCONFIG(TAG,
"%2d 0x%0*" PRIxPTR
" %-26s 0x%-10x 0x%-12x", (
int) fa->fa_id,
sizeof(uintptr_t) * 2,
102 (uintptr_t) fa->fa_dev, fa->fa_dev->name, (uint32_t) fa->fa_off, fa->fa_size);
107#if CONFIG_FLASH_MAP_LABELS
108 ESP_LOGCONFIG(TAG,
"ID | Device | Device Name "
109 "| Label | Offset | Size");
110 ESP_LOGCONFIG(TAG,
"--------------------------------------------"
111 "-----------------------------------------------");
113 ESP_LOGCONFIG(TAG,
"ID | Device | Device Name "
115 ESP_LOGCONFIG(TAG,
"-----------------------------------------"
116 "------------------------------");
118 flash_area_foreach(fa_cb,
nullptr);
122 std::string supply =
"Main supply status: ";
123 if (nrf_power_mainregstatus_get(NRF_POWER) == NRF_POWER_MAINREGSTATUS_NORMAL) {
124 supply +=
"Normal voltage.";
126 supply +=
"High voltage.";
128 ESP_LOGD(TAG,
"%s", supply.c_str());
129 device_info +=
"|" + supply;
131 std::string reg0 =
"Regulator stage 0: ";
132 if (nrf_power_mainregstatus_get(NRF_POWER) == NRF_POWER_MAINREGSTATUS_HIGH) {
133 reg0 += nrf_power_dcdcen_vddh_get(NRF_POWER) ?
"DC/DC" :
"LDO";
135 switch (NRF_UICR->REGOUT0 & UICR_REGOUT0_VOUT_Msk) {
136 case (UICR_REGOUT0_VOUT_DEFAULT << UICR_REGOUT0_VOUT_Pos):
137 reg0 +=
"1.8V (default)";
139 case (UICR_REGOUT0_VOUT_1V8 << UICR_REGOUT0_VOUT_Pos):
142 case (UICR_REGOUT0_VOUT_2V1 << UICR_REGOUT0_VOUT_Pos):
145 case (UICR_REGOUT0_VOUT_2V4 << UICR_REGOUT0_VOUT_Pos):
148 case (UICR_REGOUT0_VOUT_2V7 << UICR_REGOUT0_VOUT_Pos):
151 case (UICR_REGOUT0_VOUT_3V0 << UICR_REGOUT0_VOUT_Pos):
154 case (UICR_REGOUT0_VOUT_3V3 << UICR_REGOUT0_VOUT_Pos):
163 ESP_LOGD(TAG,
"%s", reg0.c_str());
164 device_info +=
"|" + reg0;
166 std::string reg1 =
"Regulator stage 1: ";
167 reg1 += nrf_power_dcdcen_get(NRF_POWER) ?
"DC/DC" :
"LDO";
168 ESP_LOGD(TAG,
"%s", reg1.c_str());
169 device_info +=
"|" + reg1;
171 std::string usb_power =
"USB power state: ";
172 if (nrf_power_usbregstatus_vbusdet_get(NRF_POWER)) {
173 if (nrf_power_usbregstatus_outrdy_get(NRF_POWER)) {
175 usb_power +=
"ready";
178 usb_power +=
"connected (regulator is not ready)";
182 usb_power +=
"disconected";
184 ESP_LOGD(TAG,
"%s", usb_power.c_str());
185 device_info +=
"|" + usb_power;
188 nrf_power_pof_thr_t pof_thr;
190 pof_thr = nrf_power_pofcon_get(NRF_POWER, &enabled);
191 std::string pof =
"Power-fail comparator: ";
194 case POWER_POFCON_THRESHOLD_V17:
197 case POWER_POFCON_THRESHOLD_V18:
200 case POWER_POFCON_THRESHOLD_V19:
203 case POWER_POFCON_THRESHOLD_V20:
206 case POWER_POFCON_THRESHOLD_V21:
209 case POWER_POFCON_THRESHOLD_V22:
212 case POWER_POFCON_THRESHOLD_V23:
215 case POWER_POFCON_THRESHOLD_V24:
218 case POWER_POFCON_THRESHOLD_V25:
221 case POWER_POFCON_THRESHOLD_V26:
224 case POWER_POFCON_THRESHOLD_V27:
227 case POWER_POFCON_THRESHOLD_V28:
232 if (nrf_power_mainregstatus_get(NRF_POWER) == NRF_POWER_MAINREGSTATUS_HIGH) {
234 switch (nrf_power_pofcon_vddh_get(NRF_POWER)) {
235 case NRF_POWER_POFTHRVDDH_V27:
238 case NRF_POWER_POFTHRVDDH_V28:
241 case NRF_POWER_POFTHRVDDH_V29:
244 case NRF_POWER_POFTHRVDDH_V30:
247 case NRF_POWER_POFTHRVDDH_V31:
250 case NRF_POWER_POFTHRVDDH_V32:
253 case NRF_POWER_POFTHRVDDH_V33:
256 case NRF_POWER_POFTHRVDDH_V34:
259 case NRF_POWER_POFTHRVDDH_V35:
262 case NRF_POWER_POFTHRVDDH_V36:
265 case NRF_POWER_POFTHRVDDH_V37:
268 case NRF_POWER_POFTHRVDDH_V38:
271 case NRF_POWER_POFTHRVDDH_V39:
274 case NRF_POWER_POFTHRVDDH_V40:
277 case NRF_POWER_POFTHRVDDH_V41:
280 case NRF_POWER_POFTHRVDDH_V42:
288 ESP_LOGD(TAG,
"%s", pof.c_str());
289 device_info +=
"|" + pof;
291 auto package = [](uint32_t value) {
294 return "QIxx - 7x7 73-pin aQFN";
296 return "QFxx - 6x6 48-pin QFN";
298 return "CKxx - 3.544 x 3.607 WLCSP";
300 return "Unspecified";
303 ESP_LOGD(TAG,
"Code page size: %u, code size: %u, device id: 0x%08x%08x", NRF_FICR->CODEPAGESIZE, NRF_FICR->CODESIZE,
304 NRF_FICR->DEVICEID[1], NRF_FICR->DEVICEID[0]);
305 ESP_LOGD(TAG,
"Encryption root: 0x%08x%08x%08x%08x, Identity Root: 0x%08x%08x%08x%08x", NRF_FICR->ER[0],
306 NRF_FICR->ER[1], NRF_FICR->ER[2], NRF_FICR->ER[3], NRF_FICR->IR[0], NRF_FICR->IR[1], NRF_FICR->IR[2],
308 ESP_LOGD(TAG,
"Device address type: %s, address: %s", (NRF_FICR->DEVICEADDRTYPE & 0x1 ?
"Random" :
"Public"),
310 ESP_LOGD(TAG,
"Part code: nRF%x, version: %c%c%c%c, package: %s", NRF_FICR->INFO.PART,
311 NRF_FICR->INFO.VARIANT >> 24 & 0xFF, NRF_FICR->INFO.VARIANT >> 16 & 0xFF, NRF_FICR->INFO.VARIANT >> 8 & 0xFF,
312 NRF_FICR->INFO.VARIANT & 0xFF, package(NRF_FICR->INFO.PACKAGE));
313 ESP_LOGD(TAG,
"RAM: %ukB, Flash: %ukB, production test: %sdone", NRF_FICR->INFO.RAM, NRF_FICR->INFO.FLASH,
314 (NRF_FICR->PRODTEST[0] == 0xBB42319F ?
"" :
"not "));
315 bool n_reset_enabled = NRF_UICR->PSELRESET[0] == NRF_UICR->PSELRESET[1] &&
316 (NRF_UICR->PSELRESET[0] & UICR_PSELRESET_CONNECT_Msk) == UICR_PSELRESET_CONNECT_Connected
317 << UICR_PSELRESET_CONNECT_Pos;
319 TAG,
"GPIO as NFC pins: %s, GPIO as nRESET pin: %s",
320 YESNO((NRF_UICR->NFCPINS & UICR_NFCPINS_PROTECT_Msk) == (UICR_NFCPINS_PROTECT_NFC << UICR_NFCPINS_PROTECT_Pos)),
321 YESNO(n_reset_enabled));
322 if (n_reset_enabled) {
323 uint8_t port = (NRF_UICR->PSELRESET[0] & UICR_PSELRESET_PORT_Msk) >> UICR_PSELRESET_PORT_Pos;
324 uint8_t pin = (NRF_UICR->PSELRESET[0] & UICR_PSELRESET_PIN_Msk) >> UICR_PSELRESET_PIN_Pos;
325 ESP_LOGD(TAG,
"nRESET port P%u.%02u", port, pin);
327#ifdef USE_BOOTLOADER_MCUBOOT
328 ESP_LOGD(TAG,
"bootloader: mcuboot");
330 ESP_LOGD(TAG,
"bootloader: Adafruit, version %u.%u.%u", (BOOTLOADER_VERSION_REGISTER >> 16) & 0xFF,
331 (BOOTLOADER_VERSION_REGISTER >> 8) & 0xFF, BOOTLOADER_VERSION_REGISTER & 0xFF);
332 ESP_LOGD(TAG,
"MBR bootloader addr 0x%08x, UICR bootloader addr 0x%08x", read_mem_u32(
MBR_BOOTLOADER_ADDR),
334 ESP_LOGD(TAG,
"MBR param page addr 0x%08x, UICR param page addr 0x%08x", read_mem_u32(
MBR_PARAM_PAGE_ADDR),
336 if (is_sd_present()) {
338 uint32_t const sd_version = sd_version_get();
341 ver[0] = sd_version / 1000000;
342 ver[1] = (sd_version - ver[0] * 1000000) / 1000;
343 ver[2] = (sd_version - ver[0] * 1000000 - ver[1] * 1000);
345 ESP_LOGD(TAG,
"SoftDevice: S%u %u.%u.%u", sd_id, ver[0], ver[1], ver[2]);
346#ifdef USE_SOFTDEVICE_ID
347#ifdef USE_SOFTDEVICE_VERSION
348 if (USE_SOFTDEVICE_ID != sd_id || USE_SOFTDEVICE_VERSION != ver[0]) {
349 ESP_LOGE(TAG,
"Built for SoftDevice S%u %u.x.y. It may crash due to mismatch of bootloader version.",
350 USE_SOFTDEVICE_ID, USE_SOFTDEVICE_VERSION);
353 if (USE_SOFTDEVICE_ID != sd_id) {
354 ESP_LOGE(TAG,
"Built for SoftDevice S%u. It may crash due to mismatch of bootloader version.", USE_SOFTDEVICE_ID);
360 auto uicr = [](
volatile uint32_t *data, uint8_t size) {
363 for (
size_t i = 0; i < size; i++) {
371 ESP_LOGD(TAG,
"NRFFW %s", uicr(NRF_UICR->NRFFW, 13).c_str());
372 ESP_LOGD(TAG,
"NRFHW %s", uicr(NRF_UICR->NRFHW, 12).c_str());
375void DebugComponent::update_platform_() {}
void log_partition_info_()
Logs information about the device's partition table.
uint32_t get_free_heap_()
std::string get_reset_reason_()
void get_device_info_(std::string &device_info)
constexpr std::uintptr_t MBR_PARAM_PAGE_ADDR
constexpr uintptr_t SD_ID_OFFSET
constexpr std::uintptr_t MBR_BOOTLOADER_ADDR
constexpr uintptr_t MBR_SIZE
constexpr uintptr_t SOFTDEVICE_INFO_STRUCT_OFFSET
constexpr uintptr_t SD_VERSION_OFFSET
constexpr uint32_t SD_MAGIC_NUMBER
std::string get_mac_address_pretty()
Get the device MAC address as a string, in colon-separated uppercase hex notation.
std::string format_hex_pretty(const uint8_t *data, size_t length, char separator, bool show_length)
Format a byte array in pretty-printed, human-readable hex format.