| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-06-07 22:54:09 +03:00
										 |  |  |  * SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: Apache-2.0 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include "FreeRTOS.h"
 | 
					
						
							|  |  |  | #include "task.h"
 | 
					
						
							|  |  |  | #include "portmacro.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-05 15:38:49 +08:00
										 |  |  | #include "esp_private/esp_int_wdt.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #include "esp_system.h"
 | 
					
						
							|  |  |  | #include "esp_heap_caps_init.h"
 | 
					
						
							|  |  |  | #include "esp_task_wdt.h"
 | 
					
						
							|  |  |  | #include "esp_task.h"
 | 
					
						
							|  |  |  | #include "esp_private/crosscore_int.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  | #include "esp_private/startup_internal.h"    /* Required by g_spiram_ok. [refactor-todo] for g_spiram_ok */
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #include "esp_log.h"
 | 
					
						
							| 
									
										
										
										
											2022-03-31 15:07:51 +08:00
										 |  |  | #include "esp_memory_utils.h"
 | 
					
						
							| 
									
										
										
										
											2022-04-14 18:06:21 +08:00
										 |  |  | #include "esp_freertos_hooks.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #include "sdkconfig.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if CONFIG_IDF_TARGET_ESP32
 | 
					
						
							|  |  |  | #include "esp32/spiram.h"
 | 
					
						
							|  |  |  | #elif CONFIG_IDF_TARGET_ESP32S2
 | 
					
						
							|  |  |  | #include "esp32s2/spiram.h"
 | 
					
						
							|  |  |  | #elif CONFIG_IDF_TARGET_ESP32S3
 | 
					
						
							|  |  |  | #include "esp32s3/spiram.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if CONFIG_SPIRAM_MALLOC_RESERVE_INTERNAL
 | 
					
						
							|  |  |  | static const char* TAG = "cpu_start"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Architecture-agnostic parts of the FreeRTOS ESP-IDF port layer can go here.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The actual call flow will be to call esp_startup_start_app() in <ARCH>/port.c, | 
					
						
							|  |  |  |  * which will then call esp_startup_start_app_common() | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Duplicate of inaccessible xSchedulerRunning; needed at startup to avoid counting nesting
 | 
					
						
							|  |  |  | volatile unsigned port_xSchedulerRunning[portNUM_PROCESSORS] = {0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // For now, running FreeRTOS on one core and a bare metal on the other (or other OSes)
 | 
					
						
							|  |  |  | // is not supported. For now CONFIG_FREERTOS_UNICORE and CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
 | 
					
						
							|  |  |  | // should mirror each other's values.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // And since this should be true, we can just check for CONFIG_FREERTOS_UNICORE.
 | 
					
						
							|  |  |  | #if CONFIG_FREERTOS_UNICORE != CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     #error "FreeRTOS and system configuration mismatch regarding the use of multiple cores."
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void main_task(void* args); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 12:03:43 +03:00
										 |  |  | #ifdef CONFIG_ESP_SYSTEM_GDBSTUB_RUNTIME
 | 
					
						
							|  |  |  | void esp_gdbstub_init(void); | 
					
						
							|  |  |  | #endif // CONFIG_ESP_SYSTEM_GDBSTUB_RUNTIME
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | extern void app_main(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void esp_startup_start_app_common(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if CONFIG_ESP_INT_WDT
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     esp_int_wdt_init(); | 
					
						
							|  |  |  |     //Initialize the interrupt watch dog for CPU0.
 | 
					
						
							|  |  |  |     esp_int_wdt_cpu_init(); | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     esp_crosscore_int_init(); | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-02 15:49:31 +08:00
										 |  |  | #if CONFIG_ESP_SYSTEM_GDBSTUB_RUNTIME && !CONFIG_IDF_TARGET_ESP32C2
 | 
					
						
							| 
									
										
										
										
											2021-04-19 12:03:43 +03:00
										 |  |  |     esp_gdbstub_init(); | 
					
						
							|  |  |  | #endif // CONFIG_ESP_SYSTEM_GDBSTUB_RUNTIME
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     portBASE_TYPE res = xTaskCreatePinnedToCore(&main_task, "main", | 
					
						
							|  |  |  |                                                 ESP_TASK_MAIN_STACK, NULL, | 
					
						
							|  |  |  |                                                 ESP_TASK_MAIN_PRIO, NULL, ESP_TASK_MAIN_CORE); | 
					
						
							|  |  |  |     assert(res == pdTRUE); | 
					
						
							|  |  |  |     (void)res; | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void main_task(void* args) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if !CONFIG_FREERTOS_UNICORE
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     // Wait for FreeRTOS initialization to finish on APP CPU, before replacing its startup stack
 | 
					
						
							|  |  |  |     while (port_xSchedulerRunning[1] == 0) { | 
					
						
							|  |  |  |         ; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     // [refactor-todo] check if there is a way to move the following block to esp_system startup
 | 
					
						
							|  |  |  |     heap_caps_enable_nonos_stack_heaps(); | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     // Now we have startup stack RAM available for heap, enable any DMA pool memory
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #if CONFIG_SPIRAM_MALLOC_RESERVE_INTERNAL
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     if (g_spiram_ok) { | 
					
						
							|  |  |  |         esp_err_t r = esp_spiram_reserve_dma_pool(CONFIG_SPIRAM_MALLOC_RESERVE_INTERNAL); | 
					
						
							|  |  |  |         if (r != ESP_OK) { | 
					
						
							|  |  |  |             ESP_EARLY_LOGE(TAG, "Could not reserve internal/DMA pool (error 0x%x)", r); | 
					
						
							|  |  |  |             abort(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-14 18:06:21 +08:00
										 |  |  |     //Initialize TWDT if configured to do so
 | 
					
						
							|  |  |  | #if CONFIG_ESP_TASK_WDT
 | 
					
						
							|  |  |  |     esp_task_wdt_config_t twdt_config = { | 
					
						
							|  |  |  |         .timeout_ms = CONFIG_ESP_TASK_WDT_TIMEOUT_S * 1000, | 
					
						
							|  |  |  |         .idle_core_mask = 0, | 
					
						
							|  |  |  | #if CONFIG_ESP_TASK_WDT_PANIC
 | 
					
						
							|  |  |  |         .trigger_panic = true, | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-04-14 18:06:21 +08:00
										 |  |  |     }; | 
					
						
							|  |  |  | #if CONFIG_ESP_TASK_WDT_CHECK_IDLE_TASK_CPU0
 | 
					
						
							|  |  |  |     twdt_config.idle_core_mask |= (1 << 0); | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-04-14 18:06:21 +08:00
										 |  |  | #if CONFIG_ESP_TASK_WDT_CHECK_IDLE_TASK_CPU1
 | 
					
						
							|  |  |  |     twdt_config.idle_core_mask |= (1 << 1); | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2022-04-14 18:06:21 +08:00
										 |  |  |     ESP_ERROR_CHECK(esp_task_wdt_init(&twdt_config)); | 
					
						
							|  |  |  | #endif // CONFIG_ESP_TASK_WDT
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 16:56:45 +08:00
										 |  |  |     app_main(); | 
					
						
							|  |  |  |     vTaskDelete(NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // -------------------- Heap Related -----------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool xPortCheckValidTCBMem(const void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return esp_ptr_internal(ptr) && esp_ptr_byte_accessible(ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool xPortcheckValidStackMem(const void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef CONFIG_SPIRAM_ALLOW_STACK_EXTERNAL_MEMORY
 | 
					
						
							|  |  |  |     return esp_ptr_byte_accessible(ptr); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return esp_ptr_internal(ptr) && esp_ptr_byte_accessible(ptr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-11-06 15:03:21 +11:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-10-16 01:08:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // ------------- FreeRTOS Static Allocation ----------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | This function is required by FreeRTOS when configSUPPORT_STATIC_ALLOCATION is | 
					
						
							|  |  |  | enabled and is used by FreeRTOS to obtain memory for its IDLE tasks. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Like the pvPortMallocTcbMem() and pvPortMallocStackMem() macros, TCB and stack | 
					
						
							|  |  |  | memory MUST be placed in internal RAM. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | void vApplicationGetIdleTaskMemory(StaticTask_t **ppxIdleTaskTCBBuffer, | 
					
						
							|  |  |  |                                    StackType_t **ppxIdleTaskStackBuffer, | 
					
						
							|  |  |  |                                    uint32_t *pulIdleTaskStackSize ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StaticTask_t *pxTCBBufferTemp; | 
					
						
							|  |  |  |     StackType_t *pxStackBufferTemp; | 
					
						
							| 
									
										
										
										
											2022-05-09 15:45:38 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     /* If the stack grows down then allocate the stack then the TCB so the stack
 | 
					
						
							|  |  |  |      * does not grow into the TCB.  Likewise if the stack grows up then allocate | 
					
						
							|  |  |  |      * the TCB then the stack. */ | 
					
						
							|  |  |  |     #if (portSTACK_GROWTH > 0)
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         //Allocate TCB and stack buffer in internal memory
 | 
					
						
							|  |  |  |         pxTCBBufferTemp = pvPortMallocTcbMem(sizeof(StaticTask_t)); | 
					
						
							|  |  |  |         pxStackBufferTemp = pvPortMallocStackMem(configIDLE_TASK_STACK_SIZE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     #else /* portSTACK_GROWTH */
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         //Allocate TCB and stack buffer in internal memory
 | 
					
						
							|  |  |  |         pxStackBufferTemp = pvPortMallocStackMem(configIDLE_TASK_STACK_SIZE); | 
					
						
							|  |  |  |         pxTCBBufferTemp = pvPortMallocTcbMem(sizeof(StaticTask_t)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     #endif /* portSTACK_GROWTH */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-16 01:08:46 +08:00
										 |  |  |     assert(pxTCBBufferTemp != NULL); | 
					
						
							|  |  |  |     assert(pxStackBufferTemp != NULL); | 
					
						
							|  |  |  |     //Write back pointers
 | 
					
						
							|  |  |  |     *ppxIdleTaskTCBBuffer = pxTCBBufferTemp; | 
					
						
							|  |  |  |     *ppxIdleTaskStackBuffer = pxStackBufferTemp; | 
					
						
							|  |  |  |     *pulIdleTaskStackSize = configIDLE_TASK_STACK_SIZE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | This function is required by FreeRTOS when configSUPPORT_STATIC_ALLOCATION is | 
					
						
							|  |  |  | enabled and is used by the FreeRTOS Timer to obtain memory for its daemone task. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Like the pvPortMallocTcbMem() and pvPortMallocStackMem() macros, TCB and stack | 
					
						
							|  |  |  | memory MUST be placed in internal RAM. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | void vApplicationGetTimerTaskMemory(StaticTask_t **ppxTimerTaskTCBBuffer, | 
					
						
							|  |  |  |                                     StackType_t **ppxTimerTaskStackBuffer, | 
					
						
							|  |  |  |                                     uint32_t *pulTimerTaskStackSize ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StaticTask_t *pxTCBBufferTemp; | 
					
						
							|  |  |  |     StackType_t *pxStackBufferTemp; | 
					
						
							| 
									
										
										
										
											2022-05-09 15:45:38 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     /* If the stack grows down then allocate the stack then the TCB so the stack
 | 
					
						
							|  |  |  |      * does not grow into the TCB.  Likewise if the stack grows up then allocate | 
					
						
							|  |  |  |      * the TCB then the stack. */ | 
					
						
							|  |  |  |     #if (portSTACK_GROWTH > 0)
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         //Allocate TCB and stack buffer in internal memory
 | 
					
						
							|  |  |  |         pxTCBBufferTemp = pvPortMallocTcbMem(sizeof(StaticTask_t)); | 
					
						
							|  |  |  |         pxStackBufferTemp = pvPortMallocStackMem(configTIMER_TASK_STACK_DEPTH); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     #else /* portSTACK_GROWTH */
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         //Allocate TCB and stack buffer in internal memory
 | 
					
						
							|  |  |  |         pxStackBufferTemp = pvPortMallocStackMem(configTIMER_TASK_STACK_DEPTH); | 
					
						
							|  |  |  |         pxTCBBufferTemp = pvPortMallocTcbMem(sizeof(StaticTask_t)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     #endif /* portSTACK_GROWTH */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-16 01:08:46 +08:00
										 |  |  |     assert(pxTCBBufferTemp != NULL); | 
					
						
							|  |  |  |     assert(pxStackBufferTemp != NULL); | 
					
						
							|  |  |  |     //Write back pointers
 | 
					
						
							|  |  |  |     *ppxTimerTaskTCBBuffer = pxTCBBufferTemp; | 
					
						
							|  |  |  |     *ppxTimerTaskStackBuffer = pxStackBufferTemp; | 
					
						
							|  |  |  |     *pulTimerTaskStackSize = configTIMER_TASK_STACK_DEPTH; | 
					
						
							|  |  |  | } |