2015-07-04 09:37:54 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Rakendra Thapa <rakendrathapa@gmail.com
|
2015-10-05 21:12:10 +02:00
|
|
|
* 2015 Marc Poulhiès <dkm@kataplop.net>
|
2015-07-04 09:37:54 +02:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cpu_lm4f120
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_timer
|
2015-07-04 09:37:54 +02:00
|
|
|
* @{
|
|
|
|
*
|
2015-07-16 02:56:33 +02:00
|
|
|
* @file timer.c
|
2015-07-04 09:37:54 +02:00
|
|
|
* @brief Implementation of the low-level timer driver for the LM4F120
|
|
|
|
*
|
|
|
|
* @author Rakendra Thapa <rakendrathapa@gmail.com>
|
2015-10-05 21:12:10 +02:00
|
|
|
* Marc Poulhiès <dkm@kataplop.net>
|
2015-07-04 09:37:54 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "periph/timer.h"
|
2015-07-10 04:06:50 +02:00
|
|
|
#include "mutex.h"
|
2015-07-04 09:37:54 +02:00
|
|
|
|
2015-07-10 04:06:50 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
2015-10-05 21:12:10 +02:00
|
|
|
|
2015-07-04 09:37:54 +02:00
|
|
|
/**
|
|
|
|
* @brief Struct holding the configuration data
|
2015-07-16 02:56:33 +02:00
|
|
|
* @{
|
2015-07-04 09:37:54 +02:00
|
|
|
*/
|
|
|
|
typedef struct {
|
2016-02-17 12:18:24 +01:00
|
|
|
timer_cb_t cb; /**< timeout callback */
|
|
|
|
void *arg; /**< argument to the callback */
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int divisor; /**< software clock divisor */
|
2015-07-04 09:37:54 +02:00
|
|
|
} timer_conf_t;
|
|
|
|
|
|
|
|
static timer_conf_t config[TIMER_NUMOF];
|
2015-07-16 02:56:33 +02:00
|
|
|
/**@}*/
|
2015-07-04 09:37:54 +02:00
|
|
|
|
2015-10-05 21:12:10 +02:00
|
|
|
#include "hw_timer.h"
|
|
|
|
|
2017-01-14 15:34:53 +01:00
|
|
|
/* enable timer interrupts */
|
|
|
|
static inline void _irq_enable(tim_t dev);
|
|
|
|
|
2015-10-05 21:12:10 +02:00
|
|
|
/* Missing from driverlib */
|
|
|
|
static inline unsigned long
|
|
|
|
PRIV_TimerPrescaleSnapshotGet(unsigned long ulbase, unsigned long ultimer) {
|
|
|
|
return((ultimer == TIMER_A) ? HWREG(ulbase + TIMER_O_TAPS) :
|
|
|
|
HWREG(ulbase + TIMER_O_TBPS));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline unsigned long long _scaled_to_ll_value(unsigned int uncorrected, unsigned int divisor)
|
|
|
|
{
|
|
|
|
const unsigned long long scaledv = (unsigned long long) uncorrected * divisor;
|
|
|
|
return scaledv;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline unsigned int _llvalue_to_scaled_value(unsigned long long corrected, unsigned int divisor)
|
|
|
|
{
|
|
|
|
const unsigned long long scaledv = corrected / divisor;
|
|
|
|
return scaledv;
|
|
|
|
}
|
|
|
|
|
2016-02-17 12:18:24 +01:00
|
|
|
int timer_init(tim_t dev, unsigned long freq, timer_cb_t cb, void *arg)
|
2015-07-04 09:37:54 +02:00
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-02-17 12:18:24 +01:00
|
|
|
config[dev].cb = cb;
|
|
|
|
config[dev].arg = arg;
|
2015-10-04 00:21:09 +02:00
|
|
|
config[dev].divisor = ROM_SysCtlClockGet() / freq;
|
2015-10-05 21:12:10 +02:00
|
|
|
|
|
|
|
unsigned int sysctl_timer;
|
|
|
|
unsigned int timer_base;
|
|
|
|
unsigned int timer_side = TIMER_A;
|
|
|
|
unsigned int timer_cfg = TIMER_CFG_SPLIT_PAIR | TIMER_CFG_A_PERIODIC_UP | TIMER_TAMR_TAMIE;
|
|
|
|
unsigned int timer_max_val;
|
|
|
|
unsigned int timer_intbit = TIMER_TIMA_TIMEOUT | TIMER_TIMA_MATCH;
|
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
sysctl_timer = SYSCTL_PERIPH_WTIMER0;
|
|
|
|
timer_base = WTIMER0_BASE;
|
|
|
|
timer_max_val = TIMER_0_MAX_VALUE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
sysctl_timer = SYSCTL_PERIPH_WTIMER1;
|
|
|
|
timer_base = WTIMER1_BASE;
|
|
|
|
timer_max_val = TIMER_1_MAX_VALUE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return -1; /* unreachable */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ROM_SysCtlPeripheralEnable(sysctl_timer);
|
|
|
|
|
|
|
|
ROM_TimerDisable(timer_base, timer_side);
|
|
|
|
ROM_TimerConfigure(timer_base, timer_cfg);
|
|
|
|
|
|
|
|
unsigned long long lltimer_val_max = _scaled_to_ll_value(timer_max_val, config[dev].divisor);
|
|
|
|
|
|
|
|
ROM_TimerPrescaleSet(timer_base, timer_side, lltimer_val_max >> 32);
|
|
|
|
ROM_TimerLoadSet(timer_base, timer_side, lltimer_val_max & 0xFFFFFFFF);
|
|
|
|
ROM_TimerIntClear(timer_base, timer_intbit);
|
|
|
|
|
|
|
|
ROM_TimerIntEnable(timer_base, timer_intbit);
|
|
|
|
|
2017-01-14 15:34:53 +01:00
|
|
|
_irq_enable(dev);
|
2015-10-05 21:12:10 +02:00
|
|
|
timer_start(dev);
|
|
|
|
|
|
|
|
return 0;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int timer_set_absolute(tim_t dev, int channel, unsigned int value)
|
|
|
|
{
|
build: fix unused parameter errors
cpu, sam0_common: fix unused parameter in periph/spi
cpu, kinetis_common: fix unused parameter in periph/spi
cpu, cc2538: fix unused param in periph/i2c
cpu, cc2538: fix unused param in periph/spi
cpu, sam3: fix unused param in periph/spi
cpu, stm32_common: fix unused param in periph/pm
cpu, stm32f3: fix unused params in periph/i2c
cpu, nrf5x_common: fix unused param in periph/gpio
cpu, nrf5x_common: fix unused param in periph/spi
cpu, lpc2387: fix unused params in periph/spi
cpu, cc2538: fix unused params in radio/netdev
cpu, cc2650: fix unused params in periph/uart
cpu, lm4f120: fix unused param in periph/spi
cpu, lm4f120: fix unused params in periph/timer
cpu, lm4f120: fix unused params in periph/uart
cpu, stm32_common: fix unused params in periph/dac
cpu, stm32l0: fix unused params in periph/i2c
cpu, msp430fxyz: fix unused params in periph/uart
cpu, mips: fix unused params
cpu, cc430: fix unused-params in periph/timer
cpu, msp430fxyz: fix unused params in periph/spi
drivers, cc2420: fix unused param
cpu, mips32r2_common: fix unused params in periph/timer
cpu, cc2538: fix unused-param in periph/i2c
cpu, mips32r2_common: fix unused-param in periph/timer
cpu, msp430fxyz: fix unused params in periph/timer
cpu, atmega_common: fix unused params in periph/spi
driver, nrfmin: fix unused params
cpu, cc2538_rf: fix unused params
driver, netdev_ieee802514: fix unused param
cpu, mip_pic32m: fix unused params
cpu, lpc2387: fix unused params in periph/pwm
tests/driver_sdcard_spi: fix unused params
cpu, sam3: fix unused param in periph/pwm
tests/driver_dynamixel: fix unused params, and style issues
cpu, cc430: fix unused param in periph/rtc
cpu, atmega_common: fix unused params in periph/i2c
2017-10-31 12:09:11 +01:00
|
|
|
(void) channel;
|
|
|
|
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int timer_base;
|
|
|
|
unsigned int timer_side = TIMER_A;
|
|
|
|
unsigned long long scaledv;
|
|
|
|
|
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
timer_base = WTIMER0_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
timer_base = WTIMER1_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return -1; /* unreachable */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ROM_TimerDisable(timer_base, timer_side);
|
|
|
|
|
|
|
|
scaledv = _scaled_to_ll_value(value, config[dev].divisor);
|
|
|
|
|
|
|
|
if (scaledv>>32){
|
|
|
|
ROM_TimerPrescaleMatchSet(timer_base, timer_side, scaledv >> 32);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
else {
|
|
|
|
ROM_TimerPrescaleMatchSet(timer_base, timer_side, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ROM_TimerMatchSet(timer_base, timer_side, (unsigned long) (scaledv & 0xFFFFFFFF));
|
|
|
|
ROM_TimerEnable(timer_base, timer_side);
|
|
|
|
|
2019-09-11 13:46:55 +02:00
|
|
|
return 0;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int timer_clear(tim_t dev, int channel)
|
|
|
|
{
|
build: fix unused parameter errors
cpu, sam0_common: fix unused parameter in periph/spi
cpu, kinetis_common: fix unused parameter in periph/spi
cpu, cc2538: fix unused param in periph/i2c
cpu, cc2538: fix unused param in periph/spi
cpu, sam3: fix unused param in periph/spi
cpu, stm32_common: fix unused param in periph/pm
cpu, stm32f3: fix unused params in periph/i2c
cpu, nrf5x_common: fix unused param in periph/gpio
cpu, nrf5x_common: fix unused param in periph/spi
cpu, lpc2387: fix unused params in periph/spi
cpu, cc2538: fix unused params in radio/netdev
cpu, cc2650: fix unused params in periph/uart
cpu, lm4f120: fix unused param in periph/spi
cpu, lm4f120: fix unused params in periph/timer
cpu, lm4f120: fix unused params in periph/uart
cpu, stm32_common: fix unused params in periph/dac
cpu, stm32l0: fix unused params in periph/i2c
cpu, msp430fxyz: fix unused params in periph/uart
cpu, mips: fix unused params
cpu, cc430: fix unused-params in periph/timer
cpu, msp430fxyz: fix unused params in periph/spi
drivers, cc2420: fix unused param
cpu, mips32r2_common: fix unused params in periph/timer
cpu, cc2538: fix unused-param in periph/i2c
cpu, mips32r2_common: fix unused-param in periph/timer
cpu, msp430fxyz: fix unused params in periph/timer
cpu, atmega_common: fix unused params in periph/spi
driver, nrfmin: fix unused params
cpu, cc2538_rf: fix unused params
driver, netdev_ieee802514: fix unused param
cpu, mip_pic32m: fix unused params
cpu, lpc2387: fix unused params in periph/pwm
tests/driver_sdcard_spi: fix unused params
cpu, sam3: fix unused param in periph/pwm
tests/driver_dynamixel: fix unused params, and style issues
cpu, cc430: fix unused param in periph/rtc
cpu, atmega_common: fix unused params in periph/i2c
2017-10-31 12:09:11 +01:00
|
|
|
(void) channel;
|
|
|
|
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int timer_intbit = TIMER_TIMA_TIMEOUT;
|
|
|
|
unsigned int timer_base;
|
|
|
|
|
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return -1;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
timer_base = WTIMER0_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
timer_base = WTIMER1_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return -1; /* unreachable */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ROM_TimerIntClear(timer_base, timer_intbit);
|
2019-09-11 13:46:55 +02:00
|
|
|
return 0;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int timer_read(tim_t dev)
|
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int timer_base;
|
|
|
|
unsigned int timer_side = TIMER_A;
|
|
|
|
unsigned long long high_bits, high_bits_dup;
|
|
|
|
unsigned long long low_bits;
|
|
|
|
unsigned long long total;
|
|
|
|
unsigned int scaled_value;
|
|
|
|
|
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return -1;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
timer_base = WTIMER0_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
timer_base = WTIMER1_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return -1; /* unreachable */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle overflow happening between the 2 register reads */
|
|
|
|
do {
|
|
|
|
high_bits = ((unsigned long long)PRIV_TimerPrescaleSnapshotGet(timer_base, timer_side)) << 32;
|
|
|
|
low_bits = (unsigned long long)ROM_TimerValueGet(timer_base, timer_side);
|
|
|
|
high_bits_dup = ((unsigned long long)PRIV_TimerPrescaleSnapshotGet(timer_base, timer_side)) << 32;
|
|
|
|
} while (high_bits != high_bits_dup);
|
|
|
|
|
|
|
|
total = high_bits + low_bits;
|
|
|
|
DEBUG("Combined %lx:%lx\n", (unsigned long) (total>>32), (unsigned long) (total & 0xFFFFFFFF));
|
|
|
|
|
|
|
|
scaled_value = _llvalue_to_scaled_value(total, config[dev].divisor);
|
|
|
|
|
|
|
|
return scaled_value;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void timer_start(tim_t dev)
|
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int timer_base;
|
|
|
|
unsigned int timer_side = TIMER_A;
|
|
|
|
|
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
timer_base = WTIMER0_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
timer_base = WTIMER1_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return; /* unreachable */
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
|
|
|
|
ROM_TimerEnable(timer_base, timer_side);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void timer_stop(tim_t dev)
|
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int timer_base;
|
|
|
|
unsigned int timer_side = TIMER_A;
|
|
|
|
|
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
timer_base = WTIMER0_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
timer_base = WTIMER1_BASE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return; /* unreachable */
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
|
|
|
|
ROM_TimerDisable(timer_base, timer_side);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
2017-01-14 15:34:53 +01:00
|
|
|
static inline void _irq_enable(tim_t dev)
|
2015-07-04 09:37:54 +02:00
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
unsigned int timer_intbase;
|
|
|
|
|
|
|
|
if (dev >= TIMER_NUMOF){
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(dev){
|
|
|
|
#if TIMER_0_EN
|
|
|
|
case TIMER_0:
|
|
|
|
timer_intbase = INT_WTIMER0A;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if TIMER_1_EN
|
|
|
|
case TIMER_1:
|
|
|
|
timer_intbase = INT_WTIMER1A;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return; /* unreachable */
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
|
|
|
|
ROM_IntPrioritySet(timer_intbase, 32);
|
|
|
|
ROM_IntEnable(timer_intbase);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#if TIMER_0_EN
|
2015-10-05 21:12:10 +02:00
|
|
|
void isr_wtimer0a(void)
|
2015-07-04 09:37:54 +02:00
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
/* Clears both IT */
|
|
|
|
ROM_TimerIntClear(WTIMER0_BASE, TIMER_TIMA_TIMEOUT | TIMER_TIMA_MATCH);
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_0].cb(config[TIMER_0].arg, 0);
|
2016-11-30 18:26:05 +01:00
|
|
|
cortexm_isr_end();
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
#endif /* TIMER_0_EN */
|
|
|
|
|
|
|
|
#if TIMER_1_EN
|
|
|
|
void isr_wtimer1a(void)
|
2015-07-04 09:37:54 +02:00
|
|
|
{
|
2015-10-05 21:12:10 +02:00
|
|
|
ROM_TimerIntClear(WTIMER1_BASE, TIMER_TIMA_TIMEOUT | TIMER_TIMA_MATCH);
|
2015-07-15 11:48:31 +02:00
|
|
|
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_1].cb(config[TIMER_0].arg, 0);
|
2016-11-30 18:26:05 +01:00
|
|
|
cortexm_isr_end();
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-10-05 21:12:10 +02:00
|
|
|
#endif /* TIMER_1_EN */
|
2015-07-16 02:56:33 +02:00
|
|
|
/** @} */
|