1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 11:32:45 +01:00
RIOT/tests/periph/rtc/main.c

247 lines
5.6 KiB
C

/*
* Copyright (C) 2015 Lari Lehtomäki
*
* 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 tests
* @{
*
* @file
* @brief Test for low-level Real Time clock drivers
*
* This test will initialize the real-time timer and trigger an alarm printing
* 'Hello' every 10 seconds
*
* @author Lari Lehtomäki <lari@lehtomaki.fi>
*
* @}
*/
#include <stdio.h>
#include <time.h>
#include <string.h>
#include "mutex.h"
#include "periph_conf.h"
#include "periph/rtc.h"
#include "periph/rtc_mem.h"
#include "xtimer.h"
#define PERIOD (2U)
#define REPEAT (4U)
#define TM_YEAR_OFFSET (1900)
static unsigned cnt = 0;
static void print_time(const char *label, const struct tm *time)
{
printf("%s %04d-%02d-%02d %02d:%02d:%02d\n", label,
time->tm_year + TM_YEAR_OFFSET,
time->tm_mon + 1,
time->tm_mday,
time->tm_hour,
time->tm_min,
time->tm_sec);
}
static void print_time_ms(const char *label, const struct tm *time, uint16_t ms)
{
printf("%s %04d-%02d-%02d %02d:%02d:%02d.%03d\n", label,
time->tm_year + TM_YEAR_OFFSET,
time->tm_mon + 1,
time->tm_mday,
time->tm_hour,
time->tm_min,
time->tm_sec,
ms);
}
static void inc_secs(struct tm *time, unsigned val)
{
time->tm_sec += val;
}
static void cb(void *arg)
{
mutex_unlock(arg);
}
#ifdef MODULE_PERIPH_RTC_MEM
static const uint8_t riot_msg_offset = 1;
static const char riot_msg[] = "RIOT";
static void _set_rtc_mem(void)
{
/* first fill the whole memory */
uint8_t size = rtc_mem_size();
while (size--) {
rtc_mem_write(size, &size, sizeof(size));
}
/* write test data */
rtc_mem_write(riot_msg_offset, riot_msg, sizeof(riot_msg) - 1);
}
static void _get_rtc_mem(void)
{
char buf[4];
rtc_mem_read(riot_msg_offset, buf, sizeof(buf));
if (memcmp(buf, riot_msg, sizeof(buf))) {
puts("RTC mem content does not match");
for (unsigned i = 0; i < sizeof(buf); ++i) {
printf("%02x - %02x\n", riot_msg[i], buf[i]);
}
return;
}
uint8_t size = rtc_mem_size();
while (size--) {
uint8_t data;
if (size >= riot_msg_offset &&
size < riot_msg_offset + sizeof(riot_msg)) {
continue;
}
rtc_mem_read(size, &data, 1);
if (data != size) {
puts("RTC mem content does not match");
printf("%02x: %02x\n", size, data);
}
}
puts("RTC mem OK");
}
#else
static inline void _set_rtc_mem(void) {}
static inline void _get_rtc_mem(void) {}
#endif
int main(void)
{
struct tm time = (struct tm){0};
uint16_t ms;
mutex_t rtc_mtx = MUTEX_INIT_LOCKED;
puts("\nRIOT RTC low-level driver test");
printf("This test will display 'Alarm!' every %u seconds for %u times\n",
PERIOD, REPEAT);
rtc_init();
_set_rtc_mem();
_get_rtc_mem();
/* read RTC to retrieve initial */
if (IS_USED(MODULE_PERIPH_RTC_MS)) {
rtc_get_time_ms(&time, &ms);
print_time_ms("Clock value is now ", &time, ms);
} else {
rtc_get_time(&time);
print_time("Clock value is now ", &time);
}
time = (struct tm){
.tm_year = 2020 - TM_YEAR_OFFSET, /* years are counted from 1900 */
.tm_mon = 1, /* 0 = January, 11 = December */
.tm_mday = 28,
.tm_hour = 23,
.tm_min = 59,
.tm_sec = 57
};
/* set RTC */
print_time(" Setting clock to ", &time);
rtc_set_time(&time);
time = (struct tm){0};
ms = 0;
/* read RTC to confirm value */
if (IS_USED(MODULE_PERIPH_RTC_MS)) {
rtc_get_time_ms(&time, &ms);
print_time_ms("Clock value is now ", &time, ms);
} else {
rtc_get_time(&time);
print_time("Clock value is now ", &time);
}
/* set initial alarm */
inc_secs(&time, PERIOD);
print_time(" Setting alarm to ", &time);
rtc_set_alarm(&time, cb, &rtc_mtx);
time = (struct tm){0};
ms = 0;
/* verify alarm */
rtc_get_alarm(&time);
print_time(" Alarm is set to ", &time);
time = (struct tm){0};
ms = 0;
/* clear alarm */
rtc_clear_alarm();
if (IS_USED(MODULE_PERIPH_RTC_MS)) {
rtc_get_time_ms(&time, &ms);
print_time_ms(" Alarm cleared at ", &time, ms);
} else {
rtc_get_time(&time);
print_time(" Alarm cleared at ", &time);
}
/* verify alarm has been cleared */
xtimer_sleep(PERIOD);
const char *message;
if (mutex_trylock(&rtc_mtx)) {
message = " Error: Alarm at ";
} else {
message = " No alarm at ";
}
time = (struct tm){0};
ms = 0;
if (IS_USED(MODULE_PERIPH_RTC_MS)) {
rtc_get_time_ms(&time, &ms);
print_time_ms(message, &time, ms);
} else {
rtc_get_time(&time);
print_time(message, &time);
}
time = (struct tm){0};
ms = 0;
/* set alarm */
rtc_get_time(&time);
inc_secs(&time, PERIOD);
rtc_set_alarm(&time, cb, &rtc_mtx);
print_time(" Setting alarm to ", &time);
puts("");
/* loop over a few alarm cycles */
do {
mutex_lock(&rtc_mtx);
puts("Alarm!");
struct tm time;
rtc_get_alarm(&time);
inc_secs(&time, PERIOD);
rtc_set_alarm(&time, cb, &rtc_mtx);
} while (++cnt < REPEAT);
_get_rtc_mem();
return 0;
}