Blame nptl/tst-tls3.c

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