Blame nptl/tst-robust8.c

Packit 6c4009
#include <pthread.h>
Packit 6c4009
#include <signal.h>
Packit 6c4009
#include <stdint.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <sys/mman.h>
Packit 6c4009
#include <sys/wait.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
Packit 6c4009
Packit 6c4009
static void prepare (void);
Packit 6c4009
#define PREPARE(argc, argv) prepare ()
Packit 6c4009
static int do_test (void);
Packit 6c4009
#define TEST_FUNCTION do_test ()
Packit 6c4009
#define TIMEOUT 5
Packit 6c4009
#include "../test-skeleton.c"
Packit 6c4009
Packit 6c4009
Packit 6c4009
static int fd;
Packit 6c4009
#define N 100
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
prepare (void)
Packit 6c4009
{
Packit 6c4009
  fd = create_temp_file ("tst-robust8", NULL);
Packit 6c4009
  if (fd == -1)
Packit 6c4009
    exit (1);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
#define THESIGNAL SIGKILL
Packit 6c4009
#define ROUNDS 5
Packit 6c4009
#define THREADS 9
Packit 6c4009
Packit 6c4009
Packit 6c4009
static const struct timespec before = { 0, 0 };
Packit 6c4009
Packit 6c4009
Packit 6c4009
static pthread_mutex_t *map;
Packit 6c4009
Packit 6c4009
Packit 6c4009
static void *
Packit 6c4009
tf (void *arg)
Packit 6c4009
{
Packit 6c4009
  long int nr = (long int) arg;
Packit 6c4009
  int fct = nr % 3;
Packit 6c4009
Packit 6c4009
  uint8_t state[N];
Packit 6c4009
  memset (state, '\0', sizeof (state));
Packit 6c4009
Packit 6c4009
  while (1)
Packit 6c4009
    {
Packit 6c4009
      int r = random () % N;
Packit 6c4009
      if (state[r] == 0)
Packit 6c4009
	{
Packit 6c4009
	  int e;
Packit 6c4009
Packit 6c4009
	  switch (fct)
Packit 6c4009
	    {
Packit 6c4009
	    case 0:
Packit 6c4009
	      e = pthread_mutex_lock (&map[r]);
Packit 6c4009
	      if (e != 0)
Packit 6c4009
		{
Packit 6c4009
		  printf ("mutex_lock of %d in thread %ld failed with %d\n",
Packit 6c4009
			  r, nr, e);
Packit 6c4009
		  exit (1);
Packit 6c4009
		}
Packit 6c4009
	      state[r] = 1;
Packit 6c4009
	      break;
Packit 6c4009
	    case 1:
Packit 6c4009
	      e = pthread_mutex_timedlock (&map[r], &before);
Packit 6c4009
	      if (e != 0 && e != ETIMEDOUT)
Packit 6c4009
		{
Packit 6c4009
		  printf ("\
Packit 6c4009
mutex_timedlock of %d in thread %ld failed with %d\n",
Packit 6c4009
			  r, nr, e);
Packit 6c4009
		  exit (1);
Packit 6c4009
		}
Packit 6c4009
	      break;
Packit 6c4009
	    default:
Packit 6c4009
	      e = pthread_mutex_trylock (&map[r]);
Packit 6c4009
	      if (e != 0 && e != EBUSY)
Packit 6c4009
		{
Packit 6c4009
		  printf ("mutex_trylock of %d in thread %ld failed with %d\n",
Packit 6c4009
			  r, nr, e);
Packit 6c4009
		  exit (1);
Packit 6c4009
		}
Packit 6c4009
	      break;
Packit 6c4009
	    }
Packit 6c4009
Packit 6c4009
	  if (e == EOWNERDEAD)
Packit 6c4009
	    pthread_mutex_consistent_np (&map[r]);
Packit 6c4009
Packit 6c4009
	  if (e == 0 || e == EOWNERDEAD)
Packit 6c4009
	    state[r] = 1;
Packit 6c4009
	}
Packit 6c4009
      else
Packit 6c4009
	{
Packit 6c4009
	  int e = pthread_mutex_unlock (&map[r]);
Packit 6c4009
	  if (e != 0)
Packit 6c4009
	    {
Packit 6c4009
	      printf ("mutex_unlock of %d in thread %ld failed with %d\n",
Packit 6c4009
		      r, nr, e);
Packit 6c4009
	      exit (1);
Packit 6c4009
	    }
Packit 6c4009
Packit 6c4009
	  state[r] = 0;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
child (int round)
Packit 6c4009
{
Packit 6c4009
  for (int thread = 1; thread <= THREADS; ++thread)
Packit 6c4009
    {
Packit 6c4009
      pthread_t th;
Packit 6c4009
      if (pthread_create (&th, NULL, tf, (void *) (long int) thread) != 0)
Packit 6c4009
	{
Packit 6c4009
	  printf ("cannot create thread %d in round %d\n", thread, round);
Packit 6c4009
	  exit (1);
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  struct timespec ts;
Packit 6c4009
  ts.tv_sec = 0;
Packit 6c4009
  ts.tv_nsec = 1000000000 / ROUNDS;
Packit 6c4009
  while (nanosleep (&ts, &ts) != 0)
Packit 6c4009
    /* nothing */;
Packit 6c4009
Packit 6c4009
  /* Time to die.  */
Packit 6c4009
  kill (getpid (), THESIGNAL);
Packit 6c4009
Packit 6c4009
  /* We better never get here.  */
Packit 6c4009
  abort ();
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
do_test (void)
Packit 6c4009
{
Packit 6c4009
  if (ftruncate (fd, N * sizeof (pthread_mutex_t)) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("cannot size new file");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  map = mmap (NULL, N * sizeof (pthread_mutex_t), PROT_READ | PROT_WRITE,
Packit 6c4009
	      MAP_SHARED, fd, 0);
Packit 6c4009
  if (map == MAP_FAILED)
Packit 6c4009
    {
Packit 6c4009
      puts ("mapping failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  pthread_mutexattr_t ma;
Packit 6c4009
  if (pthread_mutexattr_init (&ma) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("mutexattr_init failed");
Packit 6c4009
      return 0;
Packit 6c4009
    }
Packit 6c4009
  if (pthread_mutexattr_setrobust_np (&ma, PTHREAD_MUTEX_ROBUST_NP) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("mutexattr_setrobust failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
  if (pthread_mutexattr_setpshared (&ma, PTHREAD_PROCESS_SHARED) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("mutexattr_setpshared failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
#ifdef ENABLE_PI
Packit 6c4009
  if (pthread_mutexattr_setprotocol (&ma, PTHREAD_PRIO_INHERIT) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("pthread_mutexattr_setprotocol failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
  for (int round = 1; round <= ROUNDS; ++round)
Packit 6c4009
    {
Packit 6c4009
      for (int n = 0; n < N; ++n)
Packit 6c4009
	{
Packit 6c4009
	  int e = pthread_mutex_init (&map[n], &ma);
Packit 6c4009
	  if (e == ENOTSUP)
Packit 6c4009
	    {
Packit 6c4009
#ifdef ENABLE_PI
Packit 6c4009
	      puts ("cannot support pshared robust PI mutexes");
Packit 6c4009
#else
Packit 6c4009
	      puts ("cannot support pshared robust mutexes");
Packit 6c4009
#endif
Packit 6c4009
	      return 0;
Packit 6c4009
	    }
Packit 6c4009
	  if (e != 0)
Packit 6c4009
	    {
Packit 6c4009
	      printf ("mutex_init %d in round %d failed\n", n + 1, round);
Packit 6c4009
	      return 1;
Packit 6c4009
	    }
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      pid_t p = fork ();
Packit 6c4009
      if (p == -1)
Packit 6c4009
	{
Packit 6c4009
	  printf ("fork in round %d failed\n", round);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
      if (p == 0)
Packit 6c4009
	child (round);
Packit 6c4009
Packit 6c4009
      int status;
Packit 6c4009
      if (TEMP_FAILURE_RETRY (waitpid (p, &status, 0)) != p)
Packit 6c4009
	{
Packit 6c4009
	  printf ("waitpid in round %d failed\n", round);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
      if (!WIFSIGNALED (status))
Packit 6c4009
	{
Packit 6c4009
	  printf ("child did not die of a signal in round %d\n", round);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
      if (WTERMSIG (status) != THESIGNAL)
Packit 6c4009
	{
Packit 6c4009
	  printf ("child did not die of signal %d in round %d\n",
Packit 6c4009
		  THESIGNAL, round);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      for (int n = 0; n < N; ++n)
Packit 6c4009
	{
Packit 6c4009
	  int e = pthread_mutex_lock (&map[n]);
Packit 6c4009
	  if (e != 0 && e != EOWNERDEAD)
Packit 6c4009
	    {
Packit 6c4009
	      printf ("mutex_lock %d failed in round %d\n", n + 1, round);
Packit 6c4009
	      return 1;
Packit 6c4009
	    }
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      for (int n = 0; n < N; ++n)
Packit 6c4009
	if (pthread_mutex_unlock (&map[n]) != 0)
Packit 6c4009
	  {
Packit 6c4009
	    printf ("mutex_unlock %d failed in round %d\n", n + 1, round);
Packit 6c4009
	    return 1;
Packit 6c4009
	  }
Packit 6c4009
Packit 6c4009
      for (int n = 0; n < N; ++n)
Packit 6c4009
	{
Packit 6c4009
	  int e = pthread_mutex_destroy (&map[n]);
Packit 6c4009
	  if (e != 0)
Packit 6c4009
	    {
Packit 6c4009
	      printf ("mutex_destroy %d in round %d failed with %d\n",
Packit 6c4009
		      n + 1, round, e);
Packit 6c4009
	      printf("nusers = %d\n", (int) map[n].__data.__nusers);
Packit 6c4009
	      return 1;
Packit 6c4009
	    }
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (pthread_mutexattr_destroy (&ma) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("mutexattr_destroy failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (munmap (map, N * sizeof (pthread_mutex_t)) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("munmap failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}