177 lines
4.1 KiB
C
177 lines
4.1 KiB
C
/*
|
|
* Copyright (c) 2016 Open-RnD Sp. z o.o.
|
|
* Copyright (c) 2017 RnDity Sp. z o.o.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <watchdog.h>
|
|
#include <soc.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
|
|
#include "iwdg_stm32.h"
|
|
|
|
/* Minimal timeout in microseconds. */
|
|
#define IWDG_TIMEOUT_MIN 100
|
|
/* Maximal timeout in microseconds. */
|
|
#define IWDG_TIMEOUT_MAX 26214400
|
|
|
|
#define IS_IWDG_TIMEOUT(__TIMEOUT__) \
|
|
(((__TIMEOUT__) >= IWDG_TIMEOUT_MIN) && \
|
|
((__TIMEOUT__) <= IWDG_TIMEOUT_MAX))
|
|
|
|
/*
|
|
* Status register need 5 RC LSI divided by prescaler clock to be updated.
|
|
* With higher prescaler (256U), and according to HSI variation,
|
|
* we need to wait at least 6 cycles so 48 ms.
|
|
*/
|
|
|
|
#define IWDG_DEFAULT_TIMEOUT 48u
|
|
|
|
/**
|
|
* @brief Calculates prescaler & reload values.
|
|
*
|
|
* @param timeout Timeout value in microseconds.
|
|
* @param prescaler Pointer to prescaler value.
|
|
* @param reload Pointer to reload value.
|
|
*/
|
|
static void iwdg_stm32_convert_timeout(u32_t timeout,
|
|
u32_t *prescaler,
|
|
u32_t *reload)
|
|
{
|
|
assert(IS_IWDG_TIMEOUT(timeout));
|
|
|
|
u16_t divider = 0;
|
|
u8_t shift = 0;
|
|
|
|
/* Convert timeout to seconds. */
|
|
float m_timeout = (float)timeout / 1000000 * LSI_VALUE;
|
|
|
|
do {
|
|
divider = 4 << shift;
|
|
shift++;
|
|
} while ((m_timeout / divider) > 0xFFF);
|
|
|
|
/*
|
|
* Value of the 'shift' variable corresponds to the
|
|
* defines of LL_IWDG_PRESCALER_XX type.
|
|
*/
|
|
*prescaler = --shift;
|
|
*reload = (uint32_t)(m_timeout / divider) - 1;
|
|
}
|
|
|
|
static void iwdg_stm32_enable(struct device *dev)
|
|
{
|
|
IWDG_TypeDef *iwdg = IWDG_STM32_STRUCT(dev);
|
|
|
|
LL_IWDG_Enable(iwdg);
|
|
}
|
|
|
|
static void iwdg_stm32_disable(struct device *dev)
|
|
{
|
|
/* watchdog cannot be stopped once started */
|
|
ARG_UNUSED(dev);
|
|
}
|
|
|
|
static int iwdg_stm32_set_config(struct device *dev,
|
|
struct wdt_config *config)
|
|
{
|
|
IWDG_TypeDef *iwdg = IWDG_STM32_STRUCT(dev);
|
|
u32_t timeout = config->timeout;
|
|
u32_t prescaler = 0;
|
|
u32_t reload = 0;
|
|
u32_t tickstart;
|
|
|
|
assert(IS_IWDG_TIMEOUT(timeout));
|
|
|
|
iwdg_stm32_convert_timeout(timeout, &prescaler, &reload);
|
|
|
|
assert(IS_IWDG_PRESCALER(prescaler));
|
|
assert(IS_IWDG_RELOAD(reload));
|
|
|
|
LL_IWDG_EnableWriteAccess(iwdg);
|
|
|
|
LL_IWDG_SetPrescaler(iwdg, prescaler);
|
|
LL_IWDG_SetReloadCounter(iwdg, reload);
|
|
|
|
#if defined(CONFIG_SOC_SERIES_STM32F3X) || defined(CONFIG_SOC_SERIES_STM32L4X)
|
|
/* Neither STM32F1X nor STM32F4 series supports window option. */
|
|
LL_IWDG_SetWindow(iwdg, 0x0FFF);
|
|
#endif
|
|
|
|
tickstart = k_uptime_get_32();
|
|
|
|
while (LL_IWDG_IsReady(iwdg) == 0) {
|
|
if ((k_uptime_get_32() - tickstart) > IWDG_DEFAULT_TIMEOUT) {
|
|
return -ENODEV;
|
|
}
|
|
}
|
|
|
|
LL_IWDG_ReloadCounter(iwdg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void iwdg_stm32_get_config(struct device *dev,
|
|
struct wdt_config *config)
|
|
{
|
|
IWDG_TypeDef *iwdg = IWDG_STM32_STRUCT(dev);
|
|
|
|
u32_t prescaler = LL_IWDG_GetPrescaler(iwdg);
|
|
u32_t reload = LL_IWDG_GetReloadCounter(iwdg);
|
|
|
|
/* Timeout given in microseconds. */
|
|
config->timeout = (u32_t)((4 << prescaler) * (reload + 1)
|
|
* (1000000 / LSI_VALUE));
|
|
}
|
|
|
|
static void iwdg_stm32_reload(struct device *dev)
|
|
{
|
|
IWDG_TypeDef *iwdg = IWDG_STM32_STRUCT(dev);
|
|
|
|
LL_IWDG_ReloadCounter(iwdg);
|
|
}
|
|
|
|
static const struct wdt_driver_api iwdg_stm32_api = {
|
|
.enable = iwdg_stm32_enable,
|
|
.disable = iwdg_stm32_disable,
|
|
.get_config = iwdg_stm32_get_config,
|
|
.set_config = iwdg_stm32_set_config,
|
|
.reload = iwdg_stm32_reload,
|
|
};
|
|
|
|
static int iwdg_stm32_init(struct device *dev)
|
|
{
|
|
IWDG_TypeDef *iwdg = IWDG_STM32_STRUCT(dev);
|
|
struct wdt_config config;
|
|
|
|
config.timeout = CONFIG_IWDG_STM32_TIMEOUT;
|
|
|
|
LL_IWDG_Enable(iwdg);
|
|
|
|
iwdg_stm32_set_config(dev, &config);
|
|
|
|
/*
|
|
* The ST production value for the option bytes where WDG_SW bit is
|
|
* present is 0x00FF55AA, namely the Software watchdog mode is
|
|
* enabled by default.
|
|
* If the IWDG is started by either hardware option or software access,
|
|
* the LSI oscillator is forced ON and cannot be disabled.
|
|
*
|
|
* t_IWDG(ms) = t_LSI(ms) x 4 x 2^(IWDG_PR[2:0]) x (IWDG_RLR[11:0] + 1)
|
|
*/
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct iwdg_stm32_data iwdg_stm32_dev_data = {
|
|
.Instance = IWDG
|
|
};
|
|
|
|
DEVICE_AND_API_INIT(iwdg_stm32, CONFIG_IWDG_STM32_DEVICE_NAME,
|
|
iwdg_stm32_init, &iwdg_stm32_dev_data, NULL,
|
|
POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE,
|
|
&iwdg_stm32_api);
|
|
|