mirror of
https://github.com/espressif/esp-idf.git
synced 2025-07-31 19:24:33 +02:00
esp_timer: allow querying the timer before esp_timer_init is called
This commit is contained in:
committed by
Cao Sen Miao
parent
98d34e5f6d
commit
b5fe84a250
@@ -244,7 +244,6 @@ static void do_core_init(void)
|
|||||||
fail initializing it properly. */
|
fail initializing it properly. */
|
||||||
heap_caps_init();
|
heap_caps_init();
|
||||||
esp_newlib_init();
|
esp_newlib_init();
|
||||||
esp_newlib_time_init();
|
|
||||||
|
|
||||||
if (g_spiram_ok) {
|
if (g_spiram_ok) {
|
||||||
#if CONFIG_SPIRAM_BOOT_INIT && (CONFIG_SPIRAM_USE_CAPS_ALLOC || CONFIG_SPIRAM_USE_MALLOC)
|
#if CONFIG_SPIRAM_BOOT_INIT && (CONFIG_SPIRAM_USE_CAPS_ALLOC || CONFIG_SPIRAM_USE_MALLOC)
|
||||||
@@ -268,6 +267,12 @@ static void do_core_init(void)
|
|||||||
esp_brownout_init();
|
esp_brownout_init();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// esp_timer early initialization is required for esp_timer_get_time to work.
|
||||||
|
// This needs to happen before VFS initialization, since some USB_SERIAL_JTAG VFS driver uses
|
||||||
|
// esp_timer_get_time to determine timeout conditions.
|
||||||
|
esp_timer_early_init();
|
||||||
|
esp_newlib_time_init();
|
||||||
|
|
||||||
#ifdef CONFIG_VFS_SUPPORT_IO
|
#ifdef CONFIG_VFS_SUPPORT_IO
|
||||||
#ifdef CONFIG_ESP_CONSOLE_UART
|
#ifdef CONFIG_ESP_CONSOLE_UART
|
||||||
esp_vfs_dev_uart_register();
|
esp_vfs_dev_uart_register();
|
||||||
|
@@ -85,11 +85,28 @@ typedef struct {
|
|||||||
bool skip_unhandled_events; //!< Skip unhandled events for periodic timers
|
bool skip_unhandled_events; //!< Skip unhandled events for periodic timers
|
||||||
} esp_timer_create_args_t;
|
} esp_timer_create_args_t;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Minimal initialization of esp_timer
|
||||||
|
*
|
||||||
|
* @note This function is called from startup code. Applications do not need
|
||||||
|
* to call this function before using other esp_timer APIs.
|
||||||
|
*
|
||||||
|
* This function can be called very early in startup process, after this call
|
||||||
|
* only esp_timer_get_time function can be used.
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
* - ESP_OK on success
|
||||||
|
*/
|
||||||
|
esp_err_t esp_timer_early_init(void);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Initialize esp_timer library
|
* @brief Initialize esp_timer library
|
||||||
*
|
*
|
||||||
* @note This function is called from startup code. Applications do not need
|
* @note This function is called from startup code. Applications do not need
|
||||||
* to call this function before using other esp_timer APIs.
|
* to call this function before using other esp_timer APIs.
|
||||||
|
* Before calling this function, esp_timer_early_init must be called by the
|
||||||
|
* startup code.
|
||||||
*
|
*
|
||||||
* @return
|
* @return
|
||||||
* - ESP_OK on success
|
* - ESP_OK on success
|
||||||
|
@@ -28,9 +28,20 @@
|
|||||||
#include "esp_err.h"
|
#include "esp_err.h"
|
||||||
#include "esp_intr_alloc.h"
|
#include "esp_intr_alloc.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Minimal initialization of platform specific layer of esp_timer
|
||||||
|
* This function can be called very early in startup process, after this call
|
||||||
|
* only esp_timer_get_time function can be used.
|
||||||
|
* esp_timer_impl_init has to be called after this function to initialize the
|
||||||
|
* rest of esp_timer implementation.
|
||||||
|
* @return ESP_OK
|
||||||
|
*/
|
||||||
|
esp_err_t esp_timer_impl_early_init(void);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Initialize platform specific layer of esp_timer
|
* @brief Initialize platform specific layer of esp_timer
|
||||||
* @param alarm_handler function to call on timer interrupt
|
* @param alarm_handler function to call on timer interrupt
|
||||||
|
* Before calling this function, esp_timer_impl_early_init must be called.
|
||||||
* @return ESP_OK, ESP_ERR_NO_MEM, or one of the errors from interrupt allocator
|
* @return ESP_OK, ESP_ERR_NO_MEM, or one of the errors from interrupt allocator
|
||||||
*/
|
*/
|
||||||
esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler);
|
esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler);
|
||||||
|
@@ -364,6 +364,11 @@ static IRAM_ATTR inline bool is_initialized(void)
|
|||||||
return s_timer_task != NULL;
|
return s_timer_task != NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
esp_err_t esp_timer_early_init(void)
|
||||||
|
{
|
||||||
|
return esp_timer_impl_early_init();
|
||||||
|
}
|
||||||
|
|
||||||
esp_err_t esp_timer_init(void)
|
esp_err_t esp_timer_init(void)
|
||||||
{
|
{
|
||||||
esp_err_t err;
|
esp_err_t err;
|
||||||
|
@@ -146,9 +146,6 @@ uint64_t IRAM_ATTR esp_timer_impl_get_counter_reg(void)
|
|||||||
|
|
||||||
int64_t IRAM_ATTR esp_timer_impl_get_time(void)
|
int64_t IRAM_ATTR esp_timer_impl_get_time(void)
|
||||||
{
|
{
|
||||||
if (s_alarm_handler == NULL) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return esp_timer_impl_get_counter_reg() / TICKS_PER_US;
|
return esp_timer_impl_get_counter_reg() / TICKS_PER_US;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -207,13 +204,10 @@ void esp_timer_impl_advance(int64_t time_diff_us)
|
|||||||
portEXIT_CRITICAL(&s_time_update_lock);
|
portEXIT_CRITICAL(&s_time_update_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
esp_err_t esp_timer_impl_early_init(void)
|
||||||
{
|
{
|
||||||
s_alarm_handler = alarm_handler;
|
|
||||||
|
|
||||||
periph_module_enable(PERIPH_LACT);
|
periph_module_enable(PERIPH_LACT);
|
||||||
|
|
||||||
/* Reset the state */
|
|
||||||
REG_WRITE(CONFIG_REG, 0);
|
REG_WRITE(CONFIG_REG, 0);
|
||||||
REG_WRITE(LOAD_LO_REG, 0);
|
REG_WRITE(LOAD_LO_REG, 0);
|
||||||
REG_WRITE(LOAD_HI_REG, 0);
|
REG_WRITE(LOAD_HI_REG, 0);
|
||||||
@@ -221,6 +215,17 @@ esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
|||||||
REG_WRITE(ALARM_HI_REG, UINT32_MAX);
|
REG_WRITE(ALARM_HI_REG, UINT32_MAX);
|
||||||
REG_WRITE(LOAD_REG, 1);
|
REG_WRITE(LOAD_REG, 1);
|
||||||
REG_SET_BIT(INT_CLR_REG, TIMG_LACT_INT_CLR);
|
REG_SET_BIT(INT_CLR_REG, TIMG_LACT_INT_CLR);
|
||||||
|
REG_SET_FIELD(CONFIG_REG, TIMG_LACT_DIVIDER, APB_CLK_FREQ / 1000000 / TICKS_PER_US);
|
||||||
|
REG_SET_BIT(CONFIG_REG, TIMG_LACT_INCREASE |
|
||||||
|
TIMG_LACT_LEVEL_INT_EN |
|
||||||
|
TIMG_LACT_EN);
|
||||||
|
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
||||||
|
{
|
||||||
|
s_alarm_handler = alarm_handler;
|
||||||
|
|
||||||
esp_err_t err = esp_intr_alloc(INTR_SOURCE_LACT,
|
esp_err_t err = esp_intr_alloc(INTR_SOURCE_LACT,
|
||||||
ESP_INTR_FLAG_INTRDISABLED | ESP_INTR_FLAG_IRAM,
|
ESP_INTR_FLAG_INTRDISABLED | ESP_INTR_FLAG_IRAM,
|
||||||
@@ -239,10 +244,6 @@ esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
|||||||
|
|
||||||
esp_timer_impl_update_apb_freq(esp_clk_apb_freq() / 1000000);
|
esp_timer_impl_update_apb_freq(esp_clk_apb_freq() / 1000000);
|
||||||
|
|
||||||
REG_SET_BIT(CONFIG_REG, TIMG_LACT_INCREASE |
|
|
||||||
TIMG_LACT_LEVEL_INT_EN |
|
|
||||||
TIMG_LACT_EN);
|
|
||||||
|
|
||||||
// Set the step for the sleep mode when the timer will work
|
// Set the step for the sleep mode when the timer will work
|
||||||
// from a slow_clk frequency instead of the APB frequency.
|
// from a slow_clk frequency instead of the APB frequency.
|
||||||
uint32_t slowclk_ticks_per_us = esp_clk_slowclk_cal_get() * TICKS_PER_US;
|
uint32_t slowclk_ticks_per_us = esp_clk_slowclk_cal_get() * TICKS_PER_US;
|
||||||
|
@@ -65,9 +65,6 @@ uint64_t IRAM_ATTR esp_timer_impl_get_counter_reg(void)
|
|||||||
|
|
||||||
int64_t IRAM_ATTR esp_timer_impl_get_time(void)
|
int64_t IRAM_ATTR esp_timer_impl_get_time(void)
|
||||||
{
|
{
|
||||||
if (s_alarm_handler == NULL) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return systimer_hal_get_time(SYSTIMER_COUNTER_0);
|
return systimer_hal_get_time(SYSTIMER_COUNTER_0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,6 +97,23 @@ void esp_timer_impl_advance(int64_t time_us)
|
|||||||
portEXIT_CRITICAL_SAFE(&s_time_update_lock);
|
portEXIT_CRITICAL_SAFE(&s_time_update_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
esp_err_t esp_timer_impl_early_init(void)
|
||||||
|
{
|
||||||
|
systimer_hal_init(&systimer_hal);
|
||||||
|
|
||||||
|
#if !SOC_SYSTIMER_FIXED_TICKS_US
|
||||||
|
assert(rtc_clk_xtal_freq_get() == 40 && "update the step for xtal to support other XTAL:APB frequency ratios");
|
||||||
|
systimer_hal_set_steps_per_tick(&systimer_hal, 0, 2); // for xtal
|
||||||
|
systimer_hal_set_steps_per_tick(&systimer_hal, 1, 1); // for pll
|
||||||
|
#endif
|
||||||
|
|
||||||
|
systimer_hal_enable_counter(&systimer_hal, SYSTIMER_LL_COUNTER_CLOCK);
|
||||||
|
systimer_hal_select_alarm_mode(&systimer_hal, SYSTIMER_LL_ALARM_CLOCK, SYSTIMER_ALARM_MODE_ONESHOT);
|
||||||
|
systimer_hal_connect_alarm_counter(&systimer_hal, SYSTIMER_LL_ALARM_CLOCK, SYSTIMER_LL_COUNTER_CLOCK);
|
||||||
|
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
||||||
{
|
{
|
||||||
s_alarm_handler = alarm_handler;
|
s_alarm_handler = alarm_handler;
|
||||||
@@ -117,11 +131,6 @@ esp_err_t esp_timer_impl_init(intr_handler_t alarm_handler)
|
|||||||
goto err_intr_alloc;
|
goto err_intr_alloc;
|
||||||
}
|
}
|
||||||
|
|
||||||
systimer_hal_init();
|
|
||||||
systimer_hal_enable_counter(SYSTIMER_COUNTER_0);
|
|
||||||
systimer_hal_select_alarm_mode(SYSTIMER_ALARM_2, SYSTIMER_ALARM_MODE_ONESHOT);
|
|
||||||
systimer_hal_connect_alarm_counter(SYSTIMER_ALARM_2, SYSTIMER_COUNTER_0);
|
|
||||||
|
|
||||||
/* TODO: if SYSTIMER is used for anything else, access to SYSTIMER_INT_ENA_REG has to be
|
/* TODO: if SYSTIMER is used for anything else, access to SYSTIMER_INT_ENA_REG has to be
|
||||||
* protected by a shared spinlock. Since this code runs as part of early startup, this
|
* protected by a shared spinlock. Since this code runs as part of early startup, this
|
||||||
* is practically not an issue.
|
* is practically not an issue.
|
||||||
|
Reference in New Issue
Block a user