aboutsummaryrefslogtreecommitdiff
blob: c13d9f7e727f564696e7998b3de016ca71770c73 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
#include <aio.h>
#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h>

static pthread_barrier_t b;
static pthread_t main_thread;
static int flag;


static void *
tf (void *arg)
{
  int e = pthread_barrier_wait (&b);
  if (e != 0 && e != PTHREAD_BARRIER_SERIAL_THREAD)
    {
      puts ("child: barrier_wait failed");
      exit (1);
    }

  /* There is unfortunately no other way to try to make sure the other
     thread reached the aio_suspend call.  This test could fail on
     highly loaded machines.  */
  sleep (2);

  pthread_kill (main_thread, SIGUSR1);

  while (1)
    sleep (1000);

  return NULL;
}


static void
sh (int sig)
{
  flag = 1;
}


static int
do_test (void)
{
  main_thread = pthread_self ();

  struct sigaction sa;

  sa.sa_handler = sh;
  sa.sa_flags = 0;
  sigemptyset (&sa.sa_mask);

  if (sigaction (SIGUSR1, &sa, NULL) != 0)
    {
      puts ("sigaction failed");
      return 1;
    }

  if (pthread_barrier_init (&b, NULL, 2) != 0)
    {
      puts ("barrier_init");
      return 1;
    }

  int fds[2];
  if (pipe (fds) != 0)
    {
      puts ("pipe failed");
      return 1;
    }

  char buf[42];
  struct aiocb req;
  req.aio_fildes = fds[0];
  req.aio_reqprio = 0;
  req.aio_offset = 0;
  req.aio_buf = buf;
  req.aio_nbytes = sizeof (buf);
  req.aio_sigevent.sigev_notify = SIGEV_NONE;

  if (aio_read (&req) != 0)
    {
      puts ("aio_read failed");
      return 1;
    }

  pthread_t th;
  if (pthread_create (&th, NULL, tf, NULL) != 0)
    {
      puts ("create failed");
      return 1;
    }

  int e = pthread_barrier_wait (&b);
  if (e != 0 && e != PTHREAD_BARRIER_SERIAL_THREAD)
    {
      puts ("parent: barrier_wait failed");
      exit (1);
    }

  const struct aiocb *list[1];
  list[0] = &req;

  e = aio_suspend (list, 1, NULL);
  if (e != -1)
    {
      puts ("aio_suspend succeeded");
      return 1;
    }
  if (errno != EINTR)
    {
      puts ("aio_suspend did not return EINTR");
      return 1;
    }

  return 0;
}

#define TEST_FUNCTION do_test ()
#include "../test-skeleton.c"