2020-12-29 12:31:54 +08:00
// Copyright 2015-2020 Espressif Systems (Shanghai) PTE LTD
2019-12-26 16:30:03 +08:00
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
2020-12-29 12:31:54 +08:00
//
2019-12-26 16:30:03 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
# include <stdint.h>
# include <stdio.h>
# include <stdlib.h>
# include <stdbool.h>
2020-12-29 12:31:54 +08:00
# include "sdkconfig.h"
2019-12-26 16:30:03 +08:00
# include "freertos/FreeRTOS.h"
# include "freertos/task.h"
2020-12-29 12:31:54 +08:00
# include "esp_types.h"
2019-12-26 16:30:03 +08:00
# include "esp_err.h"
# include "esp_intr_alloc.h"
# include "esp_attr.h"
# include "esp_freertos_hooks.h"
# include "soc/timer_periph.h"
# include "driver/timer.h"
# include "driver/periph_ctrl.h"
# include "esp_int_wdt.h"
# include "esp_private/system_internal.h"
2020-12-29 12:31:54 +08:00
# include "hal/cpu_hal.h"
2019-12-26 16:30:03 +08:00
# include "hal/timer_types.h"
# include "hal/wdt_hal.h"
2020-12-03 17:17:43 +08:00
# include "hal/interrupt_controller_hal.h"
2019-12-26 16:30:03 +08:00
# if CONFIG_ESP_INT_WDT
2020-05-29 14:53:30 +08:00
# define WDT_INT_NUM ETS_T1_WDT_INUM
2019-12-26 16:30:03 +08:00
# define IWDT_INSTANCE WDT_MWDT1
# define IWDT_PRESCALER MWDT1_TICK_PRESCALER //Tick period of 500us if WDT source clock is 80MHz
# define IWDT_TICKS_PER_US MWDT1_TICKS_PER_US
# define IWDT_INITIAL_TIMEOUT_S 5
static wdt_hal_context_t iwdt_context ;
2020-05-29 14:53:30 +08:00
# if CONFIG_ESP32_ECO3_CACHE_LOCK_FIX
/*
2020-12-29 12:31:54 +08:00
* This parameter is used to indicate the response time of Interrupt watchdog to
2020-05-29 14:53:30 +08:00
* identify the live lock .
*/
# define IWDT_LIVELOCK_TIMEOUT_MS (20)
extern uint32_t _l4_intr_livelock_counter , _l4_intr_livelock_max ;
# endif
2019-12-26 16:30:03 +08:00
//Take care: the tick hook can also be called before esp_int_wdt_init() is called.
# if CONFIG_ESP_INT_WDT_CHECK_CPU1
//Not static; the ISR assembly checks this.
bool int_wdt_app_cpu_ticked = false ;
2020-12-29 12:31:54 +08:00
static void IRAM_ATTR tick_hook ( void )
{
if ( cpu_hal_get_core_id ( ) ! = 0 ) {
2019-12-26 16:30:03 +08:00
int_wdt_app_cpu_ticked = true ;
} else {
//Only feed wdt if app cpu also ticked.
if ( int_wdt_app_cpu_ticked ) {
//Todo: Check if there's a way to avoid reconfiguring the stages on each feed.
wdt_hal_write_protect_disable ( & iwdt_context ) ;
//Reconfigure stage timeouts
2020-05-29 14:53:30 +08:00
# if CONFIG_ESP32_ECO3_CACHE_LOCK_FIX
_l4_intr_livelock_counter = 0 ;
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE0 ,
2020-12-29 12:31:54 +08:00
CONFIG_ESP_INT_WDT_TIMEOUT_MS * 1000 / IWDT_TICKS_PER_US / ( _l4_intr_livelock_max + 1 ) , WDT_STAGE_ACTION_INT ) ; //Set timeout before interrupt
2020-05-29 14:53:30 +08:00
# else
2020-12-29 12:31:54 +08:00
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE0 , CONFIG_ESP_INT_WDT_TIMEOUT_MS * 1000 / IWDT_TICKS_PER_US , WDT_STAGE_ACTION_INT ) ; //Set timeout before interrupt
2020-05-29 14:53:30 +08:00
# endif
2020-12-29 12:31:54 +08:00
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE1 , 2 * CONFIG_ESP_INT_WDT_TIMEOUT_MS * 1000 / IWDT_TICKS_PER_US , WDT_STAGE_ACTION_RESET_SYSTEM ) ; //Set timeout before reset
2019-12-26 16:30:03 +08:00
wdt_hal_feed ( & iwdt_context ) ;
wdt_hal_write_protect_enable ( & iwdt_context ) ;
int_wdt_app_cpu_ticked = false ;
}
}
}
# else
2020-12-29 12:31:54 +08:00
static void IRAM_ATTR tick_hook ( void )
{
2020-07-29 22:03:46 +08:00
# if !CONFIG_FREERTOS_UNICORE
2020-12-29 12:31:54 +08:00
if ( cpu_hal_get_core_id ( ) ! = 0 ) {
2019-12-26 16:30:03 +08:00
return ;
}
# endif
//Todo: Check if there's a way to avoid reconfiguring the stages on each feed.
wdt_hal_write_protect_disable ( & iwdt_context ) ;
//Reconfigure stage timeouts
2020-12-29 12:31:54 +08:00
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE0 , CONFIG_ESP_INT_WDT_TIMEOUT_MS * 1000 / IWDT_TICKS_PER_US , WDT_STAGE_ACTION_INT ) ; //Set timeout before interrupt
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE1 , 2 * CONFIG_ESP_INT_WDT_TIMEOUT_MS * 1000 / IWDT_TICKS_PER_US , WDT_STAGE_ACTION_RESET_SYSTEM ) ; //Set timeout before reset
2019-12-26 16:30:03 +08:00
wdt_hal_feed ( & iwdt_context ) ;
wdt_hal_write_protect_enable ( & iwdt_context ) ;
}
# endif
2020-12-29 12:31:54 +08:00
void esp_int_wdt_init ( void )
{
2019-12-26 16:30:03 +08:00
periph_module_enable ( PERIPH_TIMG1_MODULE ) ;
//The timer configs initially are set to 5 seconds, to make sure the CPU can start up. The tick hook sets
//it to their actual value.
wdt_hal_init ( & iwdt_context , IWDT_INSTANCE , IWDT_PRESCALER , true ) ;
wdt_hal_write_protect_disable ( & iwdt_context ) ;
//1st stage timeout: interrupt
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE0 , IWDT_INITIAL_TIMEOUT_S * 1000000 / IWDT_TICKS_PER_US , WDT_STAGE_ACTION_INT ) ;
//2nd stage timeout: reset system
wdt_hal_config_stage ( & iwdt_context , WDT_STAGE1 , IWDT_INITIAL_TIMEOUT_S * 1000000 / IWDT_TICKS_PER_US , WDT_STAGE_ACTION_RESET_SYSTEM ) ;
//Enable WDT
wdt_hal_enable ( & iwdt_context ) ;
wdt_hal_write_protect_enable ( & iwdt_context ) ;
}
void esp_int_wdt_cpu_init ( void )
{
2020-12-29 12:31:54 +08:00
assert ( ( CONFIG_ESP_INT_WDT_TIMEOUT_MS > = ( portTICK_PERIOD_MS < < 1 ) ) & & " Interrupt watchdog timeout needs to meet twice the RTOS tick period! " ) ;
esp_register_freertos_tick_hook_for_cpu ( tick_hook , cpu_hal_get_core_id ( ) ) ;
2019-12-26 16:30:03 +08:00
ESP_INTR_DISABLE ( WDT_INT_NUM ) ;
2020-12-29 12:31:54 +08:00
intr_matrix_set ( cpu_hal_get_core_id ( ) , ETS_TG1_WDT_LEVEL_INTR_SOURCE , WDT_INT_NUM ) ;
2020-12-03 17:17:43 +08:00
2020-12-29 12:31:54 +08:00
/* Set the type and priority to watch dog interrupts */
# if SOC_CPU_HAS_FLEXIBLE_INTC
2020-12-25 10:44:22 -03:00
interrupt_controller_hal_set_int_type ( WDT_INT_NUM , INTR_TYPE_LEVEL ) ;
interrupt_controller_hal_set_int_level ( WDT_INT_NUM , SOC_INTERRUPT_LEVEL_MEDIUM ) ;
2020-12-03 17:17:43 +08:00
# endif
2020-05-29 14:53:30 +08:00
# if CONFIG_ESP32_ECO3_CACHE_LOCK_FIX
/*
* This is a workaround for issue 3.15 in " ESP32 ECO and workarounds for
* Bugs " document.
*/
_l4_intr_livelock_counter = 0 ;
if ( soc_has_cache_lock_bug ( ) ) {
2020-12-29 12:31:54 +08:00
assert ( ( portTICK_PERIOD_MS < < 1 ) < = IWDT_LIVELOCK_TIMEOUT_MS ) ;
assert ( CONFIG_ESP_INT_WDT_TIMEOUT_MS > = ( IWDT_LIVELOCK_TIMEOUT_MS * 3 ) ) ;
_l4_intr_livelock_max = CONFIG_ESP_INT_WDT_TIMEOUT_MS / IWDT_LIVELOCK_TIMEOUT_MS - 1 ;
2020-05-29 14:53:30 +08:00
}
# endif
2020-12-03 17:17:43 +08:00
2020-12-29 12:31:54 +08:00
// We do not register a handler for the watchdog interrupt because:
// 1. Interrupt level 4 on Xtensa architecture is not servicable from C
// 2. Instead, we set the entry of watchdog interrupt to the panic handler, see riscv/vector.S and xtensa_vectors.S
2019-12-26 16:30:03 +08:00
ESP_INTR_ENABLE ( WDT_INT_NUM ) ;
}
# endif