startup.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. /*
  2. * SPDX-FileCopyrightText: 2015-2021 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <stdint.h>
  7. #include <string.h>
  8. #include "esp_attr.h"
  9. #include "esp_err.h"
  10. #include "esp_system.h"
  11. #include "esp_log.h"
  12. #include "esp_ota_ops.h"
  13. #include "sdkconfig.h"
  14. #include "soc/soc_caps.h"
  15. #include "hal/wdt_hal.h"
  16. #include "hal/uart_types.h"
  17. #include "hal/uart_ll.h"
  18. #include "esp_system.h"
  19. #include "esp_log.h"
  20. #include "esp_heap_caps_init.h"
  21. #include "esp_spi_flash.h"
  22. #include "esp_flash_internal.h"
  23. #include "esp_newlib.h"
  24. #include "esp_vfs_dev.h"
  25. #include "esp_timer.h"
  26. #include "esp_efuse.h"
  27. #include "esp_flash_encrypt.h"
  28. #include "esp_secure_boot.h"
  29. #include "esp_sleep.h"
  30. #include "esp_xt_wdt.h"
  31. /***********************************************/
  32. // Headers for other components init functions
  33. #include "nvs_flash.h"
  34. #include "esp_phy_init.h"
  35. #include "esp_coexist_internal.h"
  36. #if CONFIG_ESP_COREDUMP_ENABLE
  37. #include "esp_core_dump.h"
  38. #endif
  39. #include "esp_app_trace.h"
  40. #include "esp_private/dbg_stubs.h"
  41. #include "esp_pm.h"
  42. #include "esp_private/pm_impl.h"
  43. #include "esp_pthread.h"
  44. #include "esp_vfs_console.h"
  45. #include "brownout.h"
  46. #include "esp_rom_sys.h"
  47. // [refactor-todo] make this file completely target-independent
  48. #if CONFIG_IDF_TARGET_ESP32
  49. #include "esp32/clk.h"
  50. #include "esp32/spiram.h"
  51. #elif CONFIG_IDF_TARGET_ESP32S2
  52. #include "esp32s2/clk.h"
  53. #include "esp32s2/spiram.h"
  54. #elif CONFIG_IDF_TARGET_ESP32S3
  55. #include "esp32s3/clk.h"
  56. #include "esp32s3/spiram.h"
  57. #elif CONFIG_IDF_TARGET_ESP32C3
  58. #include "esp32c3/clk.h"
  59. #elif CONFIG_IDF_TARGET_ESP32H2
  60. #include "esp32h2/clk.h"
  61. #endif
  62. /***********************************************/
  63. #include "esp_private/startup_internal.h"
  64. // Ensure that system configuration matches the underlying number of cores.
  65. // This should enable us to avoid checking for both everytime.
  66. #if !(SOC_CPU_CORES_NUM > 1) && !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  67. #error "System has been configured to run on multiple cores, but target SoC only has a single core."
  68. #endif
  69. uint64_t g_startup_time = 0;
  70. #if SOC_APB_BACKUP_DMA
  71. // APB DMA lock initialising API
  72. extern void esp_apb_backup_dma_lock_init(void);
  73. #endif
  74. // App entry point for core 0
  75. extern void esp_startup_start_app(void);
  76. // Entry point for core 0 from hardware init (port layer)
  77. void start_cpu0(void) __attribute__((weak, alias("start_cpu0_default"))) __attribute__((noreturn));
  78. #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  79. // Entry point for core [1..X] from hardware init (port layer)
  80. void start_cpu_other_cores(void) __attribute__((weak, alias("start_cpu_other_cores_default"))) __attribute__((noreturn));
  81. // App entry point for core [1..X]
  82. void esp_startup_start_app_other_cores(void) __attribute__((weak, alias("esp_startup_start_app_other_cores_default"))) __attribute__((noreturn));
  83. static volatile bool s_system_inited[SOC_CPU_CORES_NUM] = { false };
  84. const sys_startup_fn_t g_startup_fn[SOC_CPU_CORES_NUM] = { [0] = start_cpu0,
  85. #if SOC_CPU_CORES_NUM > 1
  86. [1 ... SOC_CPU_CORES_NUM - 1] = start_cpu_other_cores
  87. #endif
  88. };
  89. static volatile bool s_system_full_inited = false;
  90. #else
  91. const sys_startup_fn_t g_startup_fn[1] = { start_cpu0 };
  92. #endif
  93. #ifdef CONFIG_COMPILER_CXX_EXCEPTIONS
  94. // workaround for C++ exception crashes
  95. void _Unwind_SetNoFunctionContextInstall(unsigned char enable) __attribute__((weak, alias("_Unwind_SetNoFunctionContextInstall_Default")));
  96. // workaround for C++ exception large memory allocation
  97. void _Unwind_SetEnableExceptionFdeSorting(unsigned char enable);
  98. static IRAM_ATTR void _Unwind_SetNoFunctionContextInstall_Default(unsigned char enable __attribute__((unused)))
  99. {
  100. (void)0;
  101. }
  102. #endif // CONFIG_COMPILER_CXX_EXCEPTIONS
  103. static const char* TAG = "cpu_start";
  104. /**
  105. * This function overwrites a the same function of libsupc++ (part of libstdc++).
  106. * Consequently, libsupc++ will then follow our configured exception emergency pool size.
  107. *
  108. * It will be called even with -fno-exception for user code since the stdlib still uses exceptions.
  109. */
  110. size_t __cxx_eh_arena_size_get(void)
  111. {
  112. #ifdef CONFIG_COMPILER_CXX_EXCEPTIONS
  113. return CONFIG_COMPILER_CXX_EXCEPTIONS_EMG_POOL_SIZE;
  114. #else
  115. return 0;
  116. #endif
  117. }
  118. /**
  119. * Xtensa gcc is configured to emit a .ctors section, RISC-V gcc is configured with --enable-initfini-array
  120. * so it emits an .init_array section instead.
  121. * But the init_priority sections will be sorted for iteration in ascending order during startup.
  122. * The rest of the init_array sections is sorted for iteration in descending order during startup, however.
  123. * Hence a different section is generated for the init_priority functions which is looped
  124. * over in ascending direction instead of descending direction.
  125. * The RISC-V-specific behavior is dependent on the linker script ld/esp32c3/sections.ld.in.
  126. */
  127. static void do_global_ctors(void)
  128. {
  129. #if __riscv
  130. extern void (*__init_priority_array_start)(void);
  131. extern void (*__init_priority_array_end)(void);
  132. #endif
  133. extern void (*__init_array_start)(void);
  134. extern void (*__init_array_end)(void);
  135. #ifdef CONFIG_COMPILER_CXX_EXCEPTIONS
  136. struct object { long placeholder[ 10 ]; };
  137. void __register_frame_info (const void *begin, struct object *ob);
  138. extern char __eh_frame[];
  139. static struct object ob;
  140. __register_frame_info( __eh_frame, &ob );
  141. #endif // CONFIG_COMPILER_CXX_EXCEPTIONS
  142. void (**p)(void);
  143. #if __riscv
  144. for (p = &__init_priority_array_start; p < &__init_priority_array_end; ++p) {
  145. ESP_EARLY_LOGD(TAG, "calling init function: %p", *p);
  146. (*p)();
  147. }
  148. #endif
  149. for (p = &__init_array_end - 1; p >= &__init_array_start; --p) {
  150. ESP_EARLY_LOGD(TAG, "calling init function: %p", *p);
  151. (*p)();
  152. }
  153. }
  154. static void do_system_init_fn(void)
  155. {
  156. extern esp_system_init_fn_t _esp_system_init_fn_array_start;
  157. extern esp_system_init_fn_t _esp_system_init_fn_array_end;
  158. esp_system_init_fn_t *p;
  159. for (p = &_esp_system_init_fn_array_end - 1; p >= &_esp_system_init_fn_array_start; --p) {
  160. if (p->cores & BIT(cpu_hal_get_core_id())) {
  161. (*(p->fn))();
  162. }
  163. }
  164. #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  165. s_system_inited[cpu_hal_get_core_id()] = true;
  166. #endif
  167. }
  168. #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  169. static void esp_startup_start_app_other_cores_default(void)
  170. {
  171. while (1) {
  172. esp_rom_delay_us(UINT32_MAX);
  173. }
  174. }
  175. static void IRAM_ATTR start_cpu_other_cores_default(void)
  176. {
  177. do_system_init_fn();
  178. while (!s_system_full_inited) {
  179. esp_rom_delay_us(100);
  180. }
  181. esp_startup_start_app_other_cores();
  182. }
  183. #endif
  184. static void do_core_init(void)
  185. {
  186. /* Initialize heap allocator. WARNING: This *needs* to happen *after* the app cpu has booted.
  187. If the heap allocator is initialized first, it will put free memory linked list items into
  188. memory also used by the ROM. Starting the app cpu will let its ROM initialize that memory,
  189. corrupting those linked lists. Initializing the allocator *after* the app cpu has booted
  190. works around this problem.
  191. With SPI RAM enabled, there's a second reason: half of the SPI RAM will be managed by the
  192. app CPU, and when that is not up yet, the memory will be inaccessible and heap_caps_init may
  193. fail initializing it properly. */
  194. heap_caps_init();
  195. // When apptrace module is enabled, there will be SEGGER_SYSVIEW calls in the newlib init.
  196. // SEGGER_SYSVIEW relies on apptrace module
  197. // apptrace module uses esp_timer_get_time to determine timeout conditions.
  198. // esp_timer early initialization is required for esp_timer_get_time to work.
  199. esp_timer_early_init();
  200. esp_newlib_init();
  201. if (g_spiram_ok) {
  202. #if CONFIG_SPIRAM_BOOT_INIT && (CONFIG_SPIRAM_USE_CAPS_ALLOC || CONFIG_SPIRAM_USE_MALLOC)
  203. esp_err_t r=esp_spiram_add_to_heapalloc();
  204. if (r != ESP_OK) {
  205. ESP_EARLY_LOGE(TAG, "External RAM could not be added to heap!");
  206. abort();
  207. }
  208. #if CONFIG_SPIRAM_USE_MALLOC
  209. heap_caps_malloc_extmem_enable(CONFIG_SPIRAM_MALLOC_ALWAYSINTERNAL);
  210. #endif
  211. #endif
  212. }
  213. #if CONFIG_ESP32_BROWNOUT_DET || \
  214. CONFIG_ESP32S2_BROWNOUT_DET || \
  215. CONFIG_ESP32S3_BROWNOUT_DET || \
  216. CONFIG_ESP32C3_BROWNOUT_DET || \
  217. CONFIG_ESP32H2_BROWNOUT_DET
  218. // [refactor-todo] leads to call chain rtc_is_register (driver) -> esp_intr_alloc (esp32/esp32s2) ->
  219. // malloc (newlib) -> heap_caps_malloc (heap), so heap must be at least initialized
  220. esp_brownout_init();
  221. #endif
  222. esp_newlib_time_init();
  223. #if CONFIG_VFS_SUPPORT_IO
  224. // VFS console register.
  225. esp_err_t vfs_err = esp_vfs_console_register();
  226. assert(vfs_err == ESP_OK && "Failed to register vfs console");
  227. #endif
  228. #if defined(CONFIG_VFS_SUPPORT_IO) && !defined(CONFIG_ESP_CONSOLE_NONE)
  229. const static char *default_stdio_dev = "/dev/console/";
  230. esp_reent_init(_GLOBAL_REENT);
  231. _GLOBAL_REENT->_stdin = fopen(default_stdio_dev, "r");
  232. _GLOBAL_REENT->_stdout = fopen(default_stdio_dev, "w");
  233. _GLOBAL_REENT->_stderr = fopen(default_stdio_dev, "w");
  234. #else // defined(CONFIG_VFS_SUPPORT_IO) && !defined(CONFIG_ESP_CONSOLE_NONE)
  235. _REENT_SMALL_CHECK_INIT(_GLOBAL_REENT);
  236. #endif // defined(CONFIG_VFS_SUPPORT_IO) && !defined(CONFIG_ESP_CONSOLE_NONE)
  237. esp_err_t err __attribute__((unused));
  238. err = esp_pthread_init();
  239. assert(err == ESP_OK && "Failed to init pthread module!");
  240. spi_flash_init();
  241. /* init default OS-aware flash access critical section */
  242. spi_flash_guard_set(&g_flash_guard_default_ops);
  243. esp_flash_app_init();
  244. esp_err_t flash_ret = esp_flash_init_default_chip();
  245. assert(flash_ret == ESP_OK);
  246. (void)flash_ret;
  247. #ifdef CONFIG_EFUSE_VIRTUAL
  248. ESP_LOGW(TAG, "eFuse virtual mode is enabled. If Secure boot or Flash encryption is enabled then it does not provide any security. FOR TESTING ONLY!");
  249. #ifdef CONFIG_EFUSE_VIRTUAL_KEEP_IN_FLASH
  250. const esp_partition_t *efuse_partition = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_EFUSE_EM, NULL);
  251. if (efuse_partition) {
  252. esp_efuse_init_virtual_mode_in_flash(efuse_partition->address, efuse_partition->size);
  253. }
  254. #endif
  255. #endif
  256. #if CONFIG_SECURE_DISABLE_ROM_DL_MODE
  257. err = esp_efuse_disable_rom_download_mode();
  258. assert(err == ESP_OK && "Failed to disable ROM download mode");
  259. #endif
  260. #if CONFIG_SECURE_ENABLE_SECURE_ROM_DL_MODE
  261. err = esp_efuse_enable_rom_secure_download_mode();
  262. assert(err == ESP_OK && "Failed to enable Secure Download mode");
  263. #endif
  264. #if CONFIG_ESP32_DISABLE_BASIC_ROM_CONSOLE
  265. esp_efuse_disable_basic_rom_console();
  266. #endif
  267. #ifdef CONFIG_SECURE_FLASH_ENC_ENABLED
  268. esp_flash_encryption_init_checks();
  269. #endif
  270. #if defined(CONFIG_SECURE_BOOT) || defined(CONFIG_SECURE_SIGNED_ON_UPDATE_NO_SECURE_BOOT)
  271. // Note: in some configs this may read flash, so placed after flash init
  272. esp_secure_boot_init_checks();
  273. #endif
  274. #if CONFIG_ESP_XT_WDT
  275. esp_xt_wdt_config_t cfg = {
  276. .timeout = CONFIG_ESP_XT_WDT_TIMEOUT,
  277. .auto_backup_clk_enable = CONFIG_ESP_XT_WDT_BACKUP_CLK_ENABLE,
  278. };
  279. err = esp_xt_wdt_init(&cfg);
  280. assert(err == ESP_OK && "Failed to init xtwdt");
  281. #endif
  282. }
  283. static void do_secondary_init(void)
  284. {
  285. #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  286. // The port layer transferred control to this function with other cores 'paused',
  287. // resume execution so that cores might execute component initialization functions.
  288. startup_resume_other_cores();
  289. #endif
  290. // Execute initialization functions esp_system_init_fn_t assigned to the main core. While
  291. // this is happening, all other cores are executing the initialization functions
  292. // assigned to them since they have been resumed already.
  293. do_system_init_fn();
  294. #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  295. // Wait for all cores to finish secondary init.
  296. volatile bool system_inited = false;
  297. while (!system_inited) {
  298. system_inited = true;
  299. for (int i = 0; i < SOC_CPU_CORES_NUM; i++) {
  300. system_inited &= s_system_inited[i];
  301. }
  302. esp_rom_delay_us(100);
  303. }
  304. #endif
  305. }
  306. static void start_cpu0_default(void)
  307. {
  308. ESP_EARLY_LOGI(TAG, "Pro cpu start user code");
  309. int cpu_freq = esp_clk_cpu_freq();
  310. ESP_EARLY_LOGI(TAG, "cpu freq: %d", cpu_freq);
  311. // Display information about the current running image.
  312. if (LOG_LOCAL_LEVEL >= ESP_LOG_INFO) {
  313. const esp_app_desc_t *app_desc = esp_ota_get_app_description();
  314. ESP_EARLY_LOGI(TAG, "Application information:");
  315. #ifndef CONFIG_APP_EXCLUDE_PROJECT_NAME_VAR
  316. ESP_EARLY_LOGI(TAG, "Project name: %s", app_desc->project_name);
  317. #endif
  318. #ifndef CONFIG_APP_EXCLUDE_PROJECT_VER_VAR
  319. ESP_EARLY_LOGI(TAG, "App version: %s", app_desc->version);
  320. #endif
  321. #ifdef CONFIG_BOOTLOADER_APP_SECURE_VERSION
  322. ESP_EARLY_LOGI(TAG, "Secure version: %d", app_desc->secure_version);
  323. #endif
  324. #ifdef CONFIG_APP_COMPILE_TIME_DATE
  325. ESP_EARLY_LOGI(TAG, "Compile time: %s %s", app_desc->date, app_desc->time);
  326. #endif
  327. char buf[17];
  328. esp_ota_get_app_elf_sha256(buf, sizeof(buf));
  329. ESP_EARLY_LOGI(TAG, "ELF file SHA256: %s...", buf);
  330. ESP_EARLY_LOGI(TAG, "ESP-IDF: %s", app_desc->idf_ver);
  331. }
  332. // Initialize core components and services.
  333. do_core_init();
  334. // Execute constructors.
  335. do_global_ctors();
  336. // Execute init functions of other components; blocks
  337. // until all cores finish (when !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE).
  338. do_secondary_init();
  339. // Now that the application is about to start, disable boot watchdog
  340. #ifndef CONFIG_BOOTLOADER_WDT_DISABLE_IN_USER_CODE
  341. wdt_hal_context_t rtc_wdt_ctx = {.inst = WDT_RWDT, .rwdt_dev = &RTCCNTL};
  342. wdt_hal_write_protect_disable(&rtc_wdt_ctx);
  343. wdt_hal_disable(&rtc_wdt_ctx);
  344. wdt_hal_write_protect_enable(&rtc_wdt_ctx);
  345. #endif
  346. #if SOC_CPU_CORES_NUM > 1 && !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE
  347. s_system_full_inited = true;
  348. #endif
  349. esp_startup_start_app();
  350. while (1);
  351. }
  352. IRAM_ATTR ESP_SYSTEM_INIT_FN(init_components0, BIT(0))
  353. {
  354. esp_timer_init();
  355. #if CONFIG_ESP_SLEEP_GPIO_RESET_WORKAROUND && !CONFIG_PM_SLP_DISABLE_GPIO
  356. // Configure to isolate (disable the Input/Output/Pullup/Pulldown
  357. // function of the pin) all GPIO pins in sleep state
  358. esp_sleep_config_gpio_isolate();
  359. // Enable automatic switching of GPIO configuration
  360. esp_sleep_enable_gpio_switch(true);
  361. #endif
  362. #if CONFIG_APPTRACE_ENABLE
  363. esp_err_t err = esp_apptrace_init();
  364. assert(err == ESP_OK && "Failed to init apptrace module on PRO CPU!");
  365. #endif
  366. #if CONFIG_APPTRACE_SV_ENABLE
  367. SEGGER_SYSVIEW_Conf();
  368. #endif
  369. #if CONFIG_ESP_DEBUG_STUBS_ENABLE
  370. esp_dbg_stubs_init();
  371. #endif
  372. #if defined(CONFIG_PM_ENABLE)
  373. esp_pm_impl_init();
  374. #endif
  375. #if CONFIG_ESP_COREDUMP_ENABLE
  376. esp_core_dump_init();
  377. #endif
  378. #if SOC_APB_BACKUP_DMA
  379. esp_apb_backup_dma_lock_init();
  380. #endif
  381. #if CONFIG_SW_COEXIST_ENABLE || CONFIG_EXTERNAL_COEX_ENABLE
  382. esp_coex_adapter_register(&g_coex_adapter_funcs);
  383. coex_pre_init();
  384. #endif
  385. #ifdef CONFIG_COMPILER_CXX_EXCEPTIONS
  386. ESP_EARLY_LOGD(TAG, "Setting C++ exception workarounds.");
  387. _Unwind_SetNoFunctionContextInstall(1);
  388. _Unwind_SetEnableExceptionFdeSorting(0);
  389. #endif // CONFIG_COMPILER_CXX_EXCEPTIONS
  390. }