blob: 58962da54145658e8d4177fc825daedf85d46818 [file] [log] [blame]
/*
* Copyright (c) 2004, Bull S.A.. All rights reserved.
* Created by: Sebastien Decugis
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2 of the GNU General Public License as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
* This file tests the following assertion:
*
* The pthread_cond_broadcast function unblocks all the threads blocked on the
* conditional variable.
* The steps are:
* -> Create as many threads as possible which will wait on a condition
* variable
* -> broadcast the condition and check that all threads are awaken
*
* The test will fail when the threads are not terminated within a certain
* duration.
*
*/
#define _POSIX_C_SOURCE 200112L
#ifndef WITHOUT_XOPEN
#define _XOPEN_SOURCE 600
#endif
#include <pthread.h>
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include <string.h>
#include <time.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <semaphore.h>
#include <sys/sysinfo.h>
#include "../testfrmw/testfrmw.h"
#include "../testfrmw/testfrmw.c"
#define UNRESOLVED_KILLALL(error, text) { \
if (td->fork) { \
children_t *tmp; \
while (children->next != NULL) { \
tmp = children->next; \
children->next = tmp->next; \
kill(tmp->data.p, SIGKILL); \
free(tmp); \
} \
} \
UNRESOLVED(error, text); \
}
#define FAILED_KILLALL(text) { \
if (td->fork) { \
children_t *tmp; \
while (children->next != NULL) { \
tmp = children->next; \
children->next = tmp->next; \
kill(tmp->data.p, SIGKILL); \
free(tmp); \
} \
} \
FAILED(text); \
}
#ifndef VERBOSE
#define VERBOSE 1
#endif
/* Do not create more than this amount of children: */
static int max_process_children = 200;
static int max_thread_children = 1000;
#define TIMEOUT (180)
#ifndef WITHOUT_ALTCLK
#define USE_ALTCLK /* make tests with MONOTONIC CLOCK if supported */
#endif
#ifdef WITHOUT_XOPEN
/* We define those to avoid compilation errors, but they won't be used */
#define PTHREAD_MUTEX_DEFAULT 0
#define PTHREAD_MUTEX_NORMAL 0
#define PTHREAD_MUTEX_ERRORCHECK 0
#define PTHREAD_MUTEX_RECURSIVE 0
#endif
struct _scenar {
int m_type;
int mc_pshared;
int c_clock;
int fork;
char *descr;
} scenarii[] = {
{
PTHREAD_MUTEX_DEFAULT, 0, 0, 0, "Default mutex"}, {
PTHREAD_MUTEX_NORMAL, 0, 0, 0, "Normal mutex"}, {
PTHREAD_MUTEX_ERRORCHECK, 0, 0, 0, "Errorcheck mutex"}, {
PTHREAD_MUTEX_RECURSIVE, 0, 0, 0, "Recursive mutex"}, {
PTHREAD_MUTEX_DEFAULT, 1, 0, 0, "PShared default mutex"}, {
PTHREAD_MUTEX_NORMAL, 1, 0, 0, "Pshared normal mutex"}, {
PTHREAD_MUTEX_ERRORCHECK, 1, 0, 0, "Pshared errorcheck mutex"}, {
PTHREAD_MUTEX_RECURSIVE, 1, 0, 0, "Pshared recursive mutex"}, {
PTHREAD_MUTEX_DEFAULT, 1, 0, 1,
"Pshared default mutex across processes"}, {
PTHREAD_MUTEX_NORMAL, 1, 0, 1, "Pshared normal mutex across processes"},
{
PTHREAD_MUTEX_ERRORCHECK, 1, 0, 1,
"Pshared errorcheck mutex across processes"}, {
PTHREAD_MUTEX_RECURSIVE, 1, 0, 1,
"Pshared recursive mutex across processes"},
#ifdef USE_ALTCLK
{
PTHREAD_MUTEX_DEFAULT, 1, 1, 1,
"Pshared default mutex and alt clock condvar across processes"},
{
PTHREAD_MUTEX_NORMAL, 1, 1, 1,
"Pshared normal mutex and alt clock condvar across processes"},
{
PTHREAD_MUTEX_ERRORCHECK, 1, 1, 1,
"Pshared errorcheck mutex and alt clock condvar across processes"},
{
PTHREAD_MUTEX_RECURSIVE, 1, 1, 1,
"Pshared recursive mutex and alt clock condvar across processes"},
{
PTHREAD_MUTEX_DEFAULT, 0, 1, 0, "Default mutex and alt clock condvar"},
{
PTHREAD_MUTEX_NORMAL, 0, 1, 0, "Normal mutex and alt clock condvar"},
{
PTHREAD_MUTEX_ERRORCHECK, 0, 1, 0,
"Errorcheck mutex and alt clock condvar"}, {
PTHREAD_MUTEX_RECURSIVE, 0, 1, 0,
"Recursive mutex and alt clock condvar"}, {
PTHREAD_MUTEX_DEFAULT, 1, 1, 0,
"PShared default mutex and alt clock condvar"}, {
PTHREAD_MUTEX_NORMAL, 1, 1, 0,
"Pshared normal mutex and alt clock condvar"}, {
PTHREAD_MUTEX_ERRORCHECK, 1, 1, 0,
"Pshared errorcheck mutex and alt clock condvar"}, {
PTHREAD_MUTEX_RECURSIVE, 1, 1, 0,
"Pshared recursive mutex and alt clock condvar"},
#endif
};
#define NSCENAR (sizeof(scenarii)/sizeof(scenarii[0]))
struct testdata {
int count; /* number of children currently waiting */
pthread_cond_t cnd;
pthread_mutex_t mtx;
int predicate; /* Boolean associated to the condvar */
clockid_t cid; /* clock used in the condvar */
char fork; /* the children are processes */
};
struct testdata *td;
/* Child function (either in a thread or in a process) */
static void *child(void *arg)
{
int ret = 0;
struct timespec ts;
char timed;
/* lock the mutex */
ret = pthread_mutex_lock(&td->mtx);
if (ret != 0)
UNRESOLVED(ret, "Failed to lock mutex in child");
/* increment count */
td->count++;
timed = td->count & 1;
if (timed) {
/* get current time if we are a timedwait */
ret = clock_gettime(td->cid, &ts);
if (ret != 0)
UNRESOLVED(errno, "Unable to read clock");
ts.tv_sec += TIMEOUT;
}
do {
/* Wait while the predicate is false */
if (timed)
ret = pthread_cond_timedwait(&td->cnd, &td->mtx, &ts);
else
ret = pthread_cond_wait(&td->cnd, &td->mtx);
#if VERBOSE > 5
output("[child] Wokenup timed=%i, Predicate=%i, ret=%i\n",
timed, td->predicate, ret);
#endif
} while ((ret == 0) && (td->predicate == 0));
if (ret == ETIMEDOUT) {
ret = pthread_mutex_unlock(&td->mtx);
if (ret != 0)
UNRESOLVED(ret, "Failed to unlock the mutex.");
FAILED("Timeout occured. This means a cond signal was lost -- "
"or parent died");
}
if (ret != 0)
UNRESOLVED(ret, "Failed to wait for the cond");
/* unlock the mutex */
ret = pthread_mutex_unlock(&td->mtx);
if (ret != 0)
UNRESOLVED(ret, "Failed to unlock the mutex.");
return NULL;
}
/* Structure used to store the children information */
typedef struct _children {
union {
pid_t p;
pthread_t t;
} data;
struct _children *next;
} children_t;
children_t sentinel = {.next = NULL };
children_t *children = &sentinel;
/* Timeout thread */
sem_t sem_tmr;
void *timer(void *arg)
{
unsigned int to = TIMEOUT;
int ret;
do {
ret = sem_wait(&sem_tmr);
} while ((ret != 0) && (errno == EINTR));
if (ret != 0)
UNRESOLVED(errno, "Failed to wait for the semaphore "
"in timer thread.");
do {
to = sleep(to);
} while (to > 0);
FAILED_KILLALL("Operation timed out. A signal was lost.");
return NULL; /* For compiler */
}
#ifdef __linux__
static void children_number(void)
{
struct sysinfo sysinformation;
int ret;
int avail_number;
unsigned long per_process;
unsigned long min_stack;
min_stack = sysconf(_SC_THREAD_STACK_MIN);
ret = sysinfo(&sysinformation);
if (ret != 0)
UNRESOLVED(ret, "Failed to get system information.");
per_process = min_stack * max_thread_children;
if (per_process > sysinformation.freeram)
UNTESTED("Not enough memory.");
avail_number = sysinformation.freeram / per_process;
if (avail_number < 10)
UNTESTED("Not enough memory.");
max_process_children = (avail_number < max_process_children ?
avail_number : max_process_children);
return;
}
#else
static void children_number(void)
{
return;
}
#endif
int main(void)
{
int ret;
pthread_mutexattr_t ma;
pthread_condattr_t ca;
int scenar;
long pshared, monotonic, cs, mf;
int child_count;
children_t *tmp, *cur;
int ch;
pid_t pid;
int status;
pthread_t t_timer;
pthread_attr_t ta;
struct testdata alternativ;
output_init();
children_number();
/* check the system abilities */
pshared = sysconf(_SC_THREAD_PROCESS_SHARED);
cs = sysconf(_SC_CLOCK_SELECTION);
monotonic = sysconf(_SC_MONOTONIC_CLOCK);
mf = sysconf(_SC_MAPPED_FILES);
#if VERBOSE > 0
output("Test starting\n");
output("System abilities:\n");
output(" TPS : %li\n", pshared);
output(" CS : %li\n", cs);
output(" MON : %li\n", monotonic);
output(" MF : %li\n", mf);
if ((mf < 0) || (pshared < 0))
output("Process-shared attributes won't be tested\n");
if ((cs < 0) || (monotonic < 0))
output("Alternative clock won't be tested\n");
fflush(stdout);
#endif
if (monotonic < 0)
cs = -1;
#ifndef USE_ALTCLK
if (cs > 0)
output("Implementation supports the MONOTONIC CLOCK "
"but option is disabled in test.\n");
#endif
/*
* Allocate space for the testdata structure
*/
if (mf < 0) {
/* Cannot mmap a file, we use an alternative method */
td = &alternativ;
pshared = -1; /* We won't do this testing anyway */
#if VERBOSE > 0
output("Testdata allocated in the process memory.\n");
#endif
} else {
/* We will place the test data in a mmaped file */
char filename[] = "/tmp/cond_broadcast-XXXXXX";
size_t sz, ps;
void *mmaped;
int fd;
char *tmp;
/* We now create the temp files */
fd = mkstemp(filename);
if (fd == -1)
UNRESOLVED(errno,
"Temporary file could not be created");
/* and make sure the file will be deleted when closed */
unlink(filename);
#if VERBOSE > 1
output("Temp file created (%s).\n", filename);
#endif
ps = (size_t) sysconf(_SC_PAGESIZE);
sz = ((sizeof(struct testdata) / ps) + 1) * ps;
tmp = calloc(1, sz);
if (tmp == NULL)
UNRESOLVED(errno, "Memory allocation failed");
/* Write the data to the file. */
if (write(fd, tmp, sz) != (ssize_t) sz)
UNRESOLVED(sz, "Writting to the file failed");
free(tmp);
/* Now we can map the file in memory */
mmaped = mmap(NULL, sz, PROT_READ | PROT_WRITE, MAP_SHARED,
fd, 0);
if (mmaped == MAP_FAILED)
UNRESOLVED(errno, "mmap failed");
td = (struct testdata *)mmaped;
/* Our datatest structure is now in shared memory */
#if VERBOSE > 1
output("Testdata allocated in shared memory (%ib).\n",
sizeof(struct testdata));
#endif
}
/* Initialize the thread attribute object */
ret = pthread_attr_init(&ta);
if (ret != 0)
UNRESOLVED(ret, "[parent] Failed to initialize a thread "
"attribute object");
ret = pthread_attr_setstacksize(&ta, sysconf(_SC_THREAD_STACK_MIN));
if (ret != 0)
UNRESOLVED(ret, "[parent] Failed to set thread stack size");
/* Initialize the semaphore for the timer thread */
ret = sem_init(&sem_tmr, 0, 0);
if (ret != 0)
UNRESOLVED(ret, "Failed to initialize the semaphore");
/* Do the test for each test scenario */
for (scenar = 0; scenar < NSCENAR; scenar++) {
/* set / reset everything */
td->fork = 0;
ret = pthread_mutexattr_init(&ma);
if (ret != 0)
UNRESOLVED(ret, "[parent] Unable to initialize the "
"mutex attribute object");
ret = pthread_condattr_init(&ca);
if (ret != 0)
UNRESOLVED(ret, "[parent] Unable to initialize the "
"cond attribute object");
#ifndef WITHOUT_XOPEN
/* Set the mutex type */
ret = pthread_mutexattr_settype(&ma, scenarii[scenar].m_type);
if (ret != 0)
UNRESOLVED(ret, "[parent] Unable to set mutex type");
#endif
/* Set the pshared attributes, if supported */
if ((pshared > 0) && (scenarii[scenar].mc_pshared != 0)) {
ret = pthread_mutexattr_setpshared(&ma,
PTHREAD_PROCESS_SHARED);
if (ret != 0)
UNRESOLVED(ret, "[parent] Unable to set the "
"mutex process-shared");
ret = pthread_condattr_setpshared(&ca,
PTHREAD_PROCESS_SHARED);
if (ret != 0)
UNRESOLVED(ret, "[parent] Unable to set the "
"cond var process-shared");
}
/* Set the alternative clock, if supported */
#ifdef USE_ALTCLK
if ((cs > 0) && (scenarii[scenar].c_clock != 0)) {
ret = pthread_condattr_setclock(&ca, CLOCK_MONOTONIC);
if (ret != 0)
UNRESOLVED(ret, "[parent] Unable to set the "
"monotonic clock for the cond");
}
ret = pthread_condattr_getclock(&ca, &td->cid);
if (ret != 0)
UNRESOLVED(ret, "Unable to get clock from cond attr");
#else
td->cid = CLOCK_REALTIME;
#endif
/* Tell whether the test will be across processes */
if ((pshared > 0) && (scenarii[scenar].fork != 0))
td->fork = 1;
/* initialize the condvar */
ret = pthread_cond_init(&td->cnd, &ca);
if (ret != 0)
UNRESOLVED(ret, "Cond init failed");
/* initialize the mutex */
ret = pthread_mutex_init(&td->mtx, &ma);
if (ret != 0)
UNRESOLVED(ret, "Mutex init failed");
/* Destroy the attributes */
ret = pthread_condattr_destroy(&ca);
if (ret != 0)
UNRESOLVED(ret,
"Failed to destroy the cond var "
"attribute object");
ret = pthread_mutexattr_destroy(&ma);
if (ret != 0)
UNRESOLVED(ret,
"Failed to destroy the mutex "
"attribute object");
#if VERBOSE > 2
output("[parent] Starting test %s\n", scenarii[scenar].descr);
#endif
td->count = 0;
child_count = 0;
cur = children;
/* create the timeout thread */
ret = pthread_create(&t_timer, NULL, timer, NULL);
if (ret != 0)
UNRESOLVED_KILLALL(ret,
"Unable to create timer thread");
/* Create all the children */
if (td->fork == 0) {
do {
tmp = malloc(sizeof(children_t));
if (tmp != NULL) {
ret = pthread_create(&(tmp->data.t),
&ta, child, NULL);
if (ret != 0) {
free(tmp);
} else {
child_count++;
tmp->next = NULL;
cur->next = tmp;
cur = tmp;
}
} else {
ret = errno;
}
} while ((ret == 0)
&& (child_count < max_thread_children));
#if VERBOSE > 2
output("[parent] Created %i children threads\n",
child_count);
#endif
if (child_count == 0)
UNRESOLVED(ret, "Unable to create any thread");
} else {
do {
tmp = malloc(sizeof(children_t));
if (tmp != NULL) {
tmp->data.p = fork();
if (tmp->data.p == 0) {
child(NULL);
exit(0);
}
if (tmp->data.p == -1) {
ret = errno;
free(tmp);
} else {
ret = 0;
child_count++;
tmp->next = NULL;
cur->next = tmp;
cur = tmp;
}
} else {
ret = errno;
}
} while ((ret == 0)
&& (child_count < max_process_children));
#if VERBOSE > 2
output("[parent] Created %i children processes\n",
child_count);
#endif
if (child_count == 0)
UNRESOLVED(ret, "Unable to create any process");
}
/* Make sure all children are waiting */
ret = pthread_mutex_lock(&td->mtx);
if (ret != 0)
UNRESOLVED_KILLALL(ret, "Failed to lock mutex");
ch = td->count;
while (ch < child_count) {
ret = pthread_mutex_unlock(&td->mtx);
if (ret != 0)
UNRESOLVED_KILLALL(ret,
"Failed to unlock mutex");
sched_yield();
ret = pthread_mutex_lock(&td->mtx);
if (ret != 0)
UNRESOLVED_KILLALL(ret, "Failed to lock mutex");
ch = td->count;
}
#if VERBOSE > 4
output("[parent] All children are waiting\n");
#endif
/* start the timer count */
do {
ret = sem_post(&sem_tmr);
} while ((ret != 0) && (errno == EINTR));
if (ret != 0)
UNRESOLVED_KILLALL(errno,
"Failed to post the semaphore.");
/* Wakeup the children */
td->predicate = 1;
ret = pthread_cond_broadcast(&td->cnd);
if (ret != 0)
UNRESOLVED_KILLALL(ret,
"Failed to broadcast the condition.");
#if VERBOSE > 4
output("[parent] Condition was signaled\n");
#endif
ret = pthread_mutex_unlock(&td->mtx);
if (ret != 0)
UNRESOLVED_KILLALL(ret, "Failed to unlock mutex");
#if VERBOSE > 4
output("[parent] Joining the children\n");
#endif
/* join the children */
while (children->next != NULL) {
tmp = children->next;
children->next = tmp->next;
if (td->fork == 0) {
ret |= pthread_join(tmp->data.t, NULL);
} else {
pid = waitpid(tmp->data.p, &status, 0);
if (pid != tmp->data.p) {
ret = errno;
output("Waitpid failed (expected: %i, "
"got: %i)\n", tmp->data.p, pid);
free(tmp);
UNRESOLVED_KILLALL(ret,
"Waitpid failed");
}
if (WIFEXITED(status)) {
/* the child should return only failed or unresolved or passed */
if (ret != PTS_FAIL)
ret |= WEXITSTATUS(status);
}
}
free(tmp);
}
if (ret != 0) {
output_fini();
exit(ret);
}
#if VERBOSE > 4
output("[parent] All children terminated\n");
#endif
/* cancel the timeout thread */
ret = pthread_cancel(t_timer);
if (ret != 0)
/* Strange error here... the thread cannot be terminated
* (app would be killed)
*/
UNRESOLVED(ret, "Failed to cancel the timeout handler");
/* join the timeout thread */
ret = pthread_join(t_timer, NULL);
if (ret != 0)
UNRESOLVED(ret, "Failed to join the timeout handler");
/* Destroy the datas */
ret = pthread_cond_destroy(&td->cnd);
if (ret != 0)
UNRESOLVED(ret, "Failed to destroy the condvar");
ret = pthread_mutex_destroy(&td->mtx);
if (ret != 0)
UNRESOLVED(ret, "Failed to destroy the mutex");
}
/* Destroy global data */
ret = pthread_attr_destroy(&ta);
if (ret != 0)
UNRESOLVED(ret, "Final thread attr destroy failed");
ret = sem_destroy(&sem_tmr);
if (ret != 0)
UNRESOLVED(errno, "Final semaphore destroy failed");
/* exit */
PASSED;
}