2019-05-10 15:28:44 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 ML!PA Consulting GmbH
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cpu_sam0_common
|
|
|
|
* @ingroup drivers_periph_wdt
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file wdt.c
|
|
|
|
* @brief Low-level WDT driver implementation
|
|
|
|
*
|
|
|
|
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "periph/wdt.h"
|
|
|
|
#include "pm_layered.h"
|
|
|
|
#include "board.h"
|
|
|
|
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#ifndef WDT_CLOCK_HZ
|
|
|
|
#define WDT_CLOCK_HZ 1024
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* work around inconsistency in header files */
|
|
|
|
#ifndef WDT_CONFIG_PER_8_Val
|
|
|
|
#define WDT_CONFIG_PER_8_Val WDT_CONFIG_PER_CYC8_Val
|
|
|
|
#endif
|
|
|
|
#ifndef WDT_CONFIG_PER_8K_Val
|
|
|
|
#define WDT_CONFIG_PER_8K_Val WDT_CONFIG_PER_CYC8192_Val
|
|
|
|
#endif
|
|
|
|
#ifndef WDT_CONFIG_PER_16K_Val
|
|
|
|
#define WDT_CONFIG_PER_16K_Val WDT_CONFIG_PER_CYC16384_Val
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static inline void _set_enable(bool on)
|
|
|
|
{
|
|
|
|
/* work around strange watchdog behaviour if IDLE2 is used on samd21 */
|
|
|
|
#ifdef CPU_FAM_SAMD21
|
|
|
|
if (on) {
|
|
|
|
pm_block(1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WDT_CTRLA_ENABLE
|
|
|
|
WDT->CTRLA.bit.ENABLE = on;
|
|
|
|
#else
|
|
|
|
WDT->CTRL.bit.ENABLE = on;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _wait_syncbusy(void)
|
|
|
|
{
|
|
|
|
#ifdef WDT_STATUS_SYNCBUSY
|
|
|
|
while (WDT->STATUS.bit.SYNCBUSY) {}
|
|
|
|
#else
|
|
|
|
while (WDT->SYNCBUSY.reg) {}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t ms_to_per(uint32_t ms)
|
|
|
|
{
|
|
|
|
const uint32_t cycles = (ms * WDT_CLOCK_HZ) / 1024;
|
|
|
|
|
|
|
|
/* Minimum WDT period is 8 clock cycles (register value 0) */
|
|
|
|
if (cycles <= 8) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Round up to next pow2 and calculate the register value */
|
|
|
|
return 29 - __builtin_clz(cycles - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CPU_SAMD21
|
|
|
|
static void _wdt_clock_setup(void)
|
|
|
|
{
|
2019-12-16 19:41:16 +01:00
|
|
|
/* Connect to GCLK3 (~1.024 kHz) */
|
2019-05-10 15:28:44 +02:00
|
|
|
GCLK->CLKCTRL.reg = GCLK_CLKCTRL_ID_WDT
|
2019-12-16 19:41:16 +01:00
|
|
|
| GCLK_CLKCTRL_GEN(SAM0_GCLK_1KHZ)
|
2019-05-10 15:28:44 +02:00
|
|
|
| GCLK_CLKCTRL_CLKEN;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static void _wdt_clock_setup(void)
|
|
|
|
{
|
|
|
|
/* nothing to do here */
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void wdt_init(void)
|
|
|
|
{
|
|
|
|
_wdt_clock_setup();
|
|
|
|
#ifdef MCLK
|
|
|
|
MCLK->APBAMASK.bit.WDT_ = 1;
|
|
|
|
#else
|
|
|
|
PM->APBAMASK.bit.WDT_ = 1;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
_set_enable(0);
|
|
|
|
NVIC_EnableIRQ(WDT_IRQn);
|
|
|
|
}
|
|
|
|
|
|
|
|
void wdt_setup_reboot(uint32_t min_time, uint32_t max_time)
|
|
|
|
{
|
|
|
|
uint32_t per, win;
|
|
|
|
|
|
|
|
if (max_time == 0) {
|
|
|
|
DEBUG("invalid period: max_time = %"PRIu32"\n", max_time);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
per = ms_to_per(max_time);
|
|
|
|
|
|
|
|
if (per > WDT_CONFIG_PER_16K_Val) {
|
|
|
|
DEBUG("invalid period: max_time = %"PRIu32"\n", max_time);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (min_time) {
|
|
|
|
win = ms_to_per(min_time);
|
|
|
|
|
|
|
|
if (win > WDT_CONFIG_PER_8K_Val) {
|
|
|
|
DEBUG("invalid period: min_time = %"PRIu32"\n", min_time);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (per < win) {
|
|
|
|
per = win + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef WDT_CTRLA_WEN
|
|
|
|
WDT->CTRLA.bit.WEN = 1;
|
|
|
|
#else
|
|
|
|
WDT->CTRL.bit.WEN = 1;
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
win = 0;
|
|
|
|
#ifdef WDT_CTRLA_WEN
|
|
|
|
WDT->CTRLA.bit.WEN = 0;
|
|
|
|
#else
|
|
|
|
WDT->CTRL.bit.WEN = 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
WDT->INTFLAG.reg = WDT_INTFLAG_EW;
|
|
|
|
|
|
|
|
DEBUG("watchdog window: %"PRIu32", period: %"PRIu32"\n", win, per);
|
|
|
|
|
|
|
|
WDT->CONFIG.reg = WDT_CONFIG_WINDOW(win) | WDT_CONFIG_PER(per);
|
|
|
|
_wait_syncbusy();
|
|
|
|
}
|
|
|
|
|
|
|
|
void wdt_stop(void)
|
|
|
|
{
|
|
|
|
_set_enable(0);
|
|
|
|
_wait_syncbusy();
|
|
|
|
}
|
|
|
|
|
|
|
|
void wdt_start(void)
|
|
|
|
{
|
|
|
|
_set_enable(1);
|
|
|
|
_wait_syncbusy();
|
|
|
|
}
|
|
|
|
|
|
|
|
void wdt_kick(void)
|
|
|
|
{
|
|
|
|
WDT->CLEAR.reg = WDT_CLEAR_CLEAR_KEY_Val;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_PERIPH_WDT_CB
|
|
|
|
static wdt_cb_t cb;
|
|
|
|
static void* cb_arg;
|
|
|
|
|
|
|
|
void wdt_setup_reboot_with_callback(uint32_t min_time, uint32_t max_time,
|
|
|
|
wdt_cb_t wdt_cb, void *arg)
|
|
|
|
{
|
|
|
|
uint32_t per = ms_to_per(max_time);
|
|
|
|
|
|
|
|
if (per == WDT_CONFIG_PER_8_Val && wdt_cb) {
|
|
|
|
DEBUG("period too short for early warning\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cb = wdt_cb;
|
|
|
|
cb_arg = arg;
|
|
|
|
|
|
|
|
if (cb != NULL) {
|
|
|
|
uint32_t warning_offset = ms_to_per(WDT_WARNING_PERIOD);
|
|
|
|
|
|
|
|
if (warning_offset == 0) {
|
|
|
|
warning_offset = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (warning_offset >= per) {
|
|
|
|
warning_offset = per - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
WDT->INTENSET.reg = WDT_INTENSET_EW;
|
|
|
|
WDT->EWCTRL.bit.EWOFFSET = per - warning_offset;
|
|
|
|
} else {
|
|
|
|
WDT->INTENCLR.reg = WDT_INTENCLR_EW;
|
|
|
|
}
|
|
|
|
|
|
|
|
wdt_setup_reboot(min_time, max_time);
|
|
|
|
}
|
|
|
|
|
|
|
|
void isr_wdt(void)
|
|
|
|
{
|
|
|
|
WDT->INTFLAG.reg = WDT_INTFLAG_EW;
|
|
|
|
|
|
|
|
if (cb != NULL) {
|
|
|
|
cb(cb_arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
cortexm_isr_end();
|
|
|
|
}
|
|
|
|
#endif /* MODULE_PERIPH_WDT_CB */
|