249 lines
5.7 KiB
C
249 lines
5.7 KiB
C
/*
|
|
* Copyright (c) 2021 - 2022 HPMicro
|
|
*
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*
|
|
*/
|
|
|
|
#include <rthw.h>
|
|
#include <rtdevice.h>
|
|
#include <rtdbg.h>
|
|
#include "board.h"
|
|
#include "drv_wdt.h"
|
|
#include "hpm_wdg_drv.h"
|
|
#include "hpm_sysctl_drv.h"
|
|
|
|
|
|
#ifdef BSP_USING_WDG
|
|
|
|
|
|
typedef struct hpm_wdog
|
|
{
|
|
WDG_Type *wdog_base;
|
|
char *device_name;
|
|
clock_name_t clock_name;
|
|
uint32_t irq_num;
|
|
rt_watchdog_t *wdog;
|
|
}hpm_wdog_t;
|
|
|
|
static rt_err_t hpm_wdog_init(rt_watchdog_t *wdt);
|
|
static rt_err_t hpm_wdog_open(rt_watchdog_t *wdt, rt_uint16_t oflag);
|
|
static rt_err_t hpm_wdog_close(rt_watchdog_t *wdt);
|
|
static rt_err_t hpm_wdog_refresh(rt_watchdog_t *wdt);
|
|
static rt_err_t hpm_wdog_control(rt_watchdog_t *wdt, int cmd, void *args);
|
|
|
|
static void hpm_wdog_isr(rt_watchdog_t *wdt);
|
|
|
|
static wdg_control_t wdog_ctrl = {
|
|
.reset_interval = reset_interval_clock_period_mult_16k,
|
|
.interrupt_interval = interrupt_interval_clock_period_multi_8k,
|
|
.reset_enable = true,
|
|
.interrupt_enable = false,
|
|
.clksrc = wdg_clksrc_extclk,
|
|
.wdg_enable = false,
|
|
};
|
|
|
|
#if defined(BSP_USING_WDG0)
|
|
rt_watchdog_t wdog0;
|
|
void wdog0_isr(void)
|
|
{
|
|
hpm_wdog_isr(&wdog0);
|
|
}
|
|
SDK_DECLARE_EXT_ISR_M(IRQn_WDOG0, wdog0_isr)
|
|
#endif
|
|
|
|
#if defined(BSP_USING_WDG1)
|
|
rt_watchdog_t wdog1;
|
|
void wdog1_isr(void)
|
|
{
|
|
hpm_wdog_isr(&wdog1);
|
|
}
|
|
SDK_DECLARE_EXT_ISR_M(IRQn_WDOG1, wdog1_isr)
|
|
#endif
|
|
|
|
#if defined(BSP_USING_WDG2)
|
|
rt_watchdog_t wdog2;
|
|
void wdog2_isr(void)
|
|
{
|
|
hpm_wdog_isr(&wdog2);
|
|
}
|
|
SDK_DECLARE_EXT_ISR_M(IRQn_WDOG2, wdog2_isr)
|
|
#endif
|
|
|
|
#if defined(BSP_USING_WDG3)
|
|
rt_watchdog_t wdog3;
|
|
void wdog3_isr(void)
|
|
{
|
|
hpm_wdog_isr(&wdog3);
|
|
}
|
|
SDK_DECLARE_EXT_ISR_M(IRQn_WDOG3, wdog3_isr)
|
|
#endif
|
|
|
|
static hpm_wdog_t wdogs[] = {
|
|
#ifdef BSP_USING_WDG0
|
|
{
|
|
.wdog_base = HPM_WDG0,
|
|
.device_name = "wdt0",
|
|
.clock_name = clock_watchdog0,
|
|
.irq_num = IRQn_WDG0,
|
|
.wdog = &wdog0,
|
|
},
|
|
#endif
|
|
|
|
#ifdef BSP_USING_WDG1
|
|
{
|
|
.wdog_base = HPM_WDG1,
|
|
.device_name = "wdt1",
|
|
.clock_name = clock_watchdog1,
|
|
.irq_num = IRQn_WDG1,
|
|
.wdog = &wdog1,
|
|
},
|
|
#endif
|
|
|
|
#ifdef BSP_USING_WDG2
|
|
{
|
|
.wdog_base = HPM_WDG2,
|
|
.device_name = "wdt2",
|
|
.clock_name = clock_watchdog2,
|
|
.irq_num = IRQn_WDG2,
|
|
.wdog = &wdog2,
|
|
},
|
|
#endif
|
|
|
|
#ifdef BSP_USING_WDG3
|
|
{
|
|
.wdog_name = HPM_WDG3,
|
|
.device_name = "wdt3",
|
|
.clock_name = clock_watchdog3,
|
|
.irq_num = IRQn_WDG3,
|
|
.wdog = &wdog3,
|
|
},
|
|
#endif
|
|
};
|
|
|
|
static struct rt_watchdog_ops hpm_wdog_ops = {
|
|
.init = hpm_wdog_init,
|
|
.control = hpm_wdog_control,
|
|
};
|
|
|
|
static rt_err_t hpm_wdog_init(rt_watchdog_t *wdt)
|
|
{
|
|
hpm_wdog_t *hpm_wdog = (hpm_wdog_t*)wdt->parent.user_data;
|
|
WDG_Type *base = hpm_wdog->wdog_base;
|
|
|
|
wdg_init(base, &wdog_ctrl);
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t hpm_wdog_open(rt_watchdog_t *wdt, rt_uint16_t oflag)
|
|
{
|
|
hpm_wdog_t *hpm_wdog = (hpm_wdog_t*)wdt->parent.user_data;
|
|
WDG_Type *base = hpm_wdog->wdog_base;
|
|
|
|
rt_enter_critical();
|
|
wdg_enable(base);
|
|
rt_exit_critical();
|
|
}
|
|
|
|
static rt_err_t hpm_wdog_close(rt_watchdog_t *wdt)
|
|
{
|
|
hpm_wdog_t *hpm_wdog = (hpm_wdog_t*)wdt->parent.user_data;
|
|
WDG_Type *base = hpm_wdog->wdog_base;
|
|
|
|
rt_enter_critical();
|
|
wdg_disable(base);
|
|
rt_exit_critical();
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t hpm_wdog_refresh(rt_watchdog_t *wdt)
|
|
{
|
|
hpm_wdog_t *hpm_wdog = (hpm_wdog_t*)wdt->parent.user_data;
|
|
WDG_Type *base = hpm_wdog->wdog_base;
|
|
|
|
rt_enter_critical();
|
|
wdg_restart(base);
|
|
rt_exit_critical();
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_err_t hpm_wdog_control(rt_watchdog_t *wdt, int cmd, void *args)
|
|
{
|
|
rt_err_t ret = RT_EOK;
|
|
|
|
hpm_wdog_t *hpm_wdog = (hpm_wdog_t*)wdt->parent.user_data;
|
|
WDG_Type *base = hpm_wdog->wdog_base;
|
|
|
|
uint32_t temp;
|
|
switch (cmd)
|
|
{
|
|
case RT_DEVICE_CTRL_WDT_GET_TIMEOUT:
|
|
temp = wdg_get_total_reset_interval_in_us(base, WDG_EXT_CLK_FREQ);
|
|
temp /= 1000000UL; /* Convert to seconds */
|
|
*(uint32_t *)args = temp;
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_SET_TIMEOUT:
|
|
RT_ASSERT(*(uint32_t *)args != 0);
|
|
temp = *(uint32_t *)args;
|
|
temp *= 1000000U; /* Convert to microseconds */
|
|
wdog_ctrl.interrupt_interval = wdg_convert_interrupt_interval_from_us(WDG_EXT_CLK_FREQ, temp);
|
|
wdog_ctrl.reset_interval = reset_interval_clock_period_mult_128;
|
|
wdog_ctrl.reset_enable = true;
|
|
wdog_ctrl.interrupt_enable = true;
|
|
wdog_ctrl.clksrc = wdg_clksrc_extclk;
|
|
wdog_ctrl.wdg_enable = false;
|
|
hpm_wdog_init(wdt);
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_KEEPALIVE:
|
|
hpm_wdog_refresh(wdt);
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_START:
|
|
hpm_wdog_open(wdt, *(uint16_t*)args);
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_STOP:
|
|
hpm_wdog_close(wdt);
|
|
break;
|
|
default:
|
|
ret = RT_EINVAL;
|
|
break;
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
void hpm_wdog_isr(rt_watchdog_t *wdt)
|
|
{
|
|
hpm_wdog_t *hpm_wdog = (hpm_wdog_t*)wdt->parent.user_data;
|
|
WDG_Type *base = hpm_wdog->wdog_base;
|
|
|
|
uint32_t status = wdg_get_status(base);
|
|
|
|
if (IS_HPM_BITMASK_SET(status, WDG_ST_INTEXPIRED_MASK)) {
|
|
wdg_clear_status(base, WDG_ST_INTEXPIRED_MASK);
|
|
}
|
|
}
|
|
|
|
int rt_hw_wdt_init(void)
|
|
{
|
|
rt_err_t err = RT_EOK;
|
|
|
|
#if defined(BSP_USING_WDG)
|
|
for (uint32_t i = 0; i < sizeof(wdogs) / sizeof(wdogs[0]); i++)
|
|
{
|
|
wdogs[i].wdog->ops = &hpm_wdog_ops;
|
|
clock_add_to_group(wdogs[i].clock_name, 0);
|
|
err = rt_hw_watchdog_register(wdogs[i].wdog, wdogs[i].device_name, RT_DEVICE_FLAG_RDWR, (void *)&wdogs[i]);
|
|
if (err != RT_EOK)
|
|
{
|
|
LOG_E("rt device %s failed, status=%d\n", wdogs[i].device_name, err);
|
|
}
|
|
}
|
|
#endif
|
|
return err;
|
|
}
|
|
|
|
INIT_BOARD_EXPORT(rt_hw_wdt_init);
|
|
#endif /* RT_USING_WDT */ |