2014-11-19 16:31:33 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* 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_lpc1768
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_timer
|
2014-11-19 16:31:33 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of the low-level timer driver for the LPC1768
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2014-12-08 20:48:58 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
2014-11-19 16:31:33 +01:00
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "periph/timer.h"
|
|
|
|
|
|
|
|
/* guard file in case no timers are defined */
|
|
|
|
#if TIMER_0_EN
|
|
|
|
|
2014-12-08 20:48:58 +01:00
|
|
|
/**
|
|
|
|
* @name Timer channel interrupt flags
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#define MR0_FLAG (0x01) /**< match for channel 0 */
|
|
|
|
#define MR1_FLAG (0x02) /**< match for channel 1 */
|
|
|
|
#define MR2_FLAG (0x04) /**< match for channel 2 */
|
|
|
|
#define MR3_FLAG (0x08) /**< match for channel 3 */
|
|
|
|
/** @} */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief UART device configurations
|
|
|
|
*/
|
2016-02-17 12:18:24 +01:00
|
|
|
static timer_isr_ctx_t config[TIMER_NUMOF];
|
2014-12-08 20:48:58 +01:00
|
|
|
|
2016-02-17 12:18:24 +01:00
|
|
|
int timer_init(tim_t dev, unsigned long freq, timer_cb_t cb, void *arg)
|
2014-11-19 16:31:33 +01:00
|
|
|
{
|
2014-12-08 20:48:58 +01:00
|
|
|
if (dev == TIMER_0) {
|
|
|
|
/* save callback */
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_0].cb = cb;
|
|
|
|
config[TIMER_0].arg = arg;
|
2014-12-08 20:48:58 +01:00
|
|
|
/* enable power for timer */
|
|
|
|
TIMER_0_CLKEN();
|
|
|
|
/* let timer run with full frequency */
|
|
|
|
TIMER_0_PLKSEL();
|
|
|
|
/* set to timer mode */
|
|
|
|
TIMER_0_DEV->CTCR = 0;
|
|
|
|
/* configure prescaler */
|
2015-10-04 00:21:33 +02:00
|
|
|
TIMER_0_DEV->PR = (TIMER_0_FREQ / freq) - 1;
|
2014-12-08 20:48:58 +01:00
|
|
|
/* configure and enable timer interrupts */
|
|
|
|
NVIC_SetPriority(TIMER_0_IRQ, TIMER_IRQ_PRIO);
|
|
|
|
NVIC_EnableIRQ(TIMER_0_IRQ);
|
|
|
|
/* enable timer */
|
|
|
|
TIMER_0_DEV->TCR |= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
2014-11-19 16:31:33 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int timer_set_absolute(tim_t dev, int channel, unsigned int value)
|
|
|
|
{
|
2014-12-08 20:48:58 +01:00
|
|
|
if (dev == TIMER_0) {
|
|
|
|
switch (channel) {
|
|
|
|
case 0:
|
|
|
|
TIMER_0_DEV->MR0 = value;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
TIMER_0_DEV->MR1 = value;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
TIMER_0_DEV->MR2 = value;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
TIMER_0_DEV->MR3 = value;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
TIMER_0_DEV->MCR |= (1 << (channel * 3));
|
|
|
|
return 1;
|
|
|
|
}
|
2014-11-19 16:31:33 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int timer_clear(tim_t dev, int channel)
|
|
|
|
{
|
2014-12-08 20:48:58 +01:00
|
|
|
if (dev == TIMER_0 && channel >= 0 && channel < TIMER_0_CHANNELS) {
|
|
|
|
TIMER_0_DEV->MCR &= ~(1 << (channel * 3));
|
|
|
|
return 1;
|
|
|
|
}
|
2014-11-19 16:31:33 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int timer_read(tim_t dev)
|
|
|
|
{
|
2014-12-08 20:48:58 +01:00
|
|
|
if (dev == TIMER_0) {
|
|
|
|
return (unsigned int)TIMER_0_DEV->TC;
|
|
|
|
}
|
2014-11-19 16:31:33 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void timer_start(tim_t dev)
|
|
|
|
{
|
2014-12-08 20:48:58 +01:00
|
|
|
if (dev == TIMER_0) {
|
|
|
|
TIMER_0_DEV->TCR |= 1;
|
|
|
|
}
|
2014-11-19 16:31:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void timer_stop(tim_t dev)
|
|
|
|
{
|
2014-12-08 20:48:58 +01:00
|
|
|
if (dev == TIMER_0) {
|
|
|
|
TIMER_0_DEV->TCR &= ~(1);
|
|
|
|
}
|
2014-11-19 16:31:33 +01:00
|
|
|
}
|
|
|
|
|
2014-12-08 20:48:58 +01:00
|
|
|
#if TIMER_0_EN
|
|
|
|
void TIMER_0_ISR(void)
|
|
|
|
{
|
|
|
|
if (TIMER_0_DEV->IR & MR0_FLAG) {
|
|
|
|
TIMER_0_DEV->IR |= (MR0_FLAG);
|
|
|
|
TIMER_0_DEV->MCR &= ~(1 << 0);
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_0].cb(config[TIMER_0].arg, 0);
|
2014-12-08 20:48:58 +01:00
|
|
|
}
|
|
|
|
if (TIMER_0_DEV->IR & MR1_FLAG) {
|
|
|
|
TIMER_0_DEV->IR |= (MR1_FLAG);
|
|
|
|
TIMER_0_DEV->MCR &= ~(1 << 3);
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_0].cb(config[TIMER_0].arg, 1);
|
2014-12-08 20:48:58 +01:00
|
|
|
}
|
|
|
|
if (TIMER_0_DEV->IR & MR2_FLAG) {
|
|
|
|
TIMER_0_DEV->IR |= (MR2_FLAG);
|
|
|
|
TIMER_0_DEV->MCR &= ~(1 << 6);
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_0].cb(config[TIMER_0].arg, 2);
|
2014-12-08 20:48:58 +01:00
|
|
|
}
|
|
|
|
if (TIMER_0_DEV->IR & MR3_FLAG) {
|
|
|
|
TIMER_0_DEV->IR |= (MR3_FLAG);
|
|
|
|
TIMER_0_DEV->MCR &= ~(1 << 9);
|
2016-02-17 12:18:24 +01:00
|
|
|
config[TIMER_0].cb(config[TIMER_0].arg, 3);
|
2014-12-08 20:48:58 +01:00
|
|
|
}
|
2016-11-30 18:26:05 +01:00
|
|
|
cortexm_isr_end();
|
2014-11-19 16:31:33 +01:00
|
|
|
}
|
2014-12-08 20:48:58 +01:00
|
|
|
#endif
|
2014-11-19 16:31:33 +01:00
|
|
|
|
|
|
|
#endif /* TIMER_0_EN */
|