source: trunk/src/grep/gnulib-tests/glthread/thread.h

Last change on this file was 3529, checked in by bird, 4 years ago

Imported grep 3.7 from grep-3.7.tar.gz (sha256: c22b0cf2d4f6bbe599c902387e8058990e1eee99aef333a203829e5fd3dbb342), applying minimal auto-props.

  • Property svn:eol-style set to native
File size: 11.1 KB
Line 
1/* Creating and controlling threads.
2 Copyright (C) 2005-2021 Free Software Foundation, Inc.
3
4 This file is free software: you can redistribute it and/or modify
5 it under the terms of the GNU Lesser General Public License as
6 published by the Free Software Foundation; either version 2.1 of the
7 License, or (at your option) any later version.
8
9 This file 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
12 GNU Lesser General Public License for more details.
13
14 You should have received a copy of the GNU Lesser General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
16
17/* Written by Bruno Haible <bruno@clisp.org>, 2005.
18 Based on GCC's gthr-posix.h, gthr-posix95.h, gthr-win32.h. */
19
20/* This file contains primitives for creating and controlling threads.
21
22 Thread data type: gl_thread_t.
23
24 Creating a thread:
25 thread = gl_thread_create (func, arg);
26 Or with control of error handling:
27 err = glthread_create (&thread, func, arg);
28 extern int glthread_create (gl_thread_t *result,
29 void *(*func) (void *), void *arg);
30
31 Querying and changing the signal mask of a thread (not supported on all
32 platforms):
33 gl_thread_sigmask (how, newmask, oldmask);
34 Or with control of error handling:
35 err = glthread_sigmask (how, newmask, oldmask);
36 extern int glthread_sigmask (int how, const sigset_t *newmask, sigset_t *oldmask);
37
38 Waiting for termination of another thread:
39 gl_thread_join (thread, &return_value);
40 Or with control of error handling:
41 err = glthread_join (thread, &return_value);
42 extern int glthread_join (gl_thread_t thread, void **return_value_ptr);
43
44 Getting a reference to the current thread:
45 current = gl_thread_self ();
46 extern gl_thread_t gl_thread_self (void);
47
48 Getting a reference to the current thread as a pointer, for debugging:
49 ptr = gl_thread_self_pointer ();
50 extern void * gl_thread_self_pointer (void);
51
52 Terminating the current thread:
53 gl_thread_exit (return_value);
54 extern _Noreturn void gl_thread_exit (void *return_value);
55
56 Requesting custom code to be executed at fork() time (not supported on all
57 platforms):
58 gl_thread_atfork (prepare_func, parent_func, child_func);
59 Or with control of error handling:
60 err = glthread_atfork (prepare_func, parent_func, child_func);
61 extern int glthread_atfork (void (*prepare_func) (void),
62 void (*parent_func) (void),
63 void (*child_func) (void));
64 Note that even on platforms where this is supported, use of fork() and
65 threads together is problematic, see
66 <https://lists.gnu.org/r/bug-gnulib/2008-08/msg00062.html>
67 */
68
69
70#ifndef _GLTHREAD_THREAD_H
71#define _GLTHREAD_THREAD_H
72
73#include <errno.h>
74#include <stdlib.h>
75
76#if !defined c11_threads_in_use
77# if HAVE_THREADS_H && USE_POSIX_THREADS_FROM_LIBC
78# define c11_threads_in_use() 1
79# elif HAVE_THREADS_H && USE_POSIX_THREADS_WEAK
80# include <threads.h>
81# pragma weak thrd_exit
82# define c11_threads_in_use() (thrd_exit != NULL)
83# else
84# define c11_threads_in_use() 0
85# endif
86#endif
87
88/* ========================================================================= */
89
90#if USE_ISOC_THREADS
91
92/* Use the ISO C threads library. */
93
94# include <threads.h>
95
96# ifdef __cplusplus
97extern "C" {
98# endif
99
100/* -------------------------- gl_thread_t datatype -------------------------- */
101
102typedef struct thrd_with_exitvalue *gl_thread_t;
103extern int glthread_create (gl_thread_t *threadp,
104 void *(*func) (void *), void *arg);
105# define glthread_sigmask(HOW, SET, OSET) \
106 pthread_sigmask (HOW, SET, OSET)
107extern int glthread_join (gl_thread_t thread, void **return_value_ptr);
108extern gl_thread_t gl_thread_self (void);
109# define gl_thread_self_pointer() \
110 (void *) gl_thread_self ()
111extern _Noreturn void gl_thread_exit (void *return_value);
112# define glthread_atfork(PREPARE_FUNC, PARENT_FUNC, CHILD_FUNC) 0
113
114# ifdef __cplusplus
115}
116# endif
117
118#endif
119
120/* ========================================================================= */
121
122#if USE_POSIX_THREADS || USE_ISOC_AND_POSIX_THREADS
123
124/* Use the POSIX threads library. */
125
126# include <pthread.h>
127
128/* On IRIX, pthread_atfork is declared in <unistd.h>, not in <pthread.h>. */
129# if defined __sgi
130# include <unistd.h>
131# endif
132
133# if USE_POSIX_THREADS_WEAK
134/* Compilers other than GCC need to see the declaration of pthread_sigmask
135 before the "#pragma weak pthread_sigmask" below. */
136# include <signal.h>
137# endif
138
139# ifdef __cplusplus
140extern "C" {
141# endif
142
143# if PTHREAD_IN_USE_DETECTION_HARD
144
145/* The pthread_in_use() detection needs to be done at runtime. */
146# define pthread_in_use() \
147 glthread_in_use ()
148extern int glthread_in_use (void);
149
150# endif
151
152# if USE_POSIX_THREADS_WEAK
153
154/* Use weak references to the POSIX threads library. */
155
156/* Weak references avoid dragging in external libraries if the other parts
157 of the program don't use them. Here we use them, because we don't want
158 every program that uses libintl to depend on libpthread. This assumes
159 that libpthread would not be loaded after libintl; i.e. if libintl is
160 loaded first, by an executable that does not depend on libpthread, and
161 then a module is dynamically loaded that depends on libpthread, libintl
162 will not be multithread-safe. */
163
164/* The way to test at runtime whether libpthread is present is to test
165 whether a function pointer's value, such as &pthread_mutex_init, is
166 non-NULL. However, some versions of GCC have a bug through which, in
167 PIC mode, &foo != NULL always evaluates to true if there is a direct
168 call to foo(...) in the same function. To avoid this, we test the
169 address of a function in libpthread that we don't use. */
170
171# ifndef pthread_sigmask /* Do not declare rpl_pthread_sigmask weak. */
172# pragma weak pthread_sigmask
173# endif
174
175# pragma weak pthread_join
176# ifndef pthread_self
177# pragma weak pthread_self
178# endif
179# pragma weak pthread_exit
180# if HAVE_PTHREAD_ATFORK
181# pragma weak pthread_atfork
182# endif
183
184# if !PTHREAD_IN_USE_DETECTION_HARD
185# pragma weak pthread_mutexattr_gettype
186# define pthread_in_use() \
187 (pthread_mutexattr_gettype != NULL || c11_threads_in_use ())
188# endif
189
190# else
191
192# if !PTHREAD_IN_USE_DETECTION_HARD
193# define pthread_in_use() 1
194# endif
195
196# endif
197
198/* -------------------------- gl_thread_t datatype -------------------------- */
199
200/* This choice of gl_thread_t assumes that
201 pthread_equal (a, b) is equivalent to ((a) == (b)).
202 This is the case on all platforms in use in 2008. */
203typedef pthread_t gl_thread_t;
204# define glthread_create(THREADP, FUNC, ARG) \
205 (pthread_in_use () ? pthread_create (THREADP, NULL, FUNC, ARG) : ENOSYS)
206# define glthread_sigmask(HOW, SET, OSET) \
207 (pthread_in_use () ? pthread_sigmask (HOW, SET, OSET) : 0)
208# define glthread_join(THREAD, RETVALP) \
209 (pthread_in_use () ? pthread_join (THREAD, RETVALP) : 0)
210# ifdef PTW32_VERSION
211 /* In pthreads-win32, pthread_t is a struct with a pointer field 'p' and
212 other fields. */
213# define gl_thread_self() \
214 (pthread_in_use () ? pthread_self () : gl_null_thread)
215# define gl_thread_self_pointer() \
216 (pthread_in_use () ? pthread_self ().p : NULL)
217extern const gl_thread_t gl_null_thread;
218# elif defined __MVS__
219 /* On IBM z/OS, pthread_t is a struct with an 8-byte '__' field.
220 The first three bytes of this field appear to uniquely identify a
221 pthread_t, though not necessarily representing a pointer. */
222# define gl_thread_self() \
223 (pthread_in_use () ? pthread_self () : gl_null_thread)
224# define gl_thread_self_pointer() \
225 (pthread_in_use () ? *((void **) pthread_self ().__) : NULL)
226extern const gl_thread_t gl_null_thread;
227# else
228# define gl_thread_self() \
229 (pthread_in_use () ? pthread_self () : (pthread_t) 0)
230# define gl_thread_self_pointer() \
231 (pthread_in_use () ? (void *) pthread_self () : NULL)
232# endif
233# define gl_thread_exit(RETVAL) \
234 (void) (pthread_in_use () ? (pthread_exit (RETVAL), 0) : 0)
235
236# if HAVE_PTHREAD_ATFORK
237# define glthread_atfork(PREPARE_FUNC, PARENT_FUNC, CHILD_FUNC) \
238 (pthread_in_use () ? pthread_atfork (PREPARE_FUNC, PARENT_FUNC, CHILD_FUNC) : 0)
239# else
240# define glthread_atfork(PREPARE_FUNC, PARENT_FUNC, CHILD_FUNC) 0
241# endif
242
243# ifdef __cplusplus
244}
245# endif
246
247#endif
248
249/* ========================================================================= */
250
251#if USE_WINDOWS_THREADS
252
253# define WIN32_LEAN_AND_MEAN /* avoid including junk */
254# include <windows.h>
255
256# include "windows-thread.h"
257
258# ifdef __cplusplus
259extern "C" {
260# endif
261
262/* -------------------------- gl_thread_t datatype -------------------------- */
263
264typedef glwthread_thread_t gl_thread_t;
265# define glthread_create(THREADP, FUNC, ARG) \
266 glwthread_thread_create (THREADP, 0, FUNC, ARG)
267# define glthread_sigmask(HOW, SET, OSET) \
268 /* unsupported */ 0
269# define glthread_join(THREAD, RETVALP) \
270 glwthread_thread_join (THREAD, RETVALP)
271# define gl_thread_self() \
272 glwthread_thread_self ()
273# define gl_thread_self_pointer() \
274 gl_thread_self ()
275# define gl_thread_exit(RETVAL) \
276 glwthread_thread_exit (RETVAL)
277# define glthread_atfork(PREPARE_FUNC, PARENT_FUNC, CHILD_FUNC) 0
278
279# ifdef __cplusplus
280}
281# endif
282
283#endif
284
285/* ========================================================================= */
286
287#if !(USE_ISOC_THREADS || USE_POSIX_THREADS || USE_ISOC_AND_POSIX_THREADS || USE_WINDOWS_THREADS)
288
289/* Provide dummy implementation if threads are not supported. */
290
291typedef int gl_thread_t;
292# define glthread_create(THREADP, FUNC, ARG) ENOSYS
293# define glthread_sigmask(HOW, SET, OSET) 0
294# define glthread_join(THREAD, RETVALP) 0
295# define gl_thread_self() 0
296# define gl_thread_self_pointer() \
297 ((void *) gl_thread_self ())
298# define gl_thread_exit(RETVAL) (void)0
299# define glthread_atfork(PREPARE_FUNC, PARENT_FUNC, CHILD_FUNC) 0
300
301#endif
302
303/* ========================================================================= */
304
305/* Macros with built-in error handling. */
306
307#ifdef __cplusplus
308extern "C" {
309#endif
310
311extern gl_thread_t gl_thread_create (void *(*func) (void *arg), void *arg);
312#define gl_thread_sigmask(HOW, SET, OSET) \
313 do \
314 { \
315 if (glthread_sigmask (HOW, SET, OSET)) \
316 abort (); \
317 } \
318 while (0)
319#define gl_thread_join(THREAD, RETVAL) \
320 do \
321 { \
322 if (glthread_join (THREAD, RETVAL)) \
323 abort (); \
324 } \
325 while (0)
326#define gl_thread_atfork(PREPARE, PARENT, CHILD) \
327 do \
328 { \
329 if (glthread_atfork (PREPARE, PARENT, CHILD)) \
330 abort (); \
331 } \
332 while (0)
333
334#ifdef __cplusplus
335}
336#endif
337
338#endif /* _GLTHREAD_THREAD_H */
Note: See TracBrowser for help on using the repository browser.