/* Test of condition variables in multithreaded situations.
Copyright (C) 2008-2024 Free Software Foundation, Inc.
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see . */
#include
#if USE_ISOC_THREADS || USE_POSIX_THREADS || USE_ISOC_AND_POSIX_THREADS || USE_WINDOWS_THREADS
/* Which tests to perform.
Uncomment some of these, to verify that all tests crash if no locking
is enabled. */
#define DO_TEST_COND 1
#define DO_TEST_TIMEDCOND 1
/* Whether to help the scheduler through explicit sched_yield().
Uncomment this to see if the operating system has a fair scheduler. */
#define EXPLICIT_YIELD 1
/* Whether to print debugging messages. */
#define ENABLE_DEBUGGING 0
#include
#include
#include
#include
#include
#if EXPLICIT_YIELD
# include
#endif
#if HAVE_DECL_ALARM
# include
# include
#endif
#include "virtualbox.h"
#include "macros.h"
#if ENABLE_DEBUGGING
# define dbgprintf printf
#else
# define dbgprintf if (0) printf
#endif
#if EXPLICIT_YIELD
# define yield() sched_yield ()
#else
# define yield()
#endif
/*
* Condition check
*/
/* Marked volatile so that different threads see the same value. This is
good enough in practice, although in theory stdatomic.h should be used. */
static int volatile cond_value;
static pthread_cond_t condtest;
static pthread_mutex_t lockcond;
static void *
pthread_cond_wait_routine (void *arg)
{
ASSERT (pthread_mutex_lock (&lockcond) == 0);
if (cond_value)
{
/* The main thread already slept, and nevertheless this thread comes
too late. */
*(int *)arg = 1;
}
else
{
do
{
ASSERT (pthread_cond_wait (&condtest, &lockcond) == 0);
}
while (!cond_value);
}
ASSERT (pthread_mutex_unlock (&lockcond) == 0);
cond_value = 2;
return NULL;
}
static int
test_pthread_cond_wait ()
{
int skipped = 0;
pthread_t thread;
int ret;
cond_value = 0;
/* Create a separate thread. */
ASSERT (pthread_create (&thread, NULL, pthread_cond_wait_routine, &skipped)
== 0);
/* Sleep for 2 seconds. */
{
struct timespec remaining;
remaining.tv_sec = 2;
remaining.tv_nsec = 0;
do
{
yield ();
ret = nanosleep (&remaining, &remaining);
ASSERT (ret >= -1);
}
while (ret == -1 && (remaining.tv_sec != 0 || remaining.tv_nsec != 0));
}
/* Tell one of the waiting threads (if any) to continue. */
ASSERT (pthread_mutex_lock (&lockcond) == 0);
cond_value = 1;
ASSERT (pthread_cond_signal (&condtest) == 0);
ASSERT (pthread_mutex_unlock (&lockcond) == 0);
ASSERT (pthread_join (thread, NULL) == 0);
if (cond_value != 2)
abort ();
return skipped;
}
/*
* Timed Condition check
*/
/* Marked volatile so that different threads see the same value. This is
good enough in practice, although in theory stdatomic.h should be used. */
static int volatile cond_timed_out;
/* Stores in *TS the current time plus 1 second. */
static void
get_ts (struct timespec *ts)
{
struct timeval now;
gettimeofday (&now, NULL);
ts->tv_sec = now.tv_sec + 1;
ts->tv_nsec = now.tv_usec * 1000;
}
static void *
pthread_cond_timedwait_routine (void *arg)
{
int ret;
struct timespec ts;
ASSERT (pthread_mutex_lock (&lockcond) == 0);
if (cond_value)
{
/* The main thread already slept, and nevertheless this thread comes
too late. */
*(int *)arg = 1;
}
else
{
do
{
get_ts (&ts);
ret = pthread_cond_timedwait (&condtest, &lockcond, &ts);
if (ret == ETIMEDOUT)
cond_timed_out = 1;
}
while (!cond_value);
}
ASSERT (pthread_mutex_unlock (&lockcond) == 0);
return NULL;
}
static int
test_pthread_cond_timedwait (void)
{
int skipped = 0;
pthread_t thread;
int ret;
cond_value = cond_timed_out = 0;
/* Create a separate thread. */
ASSERT (pthread_create (&thread, NULL,
pthread_cond_timedwait_routine, &skipped)
== 0);
/* Sleep for 2 seconds. */
{
struct timespec remaining;
remaining.tv_sec = 2;
remaining.tv_nsec = 0;
do
{
yield ();
ret = nanosleep (&remaining, &remaining);
ASSERT (ret >= -1);
}
while (ret == -1 && (remaining.tv_sec != 0 || remaining.tv_nsec != 0));
}
/* Tell one of the waiting threads (if any) to continue. */
ASSERT (pthread_mutex_lock (&lockcond) == 0);
cond_value = 1;
ASSERT (pthread_cond_signal (&condtest) == 0);
ASSERT (pthread_mutex_unlock (&lockcond) == 0);
ASSERT (pthread_join (thread, NULL) == 0);
if (!cond_timed_out)
abort ();
return skipped;
}
int
main ()
{
/* This test occasionally fails on Linux (glibc or musl libc), in a
VirtualBox VM with paravirtualization = Default or KVM, with ≥ 2 CPUs.
Skip the test in this situation. */
if (is_running_under_virtualbox_kvm () && num_cpus () > 1)
{
fputs ("Skipping test: avoiding VirtualBox bug with KVM paravirtualization\n",
stderr);
return 77;
}
#if HAVE_DECL_ALARM
/* Declare failure if test takes too long, by using default abort
caused by SIGALRM. */
int alarm_value = 600;
signal (SIGALRM, SIG_DFL);
alarm (alarm_value);
#endif
ASSERT (pthread_cond_init (&condtest, NULL) == 0);
{
pthread_mutexattr_t attr;
ASSERT (pthread_mutexattr_init (&attr) == 0);
ASSERT (pthread_mutexattr_settype (&attr, PTHREAD_MUTEX_NORMAL) == 0);
ASSERT (pthread_mutex_init (&lockcond, &attr) == 0);
ASSERT (pthread_mutexattr_destroy (&attr) == 0);
}
#if DO_TEST_COND
printf ("Starting test_pthread_cond_wait ..."); fflush (stdout);
{
int skipped = test_pthread_cond_wait ();
printf (skipped ? " SKIP\n" : " OK\n"); fflush (stdout);
}
#endif
#if DO_TEST_TIMEDCOND
printf ("Starting test_pthread_cond_timedwait ..."); fflush (stdout);
{
int skipped = test_pthread_cond_timedwait ();
printf (skipped ? " SKIP\n" : " OK\n"); fflush (stdout);
}
#endif
return test_exit_status;
}
#else
/* No multithreading available. */
#include
int
main ()
{
fputs ("Skipping test: multithreading not enabled\n", stderr);
return 77;
}
#endif