Merge branch 'bugfix/rtc_clk_cpu_freq_set_xtal_behavior_v5.3' into 'release/v5.3'

fix(clk): rtc_clk_cpu_freq_set_xtal will always disable CPU's PLL (v5.3)

See merge request espressif/esp-idf!35944
This commit is contained in:
morris
2024-12-26 10:48:02 +08:00
11 changed files with 26 additions and 35 deletions

View File

@@ -18,8 +18,8 @@ extern "C" {
* @brief Switch CPU clock source to XTAL, and let cpu frequency equal to main XTAL frequency. * @brief Switch CPU clock source to XTAL, and let cpu frequency equal to main XTAL frequency.
* *
* This function does not disable CPU's source PLL. If the PLL requires to be disabled to save power, please call * This function does not disable CPU's source PLL. If the PLL requires to be disabled to save power, please call
* `rtc_clk_cpu_freq_set_xtal` instead. It does one extra check (if necessary) to see whether can disable the * `rtc_clk_cpu_freq_set_xtal` instead. It will always disable the corresponding PLL after switching the CPU clock
* corresponding PLL after switching the CPU clock source to XTAL. * source to XTAL (except for S2).
* *
* Currently, this function should only be called in `esp_restart_noos` and `esp_restart_noos_dig` to switch the CPU * Currently, this function should only be called in `esp_restart_noos` and `esp_restart_noos_dig` to switch the CPU
* clock source back to XTAL (by default) before reset. * clock source back to XTAL (by default) before reset.
@@ -32,12 +32,6 @@ void rtc_clk_cpu_set_to_default_config(void);
* Currently, this function is only used for tracking whether USB Serial/JTAG is using the 48MHz PHY clock * Currently, this function is only used for tracking whether USB Serial/JTAG is using the 48MHz PHY clock
* *
* Note: Calling this function only helps to not disable the BBPLL clock in `rtc_clk_cpu_freq_set_config`. * Note: Calling this function only helps to not disable the BBPLL clock in `rtc_clk_cpu_freq_set_config`.
* For light and deep sleep, whether to disable the BBPLL in the internal call to `rtc_clk_cpu_freq_set_xtal`
* varies for targets.
* On ESP32C3/S3, USB CDC device can not function properly during sleep due to the lack of APB clock. Therefore.
* `rtc_clk_cpu_freq_set_xtal` will always disable BBPLL, no matter whether BBPLL has any consumer.
* On ESP32C6/H2, USB CDC device can maintain the minimum connection with the host during sleep, so
* `rtc_clk_cpu_freq_set_xtal` will check for BBPLL consumers, and keep BBPLL if USB Serial/JTAG is in use.
*/ */
void rtc_clk_bbpll_add_consumer(void); void rtc_clk_bbpll_add_consumer(void);

View File

@@ -340,8 +340,9 @@ void rtc_clk_cpu_freq_get_config(rtc_cpu_freq_config_t *out_config);
* rtc_clk_cpu_freq_set_config when a switch to XTAL is needed. * rtc_clk_cpu_freq_set_config when a switch to XTAL is needed.
* Assumes that XTAL frequency has been determined — don't call in startup code. * Assumes that XTAL frequency has been determined — don't call in startup code.
* *
* @note On ESP32C5, this function will check whether BBPLL can be disabled. If there is no consumer, then BBPLL will be * @note This function always disables BBPLL after switching the CPU clock source to XTAL for power saving purpose.
* turned off. The behaviour is the same as using rtc_clk_cpu_freq_set_config to switch cpu clock source to XTAL. * If this is unwanted, please use rtc_clk_cpu_freq_set_config. It helps to check whether USB Serial JTAG is in use,
* if so, then BBPLL will not be turned off.
*/ */
void rtc_clk_cpu_freq_set_xtal(void); void rtc_clk_cpu_freq_set_xtal(void);

View File

@@ -455,10 +455,7 @@ void rtc_clk_cpu_freq_set_config_fast(const rtc_cpu_freq_config_t *config)
void rtc_clk_cpu_freq_set_xtal(void) void rtc_clk_cpu_freq_set_xtal(void)
{ {
rtc_clk_cpu_set_to_default_config(); rtc_clk_cpu_set_to_default_config();
// We don't turn off the bbpll if some consumers depend on bbpll rtc_clk_bbpll_disable();
if (!s_bbpll_digi_consumers_ref_count) {
rtc_clk_bbpll_disable();
}
} }
void rtc_clk_cpu_set_to_default_config(void) void rtc_clk_cpu_set_to_default_config(void)
@@ -466,6 +463,7 @@ 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); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
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_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)

