Blame nptl/tst-cnd-broadcast.c

Packit 6c4009
/* C11 threads condition broadcast variable tests.
Packit 6c4009
   Copyright (C) 2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <threads.h>
Packit 6c4009
#include <stdbool.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
Packit 6c4009
#include <support/check.h>
Packit 6c4009
Packit 6c4009
/* Condition variable where child threads will wait.  */
Packit 6c4009
static cnd_t cond;
Packit 6c4009
Packit 6c4009
/* Mutex to control wait on cond.  */
Packit 6c4009
static mtx_t mutex;
Packit 6c4009
Packit 6c4009
/* Number of threads which have entered the cnd_wait region.  */
Packit 6c4009
static unsigned int waiting_threads;
Packit 6c4009
Packit 6c4009
/* Code executed by each thread.  */
Packit 6c4009
static int
Packit 6c4009
child_wait (void* data)
Packit 6c4009
{
Packit 6c4009
  /* Wait until parent thread sends broadcast here.  */
Packit 6c4009
  mtx_lock (&mutex);
Packit 6c4009
  ++waiting_threads;
Packit 6c4009
  cnd_wait (&cond, &mutex);
Packit 6c4009
  mtx_unlock (&mutex);
Packit 6c4009
Packit 6c4009
  thrd_exit (thrd_success);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#define N 5
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
do_test (void)
Packit 6c4009
{
Packit 6c4009
  thrd_t ids[N];
Packit 6c4009
  unsigned char i;
Packit 6c4009
Packit 6c4009
  if (cnd_init (&cond) != thrd_success)
Packit 6c4009
    FAIL_EXIT1 ("cnd_init failed");
Packit 6c4009
  if (mtx_init (&mutex, mtx_plain) != thrd_success)
Packit 6c4009
    FAIL_EXIT1 ("mtx_init failed");
Packit 6c4009
Packit 6c4009
  /* Create N new threads.  */
Packit 6c4009
  for (i = 0; i < N; ++i)
Packit 6c4009
    {
Packit 6c4009
      if (thrd_create (&ids[i], child_wait, NULL) != thrd_success)
Packit 6c4009
	FAIL_EXIT1 ("thrd_create failed");
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Wait for other threads to reach their wait func.  */
Packit 6c4009
  while (true)
Packit 6c4009
    {
Packit 6c4009
      mtx_lock (&mutex);
Packit 6c4009
      TEST_VERIFY (waiting_threads <= N);
Packit 6c4009
      bool done_waiting = waiting_threads == N;
Packit 6c4009
      mtx_unlock (&mutex);
Packit 6c4009
      if (done_waiting)
Packit 6c4009
	break;
Packit 6c4009
      thrd_sleep (&((struct timespec){.tv_nsec = 100 * 1000 * 1000}), NULL);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  mtx_lock (&mutex);
Packit 6c4009
  if (cnd_broadcast (&cond) != thrd_success)
Packit 6c4009
    FAIL_EXIT1 ("cnd_broadcast failed");
Packit 6c4009
  mtx_unlock (&mutex);
Packit 6c4009
Packit 6c4009
  for (i = 0; i < N; ++i)
Packit 6c4009
    {
Packit 6c4009
      if (thrd_join (ids[i], NULL) != thrd_success)
Packit 6c4009
	FAIL_EXIT1 ("thrd_join failed");
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  mtx_destroy (&mutex);
Packit 6c4009
  cnd_destroy (&cond;;
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#include <support/test-driver.c>