1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-17 05:12:57 +01:00

cpu/rpx0xx: implement periph timer

This commit is contained in:
Fabian Hüßler 2021-07-02 09:17:42 +02:00
parent 6e0c8db99f
commit e89ef368c1
6 changed files with 309 additions and 0 deletions

View File

@ -1,4 +1,5 @@
CPU := rpx0xx
# Put defined MCU peripherals here (in alphabetical order)
FEATURES_PROVIDED += periph_timer
FEATURES_PROVIDED += periph_uart

View File

@ -20,6 +20,7 @@
#include <stdint.h>
#include "kernel_defines.h"
#include "cpu.h"
#include "periph_cpu.h"
@ -47,6 +48,36 @@ static const uart_conf_t uart_config[] = {
#define UART_NUMOF ARRAY_SIZE(uart_config)
static const timer_channel_conf_t timer0_channel_config[] = {
{
.irqn = TIMER_IRQ_0_IRQn
},
{
.irqn = TIMER_IRQ_1_IRQn
},
{
.irqn = TIMER_IRQ_2_IRQn
},
{
.irqn = TIMER_IRQ_3_IRQn
}
};
static const timer_conf_t timer_config[] = {
{
.dev = TIMER,
.ch = timer0_channel_config,
.ch_numof = ARRAY_SIZE(timer0_channel_config)
}
};
#define TIMER_0_ISRA isr_timer0
#define TIMER_0_ISRB isr_timer1
#define TIMER_0_ISRC isr_timer2
#define TIMER_0_ISRD isr_timer3
#define TIMER_NUMOF ARRAY_SIZE(timer_config)
#ifdef __cplusplus
}
#endif

View File

@ -11,6 +11,8 @@ config CPU_FAM_RPX0XX
select HAS_CPU_RPX0XX
select HAS_PERIPH_GPIO
select HAS_PERIPH_GPIO_IRQ
select HAS_PERIPH_TIMER
select HAS_PERIPH_TIMER_PERIODIC
select HAS_PERIPH_UART_MODECFG
select HAS_PERIPH_UART_RECONFIGURE

View File

@ -5,5 +5,6 @@ include $(RIOTCPU)/cortexm_common/Makefile.features
FEATURES_PROVIDED += periph_gpio
FEATURES_PROVIDED += periph_gpio_irq
FEATURES_PROVIDED += periph_timer_periodic
FEATURES_PROVIDED += periph_uart_reconfigure
FEATURES_PROVIDED += periph_uart_modecfg

View File

