/**************************************************************************//** * * @copyright (C) 2020 Nuvoton Technology Corp. All rights reserved. * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2022-3-15 Wayne First version * ******************************************************************************/ #include #if defined(BSP_USING_TIMER) && defined(RT_USING_HWTIMER) #include #include "NuMicro.h" /* Private define ---------------------------------------------------------------*/ #define NU_TIMER_DEVICE(timer) (nu_timer_t)(timer) enum { TIMER_START = -1, #if defined(BSP_USING_TIMER0) TIMER0_IDX, #endif #if defined(BSP_USING_TIMER1) TIMER1_IDX, #endif #if defined(BSP_USING_TIMER2) TIMER2_IDX, #endif #if defined(BSP_USING_TIMER3) TIMER3_IDX, #endif TIMER_CNT }; /* Private typedef --------------------------------------------------------------*/ struct nu_timer { rt_hwtimer_t parent; char *name; TIMER_T *base; IRQn_Type irqn; uint32_t rstidx; uint32_t modid; }; typedef struct nu_timer *nu_timer_t; /* Private functions ------------------------------------------------------------*/ static void nu_timer_init(rt_hwtimer_t *timer, rt_uint32_t state); static rt_err_t nu_timer_start(rt_hwtimer_t *timer, rt_uint32_t cnt, rt_hwtimer_mode_t opmode); static void nu_timer_stop(rt_hwtimer_t *timer); static rt_uint32_t nu_timer_count_get(rt_hwtimer_t *timer); static rt_err_t nu_timer_control(rt_hwtimer_t *timer, rt_uint32_t cmd, void *args); /* Public functions -------------------------------------------------------------*/ /* Private variables ------------------------------------------------------------*/ static struct nu_timer nu_timer_arr [] = { #if defined(BSP_USING_TIMER0) { .name = "timer0", .base = TIMER0, .irqn = TMR0_IRQn, .rstidx = TMR0_RST, .modid = TMR0_MODULE }, #endif #if defined(BSP_USING_TIMER1) { .name = "timer1", .base = TIMER1, .irqn = TMR1_IRQn, .rstidx = TMR1_RST, .modid = TMR1_MODULE }, #endif #if defined(BSP_USING_TIMER2) { .name = "timer2", .base = TIMER2, .irqn = TMR2_IRQn, .rstidx = TMR2_RST, .modid = TMR2_MODULE }, #endif #if defined(BSP_USING_TIMER3) { .name = "timer3", .base = TIMER3, .irqn = TMR3_IRQn, .rstidx = TMR3_RST, .modid = TMR3_MODULE }, #endif }; static struct rt_hwtimer_info nu_timer_info = { 12000000, /* maximum count frequency */ 46875, /* minimum count frequency */ 0xFFFFFF, /* the maximum counter value */ HWTIMER_CNTMODE_UP, /* Increment or Decreasing count mode */ }; static struct rt_hwtimer_ops nu_timer_ops = { nu_timer_init, nu_timer_start, nu_timer_stop, nu_timer_count_get, nu_timer_control }; /* Functions define ------------------------------------------------------------*/ static void nu_timer_init(rt_hwtimer_t *timer, rt_uint32_t state) { nu_timer_t psNuTmr = NU_TIMER_DEVICE(timer); RT_ASSERT(psNuTmr != RT_NULL); if (1 == state) { uint32_t timer_clk; struct rt_hwtimer_info *info = &nu_timer_info; timer_clk = TIMER_GetModuleClock(psNuTmr->base); info->maxfreq = timer_clk; info->minfreq = timer_clk / 256; TIMER_Open(psNuTmr->base, TIMER_ONESHOT_MODE, 1); TIMER_EnableInt(psNuTmr->base); NVIC_EnableIRQ(psNuTmr->irqn); } else { NVIC_DisableIRQ(psNuTmr->irqn); TIMER_DisableInt(psNuTmr->base); TIMER_Close(psNuTmr->base); } } static rt_err_t nu_timer_start(rt_hwtimer_t *timer, rt_uint32_t cnt, rt_hwtimer_mode_t opmode) { rt_err_t ret = RT_EINVAL; rt_uint32_t u32OpMode; nu_timer_t psNuTmr = NU_TIMER_DEVICE(timer); RT_ASSERT(psNuTmr != RT_NULL); if (cnt <= 1 || cnt > 0xFFFFFF) { goto exit_nu_timer_start; } switch (opmode) { case HWTIMER_MODE_PERIOD: u32OpMode = TIMER_PERIODIC_MODE; break; case HWTIMER_MODE_ONESHOT: u32OpMode = TIMER_ONESHOT_MODE; break; default: goto exit_nu_timer_start; } TIMER_SET_CMP_VALUE(psNuTmr->base, cnt); TIMER_SET_OPMODE(psNuTmr->base, u32OpMode); TIMER_EnableInt(psNuTmr->base); NVIC_EnableIRQ(psNuTmr->irqn); TIMER_Start(psNuTmr->base); ret = RT_EOK; exit_nu_timer_start: return -(ret); } static void nu_timer_stop(rt_hwtimer_t *timer) { nu_timer_t psNuTmr = NU_TIMER_DEVICE(timer); RT_ASSERT(psNuTmr != RT_NULL); NVIC_DisableIRQ(psNuTmr->irqn); TIMER_DisableInt(psNuTmr->base); TIMER_Stop(psNuTmr->base); TIMER_ResetCounter(psNuTmr->base); } static rt_uint32_t nu_timer_count_get(rt_hwtimer_t *timer) { nu_timer_t psNuTmr = NU_TIMER_DEVICE(timer); RT_ASSERT(psNuTmr != RT_NULL); return TIMER_GetCounter(psNuTmr->base); } static rt_err_t nu_timer_control(rt_hwtimer_t *timer, rt_uint32_t cmd, void *args) { rt_err_t ret = RT_EOK; nu_timer_t psNuTmr = NU_TIMER_DEVICE(timer); RT_ASSERT(psNuTmr != RT_NULL); switch (cmd) { case HWTIMER_CTRL_FREQ_SET: { uint32_t clk; uint32_t pre; clk = TIMER_GetModuleClock(psNuTmr->base); pre = clk / *((uint32_t *)args) - 1; TIMER_SET_PRESCALE_VALUE(psNuTmr->base, pre); *((uint32_t *)args) = clk / (pre + 1) ; } break; case HWTIMER_CTRL_STOP: TIMER_Stop(psNuTmr->base); break; default: ret = RT_EINVAL; break; } return -(ret); } /** * All UART interrupt service routine */ static void nu_timer_isr(nu_timer_t psNuTmr) { RT_ASSERT(psNuTmr != RT_NULL); if (TIMER_GetIntFlag(psNuTmr->base)) { TIMER_ClearIntFlag(psNuTmr->base); rt_device_hwtimer_isr(&psNuTmr->parent); } } int rt_hw_timer_init(void) { int i; rt_err_t ret = RT_EOK; for (i = (TIMER_START + 1); i < TIMER_CNT; i++) { CLK_EnableModuleClock(nu_timer_arr[i].modid); SYS_ResetModule(nu_timer_arr[i].rstidx); /* Register Timer information. */ nu_timer_arr[i].parent.info = &nu_timer_info; /* Register Timer operation. */ nu_timer_arr[i].parent.ops = &nu_timer_ops; /* Register RT hwtimer device. */ ret = rt_device_hwtimer_register(&nu_timer_arr[i].parent, nu_timer_arr[i].name, &nu_timer_arr[i]); RT_ASSERT(ret == RT_EOK); } return 0; } INIT_BOARD_EXPORT(rt_hw_timer_init); #if defined(BSP_USING_TIMER0) void TMR0_IRQHandler(void) { rt_interrupt_enter(); nu_timer_isr((void *)&nu_timer_arr[TIMER0_IDX]); rt_interrupt_leave(); } #endif #if defined(BSP_USING_TIMER1) void TMR1_IRQHandler(void) { rt_interrupt_enter(); nu_timer_isr((void *)&nu_timer_arr[TIMER1_IDX]); rt_interrupt_leave(); } #endif #if defined(BSP_USING_TIMER2) void TMR2_IRQHandler(void) { rt_interrupt_enter(); nu_timer_isr((void *)&nu_timer_arr[TIMER2_IDX]); rt_interrupt_leave(); } #endif #if defined(BSP_USING_TIMER3) void TMR3_IRQHandler(void) { rt_interrupt_enter(); nu_timer_isr((void *)&nu_timer_arr[TIMER3_IDX]); rt_interrupt_leave(); } #endif #endif //#if (defined(BSP_USING_TIMER) && defined(RT_USING_HWTIMER))