Blame nptl/tst-cancel20.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 Jakub Jelinek <jakub@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 <errno.h>
Packit Service 82fcde
#include <pthread.h>
Packit Service 82fcde
#include <signal.h>
Packit Service 82fcde
#include <stdio.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <unistd.h>
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static int fd[4];
Packit Service 82fcde
static pthread_barrier_t b;
Packit Service 82fcde
volatile int in_sh_body;
Packit Service 82fcde
unsigned long cleanups;
Packit Service 82fcde
Packit Service 82fcde
static void
Packit Service 82fcde
cl (void *arg)
Packit Service 82fcde
{
Packit Service 82fcde
  cleanups = (cleanups << 4) | (long) arg;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static void __attribute__((noinline))
Packit Service 82fcde
sh_body (void)
Packit Service 82fcde
{
Packit Service 82fcde
  char c;
Packit Service 82fcde
Packit Service 82fcde
  pthread_cleanup_push (cl, (void *) 1L);
Packit Service 82fcde
Packit Service 82fcde
  in_sh_body = 1;
Packit Service 82fcde
  if (read (fd[2], &c, 1) == 1)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("read succeeded");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  pthread_cleanup_pop (0);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static void
Packit Service 82fcde
sh (int sig)
Packit Service 82fcde
{
Packit Service 82fcde
  pthread_cleanup_push (cl, (void *) 2L);
Packit Service 82fcde
  sh_body ();
Packit Service 82fcde
  in_sh_body = 0;
Packit Service 82fcde
Packit Service 82fcde
  pthread_cleanup_pop (0);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static void __attribute__((noinline))
Packit Service 82fcde
tf_body (void)
Packit Service 82fcde
{
Packit Service 82fcde
  char c;
Packit Service 82fcde
Packit Service 82fcde
  pthread_cleanup_push (cl, (void *) 3L);
Packit Service 82fcde
Packit Service 82fcde
  int r = pthread_barrier_wait (&b);
Packit Service 82fcde
  if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("child thread: barrier_wait failed");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (read (fd[0], &c, 1) == 1)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("read succeeded");
Packit Service 82fcde
      exit (1);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  read (fd[0], &c, 1);
Packit Service 82fcde
Packit Service 82fcde
  pthread_cleanup_pop (0);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static void *
Packit Service 82fcde
tf (void *arg)
Packit Service 82fcde
{
Packit Service 82fcde
  pthread_cleanup_push (cl, (void *) 4L);
Packit Service 82fcde
  tf_body ();
Packit Service 82fcde
  pthread_cleanup_pop (0);
Packit Service 82fcde
  return NULL;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static int
Packit Service 82fcde
do_one_test (void)
Packit Service 82fcde
{
Packit Service 82fcde
  in_sh_body = 0;
Packit Service 82fcde
  cleanups = 0;
Packit Service 82fcde
  if (pipe (fd) != 0 || pipe (fd + 2) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("pipe failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  pthread_t th;
Packit Service 82fcde
  if (pthread_create (&th, NULL, tf, 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
  int r = pthread_barrier_wait (&b);
Packit Service 82fcde
  if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("parent thread: barrier_wait failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  sleep (1);
Packit Service 82fcde
Packit Service 82fcde
  r = pthread_kill (th, SIGHUP);
Packit Service 82fcde
  if (r)
Packit Service 82fcde
    {
Packit Service 82fcde
      errno = r;
Packit Service 82fcde
      printf ("pthread_kill failed %m\n");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  while (in_sh_body == 0)
Packit Service 82fcde
    sleep (1);
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_cancel (th) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("cancel failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  void *ret;
Packit Service 82fcde
  if (pthread_join (th, &ret) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("join failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (ret != PTHREAD_CANCELED)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("result is wrong");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (cleanups != 0x1234L)
Packit Service 82fcde
    {
Packit Service 82fcde
      printf ("called cleanups %lx\n", cleanups);
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* The pipe closing must be issued after the cancellation handling to avoid
Packit Service 82fcde
     a race condition where the cancellation runs after both pipe ends are
Packit Service 82fcde
     closed.  In this case the read syscall returns EOF and the cancellation
Packit Service 82fcde
     must not act.  */
Packit Service 82fcde
  close (fd[0]);
Packit Service 82fcde
  close (fd[1]);
Packit Service 82fcde
  close (fd[2]);
Packit Service 82fcde
  close (fd[3]);
Packit Service 82fcde
Packit Service 82fcde
  return 0;
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
  stack_t ss;
Packit Service 82fcde
  ss.ss_sp = malloc (2 * SIGSTKSZ);
Packit Service 82fcde
  if (ss.ss_sp == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("failed to allocate alternate stack");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
  ss.ss_flags = 0;
Packit Service 82fcde
  ss.ss_size = 2 * SIGSTKSZ;
Packit Service 82fcde
  if (sigaltstack (&ss, NULL) < 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      printf ("sigaltstack failed %m\n");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (pthread_barrier_init (&b, NULL, 2) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("barrier_init failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  struct sigaction sa;
Packit Service 82fcde
  sa.sa_handler = sh;
Packit Service 82fcde
  sigemptyset (&sa.sa_mask);
Packit Service 82fcde
  sa.sa_flags = 0;
Packit Service 82fcde
Packit Service 82fcde
  if (sigaction (SIGHUP, &sa, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("sigaction failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  puts ("sa_flags = 0 test");
Packit Service 82fcde
  if (do_one_test ())
Packit Service 82fcde
    return 1;
Packit Service 82fcde
Packit Service 82fcde
  sa.sa_handler = sh;
Packit Service 82fcde
  sigemptyset (&sa.sa_mask);
Packit Service 82fcde
  sa.sa_flags = SA_ONSTACK;
Packit Service 82fcde
Packit Service 82fcde
  if (sigaction (SIGHUP, &sa, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("sigaction failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  puts ("sa_flags = SA_ONSTACK test");
Packit Service 82fcde
  if (do_one_test ())
Packit Service 82fcde
    return 1;
Packit Service 82fcde
Packit Service 82fcde
#ifdef SA_SIGINFO
Packit Service 82fcde
  sa.sa_sigaction = (void (*)(int, siginfo_t *, void *)) sh;
Packit Service 82fcde
  sigemptyset (&sa.sa_mask);
Packit Service 82fcde
  sa.sa_flags = SA_SIGINFO;
Packit Service 82fcde
Packit Service 82fcde
  if (sigaction (SIGHUP, &sa, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("sigaction failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  puts ("sa_flags = SA_SIGINFO test");
Packit Service 82fcde
  if (do_one_test ())
Packit Service 82fcde
    return 1;
Packit Service 82fcde
Packit Service 82fcde
  sa.sa_sigaction = (void (*)(int, siginfo_t *, void *)) sh;
Packit Service 82fcde
  sigemptyset (&sa.sa_mask);
Packit Service 82fcde
  sa.sa_flags = SA_SIGINFO | SA_ONSTACK;
Packit Service 82fcde
Packit Service 82fcde
  if (sigaction (SIGHUP, &sa, NULL) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      puts ("sigaction failed");
Packit Service 82fcde
      return 1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  puts ("sa_flags = SA_SIGINFO|SA_ONSTACK test");
Packit Service 82fcde
  if (do_one_test ())
Packit Service 82fcde
    return 1;
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
#define TIMEOUT 40
Packit Service 82fcde
#define TEST_FUNCTION do_test ()
Packit Service 82fcde
#include "../test-skeleton.c"