nrf52_wdt: split into low-level API and watchdog driver lower-half
This separation allows to interact with the watchdog from OS code, for example initiating the watchdog very early on boot. Moreover, these changes make the lower-half driver support an already running watchdog, which may happen if there's a bootloader which already started it.
This commit is contained in:
parent
d6686b7409
commit
4cbfbd0c74
@ -115,6 +115,10 @@ endif
|
|||||||
|
|
||||||
ifeq ($(CONFIG_NRF52_WDT),y)
|
ifeq ($(CONFIG_NRF52_WDT),y)
|
||||||
CHIP_CSRCS += nrf52_wdt.c
|
CHIP_CSRCS += nrf52_wdt.c
|
||||||
|
|
||||||
|
ifeq ($(CONFIG_WATCHDOG),y)
|
||||||
|
CHIP_CSRCS += nrf52_wdt_lowerhalf.c
|
||||||
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_NRF52_RNG),y)
|
ifeq ($(CONFIG_NRF52_RNG),y)
|
||||||
@ -154,4 +158,4 @@ endif
|
|||||||
|
|
||||||
ifeq ($(CONFIG_NRF52_SAADC),y)
|
ifeq ($(CONFIG_NRF52_SAADC),y)
|
||||||
CHIP_CSRCS += nrf52_adc.c
|
CHIP_CSRCS += nrf52_adc.c
|
||||||
endif
|
endif
|
||||||
|
@ -106,7 +106,7 @@
|
|||||||
/* CONFIG Register */
|
/* CONFIG Register */
|
||||||
|
|
||||||
#define WDT_CONFIG_SLEEP (1 << 0) /* Bit 0: */
|
#define WDT_CONFIG_SLEEP (1 << 0) /* Bit 0: */
|
||||||
#define WDT_CONFIG_HALT (1 << 0) /* Bit 3: */
|
#define WDT_CONFIG_HALT (1 << 3) /* Bit 3: */
|
||||||
|
|
||||||
/* RR[x] Register */
|
/* RR[x] Register */
|
||||||
|
|
||||||
|
@ -39,16 +39,6 @@
|
|||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#include <nuttx/config.h>
|
#include <nuttx/config.h>
|
||||||
#include <nuttx/arch.h>
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <debug.h>
|
|
||||||
|
|
||||||
#include <nuttx/irq.h>
|
|
||||||
#include <nuttx/clock.h>
|
|
||||||
#include <nuttx/timers/watchdog.h>
|
|
||||||
#include <arch/board/board.h>
|
|
||||||
|
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
#include "arm_arch.h"
|
#include "arm_arch.h"
|
||||||
@ -57,106 +47,30 @@
|
|||||||
#include "hardware/nrf52_wdt.h"
|
#include "hardware/nrf52_wdt.h"
|
||||||
#include "nrf52_wdt.h"
|
#include "nrf52_wdt.h"
|
||||||
|
|
||||||
#if defined(CONFIG_WATCHDOG) && defined(CONFIG_NRF52_WDT)
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Pre-processor Definitions
|
* Pre-processor Definitions
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/* Clocking *****************************************************************/
|
|
||||||
|
|
||||||
/* The watchdog is always running under 32.768KHz:
|
|
||||||
*
|
|
||||||
* Fmin = Flsi / 32768
|
|
||||||
*
|
|
||||||
* So the maximum delay (in milliseconds) is then:
|
|
||||||
*
|
|
||||||
* 1000 * WDT_MAX_VALUE / 32768
|
|
||||||
*
|
|
||||||
* For example, if Flsi = 30Khz (the nominal, uncalibrated value), then the
|
|
||||||
* maximum delay is:
|
|
||||||
*
|
|
||||||
* Fmin = 0.03 ms
|
|
||||||
* Fmax = 131702 Sec
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define WDT_FMIN (1)
|
|
||||||
#define WDT_MAXTIMEOUT (1000 * (UINT32_MAX / 32768))
|
|
||||||
|
|
||||||
/* Configuration ************************************************************/
|
|
||||||
|
|
||||||
/* There are the options:
|
|
||||||
*
|
|
||||||
* WDT_CONFIG_MODE - WDT behavior in CPU SLEEP or HALT mode
|
|
||||||
* <1=> Run in SLEEP, Pause in HALT
|
|
||||||
* <8=> Pause in SLEEP, Run in HALT
|
|
||||||
* <9=> Run in SLEEP and HALT
|
|
||||||
* <0=> Pause in SLEEP and HALT
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define NRF_WDT_BEHAVIOUR_PAUSE_SLEEP_HALT 0
|
|
||||||
#define WDT_CONFIG_MODE NRF_WDT_BEHAVIOUR_PAUSE_SLEEP_HALT
|
|
||||||
|
|
||||||
#define WDT_CONFIG_IRQ_PRIORITY 7
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Private Types
|
* Private Types
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/* This structure provides the private representation of the "lower-half"
|
|
||||||
* driver state structure. This structure must be cast-compatible with the
|
|
||||||
* well-known watchdog_lowerhalf_s structure.
|
|
||||||
*/
|
|
||||||
|
|
||||||
struct nrf52_wdt_lowerhalf_s
|
|
||||||
{
|
|
||||||
FAR const struct watchdog_ops_s *ops; /* Lower half operations */
|
|
||||||
uint32_t timeout; /* The (actual) timeout */
|
|
||||||
uint32_t lastreset; /* The last reset time */
|
|
||||||
bool started; /* true: The watchdog timer has been started */
|
|
||||||
uint16_t reload; /* Timer reload value */
|
|
||||||
uint16_t mode; /* watchdog mode under sleep and halt of CPU */
|
|
||||||
};
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Private Function Prototypes
|
* Private Function Prototypes
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/* "Lower half" driver methods **********************************************/
|
|
||||||
|
|
||||||
static int nrf52_start(FAR struct watchdog_lowerhalf_s *lower);
|
|
||||||
static int nrf52_stop(FAR struct watchdog_lowerhalf_s *lower);
|
|
||||||
static int nrf52_keepalive(FAR struct watchdog_lowerhalf_s *lower);
|
|
||||||
static int nrf52_getstatus(FAR struct watchdog_lowerhalf_s *lower,
|
|
||||||
FAR struct watchdog_status_s *status);
|
|
||||||
static int nrf52_settimeout(FAR struct watchdog_lowerhalf_s *lower,
|
|
||||||
uint32_t timeout);
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Private Data
|
* Private Data
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/* "Lower half" driver methods */
|
|
||||||
|
|
||||||
static const struct watchdog_ops_s g_wdtops =
|
|
||||||
{
|
|
||||||
.start = nrf52_start,
|
|
||||||
.stop = nrf52_stop,
|
|
||||||
.keepalive = nrf52_keepalive,
|
|
||||||
.getstatus = nrf52_getstatus,
|
|
||||||
.settimeout = nrf52_settimeout,
|
|
||||||
.capture = NULL,
|
|
||||||
.ioctl = NULL,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* "Lower half" driver state */
|
|
||||||
|
|
||||||
static struct nrf52_wdt_lowerhalf_s g_wdtdev;
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Private Functions
|
* Private Functions
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Public Functions
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: nrf52_wdt_int_enable
|
* Name: nrf52_wdt_int_enable
|
||||||
*
|
*
|
||||||
@ -171,7 +85,7 @@ static struct nrf52_wdt_lowerhalf_s g_wdtdev;
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void nrf52_wdt_int_enable(uint32_t int_mask)
|
void nrf52_wdt_int_enable(uint32_t int_mask)
|
||||||
{
|
{
|
||||||
putreg32(int_mask, NRF52_WDT_INTENSET);
|
putreg32(int_mask, NRF52_WDT_INTENSET);
|
||||||
}
|
}
|
||||||
@ -190,11 +104,33 @@ static void nrf52_wdt_int_enable(uint32_t int_mask)
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void nrf52_wdt_task_trigger(void)
|
void nrf52_wdt_task_trigger(void)
|
||||||
{
|
{
|
||||||
putreg32(1, NRF52_WDT_TASKS_START);
|
putreg32(1, NRF52_WDT_TASKS_START);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_reload_register_enable
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Enable/disable a given reload register
|
||||||
|
*
|
||||||
|
* Input Parameter:
|
||||||
|
* rr_register - Reload request register
|
||||||
|
* enable - true to enable, false to disable
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void nrf52_wdt_reload_register_enable(int rr_register, bool enable)
|
||||||
|
{
|
||||||
|
modifyreg32(NRF52_WDT_RREN,
|
||||||
|
enable ? 0 : WDT_RREN_RR(rr_register),
|
||||||
|
enable ? WDT_RREN_RR(rr_register) : 0);
|
||||||
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: nrf52_wdt_reload_request_set
|
* Name: nrf52_wdt_reload_request_set
|
||||||
*
|
*
|
||||||
@ -209,7 +145,7 @@ static void nrf52_wdt_task_trigger(void)
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void nrf52_wdt_reload_request_set(int rr_register)
|
void nrf52_wdt_reload_request_set(int rr_register)
|
||||||
{
|
{
|
||||||
/* Each register is 32-bit (4 bytes), then multiply by 4 to get offset */
|
/* Each register is 32-bit (4 bytes), then multiply by 4 to get offset */
|
||||||
|
|
||||||
@ -230,7 +166,7 @@ static void nrf52_wdt_reload_request_set(int rr_register)
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void nrf52_wdt_behaviour_set(int behaviour)
|
void nrf52_wdt_behaviour_set(int behaviour)
|
||||||
{
|
{
|
||||||
putreg32(behaviour, NRF52_WDT_CONFIG);
|
putreg32(behaviour, NRF52_WDT_CONFIG);
|
||||||
}
|
}
|
||||||
@ -249,281 +185,39 @@ static void nrf52_wdt_behaviour_set(int behaviour)
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void nrf52_wdt_reload_value_set(uint32_t reload_value)
|
void nrf52_wdt_reload_value_set(uint32_t reload_value)
|
||||||
{
|
{
|
||||||
putreg32(reload_value, NRF52_WDT_CRV);
|
putreg32(reload_value, NRF52_WDT_CRV);
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: nrf52_start
|
* Name: nrf52_wdt_reload_value_get
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Start the watchdog timer, resetting the time to the current timeout,
|
* Read the watchdog reload value
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* lower - A pointer the publicly visible representation of the
|
|
||||||
* "lower-half" driver state structure.
|
|
||||||
*
|
*
|
||||||
* Returned Values:
|
* Returned Values:
|
||||||
* Zero on success; a negated errno value on failure.
|
* Watchdog counter initial value
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static int nrf52_start(FAR struct watchdog_lowerhalf_s *lower)
|
uint32_t nrf52_wdt_reload_value_get(void)
|
||||||
{
|
{
|
||||||
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
return getreg32(NRF52_WDT_CRV);
|
||||||
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
|
||||||
irqstate_t flags;
|
|
||||||
|
|
||||||
wdinfo("Entry: started=%d\n");
|
|
||||||
DEBUGASSERT(priv);
|
|
||||||
|
|
||||||
/* Have we already been started? */
|
|
||||||
|
|
||||||
if (!priv->started)
|
|
||||||
{
|
|
||||||
flags = enter_critical_section();
|
|
||||||
priv->lastreset = clock_systime_ticks();
|
|
||||||
priv->started = true;
|
|
||||||
nrf52_wdt_int_enable(WDT_INT_TIMEOUT);
|
|
||||||
nrf52_wdt_task_trigger();
|
|
||||||
leave_critical_section(flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
return OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: nrf52_stop
|
* Name: nrf52_wdt_running
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Stop the watchdog timer
|
* Check if watchdog is running
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* lower - A pointer the publicly visible representation of the
|
|
||||||
* "lower-half" driver state structure.
|
|
||||||
*
|
*
|
||||||
* Returned Values:
|
* Returned Values:
|
||||||
* Zero on success; a negated errno value on failure.
|
* true if watchdog is running, false otherwise
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static int nrf52_stop(FAR struct watchdog_lowerhalf_s *lower)
|
bool nrf52_wdt_running(void)
|
||||||
{
|
{
|
||||||
/* There is no way to disable the WDT timer once it has been started */
|
return getreg32(NRF52_WDT_RUNSTATUS);
|
||||||
|
|
||||||
wdinfo("Entry\n");
|
|
||||||
return -ENOSYS;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: nrf52_keepalive
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Reset the watchdog timer to the current timeout value, prevent any
|
|
||||||
* imminent watchdog timeouts. This is sometimes referred as "pinging"
|
|
||||||
* the watchdog timer or "petting the dog".
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* lower - A pointer the publicly visible representation of the
|
|
||||||
* "lower-half" driver state structure.
|
|
||||||
*
|
|
||||||
* Returned Values:
|
|
||||||
* Zero on success; a negated errno value on failure.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
static int nrf52_keepalive(FAR struct watchdog_lowerhalf_s *lower)
|
|
||||||
{
|
|
||||||
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
|
||||||
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
|
||||||
irqstate_t flags;
|
|
||||||
|
|
||||||
wdinfo("Entry\n");
|
|
||||||
|
|
||||||
/* Reload the WDT timer */
|
|
||||||
|
|
||||||
flags = enter_critical_section();
|
|
||||||
|
|
||||||
priv->lastreset = clock_systime_ticks();
|
|
||||||
nrf52_wdt_reload_request_set(0);
|
|
||||||
|
|
||||||
leave_critical_section(flags);
|
|
||||||
|
|
||||||
return OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: nrf52_getstatus
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Get the current watchdog timer status
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* lower - A pointer the publicly visible representation of
|
|
||||||
* the "lower-half" driver state structure.
|
|
||||||
* status - The location to return the watchdog status information.
|
|
||||||
*
|
|
||||||
* Returned Values:
|
|
||||||
* Zero on success; a negated errno value on failure.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
static int nrf52_getstatus(FAR struct watchdog_lowerhalf_s *lower,
|
|
||||||
FAR struct watchdog_status_s *status)
|
|
||||||
{
|
|
||||||
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
|
||||||
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
|
||||||
uint32_t ticks;
|
|
||||||
uint32_t elapsed;
|
|
||||||
|
|
||||||
wdinfo("Entry\n");
|
|
||||||
DEBUGASSERT(priv);
|
|
||||||
|
|
||||||
/* Return the status bit */
|
|
||||||
|
|
||||||
status->flags = WDFLAGS_RESET;
|
|
||||||
if (priv->started)
|
|
||||||
{
|
|
||||||
status->flags |= WDFLAGS_ACTIVE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Return the actual timeout in milliseconds */
|
|
||||||
|
|
||||||
status->timeout = priv->timeout;
|
|
||||||
|
|
||||||
/* Get the elapsed time since the last ping */
|
|
||||||
|
|
||||||
ticks = clock_systime_ticks() - priv->lastreset;
|
|
||||||
elapsed = (int32_t)TICK2MSEC(ticks);
|
|
||||||
|
|
||||||
if (elapsed > priv->timeout)
|
|
||||||
{
|
|
||||||
elapsed = priv->timeout;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Return the approximate time until the watchdog timer expiration */
|
|
||||||
|
|
||||||
status->timeleft = priv->timeout - elapsed;
|
|
||||||
|
|
||||||
wdinfo("Status :\n");
|
|
||||||
wdinfo(" flags : %08x\n", status->flags);
|
|
||||||
wdinfo(" timeout : %d\n", status->timeout);
|
|
||||||
wdinfo(" timeleft : %d\n", status->timeleft);
|
|
||||||
return OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: nrf52_settimeout
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Set a new timeout value (and reset the watchdog timer)
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* lower - A pointer the publicly visible representation of
|
|
||||||
* the "lower-half" driver state structure.
|
|
||||||
* timeout - The new timeout value in milliseconds.
|
|
||||||
*
|
|
||||||
* Returned Values:
|
|
||||||
* Zero on success; a negated errno value on failure.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
static int nrf52_settimeout(FAR struct watchdog_lowerhalf_s *lower,
|
|
||||||
uint32_t timeout)
|
|
||||||
{
|
|
||||||
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
|
||||||
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
|
||||||
|
|
||||||
wdinfo("Entry: timeout=%d\n", timeout);
|
|
||||||
DEBUGASSERT(priv);
|
|
||||||
|
|
||||||
/* Can this timeout be represented? */
|
|
||||||
|
|
||||||
if (timeout < 1 || timeout > WDT_MAXTIMEOUT)
|
|
||||||
{
|
|
||||||
wderr("ERROR: Cannot represent timeout=%d > %d\n",
|
|
||||||
timeout, WDT_MAXTIMEOUT);
|
|
||||||
return -ERANGE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->started)
|
|
||||||
{
|
|
||||||
wdwarn("WARNING: Watchdog is already started\n");
|
|
||||||
return -EBUSY;
|
|
||||||
}
|
|
||||||
|
|
||||||
nrf52_wdt_behaviour_set(priv->mode);
|
|
||||||
priv->timeout = timeout;
|
|
||||||
|
|
||||||
nrf52_wdt_reload_value_set(((uint64_t) timeout * 32768) / 1000);
|
|
||||||
|
|
||||||
up_enable_irq(NRF52_IRQ_WDT);
|
|
||||||
|
|
||||||
wdinfo(" mode=%d priority=%d\n", WDT_CONFIG_MODE,
|
|
||||||
WDT_CONFIG_IRQ_PRIORITY);
|
|
||||||
|
|
||||||
return OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Public Functions
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: nrf52_wdt_initialize
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Initialize the WDT watchdog time. The watchdog timer is initialized and
|
|
||||||
* registers as 'devpath. The initial state of the watchdog time is
|
|
||||||
* disabled.
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* devpath - The full path to the watchdog. This should be of the form
|
|
||||||
* /dev/watchdog0
|
|
||||||
* mode_sleep - The behaviour of watchdog when CPU enter sleep mode
|
|
||||||
* mode_halt - The behaviour of watchdog when CPU was HALT by
|
|
||||||
* debugger
|
|
||||||
*
|
|
||||||
* Returned Values:
|
|
||||||
* Zero (OK) is returned on success; a negated errno value is returned on
|
|
||||||
* any failure.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
int nrf52_wdt_initialize(FAR const char *devpath, int16_t mode_sleep,
|
|
||||||
int16_t mode_halt)
|
|
||||||
{
|
|
||||||
FAR struct nrf52_wdt_lowerhalf_s *priv = &g_wdtdev;
|
|
||||||
FAR void *handle;
|
|
||||||
|
|
||||||
wdinfo("Entry: devpath=%s, mode_sleep=%d, mode_halt=%d\n", devpath,
|
|
||||||
mode_sleep, mode_halt);
|
|
||||||
|
|
||||||
/* Initialize the driver state structure. */
|
|
||||||
|
|
||||||
priv->ops = &g_wdtops;
|
|
||||||
priv->started = false;
|
|
||||||
priv->mode = (mode_halt << WDT_CONFIG_HALT) |
|
|
||||||
(mode_sleep << WDT_CONFIG_SLEEP);
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
/* Request LSECLK firstly */
|
|
||||||
|
|
||||||
nrf52_clock_init();
|
|
||||||
|
|
||||||
/* Select the lower power external 32,768Hz (Low-Speed External, LSE)
|
|
||||||
* oscillator as RTC Clock Source and enable the Clock.
|
|
||||||
*/
|
|
||||||
|
|
||||||
nrf52_clock_lsclk_start();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Register the watchdog driver as /dev/watchdog0 */
|
|
||||||
|
|
||||||
handle = watchdog_register(devpath,
|
|
||||||
(FAR struct watchdog_lowerhalf_s *)priv);
|
|
||||||
return (handle != NULL) ? OK : -ENODEV;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /* CONFIG_WATCHDOG && CONFIG_nrf52_IWDT */
|
|
||||||
|
@ -41,62 +41,137 @@
|
|||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#include <nuttx/config.h>
|
#include <nuttx/config.h>
|
||||||
|
#include <stdbool.h>
|
||||||
#ifdef CONFIG_WATCHDOG
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Public Types
|
* Public Types
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
enum wdt_behaviour_e
|
|
||||||
{
|
|
||||||
WDG_PAUSE = 0,
|
|
||||||
WDG_RUN = 1
|
|
||||||
};
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Public Function Prototypes
|
* Public Function Prototypes
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#undef EXTERN
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
#define EXTERN extern "C"
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#else
|
|
||||||
#define EXTERN extern
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: nrf52_wdt_initialize
|
* Name: nrf52_wdt_int_enable
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Initialize the WDT watchdog time. The watchdog timer is initialized and
|
* Enable watchdog interrupt.
|
||||||
* registers as 'devpath. The initial state of the watchdog time is
|
|
||||||
* disabled.
|
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameter:
|
||||||
* devpath - The full path to the watchdog. This should be of the form
|
* int_mask - Interrupt mask
|
||||||
* /dev/watchdog0
|
|
||||||
* mode_sleep - The behaviour of watchdog when CPU enter sleep mode
|
|
||||||
* mode_halt - The behaviour of watchdog when CPU was HALT by
|
|
||||||
* debugger
|
|
||||||
*
|
*
|
||||||
* Returned Values:
|
* Returned Values:
|
||||||
* Zero (OK) is returned on success; a negated errno value is returned on
|
* None
|
||||||
* any failure.
|
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#ifdef CONFIG_NRF52_WDT
|
void nrf52_wdt_int_enable(uint32_t int_mask);
|
||||||
int nrf52_wdt_initialize(FAR const char *devpath, int16_t mode_sleep,
|
|
||||||
int16_t mode_halt);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#undef EXTERN
|
/****************************************************************************
|
||||||
#if defined(__cplusplus)
|
* Name: nrf52_wdt_task_trigger
|
||||||
}
|
*
|
||||||
#endif
|
* Description:
|
||||||
|
* Starts the watchdog
|
||||||
|
*
|
||||||
|
* Input Parameter:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void nrf52_wdt_task_trigger(void);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_reload_register_enable
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Enable/disable a given reload register
|
||||||
|
*
|
||||||
|
* Input Parameter:
|
||||||
|
* rr_register - Reload request register
|
||||||
|
* enable - true to enable, false to disable
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void nrf52_wdt_reload_register_enable(int rr_register, bool enable);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_reload_request_set
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Setting a specific reload request register
|
||||||
|
*
|
||||||
|
* Input Parameter:
|
||||||
|
* rr_register - Reload request register to set
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void nrf52_wdt_reload_request_set(int rr_register);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_behaviour_set
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Configure the watchdog behaviour when the CPU is sleeping or halted
|
||||||
|
*
|
||||||
|
* Input Parameter:
|
||||||
|
* behavior - The behaviour to be configured
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void nrf52_wdt_behaviour_set(int behaviour);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_reload_value_set
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Setup the watchdog reload value
|
||||||
|
*
|
||||||
|
* Input Parameter:
|
||||||
|
* reload_value - Watchdog counter initial value
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void nrf52_wdt_reload_value_set(uint32_t reload_value);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_reload_value_get
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Read the watchdog reload value
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Watchdog counter initial value
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
uint32_t nrf52_wdt_reload_value_get(void);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_running
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Check if watchdog is running
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* true if watchdog is running, false otherwise
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
bool nrf52_wdt_running(void);
|
||||||
|
|
||||||
#endif /* CONFIG_WATCHDOG */
|
|
||||||
#endif /* __ARCH_ARM_SRC_NRF52_NRF52_WDT_H */
|
#endif /* __ARCH_ARM_SRC_NRF52_NRF52_WDT_H */
|
||||||
|
427
arch/arm/src/nrf52/nrf52_wdt_lowerhalf.c
Normal file
427
arch/arm/src/nrf52/nrf52_wdt_lowerhalf.c
Normal file
@ -0,0 +1,427 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
* arch/arm/src/nrf52/nrf52_wdt_lowerhalf.c
|
||||||
|
*
|
||||||
|
* Copyright (C) 2018 Zglue Inc. All rights reserved.
|
||||||
|
* Author: Levin Li <zhiqiang@zglue.com>
|
||||||
|
* Author: Alan Carvalho de Assis <acassis@gmail.com>
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in
|
||||||
|
* the documentation and/or other materials provided with the
|
||||||
|
* distribution.
|
||||||
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
||||||
|
* used to endorse or promote products derived from this software
|
||||||
|
* without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||||
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||||||
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
||||||
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||||||
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||||
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||||||
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Included Files
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include <nuttx/config.h>
|
||||||
|
#include <nuttx/arch.h>
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <debug.h>
|
||||||
|
|
||||||
|
#include <nuttx/irq.h>
|
||||||
|
#include <nuttx/clock.h>
|
||||||
|
#include <nuttx/timers/watchdog.h>
|
||||||
|
#include <arch/board/board.h>
|
||||||
|
|
||||||
|
#include "nrf52_wdt.h"
|
||||||
|
#include "hardware/nrf52_wdt.h"
|
||||||
|
#include "nrf52_wdt_lowerhalf.h"
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Pre-processor Definitions
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/* Clocking *****************************************************************/
|
||||||
|
|
||||||
|
/* The watchdog is always running under 32.768KHz:
|
||||||
|
*
|
||||||
|
* Fmin = Flsi / 32768
|
||||||
|
*
|
||||||
|
* So the maximum delay (in milliseconds) is then:
|
||||||
|
*
|
||||||
|
* 1000 * WDT_MAX_VALUE / 32768
|
||||||
|
*
|
||||||
|
* For example, if Flsi = 30Khz (the nominal, uncalibrated value), then the
|
||||||
|
* maximum delay is:
|
||||||
|
*
|
||||||
|
* Fmin = 0.03 ms
|
||||||
|
* Fmax = 131702 Sec
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define WDT_FMIN (1)
|
||||||
|
#define WDT_MAXTIMEOUT (1000 * (UINT32_MAX / 32768))
|
||||||
|
|
||||||
|
/* Configuration ************************************************************/
|
||||||
|
|
||||||
|
/* There are the options:
|
||||||
|
*
|
||||||
|
* WDT_CONFIG_MODE - WDT behavior in CPU SLEEP or HALT mode
|
||||||
|
* <1=> Run in SLEEP, Pause in HALT
|
||||||
|
* <8=> Pause in SLEEP, Run in HALT
|
||||||
|
* <9=> Run in SLEEP and HALT
|
||||||
|
* <0=> Pause in SLEEP and HALT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define NRF_WDT_BEHAVIOUR_PAUSE_SLEEP_HALT 0
|
||||||
|
#define WDT_CONFIG_MODE NRF_WDT_BEHAVIOUR_PAUSE_SLEEP_HALT
|
||||||
|
|
||||||
|
#define WDT_CONFIG_IRQ_PRIORITY 7
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Private Types
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/* This structure provides the private representation of the "lower-half"
|
||||||
|
* driver state structure. This structure must be cast-compatible with the
|
||||||
|
* well-known watchdog_lowerhalf_s structure.
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct nrf52_wdt_lowerhalf_s
|
||||||
|
{
|
||||||
|
FAR const struct watchdog_ops_s *ops; /* Lower half operations */
|
||||||
|
uint32_t lastreset; /* The last reset time */
|
||||||
|
bool started; /* true: The watchdog timer has been started */
|
||||||
|
uint16_t mode; /* watchdog mode under sleep and halt of CPU */
|
||||||
|
};
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Private Function Prototypes
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/* "Lower half" driver methods **********************************************/
|
||||||
|
|
||||||
|
static int nrf52_start(FAR struct watchdog_lowerhalf_s *lower);
|
||||||
|
static int nrf52_stop(FAR struct watchdog_lowerhalf_s *lower);
|
||||||
|
static int nrf52_keepalive(FAR struct watchdog_lowerhalf_s *lower);
|
||||||
|
static int nrf52_getstatus(FAR struct watchdog_lowerhalf_s *lower,
|
||||||
|
FAR struct watchdog_status_s *status);
|
||||||
|
static int nrf52_settimeout(FAR struct watchdog_lowerhalf_s *lower,
|
||||||
|
uint32_t timeout);
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Private Data
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/* "Lower half" driver methods */
|
||||||
|
|
||||||
|
static const struct watchdog_ops_s g_wdtops =
|
||||||
|
{
|
||||||
|
.start = nrf52_start,
|
||||||
|
.stop = nrf52_stop,
|
||||||
|
.keepalive = nrf52_keepalive,
|
||||||
|
.getstatus = nrf52_getstatus,
|
||||||
|
.settimeout = nrf52_settimeout,
|
||||||
|
.capture = NULL,
|
||||||
|
.ioctl = NULL,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* "Lower half" driver state */
|
||||||
|
|
||||||
|
static struct nrf52_wdt_lowerhalf_s g_wdtdev =
|
||||||
|
{
|
||||||
|
.lastreset = 0,
|
||||||
|
.started = false,
|
||||||
|
};
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Public Data
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Private Functions
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_start
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Start the watchdog timer, resetting the time to the current timeout,
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* lower - A pointer the publicly visible representation of the
|
||||||
|
* "lower-half" driver state structure.
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero on success; a negated errno value on failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static int nrf52_start(FAR struct watchdog_lowerhalf_s *lower)
|
||||||
|
{
|
||||||
|
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
||||||
|
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
||||||
|
irqstate_t flags;
|
||||||
|
|
||||||
|
wdinfo("Entry: started=%d\n");
|
||||||
|
DEBUGASSERT(priv);
|
||||||
|
|
||||||
|
/* Have we already been started? */
|
||||||
|
|
||||||
|
if (!priv->started)
|
||||||
|
{
|
||||||
|
flags = enter_critical_section();
|
||||||
|
priv->lastreset = clock_systime_ticks();
|
||||||
|
priv->started = true;
|
||||||
|
nrf52_wdt_int_enable(WDT_INT_TIMEOUT);
|
||||||
|
nrf52_wdt_task_trigger();
|
||||||
|
leave_critical_section(flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_stop
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Stop the watchdog timer
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* lower - A pointer the publicly visible representation of the
|
||||||
|
* "lower-half" driver state structure.
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero on success; a negated errno value on failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static int nrf52_stop(FAR struct watchdog_lowerhalf_s *lower)
|
||||||
|
{
|
||||||
|
/* There is no way to disable the WDT timer once it has been started */
|
||||||
|
|
||||||
|
wdinfo("Entry\n");
|
||||||
|
return -ENOSYS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_keepalive
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Reset the watchdog timer to the current timeout value, prevent any
|
||||||
|
* imminent watchdog timeouts. This is sometimes referred as "pinging"
|
||||||
|
* the watchdog timer or "petting the dog".
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* lower - A pointer the publicly visible representation of the
|
||||||
|
* "lower-half" driver state structure.
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero on success; a negated errno value on failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static int nrf52_keepalive(FAR struct watchdog_lowerhalf_s *lower)
|
||||||
|
{
|
||||||
|
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
||||||
|
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
||||||
|
irqstate_t flags;
|
||||||
|
|
||||||
|
wdinfo("Entry\n");
|
||||||
|
|
||||||
|
/* Reload the WDT timer */
|
||||||
|
|
||||||
|
flags = enter_critical_section();
|
||||||
|
|
||||||
|
priv->lastreset = clock_systime_ticks();
|
||||||
|
nrf52_wdt_reload_request_set(0);
|
||||||
|
|
||||||
|
leave_critical_section(flags);
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_getstatus
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Get the current watchdog timer status
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* lower - A pointer the publicly visible representation of
|
||||||
|
* the "lower-half" driver state structure.
|
||||||
|
* status - The location to return the watchdog status information.
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero on success; a negated errno value on failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static int nrf52_getstatus(FAR struct watchdog_lowerhalf_s *lower,
|
||||||
|
FAR struct watchdog_status_s *status)
|
||||||
|
{
|
||||||
|
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
||||||
|
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
||||||
|
uint32_t ticks;
|
||||||
|
uint32_t elapsed;
|
||||||
|
|
||||||
|
wdinfo("Entry\n");
|
||||||
|
DEBUGASSERT(priv);
|
||||||
|
|
||||||
|
/* Return the status bit */
|
||||||
|
|
||||||
|
status->flags = WDFLAGS_RESET;
|
||||||
|
if (priv->started)
|
||||||
|
{
|
||||||
|
status->flags |= WDFLAGS_ACTIVE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Return the actual timeout in milliseconds */
|
||||||
|
|
||||||
|
status->timeout = (nrf52_wdt_reload_value_get() / 32768) * 1000;
|
||||||
|
|
||||||
|
/* Get the elapsed time since the last ping */
|
||||||
|
|
||||||
|
ticks = clock_systime_ticks() - priv->lastreset;
|
||||||
|
elapsed = (int32_t)TICK2MSEC(ticks);
|
||||||
|
|
||||||
|
if (elapsed > status->timeout)
|
||||||
|
{
|
||||||
|
elapsed = status->timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Return the approximate time until the watchdog timer expiration */
|
||||||
|
|
||||||
|
status->timeleft = status->timeout - elapsed;
|
||||||
|
|
||||||
|
wdinfo("Status :\n");
|
||||||
|
wdinfo(" flags : %08x\n", status->flags);
|
||||||
|
wdinfo(" timeout : %d\n", status->timeout);
|
||||||
|
wdinfo(" timeleft : %d\n", status->timeleft);
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_settimeout
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Set a new timeout value (and reset the watchdog timer)
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* lower - A pointer the publicly visible representation of
|
||||||
|
* the "lower-half" driver state structure.
|
||||||
|
* timeout - The new timeout value in milliseconds.
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero on success; a negated errno value on failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static int nrf52_settimeout(FAR struct watchdog_lowerhalf_s *lower,
|
||||||
|
uint32_t timeout)
|
||||||
|
{
|
||||||
|
FAR struct nrf52_wdt_lowerhalf_s *priv =
|
||||||
|
(FAR struct nrf52_wdt_lowerhalf_s *)lower;
|
||||||
|
|
||||||
|
wdinfo("Entry: timeout=%d\n", timeout);
|
||||||
|
DEBUGASSERT(priv);
|
||||||
|
|
||||||
|
/* Can this timeout be represented? */
|
||||||
|
|
||||||
|
if (timeout < 1 || timeout > WDT_MAXTIMEOUT)
|
||||||
|
{
|
||||||
|
wderr("ERROR: Cannot represent timeout=%d > %d\n",
|
||||||
|
timeout, WDT_MAXTIMEOUT);
|
||||||
|
return -ERANGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (priv->started)
|
||||||
|
{
|
||||||
|
wdwarn("WARNING: Watchdog is already started\n");
|
||||||
|
return -EBUSY;
|
||||||
|
}
|
||||||
|
|
||||||
|
nrf52_wdt_behaviour_set(priv->mode);
|
||||||
|
|
||||||
|
nrf52_wdt_reload_value_set(((uint64_t) timeout * 32768) / 1000);
|
||||||
|
|
||||||
|
up_enable_irq(NRF52_IRQ_WDT);
|
||||||
|
|
||||||
|
wdinfo(" mode=%d priority=%d\n", WDT_CONFIG_MODE,
|
||||||
|
WDT_CONFIG_IRQ_PRIORITY);
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Public Functions
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_initialize
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Initialize the WDT watchdog time. The watchdog timer is initialized and
|
||||||
|
* registers as 'devpath. The initial state of the watchdog time is
|
||||||
|
* disabled.
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* devpath - The full path to the watchdog. This should be of the form
|
||||||
|
* /dev/watchdog0
|
||||||
|
* mode_sleep - The behaviour of watchdog when CPU enter sleep mode
|
||||||
|
* mode_halt - The behaviour of watchdog when CPU was HALT by
|
||||||
|
* debugger
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero (OK) is returned on success; a negated errno value is returned on
|
||||||
|
* any failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
int nrf52_wdt_initialize(FAR const char *devpath, int16_t mode_sleep,
|
||||||
|
int16_t mode_halt)
|
||||||
|
{
|
||||||
|
FAR struct nrf52_wdt_lowerhalf_s *priv = &g_wdtdev;
|
||||||
|
FAR void *handle;
|
||||||
|
|
||||||
|
wdinfo("Entry: devpath=%s, mode_sleep=%d, mode_halt=%d\n", devpath,
|
||||||
|
mode_sleep, mode_halt);
|
||||||
|
|
||||||
|
/* Initialize the driver state structure. */
|
||||||
|
|
||||||
|
priv->ops = &g_wdtops;
|
||||||
|
priv->started = nrf52_wdt_running();
|
||||||
|
|
||||||
|
if (priv->started)
|
||||||
|
{
|
||||||
|
wdwarn("Watchdog is already running, "
|
||||||
|
"configuration cannot be changed\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This will only take effect if watchdog is not yet running */
|
||||||
|
|
||||||
|
priv->mode = (mode_halt << WDT_CONFIG_HALT) |
|
||||||
|
(mode_sleep << WDT_CONFIG_SLEEP);
|
||||||
|
|
||||||
|
/* Register the watchdog driver as /dev/watchdog0 */
|
||||||
|
|
||||||
|
handle = watchdog_register(devpath,
|
||||||
|
(FAR struct watchdog_lowerhalf_s *)priv);
|
||||||
|
return (handle != NULL) ? OK : -ENODEV;
|
||||||
|
}
|
104
arch/arm/src/nrf52/nrf52_wdt_lowerhalf.h
Normal file
104
arch/arm/src/nrf52/nrf52_wdt_lowerhalf.h
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
* arch/arm/src/nrf52/nrf52_wdt_lowerhalf.c
|
||||||
|
*
|
||||||
|
* Copyright (C) 2018 Zglue Inc. All rights reserved.
|
||||||
|
* Author: Levin Li <zhiqiang@zglue.com>
|
||||||
|
* Author: Alan Carvalho de Assis <acassis@gmail.com>
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in
|
||||||
|
* the documentation and/or other materials provided with the
|
||||||
|
* distribution.
|
||||||
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
||||||
|
* used to endorse or promote products derived from this software
|
||||||
|
* without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||||
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||||||
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
||||||
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||||||
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||||
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||||||
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef __ARCH_ARM_NRF52_WDT_LOWERHALF_H
|
||||||
|
#define __ARCH_ARM_NRF52_WDT_LOWERHALF_H
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Included Files
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include <nuttx/config.h>
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Pre-processor Definitions
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Public Types
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
enum wdt_behaviour_e
|
||||||
|
{
|
||||||
|
WDG_PAUSE = 0,
|
||||||
|
WDG_RUN = 1
|
||||||
|
};
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Public Data
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
#define EXTERN extern "C"
|
||||||
|
extern "C"
|
||||||
|
{
|
||||||
|
#else
|
||||||
|
#define EXTERN extern
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Inline Functions
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Public Function Prototypes
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: nrf52_wdt_initialize
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Initialize the WDT watchdog time. The watchdog timer is initialized and
|
||||||
|
* registers as 'devpath. The initial state of the watchdog time is
|
||||||
|
* disabled.
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* devpath - The full path to the watchdog. This should be of the form
|
||||||
|
* /dev/watchdog0
|
||||||
|
* mode_sleep - The behaviour of watchdog when CPU enter sleep mode
|
||||||
|
* mode_halt - The behaviour of watchdog when CPU was HALT by
|
||||||
|
* debugger
|
||||||
|
*
|
||||||
|
* Returned Values:
|
||||||
|
* Zero (OK) is returned on success; a negated errno value is returned on
|
||||||
|
* any failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
int nrf52_wdt_initialize(FAR const char *devpath, int16_t mode_sleep,
|
||||||
|
int16_t mode_halt);
|
||||||
|
|
||||||
|
#endif /* __ARCH_ARM_NRF52_WDT_LOWERHALF_H */
|
@ -43,7 +43,7 @@
|
|||||||
#include <syslog.h>
|
#include <syslog.h>
|
||||||
|
|
||||||
#ifdef CONFIG_NRF52_WDT
|
#ifdef CONFIG_NRF52_WDT
|
||||||
# include "nrf52_wdt.h"
|
# include "nrf52_wdt_lowerhalf.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_USERLED
|
#ifdef CONFIG_USERLED
|
||||||
|
@ -43,7 +43,7 @@
|
|||||||
#include <syslog.h>
|
#include <syslog.h>
|
||||||
|
|
||||||
#ifdef CONFIG_NRF52_WDT
|
#ifdef CONFIG_NRF52_WDT
|
||||||
# include "nrf52_wdt.h"
|
# include "nrf52_wdt_lowerhalf.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_USERLED
|
#ifdef CONFIG_USERLED
|
||||||
|
@ -28,7 +28,7 @@
|
|||||||
#include <syslog.h>
|
#include <syslog.h>
|
||||||
|
|
||||||
#ifdef CONFIG_NRF52_WDT
|
#ifdef CONFIG_NRF52_WDT
|
||||||
# include "nrf52_wdt.h"
|
# include "nrf52_wdt_lowerhalf.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_USERLED
|
#ifdef CONFIG_USERLED
|
||||||
|
@ -28,7 +28,7 @@
|
|||||||
#include <syslog.h>
|
#include <syslog.h>
|
||||||
|
|
||||||
#ifdef CONFIG_NRF52_WDT
|
#ifdef CONFIG_NRF52_WDT
|
||||||
# include "nrf52_wdt.h"
|
# include "nrf52_wdt_lowerhalf.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_USERLED
|
#ifdef CONFIG_USERLED
|
||||||
|
Loading…
Reference in New Issue
Block a user