/*
 * 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 "test_utils/expect.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 'expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_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);
    expect(!_triggers);

    _print_result("sizeof(xtimer_t)", NUMOF_TIMERS, sizeof(_timers));

    puts("done.");

    return 0;
}