2014-02-21 14:51:17 +01:00
|
|
|
/*
|
2017-10-25 14:22:07 +02:00
|
|
|
* Copyright (C) 2014-2017 Hamburg University of Applied Sciences
|
2014-06-18 15:06:23 +02:00
|
|
|
*
|
2014-07-31 19:45:27 +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.
|
2014-06-18 15:06:23 +02:00
|
|
|
*/
|
2014-02-21 14:51:17 +01:00
|
|
|
|
|
|
|
/**
|
2014-06-18 15:06:23 +02:00
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief riot thread test application
|
|
|
|
*
|
|
|
|
* @author Raphael Hiesgen <raphael.hiesgen@haw-hamburg.de>
|
2017-10-25 14:22:07 +02:00
|
|
|
* @author Sebastian Meiling <s@mlng.net>
|
2014-06-18 15:06:23 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
2014-02-21 14:51:17 +01:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <inttypes.h>
|
2014-07-15 18:47:39 +02:00
|
|
|
|
|
|
|
#include "thread.h"
|
|
|
|
#include "mutex.h"
|
2014-02-21 14:51:17 +01:00
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
#ifndef PROBLEM
|
2014-02-21 14:51:17 +01:00
|
|
|
#define PROBLEM 12
|
2017-10-25 14:22:07 +02:00
|
|
|
#endif
|
2014-02-21 14:51:17 +01:00
|
|
|
|
2023-10-26 10:34:01 +02:00
|
|
|
static mutex_t mtx = MUTEX_INIT;
|
|
|
|
static uint32_t storage = 1;
|
|
|
|
static char stacks[PROBLEM][THREAD_STACKSIZE_DEFAULT];
|
2014-02-21 14:51:17 +01:00
|
|
|
|
2023-10-26 10:34:01 +02:00
|
|
|
static void *run(void *arg)
|
2014-02-21 14:51:17 +01:00
|
|
|
{
|
2023-10-26 10:34:01 +02:00
|
|
|
(void)arg;
|
2014-03-04 20:20:01 +01:00
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
msg_t m, final;
|
2014-07-06 22:57:56 +02:00
|
|
|
kernel_pid_t me = thread_getpid();
|
2017-10-25 14:22:07 +02:00
|
|
|
|
|
|
|
printf("T-%02d: alive\n", me);
|
|
|
|
|
2014-07-25 08:17:06 +02:00
|
|
|
msg_receive(&m);
|
2017-10-25 14:22:07 +02:00
|
|
|
printf("T-%02d: got arg %" PRIu32 "\n", me, m.content.value);
|
2014-02-21 14:51:17 +01:00
|
|
|
|
2023-10-26 10:34:01 +02:00
|
|
|
/* Accessing shared variable `storage` requires a critical section to avoid
|
|
|
|
* data races. The mutex provides this and `mutex_lock()`/`mutex_unlock()`
|
|
|
|
* are an implicit memory barrier that will ensure that `storage` is read
|
|
|
|
* indeed from memory and the new value is written back to memory within
|
|
|
|
* the critical section. The use of `volatile` is, hence, not needed here
|
|
|
|
* (and in fact incorrect). */
|
2014-07-25 08:17:06 +02:00
|
|
|
mutex_lock(&mtx);
|
2014-03-04 20:20:01 +01:00
|
|
|
storage *= m.content.value;
|
2014-02-21 14:51:17 +01:00
|
|
|
mutex_unlock(&mtx);
|
|
|
|
|
|
|
|
final.content.value = me;
|
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
if (msg_send(&final, m.sender_pid) < 0) {
|
|
|
|
printf("T-%02d: send reply to main failed!\n", me);
|
2014-02-21 14:51:17 +01:00
|
|
|
}
|
2014-03-04 20:20:01 +01:00
|
|
|
|
|
|
|
return NULL;
|
2014-02-21 14:51:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
msg_t args[PROBLEM];
|
2017-10-25 14:22:07 +02:00
|
|
|
kernel_pid_t ths;
|
|
|
|
uint32_t factorial = 1;
|
|
|
|
|
|
|
|
printf("[START] compute %d! (factorial).\n", PROBLEM);
|
2014-02-21 14:51:17 +01:00
|
|
|
|
|
|
|
for (int i = 0; i < PROBLEM; ++i) {
|
2017-10-25 14:22:07 +02:00
|
|
|
int arg = i + 1;
|
|
|
|
factorial *= arg;
|
|
|
|
printf("MAIN: create thread, arg: %d\n", arg);
|
|
|
|
ths = thread_create(stacks[i], sizeof(stacks[i]),
|
2023-10-26 10:34:01 +02:00
|
|
|
THREAD_PRIORITY_MAIN - 1,
|
2024-03-07 15:51:39 +01:00
|
|
|
THREAD_CREATE_WOUT_YIELD,
|
2023-10-26 10:34:01 +02:00
|
|
|
run, NULL, "thread");
|
2014-02-21 14:51:17 +01:00
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
if (ths < 0) {
|
|
|
|
puts("[ERROR]");
|
|
|
|
return 2;
|
2014-02-21 14:51:17 +01:00
|
|
|
}
|
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
printf("MAIN: msg to T-%d\n", ths);
|
|
|
|
args[i].content.value = arg;
|
|
|
|
if (msg_send(&args[i], ths) < 0) {
|
|
|
|
puts("[ERROR]");
|
|
|
|
return 3;
|
2014-02-21 14:51:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < PROBLEM; ++i) {
|
|
|
|
msg_t msg;
|
|
|
|
msg_receive(&msg);
|
2017-10-25 14:22:07 +02:00
|
|
|
printf("MAIN: reply from T-%" PRIu32 "\n", msg.content.value);
|
2014-02-21 14:51:17 +01:00
|
|
|
}
|
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
printf("MAIN: %d! = %" PRIu32 "\n", PROBLEM, storage);
|
2014-02-21 14:51:17 +01:00
|
|
|
|
2017-10-25 14:22:07 +02:00
|
|
|
if (storage != factorial) {
|
|
|
|
printf("[ERROR] expected %" PRIu32 "\n", factorial);
|
|
|
|
return 1;
|
2014-02-21 14:51:17 +01:00
|
|
|
}
|
2017-10-25 14:22:07 +02:00
|
|
|
puts("[SUCCESS]");
|
2014-02-21 14:51:17 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|