/* * Copyright (C) 2016 Kaspar Schleiser * 2018 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 event test application * * @author Kaspar Schleiser * @author Hauke Petersen * * @} */ #include #include "test_utils/expect.h" #include "timex.h" #include "thread.h" #include "event.h" #include "event/timeout.h" #include "event/callback.h" #if IS_USED(MODULE_ZTIMER_USEC) #include "ztimer.h" #else #include "xtimer.h" #endif #define STACKSIZE THREAD_STACKSIZE_DEFAULT /* in order to actually test @ref event_sync(), the waiter's prio should be lower * than main s.t. it doesn't start executing right after events are enqueued */ #define PRIO (THREAD_PRIORITY_MAIN + 1) #define DELAYED_QUEUES_NUMOF 2 static char stack[STACKSIZE]; static unsigned order = 0; static uint32_t before; static void callback(event_t *arg); static void custom_callback(event_t *event); static void timed_callback(void *arg); static void forbidden_callback(void *arg); static void delayed_callback1(event_t *arg); static void delayed_callback2(event_t *arg); static void delayed_callback3(event_t *arg); static event_t event = { .handler = callback }; static event_t event2 = { .handler = callback }; static event_t delayed_event1 = { .handler = delayed_callback1 }; static event_t delayed_event2 = { .handler = delayed_callback2 }; static event_t delayed_event3 = { .handler = delayed_callback3 }; static void callback(event_t *arg) { order++; expect(order == 4); expect(arg == &event); printf("triggered 0x%08" PRIxPTR "\n", (uintptr_t)arg); } typedef struct { event_t super; const char *text; } custom_event_t; static custom_event_t custom_event = { .super.handler = custom_callback, .text = "CUSTOM CALLBACK" }; static event_callback_t* event_callback_ptr; static event_callback_t noevent_callback = EVENT_CALLBACK_INIT(forbidden_callback, 0); static void custom_callback(event_t *event) { order++; expect(order == 5); expect(event == (event_t *)&custom_event); custom_event_t *custom_event = (custom_event_t *)event; printf("triggered custom event with text: \"%s\"\n", custom_event->text); } static void timed_callback(void *arg) { order++; expect(order == 6); expect(arg == event_callback_ptr->arg); #if IS_USED(MODULE_ZTIMER_USEC) uint32_t now = ztimer_now(ZTIMER_USEC); #else uint32_t now = xtimer_now_usec(); #endif expect((now - before >= 100000LU)); printf("triggered timed callback with arg 0x%08" PRIxPTR " after %" PRIu32 "us\n", (uintptr_t)arg, now - before); puts("[SUCCESS]"); } static void forbidden_callback(void *arg) { (void)arg; /* this callback should never be triggered! */ puts("call to forbidden callback"); puts("[FAILED]"); while (1) { expect(false); } } static void delayed_callback1(event_t *arg) { order++; expect(order == 2); expect(arg == &delayed_event1); printf("triggered delayed event %p\n", (void *)arg); } static void delayed_callback2(event_t *arg) { order++; expect(order == 3); expect(arg == &delayed_event2); printf("triggered delayed event %p\n", (void *)arg); } static void delayed_callback3(event_t *arg) { order++; expect(order == 1); expect(arg == &delayed_event3); printf("triggered delayed event %p\n", (void *)arg); } static void *claiming_thread(void *arg) { event_queue_t *dqs = arg; printf("claiming event queues %p\n", (void *)dqs); event_queues_claim(dqs, DELAYED_QUEUES_NUMOF); printf("launching event queue for queues %p\n", (void *)dqs); event_loop_multi(dqs, DELAYED_QUEUES_NUMOF); return NULL; } int main(void) { puts("[START] event test application.\n"); /* initialize an event callback */ event_callback_t event_callback; event_callback_ptr = &event_callback; event_callback_init(&event_callback, timed_callback, (void*)0x12345678); /* test creation of delayed claiming of a detached event queue */ event_queue_t dqs[DELAYED_QUEUES_NUMOF] = { EVENT_QUEUE_INIT_DETACHED, EVENT_QUEUE_INIT_DETACHED }; printf("initializing detached event queues %p\n", (void *)dqs); event_queue_init_detached(&dqs[0]); event_queue_init_detached(&dqs[1]); printf("running thread that will claim event queues %p\n", (void *)&dqs); thread_create(stack, sizeof(stack), PRIO, 0, claiming_thread, dqs, "ct"); printf("posting %p to delayed queue at index 1\n", (void *)&delayed_event1); event_post(&dqs[1], &delayed_event1); printf("posting %p to delayed queue at index 1\n", (void *)&delayed_event2); event_post(&dqs[1], &delayed_event2); printf("posting %p to delayed queue at index 0\n", (void *)&delayed_event3); event_post(&dqs[0], &delayed_event3); event_sync(&dqs[1]); expect(order == 3); printf("synced with %p\n", (void *)&delayed_event3); /* test posting different kind of events in order to a statically * initialized queue */ event_queue_t queue = EVENT_QUEUE_INIT; printf("posting 0x%08" PRIxPTR "\n", (uintptr_t)&event); event_post(&queue, &event); printf("posting 0x%08" PRIxPTR "\n", (uintptr_t)&event2); event_post(&queue, &event2); printf("canceling 0x%08" PRIxPTR "\n", (uintptr_t)&event2); event_cancel(&queue, &event2); puts("posting custom event"); event_post(&queue, (event_t *)&custom_event); event_timeout_t event_timeout; puts("posting timed callback with timeout 1sec"); event_timeout_init(&event_timeout, &queue, (event_t *)&event_callback); #if IS_USED(MODULE_ZTIMER_USEC) before = ztimer_now(ZTIMER_USEC); #else before = xtimer_now_usec(); #endif event_timeout_set(&event_timeout, (1 * US_PER_SEC)); expect(event_timeout_is_pending(&event_timeout)); event_timeout_t event_timeout_canceled; puts("posting timed callback with timeout 0.5sec and canceling it again"); event_timeout_init(&event_timeout_canceled, &queue, (event_t *)&noevent_callback); event_timeout_set(&event_timeout_canceled, 500 * US_PER_MS); event_timeout_clear(&event_timeout_canceled); expect(!event_timeout_is_pending(&event_timeout_canceled)); puts("launching event queue"); event_loop(&queue); return 0; }