1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/native/periph/rtc.c
Benjamin Valentin bdc43c245c cpu/native: RTC: implement rtc_set_alarm()
Use xtimer to simulate an RTC timer.
This allows to simulate software that makes use of `rtc_set_alarm()`
on native.
2020-08-16 23:28:30 +02:00

191 lines
3.8 KiB
C

/*
* Copyright (C) 2013, 2014 Ludwig Knüpfer <ludwig.knuepfer@fu-berlin.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 cpu_native
* @ingroup drivers_periph_rtc
* @{
*
* @file
* @brief Native CPU periph/rtc.h implementation
*
* The implementation uses POSIX system calls to emulate a real-time
* clock based on the system clock.
*
* @author Ludwig Knüpfer <ludwig.knuepfer@fu-berlin.de>
*
* @}
*/
#include <time.h>
#include <stdlib.h>
#include <string.h>
#include <err.h>
#include "periph/rtc.h"
#include "cpu.h"
#include "xtimer.h"
#include "native_internal.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
static int _native_rtc_initialized = 0;
static int _native_rtc_powered = 0;
static struct tm _native_rtc_alarm;
static rtc_alarm_cb_t _native_rtc_alarm_callback;
static void *_native_rtc_alarm_argument;
static xtimer_t _timer;
void rtc_init(void)
{
DEBUG("rtc_init\n");
memset(&_native_rtc_alarm, 0, sizeof(_native_rtc_alarm));
_native_rtc_alarm_callback = NULL;
_native_rtc_alarm_argument = NULL;
_native_rtc_initialized = 1;
printf("Native RTC initialized.\n");
rtc_poweron();
}
void rtc_poweron(void)
{
DEBUG("rtc_poweron\n");
if (!_native_rtc_initialized) {
warnx("rtc_poweron: not initialized");
return;
}
_native_rtc_powered = 1;
}
void rtc_poweroff(void)
{
DEBUG("rtc_poweroff()\n");
if (!_native_rtc_initialized) {
warnx("rtc_poweroff: not initialized");
}
if (!_native_rtc_powered) {
warnx("rtc_poweroff: not powered on");
}
_native_rtc_powered = 0;
}
/* TODO: implement time setting using a delta */
int rtc_set_time(struct tm *ttime)
{
(void) ttime;
DEBUG("rtc_set_time()\n");
if (!_native_rtc_initialized) {
warnx("rtc_set_time: not initialized");
return -1;
}
if (!_native_rtc_powered) {
warnx("rtc_set_time: not powered on");
return -1;
}
warnx("rtc_set_time: not implemented");
return -1;
}
int rtc_get_time(struct tm *ttime)
{
time_t t;
if (!_native_rtc_initialized) {
warnx("rtc_get_time: not initialized");
return -1;
}
if (!_native_rtc_powered) {
warnx("rtc_get_time: not powered on");
return -1;
}
_native_syscall_enter();
t = time(NULL);
if (localtime_r(&t, ttime) == NULL) {
err(EXIT_FAILURE, "rtc_get_time: localtime_r");
}
_native_syscall_leave();
return 0;
}
int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
{
(void) time;
(void) cb;
(void) arg;
if (!_native_rtc_initialized) {
warnx("rtc_set_alarm: not initialized");
return -1;
}
if (!_native_rtc_powered) {
warnx("rtc_set_alarm: not powered on");
return -1;
}
struct tm now;
rtc_get_time(&now);
_native_rtc_alarm = *time;
_timer.callback = cb;
_timer.arg = arg;
xtimer_set64(&_timer, (mktime(time) - mktime(&now)) * US_PER_SEC);
return 0;
}
int rtc_get_alarm(struct tm *time)
{
(void) time;
if (!_native_rtc_initialized) {
warnx("rtc_get_alarm: not initialized");
return -1;
}
if (!_native_rtc_powered) {
warnx("rtc_get_alarm: not powered on");
return -1;
}
*time = _native_rtc_alarm;
return 0;
}
void rtc_clear_alarm(void)
{
DEBUG("rtc_clear_alarm()\n");
if (!_native_rtc_initialized) {
warnx("rtc_clear_alarm: not initialized");
}
if (!_native_rtc_powered) {
warnx("rtc_clear_alarm: not powered on");
}
xtimer_remove(&_timer);
memset(&_native_rtc_alarm, 0, sizeof(_native_rtc_alarm));
}