Blame nptl/tst-tls2.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 <errno.h>
Packit 6c4009
#include <pthread.h>
Packit 6c4009
#include <signal.h>
Packit 6c4009
#include <semaphore.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
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 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
static __thread void (*fp) (void) __attribute__ ((tls_model ("local-exec")));
Packit 6c4009
Packit 6c4009
Packit 6c4009
static sem_t s;
Packit 6c4009
Packit 6c4009
Packit 6c4009
#define THE_SIG SIGUSR1
Packit 6c4009
static void
Packit 6c4009
handler (int sig)
Packit 6c4009
{
Packit 6c4009
  if (sig != THE_SIG)
Packit 6c4009
    {
Packit 6c4009
      write_message ("wrong signal\n");
Packit 6c4009
      _exit (1);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  fp ();
Packit 6c4009
Packit 6c4009
  if (sem_post (&s) != 0)
Packit 6c4009
    {
Packit 6c4009
      write_message ("sem_post failed\n");
Packit 6c4009
      _exit (1);
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
static pthread_barrier_t b;
Packit 6c4009
Packit 6c4009
#define TOTAL_SIGS 1000
Packit 6c4009
static int nsigs;
Packit 6c4009
Packit 6c4009
Packit 6c4009
static void *
Packit 6c4009
tf (void *arg)
Packit 6c4009
{
Packit 6c4009
  fp = arg;
Packit 6c4009
Packit 6c4009
  pthread_barrier_wait (&b);
Packit 6c4009
Packit 6c4009
  pthread_barrier_wait (&b);
Packit 6c4009
Packit 6c4009
  if (nsigs != TOTAL_SIGS)
Packit 6c4009
    {
Packit 6c4009
      puts ("barrier_wait prematurely returns");
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return NULL;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
do_test (void)
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
  struct sigaction sa;
Packit 6c4009
  sa.sa_handler = handler;
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, 1 * 1024 * 1024) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("attr_setstacksize failed");
Packit 6c4009
      return 1;
Packit 6c4009
    }
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
  if (pthread_attr_destroy (&a) != 0)
Packit 6c4009
    {
Packit 6c4009
      puts ("attr_destroy failed");
Packit 6c4009
      exit (1);
Packit 6c4009
    }
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
  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
  return 0;
Packit 6c4009
}