2018-02-01 09:53:04 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2020 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2020 Freie Universität Berlin
|
|
|
|
* 2020 Inria
|
|
|
|
*
|
|
|
|
* 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 sys_ztimer_periph_rtt
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief ztimer periph/rtt implementation
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
2020-04-22 10:02:17 +02:00
|
|
|
#include "assert.h"
|
2020-07-17 20:10:27 +02:00
|
|
|
|
|
|
|
#include "irq.h"
|
2018-02-01 09:53:04 +01:00
|
|
|
#include "periph/rtt.h"
|
|
|
|
#include "ztimer/periph_rtt.h"
|
|
|
|
|
2020-10-22 11:35:22 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2018-02-01 09:53:04 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static void _ztimer_periph_rtt_callback(void *arg)
|
|
|
|
{
|
|
|
|
ztimer_handler((ztimer_clock_t *)arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _ztimer_periph_rtt_set(ztimer_clock_t *clock, uint32_t val)
|
|
|
|
{
|
2020-06-11 09:30:25 +02:00
|
|
|
if (val < RTT_MIN_OFFSET) {
|
2018-02-01 09:53:04 +01:00
|
|
|
/* the rtt might advance right between the call to rtt_get_counter()
|
|
|
|
* and rtt_set_alarm(). If that happens with val==1, we'd set an alarm
|
|
|
|
* to the current time, which would then underflow. To avoid this, we
|
|
|
|
* set the alarm at least two ticks in the future. TODO: confirm this
|
|
|
|
* is sufficient, or conceive logic to lower this value.
|
|
|
|
*
|
2020-06-11 09:30:25 +02:00
|
|
|
* @note RTT_MIN_OFFSET defaults to 2, but some platforms might have
|
2018-02-01 09:53:04 +01:00
|
|
|
* different values.
|
|
|
|
*/
|
2020-06-11 09:30:25 +02:00
|
|
|
val = RTT_MIN_OFFSET;
|
2018-02-01 09:53:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned state = irq_disable();
|
2020-04-22 10:02:17 +02:00
|
|
|
|
|
|
|
/* ensure RTT_MAX_VALUE is (2^n - 1) */
|
|
|
|
static_assert((RTT_MAX_VALUE == UINT32_MAX) ||
|
|
|
|
!((RTT_MAX_VALUE + 1) & (RTT_MAX_VALUE)),
|
|
|
|
"RTT_MAX_VALUE needs to be (2^n) - 1");
|
|
|
|
|
|
|
|
rtt_set_alarm(
|
|
|
|
(rtt_get_counter() + val) & RTT_MAX_VALUE, _ztimer_periph_rtt_callback,
|
|
|
|
clock);
|
|
|
|
|
2018-02-01 09:53:04 +01:00
|
|
|
irq_restore(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t _ztimer_periph_rtt_now(ztimer_clock_t *clock)
|
|
|
|
{
|
|
|
|
(void)clock;
|
|
|
|
return rtt_get_counter();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _ztimer_periph_rtt_cancel(ztimer_clock_t *clock)
|
|
|
|
{
|
|
|
|
(void)clock;
|
|
|
|
rtt_clear_alarm();
|
|
|
|
}
|
|
|
|
|
|
|
|
static const ztimer_ops_t _ztimer_periph_rtt_ops = {
|
|
|
|
.set = _ztimer_periph_rtt_set,
|
|
|
|
.now = _ztimer_periph_rtt_now,
|
|
|
|
.cancel = _ztimer_periph_rtt_cancel,
|
|
|
|
};
|
|
|
|
|
|
|
|
void ztimer_periph_rtt_init(ztimer_periph_rtt_t *clock)
|
|
|
|
{
|
|
|
|
clock->ops = &_ztimer_periph_rtt_ops;
|
|
|
|
clock->max_value = RTT_MAX_VALUE;
|
|
|
|
rtt_init();
|
|
|
|
rtt_poweron();
|
2020-05-04 15:47:37 +02:00
|
|
|
ztimer_init_extend(clock);
|
2018-02-01 09:53:04 +01:00
|
|
|
}
|