mirror of https://github.com/ARMmbed/mbed-os.git
440 lines
14 KiB
C
440 lines
14 KiB
C
/**************************************************************************//**
|
|
* @file timer.c
|
|
* @version V3.00
|
|
* @brief Timer Controller(Timer) driver source file
|
|
*
|
|
* @copyright (C) 2017 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
|
|
{
|
|
if(u32Clk > 128000000UL)
|
|
{
|
|
u32Prescale = 15UL; /* real prescaler value is 16 */
|
|
u32Clk >>= 4;
|
|
}
|
|
else if(u32Clk > 64000000UL)
|
|
{
|
|
u32Prescale = 7UL; /* real prescaler value is 8 */
|
|
u32Clk >>= 3;
|
|
}
|
|
else if(u32Clk > 32000000UL)
|
|
{
|
|
u32Prescale = 3UL; /* real prescaler value is 4 */
|
|
u32Clk >>= 2;
|
|
}
|
|
else if(u32Clk > 16000000UL)
|
|
{
|
|
u32Prescale = 1UL; /* real prescaler value is 2 */
|
|
u32Clk >>= 1;
|
|
}
|
|
|
|
u32Cmpr = u32Clk / u32Freq;
|
|
}
|
|
|
|
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
|
|
{
|
|
if(u32Clk > 128000000UL)
|
|
{
|
|
u32Prescale = 15UL; /* real prescaler value is 16 */
|
|
u32Clk >>= 4;
|
|
}
|
|
else if(u32Clk > 64000000UL)
|
|
{
|
|
u32Prescale = 7UL; /* real prescaler value is 8 */
|
|
u32Clk >>= 3;
|
|
}
|
|
else if(u32Clk > 32000000UL)
|
|
{
|
|
u32Prescale = 3UL; /* real prescaler value is 4 */
|
|
u32Clk >>= 2;
|
|
}
|
|
else if(u32Clk > 16000000UL)
|
|
{
|
|
u32Prescale = 1UL; /* real prescaler value is 2 */
|
|
u32Clk >>= 1;
|
|
}
|
|
|
|
if(u32Usec < 250UL)
|
|
{
|
|
u32Cmpr = (u32Usec * u32Clk) / 1000000UL;
|
|
}
|
|
else
|
|
{
|
|
u32NsecPerTick = 1000000000UL / u32Clk;
|
|
u32Cmpr = (u32Usec * 1000UL) / u32NsecPerTick;
|
|
}
|
|
}
|
|
|
|
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
|
|
* - \ref TIMER_CAPTURE_EVENT_GET_LOW_PERIOD
|
|
* - \ref TIMER_CAPTURE_EVENT_GET_HIGH_PERIOD
|
|
*
|
|
* @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 = __HIRC;
|
|
const uint32_t au32Clk[] = {__HXT, __LXT, 0UL, 0UL, 0UL, __LIRC, 0UL, __HIRC};
|
|
|
|
if(timer == TIMER0)
|
|
{
|
|
u32Src = CLK_GetModuleClockSource(TMR0_MODULE);
|
|
}
|
|
else if(timer == TIMER1)
|
|
{
|
|
u32Src = CLK_GetModuleClockSource(TMR1_MODULE);
|
|
}
|
|
else if((timer == TIMER2) || (timer == TIMER2_NS))
|
|
{
|
|
u32Src = CLK_GetModuleClockSource(TMR2_MODULE);
|
|
}
|
|
else if((timer == TIMER3) || (timer == TIMER3_NS))
|
|
{
|
|
u32Src = CLK_GetModuleClockSource(TMR3_MODULE);
|
|
}
|
|
else
|
|
{
|
|
u32Clk = 0UL;
|
|
}
|
|
|
|
if(u32Clk == 0UL)
|
|
{
|
|
; /* Invalid timer channel */
|
|
}
|
|
else
|
|
{
|
|
if(u32Src == 2UL)
|
|
{
|
|
if((timer == TIMER0) || (timer == TIMER1))
|
|
{
|
|
u32Clk = CLK_GetPCLK0Freq();
|
|
}
|
|
else
|
|
{
|
|
u32Clk = CLK_GetPCLK1Freq();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
u32Clk = au32Clk[u32Src];
|
|
}
|
|
}
|
|
|
|
return u32Clk;
|
|
}
|
|
|
|
/**
|
|
* @brief Enable Timer Frequency Counter Function
|
|
*
|
|
* @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
|
|
* @param[in] u32DropCount This parameter has no effect in this BSP
|
|
* @param[in] u32Timeout This parameter has no effect in this 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 */
|
|
|
|
if(timer == TIMER0)
|
|
{
|
|
t = TIMER1;
|
|
}
|
|
else if(timer == TIMER2)
|
|
{
|
|
t = TIMER3;
|
|
}
|
|
else if(timer == TIMER2_NS)
|
|
{
|
|
t = TIMER3_NS;
|
|
}
|
|
else
|
|
{
|
|
t = 0UL ;
|
|
}
|
|
|
|
if(t != 0UL)
|
|
{
|
|
t->CMP = 0xFFFFFFUL;
|
|
t->EXTCTL = u32EnableInt ? TIMER_EXTCTL_CAPIEN_Msk : 0UL;
|
|
timer->CTL = TIMER_CTL_INTRGEN_Msk | TIMER_CTL_CNTEN_Msk;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Disable Timer Frequency Counter Function
|
|
*
|
|
* @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
|
|
*
|
|
* @return None
|
|
*
|
|
* @brief This function is used to disable the Timer frequency counter function.
|
|
*/
|
|
void TIMER_DisableFreqCounter(TIMER_T *timer)
|
|
{
|
|
timer->CTL &= ~TIMER_CTL_INTRGEN_Msk;
|
|
}
|
|
|
|
/**
|
|
* @brief Select Interrupt Source to Trigger others Module
|
|
*
|
|
* @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
|
|
* @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
|
|
*
|
|
* @brief This function is used to select the interrupt source used to trigger other modules.
|
|
*/
|
|
void TIMER_SetTriggerSource(TIMER_T *timer, uint32_t u32Src)
|
|
{
|
|
timer->TRGCTL = (timer->TRGCTL & ~TIMER_TRGCTL_TRGSSEL_Msk) | u32Src;
|
|
}
|
|
|
|
/**
|
|
* @brief Set Modules Trigger by Timer Interrupt
|
|
*
|
|
* @param[in] timer The pointer of the specified Timer module. It could be TIMER0, TIMER1, TIMER2, TIMER3.
|
|
* @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
|
|
*
|
|
* @details This function is used to set EPWM, EADC, DAC and PDMA module triggered by timer interrupt event.
|
|
*/
|
|
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 */
|
|
|
|
/*** (C) COPYRIGHT 2017 Nuvoton Technology Corp. ***/
|