/***************************************************************************** * Copyright 2003 - 2008 Broadcom Corporation. All rights reserved. * * Unless you and Broadcom execute a separate written software license * agreement governing use of this software, this software is licensed to you * under the terms of the GNU General Public License version 2, available at * http://www.broadcom.com/licenses/GPLv2.php (the "GPL"). * * Notwithstanding the above, under no circumstances may you combine this * software in any way with any other Broadcom software provided under a * license other than the GPL, without Broadcom's express prior written * consent. *****************************************************************************/ /****************************************************************************/ /** * @file tmrHw.c * * @brief Low level Timer driver routines * * @note * * These routines provide basic timer functionality only. */ /****************************************************************************/ /* ---- Include Files ---------------------------------------------------- */ #include <csp/errno.h> #include <csp/stdint.h> #include <csp/tmrHw.h> #include <mach/csp/tmrHw_reg.h> #define tmrHw_ASSERT(a) if (!(a)) *(char *)0 = 0 #define tmrHw_MILLISEC_PER_SEC (1000) #define tmrHw_LOW_1_RESOLUTION_COUNT (tmrHw_LOW_RESOLUTION_CLOCK / tmrHw_MILLISEC_PER_SEC) #define tmrHw_LOW_1_MAX_MILLISEC (0xFFFFFFFF / tmrHw_LOW_1_RESOLUTION_COUNT) #define tmrHw_LOW_16_RESOLUTION_COUNT (tmrHw_LOW_1_RESOLUTION_COUNT / 16) #define tmrHw_LOW_16_MAX_MILLISEC (0xFFFFFFFF / tmrHw_LOW_16_RESOLUTION_COUNT) #define tmrHw_LOW_256_RESOLUTION_COUNT (tmrHw_LOW_1_RESOLUTION_COUNT / 256) #define tmrHw_LOW_256_MAX_MILLISEC (0xFFFFFFFF / tmrHw_LOW_256_RESOLUTION_COUNT) #define tmrHw_HIGH_1_RESOLUTION_COUNT (tmrHw_HIGH_RESOLUTION_CLOCK / tmrHw_MILLISEC_PER_SEC) #define tmrHw_HIGH_1_MAX_MILLISEC (0xFFFFFFFF / tmrHw_HIGH_1_RESOLUTION_COUNT) #define tmrHw_HIGH_16_RESOLUTION_COUNT (tmrHw_HIGH_1_RESOLUTION_COUNT / 16) #define tmrHw_HIGH_16_MAX_MILLISEC (0xFFFFFFFF / tmrHw_HIGH_16_RESOLUTION_COUNT) #define tmrHw_HIGH_256_RESOLUTION_COUNT (tmrHw_HIGH_1_RESOLUTION_COUNT / 256) #define tmrHw_HIGH_256_MAX_MILLISEC (0xFFFFFFFF / tmrHw_HIGH_256_RESOLUTION_COUNT) static void ResetTimer(tmrHw_ID_t timerId) __attribute__ ((section(".aramtext"))); static int tmrHw_divide(int num, int denom) __attribute__ ((section(".aramtext"))); /****************************************************************************/ /** * @brief Get timer capability * * This function returns various capabilities/attributes of a timer * * @return Capability * */ /****************************************************************************/ uint32_t tmrHw_getTimerCapability(tmrHw_ID_t timerId, /* [ IN ] Timer Id */ tmrHw_CAPABILITY_e capability /* [ IN ] Timer capability */ ) { switch (capability) { case tmrHw_CAPABILITY_CLOCK: return (timerId <= 1) ? tmrHw_LOW_RESOLUTION_CLOCK : tmrHw_HIGH_RESOLUTION_CLOCK; case tmrHw_CAPABILITY_RESOLUTION: return 32; default: return 0; } return 0; } /****************************************************************************/ /** * @brief Resets a timer * * This function initializes timer * * @return void * */ /****************************************************************************/ static void ResetTimer(tmrHw_ID_t timerId /* [ IN ] Timer Id */ ) { /* Reset timer */ pTmrHw[timerId].LoadValue = 0; pTmrHw[timerId].CurrentValue = 0xFFFFFFFF; pTmrHw[timerId].Control = 0; pTmrHw[timerId].BackgroundLoad = 0; /* Always configure as a 32 bit timer */ pTmrHw[timerId].Control |= tmrHw_CONTROL_32BIT; /* Clear interrupt only if raw status interrupt is set */ if (pTmrHw[timerId].RawInterruptStatus) { pTmrHw[timerId].InterruptClear = 0xFFFFFFFF; } } /****************************************************************************/ /** * @brief Sets counter value for an interval in ms * * @return On success: Effective counter value set * On failure: 0 * */ /****************************************************************************/ static tmrHw_INTERVAL_t SetTimerPeriod(tmrHw_ID_t timerId, /* [ IN ] Timer Id */ tmrHw_INTERVAL_t msec /* [ IN ] Interval in milli-second */ ) { uint32_t scale = 0; uint32_t count = 0; if (timerId == 0 || timerId == 1) { if (msec <= tmrHw_LOW_1_MAX_MILLISEC) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_1; scale = tmrHw_LOW_1_RESOLUTION_COUNT; } else if (msec <= tmrHw_LOW_16_MAX_MILLISEC) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_16; scale = tmrHw_LOW_16_RESOLUTION_COUNT; } else if (msec <= tmrHw_LOW_256_MAX_MILLISEC) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_256; scale = tmrHw_LOW_256_RESOLUTION_COUNT; } else { return 0; } count = msec * scale; /* Set counter value */ pTmrHw[timerId].LoadValue = count; pTmrHw[timerId].BackgroundLoad = count; } else if (timerId == 2 || timerId == 3) { if (msec <= tmrHw_HIGH_1_MAX_MILLISEC) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_1; scale = tmrHw_HIGH_1_RESOLUTION_COUNT; } else if (msec <= tmrHw_HIGH_16_MAX_MILLISEC) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_16; scale = tmrHw_HIGH_16_RESOLUTION_COUNT; } else if (msec <= tmrHw_HIGH_256_MAX_MILLISEC) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_256; scale = tmrHw_HIGH_256_RESOLUTION_COUNT; } else { return 0; } count = msec * scale; /* Set counter value */ pTmrHw[timerId].LoadValue = count; pTmrHw[timerId].BackgroundLoad = count; } return count / scale; } /****************************************************************************/ /** * @brief Configures a periodic timer in terms of timer interrupt rate * * This function initializes a periodic timer to generate specific number of * timer interrupt per second * * @return On success: Effective timer frequency * On failure: 0 * */ /****************************************************************************/ tmrHw_RATE_t tmrHw_setPeriodicTimerRate(tmrHw_ID_t timerId, /* [ IN ] Timer Id */ tmrHw_RATE_t rate /* [ IN ] Number of timer interrupt per second */ ) { uint32_t resolution = 0; uint32_t count = 0; ResetTimer(timerId); /* Set timer mode periodic */ pTmrHw[timerId].Control |= tmrHw_CONTROL_PERIODIC; pTmrHw[timerId].Control &= ~tmrHw_CONTROL_ONESHOT; /* Set timer in highest resolution */ pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_1; if (rate && (timerId == 0 || timerId == 1)) { if (rate > tmrHw_LOW_RESOLUTION_CLOCK) { return 0; } resolution = tmrHw_LOW_RESOLUTION_CLOCK; } else if (rate && (timerId == 2 || timerId == 3)) { if (rate > tmrHw_HIGH_RESOLUTION_CLOCK) { return 0; } else { resolution = tmrHw_HIGH_RESOLUTION_CLOCK; } } else { return 0; } /* Find the counter value */ count = resolution / rate; /* Set counter value */ pTmrHw[timerId].LoadValue = count; pTmrHw[timerId].BackgroundLoad = count; return resolution / count; } /****************************************************************************/ /** * @brief Configures a periodic timer to generate timer interrupt after * certain time interval * * This function initializes a periodic timer to generate timer interrupt * after every time interval in millisecond * * @return On success: Effective interval set in milli-second * On failure: 0 * */ /****************************************************************************/ tmrHw_INTERVAL_t tmrHw_setPeriodicTimerInterval(tmrHw_ID_t timerId, /* [ IN ] Timer Id */ tmrHw_INTERVAL_t msec /* [ IN ] Interval in milli-second */ ) { ResetTimer(timerId); /* Set timer mode periodic */ pTmrHw[timerId].Control |= tmrHw_CONTROL_PERIODIC; pTmrHw[timerId].Control &= ~tmrHw_CONTROL_ONESHOT; return SetTimerPeriod(timerId, msec); } /****************************************************************************/ /** * @brief Configures a periodic timer to generate timer interrupt just once * after certain time interval * * This function initializes a periodic timer to generate a single ticks after * certain time interval in millisecond * * @return On success: Effective interval set in milli-second * On failure: 0 * */ /****************************************************************************/ tmrHw_INTERVAL_t tmrHw_setOneshotTimerInterval(tmrHw_ID_t timerId, /* [ IN ] Timer Id */ tmrHw_INTERVAL_t msec /* [ IN ] Interval in milli-second */ ) { ResetTimer(timerId); /* Set timer mode oneshot */ pTmrHw[timerId].Control |= tmrHw_CONTROL_PERIODIC; pTmrHw[timerId].Control |= tmrHw_CONTROL_ONESHOT; return SetTimerPeriod(timerId, msec); } /****************************************************************************/ /** * @brief Configures a timer to run as a free running timer * * This function initializes a timer to run as a free running timer * * @return Timer resolution (count / sec) * */ /****************************************************************************/ tmrHw_RATE_t tmrHw_setFreeRunningTimer(tmrHw_ID_t timerId, /* [ IN ] Timer Id */ uint32_t divider /* [ IN ] Dividing the clock frequency */ ) { uint32_t scale = 0; ResetTimer(timerId); /* Set timer as free running mode */ pTmrHw[timerId].Control &= ~tmrHw_CONTROL_PERIODIC; pTmrHw[timerId].Control &= ~tmrHw_CONTROL_ONESHOT; if (divider >= 64) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_256; scale = 256; } else if (divider >= 8) { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_16; scale = 16; } else { pTmrHw[timerId].Control |= tmrHw_CONTROL_PRESCALE_1; scale = 1; } if (timerId == 0 || timerId == 1) { return tmrHw_divide(tmrHw_LOW_RESOLUTION_CLOCK, scale); } else if (timerId == 2 || timerId == 3) { return tmrHw_divide(tmrHw_HIGH_RESOLUTION_CLOCK, scale); } return 0; } /****************************************************************************/ /** * @brief Starts a timer * * This function starts a preconfigured timer * * @return -1 - On Failure * 0 - On Success * */ /****************************************************************************/ int tmrHw_startTimer(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { pTmrHw[timerId].Control |= tmrHw_CONTROL_TIMER_ENABLE; return 0; } /****************************************************************************/ /** * @brief Stops a timer * * This function stops a running timer * * @return -1 - On Failure * 0 - On Success * */ /****************************************************************************/ int tmrHw_stopTimer(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { pTmrHw[timerId].Control &= ~tmrHw_CONTROL_TIMER_ENABLE; return 0; } /****************************************************************************/ /** * @brief Gets current timer count * * This function returns the current timer value * * @return Current downcounting timer value * */ /****************************************************************************/ uint32_t tmrHw_GetCurrentCount(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { /* return 32 bit timer value */ switch (pTmrHw[timerId].Control & tmrHw_CONTROL_MODE_MASK) { case tmrHw_CONTROL_FREE_RUNNING: if (pTmrHw[timerId].CurrentValue) { return tmrHw_MAX_COUNT - pTmrHw[timerId].CurrentValue; } break; case tmrHw_CONTROL_PERIODIC: case tmrHw_CONTROL_ONESHOT: return pTmrHw[timerId].BackgroundLoad - pTmrHw[timerId].CurrentValue; } return 0; } /****************************************************************************/ /** * @brief Gets timer count rate * * This function returns the number of counts per second * * @return Count rate * */ /****************************************************************************/ tmrHw_RATE_t tmrHw_getCountRate(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { uint32_t divider = 0; switch (pTmrHw[timerId].Control & tmrHw_CONTROL_PRESCALE_MASK) { case tmrHw_CONTROL_PRESCALE_1: divider = 1; break; case tmrHw_CONTROL_PRESCALE_16: divider = 16; break; case tmrHw_CONTROL_PRESCALE_256: divider = 256; break; default: tmrHw_ASSERT(0); } if (timerId == 0 || timerId == 1) { return tmrHw_divide(tmrHw_LOW_RESOLUTION_CLOCK, divider); } else { return tmrHw_divide(tmrHw_HIGH_RESOLUTION_CLOCK, divider); } return 0; } /****************************************************************************/ /** * @brief Enables timer interrupt * * This function enables the timer interrupt * * @return N/A * */ /****************************************************************************/ void tmrHw_enableInterrupt(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { pTmrHw[timerId].Control |= tmrHw_CONTROL_INTERRUPT_ENABLE; } /****************************************************************************/ /** * @brief Disables timer interrupt * * This function disable the timer interrupt * * @return N/A * */ /****************************************************************************/ void tmrHw_disableInterrupt(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { pTmrHw[timerId].Control &= ~tmrHw_CONTROL_INTERRUPT_ENABLE; } /****************************************************************************/ /** * @brief Clears the interrupt * * This function clears the timer interrupt * * @return N/A * * @note * Must be called under the context of ISR */ /****************************************************************************/ void tmrHw_clearInterrupt(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { pTmrHw[timerId].InterruptClear = 0x1; } /****************************************************************************/ /** * @brief Gets the interrupt status * * This function returns timer interrupt status * * @return Interrupt status */ /****************************************************************************/ tmrHw_INTERRUPT_STATUS_e tmrHw_getInterruptStatus(tmrHw_ID_t timerId /* [ IN ] Timer id */ ) { if (pTmrHw[timerId].InterruptStatus) { return tmrHw_INTERRUPT_STATUS_SET; } else { return tmrHw_INTERRUPT_STATUS_UNSET; } } /****************************************************************************/ /** * @brief Indentifies a timer causing interrupt * * This functions returns a timer causing interrupt * * @return 0xFFFFFFFF : No timer causing an interrupt * ! 0xFFFFFFFF : timer causing an interrupt * @note * tmrHw_clearIntrrupt() must be called with a valid timer id after calling this function */ /****************************************************************************/ tmrHw_ID_t tmrHw_getInterruptSource(void /* void */ ) { int i; for (i = 0; i < tmrHw_TIMER_NUM_COUNT; i++) { if (pTmrHw[i].InterruptStatus) { return i; } } return 0xFFFFFFFF; } /****************************************************************************/ /** * @brief Displays specific timer registers * * * @return void * */ /****************************************************************************/ void tmrHw_printDebugInfo(tmrHw_ID_t timerId, /* [ IN ] Timer id */ int (*fpPrint) (const char *, ...) /* [ IN ] Print callback function */ ) { (*fpPrint) ("Displaying register contents \n\n"); (*fpPrint) ("Timer %d: Load value 0x%X\n", timerId, pTmrHw[timerId].LoadValue); (*fpPrint) ("Timer %d: Background load value 0x%X\n", timerId, pTmrHw[timerId].BackgroundLoad); (*fpPrint) ("Timer %d: Control 0x%X\n", timerId, pTmrHw[timerId].Control); (*fpPrint) ("Timer %d: Interrupt clear 0x%X\n", timerId, pTmrHw[timerId].InterruptClear); (*fpPrint) ("Timer %d: Interrupt raw interrupt 0x%X\n", timerId, pTmrHw[timerId].RawInterruptStatus); (*fpPrint) ("Timer %d: Interrupt status 0x%X\n", timerId, pTmrHw[timerId].InterruptStatus); } /****************************************************************************/ /** * @brief Use a timer to perform a busy wait delay for a number of usecs. * * @return N/A */ /****************************************************************************/ void tmrHw_udelay(tmrHw_ID_t timerId, /* [ IN ] Timer id */ unsigned long usecs /* [ IN ] usec to delay */ ) { tmrHw_RATE_t usec_tick_rate; tmrHw_COUNT_t start_time; tmrHw_COUNT_t delta_time; start_time = tmrHw_GetCurrentCount(timerId); usec_tick_rate = tmrHw_divide(tmrHw_getCountRate(timerId), 1000000); delta_time = usecs * usec_tick_rate; /* Busy wait */ while (delta_time > (tmrHw_GetCurrentCount(timerId) - start_time)) ; } /****************************************************************************/ /** * @brief Local Divide function * * This function does the divide * * @return divide value * */ /****************************************************************************/ static int tmrHw_divide(int num, int denom) { int r; int t = 1; /* Shift denom and t up to the largest value to optimize algorithm */ /* t contains the units of each divide */ while ((denom & 0x40000000) == 0) { /* fails if denom=0 */ denom = denom << 1; t = t << 1; } /* Initialize the result */ r = 0; do { /* Determine if there exists a positive remainder */ if ((num - denom) >= 0) { /* Accumlate t to the result and calculate a new remainder */ num = num - denom; r = r + t; } /* Continue to shift denom and shift t down to 0 */ denom = denom >> 1; t = t >> 1; } while (t != 0); return r; }