2021-08-10 13:16:02 +02:00
|
|
|
/*
|
2021-12-22 22:18:43 +08:00
|
|
|
* SPDX-FileCopyrightText: 2018-2022 Espressif Systems (Shanghai) CO LTD
|
2021-08-10 13:16:02 +02:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
2020-03-27 19:22:47 +08:00
|
|
|
|
2022-01-12 12:23:47 +05:30
|
|
|
#include "rtc_wdt.h"
|
2020-03-27 19:22:47 +08:00
|
|
|
#include "soc/rtc.h"
|
2021-12-22 22:18:43 +08:00
|
|
|
#include "hal/efuse_ll.h"
|
2020-03-27 19:22:47 +08:00
|
|
|
|
2022-02-28 18:30:36 +08:00
|
|
|
#if CONFIG_IDF_TARGET_ESP32 || CONFIG_IDF_TARGET_ESP32S2
|
2020-03-27 19:22:47 +08:00
|
|
|
|
|
|
|
bool rtc_wdt_get_protect_status(void)
|
|
|
|
{
|
|
|
|
return READ_PERI_REG(RTC_CNTL_WDTWPROTECT_REG) != RTC_CNTL_WDT_WKEY_VALUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_wdt_protect_off(void)
|
|
|
|
{
|
|
|
|
WRITE_PERI_REG(RTC_CNTL_WDTWPROTECT_REG, RTC_CNTL_WDT_WKEY_VALUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_wdt_protect_on(void)
|
|
|
|
{
|
|
|
|
WRITE_PERI_REG(RTC_CNTL_WDTWPROTECT_REG, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rtc_wdt_enable(void)
|
|
|
|
{
|
|
|
|
REG_SET_BIT(RTC_CNTL_WDTFEED_REG, RTC_CNTL_WDT_FEED);
|
|
|
|
SET_PERI_REG_MASK(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_EN | RTC_CNTL_WDT_PAUSE_IN_SLP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_wdt_flashboot_mode_enable(void)
|
|
|
|
{
|
|
|
|
REG_SET_BIT(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_FLASHBOOT_MOD_EN);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_wdt_disable(void)
|
|
|
|
{
|
|
|
|
bool protect = rtc_wdt_get_protect_status();
|
|
|
|
if (protect) {
|
|
|
|
rtc_wdt_protect_off();
|
|
|
|
}
|
|
|
|
REG_SET_BIT(RTC_CNTL_WDTFEED_REG, RTC_CNTL_WDT_FEED);
|
|
|
|
rtc_wdt_set_stage(RTC_WDT_STAGE0, RTC_WDT_STAGE_ACTION_OFF);
|
|
|
|
rtc_wdt_set_stage(RTC_WDT_STAGE1, RTC_WDT_STAGE_ACTION_OFF);
|
|
|
|
rtc_wdt_set_stage(RTC_WDT_STAGE2, RTC_WDT_STAGE_ACTION_OFF);
|
|
|
|
rtc_wdt_set_stage(RTC_WDT_STAGE3, RTC_WDT_STAGE_ACTION_OFF);
|
|
|
|
REG_CLR_BIT(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_FLASHBOOT_MOD_EN);
|
|
|
|
REG_CLR_BIT(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_EN);
|
|
|
|
if (protect) {
|
|
|
|
rtc_wdt_protect_on();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_wdt_feed(void)
|
|
|
|
{
|
|
|
|
bool protect = rtc_wdt_get_protect_status();
|
|
|
|
if (protect) {
|
|
|
|
rtc_wdt_protect_off();
|
|
|
|
}
|
|
|
|
REG_SET_BIT(RTC_CNTL_WDTFEED_REG, RTC_CNTL_WDT_FEED);
|
|
|
|
if (protect) {
|
|
|
|
rtc_wdt_protect_on();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t get_addr_reg(rtc_wdt_stage_t stage)
|
|
|
|
{
|
|
|
|
uint32_t reg;
|
|
|
|
if (stage == RTC_WDT_STAGE0) {
|
|
|
|
reg = RTC_CNTL_WDTCONFIG1_REG;
|
|
|
|
} else if (stage == RTC_WDT_STAGE1) {
|
|
|
|
reg = RTC_CNTL_WDTCONFIG2_REG;
|
|
|
|
} else if (stage == RTC_WDT_STAGE2) {
|
|
|
|
reg = RTC_CNTL_WDTCONFIG3_REG;
|
|
|
|
} else {
|
2022-07-28 11:15:36 +08:00
|
|
|
reg = RTC_CNTL_WDTCONFIG4_REG;
|
2020-03-27 19:22:47 +08:00
|
|
|
}
|
|
|
|
return reg;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t rtc_wdt_set_time(rtc_wdt_stage_t stage, unsigned int timeout_ms)
|
|
|
|
{
|
|
|
|
if (stage > 3) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
uint32_t timeout = (uint32_t) ((uint64_t) rtc_clk_slow_freq_get_hz() * timeout_ms / 1000);
|
2022-01-18 17:34:15 +08:00
|
|
|
#if !CONFIG_IDF_TARGET_ESP32
|
2020-03-27 19:22:47 +08:00
|
|
|
if (stage == RTC_WDT_STAGE0) {
|
2021-12-22 22:18:43 +08:00
|
|
|
timeout = timeout >> (1 + efuse_ll_get_wdt_delay_sel());
|
2020-03-27 19:22:47 +08:00
|
|
|
}
|
2022-01-18 17:34:15 +08:00
|
|
|
#endif
|
2020-03-27 19:22:47 +08:00
|
|
|
WRITE_PERI_REG(get_addr_reg(stage), timeout);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
2022-07-28 11:15:36 +08:00
|
|
|
esp_err_t rtc_wdt_get_timeout(rtc_wdt_stage_t stage, unsigned int *timeout_ms)
|
2020-03-27 19:22:47 +08:00
|
|
|
{
|
|
|
|
if (stage > 3) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
uint32_t time_tick;
|
2022-07-28 11:15:36 +08:00
|
|
|
uint32_t rtc_slow_freq = rtc_clk_slow_freq_get_hz();
|
|
|
|
if (rtc_slow_freq == 0) {
|
|
|
|
return ESP_ERR_INVALID_STATE;
|
|
|
|
}
|
2020-03-27 19:22:47 +08:00
|
|
|
time_tick = READ_PERI_REG(get_addr_reg(stage));
|
2022-07-28 11:15:36 +08:00
|
|
|
*timeout_ms = time_tick * 1000 / rtc_slow_freq;
|
2020-03-27 19:22:47 +08:00
|
|
|
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t rtc_wdt_set_stage(rtc_wdt_stage_t stage, rtc_wdt_stage_action_t stage_sel)
|
|
|
|
{
|
|
|
|
if (stage > 3 || stage_sel > 4) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
if (stage == RTC_WDT_STAGE0) {
|
|
|
|
REG_SET_FIELD(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_STG0, stage_sel);
|
|
|
|
} else if (stage == RTC_WDT_STAGE1) {
|
|
|
|
REG_SET_FIELD(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_STG1, stage_sel);
|
|
|
|
} else if (stage == RTC_WDT_STAGE2) {
|
|
|
|
REG_SET_FIELD(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_STG2, stage_sel);
|
|
|
|
} else {
|
|
|
|
REG_SET_FIELD(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_STG3, stage_sel);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t rtc_wdt_set_length_of_reset_signal(rtc_wdt_reset_sig_t reset_src, rtc_wdt_length_sig_t reset_signal_length)
|
|
|
|
{
|
|
|
|
if (reset_src > 1 || reset_signal_length > 7) {
|
|
|
|
return ESP_ERR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
if (reset_src == 0) {
|
|
|
|
REG_SET_FIELD(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_SYS_RESET_LENGTH, reset_signal_length);
|
|
|
|
} else {
|
|
|
|
REG_SET_FIELD(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_CPU_RESET_LENGTH, reset_signal_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool rtc_wdt_is_on(void)
|
|
|
|
{
|
|
|
|
return (REG_GET_BIT(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_EN) != 0) || (REG_GET_BIT(RTC_CNTL_WDTCONFIG0_REG, RTC_CNTL_WDT_FLASHBOOT_MOD_EN) != 0);
|
|
|
|
}
|
2022-01-18 17:34:15 +08:00
|
|
|
|
2022-02-28 18:30:36 +08:00
|
|
|
#endif // CONFIG_IDF_TARGET_ESP32 || CONFIG_IDF_TARGET_ESP32S2
|