mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-10-26 19:45:00 +00:00 
			
		
		
		
	gcov: add gcov callback into the ipc task
This commit is contained in:
		| @@ -30,7 +30,8 @@ endif() | |||||||
|  |  | ||||||
| idf_component_register(SRCS "${srcs}" | idf_component_register(SRCS "${srcs}" | ||||||
|                        INCLUDE_DIRS "${include_dirs}" |                        INCLUDE_DIRS "${include_dirs}" | ||||||
|                        PRIV_REQUIRES soc |                        PRIV_INCLUDE_DIRS "${priv_include_dirs}" | ||||||
|  |                        PRIV_REQUIRES soc esp_ipc | ||||||
|                        LDFRAGMENTS linker.lf) |                        LDFRAGMENTS linker.lf) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,6 +24,7 @@ | |||||||
| #include "esp_app_trace.h" | #include "esp_app_trace.h" | ||||||
| #include "esp_freertos_hooks.h" | #include "esp_freertos_hooks.h" | ||||||
| #include "esp_private/dbg_stubs.h" | #include "esp_private/dbg_stubs.h" | ||||||
|  | #include "esp_ipc.h" | ||||||
| #include "hal/wdt_hal.h" | #include "hal/wdt_hal.h" | ||||||
| #if CONFIG_IDF_TARGET_ESP32 | #if CONFIG_IDF_TARGET_ESP32 | ||||||
| #include "esp32/rom/libc_stubs.h" | #include "esp32/rom/libc_stubs.h" | ||||||
| @@ -49,6 +50,8 @@ void gcov_dump_task(void *pvParameter) | |||||||
|     int dump_result = 0; |     int dump_result = 0; | ||||||
|     bool *running = (bool *)pvParameter; |     bool *running = (bool *)pvParameter; | ||||||
|  |  | ||||||
|  |     ESP_EARLY_LOGV(TAG, "%s stack use in %d", __FUNCTION__, uxTaskGetStackHighWaterMark(NULL)); | ||||||
|  |  | ||||||
|     ESP_EARLY_LOGV(TAG, "Alloc apptrace down buf %d bytes", ESP_GCOV_DOWN_BUF_SIZE); |     ESP_EARLY_LOGV(TAG, "Alloc apptrace down buf %d bytes", ESP_GCOV_DOWN_BUF_SIZE); | ||||||
|     void *down_buf = malloc(ESP_GCOV_DOWN_BUF_SIZE); |     void *down_buf = malloc(ESP_GCOV_DOWN_BUF_SIZE); | ||||||
|     if (down_buf == NULL) { |     if (down_buf == NULL) { | ||||||
| @@ -81,15 +84,26 @@ gcov_exit: | |||||||
|     if (running) { |     if (running) { | ||||||
|         *running = false; |         *running = false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     ESP_EARLY_LOGV(TAG, "%s stack use out %d", __FUNCTION__, uxTaskGetStackHighWaterMark(NULL)); | ||||||
|  |  | ||||||
|     vTaskDelete(NULL); |     vTaskDelete(NULL); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void gcov_create_task_hook(void) | void gcov_create_task(void *arg) | ||||||
| { | { | ||||||
|  |     ESP_EARLY_LOGV(TAG, "%s", __FUNCTION__); | ||||||
|  |     xTaskCreatePinnedToCore(&gcov_dump_task, "gcov_dump_task", 2048, (void *)&s_gcov_task_running, configMAX_PRIORITIES - 1, NULL, 0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void gcov_create_task_tick_hook(void) | ||||||
|  | { | ||||||
|  |     extern esp_err_t esp_ipc_start_gcov_from_isr(uint32_t cpu_id, esp_ipc_func_t func, void* arg); | ||||||
|     if (s_create_gcov_task) { |     if (s_create_gcov_task) { | ||||||
|         xTaskCreatePinnedToCore(&gcov_dump_task, "gcov_dump_task", 2048, &s_gcov_task_running, configMAX_PRIORITIES - 1, NULL, 0); |         if (esp_ipc_start_gcov_from_isr(xPortGetCoreID(), &gcov_create_task, NULL) == ESP_OK) { | ||||||
|             s_create_gcov_task = false; |             s_create_gcov_task = false; | ||||||
|         } |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -114,19 +128,21 @@ int gcov_rtio_atexit(void (*function)(void) __attribute__ ((unused))) | |||||||
|     if (esp_dbg_stub_entry_get(ESP_DBG_STUB_CAPABILITIES, &capabilities) == ESP_OK) { |     if (esp_dbg_stub_entry_get(ESP_DBG_STUB_CAPABILITIES, &capabilities) == ESP_OK) { | ||||||
|         esp_dbg_stub_entry_set(ESP_DBG_STUB_CAPABILITIES, capabilities | ESP_DBG_STUB_CAP_GCOV_TASK); |         esp_dbg_stub_entry_set(ESP_DBG_STUB_CAPABILITIES, capabilities | ESP_DBG_STUB_CAP_GCOV_TASK); | ||||||
|     } |     } | ||||||
|     esp_register_freertos_tick_hook(gcov_create_task_hook); |     esp_register_freertos_tick_hook(gcov_create_task_tick_hook); | ||||||
|     return ESP_OK; |     return ESP_OK; | ||||||
| } | } | ||||||
|  |  | ||||||
| void esp_gcov_dump(void) | void esp_gcov_dump(void) | ||||||
| { | { | ||||||
|  |     ESP_EARLY_LOGV(TAG, "%s", __FUNCTION__); | ||||||
|  |  | ||||||
|     while (!esp_apptrace_host_is_connected(ESP_APPTRACE_DEST_TRAX)) { |     while (!esp_apptrace_host_is_connected(ESP_APPTRACE_DEST_TRAX)) { | ||||||
|         vTaskDelay(pdMS_TO_TICKS(10)); |         vTaskDelay(pdMS_TO_TICKS(10)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /* We are not in isr context here. Waiting for the completion is safe */ |     /* We are not in isr context here. Waiting for the completion is safe */ | ||||||
|     s_create_gcov_task = true; |  | ||||||
|     s_gcov_task_running = true; |     s_gcov_task_running = true; | ||||||
|  |     s_create_gcov_task = true; | ||||||
|     while (s_gcov_task_running) { |     while (s_gcov_task_running) { | ||||||
|         vTaskDelay(pdMS_TO_TICKS(10)); |         vTaskDelay(pdMS_TO_TICKS(10)); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -82,6 +82,7 @@ void esp_dbg_stubs_init(void) | |||||||
|     ESP_LOGV(TAG, "%s stubs %x", __func__, eri_read(ESP_DBG_STUBS_TRAX_REG)); |     ESP_LOGV(TAG, "%s stubs %x", __func__, eri_read(ESP_DBG_STUBS_TRAX_REG)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // TODO: add lock mechanism. Not now but in the future ESP_DBG_STUB_CAPABILITIES can be set from different places. | ||||||
| esp_err_t esp_dbg_stub_entry_set(esp_dbg_stub_id_t id, uint32_t entry) | esp_err_t esp_dbg_stub_entry_set(esp_dbg_stub_id_t id, uint32_t entry) | ||||||
| { | { | ||||||
|     if (id < ESP_DBG_STUB_ENTRY_FIRST || id >= ESP_DBG_STUB_ENTRY_MAX) { |     if (id < ESP_DBG_STUB_ENTRY_FIRST || id >= ESP_DBG_STUB_ENTRY_MAX) { | ||||||
|   | |||||||
| @@ -40,6 +40,11 @@ static volatile esp_ipc_wait_t s_ipc_wait[portNUM_PROCESSORS];// This variable t | |||||||
|                                                              //   s_ipc_ack semaphore: before s_func is called, or |                                                              //   s_ipc_ack semaphore: before s_func is called, or | ||||||
|                                                              //   after it returns |                                                              //   after it returns | ||||||
|  |  | ||||||
|  | #if CONFIG_APPTRACE_GCOV_ENABLE | ||||||
|  | static volatile esp_ipc_func_t s_gcov_func = NULL;           // Gcov dump starter function which should be called by high priority task | ||||||
|  | static void * volatile s_gcov_func_arg;                      // Argument to pass into s_gcov_func | ||||||
|  | #endif | ||||||
|  |  | ||||||
| static void IRAM_ATTR ipc_task(void* arg) | static void IRAM_ATTR ipc_task(void* arg) | ||||||
| { | { | ||||||
|     const int cpuid = (int) arg; |     const int cpuid = (int) arg; | ||||||
| @@ -53,6 +58,13 @@ static void IRAM_ATTR ipc_task(void* arg) | |||||||
|             abort(); |             abort(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | #if CONFIG_APPTRACE_GCOV_ENABLE | ||||||
|  |         if (s_gcov_func) { | ||||||
|  |             (*s_gcov_func)(s_gcov_func_arg); | ||||||
|  |             s_gcov_func = NULL; | ||||||
|  |         } | ||||||
|  | #endif | ||||||
|  |         if (s_func[cpuid]) { | ||||||
|             esp_ipc_func_t func = s_func[cpuid]; |             esp_ipc_func_t func = s_func[cpuid]; | ||||||
|             void* arg = s_func_arg[cpuid]; |             void* arg = s_func_arg[cpuid]; | ||||||
|  |  | ||||||
| @@ -63,6 +75,9 @@ static void IRAM_ATTR ipc_task(void* arg) | |||||||
|             if (s_ipc_wait[cpuid] == IPC_WAIT_FOR_END) { |             if (s_ipc_wait[cpuid] == IPC_WAIT_FOR_END) { | ||||||
|                 xSemaphoreGive(s_ipc_ack[cpuid]); |                 xSemaphoreGive(s_ipc_ack[cpuid]); | ||||||
|             } |             } | ||||||
|  |             s_func[cpuid] = NULL; | ||||||
|  |         } | ||||||
|  |  | ||||||
|     } |     } | ||||||
|     // TODO: currently this is unreachable code. Introduce esp_ipc_uninit |     // TODO: currently this is unreachable code. Introduce esp_ipc_uninit | ||||||
|     // function which will signal to both tasks that they can shut down. |     // function which will signal to both tasks that they can shut down. | ||||||
| @@ -87,6 +102,7 @@ static void esp_ipc_init(void) __attribute__((constructor)); | |||||||
| static void esp_ipc_init(void) | static void esp_ipc_init(void) | ||||||
| { | { | ||||||
|     char task_name[15]; |     char task_name[15]; | ||||||
|  |  | ||||||
|     for (int i = 0; i < portNUM_PROCESSORS; ++i) { |     for (int i = 0; i < portNUM_PROCESSORS; ++i) { | ||||||
|         snprintf(task_name, sizeof(task_name), "ipc%d", i); |         snprintf(task_name, sizeof(task_name), "ipc%d", i); | ||||||
|         s_ipc_mutex[i] = xSemaphoreCreateMutex(); |         s_ipc_mutex[i] = xSemaphoreCreateMutex(); | ||||||
| @@ -143,3 +159,18 @@ esp_err_t esp_ipc_call_blocking(uint32_t cpu_id, esp_ipc_func_t func, void* arg) | |||||||
| { | { | ||||||
|     return esp_ipc_call_and_wait(cpu_id, func, arg, IPC_WAIT_FOR_END); |     return esp_ipc_call_and_wait(cpu_id, func, arg, IPC_WAIT_FOR_END); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // currently this is only called from gcov component | ||||||
|  | #if CONFIG_APPTRACE_GCOV_ENABLE | ||||||
|  | esp_err_t esp_ipc_start_gcov_from_isr(uint32_t cpu_id, esp_ipc_func_t func, void* arg) | ||||||
|  | { | ||||||
|  |     if (xTaskGetSchedulerState() != taskSCHEDULER_RUNNING) { | ||||||
|  |         return ESP_ERR_INVALID_STATE; | ||||||
|  |     } | ||||||
|  |     s_gcov_func = func; | ||||||
|  |     s_gcov_func_arg = arg; | ||||||
|  |     xSemaphoreGiveFromISR(s_ipc_sem[cpu_id], NULL); | ||||||
|  |  | ||||||
|  |     return ESP_OK; | ||||||
|  | } | ||||||
|  | #endif  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Erhan Kurubas
					Erhan Kurubas