1 /* Wait for thread termination.
2 Copyright (C) 2000-2023 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 <https://www.gnu.org/licenses/>. */
23 #include <pt-internal.h>
25 /* Make calling thread wait for termination of thread THREAD. Return
26 the exit status of the thread in *STATUS. */
28 __pthread_join_common (pthread_t thread
, void **status
, int try,
30 const struct timespec
*abstime
)
32 struct __pthread
*pthread
;
35 /* Lookup the thread structure for THREAD. */
36 pthread
= __pthread_getid (thread
);
40 if (pthread
== _pthread_self ())
43 __pthread_mutex_lock (&pthread
->state_lock
);
47 pthread_cleanup_push ((void (*)(void *)) __pthread_mutex_unlock
,
48 &pthread
->state_lock
);
50 /* Rely on pthread_cond_wait being a cancellation point to make
51 pthread_join one too. */
52 while (pthread
->state
== PTHREAD_JOINABLE
&& err
!= ETIMEDOUT
)
53 err
= __pthread_cond_clockwait (&pthread
->state_cond
,
57 pthread_cleanup_pop (0);
60 switch (pthread
->state
)
62 case PTHREAD_JOINABLE
:
63 __pthread_mutex_unlock (&pthread
->state_lock
);
69 /* THREAD has already exited. Salvage its exit status. */
71 *status
= pthread
->status
;
73 __pthread_mutex_unlock (&pthread
->state_lock
);
75 __pthread_dealloc (pthread
);
79 /* Thou shalt not join non-joinable threads! */
80 __pthread_mutex_unlock (&pthread
->state_lock
);
89 __pthread_join (pthread_t thread
, void **status
)
91 return __pthread_join_common (thread
, status
, 0, CLOCK_REALTIME
, NULL
);
93 weak_alias (__pthread_join
, pthread_join
);
96 __pthread_tryjoin_np (pthread_t thread
, void **status
)
98 return __pthread_join_common (thread
, status
, 1, CLOCK_REALTIME
, NULL
);
100 weak_alias (__pthread_tryjoin_np
, pthread_tryjoin_np
);
103 __pthread_timedjoin_np (pthread_t thread
, void **status
,
104 const struct timespec
*abstime
)
106 return __pthread_join_common (thread
, status
, 0, CLOCK_REALTIME
, abstime
);
108 weak_alias (__pthread_timedjoin_np
, pthread_timedjoin_np
);
111 __pthread_clockjoin_np (pthread_t thread
, void **status
,
113 const struct timespec
*abstime
)
115 return __pthread_join_common (thread
, status
, 0, clockid
, abstime
);
117 weak_alias (__pthread_clockjoin_np
, pthread_clockjoin_np
);