1 /* Threads compatibility routines for libgcc2 and libobjc. */
2 /* Compile this one with gcc. */
3 /* Copyright (C) 1997-2024 Free Software Foundation, Inc.
5 This file is part of GCC.
7 GCC is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 3, or (at your option) any later
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
17 Under Section 7 of GPL version 3, you are granted additional
18 permissions described in the GCC Runtime Library Exception, version
19 3.1, as published by the Free Software Foundation.
21 You should have received a copy of the GNU General Public License and
22 a copy of the GCC Runtime Library Exception along with this program;
23 see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
24 <http://www.gnu.org/licenses/>. */
26 #ifndef GCC_GTHR_POSIX_H
27 #define GCC_GTHR_POSIX_H
29 /* POSIX threads specific definitions.
30 Easy, since the interface is just one-to-one mapping. */
33 #define __GTHREADS_CXX0X 1
37 #if ((defined(_LIBOBJC) || defined(_LIBOBJC_WEAK)) \
38 || !defined(_GTHREAD_USE_MUTEX_TIMEDLOCK))
40 # if defined(_POSIX_TIMEOUTS) && _POSIX_TIMEOUTS >= 0
41 # define _GTHREAD_USE_MUTEX_TIMEDLOCK 1
43 # define _GTHREAD_USE_MUTEX_TIMEDLOCK 0
47 #ifdef __has_attribute
48 # if __has_attribute(__always_inline__)
49 # define __GTHREAD_ALWAYS_INLINE __attribute__((__always_inline__))
52 #ifndef __GTHREAD_ALWAYS_INLINE
53 # define __GTHREAD_ALWAYS_INLINE
57 # define __GTHREAD_INLINE inline __GTHREAD_ALWAYS_INLINE
59 # define __GTHREAD_INLINE static inline
62 typedef pthread_t __gthread_t
;
63 typedef pthread_key_t __gthread_key_t
;
64 typedef pthread_once_t __gthread_once_t
;
65 typedef pthread_mutex_t __gthread_mutex_t
;
67 typedef pthread_rwlock_t __gthread_rwlock_t
;
69 typedef pthread_mutex_t __gthread_recursive_mutex_t
;
70 typedef pthread_cond_t __gthread_cond_t
;
71 typedef struct timespec __gthread_time_t
;
73 /* POSIX like conditional variables are supported. Please look at comments
74 in gthr.h for details. */
75 #define __GTHREAD_HAS_COND 1
77 #define __GTHREAD_MUTEX_INIT PTHREAD_MUTEX_INITIALIZER
78 #define __GTHREAD_MUTEX_INIT_FUNCTION __gthread_mutex_init_function
80 #define __GTHREAD_RWLOCK_INIT PTHREAD_RWLOCK_INITIALIZER
82 #define __GTHREAD_ONCE_INIT PTHREAD_ONCE_INIT
83 #if defined(PTHREAD_RECURSIVE_MUTEX_INITIALIZER)
84 #define __GTHREAD_RECURSIVE_MUTEX_INIT PTHREAD_RECURSIVE_MUTEX_INITIALIZER
85 #elif defined(PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP)
86 #define __GTHREAD_RECURSIVE_MUTEX_INIT PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP
88 #define __GTHREAD_RECURSIVE_MUTEX_INIT_FUNCTION __gthread_recursive_mutex_init_function
90 #define __GTHREAD_COND_INIT PTHREAD_COND_INITIALIZER
91 #define __GTHREAD_TIME_INIT {0,0}
93 #ifdef _GTHREAD_USE_MUTEX_INIT_FUNC
94 # undef __GTHREAD_MUTEX_INIT
96 #ifdef _GTHREAD_USE_RECURSIVE_MUTEX_INIT_FUNC
97 # undef __GTHREAD_RECURSIVE_MUTEX_INIT
98 # undef __GTHREAD_RECURSIVE_MUTEX_INIT_FUNCTION
99 # define __GTHREAD_RECURSIVE_MUTEX_INIT_FUNCTION __gthread_recursive_mutex_init_function
101 #ifdef _GTHREAD_USE_COND_INIT_FUNC
102 # undef __GTHREAD_COND_INIT
103 # define __GTHREAD_COND_INIT_FUNCTION __gthread_cond_init_function
106 #if SUPPORTS_WEAK && GTHREAD_USE_WEAK
107 # ifndef __gthrw_pragma
108 # define __gthrw_pragma(pragma)
110 # define __gthrw2(name,name2,type) \
111 static __typeof(type) name \
112 __attribute__ ((__weakref__(#name2), __copy__ (type))); \
113 __gthrw_pragma(weak type)
114 # define __gthrw_(name) __gthrw_ ## name
116 # define __gthrw2(name,name2,type)
117 # define __gthrw_(name) name
120 /* Typically, __gthrw_foo is a weak reference to symbol foo. */
121 #define __gthrw(name) __gthrw2(__gthrw_ ## name,name,name)
123 __gthrw(pthread_once
)
124 __gthrw(pthread_getspecific
)
125 __gthrw(pthread_setspecific
)
127 __gthrw(pthread_create
)
128 __gthrw(pthread_join
)
129 __gthrw(pthread_equal
)
130 __gthrw(pthread_self
)
131 __gthrw(pthread_detach
)
133 __gthrw(pthread_cancel
)
137 __gthrw(pthread_mutex_lock
)
138 __gthrw(pthread_mutex_trylock
)
139 #if _GTHREAD_USE_MUTEX_TIMEDLOCK
140 __gthrw(pthread_mutex_timedlock
)
142 __gthrw(pthread_mutex_unlock
)
143 __gthrw(pthread_mutex_init
)
144 __gthrw(pthread_mutex_destroy
)
146 __gthrw(pthread_cond_init
)
147 __gthrw(pthread_cond_broadcast
)
148 __gthrw(pthread_cond_signal
)
149 __gthrw(pthread_cond_wait
)
150 __gthrw(pthread_cond_timedwait
)
151 __gthrw(pthread_cond_destroy
)
153 __gthrw(pthread_key_create
)
154 __gthrw(pthread_key_delete
)
155 __gthrw(pthread_mutexattr_init
)
156 __gthrw(pthread_mutexattr_settype
)
157 __gthrw(pthread_mutexattr_destroy
)
160 __gthrw(pthread_rwlock_rdlock
)
161 __gthrw(pthread_rwlock_tryrdlock
)
162 __gthrw(pthread_rwlock_wrlock
)
163 __gthrw(pthread_rwlock_trywrlock
)
164 __gthrw(pthread_rwlock_unlock
)
167 #if defined(_LIBOBJC) || defined(_LIBOBJC_WEAK)
169 __gthrw(pthread_exit
)
170 #ifdef _POSIX_PRIORITY_SCHEDULING
171 #ifdef _POSIX_THREAD_PRIORITY_SCHEDULING
172 __gthrw(sched_get_priority_max
)
173 __gthrw(sched_get_priority_min
)
174 #endif /* _POSIX_THREAD_PRIORITY_SCHEDULING */
175 #endif /* _POSIX_PRIORITY_SCHEDULING */
176 __gthrw(pthread_attr_destroy
)
177 __gthrw(pthread_attr_init
)
178 __gthrw(pthread_attr_setdetachstate
)
179 #ifdef _POSIX_THREAD_PRIORITY_SCHEDULING
180 __gthrw(pthread_getschedparam
)
181 __gthrw(pthread_setschedparam
)
182 #endif /* _POSIX_THREAD_PRIORITY_SCHEDULING */
183 #endif /* _LIBOBJC || _LIBOBJC_WEAK */
185 #if SUPPORTS_WEAK && GTHREAD_USE_WEAK
187 /* On Solaris 2.6 up to 9, the libc exposes a POSIX threads interface even if
188 -pthreads is not specified. The functions are dummies and most return an
189 error value. However pthread_once returns 0 without invoking the routine
190 it is passed so we cannot pretend that the interface is active if -pthreads
191 is not specified. On Solaris 2.5.1, the interface is not exposed at all so
192 we need to play the usual game with weak symbols. On Solaris 10 and up, a
193 working interface is always exposed. On FreeBSD 6 and later, libc also
194 exposes a dummy POSIX threads interface, similar to what Solaris 2.6 up
195 to 9 does. FreeBSD >= 700014 even provides a pthread_cancel stub in libc,
196 which means the alternate __gthread_active_p below cannot be used there. */
198 #if defined(__FreeBSD__) || (defined(__sun) && defined(__svr4__))
200 #pragma GCC visibility push(hidden)
201 __GTHREAD_INLINE
volatile int *
202 __gthread_active (void)
204 static volatile int __gthread_active_var
= -1;
205 return &__gthread_active_var
;
207 #pragma GCC visibility pop
209 __GTHREAD_INLINE
void
210 __gthread_trigger (void)
212 *__gthread_active () = 1;
215 #pragma GCC visibility push(hidden)
217 __gthread_active_p (void)
219 static pthread_mutex_t __gthread_active_mutex
= PTHREAD_MUTEX_INITIALIZER
;
220 static pthread_once_t __gthread_active_once
= PTHREAD_ONCE_INIT
;
222 /* Avoid reading __gthread_active twice on the main code path. */
223 int __gthread_active_latest_value
= *__gthread_active ();
225 /* This test is not protected to avoid taking a lock on the main code
226 path so every update of __gthread_active in a threaded program must
227 be atomic with regard to the result of the test. */
228 if (__builtin_expect (__gthread_active_latest_value
< 0, 0))
230 if (__gthrw_(pthread_once
))
232 /* If this really is a threaded program, then we must ensure that
233 __gthread_active has been set to 1 before exiting this block. */
234 __gthrw_(pthread_mutex_lock
) (&__gthread_active_mutex
);
235 __gthrw_(pthread_once
) (&__gthread_active_once
, __gthread_trigger
);
236 __gthrw_(pthread_mutex_unlock
) (&__gthread_active_mutex
);
239 /* Make sure we'll never enter this block again. */
240 if (*__gthread_active () < 0)
241 *__gthread_active () = 0;
243 __gthread_active_latest_value
= *__gthread_active ();
246 return __gthread_active_latest_value
!= 0;
248 #pragma GCC visibility pop
250 #else /* neither FreeBSD nor Solaris */
252 /* For a program to be multi-threaded the only thing that it certainly must
253 be using is pthread_create. However, there may be other libraries that
254 intercept pthread_create with their own definitions to wrap pthreads
255 functionality for some purpose. In those cases, pthread_create being
256 defined might not necessarily mean that libpthread is actually linked
259 For the GNU C library, we can use a known internal name. This is always
260 available in the ABI, but no other library would define it. That is
261 ideal, since any public pthread function might be intercepted just as
262 pthread_create might be. __pthread_key_create is an "internal"
263 implementation symbol, but it is part of the public exported ABI. Also,
264 it's among the symbols that the static libpthread.a always links in
265 whenever pthread_create is used, so there is no danger of a false
266 negative result in any statically-linked, multi-threaded program.
268 For others, we choose pthread_cancel as a function that seems unlikely
269 to be redefined by an interceptor library. The bionic (Android) C
270 library does not provide pthread_cancel, so we do use pthread_create
271 there (and interceptor libraries lose). */
274 __gthrw2(__gthrw_(__pthread_key_create
),
275 __pthread_key_create
,
277 # define GTHR_ACTIVE_PROXY __gthrw_(__pthread_key_create)
278 #elif defined (__BIONIC__)
279 # define GTHR_ACTIVE_PROXY __gthrw_(pthread_create)
281 # define GTHR_ACTIVE_PROXY __gthrw_(pthread_cancel)
284 #pragma GCC visibility push(hidden)
286 __gthread_active_p (void)
288 static void *const __gthread_active_ptr
289 = __extension__ (void *) >HR_ACTIVE_PROXY
;
290 return __gthread_active_ptr
!= 0;
292 #pragma GCC visibility pop
294 #endif /* FreeBSD or Solaris */
296 #else /* not SUPPORTS_WEAK */
298 /* Similar to Solaris, HP-UX 11 for PA-RISC provides stubs for pthread
299 calls in shared flavors of the HP-UX C library. Most of the stubs
300 have no functionality. The details are described in the "libc cumulative
301 patch" for each subversion of HP-UX 11. There are two special interfaces
302 provided for checking whether an application is linked to a shared pthread
303 library or not. However, these interfaces aren't available in early
304 libpthread libraries. We also need a test that works for archive
305 libraries. We can't use pthread_once as some libc versions call the
306 init function. We also can't use pthread_create or pthread_attr_init
307 as these create a thread and thereby prevent changing the default stack
308 size. The function pthread_default_stacksize_np is available in both
309 the archive and shared versions of libpthread. It can be used to
310 determine the default pthread stack size. There is a stub in some
311 shared libc versions which returns a zero size if pthreads are not
312 active. We provide an equivalent stub to handle cases where libc
313 doesn't provide one. */
315 #if defined(__hppa__) && defined(__hpux__)
317 #pragma GCC visibility push(hidden)
318 __GTHREAD_INLINE
volatile int *
319 __gthread_active (void)
321 static volatile int __gthread_active_var
= -1;
322 return &__gthread_active_var
;
324 #pragma GCC visibility pop
327 __gthread_active_p (void)
329 /* Avoid reading __gthread_active twice on the main code path. */
330 int __gthread_active_latest_value
= *__gthread_active ();
333 if (__builtin_expect (__gthread_active_latest_value
< 0, 0))
335 pthread_default_stacksize_np (0, &__s
);
336 *__gthread_active () = __s
? 1 : 0;
337 __gthread_active_latest_value
= *__gthread_active ();
340 return __gthread_active_latest_value
!= 0;
343 #else /* not hppa-hpux */
346 __gthread_active_p (void)
351 #endif /* hppa-hpux */
353 #endif /* SUPPORTS_WEAK */
357 /* This is the config.h file in libobjc/ */
364 /* Key structure for maintaining thread specific storage */
365 static pthread_key_t _objc_thread_storage
;
366 static pthread_attr_t _objc_thread_attribs
;
368 /* Thread local storage for a single thread */
369 static void *thread_local_storage
= NULL
;
371 /* Backend initialization functions */
373 /* Initialize the threads subsystem. */
375 __gthread_objc_init_thread_system (void)
377 if (__gthread_active_p ())
379 /* Initialize the thread storage key. */
380 if (__gthrw_(pthread_key_create
) (&_objc_thread_storage
, NULL
) == 0)
382 /* The normal default detach state for threads is
383 * PTHREAD_CREATE_JOINABLE which causes threads to not die
384 * when you think they should. */
385 if (__gthrw_(pthread_attr_init
) (&_objc_thread_attribs
) == 0
386 && __gthrw_(pthread_attr_setdetachstate
) (&_objc_thread_attribs
,
387 PTHREAD_CREATE_DETACHED
) == 0)
395 /* Close the threads subsystem. */
397 __gthread_objc_close_thread_system (void)
399 if (__gthread_active_p ()
400 && __gthrw_(pthread_key_delete
) (_objc_thread_storage
) == 0
401 && __gthrw_(pthread_attr_destroy
) (&_objc_thread_attribs
) == 0)
407 /* Backend thread functions */
409 /* Create a new thread of execution. */
410 static inline objc_thread_t
411 __gthread_objc_thread_detach (void (*func
)(void *), void *arg
)
413 objc_thread_t thread_id
;
414 pthread_t new_thread_handle
;
416 if (!__gthread_active_p ())
419 if (!(__gthrw_(pthread_create
) (&new_thread_handle
, &_objc_thread_attribs
,
420 (void *) func
, arg
)))
421 thread_id
= (objc_thread_t
) new_thread_handle
;
428 /* Set the current thread's priority. */
430 __gthread_objc_thread_set_priority (int priority
)
432 if (!__gthread_active_p ())
436 #ifdef _POSIX_PRIORITY_SCHEDULING
437 #ifdef _POSIX_THREAD_PRIORITY_SCHEDULING
438 pthread_t thread_id
= __gthrw_(pthread_self
) ();
440 struct sched_param params
;
441 int priority_min
, priority_max
;
443 if (__gthrw_(pthread_getschedparam
) (thread_id
, &policy
, ¶ms
) == 0)
445 if ((priority_max
= __gthrw_(sched_get_priority_max
) (policy
)) == -1)
448 if ((priority_min
= __gthrw_(sched_get_priority_min
) (policy
)) == -1)
451 if (priority
> priority_max
)
452 priority
= priority_max
;
453 else if (priority
< priority_min
)
454 priority
= priority_min
;
455 params
.sched_priority
= priority
;
458 * The solaris 7 and several other man pages incorrectly state that
459 * this should be a pointer to policy but pthread.h is universally
462 if (__gthrw_(pthread_setschedparam
) (thread_id
, policy
, ¶ms
) == 0)
465 #endif /* _POSIX_THREAD_PRIORITY_SCHEDULING */
466 #endif /* _POSIX_PRIORITY_SCHEDULING */
471 /* Return the current thread's priority. */
473 __gthread_objc_thread_get_priority (void)
475 #ifdef _POSIX_PRIORITY_SCHEDULING
476 #ifdef _POSIX_THREAD_PRIORITY_SCHEDULING
477 if (__gthread_active_p ())
480 struct sched_param params
;
482 if (__gthrw_(pthread_getschedparam
) (__gthrw_(pthread_self
) (), &policy
, ¶ms
) == 0)
483 return params
.sched_priority
;
488 #endif /* _POSIX_THREAD_PRIORITY_SCHEDULING */
489 #endif /* _POSIX_PRIORITY_SCHEDULING */
490 return OBJC_THREAD_INTERACTIVE_PRIORITY
;
493 /* Yield our process time to another thread. */
495 __gthread_objc_thread_yield (void)
497 if (__gthread_active_p ())
498 __gthrw_(sched_yield
) ();
501 /* Terminate the current thread. */
503 __gthread_objc_thread_exit (void)
505 if (__gthread_active_p ())
506 /* exit the thread */
507 __gthrw_(pthread_exit
) (&__objc_thread_exit_status
);
509 /* Failed if we reached here */
513 /* Returns an integer value which uniquely describes a thread. */
514 static inline objc_thread_t
515 __gthread_objc_thread_id (void)
517 if (__gthread_active_p ())
518 return (objc_thread_t
) __gthrw_(pthread_self
) ();
520 return (objc_thread_t
) 1;
523 /* Sets the thread's local storage pointer. */
525 __gthread_objc_thread_set_data (void *value
)
527 if (__gthread_active_p ())
528 return __gthrw_(pthread_setspecific
) (_objc_thread_storage
, value
);
531 thread_local_storage
= value
;
536 /* Returns the thread's local storage pointer. */
538 __gthread_objc_thread_get_data (void)
540 if (__gthread_active_p ())
541 return __gthrw_(pthread_getspecific
) (_objc_thread_storage
);
543 return thread_local_storage
;
546 /* Backend mutex functions */
548 /* Allocate a mutex. */
550 __gthread_objc_mutex_allocate (objc_mutex_t mutex
)
552 if (__gthread_active_p ())
554 mutex
->backend
= objc_malloc (sizeof (pthread_mutex_t
));
556 if (__gthrw_(pthread_mutex_init
) ((pthread_mutex_t
*) mutex
->backend
, NULL
))
558 objc_free (mutex
->backend
);
559 mutex
->backend
= NULL
;
567 /* Deallocate a mutex. */
569 __gthread_objc_mutex_deallocate (objc_mutex_t mutex
)
571 if (__gthread_active_p ())
576 * Posix Threads specifically require that the thread be unlocked
577 * for __gthrw_(pthread_mutex_destroy) to work.
582 count
= __gthrw_(pthread_mutex_unlock
) ((pthread_mutex_t
*) mutex
->backend
);
588 if (__gthrw_(pthread_mutex_destroy
) ((pthread_mutex_t
*) mutex
->backend
))
591 objc_free (mutex
->backend
);
592 mutex
->backend
= NULL
;
597 /* Grab a lock on a mutex. */
599 __gthread_objc_mutex_lock (objc_mutex_t mutex
)
601 if (__gthread_active_p ()
602 && __gthrw_(pthread_mutex_lock
) ((pthread_mutex_t
*) mutex
->backend
) != 0)
610 /* Try to grab a lock on a mutex. */
612 __gthread_objc_mutex_trylock (objc_mutex_t mutex
)
614 if (__gthread_active_p ()
615 && __gthrw_(pthread_mutex_trylock
) ((pthread_mutex_t
*) mutex
->backend
) != 0)
623 /* Unlock the mutex */
625 __gthread_objc_mutex_unlock (objc_mutex_t mutex
)
627 if (__gthread_active_p ()
628 && __gthrw_(pthread_mutex_unlock
) ((pthread_mutex_t
*) mutex
->backend
) != 0)
636 /* Backend condition mutex functions */
638 /* Allocate a condition. */
640 __gthread_objc_condition_allocate (objc_condition_t condition
)
642 if (__gthread_active_p ())
644 condition
->backend
= objc_malloc (sizeof (pthread_cond_t
));
646 if (__gthrw_(pthread_cond_init
) ((pthread_cond_t
*) condition
->backend
, NULL
))
648 objc_free (condition
->backend
);
649 condition
->backend
= NULL
;
657 /* Deallocate a condition. */
659 __gthread_objc_condition_deallocate (objc_condition_t condition
)
661 if (__gthread_active_p ())
663 if (__gthrw_(pthread_cond_destroy
) ((pthread_cond_t
*) condition
->backend
))
666 objc_free (condition
->backend
);
667 condition
->backend
= NULL
;
672 /* Wait on the condition */
674 __gthread_objc_condition_wait (objc_condition_t condition
, objc_mutex_t mutex
)
676 if (__gthread_active_p ())
677 return __gthrw_(pthread_cond_wait
) ((pthread_cond_t
*) condition
->backend
,
678 (pthread_mutex_t
*) mutex
->backend
);
683 /* Wake up all threads waiting on this condition. */
685 __gthread_objc_condition_broadcast (objc_condition_t condition
)
687 if (__gthread_active_p ())
688 return __gthrw_(pthread_cond_broadcast
) ((pthread_cond_t
*) condition
->backend
);
693 /* Wake up one thread waiting on this condition. */
695 __gthread_objc_condition_signal (objc_condition_t condition
)
697 if (__gthread_active_p ())
698 return __gthrw_(pthread_cond_signal
) ((pthread_cond_t
*) condition
->backend
);
706 __gthread_create (__gthread_t
*__threadid
, void *(*__func
) (void*),
709 return __gthrw_(pthread_create
) (__threadid
, NULL
, __func
, __args
);
713 __gthread_join (__gthread_t __threadid
, void **__value_ptr
)
715 return __gthrw_(pthread_join
) (__threadid
, __value_ptr
);
719 __gthread_detach (__gthread_t __threadid
)
721 return __gthrw_(pthread_detach
) (__threadid
);
725 __gthread_equal (__gthread_t __t1
, __gthread_t __t2
)
727 return __gthrw_(pthread_equal
) (__t1
, __t2
);
730 __GTHREAD_INLINE __gthread_t
731 __gthread_self (void)
733 return __gthrw_(pthread_self
) ();
737 __gthread_yield (void)
739 return __gthrw_(sched_yield
) ();
743 __gthread_once (__gthread_once_t
*__once
, void (*__func
) (void))
745 if (__gthread_active_p ())
746 return __gthrw_(pthread_once
) (__once
, __func
);
752 __gthread_key_create (__gthread_key_t
*__key
, void (*__dtor
) (void *))
754 return __gthrw_(pthread_key_create
) (__key
, __dtor
);
758 __gthread_key_delete (__gthread_key_t __key
)
760 return __gthrw_(pthread_key_delete
) (__key
);
763 __GTHREAD_INLINE
void *
764 __gthread_getspecific (__gthread_key_t __key
)
766 return __gthrw_(pthread_getspecific
) (__key
);
770 __gthread_setspecific (__gthread_key_t __key
, const void *__ptr
)
772 return __gthrw_(pthread_setspecific
) (__key
, __ptr
);
775 __GTHREAD_INLINE
void
776 __gthread_mutex_init_function (__gthread_mutex_t
*__mutex
)
778 if (__gthread_active_p ())
779 __gthrw_(pthread_mutex_init
) (__mutex
, NULL
);
783 __gthread_mutex_destroy (__gthread_mutex_t
*__mutex
)
785 if (__gthread_active_p ())
786 return __gthrw_(pthread_mutex_destroy
) (__mutex
);
792 __gthread_mutex_lock (__gthread_mutex_t
*__mutex
)
794 if (__gthread_active_p ())
795 return __gthrw_(pthread_mutex_lock
) (__mutex
);
801 __gthread_mutex_trylock (__gthread_mutex_t
*__mutex
)
803 if (__gthread_active_p ())
804 return __gthrw_(pthread_mutex_trylock
) (__mutex
);
809 #if _GTHREAD_USE_MUTEX_TIMEDLOCK
811 __gthread_mutex_timedlock (__gthread_mutex_t
*__mutex
,
812 const __gthread_time_t
*__abs_timeout
)
814 if (__gthread_active_p ())
815 return __gthrw_(pthread_mutex_timedlock
) (__mutex
, __abs_timeout
);
822 __gthread_mutex_unlock (__gthread_mutex_t
*__mutex
)
824 if (__gthread_active_p ())
825 return __gthrw_(pthread_mutex_unlock
) (__mutex
);
830 #if !defined( PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP) \
831 || defined(_GTHREAD_USE_RECURSIVE_MUTEX_INIT_FUNC)
833 __gthread_recursive_mutex_init_function (__gthread_recursive_mutex_t
*__mutex
)
835 if (__gthread_active_p ())
837 pthread_mutexattr_t __attr
;
840 __r
= __gthrw_(pthread_mutexattr_init
) (&__attr
);
842 __r
= __gthrw_(pthread_mutexattr_settype
) (&__attr
,
843 PTHREAD_MUTEX_RECURSIVE
);
845 __r
= __gthrw_(pthread_mutex_init
) (__mutex
, &__attr
);
847 __r
= __gthrw_(pthread_mutexattr_destroy
) (&__attr
);
855 __gthread_recursive_mutex_lock (__gthread_recursive_mutex_t
*__mutex
)
857 return __gthread_mutex_lock (__mutex
);
861 __gthread_recursive_mutex_trylock (__gthread_recursive_mutex_t
*__mutex
)
863 return __gthread_mutex_trylock (__mutex
);
866 #if _GTHREAD_USE_MUTEX_TIMEDLOCK
868 __gthread_recursive_mutex_timedlock (__gthread_recursive_mutex_t
*__mutex
,
869 const __gthread_time_t
*__abs_timeout
)
871 return __gthread_mutex_timedlock (__mutex
, __abs_timeout
);
876 __gthread_recursive_mutex_unlock (__gthread_recursive_mutex_t
*__mutex
)
878 return __gthread_mutex_unlock (__mutex
);
882 __gthread_recursive_mutex_destroy (__gthread_recursive_mutex_t
*__mutex
)
884 return __gthread_mutex_destroy (__mutex
);
887 #ifdef _GTHREAD_USE_COND_INIT_FUNC
888 __GTHREAD_INLINE
void
889 __gthread_cond_init_function (__gthread_cond_t
*__cond
)
891 if (__gthread_active_p ())
892 __gthrw_(pthread_cond_init
) (__cond
, NULL
);
897 __gthread_cond_broadcast (__gthread_cond_t
*__cond
)
899 return __gthrw_(pthread_cond_broadcast
) (__cond
);
903 __gthread_cond_signal (__gthread_cond_t
*__cond
)
905 return __gthrw_(pthread_cond_signal
) (__cond
);
909 __gthread_cond_wait (__gthread_cond_t
*__cond
, __gthread_mutex_t
*__mutex
)
911 return __gthrw_(pthread_cond_wait
) (__cond
, __mutex
);
915 __gthread_cond_timedwait (__gthread_cond_t
*__cond
, __gthread_mutex_t
*__mutex
,
916 const __gthread_time_t
*__abs_timeout
)
918 return __gthrw_(pthread_cond_timedwait
) (__cond
, __mutex
, __abs_timeout
);
922 __gthread_cond_wait_recursive (__gthread_cond_t
*__cond
,
923 __gthread_recursive_mutex_t
*__mutex
)
925 return __gthread_cond_wait (__cond
, __mutex
);
929 __gthread_cond_destroy (__gthread_cond_t
* __cond
)
931 return __gthrw_(pthread_cond_destroy
) (__cond
);
936 __gthread_rwlock_rdlock (__gthread_rwlock_t
*__rwlock
)
938 if (__gthread_active_p ())
939 return __gthrw_(pthread_rwlock_rdlock
) (__rwlock
);
945 __gthread_rwlock_tryrdlock (__gthread_rwlock_t
*__rwlock
)
947 if (__gthread_active_p ())
948 return __gthrw_(pthread_rwlock_tryrdlock
) (__rwlock
);
954 __gthread_rwlock_wrlock (__gthread_rwlock_t
*__rwlock
)
956 if (__gthread_active_p ())
957 return __gthrw_(pthread_rwlock_wrlock
) (__rwlock
);
963 __gthread_rwlock_trywrlock (__gthread_rwlock_t
*__rwlock
)
965 if (__gthread_active_p ())
966 return __gthrw_(pthread_rwlock_trywrlock
) (__rwlock
);
972 __gthread_rwlock_unlock (__gthread_rwlock_t
*__rwlock
)
974 if (__gthread_active_p ())
975 return __gthrw_(pthread_rwlock_unlock
) (__rwlock
);
981 #endif /* _LIBOBJC */
983 #undef __GTHREAD_INLINE
984 #undef __GTHREAD_ALWAYS_INLINE
986 #endif /* ! GCC_GTHR_POSIX_H */