1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/thread_race/main.c
Francisco Molina 416c048737 tests: add test_utils_interactive_sync when possible
- Define test_utils_interactive_sync as DEFAULT_MODULE in Makefile.tests_common
- For tests disabling autoinit, add test_utils_interactive_sync to main
- Add DISABLE_MODULE += test_utils_interactive_sync for tests requiring
  sudo,  `tests/shell`, `tests/minimal` and `tests/stdin`
- Add shell_commands to tests/periph_wdt and tests/struct_tm_utility to
  pull `r` and `s` commands
- Remove includes and usage in `tests/main.c` for tests that where
  already using test_utils_interactive_sync
2019-11-27 15:07:42 +01:00

138 lines
2.7 KiB
C

/*
* Copyright (C) 2018 Matthew Blue <matthew.blue.neuro@gmail.com>
*
* 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 Test for race conditions in context switching
*
* May have false negatives.
*
* @author Matthew Blue <matthew.blue.neuro@gmail.com>
*
* @}
*/
#include <stdio.h>
#include "irq.h"
#include "sched.h"
#include "thread.h"
#include "test_utils/interactive_sync.h"
char iqr_check_stack[THREAD_STACKSIZE_DEFAULT];
static volatile uint8_t irq_occurred;
static void *_thread_irq_check(void *arg)
{
(void)arg;
while (1) {
irq_occurred = 1;
thread_sleep();
}
return NULL;
}
/* Mostly copied from thread_wakeup() */
static void _thread_wake_wo_yield(kernel_pid_t pid)
{
unsigned old_state = irq_disable();
thread_t *other_thread = (thread_t *) thread_get(pid);
sched_set_status(other_thread, STATUS_RUNNING);
irq_restore(old_state);
}
static void _spin(void)
{
/* Volatile so it is not messed with by optimizations */
volatile uint8_t i;
for (i = 0; i < 255; i++) ;
}
int main(void)
{
test_utils_interactive_sync();
puts("Context swap race condition test application");
kernel_pid_t pid;
puts("Starting IRQ check thread");
pid = thread_create(iqr_check_stack, sizeof(iqr_check_stack),
THREAD_PRIORITY_MAIN - 1,
THREAD_CREATE_SLEEPING,
_thread_irq_check, NULL, "irqchk");
printf("Checking for working context swap (to detect false positives)... ");
irq_occurred = 0;
_thread_wake_wo_yield(pid);
thread_yield_higher();
/* Delay so we are not testing for race conditions also */
_spin();
if (irq_occurred == 1) {
puts("[Success]");
}
else {
puts("[Failed]");
return -1;
}
printf("Checking for reset of swaps (to detect false positives)... ");
irq_occurred = 0;
_thread_wake_wo_yield(pid);
thread_yield_higher();
/* Delay so we are not testing for race conditions also */
_spin();
if (irq_occurred == 1) {
puts("[Success]");
}
else {
puts("[Failed]");
return -1;
}
/* Volatile so it is not messed with by optimizations */
volatile uint8_t race_test;
printf("Checking for context swap race condition... ");
irq_occurred = 0;
_thread_wake_wo_yield(pid);
thread_yield_higher();
/* Race instruction */
race_test = irq_occurred;
if (race_test == 1) {
puts("[Success]");
}
else {
puts("[Failed]");
return -1;
}
return 0;
}