2019-12-10 10:24:01 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Kaspar Schleiser <kaspar@schleiser.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 tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief xtimer set / remove / now benchmark application
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "msg.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "xtimer.h"
|
|
|
|
|
|
|
|
#ifndef NUMOF_TIMERS
|
|
|
|
#define NUMOF_TIMERS (1000U)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef REPEAT
|
|
|
|
#define REPEAT (1000U)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef BASE
|
|
|
|
#define BASE (100000000LU)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SPREAD
|
|
|
|
#define SPREAD (10000LU)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static xtimer_t _timers[NUMOF_TIMERS];
|
|
|
|
|
|
|
|
/* This variable is set by any timer that actually triggers. As the test is
|
|
|
|
* only testing set/remove/now operations, timers are not supposed to trigger.
|
|
|
|
* Thus, after every test there's an 'assert(!_triggers)'
|
|
|
|
*/
|
|
|
|
static unsigned _triggers;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The test assumes that first, middle and last will always end up in at the
|
|
|
|
* same index within the timer queue. In order to compensate for the time that
|
|
|
|
* previous operations take themselves, the interval is corrected. The
|
|
|
|
* variables "start" and "_base" are used for that.
|
|
|
|
*/
|
|
|
|
uint32_t _base;
|
|
|
|
|
|
|
|
static void _callback(void *arg) {
|
|
|
|
unsigned *triggers = arg;
|
|
|
|
*triggers += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* returns the interval for timer 'n' that has to be set in order to insert it
|
|
|
|
* into position n */
|
|
|
|
static uint32_t _timer_val(unsigned n)
|
|
|
|
{
|
|
|
|
return _base + (SPREAD * n);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set timer 'n' to its intended position 'n' */
|
|
|
|
static void _timer_set(unsigned n)
|
|
|
|
{
|
|
|
|
xtimer_set(&_timers[n], _timer_val(n));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove timer 'n' */
|
|
|
|
static void _timer_remove(unsigned n)
|
|
|
|
{
|
|
|
|
xtimer_remove(&_timers[n]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _print_result(const char *desc, unsigned n, uint32_t total)
|
|
|
|
{
|
|
|
|
printf("%30s %8"PRIu32" / %u = %"PRIu32"\n", desc, total, n, total/n);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
puts("xtimer benchmark application.\n");
|
|
|
|
|
|
|
|
unsigned n;
|
|
|
|
uint32_t before, diff, start;
|
|
|
|
|
|
|
|
/* initializing timer structs */
|
|
|
|
for (unsigned int n = 0; n < NUMOF_TIMERS; n++) {
|
|
|
|
_timers[n].callback = _callback;
|
|
|
|
_timers[n].arg = &_triggers;
|
|
|
|
}
|
|
|
|
|
|
|
|
start = xtimer_now_usec();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test setting one set timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
_base = BASE;
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_set(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("set() one", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test removing one unset timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_remove(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("remove() one", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test setting / removing one timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_set(0);
|
|
|
|
_timer_remove(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("set() + remove() one", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test setting NUMOF_TIMERS timers with increasing targets
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (unsigned int n = 0; n < NUMOF_TIMERS; n++) {
|
|
|
|
_timer_set(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("set() many increasing target", NUMOF_TIMERS, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test re-setting first timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_set(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("re-set() first", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test setting middle timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_set(NUMOF_TIMERS/2);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("re-set() middle", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test setting last timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_set(NUMOF_TIMERS - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("re-set() last", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test removing / setting first timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_remove(0);
|
|
|
|
_timer_set(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("remove() + set() first", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test removing / setting middle timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_remove(NUMOF_TIMERS/2);
|
|
|
|
_timer_set(NUMOF_TIMERS/2);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("remove() + set() middle", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test removing / setting last timer REPEAT times
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
_base = BASE - (before - start);
|
|
|
|
for (n = 0; n < REPEAT; n++) {
|
|
|
|
_timer_remove(NUMOF_TIMERS - 1);
|
|
|
|
_timer_set(NUMOF_TIMERS - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("remove() + set() last", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test removing NUMOF_TIMERS timers (latest first)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
for (n = 0; n < NUMOF_TIMERS; n++) {
|
|
|
|
_timer_remove(NUMOF_TIMERS - n - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("remove() many decreasing", NUMOF_TIMERS, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* test xtimer_now()
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
before = xtimer_now_usec();
|
|
|
|
n = REPEAT;
|
|
|
|
while (n--) {
|
|
|
|
xtimer_now_usec();
|
|
|
|
}
|
|
|
|
|
|
|
|
diff = xtimer_now_usec() - before;
|
|
|
|
|
|
|
|
_print_result("xtimer_now()", REPEAT, diff);
|
|
|
|
assert(!_triggers);
|
|
|
|
|
2020-01-09 11:23:23 +01:00
|
|
|
_print_result("sizeof(xtimer_t)", NUMOF_TIMERS, sizeof(_timers));
|
|
|
|
|
2019-12-10 10:24:01 +01:00
|
|
|
puts("done.");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|