1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/sys/include/ztimer64.h
Kaspar Schleiser 439fcbf392 sys/ztimer64: initial implementation
Co-authored-by: Francisco Molina <femolina@uc.cl>
Co-authored-by: Marian Buschsieweke <maribu@users.noreply.github.com>
2021-12-07 23:57:56 +01:00

535 lines
18 KiB
C

/*
* Copyright (C) 2021 Inria
* 2021 Freie Universität Berlin
* 2021 Kaspar Schleiser <kaspar@schleiser.de>
*
* 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
* @defgroup sys_ztimer64 ztimer 64bit version
* @brief ztimer 64bit version
*
* # Introduction
*
* This module implements a 64bit version of the ztimer API.
* All ztimer_* functions have a ztimer64_* equivalent that behaves the same,
* but allowing offsets >= 2^32.
*
* In addition to that, ztimer64 offers functions taking absolute target times
* as parameter.
* If the absolute target time is in the past, a timer would trigger right away.
*
* Here's a table for absolute variants:
*
* | relative version | absolute version |
* |--------------------------------|--------------------------------|
* | ztimer64_set() | ztimer64_set_at() |
* | ztimer64_sleep() | ztimer64_sleep_until() |
* | ztimer64_set_msg() | ztimer64_set_msg_at() |
* | ztimer64_set_wakeup() | ztimer64_set_wakeup_at() |
* | ztimer64_set_timeout_flag() | ztimer64_set_timeout_flag_at() |
* | ztimer64_msg_receive_timeout() | ztimer64_msg_receive_until() |
* | ztimer64_mutex_lock_timeout() | ztimer64_mutex_lock_until() |
* | ztimer64_rmutex_lock_timeout() | ztimer64_rmutex_lock_until() |
*
* ## ztimer64 clocks
*
* ztimer64 provides automatic setup of ZTIMER64_SEC, ZTIMER64_MSEC and
* ZTIMER64_USEC, using the ztimer(32) clocks as base clocks.
* Enable them by depending on ztimer64_sec, ztimer64_msec and / or ztimer64_usec.
*
* @warning ztimer64 always keeps a timer running on the base clock. Depending
* on the base clock, this blocks low-power sleep modes. If the hardware
* supports it, the msec and sec timers should be using a low-power capable
* timer (e.g., periph_rtt). ztimer64_usec will almost certainly block
* low-power sleep.
*
* TODO:
* - some explicit power management
* - implement adjust_set and adjust_sleep API
* - implement setting a 64bit reference time (epoch)
*
* @{
*
* @file
* @brief ztimer 64bit API
*
* @author Kaspar Schleiser <kaspar@schleiser.de>
*/
#ifndef ZTIMER64_H
#define ZTIMER64_H
#include <stdint.h>
#include "mutex.h"
#include "msg.h"
#include "ztimer.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief ztimer64_t forward declaration
*/
typedef struct ztimer64_base ztimer64_base_t;
/**
* @brief ztimer64_clock_t forward declaration
*/
typedef struct ztimer64_clock ztimer64_clock_t;
/**
* @brief Minimum information for each timer
*/
struct ztimer64_base {
ztimer64_base_t *next; /**< next timer in list */
uint64_t target; /**< absolute target time */
};
/**
* @brief ztimer64 structure
*
* This type represents an instance of a timer, which is set on an
* underlying clock object
*/
typedef struct {
ztimer64_base_t base; /**< clock list entry */
void (*callback)(void *arg); /**< timer callback function pointer */
void *arg; /**< timer callback argument */
} ztimer64_t;
/**
* @brief ztimer64 clock structure
*/
struct ztimer64_clock {
ztimer64_base_t *first; /**< list of active timers */
ztimer_clock_t *base_clock; /**< 32bit clock backend */
ztimer_t base_timer; /**< 32bit backend timer */
uint64_t checkpoint; /**< lower timer checkpoint offset */
uint16_t adjust_set; /**< will be subtracted on every set() */
uint16_t adjust_sleep; /**< will be subtracted on every sleep(),
in addition to adjust_set */
#if MODULE_PM_LAYERED || DOXYGEN
uint8_t block_pm_mode; /**< min. pm mode to block for the clock to run */
#endif
};
/**
* @brief Get the current time from a clock
*
* @param[in] clock ztimer clock to operate on
*
* @return Current count on @p clock
*/
uint64_t ztimer64_now(ztimer64_clock_t *clock);
/**
* @brief Get absolute target time for a clock given offset
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] offset relative target time
*
* @returns absolute target time
*/
static inline uint64_t ztimer64_offset2absolute(ztimer64_clock_t *clock,
uint64_t offset)
{
unsigned state = irq_disable();
uint64_t result = ztimer64_now(clock) + offset;
irq_restore(state);
return result;
}
/**
* @brief Set a timer on a clock (absolute version)
*
* This will place @p timer in the timer targets queue of @p clock.
*
* @note The memory pointed to by @p timer is not copied and must
* remain in scope until the callback is fired or the timer
* is removed via @ref ztimer64_remove
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer entry to set
* @param[in] target absolute target time
*/
void ztimer64_set_at(ztimer64_clock_t *clock, ztimer64_t *timer,
uint64_t target);
/**
* @brief Set a timer on a clock (relative version)
*
* This will place @p timer in the timer targets queue of @p clock.
*
* @note The memory pointed to by @p timer is not copied and must
* remain in scope until the callback is fired or the timer
* is removed via @ref ztimer64_remove
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer entry to set
* @param[in] offset relative target time
*/
static inline void ztimer64_set(ztimer64_clock_t *clock, ztimer64_t *timer,
uint64_t offset)
{
ztimer64_set_at(clock, timer,
ztimer64_offset2absolute(clock, offset));
}
/**
* @brief Check if a timer is currently active
*
* @param[in] timer timer to check
*
* @return 1 if timer is active
* @return 0 if timer is not active
*/
unsigned ztimer64_is_set(const ztimer64_t *timer);
/**
* @brief Remove a timer from a clock
*
* This will place @p timer in the timer targets queue for @p clock.
*
* This function does nothing if @p timer is not found in the timer queue of
* @p clock.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer entry to remove
*/
void ztimer64_remove(ztimer64_clock_t *clock, ztimer64_t *timer);
/**
* @brief Post a message at a specified time
*
* This function sets a timer that will send a message at time @p target.
*
* @note The memory pointed to by @p timer and @p msg will not be copied, i.e.
* `*timer` and `*msg` needs to remain valid until the timer has triggered.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer ztimer64 timer struct to use
* @param[in] target absolute target time
* @param[in] msg pointer to msg that will be sent
* @param[in] target_pid pid the message will be sent to
*/
void ztimer64_set_msg_at(ztimer64_clock_t *clock, ztimer64_t *timer,
uint64_t target,
msg_t *msg, kernel_pid_t target_pid);
/**
* @brief Post a message after a delay (relative version)
*
* This function sets a timer that will send a message @p offset ticks
* from now.
*
* @note The memory pointed to by @p timer and @p msg will not be copied, i.e.
* `*timer` and `*msg` needs to remain valid until the timer has triggered.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer ztimer64 timer struct to use
* @param[in] offset relative target time
* @param[in] msg pointer to msg that will be sent
* @param[in] target_pid pid the message will be sent to
*/
static inline void ztimer64_set_msg(ztimer64_clock_t *clock, ztimer64_t *timer,
uint64_t offset,
msg_t *msg, kernel_pid_t target_pid)
{
ztimer64_set_msg_at(clock, timer,
ztimer64_offset2absolute(clock, offset),
msg, target_pid);
}
/**
* @brief receive a message (blocking, with absolute timeout)
*
* Similar to msg_receive(), but with a timeout parameter.
* The function will return after waiting at most until @p target.
*
* @note: This might function might leave a message with type MSG_ZTIMER64 in the
* thread's message queue, which must be handled (ignored).
*
* @param[in] clock ztimer64 clock to operate on
* @param[out] msg pointer to buffer which will be filled if a
* message is received
* @param[in] target absolute target, in @p clock time
*
* @return >=0 if a message was received
* @return -ETIME on timeout
*/
int ztimer64_msg_receive_until(ztimer64_clock_t *clock, msg_t *msg,
uint64_t target);
/**
* @brief receive a message (blocking, with relative timeout)
*
* Similar to msg_receive(), but with a timeout parameter.
* The function will return after waiting at most @p timeout ticks.
*
* @note: This might function might leave a message with type MSG_ZTIMER64 in the
* thread's message queue, which must be handled (ignored).
*
* @param[in] clock ztimer64 clock to operate on
* @param[out] msg pointer to buffer which will be filled if a
* message is received
* @param[in] timeout relative timeout, in @p clock time units
*
* @return >=0 if a message was received
* @return -ETIME on timeout
*/
static inline int ztimer64_msg_receive_timeout(ztimer64_clock_t *clock,
msg_t *msg,
uint64_t timeout)
{
return ztimer64_msg_receive_until(clock, msg,
ztimer64_offset2absolute(clock, timeout));
}
#define MSG_ZTIMER64 0xc83f /**< msg type used by ztimer64_msg_receive_timeout */
/**
* @brief Suspend the calling thread until the time (@p last_wakeup + @p period)
*
* This function can be used to create periodic wakeups.
*
* When the function returns, @p last_wakeup is set to
* (@p last_wakeup + @p period).
*
* @c last_wakeup should be set to ztimer64_now(@p clock) before first call of the
* function.
*
* If the time (@p last_wakeup + @p period) has already passed, the function
* sets @p last_wakeup to @p last_wakeup + @p period and returns immediately.
*
* @note if you period is smaller than 2^32 ticks, consider using the 32bit version.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] last_wakeup base time stamp for the wakeup
* @param[in] period time in ticks that will be added to @p last_wakeup
*/
void ztimer64_periodic_wakeup(ztimer64_clock_t *clock, uint64_t *last_wakeup,
uint64_t period);
/**
* @brief Put the calling thread to sleep until the specified time
*
* @param[in] clock ztimer64 clock to use
* @param[in] target wakeup time, in @p clock time
*/
void ztimer64_sleep_until(ztimer64_clock_t *clock, uint64_t target);
/**
* @brief Put the calling thread to sleep for the specified number of ticks
*
* @param[in] clock ztimer64 clock to use
* @param[in] duration duration of sleep, in @p ztimer time units
*/
static inline void ztimer64_sleep(ztimer64_clock_t *clock, uint64_t duration)
{
ztimer64_sleep_until(clock, ztimer64_offset2absolute(clock, duration));
}
/**
* @brief Busy-wait until specified target time
*
* @note: This blocks lower priority threads. Use only for *very* short delays.
*
* @param[in] clock ztimer64 clock to use
* @param[in] target time when spinning should end, in @p clock time
*/
static inline void ztimer64_spin_until(ztimer64_clock_t *clock, uint64_t target)
{
while (ztimer64_now(clock) <= target) {}
}
/**
* @brief Set a timer that wakes up a thread (absolute version)
*
* This function sets a timer that will wake up a thread when the timer has
* expired.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer struct to work with.
* @param[in] target wakeup time
* @param[in] pid pid of the thread that will be woken up
*/
void ztimer64_set_wakeup_at(ztimer64_clock_t *clock, ztimer64_t *timer,
uint64_t target,
kernel_pid_t pid);
/**
* @brief Set a timer that wakes up a thread (relative version)
*
* This function sets a timer that will wake up a thread when the timer has
* expired.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer struct to work with.
* @param[in] offset clock ticks from now
* @param[in] pid pid of the thread that will be woken up
*/
static inline void ztimer64_set_wakeup(ztimer64_clock_t *clock,
ztimer64_t *timer, uint64_t offset,
kernel_pid_t pid)
{
ztimer64_set_wakeup_at(clock, timer,
ztimer64_offset2absolute(clock, offset), pid);
}
/**
* @brief Set timeout thread flag at @p target time
*
* This function will set THREAD_FLAG_TIMEOUT on the current thread at time
* @p target.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer struct to use
* @param[in] target target in @p clock time
*/
void ztimer64_set_timeout_flag_at(ztimer64_clock_t *clock, ztimer64_t *timer,
uint64_t target);
/**
* @brief Set timeout thread flag after @p timeout
*
* This function will set THREAD_FLAG_TIMEOUT on the current thread after @p
* timeout usec have passed.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] timer timer struct to use
* @param[in] timeout timeout in ztimer64_clock's ticks
*/
static inline void ztimer64_set_timeout_flag(ztimer64_clock_t *clock,
ztimer64_t *timer,
uint64_t timeout)
{
ztimer64_set_timeout_flag_at(clock, timer,
ztimer64_offset2absolute(clock, timeout));
}
/**
* @brief Try to lock the given mutex, but give up at @p target time
*
* @param[in] clock ztimer64 clock to operate on
* @param[in,out] mutex Mutex object to lock
* @param[in] target time after which to give up
*
* @retval 0 Success, caller has the mutex
* @retval -ECANCELED Failed to obtain mutex within @p timeout
*/
int ztimer64_mutex_lock_until(ztimer64_clock_t *clock, mutex_t *mutex,
uint64_t target);
/**
* @brief Try to lock the given mutex, but give up after @p timeout
*
* @param[in] clock ztimer64 clock to operate on
* @param[in,out] mutex Mutex object to lock
* @param[in] timeout timeout after which to give up
*
* @retval 0 Success, caller has the mutex
* @retval -ECANCELED Failed to obtain mutex within @p timeout
*/
static inline int ztimer64_mutex_lock_timeout(ztimer64_clock_t *clock,
mutex_t *mutex,
uint64_t timeout)
{
return ztimer64_mutex_lock_until(clock, mutex,
ztimer64_offset2absolute(clock, timeout));
}
/**
* @brief Try to lock the given rmutex, but give up at @p time
*
* @param[in] clock ztimer64 clock to operate on
* @param[in,out] rmutex rmutex object to lock
* @param[in] target time after which to give up
*
* @retval 0 Success, caller has the rmutex
* @retval -ECANCELED Failed to obtain rmutex before @p time
*/
int ztimer64_rmutex_lock_until(ztimer64_clock_t *clock, rmutex_t *rmutex,
uint64_t target);
/**
* @brief Try to lock the given rmutex, but give up after @p timeout
*
* @param[in] clock ztimer64 clock to operate on
* @param[in,out] rmutex rmutex object to lock
* @param[in] timeout timeout after which to give up
*
* @retval 0 Success, caller has the rmutex
* @retval -ECANCELED Failed to obtain rmutex within @p timeout
*/
static inline int ztimer64_rmutex_lock_timeout(ztimer64_clock_t *clock,
rmutex_t *rmutex,
uint64_t timeout)
{
return ztimer64_rmutex_lock_until(clock, rmutex,
ztimer64_offset2absolute(clock, timeout));
}
/**
* @brief Update ztimer clock head list offset
*
* @internal
*
* @param[in] clock ztimer clock to work on
*/
void ztimer64_update_head_offset(ztimer64_clock_t *clock);
/**
* @brief Initialize the board-specific default ztimer configuration
*/
void ztimer64_init(void);
/**
* @brief Initialize @p clock to be run from @p base_clock
* @param[in,out] clock Clock to initialize
* @param[in] base_clock Base clock to use
*/
void ztimer64_clock_init(ztimer64_clock_t *clock, ztimer_clock_t *base_clock);
/* default ztimer virtual devices */
/**
* @brief Default ztimer microsecond clock
*/
extern ztimer64_clock_t *const ZTIMER64_USEC;
/**
* @brief Default ztimer millisecond clock
*/
extern ztimer64_clock_t *const ZTIMER64_MSEC;
/**
* @brief Default ztimer second clock
*/
extern ztimer64_clock_t *const ZTIMER64_SEC;
/**
* @brief Measure ztimer64 overhead
*
* This function can be used to measure the overhead incurred by ztimer64.
* It will configure a callback to trigger after @p base ticks, then return the
* number of ticks that have passed, minus @p base.
*
* @param[in] clock ztimer64 clock to operate on
* @param[in] base base interval to use
* @return (time from ztimer64_set() until callback) - base
*/
int64_t ztimer64_overhead(ztimer64_clock_t *clock, uint64_t base);
#ifdef __cplusplus
}
#endif
#endif /* ZTIMER64_H */
/** @} */