1 /* Libc stubs for pthread functions. Hurd pthread version.
2 Copyright (C) 2002-2018 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <http://www.gnu.org/licenses/>. */
22 #include <shlib-compat.h>
23 #include <pthread-functions.h>
24 #include <libc-lock.h>
27 /* Pointers to the libc functions. */
28 struct pthread_functions __libc_pthread_functions attribute_hidden
;
29 int __libc_pthread_functions_init attribute_hidden
;
32 #define FORWARD2(name, rettype, decl, params, defaction) \
36 if (!__libc_pthread_functions_init) \
39 return PTHFCT_CALL (ptr_##name, params); \
42 /* Same as FORWARD2, only without return. */
43 #define FORWARD_NORETURN(name, rettype, decl, params, defaction) \
47 if (!__libc_pthread_functions_init) \
50 PTHFCT_CALL (ptr_##name, params); \
53 #define FORWARD(name, decl, params, defretval) \
54 FORWARD2 (name, int, decl, params, return defretval)
56 FORWARD (pthread_attr_destroy
, (pthread_attr_t
*attr
), (attr
), 0)
58 FORWARD (pthread_attr_init
, (pthread_attr_t
*attr
), (attr
), 0)
60 FORWARD (pthread_attr_getdetachstate
,
61 (const pthread_attr_t
*attr
, int *detachstate
), (attr
, detachstate
),
63 FORWARD (pthread_attr_setdetachstate
, (pthread_attr_t
*attr
, int detachstate
),
64 (attr
, detachstate
), 0)
66 FORWARD (pthread_attr_getinheritsched
,
67 (const pthread_attr_t
*attr
, int *inherit
), (attr
, inherit
), 0)
68 FORWARD (pthread_attr_setinheritsched
, (pthread_attr_t
*attr
, int inherit
),
71 FORWARD (pthread_attr_getschedparam
,
72 (const pthread_attr_t
*attr
, struct sched_param
*param
),
74 FORWARD (pthread_attr_setschedparam
,
75 (pthread_attr_t
*attr
, const struct sched_param
*param
),
78 FORWARD (pthread_attr_getschedpolicy
,
79 (const pthread_attr_t
*attr
, int *policy
), (attr
, policy
), 0)
80 FORWARD (pthread_attr_setschedpolicy
, (pthread_attr_t
*attr
, int policy
),
83 FORWARD (pthread_attr_getscope
,
84 (const pthread_attr_t
*attr
, int *scope
), (attr
, scope
), 0)
85 FORWARD (pthread_attr_setscope
, (pthread_attr_t
*attr
, int scope
),
89 FORWARD (pthread_condattr_destroy
, (pthread_condattr_t
*attr
), (attr
), 0)
90 FORWARD (pthread_condattr_init
, (pthread_condattr_t
*attr
), (attr
), 0)
93 FORWARD (pthread_cond_broadcast
, (pthread_cond_t
*cond
), (cond
), 0)
94 FORWARD (pthread_cond_destroy
, (pthread_cond_t
*cond
), (cond
), 0)
95 FORWARD (pthread_cond_init
,
96 (pthread_cond_t
*cond
, const pthread_condattr_t
*cond_attr
),
98 FORWARD (pthread_cond_signal
, (pthread_cond_t
*cond
), (cond
), 0)
99 FORWARD (pthread_cond_wait
, (pthread_cond_t
*cond
, pthread_mutex_t
*mutex
),
101 FORWARD (pthread_cond_timedwait
,
102 (pthread_cond_t
*cond
, pthread_mutex_t
*mutex
,
103 const struct timespec
*abstime
), (cond
, mutex
, abstime
), 0)
105 FORWARD (pthread_equal
, (pthread_t thread1
, pthread_t thread2
),
106 (thread1
, thread2
), 1)
109 /* Use an alias to avoid warning, as pthread_exit is declared noreturn. */
110 FORWARD_NORETURN (__pthread_exit
, void, (void *retval
), (retval
),
112 strong_alias (__pthread_exit
, pthread_exit
);
115 FORWARD (pthread_getschedparam
,
116 (pthread_t target_thread
, int *policy
, struct sched_param
*param
),
117 (target_thread
, policy
, param
), 0)
118 FORWARD (pthread_setschedparam
,
119 (pthread_t target_thread
, int policy
,
120 const struct sched_param
*param
), (target_thread
, policy
, param
), 0)
123 FORWARD (pthread_mutex_destroy
, (pthread_mutex_t
*mutex
), (mutex
), 0)
125 FORWARD (pthread_mutex_init
,
126 (pthread_mutex_t
*mutex
, const pthread_mutexattr_t
*mutexattr
),
127 (mutex
, mutexattr
), 0)
129 FORWARD (pthread_mutex_lock
, (pthread_mutex_t
*mutex
), (mutex
), 0)
131 FORWARD (pthread_mutex_unlock
, (pthread_mutex_t
*mutex
), (mutex
), 0)
134 FORWARD2 (pthread_self
, pthread_t
, (void), (), return 0)
137 FORWARD (__pthread_setcancelstate
, (int state
, int *oldstate
),
138 (state
, oldstate
), 0)
139 strong_alias (__pthread_setcancelstate
, pthread_setcancelstate
);
141 FORWARD (pthread_setcanceltype
, (int type
, int *oldtype
), (type
, oldtype
), 0)
143 struct __pthread_cancelation_handler
*dummy_list
;
144 FORWARD2 (__pthread_get_cleanup_stack
, struct __pthread_cancelation_handler
**,
145 (void), (), return &dummy_list
);
148 /* Fork interaction */
152 void (*prepare
) (void);
153 void (*parent
) (void);
154 void (*child
) (void);
160 /* TODO: better locking */
161 __libc_lock_define_initialized (static, atfork_lock
);
162 static struct atfork
*fork_handlers
, *fork_last_handler
;
165 atfork_pthread_prepare (void)
167 struct atfork
*handlers
, *last_handler
;
169 __libc_lock_lock (atfork_lock
);
170 handlers
= fork_handlers
;
171 last_handler
= fork_last_handler
;
172 __libc_lock_unlock (atfork_lock
);
174 if (last_handler
== NULL
)
179 if (last_handler
->prepare
!= NULL
)
180 last_handler
->prepare ();
181 if (last_handler
== handlers
)
183 last_handler
= last_handler
->prev
;
186 text_set_element (_hurd_atfork_prepare_hook
, atfork_pthread_prepare
);
189 atfork_pthread_parent (void)
191 struct atfork
*handlers
;
193 __libc_lock_lock (atfork_lock
);
194 handlers
= fork_handlers
;
195 __libc_lock_unlock (atfork_lock
);
197 while (handlers
!= NULL
)
199 if (handlers
->parent
!= NULL
)
201 handlers
= handlers
->next
;
204 text_set_element (_hurd_atfork_parent_hook
, atfork_pthread_parent
);
207 atfork_pthread_child (void)
209 struct atfork
*handlers
;
211 __libc_lock_lock (atfork_lock
);
212 handlers
= fork_handlers
;
213 __libc_lock_unlock (atfork_lock
);
215 while (handlers
!= NULL
)
217 if (handlers
->child
!= NULL
)
219 handlers
= handlers
->next
;
222 text_set_element (_hurd_atfork_child_hook
, atfork_pthread_child
);
225 __register_atfork (void (*prepare
) (void),
226 void (*parent
) (void),
227 void (*child
) (void),
230 struct atfork
*new = malloc (sizeof (*new));
234 new->prepare
= prepare
;
235 new->parent
= parent
;
237 new->dso_handle
= dso_handle
;
240 __libc_lock_lock (atfork_lock
);
241 new->next
= fork_handlers
;
242 if (fork_handlers
!= NULL
)
243 fork_handlers
->prev
= new;
245 if (fork_last_handler
== NULL
)
246 fork_last_handler
= new;
247 __libc_lock_unlock (atfork_lock
);
251 libc_hidden_def (__register_atfork
)
254 __unregister_atfork (void *dso_handle
)
256 struct atfork
**handlers
, *prev
= NULL
, *next
;
257 __libc_lock_lock (atfork_lock
);
258 handlers
= &fork_handlers
;
259 while (*handlers
!= NULL
)
261 if ((*handlers
)->dso_handle
== dso_handle
)
263 /* Drop this handler from the list. */
264 if (*handlers
== fork_last_handler
)
266 /* Was last, new last is prev, if any. */
267 fork_last_handler
= prev
;
270 next
= (*handlers
)->next
;
277 /* Just proceed to next handler. */
279 handlers
= &prev
->next
;
282 __libc_lock_unlock (atfork_lock
);