fix(esp_hw_support): fix wrong APB clock freq on retenion

This commit is contained in:
wuzhenghui
2025-03-21 20:52:20 +08:00
parent 38d6e17516
commit ad4f0a8708
12 changed files with 69 additions and 13 deletions

View File

@ -1,5 +1,5 @@
/* /*
* SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD * SPDX-FileCopyrightText: 2022-2025 Espressif Systems (Shanghai) CO LTD
* *
* SPDX-License-Identifier: Apache-2.0 * SPDX-License-Identifier: Apache-2.0
*/ */
@ -22,11 +22,24 @@ extern "C" {
* source to XTAL (except for S2). * source to XTAL (except for S2).
* *
* Currently, this function is only called in `esp_restart_noos` and `esp_restart_noos_dig` to switch the CPU * Currently, this function is only called in `esp_restart_noos` and `esp_restart_noos_dig` to switch the CPU
* clock source back to XTAL (by default) before reset, and in `esp_sleep_start` to switch CPU clock source to XTAL * clock source back to XTAL (by default) before reset.
* before entering sleep for PMU supported chips.
*/ */
void rtc_clk_cpu_set_to_default_config(void); void rtc_clk_cpu_set_to_default_config(void);
/**
* @brief Switch CPU clock source to XTAL, the PLL has different processing methods for different chips.
* 1. For earlier chips without PMU, there is no PMU module that can turn off the CPU's PLL, so it has to be
* disabled at here to save the power consumption. Though ESP32C3/S3 has USB CDC device, it can not function
* properly during sleep due to the lack of APB clock (before C6, USJ relies on APB clock to work). Therefore,
* we will always disable CPU's PLL (i.e. BBPLL).
* 2. For PMU supported chips, CPU's PLL power can be turned off by PMU, so no need to disable the PLL at here.
* Leaving PLL on at this stage also helps USJ keep connection and retention operation (if they rely on this PLL).
* For ESP32P4, if the APB frequency is configured as the hardware default value (10MHz), this will cause the
* regdma backup/restore to not achieve optimal performance. The MEM/APB frequency divider needs to be configured
* to 40MHz to speed up the retention speed.
*/
void rtc_clk_cpu_freq_set_xtal_for_sleep(void);
/** /**
* @brief Notify that the BBPLL has a new in-use consumer * @brief Notify that the BBPLL has a new in-use consumer
* *

View File

@ -417,6 +417,11 @@ void rtc_clk_cpu_set_to_default_config(void)
rtc_clk_wait_for_slow_cycle(); rtc_clk_wait_for_slow_cycle();
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_freq_set_xtal();
}
bool rtc_clk_cpu_freq_mhz_to_config(uint32_t freq_mhz, rtc_cpu_freq_config_t* out_config) bool rtc_clk_cpu_freq_mhz_to_config(uint32_t freq_mhz, rtc_cpu_freq_config_t* out_config)
{ {
uint32_t source_freq_mhz; uint32_t source_freq_mhz;

View File

@ -286,6 +286,11 @@ void rtc_clk_cpu_set_to_default_config(void)
rtc_clk_cpu_freq_to_xtal(freq_mhz, 1); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_freq_set_xtal();
}
/** /**
* Switch to use XTAL as the CPU clock source. * Switch to use XTAL as the CPU clock source.
* Must satisfy: cpu_freq = XTAL_FREQ / div. * Must satisfy: cpu_freq = XTAL_FREQ / div.

View File

@ -316,6 +316,11 @@ void rtc_clk_cpu_set_to_default_config(void)
rtc_clk_cpu_freq_to_xtal(freq_mhz, 1); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_freq_set_xtal();
}
/** /**
* Switch to use XTAL as the CPU clock source. * Switch to use XTAL as the CPU clock source.
* Must satisfy: cpu_freq = XTAL_FREQ / div. * Must satisfy: cpu_freq = XTAL_FREQ / div.

View File

@ -473,6 +473,11 @@ void rtc_clk_cpu_set_to_default_config(void)
s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_set_to_default_config();
}
void rtc_clk_cpu_freq_to_pll_and_pll_lock_release(int cpu_freq_mhz) void rtc_clk_cpu_freq_to_pll_and_pll_lock_release(int cpu_freq_mhz)
{ {
rtc_clk_cpu_freq_to_pll_mhz(cpu_freq_mhz); rtc_clk_cpu_freq_to_pll_mhz(cpu_freq_mhz);

View File

@ -356,6 +356,11 @@ void rtc_clk_cpu_set_to_default_config(void)
s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_set_to_default_config();
}
void rtc_clk_cpu_freq_to_pll_and_pll_lock_release(int cpu_freq_mhz) void rtc_clk_cpu_freq_to_pll_and_pll_lock_release(int cpu_freq_mhz)
{ {
rtc_clk_cpu_freq_to_pll_mhz(cpu_freq_mhz); rtc_clk_cpu_freq_to_pll_mhz(cpu_freq_mhz);

View File

@ -354,6 +354,11 @@ void rtc_clk_cpu_set_to_default_config(void)
s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_set_to_default_config();
}
void rtc_clk_cpu_freq_to_pll_and_pll_lock_release(int cpu_freq_mhz) void rtc_clk_cpu_freq_to_pll_and_pll_lock_release(int cpu_freq_mhz)
{ {
rtc_clk_cpu_freq_to_pll_mhz(cpu_freq_mhz); rtc_clk_cpu_freq_to_pll_mhz(cpu_freq_mhz);

View File

@ -416,6 +416,11 @@ void rtc_clk_cpu_set_to_default_config(void)
s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_set_to_default_config();
}
soc_xtal_freq_t rtc_clk_xtal_freq_get(void) soc_xtal_freq_t rtc_clk_xtal_freq_get(void)
{ {
uint32_t xtal_freq_mhz = clk_ll_xtal_load_freq_mhz(); uint32_t xtal_freq_mhz = clk_ll_xtal_load_freq_mhz();

View File

@ -426,6 +426,13 @@ void rtc_clk_cpu_set_to_default_config(void)
int freq_mhz = (int)rtc_clk_xtal_freq_get(); int freq_mhz = (int)rtc_clk_xtal_freq_get();
rtc_clk_cpu_freq_to_xtal(freq_mhz, 1, true); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1, true);
}
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
int freq_mhz = (int)rtc_clk_xtal_freq_get();
rtc_clk_cpu_freq_to_xtal(freq_mhz, 1, false);
s_cur_cpll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep s_cur_cpll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }

View File

@ -437,6 +437,11 @@ void rtc_clk_cpu_set_to_default_config(void)
rtc_clk_cpu_freq_to_xtal(CLK_LL_XTAL_FREQ_MHZ, 1); rtc_clk_cpu_freq_to_xtal(CLK_LL_XTAL_FREQ_MHZ, 1);
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_freq_set_xtal();
}
/** /**
* Switch to use XTAL as the CPU clock source. * Switch to use XTAL as the CPU clock source.
* Must satisfy: cpu_freq = XTAL_FREQ / div. * Must satisfy: cpu_freq = XTAL_FREQ / div.

View File

@ -378,6 +378,11 @@ void rtc_clk_cpu_set_to_default_config(void)
rtc_clk_cpu_freq_to_xtal(freq_mhz, 1); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
} }
void rtc_clk_cpu_freq_set_xtal_for_sleep(void)
{
rtc_clk_cpu_freq_set_xtal();
}
/** /**
* Switch to use XTAL as the CPU clock source. * Switch to use XTAL as the CPU clock source.
* Must satisfy: cpu_freq = XTAL_FREQ / div. * Must satisfy: cpu_freq = XTAL_FREQ / div.

View File

@ -826,16 +826,7 @@ static esp_err_t IRAM_ATTR esp_sleep_start(uint32_t pd_flags, esp_sleep_mode_t m
// Save current frequency and switch to XTAL // Save current frequency and switch to XTAL
rtc_cpu_freq_config_t cpu_freq_config; rtc_cpu_freq_config_t cpu_freq_config;
rtc_clk_cpu_freq_get_config(&cpu_freq_config); rtc_clk_cpu_freq_get_config(&cpu_freq_config);
#if SOC_PMU_SUPPORTED rtc_clk_cpu_freq_set_xtal_for_sleep();
// For PMU supported chips, CPU's PLL power can be turned off by PMU, so no need to disable the PLL at here.
// Leaving PLL on at this stage also helps USJ keep connection and retention operation (if they rely on this PLL).
rtc_clk_cpu_set_to_default_config();
#else
// For earlier chips, there is no PMU module that can turn off the CPU's PLL, so it has to be disabled at here to save the power consumption.
// Though ESP32C3/S3 has USB CDC device, it can not function properly during sleep due to the lack of APB clock (before C6, USJ relies on APB clock to work).
// Therefore, we will always disable CPU's PLL (i.e. BBPLL).
rtc_clk_cpu_freq_set_xtal();
#endif
#if SOC_PM_SUPPORT_EXT0_WAKEUP #if SOC_PM_SUPPORT_EXT0_WAKEUP
// Configure pins for external wakeup // Configure pins for external wakeup