/** * Copyright (C) 2013 Ludwig Knüpfer * 2015 Kaspar Schleiser * * 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_native * @ingroup drivers_periph_timer * @{ * * @file * @brief Native CPU periph/timer.h implementation * * Uses POSIX realtime clock and POSIX itimer to mimic hardware. * * This is based on native's hwtimer implementation by Ludwig Knüpfer. * I removed the multiplexing, as ztimer does the same. (kaspar) * * @author Ludwig Knüpfer * @author Kaspar Schleiser * * @} */ #include #include #include #include #include #include #include #include "cpu.h" #include "cpu_conf.h" #include "native_internal.h" #include "periph/timer.h" #define ENABLE_DEBUG 0 #include "debug.h" #define NATIVE_TIMER_SPEED 1000000 static unsigned long time_null; static timer_cb_t _callback; static void *_cb_arg; static struct itimerval itv; /** * returns ticks for give timespec */ static unsigned long ts2ticks(struct timespec *tp) { /* TODO: check for overflow */ return (((unsigned long)tp->tv_sec * NATIVE_TIMER_SPEED) + (tp->tv_nsec / 1000)); } /** * native timer signal handler * * set new system timer, call timer interrupt handler */ void native_isr_timer(void) { DEBUG("%s\n", __func__); _callback(_cb_arg, 0); } int timer_init(tim_t dev, uint32_t freq, timer_cb_t cb, void *arg) { (void)freq; DEBUG("%s\n", __func__); if (dev >= TIMER_NUMOF) { return -1; } if (freq != NATIVE_TIMER_SPEED) { return -1; } /* initialize time delta */ time_null = 0; time_null = timer_read(0); _callback = cb; _cb_arg = arg; if (register_interrupt(SIGALRM, native_isr_timer) != 0) { DEBUG("darn!\n\n"); } return 0; } static void do_timer_set(unsigned int offset, bool periodic) { DEBUG("%s\n", __func__); if (offset && offset < NATIVE_TIMER_MIN_RES) { offset = NATIVE_TIMER_MIN_RES; } memset(&itv, 0, sizeof(itv)); itv.it_value.tv_sec = (offset / 1000000); itv.it_value.tv_usec = offset % 1000000; if (periodic) { itv.it_interval = itv.it_value; } DEBUG("timer_set(): setting %lu.%06lu\n", itv.it_value.tv_sec, itv.it_value.tv_usec); } int timer_set(tim_t dev, int channel, unsigned int offset) { DEBUG("%s\n", __func__); if (channel != 0) { return -1; } if (!offset) { offset = NATIVE_TIMER_MIN_RES; } do_timer_set(offset, false); timer_start(dev); return 0; } int timer_set_absolute(tim_t dev, int channel, unsigned int value) { uint32_t now = timer_read(dev); return timer_set(dev, channel, value - now); } int timer_set_periodic(tim_t dev, int channel, unsigned int value, uint8_t flags) { (void)flags; if (channel != 0) { return -1; } do_timer_set(value, true); if (!(flags & TIM_FLAG_SET_STOPPED)) { timer_start(dev); } return 0; } int timer_clear(tim_t dev, int channel) { (void)channel; do_timer_set(0, false); timer_start(dev); return 0; } void timer_start(tim_t dev) { (void)dev; DEBUG("%s\n", __func__); _native_syscall_enter(); if (real_setitimer(ITIMER_REAL, &itv, NULL) == -1) { err(EXIT_FAILURE, "timer_arm: setitimer"); } _native_syscall_leave(); } void timer_stop(tim_t dev) { (void)dev; DEBUG("%s\n", __func__); _native_syscall_enter(); struct itimerval zero = {0}; if (real_setitimer(ITIMER_REAL, &zero, &itv) == -1) { err(EXIT_FAILURE, "timer_arm: setitimer"); } _native_syscall_leave(); DEBUG("time left: %lu.%06lu\n", itv.it_value.tv_sec, itv.it_value.tv_usec); } unsigned int timer_read(tim_t dev) { if (dev >= TIMER_NUMOF) { return 0; } struct timespec t; DEBUG("timer_read()\n"); _native_syscall_enter(); if (clock_gettime(CLOCK_MONOTONIC, &t) == -1) { err(EXIT_FAILURE, "timer_read: clock_gettime"); } _native_syscall_leave(); return ts2ticks(&t) - time_null; }