Blame nptl/tst-rwlock15.c

Packit Service 82fcde
/* Copyright (C) 2015-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
/* This tests that a writer that is preferred -- but times out due to a
Packit Service 82fcde
   reader being present -- does not miss to wake other readers blocked on the
Packit Service 82fcde
   writer's pending lock acquisition.  */
Packit Service 82fcde
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <pthread.h>
Packit Service 82fcde
#include <stdio.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <time.h>
Packit Service 82fcde
Packit Service 82fcde
/* The bug existed in the code that strictly prefers writers over readers.  */
Packit Service 82fcde
static pthread_rwlock_t r = PTHREAD_RWLOCK_WRITER_NONRECURSIVE_INITIALIZER_NP;
Packit Service 82fcde
Packit Service 82fcde
static void *
Packit Service 82fcde
writer (void *arg)
Packit Service 82fcde
{
Packit Service 82fcde
  struct timespec ts;
Packit Service 82fcde
  if (clock_gettime (CLOCK_REALTIME, &ts) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("clock_gettime failed");
Packit Service 82fcde
      exit (EXIT_FAILURE);
Packit Service 82fcde
    }
Packit Service 82fcde
  ts.tv_sec += 1;
Packit Service 82fcde
  int e = pthread_rwlock_timedwrlock (&r, &ts);
Packit Service 82fcde
  if (e != ETIMEDOUT)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("timedwrlock did not time out");
Packit Service 82fcde
      exit (EXIT_FAILURE);
Packit Service 82fcde
    }
Packit Service 82fcde
  return NULL;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static void *
Packit Service 82fcde
reader (void *arg)
Packit Service 82fcde
{
Packit Service 82fcde
  /* This isn't a reliable way to get the interleaving we need (because a
Packit Service 82fcde
     failed trylock doesn't synchronize with the writer, and because we could
Packit Service 82fcde
     try to lock after the writer has already timed out).  However, both will
Packit Service 82fcde
     just lead to false positives.  */
Packit Service 82fcde
  int e;
Packit Service 82fcde
  while ((e = pthread_rwlock_tryrdlock (&r)) != EBUSY)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (e != 0)
Packit Service 82fcde
	exit (EXIT_FAILURE);
Packit Service 82fcde
      pthread_rwlock_unlock (&r);
Packit Service 82fcde
    }
Packit Service 82fcde
  e = pthread_rwlock_rdlock (&r);
Packit Service 82fcde
  if (e != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("reader rdlock failed");
Packit Service 82fcde
      exit (EXIT_FAILURE);
Packit Service 82fcde
    }
Packit Service 82fcde
  pthread_rwlock_unlock (&r);
Packit Service 82fcde
  return NULL;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static int
Packit Service 82fcde
do_test (void)
Packit Service 82fcde
{
Packit Service 82fcde
  /* Grab a rdlock, then create a writer and a reader, and wait until they
Packit Service 82fcde
     finished.  */
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_rwlock_rdlock (&r) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("initial rdlock failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  pthread_t thw;
Packit Service 82fcde
  if (pthread_create (&thw, NULL, writer, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("create failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
  pthread_t thr;
Packit Service 82fcde
  if (pthread_create (&thr, NULL, reader, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("create failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_join (thw, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("writer join failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
  if (pthread_join (thr, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("reader join failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
#define TEST_FUNCTION do_test ()
Packit Service 82fcde
#include "../test-skeleton.c"