2007-11-18 Roland McGrath <roland@frob.com>
[glibc.git] / stdlib / cxa_finalize.c
blob148d57f200647ad70f1f7298be1bc69c8ba48426
1 /* Copyright (C) 1999,2001,2002,2003,2005,2006 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
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.
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
12 Lesser General Public License for more details.
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. */
19 #include <assert.h>
20 #include <stdlib.h>
21 #include <atomic.h>
22 #include "exit.h"
23 #include <fork.h>
24 #include <sysdep.h>
26 /* If D is non-NULL, call all functions registered with `__cxa_atexit'
27 with the same dso handle. Otherwise, if D is NULL, call all of the
28 registered handlers. */
29 void
30 __cxa_finalize (void *d)
32 struct exit_function_list *funcs;
34 restart:
35 for (funcs = __exit_funcs; funcs; funcs = funcs->next)
37 struct exit_function *f;
39 for (f = &funcs->fns[funcs->idx - 1]; f >= &funcs->fns[0]; --f)
41 void (*cxafn) (void *arg, int status);
42 void *cxaarg;
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,
47 cxaarg = f->func.cxa.arg,
48 ! catomic_compare_and_exchange_bool_acq (&f->flavor, ef_free,
49 ef_cxa)))
51 uint64_t check = __new_exitfn_called;
53 #ifdef PTR_DEMANGLE
54 PTR_DEMANGLE (cxafn);
55 #endif
56 cxafn (cxaarg, 0);
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;
66 /* Remove the registered fork handlers. We do not have to
67 unregister anything if the program is going to terminate anyway. */
68 #ifdef UNREGISTER_ATFORK
69 if (d != NULL)
70 UNREGISTER_ATFORK (d);
71 #endif