1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 11:12:44 +01:00
RIOT/cpu/esp32/startup.c

430 lines
13 KiB
C
Raw Normal View History

2018-10-08 12:20:49 +02:00
/*
* Copyright (C) 2018 Gunar Schorcht
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup cpu_esp32
* @{
*
* @file
* @brief Implementation of the CPU initialization
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @author Jens Alfke <jens@mooseyard.com>
2018-10-08 12:20:49 +02:00
* @}
*/
#include "esp_common.h"
#include <stdlib.h>
#include <string.h>
#include <sys/reent.h>
2022-02-01 22:11:43 +01:00
/* RIOT headers have to be included before ESP-IDF headers! */
2018-10-08 12:20:49 +02:00
#include "board.h"
2022-02-01 22:11:43 +01:00
#include "esp/common_macros.h"
2018-10-08 12:20:49 +02:00
#include "exceptions.h"
#include "irq_arch.h"
#include "kernel_defines.h"
#include "kernel_init.h"
#include "log.h"
2022-02-01 22:11:43 +01:00
#include "periph_cpu.h"
#include "stdio_base.h"
2018-10-08 12:20:49 +02:00
#include "syscalls.h"
#include "thread_arch.h"
2022-02-01 22:11:43 +01:00
#include "tools.h"
2018-10-08 12:20:49 +02:00
#include "periph/cpuid.h"
#include "periph/init.h"
#include "periph/rtc.h"
2022-02-01 22:11:43 +01:00
/* ESP-IDF headers */
2018-10-08 12:20:49 +02:00
#include "driver/periph_ctrl.h"
2022-02-01 22:11:43 +01:00
#include "esp_attr.h"
#include "esp_heap_caps_init.h"
#include "esp_log.h"
#include "esp_sleep.h"
#include "esp_timer.h"
2018-10-08 12:20:49 +02:00
#include "rom/cache.h"
#include "rom/ets_sys.h"
#include "rom/rtc.h"
#include "rom/uart.h"
#include "soc/apb_ctrl_reg.h"
#include "soc/cpu.h"
#include "soc/dport_reg.h"
#include "soc/dport_access.h"
#include "soc/rtc.h"
#include "soc/rtc_cntl_reg.h"
#include "soc/rtc_cntl_struct.h"
#include "soc/timer_group_struct.h"
#include "xtensa/core-macros.h"
#include "xtensa/xtensa_api.h"
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_ESP_SPI_RAM)
#include "spiram.h"
#endif
2018-10-08 12:20:49 +02:00
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_PUF_SRAM)
2022-02-17 11:08:11 +01:00
#include "puf_sram.h"
#endif
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_STDIO_UART)
2018-10-08 12:20:49 +02:00
#include "stdio_uart.h"
#endif
#define ENABLE_DEBUG 0
#include "debug.h"
2018-10-08 12:20:49 +02:00
#define STRINGIFY(s) STRINGIFY2(s)
#define STRINGIFY2(s) #s
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_ESP_LOG_STARTUP)
#define LOG_STARTUP(format, ...) LOG_TAG_EARLY(LOG_INFO, D, __func__, format, ##__VA_ARGS__)
#else
#define LOG_STARTUP(format, ...)
#endif
2018-10-08 12:20:49 +02:00
/* following variables are defined in linker script */
extern uint8_t _bss_start;
extern uint8_t _bss_end;
extern uint8_t _sheap;
extern uint8_t _eheap;
extern uint8_t _rtc_bss_start;
extern uint8_t _rtc_bss_end;
extern uint8_t _rtc_bss_rtc_start;
extern uint8_t _rtc_bss_rtc_end;
2022-02-01 22:11:43 +01:00
extern uint8_t _iram_start;
2018-10-08 12:20:49 +02:00
/* external esp function declarations */
extern void esp_clk_init(void);
extern void esp_perip_clk_init(void);
extern void esp_reent_init(struct _reent* r);
extern uint32_t hwrand (void);
extern void bootloader_clock_configure(void);
2018-10-08 12:20:49 +02:00
2022-02-01 22:11:43 +01:00
/* external RTC function declarations since they are not declared in headers */
/* components/esp_hw_support/include/esp_private/esp_clk.h */
extern int esp_clk_apb_freq(void);
extern int esp_clk_cpu_freq(void);
extern int esp_clk_xtal_freq(void);
extern uint32_t esp_clk_slowclk_cal_get(void);
/* components/esp_hw_support/include/soc/esp32/rtc.h */
extern uint64_t esp_rtc_get_time_us(void);
/* components/esp_system/port/include/esp_clk_internal.h */
extern void IRAM_ATTR rtc_clk_select_rtc_slow_clk(rtc_slow_freq_t slow_clk);
2018-10-08 12:20:49 +02:00
/* forward declarations */
static void system_init(void);
static void intr_matrix_clear(void);
typedef int32_t esp_err_t;
2022-02-01 22:11:43 +01:00
uint64_t g_startup_time = 0;
2018-10-08 12:20:49 +02:00
/**
* @brief CPU startup function
*
* This function is the entry point in the user application. It is called
* after a system reset to startup the system.
*/
NORETURN void IRAM call_start_cpu0 (void)
{
register uint32_t *sp __asm__ ("a1"); (void)sp;
2022-02-01 22:11:43 +01:00
esp_cpu_configure_region_protection();
2018-10-08 12:20:49 +02:00
/* move exception vectors to IRAM */
2022-02-01 22:11:43 +01:00
asm volatile ("wsr %0, vecbase\n" ::"r"(&_iram_start));
2018-10-08 12:20:49 +02:00
RESET_REASON reset_reason = rtc_get_reset_reason(PRO_CPU_NUM);
/* reset from panic handler by RWDT or TG0WDT */
if (reset_reason == RTCWDT_SYS_RESET || reset_reason == TG0WDT_SYS_RESET) {
2022-02-01 22:11:43 +01:00
/* TODO esp_panic_wdt_stop was called here in former versions */
2018-10-08 12:20:49 +02:00
}
2022-02-17 11:08:11 +01:00
#ifdef MODULE_PUF_SRAM
puf_sram_init((uint8_t *)&_sheap, SEED_RAM_LEN);
#endif
2018-10-08 12:20:49 +02:00
/* Clear BSS. Please do not attempt to do any complex stuff */
/* (like early logging) before this. */
2021-11-16 16:50:28 +01:00
/* cppcheck-suppress comparePointers */
2018-10-08 12:20:49 +02:00
memset(&_bss_start, 0, (&_bss_end - &_bss_start) * sizeof(_bss_start));
/* if we are not waking up from deep sleep, clear RTC bss */
if (reset_reason != DEEPSLEEP_RESET) {
2021-11-16 16:50:28 +01:00
/* cppcheck-suppress comparePointers */
2018-10-08 12:20:49 +02:00
memset(&_rtc_bss_start, 0, (&_rtc_bss_end - &_rtc_bss_start));
}
/* initialize RTC data after power on */
if (reset_reason == POWERON_RESET || reset_reason == RTCWDT_RTC_RESET) {
2021-11-16 16:50:28 +01:00
/* cppcheck-suppress comparePointers */
2018-10-08 12:20:49 +02:00
memset(&_rtc_bss_rtc_start, 0, (&_rtc_bss_rtc_end - &_rtc_bss_rtc_start));
}
uint8_t cpu_id[CPUID_LEN];
cpuid_get ((void*)cpu_id);
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_ESP_LOG_STARTUP)
ets_printf("\n");
LOG_STARTUP("Starting ESP32 with ID: ");
2018-10-08 12:20:49 +02:00
for (unsigned i = 0; i < CPUID_LEN; i++) {
ets_printf("%02x", cpu_id[i]);
}
ets_printf("\n");
extern char* esp_get_idf_version(void);
LOG_STARTUP("ESP-IDF SDK Version %s\n\n", esp_get_idf_version());
#endif
2018-10-08 12:20:49 +02:00
2020-02-20 08:30:06 +01:00
if (reset_reason == DEEPSLEEP_RESET) {
/* the cause has to be read to clear it */
esp_sleep_wakeup_cause_t cause = esp_sleep_get_wakeup_cause();
(void)cause;
LOG_STARTUP("Restart after deep sleep, wake-up cause: %d\n", cause);
}
LOG_STARTUP("Current clocks in Hz: CPU=%d APB=%d XTAL=%d SLOW=%d\n",
2022-02-01 22:11:43 +01:00
esp_clk_cpu_freq(),
esp_clk_apb_freq(), esp_clk_xtal_freq(),
2018-10-08 12:20:49 +02:00
rtc_clk_slow_freq_get_hz());
if (IS_ACTIVE(ENABLE_DEBUG)) {
ets_printf("reset reason: %d\n", reset_reason);
ets_printf("_stack %p\n", sp);
ets_printf("_bss_start %p\n", &_bss_start);
ets_printf("_bss_end %p\n", &_bss_end);
if (!IS_ACTIVE(MODULE_ESP_IDF_HEAP)) {
ets_printf("_heap_start %p\n", &_sheap);
ets_printf("_heap_end %p\n", &_eheap);
ets_printf("_heap_free %u\n", get_free_heap_size());
}
}
2018-10-08 12:20:49 +02:00
LOG_STARTUP("PRO cpu is up (single core mode, only PRO cpu is used)\n");
2018-10-08 12:20:49 +02:00
/* disable APP cpu */
DPORT_CLEAR_PERI_REG_MASK(DPORT_APPCPU_CTRL_B_REG, DPORT_APPCPU_CLKGATE_EN);
if (IS_ACTIVE(MODULE_ESP_IDF_HEAP)) {
/* init heap */
heap_caps_init();
if (IS_ACTIVE(ENABLE_DEBUG)) {
ets_printf("Heap free: %u byte\n", get_free_heap_size());
}
}
2018-10-08 12:20:49 +02:00
/* init SPI RAM if enabled */
2022-02-01 22:11:43 +01:00
#if CONFIG_SPIRAM_SUPPORT && CONFIG_SPIRAM_BOOT_INIT
if (esp_spiram_init() != ESP_OK) {
LOG_STARTUP("Failed to initialize SPI RAM\n");
exit(1);
}
#endif
2018-10-08 12:20:49 +02:00
LOG_STARTUP("PRO cpu starts user code\n");
2018-10-08 12:20:49 +02:00
system_init();
UNREACHABLE();
}
static void IRAM system_clk_init (void)
{
/* first initialize RTC with default configuration */
rtc_config_t rtc_cfg = RTC_CONFIG_DEFAULT();
rtc_init_module(rtc_cfg);
/* configure main crystal frequency if necessary */
if (CONFIG_ESP32_XTAL_FREQ != RTC_XTAL_FREQ_AUTO &&
CONFIG_ESP32_XTAL_FREQ != rtc_clk_xtal_freq_get()) {
bootloader_clock_configure();
}
2018-10-08 12:20:49 +02:00
/* set FAST_CLK to internal low power clock of 8 MHz */
rtc_clk_fast_freq_set(RTC_FAST_FREQ_8M);
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_ESP_RTC_TIMER_32K)
/* set SLOW_CLK to external 32.768 kHz crystal clock */
2022-02-01 22:11:43 +01:00
rtc_clk_select_rtc_slow_clk(RTC_SLOW_FREQ_32K_XTAL);
#else
2018-10-08 12:20:49 +02:00
/* set SLOW_CLK to internal low power clock of 150 kHz */
rtc_clk_32k_enable(false);
2022-02-01 22:11:43 +01:00
rtc_clk_select_rtc_slow_clk(RTC_SLOW_FREQ_RTC);
#endif
2018-10-08 12:20:49 +02:00
LOG_STARTUP("Switching system clocks can lead to some unreadable characters\n");
/* wait until UART is idle to avoid losing output */
2018-10-08 12:20:49 +02:00
uart_tx_wait_idle(CONFIG_CONSOLE_UART_NUM);
/* determine configured CPU clock frequency from sdk_conf.h */
rtc_cpu_freq_t freq;
switch (CONFIG_ESP32_DEFAULT_CPU_FREQ_MHZ) {
2019-10-23 21:13:52 +02:00
case 40: freq = RTC_CPU_FREQ_XTAL; /* derived from external crystal */
2018-10-08 12:20:49 +02:00
break; /* normally 40 MHz */
case 80: freq = RTC_CPU_FREQ_80M; /* derived from PLL */
break;
case 160: freq = RTC_CPU_FREQ_160M; /* derived from PLL */
break;
case 240: freq = RTC_CPU_FREQ_240M; /* derived from PLL */
break;
default: freq = RTC_CPU_FREQ_2M; /* frequencies <= 8 MHz are
set to 2 MHz and handled later */
}
2022-02-01 22:11:43 +01:00
uint32_t freq_before = esp_clk_cpu_freq() / MHZ;
2018-10-08 12:20:49 +02:00
if (freq_before != CONFIG_ESP32_DEFAULT_CPU_FREQ_MHZ) {
2022-02-01 22:11:43 +01:00
rtc_cpu_freq_config_t clk_cfg;
rtc_clk_cpu_freq_to_config(freq, &clk_cfg);
/* set configured CPU frequency */
2022-02-01 22:11:43 +01:00
rtc_clk_cpu_freq_set_config(&clk_cfg);
2018-10-08 12:20:49 +02:00
/* Recalculate the ccount to make time calculation correct. */
uint32_t freq_after = CONFIG_ESP32_DEFAULT_CPU_FREQ_MHZ;
XTHAL_SET_CCOUNT( XTHAL_GET_CCOUNT() * freq_after / freq_before );
}
2018-10-08 12:20:49 +02:00
}
extern void IRAM_ATTR thread_yield_isr(void* arg);
static NORETURN void IRAM system_init (void)
{
/* enable cached read from flash */
Cache_Read_Enable(PRO_CPU_NUM);
/* initialize the ISR stack for usage measurements */
thread_isr_stack_init();
/* initialize clocks (CPU_CLK, APB_CLK, SLOW and FAST) */
system_clk_init();
/* disable clocks of peripherals that are not needed at startup */
esp_perip_clk_init();
2022-02-01 22:11:43 +01:00
g_startup_time = esp_rtc_get_time_us();
2018-10-08 12:20:49 +02:00
/* set configured console UART baudrate */
const int uart_clk_freq = rtc_clk_apb_freq_get();
uart_tx_wait_idle(CONFIG_CONSOLE_UART_NUM);
uart_div_modify(CONFIG_CONSOLE_UART_NUM,
(uart_clk_freq << 4) / STDIO_UART_BAUDRATE);
/* initialize system call tables of ESP32 rom and newlib */
syscalls_init();
2019-10-23 21:13:52 +02:00
/* install exception handlers */
2018-10-08 12:20:49 +02:00
init_exceptions();
/* clear interrupt matrix */
intr_matrix_clear();
/* systemwide UART initialization */
extern void uart_system_init (void);
uart_system_init();
/* Disable the hold flag of all RTC GPIO pins */
RTCCNTL.hold_force.val = 0;
/* set log levels for SDK library outputs */
extern void esp_log_level_set(const char* tag, esp_log_level_t level);
esp_log_level_set("wifi", LOG_DEBUG);
2018-10-08 12:20:49 +02:00
/* init watchdogs */
system_wdt_init();
/* init random number generator */
srand(hwrand());
/* add SPI RAM to heap if enabled */
2022-02-01 22:11:43 +01:00
#if CONFIG_SPIRAM_SUPPORT && CONFIG_SPIRAM_BOOT_INIT
esp_spiram_init_cache();
esp_spiram_add_to_heapalloc();
#endif
2018-10-08 12:20:49 +02:00
/* print some infos */
LOG_STARTUP("Used clocks in Hz: CPU=%d APB=%d XTAL=%d FAST=%d SLOW=%d\n",
2022-02-01 22:11:43 +01:00
esp_clk_cpu_freq(),
esp_clk_apb_freq(), esp_clk_xtal_freq(),
rtc_clk_fast_freq_get() == RTC_FAST_FREQ_8M ? 8 * MHZ
: esp_clk_xtal_freq()/4,
rtc_clk_slow_freq_get_hz());
LOG_STARTUP("XTAL calibration value: %d\n", esp_clk_slowclk_cal_get());
LOG_STARTUP("Heap free: %u bytes\n", get_free_heap_size());
uart_tx_wait_idle(CONFIG_CONSOLE_UART_NUM);
2018-10-08 12:20:49 +02:00
/* initialize stdio */
stdio_init();
/* disable buffering in stdio */
setvbuf(_stdout_r(_REENT), NULL, _IONBF, 0);
setvbuf(_stderr_r(_REENT), NULL, _IONBF, 0);
2018-10-08 12:20:49 +02:00
/* trigger static peripheral initialization */
periph_init();
/* print system time */
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_PERIPH_RTC)
struct tm _sys_time;
rtc_get_time(&_sys_time);
LOG_STARTUP("System time: %04d-%02d-%02d %02d:%02d:%02d\n",
_sys_time.tm_year + 1900, _sys_time.tm_mon + 1, _sys_time.tm_mday,
_sys_time.tm_hour, _sys_time.tm_min, _sys_time.tm_sec);
#endif
2018-10-08 12:20:49 +02:00
/* print the board config */
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_ESP_LOG_STARTUP)
2018-10-08 12:20:49 +02:00
print_board_config();
#endif
2018-10-08 12:20:49 +02:00
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_MTD)
/* init flash drive */
extern void spi_flash_drive_init (void);
spi_flash_drive_init();
2022-02-01 22:11:43 +01:00
#endif
/* initialize the board */
extern void board_init(void);
board_init();
2018-10-08 12:20:49 +02:00
/* route a software interrupt source to CPU as trigger for thread yields */
intr_matrix_set(PRO_CPU_NUM, ETS_FROM_CPU_INTR0_SOURCE, CPU_INUM_SOFTWARE);
/* set thread yield handler and enable the software interrupt */
xt_set_interrupt_handler(CPU_INUM_SOFTWARE, thread_yield_isr, NULL);
xt_ints_on(BIT(CPU_INUM_SOFTWARE));
/* initialize ESP system event loop */
extern void esp_event_handler_init(void);
esp_event_handler_init();
2022-02-01 22:11:43 +01:00
/* initialize ESP-IDF timer task */
esp_timer_init();
2018-10-08 12:20:49 +02:00
/* starting RIOT */
2022-02-01 22:11:43 +01:00
#if IS_USED(MODULE_ESP_LOG_STARTUP)
LOG_STARTUP("Starting RIOT kernel on PRO cpu\n");
uart_tx_wait_idle(CONFIG_CONSOLE_UART_NUM);
#else
puts("");
#endif
2018-10-08 12:20:49 +02:00
kernel_init();
UNREACHABLE();
}
static void intr_matrix_clear(void)
{
/* attach all peripheral interrupt sources (Technical Reference, Table 7) */
/* to an arbitrary CPU interrupt number (Technical Reference, Table 8) */
for (int i = ETS_WIFI_MAC_INTR_SOURCE; i <= ETS_CACHE_IA_INTR_SOURCE; i++) {
intr_matrix_set(PRO_CPU_NUM, i, ETS_INVALID_INUM);
}
}