@ -400,6 +400,29 @@ typedef struct {
IRQn_Type irqn; /**< IRQ number of the UART interface */
} uart_conf_t;
/**
* @brief Prevent shared timer functions from being used
*/
#define PERIPH_TIMER_PROVIDES_SET
/**
* @brief Configuration type of a timer channel
*/
typedef struct {
IRQn_Type irqn; /**< timer channel interrupt number */
} timer_channel_conf_t;
/**
* @brief Configuration type of a timer device @ref timer_conf_t::dev,
* having @ref timer_conf_t::ch_numof number of channels,
* each one modeled as @ref timer_channel_conf_t
*/
typedef struct {
TIMER_Type *dev; /**< pointer to timer base address */
const timer_channel_conf_t *ch; /**< pointer to timer channel configuration */
uint8_t ch_numof; /**< number of timer channels */
} timer_conf_t;
/**
* @brief Get the PAD control register for the given GPIO pin as word
*

251
cpu/rpx0xx/periph/timer.c Normal file
View File

@ -0,0 +1,251 @@
/*
* Copyright (C) 2021 Otto-von-Guericke Universität Magdeburg
*
* 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_rpx0xx
* @ingroup drivers_periph_timer
* @{
*
* @file
* @brief Timer implementation for the RPX0XX
* @details The RPX0XX has a 64 bit µs timer but timer interrupts match
* on the lower 32 bits.
*
* @author Fabian Hüßler <fabian.huessler@ovgu.de>
*
* @}
*/
#include <errno.h>
#include <assert.h>
#include <stdint.h>
#include <stdbool.h>
#include "vendor/RP2040.h"
#include "io_reg.h"
#include "timex.h"
#include "periph_conf.h"
#include "periph/timer.h"
#define DEV(d) (timer_config[d].dev)
#define ALARM(d, a) ((&(DEV(d)->ALARM0)) + (a))
static timer_cb_t _timer_ctx_cb[TIMER_NUMOF];
static void *_timer_ctx_arg[TIMER_NUMOF];
static unsigned _timer_flag_periodic[TIMER_NUMOF];
static unsigned _timer_flag_reset[TIMER_NUMOF];
static inline uint64_t _timer_read_us(tim_t dev)
{
/* This is not safe when the second core also accesses the timer */
unsigned state = irq_disable();
uint32_t lo = DEV(dev)->TIMELR; /* always read timelr to latch the value of timehr */
uint32_t hi = DEV(dev)->TIMEHR; /* read timehr to unlatch */
irq_restore(state);
return ((uint64_t)hi << 32U) | lo;
}
static inline void _timer_reset(tim_t dev)
{
unsigned state = irq_disable();
DEV(dev)->TIMELW = 0; /* always write timelw before timehw */
DEV(dev)->TIMEHW = 0; /* writes do not get copied to time until timehw is written */
irq_restore(state);
}
static inline void _timer_enable_periodic(tim_t dev, int channel, uint8_t flags)
{
_timer_flag_periodic[dev] |= (1U << channel);
if (flags & TIM_FLAG_RESET_ON_MATCH) {
_timer_flag_reset[dev] |= (1U << channel);
}
else {
_timer_flag_reset[dev] &= ~(1U << channel);
}
}
static inline void _timer_disable_periodic(tim_t dev, int channel)
{
_timer_flag_periodic[dev] &= ~(1U << channel);
}
static inline bool _timer_is_periodic(tim_t dev, int channel)
{
return !!(_timer_flag_periodic[dev] & (1U << channel));
}
static inline bool _timer_reset_on_match(tim_t dev, int channel)
{
return !!(_timer_flag_reset[dev] & (1U << channel));
}
static inline void _irq_enable(tim_t dev)
{
for (uint8_t i = 0; i < timer_config[dev].ch_numof; i++) {
NVIC_EnableIRQ(timer_config[dev].ch[i].irqn);
io_reg_atomic_set(&DEV(dev)->INTE.reg, (1U << i));
}
}
static void _isr(tim_t dev, int channel)
{
/* clear latched interrupt */
io_reg_atomic_clear(&DEV(dev)->INTR.reg, 1U << channel);
if (_timer_is_periodic(dev, channel)) {
if (_timer_reset_on_match(dev, channel)) {
_timer_reset(dev);
}
/* rearm */
*ALARM(dev, channel) = *ALARM(dev, channel);
}
if (_timer_ctx_cb[dev]) {
_timer_ctx_cb[dev](_timer_ctx_arg[dev], channel);
}
cortexm_isr_end();
}
int timer_init(tim_t dev, uint32_t freq, timer_cb_t cb, void *arg)
{
if (dev >= TIMER_NUMOF) {
return -ENODEV;
}
/* The timer must run at 1000000 Hz (µs precision)
because the number of cycles per µs is shared with the watchdog.
The reference clock (clk_ref) is divided by WATCHDOG->TICK.bits.CYCLES
to generate µs ticks.
*/
assert(freq == US_PER_SEC); (void)freq;
_timer_ctx_cb[dev] = cb;
_timer_ctx_arg[dev] = arg;
periph_reset(RESETS_RESET_timer_Msk);
periph_reset_done(RESETS_RESET_timer_Msk);
io_reg_write_dont_corrupt(&WATCHDOG->TICK.reg,
(CLOCK_XOSC / MHZ(1)) << WATCHDOG_TICK_CYCLES_Pos,
WATCHDOG_TICK_CYCLES_Msk);
_irq_enable(dev);
return 0;
}
int timer_set(tim_t dev, int channel, unsigned int timeout)
{
if (dev >= TIMER_NUMOF) {
return -ENODEV;
}
if (channel < 0 || channel >= timer_config[dev].ch_numof) {
return -EINVAL;
}
if (!timeout) {
/* execute callback immediately if timeout equals 0,
to ctach the case that a tick happens right before arming the alarm
and causes a full timer period to elaps */
if (_timer_ctx_cb[dev]) {
_timer_ctx_cb[dev](_timer_ctx_arg[dev], channel);
}
}
else {
unsigned state = irq_disable();
_timer_disable_periodic(dev, channel);
/* an alarm interrupt matches on the lower 32 bit of the 64 bit timer counter */
uint64_t target = DEV(dev)->TIMERAWL + timeout;
*ALARM(dev, channel) = (uint32_t)target;
irq_restore(state);
}
return 0;
}
int timer_set_absolute(tim_t dev, int channel, unsigned int value)
{
if (dev >= TIMER_NUMOF) {
return -ENODEV;
}
if (channel < 0 || channel >= timer_config[dev].ch_numof) {
return -EINVAL;
}
unsigned state = irq_disable();
_timer_disable_periodic(dev, channel);
*ALARM(dev, channel) = (uint32_t)value;
irq_restore(state);
return 0;
}
int timer_set_periodic(tim_t dev, int channel, unsigned int value, uint8_t flags)
{
if (dev >= TIMER_NUMOF) {
return -ENODEV;
}
if (channel < 0 || channel >= timer_config[dev].ch_numof) {
return -EINVAL;
}
if (flags & TIM_FLAG_RESET_ON_SET) {
_timer_reset(dev);
}
unsigned state = irq_disable();
_timer_enable_periodic(dev, channel, flags);
*ALARM(dev, channel) = (uint32_t)value;
irq_restore(state);
return 0;
}
int timer_clear(tim_t dev, int channel)
{
if (dev >= TIMER_NUMOF) {
return -ENODEV;
}
if (channel < 0 || channel >= timer_config[dev].ch_numof) {
return -EINVAL;
}
/* ARMED bits are write clear */
io_reg_atomic_set(&DEV(dev)->ARMED.reg, (1 << channel));
unsigned state = irq_disable();
_timer_disable_periodic(dev, channel);
irq_restore(state);
return 0;
}
unsigned int timer_read(tim_t dev)
{
if (dev >= TIMER_NUMOF) {
return -ENODEV;
}
return _timer_read_us(dev);
}
void timer_start(tim_t dev)
{
assert(dev < TIMER_NUMOF);
io_reg_atomic_clear(&DEV(dev)->PAUSE.reg, (1 << TIMER_PAUSE_PAUSE_Pos));
}
void timer_stop(tim_t dev)
{
assert(dev < TIMER_NUMOF);
io_reg_atomic_set(&DEV(dev)->PAUSE.reg, (1 << TIMER_PAUSE_PAUSE_Pos));
}
/* timer 0 IRQ0 */
void TIMER_0_ISRA(void)
{
_isr(0, 0);
}
/* timer 0 IRQ1 */
void TIMER_0_ISRB(void)
{
_isr(0, 1);
}
/* timer 0 IRQ2 */
void TIMER_0_ISRC(void)
{
_isr(0, 2);
}
/* timer 0 IRQ3 */
void TIMER_0_ISRD(void)
{
_isr(0, 3);
}