1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/tests/thread_race/main.c
Marian Buschsieweke eb1279005c
tests: Cleanup access to internal variables
Replace direct accesses to sched_active_thread and sched_active_pid with
the helper functions thread_getpid() and thread_get_active(). This serves
two purposes:

1. It makes accidental writes to those variable from outside core less likely.
2. Casting off the volatile qualifier is now well contained to those two
   functions
2020-08-24 09:08:01 +02:00

134 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"
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_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)
{
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;
}