5dd3b7427a
* bsp: cvitek: kconfig: add wdt for cv18xx_riscv Add Watchdog timer in Kconfig. Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * drv: cvitek: remove using macro from source file Building of source file should be controlled by SConscript, but not in source file itself. Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * bsp: cvitek: kconfig: add i2c for cv18xx_riscv Add I2C in Kconfig for c906B. Note, the IRQ# is different from that of c906L. Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * bsp: cvitek: kconfig: add rtc for cv18xx_riscv Add RTC in Kconfig for c906B. Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * bsp: cvitek: fix channel issue for pwm driver The original code confuses the concepts of controllers and channels. Fixed it and do some code cleanup. Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * bsp:cvitek: add i2c pinmux config for cv18xx_riscv Pinmux in driver code is controlled by SOC type, bcos driver code should be general and support all pins defined by SoC. Pinmux configuration in Kconfig is controlled by BOARD type, bcos when we operate on board, it does not expose all chip-level pin signals and we can only use part of them. Following is I2C signals exported by duo family. Details see https://milkv.io/docs/duo/overview. Note: we have not added support for duo-S. Duo === NAME I2C CV1800B/GPIO <PINNAME>__<FUNCNAME> ---- --- ------------ --------------------- GP0 I2C0_SCL XGPIOA[28] IIC0_SCL__IIC0_SCL GP1 I2C0_SDA XGPIOA[29] IIC0_SDA__IIC0_SDA GP4 I2C1_SCL PWR_GPIO[19] SD1_D2__IIC1_SCL GP9 I2C1_SCL PWR_GPIO[18] SD1_D3__IIC1_SCL GP11 I2C1_SCL XGPIOC[10] PAD_MIPIRX0N__IIC1_SCL GP5 I2C1_SDA PWR_GPIO[20] SD1_D1__IIC1_SDA GP8 I2C1_SDA PWR_GPIO[21] SD1_D0__IIC1_SDA GP10 I2C1_SDA XGPIOC[9] PAD_MIPIRX1P__IIC1_SDA GP7 I2C3_SCL PWR_GPIO[22] SD1_CMD__IIC3_SCL GP6 I2C3_SDA PWR_GPIO[23] SD1_CLK__IIC3_SDA Duo 256m ======== NAME I2C CV1800B/GPIO <PINNAME>__<FUNCNAME> ---- --- ------------ --------------------- GP4 I2C1_SCL PWR_GPIO[19] SD1_D2__IIC1_SCL GP9 I2C1_SCL PWR_GPIO[18] SD1_D3__IIC1_SCL GP5 I2C1_SDA PWR_GPIO[20] SD1_D1__IIC1_SDA GP8 I2C1_SDA PWR_GPIO[21] SD1_D0__IIC1_SDA GP11 I2C2_SCL XGPIOC[15] PAD_MIPI_TXP1__IIC2_SCL GP10 I2C2_SDA XGPIOC[14] PAD_MIPI_TXM1__IIC2_SDA GP7 I2C3_SCL PWR_GPIO[22] SD1_CMD__IIC3_SCL GP6 I2C3_SDA PWR_GPIO[23] SD1_CLK__IIC3_SDA Duo S ===== NAME I2C CV1800B/GPIO <PINNAME>__<FUNCNAME> ---- --- ------------ --------------------- J3-B18 I2C1_SCL XGPIOB[18] VIVO_D3__IIC1_SCL J3-B12 I2C1_SCL XGPIOB[12] VIVO_D9__IIC1_SCL J3-B11 I2C1_SDA XGPIOB[11] VIVO_D10__IIC1_SDA J3-B13 I2C2_SCL XGPIOB[13] VIVO_D8__IIC2_SCL J4-E1 I2C2_SCL PWR_GPIO[1] PWR_GPIO1__IIC2_SCL J3-B14 I2C2_SDA XGPIOB[14] VIVO_D7__IIC2_SDA J4-E2 I2C2_SDA PWR_GPIO[2] PWR_GPIO2__IIC2_SDA J3-B20 I2C4_SCL XGPIOB[20] VIVO_D1__IIC4_SCL J4-B1 I2C4_SCL XGPIOB[1] ADC3__IIC4_SCL J3-B21 I2C4_SDA XGPIOB[21] VIVO_D0__IIC4_SDA J4-B2 I2C4_SDA XGPIOB[2] ADC2__IIC4_SDA Signed-off-by: Chen Wang <unicorn_wang@outlook.com> Signed-off-by: flyingcys <flyingcys@163.com> * bsp:cvitek: remove using macro from source file for i2c Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * bsp:cvitek: unify menu message text for i2c as other drivers Other dirvers has no extra word "HW". Signed-off-by: Chen Wang <unicorn_wang@outlook.com> * bsp:cvitek: add i2c pinmux config for c906_little Porting what we have done in commit "bsp:cvitek: add i2c pinmux config for cv18xx_riscv" to c906_little. Signed-off-by: Chen Wang <unicorn_wang@outlook.com> --------- Signed-off-by: Chen Wang <unicorn_wang@outlook.com> Signed-off-by: flyingcys <flyingcys@163.com> Co-authored-by: flyingcys <flyingcys@163.com>
162 lines
3.9 KiB
C
162 lines
3.9 KiB
C
/*
|
|
* Copyright (c) 2006-2024, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2024/03/02 ShichengChu first version
|
|
*/
|
|
#include <rtthread.h>
|
|
#include <rtdevice.h>
|
|
#include "drv_wdt.h"
|
|
|
|
#define DBG_LEVEL DBG_LOG
|
|
#include <rtdbg.h>
|
|
#define LOG_TAG "DRV.WDT"
|
|
|
|
#define WDT_FREQ_DEFAULT 25000000UL
|
|
#define CVI_WDT_MAX_TOP 15
|
|
|
|
struct _cvi_wdt_dev
|
|
{
|
|
struct rt_watchdog_device device;
|
|
const char *name;
|
|
rt_uint32_t base;
|
|
};
|
|
|
|
static struct _cvi_wdt_dev _wdt_dev[] =
|
|
{
|
|
#ifdef BSP_USING_WDT0
|
|
{
|
|
.name = "wdt0",
|
|
.base = CVI_WDT0_BASE
|
|
},
|
|
#endif /* BSP_USING_WDT0 */
|
|
#ifdef BSP_USING_WDT1
|
|
{
|
|
.name = "wdt1",
|
|
.base = CVI_WDT1_BASE
|
|
},
|
|
#endif /* BSP_USING_WDT1 */
|
|
#ifdef BSP_USING_WDT2
|
|
{
|
|
.name = "wdt2",
|
|
.base = CVI_WDT2_BASE
|
|
},
|
|
#endif /* BSP_USING_WDT2 */
|
|
};
|
|
struct rt_watchdog_device wdt_device;
|
|
|
|
static rt_err_t _wdt_init(rt_watchdog_t *wdt_device)
|
|
{
|
|
cvi_wdt_top_setting();
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
rt_inline int wdt_top_in_ms(unsigned int top)
|
|
{
|
|
/*
|
|
* There are 16 possible timeout values in 0..15 where the number of
|
|
* cycles is 2 ^ (16 + i) and the watchdog counts down.
|
|
*/
|
|
// pr_debug("wdt top in seconds: %d/%d=%d\n", (1U << (16 + top)), chip->clk_hz, (1U << (16 + top)) / chip->clk_hz);
|
|
return (1U << (16 + top)) / (WDT_FREQ_DEFAULT / 1000);
|
|
}
|
|
|
|
static rt_err_t csi_wdt_set_timeout(unsigned long reg_base, uint32_t ms)
|
|
{
|
|
rt_err_t ret = RT_EOK;
|
|
int i, top_val = CVI_WDT_MAX_TOP;
|
|
|
|
/*
|
|
* Iterate over the timeout values until we find the closest match. We
|
|
* always look for >=.
|
|
*/
|
|
for (i = 0; i <= CVI_WDT_MAX_TOP; ++i)
|
|
if (wdt_top_in_ms(i) >= ms) {
|
|
top_val = i;
|
|
break;
|
|
}
|
|
|
|
if (i < CVI_WDT_MAX_TOP)
|
|
{
|
|
/*
|
|
* Set the new value in the watchdog. Some versions of wdt_chip
|
|
* have TOPINIT in the TIMEOUT_RANGE register (as per
|
|
* CP_WDT_DUAL_TOP in WDT_COMP_PARAMS_1). On those we
|
|
* effectively get a pat of the watchdog right here.
|
|
*/
|
|
cvi_wdt_set_timeout(reg_base, top_val);
|
|
cvi_wdt_start_en(reg_base);
|
|
}
|
|
else
|
|
{
|
|
ret = -RT_ERROR;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
static rt_err_t _wdt_control(rt_watchdog_t *wdt_device, int cmd, void *arg)
|
|
{
|
|
RT_ASSERT(wdt_device != RT_NULL);
|
|
|
|
struct _cvi_wdt_dev *wdt = rt_container_of(wdt_device, struct _cvi_wdt_dev, device);
|
|
rt_uint32_t reg_base = wdt->base;
|
|
|
|
switch (cmd)
|
|
{
|
|
case RT_DEVICE_CTRL_WDT_KEEPALIVE:
|
|
cvi_wdt_feed_en(reg_base);
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_SET_TIMEOUT:
|
|
csi_wdt_set_timeout(reg_base, *(rt_uint32_t *)arg);
|
|
wdt_device->parent.user_data = (rt_uint32_t)(*(rt_uint32_t *)arg);
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_GET_TIMEOUT:
|
|
*(rt_uint32_t *)arg = (rt_uint32_t)wdt_device->parent.user_data;
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_GET_TIMELEFT:
|
|
*(rt_uint32_t *)arg = (cvi_wdt_get_counter_value(reg_base) / (WDT_FREQ_DEFAULT / 1000U));
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_START:
|
|
cvi_wdt_set_respond_system_reset(reg_base);
|
|
cvi_wdt_start_en(reg_base);
|
|
break;
|
|
case RT_DEVICE_CTRL_WDT_STOP:
|
|
cvi_wdt_start_dis(reg_base);
|
|
break;
|
|
default:
|
|
LOG_W("This command is not supported.");
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static const struct rt_watchdog_ops _wdt_ops =
|
|
{
|
|
.init = _wdt_init,
|
|
.control = _wdt_control
|
|
};
|
|
|
|
int rt_hw_wdt_init(void)
|
|
{
|
|
rt_uint8_t i;
|
|
for (i = 0; i < sizeof(_wdt_dev) / sizeof(_wdt_dev[0]); i ++)
|
|
{
|
|
_wdt_dev[i].device.ops = &_wdt_ops;
|
|
if (rt_hw_watchdog_register(&_wdt_dev[i].device, _wdt_dev[i].name, RT_DEVICE_FLAG_RDWR, RT_NULL) != RT_EOK)
|
|
{
|
|
LOG_E("%s register failed!", _wdt_dev[i].name);
|
|
return -RT_ERROR;
|
|
}
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
INIT_BOARD_EXPORT(rt_hw_wdt_init);
|