mirror of https://github.com/ARMmbed/mbed-os.git
Watchdog: refactor driver
Watchdog is hardware driver. It interacts with HAL - provides wrapper to interact with the peripheral. Provides basic functionality: start/stop, get timeout/max timeout. It is automatically kicked by a timer, according to the timeout set in ctor.pull/10857/head
parent
c497010a56
commit
7b0915c7d4
|
@ -17,92 +17,119 @@
|
||||||
#ifdef DEVICE_WATCHDOG
|
#ifdef DEVICE_WATCHDOG
|
||||||
|
|
||||||
#include "Watchdog.h"
|
#include "Watchdog.h"
|
||||||
|
#include "VirtualWatchdog.h"
|
||||||
|
|
||||||
|
#define MS_TO_US(x) ((x) * 1000) //macro to convert millisecond to microsecond
|
||||||
|
|
||||||
namespace mbed {
|
namespace mbed {
|
||||||
|
|
||||||
Watchdog *Watchdog::_first = NULL;
|
#if DEVICE_LPTICKER
|
||||||
|
/** Create singleton instance of LowPowerTicker for watchdog periodic call back of kick.
|
||||||
|
*/
|
||||||
|
SingletonPtr<LowPowerTicker> Watchdog::_ticker;
|
||||||
|
#else
|
||||||
|
/** Create singleton instance of Ticker for watchdog periodic call back of kick.
|
||||||
|
*/
|
||||||
|
SingletonPtr<Ticker> Watchdog::_ticker;
|
||||||
|
#endif
|
||||||
|
|
||||||
Watchdog::Watchdog(uint32_t timeout, const char *const str): _name(str)
|
bool Watchdog::_running = false;
|
||||||
{
|
static const uint32_t elapsed_ms = MBED_CONF_TARGET_WATCHDOG_TIMEOUT / 2;
|
||||||
_current_count = 0;
|
|
||||||
_is_initialized = false;
|
|
||||||
_next = NULL;
|
|
||||||
_max_timeout = timeout;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Watchdog::add_to_list()
|
Watchdog::Watchdog()
|
||||||
{
|
{
|
||||||
this->_next = _first;
|
|
||||||
_first = this;
|
|
||||||
_is_initialized = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Watchdog::start()
|
|
||||||
{
|
|
||||||
MBED_ASSERT(!_is_initialized);
|
|
||||||
core_util_critical_section_enter();
|
core_util_critical_section_enter();
|
||||||
add_to_list();
|
if (_running) {
|
||||||
|
// error function does not return thus exit critical section and print an error
|
||||||
core_util_critical_section_exit();
|
core_util_critical_section_exit();
|
||||||
}
|
MBED_ERROR(MBED_MAKE_ERROR(MBED_MODULE_DRIVER, MBED_ERROR_CODE_OUT_OF_RESOURCES), "There's already watchdog in the system");
|
||||||
|
}
|
||||||
|
|
||||||
void Watchdog::kick()
|
|
||||||
{
|
|
||||||
MBED_ASSERT(_is_initialized);
|
|
||||||
core_util_critical_section_enter();
|
|
||||||
_current_count = 0;
|
|
||||||
core_util_critical_section_exit();
|
core_util_critical_section_exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Watchdog::stop()
|
|
||||||
{
|
|
||||||
MBED_ASSERT(_is_initialized);
|
|
||||||
core_util_critical_section_enter();
|
|
||||||
remove_from_list();
|
|
||||||
core_util_critical_section_exit();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Watchdog::remove_from_list()
|
|
||||||
{
|
|
||||||
Watchdog *cur_ptr = _first,
|
|
||||||
*prev_ptr = NULL;
|
|
||||||
while (cur_ptr != NULL) {
|
|
||||||
if (cur_ptr == this) {
|
|
||||||
if (cur_ptr == _first) {
|
|
||||||
prev_ptr = _first;
|
|
||||||
_first = cur_ptr->_next;
|
|
||||||
prev_ptr->_next = NULL;
|
|
||||||
} else {
|
|
||||||
prev_ptr->_next = cur_ptr->_next;
|
|
||||||
cur_ptr->_next = NULL;
|
|
||||||
}
|
|
||||||
_is_initialized = false;
|
|
||||||
break;
|
|
||||||
} else {
|
|
||||||
prev_ptr = cur_ptr;
|
|
||||||
cur_ptr = cur_ptr->_next;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Watchdog::process(uint32_t elapsed_ms)
|
|
||||||
{
|
|
||||||
Watchdog *cur_ptr = _first;
|
|
||||||
while (cur_ptr != NULL) {
|
|
||||||
if (cur_ptr->_current_count > cur_ptr->_max_timeout) {
|
|
||||||
system_reset();
|
|
||||||
} else {
|
|
||||||
cur_ptr->_current_count += elapsed_ms;
|
|
||||||
}
|
|
||||||
cur_ptr = cur_ptr->_next;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Watchdog::~Watchdog()
|
Watchdog::~Watchdog()
|
||||||
{
|
{
|
||||||
if (_is_initialized) {
|
core_util_critical_section_enter();
|
||||||
stop();
|
if (_running) {
|
||||||
|
if (stop()) {
|
||||||
|
// some targets might not support stop, thus keep running true in that case
|
||||||
|
_running = false;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
core_util_critical_section_exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Watchdog::start()
|
||||||
|
{
|
||||||
|
watchdog_status_t sts;
|
||||||
|
MBED_ASSERT(MBED_CONF_TARGET_WATCHDOG_TIMEOUT < get_max_timeout());
|
||||||
|
core_util_critical_section_enter();
|
||||||
|
if (_running) {
|
||||||
|
core_util_critical_section_exit();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
watchdog_config_t config;
|
||||||
|
config.timeout_ms = MBED_CONF_TARGET_WATCHDOG_TIMEOUT;
|
||||||
|
sts = hal_watchdog_init(&config);
|
||||||
|
if (sts == WATCHDOG_STATUS_OK) {
|
||||||
|
_running = true;
|
||||||
|
}
|
||||||
|
core_util_critical_section_exit();
|
||||||
|
if (_running) {
|
||||||
|
us_timestamp_t timeout = (MS_TO_US(((elapsed_ms <= 0) ? 1 : elapsed_ms)));
|
||||||
|
_ticker->attach_us(callback(&Watchdog::kick), timeout);
|
||||||
|
}
|
||||||
|
return _running;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Watchdog::stop()
|
||||||
|
{
|
||||||
|
watchdog_status_t sts;
|
||||||
|
bool msts = true;
|
||||||
|
core_util_critical_section_enter();
|
||||||
|
if (_running) {
|
||||||
|
sts = hal_watchdog_stop();
|
||||||
|
if (sts != WATCHDOG_STATUS_OK) {
|
||||||
|
msts = false;
|
||||||
|
} else {
|
||||||
|
_ticker->detach();
|
||||||
|
_running = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
msts = false;
|
||||||
|
}
|
||||||
|
core_util_critical_section_exit();
|
||||||
|
return msts;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Watchdog::kick()
|
||||||
|
{
|
||||||
|
core_util_critical_section_enter();
|
||||||
|
hal_watchdog_kick();
|
||||||
|
// VirtualWatchdog will access the watchdog process method to verify
|
||||||
|
// all registered users/threads in alive state */
|
||||||
|
VirtualWatchdog::process(((elapsed_ms <= 0) ? 1 : elapsed_ms));
|
||||||
|
core_util_critical_section_exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Watchdog::is_running() const
|
||||||
|
{
|
||||||
|
return _running;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t Watchdog::get_timeout()
|
||||||
|
{
|
||||||
|
return hal_watchdog_get_reload_value();
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t Watchdog::get_max_timeout()
|
||||||
|
{
|
||||||
|
const watchdog_features_t features = hal_watchdog_get_platform_features();
|
||||||
|
return features.max_timeout;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace mbed
|
} // namespace mbed
|
||||||
|
|
|
@ -20,98 +20,95 @@
|
||||||
|
|
||||||
#ifdef DEVICE_WATCHDOG
|
#ifdef DEVICE_WATCHDOG
|
||||||
|
|
||||||
#include <cstdio>
|
|
||||||
#include "mbed_error.h"
|
#include "mbed_error.h"
|
||||||
#include "platform/mbed_critical.h"
|
|
||||||
#include "platform/mbed_power_mgmt.h"
|
|
||||||
#include "mbed_assert.h"
|
#include "mbed_assert.h"
|
||||||
|
#include "platform/mbed_critical.h"
|
||||||
|
#include "watchdog_api.h"
|
||||||
|
#include <cstdio>
|
||||||
|
|
||||||
namespace mbed {
|
namespace mbed {
|
||||||
|
|
||||||
/** \addtogroup drivers */
|
/** \addtogroup drivers */
|
||||||
/** Services that need to monitor nonblocking periodic behavior or
|
/** Hardware system timer that will reset the system in the case of system failures or
|
||||||
* malfunctions, such as Wi-Fi and TLS, use a software watchdog.
|
* malfunctions.
|
||||||
* Normally, one instance of a software watchdog gets created during hardware watchdog startup, so you can access the software watchdog.
|
|
||||||
* The hardware watchdog periodically calls the "process" method of the services to check nonblocking periodic behavior.
|
|
||||||
*
|
*
|
||||||
* Example:
|
* Example:
|
||||||
* @code
|
* @code
|
||||||
*
|
*
|
||||||
* Watchdog watchdog(300,"Software Watchdog");
|
* Watchdog watchdog();
|
||||||
* watchdog.start();
|
* watchdog.start();
|
||||||
*
|
*
|
||||||
* while (true) {
|
* while (true) {
|
||||||
* watchdog.kick();
|
|
||||||
*
|
|
||||||
* // Application code
|
* // Application code
|
||||||
* }
|
* }
|
||||||
* @endcode
|
* @endcode
|
||||||
* @ingroup drivers
|
* @ingroup drivers
|
||||||
*/
|
*/
|
||||||
class Watchdog {
|
class Watchdog : private NonCopyable<Watchdog> {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/** Constructor configured with timeout and name for this software watchdog instance.
|
/** Constructor configured with timeout
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
Watchdog(uint32_t timeout = 1, const char *const str = NULL);
|
Watchdog();
|
||||||
~Watchdog();
|
~Watchdog();
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/** Start an independent watchdog timer with specified parameters.
|
/** Start the watchdog timer
|
||||||
*
|
*
|
||||||
* Assert for multiple calls of start.
|
*
|
||||||
|
* @return status true if the watchdog timer was started
|
||||||
|
* successfully. assert if one of the input parameters is out of range for the current platform.
|
||||||
|
* false if watchdog timer was not started
|
||||||
*/
|
*/
|
||||||
void start();
|
bool start();
|
||||||
|
|
||||||
/** This stops the watchdog timer.
|
/** Stops the watchdog timer
|
||||||
*
|
*
|
||||||
* Calling this function disables any running
|
* Calling this function will attempt to disable any currently running
|
||||||
* watchdog timers if the platform supports them.
|
* watchdog timers if supported by the current platform.
|
||||||
*
|
*
|
||||||
* Assert without calling start.
|
* @return Returns true if the watchdog timer was successfully
|
||||||
|
* stopped, Returns false if the watchdog cannot be disabled
|
||||||
|
* on the current platform or if the timer was never started.
|
||||||
*/
|
*/
|
||||||
void stop();
|
bool stop();
|
||||||
|
|
||||||
/** This function refreshes the watchdog timer.
|
/** Get the watchdog timer refresh value
|
||||||
*
|
*
|
||||||
* Call this function periodically before the watchdog times out.
|
* This function returns the refresh timeout of the watchdog timer.
|
||||||
* Otherwise, the system resets.
|
|
||||||
*
|
*
|
||||||
* If the watchdog timer is not running, this function does nothing.
|
* @return Reload value for the watchdog timer in milliseconds.
|
||||||
*/
|
*/
|
||||||
void kick();
|
uint32_t get_timeout() const;
|
||||||
|
|
||||||
/** mbed_watchdog_manager (runs by periodic call from ticker) used this API interface
|
/** Get the maximum refresh value for the current platform in milliseconds
|
||||||
* to go through all the registered user/threads of watchdog.
|
|
||||||
*
|
*
|
||||||
* @param elapsed_ms completed ticker callback elapsed milliseconds
|
* @return Maximum refresh value supported by the watchdog for the current
|
||||||
*
|
* platform in milliseconds
|
||||||
* Call this function from mbed_watchdog_manager_kick to monitor all the
|
|
||||||
* user/threads alive states.
|
|
||||||
*
|
|
||||||
* Otherwise, the system resets.
|
|
||||||
*/
|
*/
|
||||||
static void process(uint32_t elapsed_ms);
|
uint32_t get_max_timeout() const;
|
||||||
protected :
|
|
||||||
|
|
||||||
/** Use add_to_list to store the registered user in the list.
|
/** Check if watchdog is already running
|
||||||
* This API is only used to call from start.
|
|
||||||
*/
|
|
||||||
void add_to_list();
|
|
||||||
|
|
||||||
/** Use remove_from_list to remove the entry from the list.
|
|
||||||
* This API is only used to call from stop.
|
|
||||||
*
|
*
|
||||||
|
* @return Maximum refresh value supported by the watchdog for the current
|
||||||
|
* platform in milliseconds
|
||||||
*/
|
*/
|
||||||
void remove_from_list();
|
bool is_running() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
uint32_t _max_timeout; //_max_timeout initialized via constructor while creating instance of this class
|
static void kick();
|
||||||
const char *_name; //To store the details of user
|
static uint32_t _elapsed_ms;
|
||||||
uint32_t _current_count; //this parameter is used to reset everytime threads/user calls kick
|
static bool _running;
|
||||||
bool _is_initialized; //To control start and stop functionality
|
#if DEVICE_LPTICKER
|
||||||
static Watchdog *_first; //List to store the user/threads who called start
|
/** Create singleton instance of LowPowerTicker for watchdog periodic call back of kick.
|
||||||
Watchdog *_next;
|
*/
|
||||||
|
static SingletonPtr<LowPowerTicker> _ticker;
|
||||||
|
#else
|
||||||
|
/** Create singleton instance of Ticker for watchdog periodic call back of kick.
|
||||||
|
*/
|
||||||
|
static SingletonPtr<Ticker> _ticker;
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace mbed
|
} // namespace mbed
|
||||||
|
|
|
@ -48,10 +48,9 @@
|
||||||
"help": "Run tickless from the microsecond ticker rather than the low power ticker. Running tickless off of the microsecond ticker improves interrupt latency on targets which use lpticker_delay_ticks",
|
"help": "Run tickless from the microsecond ticker rather than the low power ticker. Running tickless off of the microsecond ticker improves interrupt latency on targets which use lpticker_delay_ticks",
|
||||||
"value": false
|
"value": false
|
||||||
},
|
},
|
||||||
"hw-watchdog_timeout": {
|
"watchdog-timeout": {
|
||||||
"help": "Define the timeout in ms value LowPowerTicker to do HW kick",
|
"help": "Define the timeout in ms to do HW kick",
|
||||||
"value": "800",
|
"value": "800"
|
||||||
"macro_name": "HW_WATCHDOG_TIMEOUT"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in New Issue