Blame nptl/tst-tls3.c

Packit Service 82fcde
/* Copyright (C) 2003-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Ulrich Drepper <drepper@redhat.com>, 2003.
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
#include <dlfcn.h>
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <pthread.h>
Packit Service 82fcde
#include <signal.h>
Packit Service 82fcde
#include <semaphore.h>
Packit Service 82fcde
#include <stdint.h>
Packit Service 82fcde
#include <stdio.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <unistd.h>
Packit Service 82fcde
#include <pthreaddef.h>
Packit Service 82fcde
Packit Service 82fcde
#define THE_SIG SIGUSR1
Packit Service 82fcde
Packit Service 82fcde
/* The stack size can be overriden.  With a sufficiently large stack
Packit Service 82fcde
   size, thread stacks for terminated threads are freed, but this does
Packit Service 82fcde
   not happen with the default size of 1 MiB.  */
Packit Service 82fcde
enum { default_stack_size_in_mb = 1 };
Packit Service 82fcde
static long stack_size_in_mb;
Packit Service 82fcde
Packit Service 82fcde
#define N 10
Packit Service 82fcde
static pthread_t th[N];
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static int do_test (void);
Packit Service 82fcde
Packit Service 82fcde
#define TIMEOUT 5
Packit Service 82fcde
#define TEST_FUNCTION do_test ()
Packit Service 82fcde
#include "../test-skeleton.c"
Packit Service 82fcde
Packit Service 82fcde
#define CB(n) \
Packit Service 82fcde
static void								      \
Packit Service 82fcde
cb##n (void)								      \
Packit Service 82fcde
{									      \
Packit Service 82fcde
  if (th[n] != pthread_self ())						      \
Packit Service 82fcde
    {									      \
Packit Service 82fcde
      write_message ("wrong callback\n");				      \
Packit Service 82fcde
      _exit (1);							      \
Packit Service 82fcde
    }									      \
Packit Service 82fcde
}
Packit Service 82fcde
CB (0)
Packit Service 82fcde
CB (1)
Packit Service 82fcde
CB (2)
Packit Service 82fcde
CB (3)
Packit Service 82fcde
CB (4)
Packit Service 82fcde
CB (5)
Packit Service 82fcde
CB (6)
Packit Service 82fcde
CB (7)
Packit Service 82fcde
CB (8)
Packit Service 82fcde
CB (9)
Packit Service 82fcde
static void (*cbs[]) (void) =
Packit Service 82fcde
{
Packit Service 82fcde
  cb0, cb1, cb2, cb3, cb4, cb5, cb6, cb7, cb8, cb9
Packit Service 82fcde
};
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
sem_t s;
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
pthread_barrier_t b;
Packit Service 82fcde
Packit Service 82fcde
#define TOTAL_SIGS 1000
Packit Service 82fcde
int nsigs;
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
do_test (void)
Packit Service 82fcde
{
Packit Service 82fcde
  if (stack_size_in_mb == 0)
Packit Service 82fcde
    stack_size_in_mb = default_stack_size_in_mb;
Packit Service 82fcde
Packit Service 82fcde
  if ((uintptr_t) pthread_self () & (TCB_ALIGNMENT - 1))
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("initial thread's struct pthread not aligned enough");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_barrier_init (&b, NULL, N + 1) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("barrier_init failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (sem_init (&s, 0, 0) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("sem_init failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  void *h = dlopen ("tst-tls3mod.so", RTLD_LAZY);
Packit Service 82fcde
  if (h == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("dlopen failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  void *(*tf) (void *) = dlsym (h, "tf");
Packit Service 82fcde
  if (tf == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("dlsym for tf failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  struct sigaction sa;
Packit Service 82fcde
  sa.sa_handler = dlsym (h, "handler");
Packit Service 82fcde
  if (sa.sa_handler == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("dlsym for handler failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
  sigemptyset (&sa.sa_mask);
Packit Service 82fcde
  sa.sa_flags = 0;
Packit Service 82fcde
  if (sigaction (THE_SIG, &sa, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("sigaction failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  pthread_attr_t a;
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_attr_init (&a) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("attr_init failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_attr_setstacksize (&a, stack_size_in_mb * 1024 * 1024) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("attr_setstacksize failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  int r;
Packit Service 82fcde
  for (r = 0; r < 10; ++r)
Packit Service 82fcde
    {
Packit Service 82fcde
      int i;
Packit Service 82fcde
      for (i = 0; i < N; ++i)
Packit Service 82fcde
	if (pthread_create (&th[i], &a, tf, cbs[i]) != 0)
Packit Service 82fcde
	  {
Packit Service 82fcde
	    puts ("pthread_create failed");
Packit Service 82fcde
	    exit (1);
Packit Service 82fcde
	  }
Packit Service 82fcde
Packit Service 82fcde
      nsigs = 0;
Packit Service 82fcde
Packit Service 82fcde
      pthread_barrier_wait (&b);
Packit Service 82fcde
Packit Service 82fcde
      sigset_t ss;
Packit Service 82fcde
      sigemptyset (&ss);
Packit Service 82fcde
      sigaddset (&ss, THE_SIG);
Packit Service 82fcde
      if (pthread_sigmask (SIG_BLOCK, &ss, NULL) != 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  puts ("pthread_sigmask failed");
Packit Service 82fcde
	  exit (1);
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      /* Start sending signals.  */
Packit Service 82fcde
      for (i = 0; i < TOTAL_SIGS; ++i)
Packit Service 82fcde
	{
Packit Service 82fcde
	  if (kill (getpid (), THE_SIG) != 0)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      puts ("kill failed");
Packit Service 82fcde
	      exit (1);
Packit Service 82fcde
	    }
Packit Service 82fcde
Packit Service 82fcde
	  if (TEMP_FAILURE_RETRY (sem_wait (&s)) != 0)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      puts ("sem_wait failed");
Packit Service 82fcde
	      exit (1);
Packit Service 82fcde
	    }
Packit Service 82fcde
Packit Service 82fcde
	  ++nsigs;
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      pthread_barrier_wait (&b);
Packit Service 82fcde
Packit Service 82fcde
      if (pthread_sigmask (SIG_UNBLOCK, &ss, NULL) != 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  puts ("pthread_sigmask failed");
Packit Service 82fcde
	  exit (1);
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      for (i = 0; i < N; ++i)
Packit Service 82fcde
	if (pthread_join (th[i], NULL) != 0)
Packit Service 82fcde
	  {
Packit Service 82fcde
	    puts ("join failed");
Packit Service 82fcde
	    exit (1);
Packit Service 82fcde
	  }
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_attr_destroy (&a) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("attr_destroy failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}