2 Copyright (C) 2000-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/>. */
24 #include <pt-internal.h>
29 /* Terminate the current thread and make STATUS available to any
30 thread that might join it. */
32 __pthread_exit (void *status
)
34 struct __pthread
*self
= _pthread_self ();
35 struct __pthread_cancelation_handler
**handlers
;
38 /* Run any cancelation handlers. According to POSIX, the
39 cancellation cleanup handlers should be called with cancellation
41 __pthread_setcancelstate (PTHREAD_CANCEL_DISABLE
, &oldstate
);
43 for (handlers
= __pthread_get_cleanup_stack ();
45 *handlers
= (*handlers
)->__next
)
46 (*handlers
)->__handler ((*handlers
)->__arg
);
48 __pthread_setcancelstate (oldstate
, &oldstate
);
50 /* Decrease the number of threads. We use an atomic operation to
51 make sure that only the last thread calls `exit'. */
52 if (atomic_decrement_and_test (&__pthread_total
))
53 /* We are the last thread. */
56 /* Note that after this point the process can be terminated at any
57 point if another thread calls `pthread_exit' and happens to be
60 __pthread_mutex_lock (&self
->state_lock
);
62 if (self
->cancel_state
== PTHREAD_CANCEL_ENABLE
&& self
->cancel_pending
)
63 status
= PTHREAD_CANCELED
;
68 assert (!"Consistency error: unexpected self->state");
72 case PTHREAD_DETACHED
:
73 __pthread_mutex_unlock (&self
->state_lock
);
77 case PTHREAD_JOINABLE
:
78 /* We need to stay around for a while since another thread
79 might want to join us. */
80 self
->state
= PTHREAD_EXITED
;
82 /* We need to remember the exit status. A thread joining us
84 self
->status
= status
;
86 /* Broadcast the condition. This will wake up threads that are
87 waiting to join us. */
88 __pthread_cond_broadcast (&self
->state_cond
);
89 __pthread_mutex_unlock (&self
->state_lock
);
94 /* Destroy any thread specific data. */
95 __pthread_destroy_specific (self
);
97 /* Destroy any signal state. */
98 __pthread_sigstate_destroy (self
);
100 /* Self terminating requires TLS, so defer the release of the TCB until
101 the thread structure is reused. */
103 /* Release kernel resources, including the kernel thread and the stack,
104 and drop the self reference. */
105 __pthread_thread_terminate (self
);
111 strong_alias (__pthread_exit
, pthread_exit
);