View File

@@ -338,8 +338,9 @@ void rtc_clk_cpu_freq_get_config(rtc_cpu_freq_config_t *out_config);
* rtc_clk_cpu_freq_set_config when a switch to XTAL is needed. * rtc_clk_cpu_freq_set_config when a switch to XTAL is needed.
* Assumes that XTAL frequency has been determined — don't call in startup code. * Assumes that XTAL frequency has been determined — don't call in startup code.
* *
* @note On ESP32C6, this function will check whether BBPLL can be disabled. If there is no consumer, then BBPLL will be * @note This function always disables BBPLL after switching the CPU clock source to XTAL for power saving purpose.
* turned off. The behaviour is the same as using rtc_clk_cpu_freq_set_config to switch cpu clock source to XTAL. * If this is unwanted, please use rtc_clk_cpu_freq_set_config. It helps to check whether USB Serial JTAG is in use,
* if so, then BBPLL will not be turned off.
*/ */
void rtc_clk_cpu_freq_set_xtal(void); void rtc_clk_cpu_freq_set_xtal(void);

View File

@@ -338,10 +338,7 @@ void rtc_clk_cpu_freq_set_config_fast(const rtc_cpu_freq_config_t *config)
void rtc_clk_cpu_freq_set_xtal(void) void rtc_clk_cpu_freq_set_xtal(void)
{ {
rtc_clk_cpu_set_to_default_config(); rtc_clk_cpu_set_to_default_config();
// We don't turn off the bbpll if some consumers depend on bbpll rtc_clk_bbpll_disable();
if (!s_bbpll_digi_consumers_ref_count) {
rtc_clk_bbpll_disable();
}
} }
void rtc_clk_cpu_set_to_default_config(void) void rtc_clk_cpu_set_to_default_config(void)
@@ -349,6 +346,7 @@ 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); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
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_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)

View File

@@ -338,8 +338,9 @@ void rtc_clk_cpu_freq_get_config(rtc_cpu_freq_config_t *out_config);
* rtc_clk_cpu_freq_set_config when a switch to XTAL is needed. * rtc_clk_cpu_freq_set_config when a switch to XTAL is needed.
* Assumes that XTAL frequency has been determined — don't call in startup code. * Assumes that XTAL frequency has been determined — don't call in startup code.
* *
* @note On ESP32C61, this function will check whether BBPLL can be disabled. If there is no consumer, then BBPLL will be * @note This function always disables BBPLL after switching the CPU clock source to XTAL for power saving purpose.
* turned off. The behaviour is the same as using rtc_clk_cpu_freq_set_config to switch cpu clock source to XTAL. * If this is unwanted, please use rtc_clk_cpu_freq_set_config. It helps to check whether USB Serial JTAG is in use,
* if so, then BBPLL will not be turned off.
*/ */
void rtc_clk_cpu_freq_set_xtal(void); void rtc_clk_cpu_freq_set_xtal(void);

View File

@@ -343,10 +343,7 @@ void rtc_clk_cpu_freq_set_config_fast(const rtc_cpu_freq_config_t *config)
void rtc_clk_cpu_freq_set_xtal(void) void rtc_clk_cpu_freq_set_xtal(void)
{ {
rtc_clk_cpu_set_to_default_config(); rtc_clk_cpu_set_to_default_config();
// We don't turn off the bbpll if some consumers depend on bbpll rtc_clk_bbpll_disable();
if (!s_bbpll_digi_consumers_ref_count) {
rtc_clk_bbpll_disable();
}
} }
void rtc_clk_cpu_set_to_default_config(void) void rtc_clk_cpu_set_to_default_config(void)
@@ -354,6 +351,7 @@ 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); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
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_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)

