mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
276 lines
7.5 KiB
C
276 lines
7.5 KiB
C
|
/*
|
||
|
* Copyright (C) 2019 Freie Universität Berlin,
|
||
|
*
|
||
|
* 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 Julian Holzwarth <julian.holzwarth@fu-berlin.de>
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include "thread.h"
|
||
|
#include "test_utils/interactive_sync.h"
|
||
|
#ifdef MODULE_PS
|
||
|
#include "ps.h"
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
#define TEST_THREAD_STACKSIZE ((2 * THREAD_STACKSIZE_IDLE) + \
|
||
|
THREAD_EXTRA_STACKSIZE_PRINTF)
|
||
|
|
||
|
/* stacks for testing threads */
|
||
|
static char t2_stack[TEST_THREAD_STACKSIZE];
|
||
|
static char t3_stack[TEST_THREAD_STACKSIZE];
|
||
|
static char t4_stack[TEST_THREAD_STACKSIZE];
|
||
|
|
||
|
/* function for testing threads */
|
||
|
void *second_thread(void *arg)
|
||
|
{
|
||
|
printf("Thread: %d is starting\n", (int)arg);
|
||
|
printf("Thread: %d calls zombify\n", (int)arg);
|
||
|
thread_zombify();
|
||
|
puts("ERROR zombie runs again!");
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
int main(void)
|
||
|
{
|
||
|
test_utils_interactive_sync();
|
||
|
|
||
|
/* save thread count on test start */
|
||
|
int current_thread_count;
|
||
|
int start_thread_count = sched_num_threads;
|
||
|
|
||
|
printf("Number of threads before the test = %d\n", start_thread_count);
|
||
|
|
||
|
/* creating threads for testing */
|
||
|
puts("Creating first thread (sleeping)");
|
||
|
kernel_pid_t first_pid = thread_create(
|
||
|
t2_stack, sizeof(t2_stack),
|
||
|
THREAD_PRIORITY_MAIN - 1,
|
||
|
THREAD_CREATE_SLEEPING | THREAD_CREATE_STACKTEST,
|
||
|
second_thread, (void *)1, "nr1");
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
puts("Creating second thread (sleeping)");
|
||
|
kernel_pid_t second_pid = thread_create(
|
||
|
t3_stack, sizeof(t3_stack),
|
||
|
THREAD_PRIORITY_MAIN - 1,
|
||
|
THREAD_CREATE_SLEEPING | THREAD_CREATE_STACKTEST,
|
||
|
second_thread, (void *)2, "nr2");
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
puts("Creating third thread (sleeping)");
|
||
|
kernel_pid_t third_pid = thread_create(
|
||
|
t4_stack, sizeof(t4_stack),
|
||
|
THREAD_PRIORITY_MAIN - 1,
|
||
|
THREAD_CREATE_SLEEPING | THREAD_CREATE_STACKTEST,
|
||
|
second_thread, (void *)3, "nr3");
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
current_thread_count = sched_num_threads;
|
||
|
printf("Current number of threads = %d\n", current_thread_count);
|
||
|
|
||
|
/* check if all threads got created */
|
||
|
if (start_thread_count + 3 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
|
||
|
/* wakeup first thread and check if the thread zombified itself */
|
||
|
puts("Waking up first thread");
|
||
|
thread_wakeup(first_pid);
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
current_thread_count = sched_num_threads;
|
||
|
printf("Current number of threads = %d\n", current_thread_count);
|
||
|
if (thread_getstatus(first_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: first thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (start_thread_count + 3 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* wakeup second thread and check if the thread zombified itself */
|
||
|
puts("Waking up second thread");
|
||
|
thread_wakeup(second_pid);
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
current_thread_count = sched_num_threads;
|
||
|
printf("Current number of threads = %d\n", current_thread_count);
|
||
|
|
||
|
if (thread_getstatus(first_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: first thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (thread_getstatus(second_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: second thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (start_thread_count + 3 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* kill first zombie thread and check if the thread still exist (it should not) */
|
||
|
puts("Kill first thread");
|
||
|
|
||
|
if (thread_kill_zombie(first_pid) != 1) {
|
||
|
puts("Error thread_kill_zombie returned an error");
|
||
|
}
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
current_thread_count = sched_num_threads;
|
||
|
printf("Current number of threads = %d\n", current_thread_count);
|
||
|
|
||
|
if (thread_getstatus(first_pid) != STATUS_NOT_FOUND) {
|
||
|
puts("Error first Thread does still exist");
|
||
|
}
|
||
|
else {
|
||
|
if (thread_getstatus(second_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: second thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (start_thread_count + 2 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* wakeup third thread and check if the thread zombified itself */
|
||
|
puts("Waking up third thread");
|
||
|
thread_wakeup(third_pid);
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
if (thread_getstatus(first_pid) != STATUS_NOT_FOUND) {
|
||
|
puts("Error first Thread does still exist");
|
||
|
}
|
||
|
else {
|
||
|
if (thread_getstatus(second_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: third thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (thread_getstatus(third_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: second thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (start_thread_count + 2 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* check if threads are created normally after killing a zombie */
|
||
|
puts("Creating fourth thread (sleeping)");
|
||
|
kernel_pid_t last_pid = thread_create(
|
||
|
t2_stack, sizeof(t2_stack),
|
||
|
THREAD_PRIORITY_MAIN - 1,
|
||
|
THREAD_CREATE_SLEEPING | THREAD_CREATE_STACKTEST,
|
||
|
second_thread, (void *)4, "nr4");
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
if (thread_getstatus(last_pid) != STATUS_SLEEPING) {
|
||
|
puts("Error last Thread is not sleeping");
|
||
|
}
|
||
|
else {
|
||
|
if (last_pid != first_pid) {
|
||
|
puts("Error thread did not reuse first thread pid");
|
||
|
}
|
||
|
else {
|
||
|
if (thread_getstatus(second_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: third thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (thread_getstatus(third_pid) != STATUS_ZOMBIE) {
|
||
|
puts("Error Wrong Status: second thread is not a zombie!");
|
||
|
}
|
||
|
else {
|
||
|
if (start_thread_count + 2 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef MODULE_PS
|
||
|
ps();
|
||
|
puts("");
|
||
|
#endif /* MODULE_PS */
|
||
|
|
||
|
current_thread_count = sched_num_threads;
|
||
|
printf("Current number of threads = %d\n", current_thread_count);
|
||
|
|
||
|
/* check if all threads got created */
|
||
|
if (start_thread_count + 3 != current_thread_count) {
|
||
|
puts("Error wrong thread count");
|
||
|
}
|
||
|
else {
|
||
|
puts("OK");
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|