123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533 |
- /***************************************************************************//**
- * @file
- * @brief Low Energy Timer (LETIMER) Peripheral API
- * @author Energy Micro AS
- * @version 3.0.0
- *******************************************************************************
- * @section License
- * <b>(C) Copyright 2012 Energy Micro AS, http://www.energymicro.com</b>
- *******************************************************************************
- *
- * Permission is granted to anyone to use this software for any purpose,
- * including commercial applications, and to alter it and redistribute it
- * freely, subject to the following restrictions:
- *
- * 1. The origin of this software must not be misrepresented; you must not
- * claim that you wrote the original software.
- * 2. Altered source versions must be plainly marked as such, and must not be
- * misrepresented as being the original software.
- * 3. This notice may not be removed or altered from any source distribution.
- *
- * DISCLAIMER OF WARRANTY/LIMITATION OF REMEDIES: Energy Micro AS has no
- * obligation to support this Software. Energy Micro AS is providing the
- * Software "AS IS", with no express or implied warranties of any kind,
- * including, but not limited to, any implied warranties of merchantability
- * or fitness for any particular purpose or warranties against infringement
- * of any proprietary rights of a third party.
- *
- * Energy Micro AS will not be liable for any consequential, incidental, or
- * special damages, or any other relief, or for any claim by any third party,
- * arising from your use of this Software.
- *
- ******************************************************************************/
- #include "em_letimer.h"
- #include "em_cmu.h"
- #include "em_assert.h"
- /***************************************************************************//**
- * @addtogroup EM_Library
- * @{
- ******************************************************************************/
- /***************************************************************************//**
- * @addtogroup LETIMER
- * @brief Low Energy Timer (LETIMER) Peripheral API
- * @{
- ******************************************************************************/
- /*******************************************************************************
- ******************************* DEFINES ***********************************
- ******************************************************************************/
- /** @cond DO_NOT_INCLUDE_WITH_DOXYGEN */
- /** Validation of valid comparator register for assert statements. */
- #define LETIMER_COMP_REG_VALID(reg) (((reg) <= 1))
- /** Validation of LETIMER register block pointer reference for assert statements. */
- #define LETIMER_REF_VALID(ref) ((ref) == LETIMER0)
- /** Validation of valid repeat counter register for assert statements. */
- #define LETIMER_REP_REG_VALID(reg) (((reg) <= 1))
- /** @endcond */
- /*******************************************************************************
- ************************** LOCAL FUNCTIONS ********************************
- ******************************************************************************/
- /** @cond DO_NOT_INCLUDE_WITH_DOXYGEN */
- #if defined(_EFM32_GECKO_FAMILY)
- /***************************************************************************//**
- * @brief
- * Wait for ongoing sync of register(s) to low frequency domain to complete.
- *
- * @note
- * This only applies to the Gecko Family, see the reference manual
- * chapter about Access to Low Energy Peripherals (Asynchronos Registers)
- * for details.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block
- *
- * @param[in] mask
- * Bitmask corresponding to SYNCBUSY register defined bits, indicating
- * registers that must complete any ongoing synchronization.
- ******************************************************************************/
- __STATIC_INLINE void LETIMER_Sync(LETIMER_TypeDef *letimer, uint32_t mask)
- {
- /* Avoid deadlock if modifying the same register twice when freeze mode is */
- /* activated. */
- if (letimer->FREEZE & LETIMER_FREEZE_REGFREEZE)
- return;
- /* Wait for any pending previous write operation to have been completed */
- /* in low frequency domain, only required for Gecko Family of devices */
- while (letimer->SYNCBUSY & mask)
- ;
- }
- #endif
- /** @endcond */
- /*******************************************************************************
- ************************** GLOBAL FUNCTIONS *******************************
- ******************************************************************************/
- /***************************************************************************//**
- * @brief
- * Get LETIMER compare register value.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block
- *
- * @param[in] comp
- * Compare register to get, either 0 or 1
- *
- * @return
- * Compare register value, 0 if invalid register selected.
- ******************************************************************************/
- uint32_t LETIMER_CompareGet(LETIMER_TypeDef *letimer, unsigned int comp)
- {
- uint32_t ret;
- EFM_ASSERT(LETIMER_REF_VALID(letimer) && LETIMER_COMP_REG_VALID(comp));
- /* Initialize selected compare value */
- switch (comp)
- {
- case 0:
- ret = letimer->COMP0;
- break;
- case 1:
- ret = letimer->COMP1;
- break;
- default:
- /* Unknown compare register selected */
- ret = 0;
- break;
- }
- return(ret);
- }
- /***************************************************************************//**
- * @brief
- * Set LETIMER compare register value.
- *
- * @note
- * The setting of a compare register requires synchronization into the
- * low frequency domain. If the same register is modified before a previous
- * update has completed, this function will stall until the previous
- * synchronization has completed. This only applies to the Gecko Family, see
- * comment in the LETIMER_Sync() internal function call.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block
- *
- * @param[in] comp
- * Compare register to set, either 0 or 1
- *
- * @param[in] value
- * Initialization value (<= 0x0000ffff)
- ******************************************************************************/
- void LETIMER_CompareSet(LETIMER_TypeDef *letimer,
- unsigned int comp,
- uint32_t value)
- {
- volatile uint32_t *compReg;
- uint32_t syncbusy;
- EFM_ASSERT(LETIMER_REF_VALID(letimer) &&
- LETIMER_COMP_REG_VALID(comp) &&
- ((value & ~(_LETIMER_COMP0_COMP0_MASK >> _LETIMER_COMP0_COMP0_SHIFT)) == 0));
- /* Initialize selected compare value */
- switch (comp)
- {
- case 0:
- compReg = &(letimer->COMP0);
- syncbusy = LETIMER_SYNCBUSY_COMP0;
- break;
- case 1:
- compReg = &(letimer->COMP1);
- syncbusy = LETIMER_SYNCBUSY_COMP1;
- break;
- default:
- /* Unknown compare register selected, abort */
- return;
- }
- #if defined(_EFM32_GECKO_FAMILY)
- /* LF register about to be modified require sync. busy check */
- LETIMER_Sync(letimer, syncbusy);
- #endif
- *compReg = value;
- }
- /***************************************************************************//**
- * @brief
- * Start/stop LETIMER.
- *
- * @note
- * The enabling/disabling of the LETIMER modifies the LETIMER CMD register
- * which requires synchronization into the low frequency domain. If this
- * register is modified before a previous update to the same register has
- * completed, this function will stall until the previous synchronization has
- * completed. This only applies to the Gecko Family, see comment in the
- * LETIMER_Sync() internal function call.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block.
- *
- * @param[in] enable
- * true to enable counting, false to disable.
- ******************************************************************************/
- void LETIMER_Enable(LETIMER_TypeDef *letimer, bool enable)
- {
- EFM_ASSERT(LETIMER_REF_VALID(letimer));
- #if defined(_EFM32_GECKO_FAMILY)
- /* LF register about to be modified require sync. busy check */
- LETIMER_Sync(letimer, LETIMER_SYNCBUSY_CMD);
- #endif
- if (enable)
- {
- letimer->CMD = LETIMER_CMD_START;
- }
- else
- {
- letimer->CMD = LETIMER_CMD_STOP;
- }
- }
- /***************************************************************************//**
- * @brief
- * LETIMER register synchronization freeze control.
- *
- * @details
- * Some LETIMER registers require synchronization into the low frequency (LF)
- * domain. The freeze feature allows for several such registers to be
- * modified before passing them to the LF domain simultaneously (which
- * takes place when the freeze mode is disabled).
- *
- * @note
- * When enabling freeze mode, this function will wait for all current
- * ongoing LETIMER synchronization to LF domain to complete (Normally
- * synchronization will not be in progress.) However for this reason, when
- * using freeze mode, modifications of registers requiring LF synchronization
- * should be done within one freeze enable/disable block to avoid unecessary
- * stalling.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block.
- *
- * @param[in] enable
- * @li true - enable freeze, modified registers are not propagated to the
- * LF domain
- * @li false - disables freeze, modified registers are propagated to LF
- * domain
- ******************************************************************************/
- void LETIMER_FreezeEnable(LETIMER_TypeDef *letimer, bool enable)
- {
- if (enable)
- {
- /*
- * Wait for any ongoing LF synchronization to complete. This is just to
- * protect against the rare case when a user
- * - modifies a register requiring LF sync
- * - then enables freeze before LF sync completed
- * - then modifies the same register again
- * since modifying a register while it is in sync progress should be
- * avoided.
- */
- while (letimer->SYNCBUSY)
- ;
- letimer->FREEZE = LETIMER_FREEZE_REGFREEZE;
- }
- else
- {
- letimer->FREEZE = 0;
- }
- }
- /***************************************************************************//**
- * @brief
- * Initialize LETIMER.
- *
- * @details
- * Note that the compare/repeat values must be set separately with
- * LETIMER_CompareSet() and LETIMER_RepeatSet(). That should probably be done
- * prior to the use of this function if configuring the LETIMER to start when
- * initialization is completed.
- *
- * @note
- * The initialization of the LETIMER modifies the LETIMER CTRL/CMD registers
- * which require synchronization into the low frequency domain. If any of those
- * registers are modified before a previous update to the same register has
- * completed, this function will stall until the previous synchronization has
- * completed. This only applies to the Gecko Family, see comment in the
- * LETIMER_Sync() internal function call.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block.
- *
- * @param[in] init
- * Pointer to LETIMER initialization structure.
- ******************************************************************************/
- void LETIMER_Init(LETIMER_TypeDef *letimer, const LETIMER_Init_TypeDef *init)
- {
- uint32_t tmp = 0;
- EFM_ASSERT(LETIMER_REF_VALID(letimer));
- /* Stop timer if specified to be disabled and running */
- if (!(init->enable) && (letimer->STATUS & LETIMER_STATUS_RUNNING))
- {
- #if defined(_EFM32_GECKO_FAMILY)
- /* LF register about to be modified require sync. busy check */
- LETIMER_Sync(letimer, LETIMER_SYNCBUSY_CMD);
- #endif
- letimer->CMD = LETIMER_CMD_STOP;
- }
- /* Configure DEBUGRUN flag, sets whether or not counter should be
- * updated when debugger is active */
- if (init->debugRun)
- {
- tmp |= LETIMER_CTRL_DEBUGRUN;
- }
- if (init->rtcComp0Enable)
- {
- tmp |= LETIMER_CTRL_RTCC0TEN;
- }
- if (init->rtcComp1Enable)
- {
- tmp |= LETIMER_CTRL_RTCC1TEN;
- }
- if (init->comp0Top)
- {
- tmp |= LETIMER_CTRL_COMP0TOP;
- }
- if (init->bufTop)
- {
- tmp |= LETIMER_CTRL_BUFTOP;
- }
- if (init->out0Pol)
- {
- tmp |= LETIMER_CTRL_OPOL0;
- }
- if (init->out1Pol)
- {
- tmp |= LETIMER_CTRL_OPOL1;
- }
- tmp |= init->ufoa0 << _LETIMER_CTRL_UFOA0_SHIFT;
- tmp |= init->ufoa1 << _LETIMER_CTRL_UFOA1_SHIFT;
- tmp |= init->repMode << _LETIMER_CTRL_REPMODE_SHIFT;
- #if defined(_EFM32_GECKO_FAMILY)
- /* LF register about to be modified require sync. busy check */
- LETIMER_Sync(letimer, LETIMER_SYNCBUSY_CTRL);
- #endif
- letimer->CTRL = tmp;
- /* Start timer if specified to be enabled and not already running */
- if (init->enable && !(letimer->STATUS & LETIMER_STATUS_RUNNING))
- {
- #if defined(_EFM32_GECKO_FAMILY)
- /* LF register about to be modified require sync. busy check */
- LETIMER_Sync(letimer, LETIMER_SYNCBUSY_CMD);
- #endif
- letimer->CMD = LETIMER_CMD_START;
- }
- }
- /***************************************************************************//**
- * @brief
- * Get LETIMER repeat register value.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block
- *
- * @param[in] rep
- * Repeat register to get, either 0 or 1
- *
- * @return
- * Repeat register value, 0 if invalid register selected.
- ******************************************************************************/
- uint32_t LETIMER_RepeatGet(LETIMER_TypeDef *letimer, unsigned int rep)
- {
- uint32_t ret;
- EFM_ASSERT(LETIMER_REF_VALID(letimer) && LETIMER_REP_REG_VALID(rep));
- /* Initialize selected compare value */
- switch (rep)
- {
- case 0:
- ret = letimer->REP0;
- break;
- case 1:
- ret = letimer->REP1;
- break;
- default:
- /* Unknown compare register selected */
- ret = 0;
- break;
- }
- return(ret);
- }
- /***************************************************************************//**
- * @brief
- * Set LETIMER repeat counter register value.
- *
- * @note
- * The setting of a repeat counter register requires synchronization into the
- * low frequency domain. If the same register is modified before a previous
- * update has completed, this function will stall until the previous
- * synchronization has completed. This only applies to the Gecko Family, see
- * comment in the LETIMER_Sync() internal function call.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block
- *
- * @param[in] rep
- * Repeat counter register to set, either 0 or 1
- *
- * @param[in] value
- * Initialization value (<= 0x0000ffff)
- ******************************************************************************/
- void LETIMER_RepeatSet(LETIMER_TypeDef *letimer,
- unsigned int rep,
- uint32_t value)
- {
- volatile uint32_t *repReg;
- #if defined(_EFM32_GECKO_FAMILY)
- uint32_t syncbusy;
- #endif
- EFM_ASSERT(LETIMER_REF_VALID(letimer) &&
- LETIMER_REP_REG_VALID(rep) &&
- ((value & ~(_LETIMER_REP0_REP0_MASK >> _LETIMER_REP0_REP0_SHIFT)) == 0));
- /* Initialize selected compare value */
- switch (rep)
- {
- case 0:
- repReg = &(letimer->REP0);
- #if defined(_EFM32_GECKO_FAMILY)
- syncbusy = LETIMER_SYNCBUSY_REP0;
- #endif
- break;
- case 1:
- repReg = &(letimer->REP1);
- #if defined(_EFM32_GECKO_FAMILY)
- syncbusy = LETIMER_SYNCBUSY_REP1;
- #endif
- break;
- default:
- /* Unknown compare register selected, abort */
- return;
- }
- #if defined(_EFM32_GECKO_FAMILY)
- /* LF register about to be modified require sync. busy check */
- LETIMER_Sync(letimer, syncbusy);
- #endif
- *repReg = value;
- }
- /***************************************************************************//**
- * @brief
- * Reset LETIMER to same state as after a HW reset.
- *
- * @note
- * The ROUTE register is NOT reset by this function, in order to allow for
- * centralized setup of this feature.
- *
- * @param[in] letimer
- * Pointer to LETIMER peripheral register block.
- ******************************************************************************/
- void LETIMER_Reset(LETIMER_TypeDef *letimer)
- {
- /* Freeze registers to avoid stalling for LF synchronization */
- LETIMER_FreezeEnable(letimer, true);
- /* Make sure disabled first, before resetting other registers */
- letimer->CMD = LETIMER_CMD_STOP | LETIMER_CMD_CLEAR |
- LETIMER_CMD_CTO0 | LETIMER_CMD_CTO1;
- letimer->CTRL = _LETIMER_CTRL_RESETVALUE;
- letimer->COMP0 = _LETIMER_COMP0_RESETVALUE;
- letimer->COMP1 = _LETIMER_COMP1_RESETVALUE;
- letimer->REP0 = _LETIMER_REP0_RESETVALUE;
- letimer->REP1 = _LETIMER_REP1_RESETVALUE;
- letimer->IEN = _LETIMER_IEN_RESETVALUE;
- letimer->IFC = _LETIMER_IFC_MASK;
- /* Do not reset route register, setting should be done independently */
- /* Unfreeze registers, pass new settings on to LETIMER */
- LETIMER_FreezeEnable(letimer, false);
- }
- /** @} (end addtogroup LETIMER) */
- /** @} (end addtogroup EM_Library) */
|