View File

@@ -354,8 +354,9 @@ void rtc_clk_cpu_freq_get_config(rtc_cpu_freq_config_t *out_config);
* rtc_clk_cpu_freq_set_config when a switch to XTAL is needed. * rtc_clk_cpu_freq_set_config when a switch to XTAL is needed.
* Assumes that XTAL frequency has been determined — don't call in startup code. * Assumes that XTAL frequency has been determined — don't call in startup code.
* *
* @note On ESP32H2, this function will check whether BBPLL can be disabled. If there is no consumer, then BBPLL will be * @note This function always disables BBPLL after switching the CPU clock source to XTAL for power saving purpose.
* turned off. The behaviour is the same as using rtc_clk_cpu_freq_set_config to switch cpu clock source to XTAL. * If this is unwanted, please use rtc_clk_cpu_freq_set_config. It helps to check whether USB Serial JTAG is in use,
* if so, then BBPLL will not be turned off.
*/ */
void rtc_clk_cpu_freq_set_xtal(void); void rtc_clk_cpu_freq_set_xtal(void);

View File

@@ -398,10 +398,7 @@ void rtc_clk_cpu_freq_set_config_fast(const rtc_cpu_freq_config_t *config)
void rtc_clk_cpu_freq_set_xtal(void) void rtc_clk_cpu_freq_set_xtal(void)
{ {
rtc_clk_cpu_set_to_default_config(); rtc_clk_cpu_set_to_default_config();
// We don't turn off the bbpll if some consumers only depends on bbpll rtc_clk_bbpll_disable();
if (!s_bbpll_digi_consumers_ref_count) {
rtc_clk_bbpll_disable();
}
} }
void rtc_clk_cpu_set_to_default_config(void) void rtc_clk_cpu_set_to_default_config(void)
@@ -409,6 +406,7 @@ 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); rtc_clk_cpu_freq_to_xtal(freq_mhz, 1);
s_cur_pll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }
soc_xtal_freq_t rtc_clk_xtal_freq_get(void) soc_xtal_freq_t rtc_clk_xtal_freq_get(void)

View File

@@ -350,8 +350,8 @@ void rtc_clk_cpu_freq_get_config(rtc_cpu_freq_config_t *out_config);
* rtc_clk_cpu_freq_set_config when a switch to XTAL is needed. * rtc_clk_cpu_freq_set_config when a switch to XTAL is needed.
* Assumes that XTAL frequency has been determined — don't call in startup code. * Assumes that XTAL frequency has been determined — don't call in startup code.
* *
* @note On ESP32C6, this function will check whether BBPLL can be disabled. If there is no consumer, then BBPLL will be * @note On ESP32P4, this function always disables CPLL after switching the CPU clock source to XTAL,
* turned off. The behaviour is the same as using rtc_clk_cpu_freq_set_config to switch cpu clock source to XTAL. * since there is no peripheral relies on CPLL clock (except Flash/PSRAM if their clock source selects CPLL).
*/ */
void rtc_clk_cpu_freq_set_xtal(void); void rtc_clk_cpu_freq_set_xtal(void);
@@ -494,7 +494,7 @@ void rtc_clk_apll_enable(bool enable);
* *
* @return * @return
* - 0 Failed * - 0 Failed
* - else Sucess * - else Success
*/ */
uint32_t rtc_clk_apll_coeff_calc(uint32_t freq, uint32_t *_o_div, uint32_t *_sdm0, uint32_t *_sdm1, uint32_t *_sdm2); uint32_t rtc_clk_apll_coeff_calc(uint32_t freq, uint32_t *_o_div, uint32_t *_sdm0, uint32_t *_sdm1, uint32_t *_sdm2);

View File

@@ -426,6 +426,7 @@ 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);
s_cur_cpll_freq = 0; // no disable PLL, but set freq to 0 to trigger a PLL calibration after wake-up from sleep
} }
soc_xtal_freq_t rtc_clk_xtal_freq_get(void) soc_xtal_freq_t rtc_clk_xtal_freq_get(void)