2015-09-04 14:38:37 +02:00
|
|
|
/*
|
2015-09-07 12:05:10 +02:00
|
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
2015-09-04 14:38:37 +02:00
|
|
|
*
|
2015-09-07 12:05:10 +02:00
|
|
|
* 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.
|
2015-09-04 14:38:37 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Peripheral timer test application
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "periph/timer.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Make sure, the maximum number of timers is defined
|
|
|
|
*/
|
|
|
|
#ifndef TIMER_NUMOF
|
|
|
|
#error "TIMER_NUMOF not defined!"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define MAX_CHANNELS (10U)
|
2015-10-04 00:19:56 +02:00
|
|
|
#define TIM_SPEED (1000000ul) /* try to run with 1MHz */
|
2015-09-04 14:38:37 +02:00
|
|
|
#define CHAN_OFFSET (5000U) /* fire every 5ms */
|
2017-11-01 11:38:54 +01:00
|
|
|
#define COOKIE (100U) /* for checking if arg is passed */
|
2015-09-04 14:38:37 +02:00
|
|
|
|
|
|
|
static volatile int fired;
|
|
|
|
static volatile uint32_t sw_count;
|
|
|
|
static volatile uint32_t timeouts[MAX_CHANNELS];
|
2017-11-01 11:38:54 +01:00
|
|
|
static volatile unsigned args[MAX_CHANNELS];
|
2015-09-04 14:38:37 +02:00
|
|
|
|
2016-02-17 12:19:00 +01:00
|
|
|
static void cb(void *arg, int chan)
|
2015-09-04 14:38:37 +02:00
|
|
|
{
|
|
|
|
timeouts[chan] = sw_count;
|
2017-11-01 11:38:54 +01:00
|
|
|
args[chan] = (unsigned)arg + chan;
|
2015-09-04 14:38:37 +02:00
|
|
|
fired++;
|
|
|
|
}
|
|
|
|
|
2015-09-17 13:57:35 +02:00
|
|
|
static int test_timer(unsigned num)
|
2015-09-04 14:38:37 +02:00
|
|
|
{
|
|
|
|
int set = 0;
|
|
|
|
|
|
|
|
/* reset state */
|
|
|
|
sw_count = 0;
|
|
|
|
fired = 0;
|
|
|
|
for (unsigned i = 0; i < MAX_CHANNELS; i++) {
|
|
|
|
timeouts[i] = 0;
|
2017-11-01 11:38:54 +01:00
|
|
|
args[i] = UINT_MAX;
|
2015-09-04 14:38:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* initialize and halt timer */
|
2017-11-01 11:38:54 +01:00
|
|
|
if (timer_init(TIMER_DEV(num), TIM_SPEED, cb, (void *)(COOKIE * num)) < 0) {
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: ERROR on initialization - skipping\n\n", num);
|
2015-09-04 14:38:37 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: initialization successful\n", num);
|
2015-09-04 14:38:37 +02:00
|
|
|
}
|
|
|
|
timer_stop(TIMER_DEV(num));
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: stopped\n", num);
|
2015-09-04 14:38:37 +02:00
|
|
|
/* set each available channel */
|
|
|
|
for (unsigned i = 0; i < MAX_CHANNELS; i++) {
|
|
|
|
unsigned timeout = ((i + 1) * CHAN_OFFSET);
|
|
|
|
if (timer_set(TIMER_DEV(num), i, timeout) < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
++set;
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: set channel %u to %u\n", num, i, timeout);
|
2015-09-04 14:38:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (set == 0) {
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: ERROR setting any channel\n\n", num);
|
2015-09-04 14:38:37 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* start the timer */
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: starting\n", num);
|
2015-09-04 14:38:37 +02:00
|
|
|
timer_start(TIMER_DEV(num));
|
|
|
|
/* wait for all channels to fire */
|
|
|
|
do {
|
|
|
|
++sw_count;
|
|
|
|
} while (fired != set);
|
|
|
|
/* collect results */
|
|
|
|
for (int i = 0; i < fired; i++) {
|
2017-11-01 11:38:54 +01:00
|
|
|
if (args[i] != ((COOKIE * num) + i)) {
|
|
|
|
printf("TIMER_%u: ERROR callback argument mismatch\n\n", num);
|
|
|
|
return 0;
|
|
|
|
}
|
2015-09-17 13:57:35 +02:00
|
|
|
printf("TIMER_%u: channel %i fired at SW count %8u",
|
2015-09-04 14:38:37 +02:00
|
|
|
num, i, (unsigned)timeouts[i]);
|
|
|
|
if (i == 0) {
|
|
|
|
printf(" - init: %8u\n", (unsigned)timeouts[i]);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf(" - diff: %8u\n", (unsigned)(timeouts[i] - timeouts[i - 1]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
int res = 0;
|
|
|
|
|
|
|
|
puts("\nTest for peripheral TIMERs\n");
|
|
|
|
puts("This test will test all configured peripheral timers of the\n"
|
|
|
|
"targeted platform. For each timer, it will set each channel with\n"
|
|
|
|
"an incrementing timeout. CH0 set to 5ms, CH1 to 10ms, CH2 to 15ms\n"
|
|
|
|
"and so on.\n"
|
|
|
|
"In the output you should see that every channel fired, after an\n"
|
|
|
|
"evenly distributed amount of time -> the shown diff values should\n"
|
|
|
|
"be pretty much equal (to some jitter...)\n"
|
|
|
|
"This test does however NOT show, if the timeouts were correct in\n"
|
|
|
|
"relation to the expected real-time ~ use e.g. tests/xtimer_msg for\n"
|
|
|
|
"this.\n\n");
|
|
|
|
|
|
|
|
printf("Available timers: %i\n", TIMER_NUMOF);
|
|
|
|
|
|
|
|
/* test all configured timers */
|
|
|
|
for (unsigned i = 0; i < TIMER_NUMOF; i++) {
|
|
|
|
printf("\nTesting TIMER_%u:\n", i);
|
|
|
|
res += test_timer(i);
|
|
|
|
}
|
|
|
|
/* draw conclusion */
|
|
|
|
if (res == TIMER_NUMOF) {
|
|
|
|
puts("\nTEST SUCCEEDED");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("\nTEST FAILED");
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|