1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/tests/event_wait_timeout/main.c
Marian Buschsieweke 60aabbec25
tests/event_wait_timeout: Use C11 atomics for IPC
The test currently uses static globals for communication without any protection.
This could lead optimizing C compilers to deduce that access to those are not
needed (e.g. with LTO enabled). Using C11 atomics is the easiest way to tell
the compiler that those accesses are used for communication between two
different threads of execution (here: between the ISR and the main thread).
2020-04-23 21:16:03 +02:00

143 lines
3.7 KiB
C

/*
* Copyright (C) 2019 Freie Universität Berlin
*
* 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 application to test event wait timeout
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*
* @}
*/
#include <stdint.h>
#include <stdatomic.h>
#include <stdio.h>
#include "event.h"
#include "thread.h"
#include "xtimer.h"
#include "thread_flags.h"
#define STATIC_TIMEOUT (10U * US_PER_MS) /* 10ms */
#define TIMEOUT (50U * US_PER_MS) /* 50ms */
#define PRIO (THREAD_PRIORITY_MAIN - 5)
#define STACKSIZE (THREAD_STACKSIZE_DEFAULT)
#define FLAG_SYNC (0x0040)
static void _on_evt(event_t *evt);
static event_queue_t _evtq;
static event_t _evt = { .handler = _on_evt };
static char _stack[STACKSIZE];
static thread_t *_thread_main;
static atomic_uint _wakeup_evt = ATOMIC_VAR_INIT(0);
static atomic_uint _wakeup_timeout = ATOMIC_VAR_INIT(0);
static void _on_evt(event_t *evt)
{
(void)evt;
atomic_fetch_add(&_wakeup_evt, 1);
}
static void *_cnt_thread(void *arg)
{
(void)arg;
event_queue_init(&_evtq);
while (1) {
event_t *evt = event_wait_timeout(&_evtq, TIMEOUT);
if (evt) {
evt->handler(evt);
}
else {
atomic_fetch_add(&_wakeup_timeout, 1);
}
}
return NULL;
}
int main(void)
{
/* setup */
_thread_main = (thread_t *)thread_get(thread_getpid());
puts("[START] event_wait_timeout test application.\n");
/* test 'instant' return */
event_queue_t tmp_eq = EVENT_QUEUE_INIT;
event_t *tmp_evt = event_wait_timeout(&tmp_eq, 0);
if (tmp_evt != NULL) {
puts("[FAILED]");
return 1;
}
tmp_evt = event_wait_timeout64(&tmp_eq, 0);
if (tmp_evt != NULL) {
puts("[FAILED]");
return 1;
}
/* test return in a predefined amount of time */
puts("waiting for event with 10ms timeout...");
uint32_t before = xtimer_now_usec();
tmp_evt = event_wait_timeout(&tmp_eq, STATIC_TIMEOUT);
if (tmp_evt != NULL) {
puts("[FAILED]");
return 1;
}
uint32_t diff = xtimer_now_usec() - before;
printf("event_wait time out after %"PRIu32"us\n", diff);
puts("waiting for event with 10ms timeout (using uint64)...");
uint64_t static_timeout = STATIC_TIMEOUT;
before = xtimer_now_usec();
tmp_evt = event_wait_timeout64(&tmp_eq, static_timeout);
if (tmp_evt != NULL) {
puts("[FAILED]");
return 1;
}
diff = xtimer_now_usec() - before;
printf("event_wait time out after %"PRIu32"us\n", diff);
thread_create(_stack, sizeof(_stack), PRIO, 0, _cnt_thread, NULL, "cnt");
/* first, wait 155ms -> should lead to 3 timeout wakeups */
xtimer_usleep(155U * US_PER_MS);
/* post event 3 times -> should lead to 3 event wakeups */
for (unsigned i = 0; i < 3; i++) {
event_post(&_evtq, &_evt);
xtimer_usleep(5U * US_PER_MS);
}
/* wait for 35ms and post another event -> +1 event wakeup */
xtimer_usleep(35U * US_PER_MS);
event_post(&_evtq, &_evt);
/* finally, wait 60ms and collect results -> +1 timeout wakeup */
xtimer_usleep(60U * US_PER_MS);
unsigned events = atomic_load(&_wakeup_evt);
unsigned timeouts = atomic_load(&_wakeup_timeout);
/* rate results */
printf("finished: %u/4 events and %u/4 timeouts recorded\n",
events, timeouts);
if ((events == 4) && (timeouts == 4)) {
puts("[SUCCESS]");
}
else {
puts("[FAILED]");
}
return 0;
}