1/* Copyright (C) 2003-2024 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
8
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
13
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <https://www.gnu.org/licenses/>. */
17
18#include <errno.h>
19#include <pthread.h>
20#include <stdio.h>
21#include <stdlib.h>
22#include <string.h>
23#include <unistd.h>
24#include <sys/mman.h>
25#include <sys/wait.h>
26
27
28static pthread_barrier_t b2;
29static int fd;
30static int called;
31
32
33static void
34cl (void *arg)
35{
36 called = 1;
37}
38
39
40static void *
41tf (void *arg)
42{
43 int r = pthread_barrier_wait (barrier: &b2);
44 if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
45 {
46 puts (s: "child thread: barrier_wait failed");
47 exit (1);
48 }
49
50 pthread_cleanup_push (cl, NULL);
51
52 /* This call should never return. */
53 if (lockf (fd: fd, F_LOCK, len: 0))
54 {
55 puts (s: "child thread: lockf failed");
56 exit (1);
57 }
58
59 pthread_cleanup_pop (0);
60
61 return NULL;
62}
63
64
65static int
66do_test (void)
67{
68 char fname[] = "/tmp/cancel16XXXXXX";
69 fd = mkstemp (template: fname);
70 if (fd == -1)
71 {
72 puts (s: "mkstemp failed");
73 return 1;
74 }
75 unlink (name: fname);
76
77 char mem[sizeof (pthread_barrier_t)];
78 memset (mem, '\0', sizeof (mem));
79 if (TEMP_FAILURE_RETRY (pwrite (fd, mem, sizeof (mem), 0)) != sizeof (mem))
80 {
81 puts (s: "pwrite failed");
82 return 1;
83 }
84
85 void *p = mmap (NULL, len: sizeof (mem), PROT_READ|PROT_WRITE, MAP_SHARED, fd: fd, offset: 0);
86 if (p == MAP_FAILED)
87 {
88 puts (s: "mmap failed");
89 return 1;
90 }
91 pthread_barrier_t *b = (pthread_barrier_t *) p;
92
93 pthread_barrierattr_t ba;
94 if (pthread_barrierattr_init (attr: &ba) != 0)
95 {
96 puts (s: "barrierattr_init failed");
97 return 1;
98 }
99 if (pthread_barrierattr_setpshared (attr: &ba, pshared: 1) != 0)
100 {
101 puts (s: "barrierattr_setshared failed");
102 return 1;
103 }
104
105 if (pthread_barrier_init (barrier: b, attr: &ba, count: 2) != 0)
106 {
107 puts (s: "1st barrier_init failed");
108 return 1;
109 }
110 if (pthread_barrierattr_destroy (attr: &ba) != 0)
111 {
112 puts (s: "barrier_destroy failed");
113 return 1;
114 }
115
116 pid_t pid = fork ();
117 if (pid == 0)
118 {
119 /* Child. Lock the file and wait. */
120 if (lockf (fd: fd, F_LOCK, len: 0) != 0)
121 {
122 puts (s: "child process: lockf failed");
123 _exit (1);
124 }
125
126 int r = pthread_barrier_wait (barrier: b);
127 if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
128 {
129 puts (s: "child process: 1st barrier_wait failed");
130 _exit (1);
131 }
132
133 /* Make sure the process dies. */
134 alarm (5);
135
136 r = pthread_barrier_wait (barrier: b);
137 if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
138 {
139 puts (s: "child process: 2nd barrier_wait failed");
140 _exit (1);
141 }
142
143 _exit (0);
144 }
145 if (pid == -1)
146 {
147 puts (s: "fork failed");
148 return 1;
149 }
150
151 int r = pthread_barrier_wait (barrier: b);
152 if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
153 {
154 puts (s: "main: 1st barrier_wait failed");
155 _exit (1);
156 }
157
158 if (pthread_barrier_init (barrier: &b2, NULL, count: 2) != 0)
159 {
160 puts (s: "2nd barrier_init failed");
161 return 1;
162 }
163
164 pthread_t th;
165 if (pthread_create (newthread: &th, NULL, start_routine: tf, NULL) != 0)
166 {
167 puts (s: "create failed");
168 return 1;
169 }
170
171 r = pthread_barrier_wait (barrier: &b2);
172 if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
173 {
174 puts (s: "main: 2nd barrier_wait failed");
175 return 1;
176 }
177
178 /* Delay. */
179 sleep (seconds: 1);
180
181 if (pthread_cancel (th: th) != 0)
182 {
183 puts (s: "cancel failed");
184 return 1;
185 }
186
187 void *result;
188 if (pthread_join (th: th, thread_return: &result) != 0)
189 {
190 puts (s: "join failed");
191 return 1;
192 }
193 if (result != PTHREAD_CANCELED)
194 {
195 puts (s: "thread not canceled");
196 return 1;
197 }
198 if (called == 0)
199 {
200 puts (s: "cleanup handler not called");
201 return 1;
202 }
203
204 r = pthread_barrier_wait (barrier: b);
205 if (r != 0 && r != PTHREAD_BARRIER_SERIAL_THREAD)
206 {
207 puts (s: "main: 3rd barrier_wait failed");
208 return 1;
209 }
210
211 int status;
212 if (TEMP_FAILURE_RETRY (waitpid (pid, &status, 0)) != pid)
213 {
214 puts (s: "waitpid failed");
215 return 1;
216 }
217 if (WEXITSTATUS (status) != 0)
218 {
219 printf (format: "child process exits with %d\n", WEXITSTATUS (status));
220 return 1;
221 }
222
223 if (lockf (fd: fd, F_LOCK, len: 0) != 0)
224 {
225 puts (s: "main: lockf failed");
226 return 1;
227 }
228
229 return 0;
230}
231
232#define TEST_FUNCTION do_test ()
233#include "../test-skeleton.c"
234

source code of glibc/sysdeps/pthread/tst-cancel16.c