| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223 |
- // 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 <string.h>
- #include <stdbool.h>
- #include "rom/crc.h"
- #include "esp_panic.h"
- #include "esp_partition.h"
- #include "esp_core_dump_priv.h"
- const static DRAM_ATTR char TAG[] __attribute__((unused)) = "esp_core_dump_common";
- #if CONFIG_ESP32_ENABLE_COREDUMP
- static esp_err_t esp_core_dump_write_binary(void *frame, core_dump_write_config_t *write_cfg)
- {
- esp_err_t err;
- core_dump_task_header_t tasks[CONFIG_ESP32_CORE_DUMP_MAX_TASKS_NUM];
- uint32_t tcb_sz, task_num, tcb_sz_padded;
- bool task_is_valid = false;
- uint32_t data_len = 0, i;
- union
- {
- core_dump_header_t hdr;
- core_dump_task_header_t task_hdr;
- } dump_data;
- task_num = esp_core_dump_get_tasks_snapshot(tasks, CONFIG_ESP32_CORE_DUMP_MAX_TASKS_NUM, &tcb_sz);
- ESP_COREDUMP_LOGI("Found tasks: (%d)!", task_num);
-
- // Take TCB padding into account, actual TCB size will be stored in header
- if (tcb_sz % sizeof(uint32_t))
- tcb_sz_padded = (tcb_sz / sizeof(uint32_t) + 1) * sizeof(uint32_t);
- else
- tcb_sz_padded = tcb_sz;
-
- // Verifies all tasks in the snapshot
- for (i = 0; i < task_num; i++) {
- task_is_valid = esp_core_dump_process_tcb(frame, &tasks[i], tcb_sz);
- // Check if task tcb is corrupted
- if (!task_is_valid) {
- write_cfg->bad_tasks_num++;
- continue;
- } else {
- data_len += (tcb_sz_padded + sizeof(core_dump_task_header_t));
- }
- uint32_t len = 0;
- task_is_valid = esp_core_dump_process_stack(&tasks[i], &len);
- if (task_is_valid) {
- // Increase core dump size by task stack size
- data_len += len;
- } else {
- // If task tcb is ok but stack is corrupted
- write_cfg->bad_tasks_num++;
- }
- }
- // Add core dump header size
- data_len += sizeof(core_dump_header_t);
- ESP_COREDUMP_LOG_PROCESS("Core dump len = %lu (%d %d)", data_len, task_num, write_cfg->bad_tasks_num);
-
- // Prepare write
- if (write_cfg->prepare) {
- err = write_cfg->prepare(write_cfg->priv, &data_len);
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to prepare core dump (%d)!", err);
- return err;
- }
- }
- // Write start
- if (write_cfg->start) {
- err = write_cfg->start(write_cfg->priv);
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to start core dump (%d)!", err);
- return err;
- }
- }
- // Write header
- dump_data.hdr.data_len = data_len;
- dump_data.hdr.version = COREDUMP_VERSION;
- dump_data.hdr.tasks_num = task_num - write_cfg->bad_tasks_num;
- dump_data.hdr.tcb_sz = tcb_sz;
- err = write_cfg->write(write_cfg->priv, &dump_data, sizeof(core_dump_header_t));
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to write core dump header (%d)!", err);
- return err;
- }
- // Write tasks
- for (i = 0; i < task_num; i++) {
- if (!esp_tcb_addr_is_sane((uint32_t)tasks[i].tcb_addr, tcb_sz)) {
- ESP_COREDUMP_LOG_PROCESS("Skip TCB with bad addr %x!", tasks[i].tcb_addr);
- continue;
- }
- ESP_COREDUMP_LOG_PROCESS("Dump task %x", tasks[i].tcb_addr);
- // Save TCB address, stack base and stack top addr
- dump_data.task_hdr.tcb_addr = tasks[i].tcb_addr;
- dump_data.task_hdr.stack_start = tasks[i].stack_start;
- dump_data.task_hdr.stack_end = tasks[i].stack_end;
- err = write_cfg->write(write_cfg->priv, (void*)&dump_data, sizeof(core_dump_task_header_t));
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to write task header (%d)!", err);
- return err;
- }
- // Save TCB
- err = write_cfg->write(write_cfg->priv, tasks[i].tcb_addr, tcb_sz);
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to write TCB (%d)!", err);
- return err;
- }
- // Save task stack
- if (tasks[i].stack_start != 0 && tasks[i].stack_end != 0) {
- err = write_cfg->write(write_cfg->priv, (void*)tasks[i].stack_start,
- tasks[i].stack_end - tasks[i].stack_start);
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to write task stack (%d)!", err);
- return err;
- }
- } else {
- ESP_COREDUMP_LOG_PROCESS("Skip corrupted task %x stack!", tasks[i].tcb_addr);
- }
- }
- // write end
- if (write_cfg->end) {
- err = write_cfg->end(write_cfg->priv);
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Failed to end core dump (%d)!", err);
- return err;
- }
- }
- if (write_cfg->bad_tasks_num) {
- ESP_COREDUMP_LOGE("Skipped %d tasks with bad TCB!", write_cfg->bad_tasks_num);
- }
- return err;
- }
- inline void esp_core_dump_write(void *frame, core_dump_write_config_t *write_cfg)
- {
- esp_err_t err = esp_core_dump_write_binary(frame, write_cfg);
- if (err != ESP_OK) {
- ESP_COREDUMP_LOGE("Core dump write binary failed with error: %d", err);
- }
- }
- #endif
- void esp_core_dump_init()
- {
- #if CONFIG_ESP32_ENABLE_COREDUMP_TO_FLASH
- esp_core_dump_flash_init();
- #endif
- #if CONFIG_ESP32_ENABLE_COREDUMP_TO_UART
- ESP_COREDUMP_LOGI("Init core dump to UART");
- #endif
- }
- esp_err_t esp_core_dump_image_get(size_t* out_addr, size_t *out_size)
- {
- esp_err_t err;
- const void *core_data;
- spi_flash_mmap_handle_t core_data_handle;
- if (out_addr == NULL || out_size == NULL) {
- return ESP_ERR_INVALID_ARG;
- }
- const esp_partition_t *core_part = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_COREDUMP, NULL);
- if (!core_part) {
- ESP_LOGE(TAG, "No core dump partition found!");
- return ESP_FAIL;
- }
- if (core_part->size < sizeof(uint32_t)) {
- ESP_LOGE(TAG, "Too small core dump partition!");
- return ESP_FAIL;
- }
- err = esp_partition_mmap(core_part, 0, sizeof(uint32_t),
- SPI_FLASH_MMAP_DATA, &core_data, &core_data_handle);
- if (err != ESP_OK) {
- ESP_LOGE(TAG, "Failed to mmap core dump data (%d)!", err);
- return err;
- }
- uint32_t *dw = (uint32_t *)core_data;
- *out_size = *dw;
- spi_flash_munmap(core_data_handle);
- if (*out_size == 0xFFFFFFFF) {
- ESP_LOGD(TAG, "Blank core dump partition!");
- return ESP_ERR_INVALID_SIZE;
- } else if ((*out_size < sizeof(uint32_t)) || (*out_size > core_part->size)) {
- ESP_LOGE(TAG, "Incorrect size of core dump image: %d", *out_size);
- return ESP_ERR_INVALID_SIZE;
- }
- // remap full core dump with CRC
- err = esp_partition_mmap(core_part, 0, *out_size,
- SPI_FLASH_MMAP_DATA, &core_data, &core_data_handle);
- if (err != ESP_OK) {
- ESP_LOGE(TAG, "Failed to mmap core dump data (%d)!", err);
- return err;
- }
- uint32_t *crc = (uint32_t *)(((uint8_t *)core_data) + *out_size);
- crc--; // Point to CRC field
- // Calc CRC over core dump data except for CRC field
- core_dump_crc_t cur_crc = crc32_le(0, (uint8_t const *)core_data, *out_size - sizeof(core_dump_crc_t));
- if (*crc != cur_crc) {
- ESP_LOGE(TAG, "Core dump data CRC check failed: 0x%x -> 0x%x!", *crc, cur_crc);
- spi_flash_munmap(core_data_handle);
- return ESP_FAIL;
- }
- spi_flash_munmap(core_data_handle);
- *out_addr = core_part->address;
- return ESP_OK;
- }
|