2015-11-22 03:07:43 +01:00
|
|
|
/*
|
2017-08-24 17:35:08 +02:00
|
|
|
* Copyright (c) 2017 Nordic Semiconductor ASA
|
|
|
|
* Copyright (c) 2015 Intel Corporation
|
2015-11-22 03:07:43 +01:00
|
|
|
*
|
2017-01-19 02:01:01 +01:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2015-11-22 03:07:43 +01:00
|
|
|
*/
|
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* @brief Public API for watchdog drivers.
|
|
|
|
*/
|
|
|
|
|
2018-09-14 19:43:44 +02:00
|
|
|
#ifndef ZEPHYR_INCLUDE_WATCHDOG_H_
|
|
|
|
#define ZEPHYR_INCLUDE_WATCHDOG_H_
|
2015-11-22 03:07:43 +01:00
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @brief Watchdog Interface
|
|
|
|
* @defgroup watchdog_interface Watchdog Interface
|
2018-12-08 19:12:32 +01:00
|
|
|
* @ingroup io_interfaces
|
2017-08-24 17:35:08 +02:00
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
Introduce new sized integer typedefs
This is a start to move away from the C99 {u}int{8,16,32,64}_t types to
Zephyr defined u{8,16,32,64}_t and s{8,16,32,64}_t. This allows Zephyr
to define the sized types in a consistent manor across all the
architectures we support and not conflict with what various compilers
and libc might do with regards to the C99 types.
We introduce <zephyr/types.h> as part of this and have it include
<stdint.h> for now until we transition all the code away from the C99
types.
We go with u{8,16,32,64}_t and s{8,16,32,64}_t as there are some
existing variables defined u8 & u16 as well as to be consistent with
Zephyr naming conventions.
Jira: ZEP-2051
Change-Id: I451fed0623b029d65866622e478225dfab2c0ca8
Signed-off-by: Kumar Gala <kumar.gala@linaro.org>
2017-04-19 17:32:08 +02:00
|
|
|
#include <zephyr/types.h>
|
2015-12-03 09:43:26 +01:00
|
|
|
#include <misc/util.h>
|
2017-08-24 17:35:08 +02:00
|
|
|
#include <device.h>
|
2015-11-29 18:35:59 +01:00
|
|
|
|
2016-01-22 18:38:49 +01:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2015-11-29 18:35:59 +01:00
|
|
|
/**
|
|
|
|
* WDT clock cycles for timeout type.
|
|
|
|
*/
|
|
|
|
enum wdt_clock_timeout_cycles {
|
|
|
|
WDT_2_16_CYCLES,
|
|
|
|
WDT_2_17_CYCLES,
|
|
|
|
WDT_2_18_CYCLES,
|
|
|
|
WDT_2_19_CYCLES,
|
|
|
|
WDT_2_20_CYCLES,
|
|
|
|
WDT_2_21_CYCLES,
|
|
|
|
WDT_2_22_CYCLES,
|
|
|
|
WDT_2_23_CYCLES,
|
|
|
|
WDT_2_24_CYCLES,
|
|
|
|
WDT_2_25_CYCLES,
|
|
|
|
WDT_2_26_CYCLES,
|
|
|
|
WDT_2_27_CYCLES,
|
|
|
|
WDT_2_28_CYCLES,
|
|
|
|
WDT_2_29_CYCLES,
|
|
|
|
WDT_2_30_CYCLES,
|
|
|
|
WDT_2_31_CYCLES
|
|
|
|
};
|
|
|
|
|
2015-11-22 03:07:43 +01:00
|
|
|
/**
|
|
|
|
* WDT configuration struct.
|
|
|
|
*/
|
|
|
|
struct wdt_config {
|
2017-04-21 17:55:34 +02:00
|
|
|
u32_t timeout;
|
2015-12-03 09:43:26 +01:00
|
|
|
void (*interrupt_fn)(struct device *dev);
|
2015-11-22 03:07:43 +01:00
|
|
|
};
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Pause watchdog timer when CPU is in sleep state.
|
|
|
|
*/
|
|
|
|
#define WDT_OPT_PAUSE_IN_SLEEP BIT(0)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Pause watchdog timer when CPU is halted by the debugger.
|
|
|
|
*/
|
|
|
|
#define WDT_OPT_PAUSE_HALTED_BY_DBG BIT(1)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Watchdog reset flag bit field mask shift.
|
|
|
|
*/
|
|
|
|
#define WDT_FLAG_RESET_SHIFT (0)
|
|
|
|
/**
|
|
|
|
* @brief Watchdog reset flag bit field mask.
|
|
|
|
*/
|
|
|
|
#define WDT_FLAG_RESET_MASK (0x3 << WDT_FLAG_RESET_SHIFT)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @name Watchdog Reset Behavior.
|
|
|
|
* Reset behavior after timeout.
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
/** No reset */
|
|
|
|
#define WDT_FLAG_RESET_NONE (0 << WDT_FLAG_RESET_SHIFT)
|
|
|
|
/** CPU core reset */
|
|
|
|
#define WDT_FLAG_RESET_CPU_CORE (1 << WDT_FLAG_RESET_SHIFT)
|
|
|
|
/** Global SoC reset */
|
|
|
|
#define WDT_FLAG_RESET_SOC (2 << WDT_FLAG_RESET_SHIFT)
|
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Watchdog timeout window.
|
|
|
|
*
|
|
|
|
* Each installed timeout needs feeding within the specified time window,
|
|
|
|
* otherwise the watchdog will trigger. If the watchdog instance does not
|
|
|
|
* support window timeouts then min value must be equal to 0.
|
|
|
|
*
|
|
|
|
* @param min Lower limit of watchdog feed timeout in milliseconds.
|
|
|
|
* @param max Upper limit of watchdog feed timeout in milliseconds.
|
|
|
|
*
|
|
|
|
* @note If specified values can not be precisely set they are always
|
|
|
|
* rounded up.
|
|
|
|
*/
|
|
|
|
struct wdt_window {
|
|
|
|
u32_t min;
|
|
|
|
u32_t max;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Watchdog callback. */
|
|
|
|
typedef void (*wdt_callback_t)(struct device *dev, int channel_id);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Watchdog timeout configuration struct.
|
|
|
|
*
|
|
|
|
* @param window Timing parameters of watchdog timeout.
|
|
|
|
* @param callback Timeout callback. Passing NULL means that no callback
|
|
|
|
* will be run.
|
|
|
|
* @param next Pointer to the next timeout configuration. This pointer is used
|
|
|
|
* for watchdogs with staged timeouts functionality. Value must be
|
|
|
|
* NULL for single stage timeout.
|
|
|
|
* @param flags Bit field with following parts:
|
|
|
|
*
|
|
|
|
* reset [ 0 : 1 ] - perform specified reset after timeout/callback
|
|
|
|
*/
|
|
|
|
struct wdt_timeout_cfg {
|
|
|
|
struct wdt_window window;
|
|
|
|
wdt_callback_t callback;
|
|
|
|
#ifdef CONFIG_WDT_MULTISTAGE
|
|
|
|
struct wdt_timeout_cfg *next;
|
|
|
|
#endif
|
|
|
|
u8_t flags;
|
|
|
|
};
|
2015-11-22 03:07:43 +01:00
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @typedef wdt_api_setup
|
|
|
|
* @brief Callback API for setting up watchdog instance.
|
|
|
|
* See wdt_setup() for argument descriptions
|
|
|
|
*/
|
|
|
|
typedef int (*wdt_api_setup)(struct device *dev, u8_t options);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef wdt_api_disable
|
|
|
|
* @brief Callback API for disabling watchdog instance.
|
|
|
|
* See wdt_disable() for argument descriptions
|
|
|
|
*/
|
|
|
|
typedef int (*wdt_api_disable)(struct device *dev);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef wdt_api_install_timeout
|
|
|
|
* @brief Callback API for installing new timeout.
|
|
|
|
* See wdt_install_timeout() for argument descriptions
|
|
|
|
*/
|
|
|
|
typedef int (*wdt_api_install_timeout)(struct device *dev,
|
|
|
|
const struct wdt_timeout_cfg *cfg);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef wdt_api_feed
|
|
|
|
* @brief Callback API for feeding specified watchdog timeout.
|
|
|
|
* See (wdt_feed) for argument descriptions
|
|
|
|
*/
|
|
|
|
typedef int (*wdt_api_feed)(struct device *dev, int channel_id);
|
|
|
|
|
|
|
|
/** @cond INTERNAL_HIDDEN */
|
2015-11-22 03:07:43 +01:00
|
|
|
struct wdt_driver_api {
|
2017-08-24 17:35:08 +02:00
|
|
|
wdt_api_setup setup;
|
2015-11-22 03:07:43 +01:00
|
|
|
wdt_api_disable disable;
|
2017-08-24 17:35:08 +02:00
|
|
|
wdt_api_install_timeout install_timeout;
|
|
|
|
wdt_api_feed feed;
|
2015-11-22 03:07:43 +01:00
|
|
|
};
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @endcond
|
|
|
|
*/
|
2015-11-22 03:07:43 +01:00
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @brief Set up watchdog instance.
|
|
|
|
*
|
|
|
|
* This function is used for configuring global watchdog settings that
|
|
|
|
* affect all timeouts. It should be called after installing timeouts.
|
|
|
|
* After successful return, all installed timeouts are valid and must be
|
2018-05-23 21:06:24 +02:00
|
|
|
* serviced periodically by calling wdt_feed().
|
2017-08-24 17:35:08 +02:00
|
|
|
*
|
|
|
|
* @param dev Pointer to the device structure for the driver instance.
|
|
|
|
* @param options Configuration options as defined by the WDT_OPT_* constants
|
|
|
|
*
|
|
|
|
* @retval 0 If successful.
|
|
|
|
* @retval -ENOTSUP If any of the set options is not supported.
|
|
|
|
* @retval -EBUSY If watchdog instance has been already setup.
|
|
|
|
*/
|
|
|
|
static inline int wdt_setup(struct device *dev, u8_t options)
|
2015-11-22 03:07:43 +01:00
|
|
|
{
|
2018-03-17 12:44:40 +01:00
|
|
|
const struct wdt_driver_api *api =
|
|
|
|
(const struct wdt_driver_api *)dev->driver_api;
|
2015-11-22 03:07:43 +01:00
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
return api->setup(dev, options);
|
2015-11-22 03:07:43 +01:00
|
|
|
}
|
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @brief Disable watchdog instance.
|
|
|
|
*
|
|
|
|
* This function disables the watchdog instance and automatically uninstalls all
|
|
|
|
* timeouts. To set up a new watchdog, install timeouts and call wdt_setup()
|
|
|
|
* again. Not all watchdogs can be restarted after they are disabled.
|
|
|
|
*
|
|
|
|
* @param dev Pointer to the device structure for the driver instance.
|
|
|
|
*
|
|
|
|
* @retval 0 If successful.
|
|
|
|
* @retval -EFAULT If watchdog instance is not enabled.
|
|
|
|
* @retval -EPERM If watchdog can not be disabled directly by application code.
|
|
|
|
*/
|
|
|
|
static inline int wdt_disable(struct device *dev)
|
2015-11-22 03:07:43 +01:00
|
|
|
{
|
2018-03-17 12:44:40 +01:00
|
|
|
const struct wdt_driver_api *api =
|
|
|
|
(const struct wdt_driver_api *)dev->driver_api;
|
2015-11-22 03:07:43 +01:00
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
return api->disable(dev);
|
2015-11-22 03:07:43 +01:00
|
|
|
}
|
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @brief Install new timeout.
|
|
|
|
*
|
|
|
|
* This function must be used before wdt_setup(). Changes applied here
|
|
|
|
* have no effects until wdt_setup() is called.
|
|
|
|
*
|
|
|
|
* @param dev Pointer to the device structure for the driver instance.
|
|
|
|
* @param cfg Pointer to timeout configuration structure.
|
|
|
|
*
|
2018-06-04 01:42:08 +02:00
|
|
|
* @retval channel_id If successful, a non-negative value indicating the index
|
|
|
|
* of the channel to which the timeout was assigned. This
|
|
|
|
* value is supposed to be used as the parameter in calls to
|
|
|
|
* wdt_feed().
|
2017-08-24 17:35:08 +02:00
|
|
|
* @retval -EBUSY If timeout can not be installed while watchdog has already
|
|
|
|
* been setup.
|
|
|
|
* @retval -ENOMEM If no more timeouts can be installed.
|
|
|
|
* @retval -ENOTSUP If any of the set flags is not supported.
|
|
|
|
* @retval -EINVAL If any of the window timeout value is out of possible range.
|
|
|
|
* This value is also returned if watchdog supports only one
|
|
|
|
* timeout value for all timeouts and the supplied timeout
|
|
|
|
* window differs from windows for alarms installed so far.
|
|
|
|
*/
|
|
|
|
static inline int wdt_install_timeout(struct device *dev,
|
|
|
|
const struct wdt_timeout_cfg *cfg)
|
|
|
|
{
|
|
|
|
const struct wdt_driver_api *api =
|
|
|
|
(const struct wdt_driver_api *) dev->driver_api;
|
|
|
|
|
|
|
|
return api->install_timeout(dev, cfg);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Feed specified watchdog timeout.
|
|
|
|
*
|
|
|
|
* @param dev Pointer to the device structure for the driver instance.
|
2018-05-23 21:06:24 +02:00
|
|
|
* @param channel_id Index of the fed channel.
|
2017-08-24 17:35:08 +02:00
|
|
|
*
|
|
|
|
* @retval 0 If successful.
|
|
|
|
* @retval -EINVAL If there is no installed timeout for supplied channel.
|
|
|
|
*/
|
|
|
|
static inline int wdt_feed(struct device *dev, int channel_id)
|
|
|
|
{
|
|
|
|
const struct wdt_driver_api *api =
|
|
|
|
(const struct wdt_driver_api *)dev->driver_api;
|
|
|
|
|
|
|
|
return api->feed(dev, channel_id);
|
|
|
|
}
|
|
|
|
|
2016-01-22 18:38:49 +01:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-08-24 17:35:08 +02:00
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2018-09-14 19:43:44 +02:00
|
|
|
#endif /* _ZEPHYR_WATCHDOG_H__ */
|