2017-02-20 18:23:20 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 OTA keys S.A.
|
2021-09-28 12:27:50 +02:00
|
|
|
* 2021 Otto-von-Guericke-Universität Magdeburg
|
2017-02-20 18:23:20 +01: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Thread test application
|
|
|
|
*
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
2021-09-28 12:27:50 +02:00
|
|
|
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
2017-02-20 18:23:20 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2021-09-28 12:27:50 +02:00
|
|
|
#include <string.h>
|
2017-02-20 18:23:20 +01:00
|
|
|
|
2022-09-23 13:49:45 +02:00
|
|
|
#include "board.h"
|
|
|
|
#include "clk.h"
|
|
|
|
#include "macros/units.h"
|
2017-02-20 18:23:20 +01:00
|
|
|
#include "msg.h"
|
2022-09-23 13:49:45 +02:00
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "time_units.h"
|
2022-03-22 08:30:50 +01:00
|
|
|
#include "ztimer.h"
|
2017-02-20 18:23:20 +01:00
|
|
|
|
|
|
|
static char t1_stack[THREAD_STACKSIZE_MAIN];
|
|
|
|
static char t2_stack[THREAD_STACKSIZE_MAIN];
|
|
|
|
static char t3_stack[THREAD_STACKSIZE_MAIN];
|
|
|
|
|
|
|
|
static kernel_pid_t p1, p2, p3;
|
|
|
|
|
2022-03-22 08:30:50 +01:00
|
|
|
static ztimer_t timer;
|
|
|
|
|
2017-02-20 18:23:20 +01:00
|
|
|
static mutex_t lock = MUTEX_INIT;
|
|
|
|
|
|
|
|
static void timer_cb(void *arg)
|
|
|
|
{
|
2022-09-23 13:49:45 +02:00
|
|
|
uint32_t *timeout = arg;
|
2017-02-20 18:23:20 +01:00
|
|
|
thread_yield();
|
2022-09-23 13:49:45 +02:00
|
|
|
ztimer_set(ZTIMER_USEC, &timer, *timeout);
|
2017-02-20 18:23:20 +01:00
|
|
|
}
|
|
|
|
|
2021-09-28 12:27:50 +02:00
|
|
|
static void *thread_1_2_3(void *_arg)
|
2017-02-20 18:23:20 +01:00
|
|
|
{
|
2021-09-28 12:27:50 +02:00
|
|
|
const char *arg = _arg;
|
2017-02-20 18:23:20 +01:00
|
|
|
float f, init;
|
|
|
|
|
2021-09-28 12:27:50 +02:00
|
|
|
mutex_lock(&lock);
|
|
|
|
printf("THREAD %s start\n", arg);
|
|
|
|
mutex_unlock(&lock);
|
2017-02-20 18:23:20 +01:00
|
|
|
|
2021-09-28 12:27:50 +02:00
|
|
|
/* Use number at end of thread name, e.g. 3 for "t3", to seed the calculation */
|
|
|
|
init = 1.0 * (arg[strlen(arg) - 1] - '0');
|
2017-02-20 18:23:20 +01:00
|
|
|
f = init;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
for (unsigned long i = 0; i < 10000ul; i++) {
|
|
|
|
f = f + 1.0 / f;
|
|
|
|
}
|
2021-09-28 12:27:50 +02:00
|
|
|
/* only t1 and t3 should print */
|
|
|
|
if (strcmp("t2", arg) != 0) {
|
|
|
|
mutex_lock(&lock);
|
|
|
|
printf("%s: %f\n", arg, (double)f);
|
|
|
|
mutex_unlock(&lock);
|
2017-02-20 18:23:20 +01:00
|
|
|
}
|
|
|
|
f = init;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
2021-09-28 12:27:50 +02:00
|
|
|
const char *t1_name = "t1";
|
|
|
|
const char *t2_name = "t2";
|
|
|
|
const char *t3_name = "t3";
|
2022-03-22 08:40:53 +01:00
|
|
|
|
2022-09-23 13:49:45 +02:00
|
|
|
/* Let's not overwhelm boards by firing IRQs faster than they can handle and
|
|
|
|
* give them 50 billion CPU cycles per timeout.
|
|
|
|
*
|
|
|
|
* (Note: The `static` is required as this variable will be accessed from
|
|
|
|
* the ISR, which will occur even after the main thread has exited.) */
|
|
|
|
static uint32_t timeout = 0;
|
|
|
|
/* Note: It must be initialized dynamically, as coreclk() is not
|
|
|
|
* constant. */
|
|
|
|
timeout = 50000000000U / coreclk();
|
|
|
|
|
2017-02-20 18:23:20 +01:00
|
|
|
p1 = thread_create(t1_stack, sizeof(t1_stack), THREAD_PRIORITY_MAIN + 1,
|
2024-03-07 15:51:39 +01:00
|
|
|
THREAD_CREATE_WOUT_YIELD,
|
2021-09-28 12:27:50 +02:00
|
|
|
thread_1_2_3, (void *)t1_name, t1_name);
|
2017-02-20 18:23:20 +01:00
|
|
|
p2 = thread_create(t2_stack, sizeof(t2_stack), THREAD_PRIORITY_MAIN + 1,
|
2024-03-07 15:51:39 +01:00
|
|
|
THREAD_CREATE_WOUT_YIELD,
|
2021-09-28 12:27:50 +02:00
|
|
|
thread_1_2_3, (void *)t2_name, t2_name);
|
2017-02-20 18:23:20 +01:00
|
|
|
p3 = thread_create(t3_stack, sizeof(t3_stack), THREAD_PRIORITY_MAIN + 1,
|
2024-03-07 15:51:39 +01:00
|
|
|
THREAD_CREATE_WOUT_YIELD,
|
2021-09-28 12:27:50 +02:00
|
|
|
thread_1_2_3, (void *)t3_name, t3_name);
|
2017-02-20 18:23:20 +01:00
|
|
|
puts("THREADS CREATED\n");
|
|
|
|
|
2022-09-23 13:49:45 +02:00
|
|
|
printf("Context switch every %" PRIu32 " µs\n", timeout);
|
|
|
|
|
2017-02-20 18:23:20 +01:00
|
|
|
timer.callback = timer_cb;
|
2022-09-23 13:49:45 +02:00
|
|
|
timer.arg = &timeout;
|
|
|
|
ztimer_set(ZTIMER_USEC, &timer, timeout);
|
2017-02-20 18:23:20 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|