]> git.ipfire.org Git - thirdparty/glibc.git/blame - stdlib/cxa_finalize.c
* po/id.po: Update from translation team.
[thirdparty/glibc.git] / stdlib / cxa_finalize.c
CommitLineData
bd3754dc 1/* Copyright (C) 1999,2001,2002,2003,2005,2006 Free Software Foundation, Inc.
c41041bc
UD
2 This file is part of the GNU C Library.
3
4 The GNU C Library is free software; you can redistribute it and/or
41bdb6e2
AJ
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
c41041bc
UD
8
9 The GNU C Library 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 GNU
41bdb6e2 12 Lesser General Public License for more details.
c41041bc 13
41bdb6e2
AJ
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, write to the Free
16 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
17 02111-1307 USA. */
c41041bc
UD
18
19#include <assert.h>
20#include <stdlib.h>
4009bf40 21#include <atomic.h>
c41041bc 22#include "exit.h"
d1f69fed 23#include <fork.h>
a3c88553 24#include <sysdep.h>
c41041bc
UD
25
26/* If D is non-NULL, call all functions registered with `__cxa_atexit'
4e648ea3
UD
27 with the same dso handle. Otherwise, if D is NULL, call all of the
28 registered handlers. */
c41041bc
UD
29void
30__cxa_finalize (void *d)
31{
32 struct exit_function_list *funcs;
33
a92b1166 34 restart:
c41041bc
UD
35 for (funcs = __exit_funcs; funcs; funcs = funcs->next)
36 {
37 struct exit_function *f;
38
39 for (f = &funcs->fns[funcs->idx - 1]; f >= &funcs->fns[0]; --f)
bd3754dc
UD
40 {
41 void (*cxafn) (void *arg, int status);
a92b1166 42 void *cxaarg;
bd3754dc
UD
43
44 if ((d == NULL || d == f->func.cxa.dso_handle)
45 /* We don't want to run this cleanup more than once. */
46 && (cxafn = f->func.cxa.fn,
a92b1166 47 cxaarg = f->func.cxa.arg,
11bf311e
UD
48 ! catomic_compare_and_exchange_bool_acq (&f->flavor, ef_free,
49 ef_cxa)))
bd3754dc 50 {
a92b1166
UD
51 uint64_t check = __new_exitfn_called;
52
a3c88553 53#ifdef PTR_DEMANGLE
bd3754dc 54 PTR_DEMANGLE (cxafn);
a3c88553 55#endif
a92b1166
UD
56 cxafn (cxaarg, 0);
57
58 /* It is possible that that last exit function registered
59 more exit functions. Start the loop over. */
60 if (__builtin_expect (check != __new_exitfn_called, 0))
61 goto restart;
bd3754dc
UD
62 }
63 }
c41041bc 64 }
d1f69fed 65
4e648ea3
UD
66 /* Remove the registered fork handlers. We do not have to
67 unregister anything if the program is going to terminate anyway. */
d1f69fed 68#ifdef UNREGISTER_ATFORK
4e648ea3
UD
69 if (d != NULL)
70 UNREGISTER_ATFORK (d);
d1f69fed 71#endif
c41041bc 72}