| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353 |
- /**************************************************************************//**
- * @file timer.c
- * @brief M480 Timer Controller(Timer) driver source file
- *
- * SPDX-License-Identifier: Apache-2.0
- * @copyright (C) 2016-2020 Nuvoton Technology Corp. All rights reserved.
- *****************************************************************************/
- #include "NuMicro.h"
- /** @addtogroup Standard_Driver Standard Driver
- @{
- */
- /** @addtogroup TIMER_Driver TIMER Driver
- @{
- */
- /** @addtogroup TIMER_EXPORTED_FUNCTIONS TIMER Exported Functions
- @{
- */
- /**
- * @brief Open Timer with Operate Mode and Frequency
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- * @param[in] u32Mode Operation mode. Possible options are
- * - \ref TIMER_ONESHOT_MODE
- * - \ref TIMER_PERIODIC_MODE
- * - \ref TIMER_TOGGLE_MODE
- * - \ref TIMER_CONTINUOUS_MODE
- * @param[in] u32Freq Target working frequency
- *
- * @return Real timer working frequency
- *
- * @details This API is used to configure timer to operate in specified mode and frequency.
- * If timer cannot work in target frequency, a closest frequency will be chose and returned.
- * @note After calling this API, Timer is \b NOT running yet. But could start timer running be calling
- * \ref TIMER_Start macro or program registers directly.
- */
- uint32_t TIMER_Open(TIMER_T *timer, uint32_t u32Mode, uint32_t u32Freq)
- {
- uint32_t u32Clk = TIMER_GetModuleClock(timer);
- uint32_t u32Cmpr = 0UL, u32Prescale = 0UL;
- /* Fastest possible timer working freq is (u32Clk / 2). While cmpr = 2, prescaler = 0. */
- if(u32Freq > (u32Clk / 2UL))
- {
- u32Cmpr = 2UL;
- }
- else
- {
- u32Cmpr = u32Clk / u32Freq;
- u32Prescale = (u32Cmpr >> 24); /* for 24 bits CMPDAT */
- if (u32Prescale > 0UL)
- u32Cmpr = u32Cmpr / (u32Prescale + 1UL);
- }
- timer->CTL = u32Mode | u32Prescale;
- timer->CMP = u32Cmpr;
- return(u32Clk / (u32Cmpr * (u32Prescale + 1UL)));
- }
- /**
- * @brief Stop Timer Counting
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- *
- * @return None
- *
- * @details This API stops timer counting and disable all timer interrupt function.
- */
- void TIMER_Close(TIMER_T *timer)
- {
- timer->CTL = 0UL;
- timer->EXTCTL = 0UL;
- }
- /**
- * @brief Create a specify Delay Time
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- * @param[in] u32Usec Delay period in micro seconds. Valid values are between 100~1000000 (100 micro second ~ 1 second).
- *
- * @return None
- *
- * @details This API is used to create a delay loop for u32usec micro seconds by using timer one-shot mode.
- * @note This API overwrites the register setting of the timer used to count the delay time.
- * @note This API use polling mode. So there is no need to enable interrupt for the timer module used to generate delay.
- */
- void TIMER_Delay(TIMER_T *timer, uint32_t u32Usec)
- {
- uint32_t u32Clk = TIMER_GetModuleClock(timer);
- uint32_t u32Prescale = 0UL, delay = (SystemCoreClock / u32Clk) + 1UL;
- uint32_t u32Cmpr, u32NsecPerTick;
- /* Clear current timer configuration */
- timer->CTL = 0UL;
- timer->EXTCTL = 0UL;
- if(u32Clk <= 1000000UL) /* min delay is 1000 us if timer clock source is <= 1 MHz */
- {
- if(u32Usec < 1000UL)
- {
- u32Usec = 1000UL;
- }
- if(u32Usec > 1000000UL)
- {
- u32Usec = 1000000UL;
- }
- }
- else
- {
- if(u32Usec < 100UL)
- {
- u32Usec = 100UL;
- }
- if(u32Usec > 1000000UL)
- {
- u32Usec = 1000000UL;
- }
- }
- if(u32Clk <= 1000000UL)
- {
- u32Prescale = 0UL;
- u32NsecPerTick = 1000000000UL / u32Clk;
- u32Cmpr = (u32Usec * 1000UL) / u32NsecPerTick;
- }
- else
- {
- u32Cmpr = u32Usec * (u32Clk / 1000000UL);
- u32Prescale = (u32Cmpr >> 24); /* for 24 bits CMPDAT */
- if (u32Prescale > 0UL)
- u32Cmpr = u32Cmpr / (u32Prescale + 1UL);
- }
- timer->CMP = u32Cmpr;
- timer->CTL = TIMER_CTL_CNTEN_Msk | TIMER_ONESHOT_MODE | u32Prescale;
- /* When system clock is faster than timer clock, it is possible timer active bit cannot set in time while we check it.
- And the while loop below return immediately, so put a tiny delay here allowing timer start counting and raise active flag. */
- for(; delay > 0UL; delay--)
- {
- __NOP();
- }
- while(timer->CTL & TIMER_CTL_ACTSTS_Msk)
- {
- ;
- }
- }
- /**
- * @brief Enable Timer Capture Function
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- * @param[in] u32CapMode Timer capture mode. Could be
- * - \ref TIMER_CAPTURE_FREE_COUNTING_MODE
- * - \ref TIMER_CAPTURE_COUNTER_RESET_MODE
- * @param[in] u32Edge Timer capture trigger edge. Possible values are
- * - \ref TIMER_CAPTURE_EVENT_FALLING
- * - \ref TIMER_CAPTURE_EVENT_RISING
- * - \ref TIMER_CAPTURE_EVENT_FALLING_RISING
- * - \ref TIMER_CAPTURE_EVENT_RISING_FALLING
- *
- * @return None
- *
- * @details This API is used to enable timer capture function with specify capture trigger edge \n
- * to get current counter value or reset counter value to 0.
- * @note Timer frequency should be configured separately by using \ref TIMER_Open API, or program registers directly.
- */
- void TIMER_EnableCapture(TIMER_T *timer, uint32_t u32CapMode, uint32_t u32Edge)
- {
- timer->EXTCTL = (timer->EXTCTL & ~(TIMER_EXTCTL_CAPFUNCS_Msk | TIMER_EXTCTL_CAPEDGE_Msk)) |
- u32CapMode | u32Edge | TIMER_EXTCTL_CAPEN_Msk;
- }
- /**
- * @brief Disable Timer Capture Function
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- *
- * @return None
- *
- * @details This API is used to disable the timer capture function.
- */
- void TIMER_DisableCapture(TIMER_T *timer)
- {
- timer->EXTCTL &= ~TIMER_EXTCTL_CAPEN_Msk;
- }
- /**
- * @brief Enable Timer Counter Function
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- * @param[in] u32Edge Detection edge of counter pin. Could be ether
- * - \ref TIMER_COUNTER_EVENT_FALLING, or
- * - \ref TIMER_COUNTER_EVENT_RISING
- *
- * @return None
- *
- * @details This function is used to enable the timer counter function with specify detection edge.
- * @note Timer compare value should be configured separately by using \ref TIMER_SET_CMP_VALUE macro or program registers directly.
- * @note While using event counter function, \ref TIMER_TOGGLE_MODE cannot set as timer operation mode.
- */
- void TIMER_EnableEventCounter(TIMER_T *timer, uint32_t u32Edge)
- {
- timer->EXTCTL = (timer->EXTCTL & ~TIMER_EXTCTL_CNTPHASE_Msk) | u32Edge;
- timer->CTL |= TIMER_CTL_EXTCNTEN_Msk;
- }
- /**
- * @brief Disable Timer Counter Function
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- *
- * @return None
- *
- * @details This API is used to disable the timer event counter function.
- */
- void TIMER_DisableEventCounter(TIMER_T *timer)
- {
- timer->CTL &= ~TIMER_CTL_EXTCNTEN_Msk;
- }
- /**
- * @brief Get Timer Clock Frequency
- *
- * @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
- *
- * @return Timer clock frequency
- *
- * @details This API is used to get the timer clock frequency.
- * @note This API cannot return correct clock rate if timer source is from external clock input.
- */
- uint32_t TIMER_GetModuleClock(TIMER_T *timer)
- {
- uint32_t u32Src, u32Clk;
- const uint32_t au32Clk[] = {__HXT, __LXT, 0UL, 0UL, 0UL, __LIRC, 0UL, __HIRC};
- if(timer == TIMER0)
- {
- u32Src = (CLK->CLKSEL1 & CLK_CLKSEL1_TMR0SEL_Msk) >> CLK_CLKSEL1_TMR0SEL_Pos;
- }
- else if(timer == TIMER1)
- {
- u32Src = (CLK->CLKSEL1 & CLK_CLKSEL1_TMR1SEL_Msk) >> CLK_CLKSEL1_TMR1SEL_Pos;
- }
- else if(timer == TIMER2)
- {
- u32Src = (CLK->CLKSEL1 & CLK_CLKSEL1_TMR2SEL_Msk) >> CLK_CLKSEL1_TMR2SEL_Pos;
- }
- else /* Timer 3 */
- {
- u32Src = (CLK->CLKSEL1 & CLK_CLKSEL1_TMR3SEL_Msk) >> CLK_CLKSEL1_TMR3SEL_Pos;
- }
- if(u32Src == 2UL)
- {
- if((timer == TIMER0) || (timer == TIMER1))
- {
- u32Clk = CLK_GetPCLK0Freq();
- }
- else
- {
- u32Clk = CLK_GetPCLK1Freq();
- }
- }
- else
- {
- u32Clk = au32Clk[u32Src];
- }
- return u32Clk;
- }
- /**
- * @brief This function is used to enable the Timer frequency counter function
- * @param[in] timer The base address of Timer module. Can be \ref TIMER0 or \ref TIMER2
- * @param[in] u32DropCount This parameter has no effect in M480 series BSP
- * @param[in] u32Timeout This parameter has no effect in M480 series BSP
- * @param[in] u32EnableInt Enable interrupt assertion after capture complete or not. Valid values are TRUE and FALSE
- * @return None
- * @details This function is used to calculate input event frequency. After enable
- * this function, a pair of timers, TIMER0 and TIMER1, or TIMER2 and TIMER3
- * will be configured for this function. The mode used to calculate input
- * event frequency is mentioned as "Inter Timer Trigger Mode" in Technical
- * Reference Manual
- */
- void TIMER_EnableFreqCounter(TIMER_T *timer,
- uint32_t u32DropCount,
- uint32_t u32Timeout,
- uint32_t u32EnableInt)
- {
- TIMER_T *t; /* store the timer base to configure compare value */
- t = (timer == TIMER0) ? TIMER1 : TIMER3;
- t->CMP = 0xFFFFFFUL;
- t->EXTCTL = u32EnableInt ? TIMER_EXTCTL_CAPIEN_Msk : 0UL;
- timer->CTL = TIMER_CTL_INTRGEN_Msk | TIMER_CTL_CNTEN_Msk;
- return;
- }
- /**
- * @brief This function is used to disable the Timer frequency counter function.
- * @param[in] timer The base address of Timer module
- * @return None
- */
- void TIMER_DisableFreqCounter(TIMER_T *timer)
- {
- timer->CTL &= ~TIMER_CTL_INTRGEN_Msk;
- }
- /**
- * @brief This function is used to select the interrupt source used to trigger other modules.
- * @param[in] timer The base address of Timer module
- * @param[in] u32Src Selects the interrupt source to trigger other modules. Could be:
- * - \ref TIMER_TRGSRC_TIMEOUT_EVENT
- * - \ref TIMER_TRGSRC_CAPTURE_EVENT
- * @return None
- */
- void TIMER_SetTriggerSource(TIMER_T *timer, uint32_t u32Src)
- {
- timer->TRGCTL = (timer->TRGCTL & ~TIMER_TRGCTL_TRGSSEL_Msk) | u32Src;
- }
- /**
- * @brief This function is used to set modules trigger by timer interrupt
- * @param[in] timer The base address of Timer module
- * @param[in] u32Mask The mask of modules (EPWM, EADC, DAC and PDMA) trigger by timer. Is the combination of
- * - \ref TIMER_TRG_TO_EPWM,
- * - \ref TIMER_TRG_TO_EADC,
- * - \ref TIMER_TRG_TO_DAC, and
- * - \ref TIMER_TRG_TO_PDMA
- * @return None
- */
- void TIMER_SetTriggerTarget(TIMER_T *timer, uint32_t u32Mask)
- {
- timer->TRGCTL = (timer->TRGCTL & ~(TIMER_TRGCTL_TRGEPWM_Msk | TIMER_TRGCTL_TRGDAC_Msk | TIMER_TRGCTL_TRGEADC_Msk | TIMER_TRGCTL_TRGPDMA_Msk)) | u32Mask;
- }
- /*@}*/ /* end of group TIMER_EXPORTED_FUNCTIONS */
- /*@}*/ /* end of group TIMER_Driver */
- /*@}*/ /* end of group Standard_Driver */
|