(root)/
sed-4.9/
gnulib-tests/
glthread/
thread.h
       1  /* Creating and controlling threads.
       2     Copyright (C) 2005-2022 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
      97  extern "C" {
      98  # endif
      99  
     100  /* -------------------------- gl_thread_t datatype -------------------------- */
     101  
     102  typedef struct thrd_with_exitvalue *gl_thread_t;
     103  extern 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)
     107  extern int glthread_join (gl_thread_t thread, void **return_value_ptr);
     108  extern gl_thread_t gl_thread_self (void);
     109  # define gl_thread_self_pointer() \
     110      (void *) gl_thread_self ()
     111  extern _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
     140  extern "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 ()
     148  extern 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.  */
     203  typedef 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)
     217  extern 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)
     226  extern 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_t) 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
     259  extern "C" {
     260  # endif
     261  
     262  /* -------------------------- gl_thread_t datatype -------------------------- */
     263  
     264  typedef 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  
     291  typedef 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
     308  extern "C" {
     309  #endif
     310  
     311  extern 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 */