123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307 |
- /* *****************************************************************************
- * Copyright (C) 2016 Maxim Integrated Products, Inc., All Rights Reserved.
- *
- * Permission is hereby granted, free of charge, to any person obtaining a
- * copy of this software and associated documentation files (the "Software"),
- * to deal in the Software without restriction, including without limitation
- * the rights to use, copy, modify, merge, publish, distribute, sublicense,
- * and/or sell copies of the Software, and to permit persons to whom the
- * Software is furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included
- * in all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
- * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
- * IN NO EVENT SHALL MAXIM INTEGRATED BE LIABLE FOR ANY CLAIM, DAMAGES
- * OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
- * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
- * OTHER DEALINGS IN THE SOFTWARE.
- *
- * Except as contained in this notice, the name of Maxim Integrated
- * Products, Inc. shall not be used except as stated in the Maxim Integrated
- * Products, Inc. Branding Policy.
- *
- * The mere transfer of this software does not imply any licenses
- * of trade secrets, proprietary technology, copyrights, patents,
- * trademarks, maskwork rights, or any other form of intellectual
- * property whatsoever. Maxim Integrated Products, Inc. retains all
- * ownership rights.
- *
- * $Date: 2019-09-11 14:32:22 -0500 (Wed, 11 Sep 2019) $
- * $Revision: 46047 $
- *
- **************************************************************************** */
- /* **** Includes **** */
- #include "mxc_config.h"
- #include "mxc_assert.h"
- #include "mxc_sys.h"
- #include "tmr.h"
- /* **** Definitions **** */
- /* **** Globals **** */
- /* **** Functions **** */
- /* ************************************************************************** */
- int TMR_Init(mxc_tmr_regs_t *tmr, tmr_pres_t pres, const sys_cfg_tmr_t* sys_cfg)
- {
- MXC_ASSERT(tmr);
-
- int err;
- // System settigns
- if((err=SYS_TMR_Init(tmr, sys_cfg)) !=E_NO_ERROR)
- {
- return err;
- }
-
- // Disable timer and clear settings
- tmr->cn = 0;
-
- // Clear interrupt flag
- tmr->intr = MXC_F_TMR_INTR_IRQ_CLR;
-
- // Set the prescaler
- tmr->cn = pres;
- return err;
- }
- int TMR_Shutdown(mxc_tmr_regs_t *tmr)
- {
- MXC_ASSERT(tmr);
-
- int err;
- // System settigns
- if((err=SYS_TMR_Shutdown(tmr)) !=E_NO_ERROR)
- {
- return err;
- }
-
- // Disable timer and clear settings
- tmr->cn = 0;
-
- return err;
- }
- /* ************************************************************************** */
- void TMR_Enable(mxc_tmr_regs_t* tmr)
- {
- MXC_ASSERT(tmr);
- tmr->cn |= MXC_F_TMR_CN_TEN;
- }
- /* ************************************************************************** */
- void TMR_Disable(mxc_tmr_regs_t* tmr)
- {
- MXC_ASSERT(tmr);
- tmr->cn &= ~(MXC_F_TMR_CN_TEN);
- }
- /* ************************************************************************** */
- int TMR_Config(mxc_tmr_regs_t *tmr, const tmr_cfg_t *cfg)
- {
- MXC_ASSERT(tmr);
-
- // Configure the timer
- tmr->cn = (tmr->cn & ~(MXC_F_TMR_CN_TMODE | MXC_F_TMR_CN_TPOL)) |
- ((cfg->mode << MXC_F_TMR_CN_TMODE_POS) & MXC_F_TMR_CN_TMODE) |
- ((cfg->pol << MXC_F_TMR_CN_TPOL_POS) & MXC_F_TMR_CN_TPOL);
-
- tmr->cnt = 0x1;
- tmr->cmp = cfg->cmp_cnt;
- return E_NO_ERROR;
- }
- /* ************************************************************************** */
- int TMR_PWMConfig(mxc_tmr_regs_t *tmr, const tmr_pwm_cfg_t *cfg)
- {
- if (cfg->duty_cnt > cfg->per_cnt) {
- return E_BAD_PARAM;
- }
-
- // Configure the timer
- tmr->cn = (tmr->cn & ~(MXC_F_TMR_CN_TMODE | MXC_F_TMR_CN_TPOL)) |
- MXC_S_TMR_CN_TMODE_PWM | ((cfg->pol << MXC_F_TMR_CN_TPOL_POS) & MXC_F_TMR_CN_TPOL);
-
- tmr->cnt = 0x1;
- tmr->cmp = cfg->per_cnt;
- tmr->pwm = cfg->duty_cnt;
-
- return E_NO_ERROR;
- }
- /* ************************************************************************** */
- int TMR_PWMSetDuty(mxc_tmr_regs_t *tmr, uint32_t duty)
- {
- uint32_t cnt;
-
- // Make sure the new Duty count is less than the period count
- if (duty > tmr->cmp) {
- return E_BAD_PARAM;
- }
-
- cnt = tmr->cnt; // make sure order of volatile access is known.
- // Avoid glitching the output
- if (duty >= tmr->pwm) {
- // Wait for the count to be in the range of 1 to tmr->pwm
- while (cnt > tmr->pwm) {
- cnt = tmr->cnt; // update the volatile access variable
- }
- } else {
- // Wait for the count to pass tmr->pwm
- while (cnt < tmr->pwm) {
- cnt = tmr->cnt; // update the volatile access variable
- }
- }
- tmr->pwm = duty;
-
- return E_NO_ERROR;
- }
- /* ************************************************************************** */
- int TMR_PWMSetPeriod(mxc_tmr_regs_t *tmr, uint32_t per)
- {
- // Make sure the new Duty count is less than the period count
- if (tmr->pwm > per) {
- return E_BAD_PARAM;
- }
-
- // Wait for the count to be less than the new dut_cnt
- while (tmr->cnt >= per) {}
- tmr->cmp = per;
-
- return E_NO_ERROR;
- }
- /* ************************************************************************** */
- uint32_t TMR_GetCompare(mxc_tmr_regs_t* tmr)
- {
- return tmr->cmp;
- }
- /* ************************************************************************** */
- uint32_t TMR_GetCapture(mxc_tmr_regs_t* tmr)
- {
- return tmr->pwm;
- }
- /* ************************************************************************* */
- uint32_t TMR_GetCount(mxc_tmr_regs_t* tmr)
- {
- return tmr->cnt;
- }
- /* ************************************************************************* */
- void TMR_IntClear(mxc_tmr_regs_t* tmr)
- {
- tmr->intr = MXC_F_TMR_INTR_IRQ_CLR;
- }
- /* ************************************************************************* */
- uint32_t TMR_IntStatus(mxc_tmr_regs_t* tmr)
- {
- return tmr->intr;
- }
- /* ************************************************************************* */
- void TMR_SetCompare(mxc_tmr_regs_t *tmr, uint32_t cmp_cnt)
- {
- tmr->cmp = cmp_cnt;
- }
- /* ************************************************************************* */
- void TMR_SetCount(mxc_tmr_regs_t *tmr, uint32_t cnt)
- {
- tmr->cnt = cnt;
- }
- /* ************************************************************************* */
- int TMR_GetTicks(mxc_tmr_regs_t *tmr, uint32_t time, tmr_unit_t units, uint32_t *ticks)
- {
- uint32_t unit_div0, unit_div1;
- uint32_t timerClock;
- uint32_t prescale;
- uint64_t temp_ticks;
-
- timerClock = SYS_TMR_GetFreq(tmr);
- prescale = ((tmr->cn & MXC_F_TMR_CN_PRES) >> MXC_F_TMR_CN_PRES_POS)
- | (((tmr->cn & MXC_F_TMR_CN_PRES3) >> (MXC_F_TMR_CN_PRES3_POS))<<3);
-
- switch (units) {
- case TMR_UNIT_NANOSEC:
- unit_div0 = 1000000;
- unit_div1 = 1000;
- break;
- case TMR_UNIT_MICROSEC:
- unit_div0 = 1000;
- unit_div1 = 1000;
- break;
- case TMR_UNIT_MILLISEC:
- unit_div0 = 1;
- unit_div1 = 1000;
- break;
- case TMR_UNIT_SEC:
- unit_div0 = 1;
- unit_div1 = 1;
- break;
- default:
- return E_BAD_PARAM;
- }
-
- temp_ticks = (uint64_t)time * (timerClock / unit_div0) / (unit_div1 * (1 << (prescale & 0xF)));
-
- //make sure ticks is within a 32 bit value
- if (!(temp_ticks & 0xffffffff00000000) && (temp_ticks & 0xffffffff)) {
- *ticks = temp_ticks;
- return E_NO_ERROR;
- }
-
- return E_INVALID;
- }
- /* ************************************************************************* */
- int TMR_GetTime(mxc_tmr_regs_t *tmr, uint32_t ticks, uint32_t *time, tmr_unit_t *units)
- {
- uint64_t temp_time = 0;
- uint32_t timerClock = SYS_TMR_GetFreq(tmr);
- uint32_t prescale = ((tmr->cn & MXC_F_TMR_CN_PRES) >> MXC_F_TMR_CN_PRES_POS)
- | (((tmr->cn & MXC_F_TMR_CN_PRES3) >> (MXC_F_TMR_CN_PRES3_POS))<<3);
-
- temp_time = (uint64_t)ticks * 1000 * (1 << (prescale & 0xF)) / (timerClock / 1000000);
- if (!(temp_time & 0xffffffff00000000)) {
- *time = temp_time;
- *units = TMR_UNIT_NANOSEC;
- return E_NO_ERROR;
- }
-
- temp_time = (uint64_t)ticks * 1000 * (1 << (prescale & 0xF)) / (timerClock / 1000);
- if (!(temp_time & 0xffffffff00000000)) {
- *time = temp_time;
- *units = TMR_UNIT_MICROSEC;
- return E_NO_ERROR;
- }
-
- temp_time = (uint64_t)ticks * 1000 * (1 << (prescale & 0xF)) / timerClock;
- if (!(temp_time & 0xffffffff00000000)) {
- *time = temp_time;
- *units = TMR_UNIT_MILLISEC;
- return E_NO_ERROR;
- }
-
- temp_time = (uint64_t)ticks * (1 << (prescale & 0xF)) / timerClock;
- if (!(temp_time & 0xffffffff00000000)) {
- *time = temp_time;
- *units = TMR_UNIT_SEC;
- return E_NO_ERROR;
- }
-
- return E_INVALID;
- }
|