Fix loop construction to functions calls
[glibc.git] / stdlib / cxa_finalize.c
blobd7262a9df44f2c4236a87ba8d88a30bd00f6b51b
1 /* Copyright (C) 1999,2001-2003,2005,2006,2009 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, see
16 <http://www.gnu.org/licenses/>. */
18 #include <assert.h>
19 #include <stdlib.h>
20 #include <atomic.h>
21 #include "exit.h"
22 #include <fork.h>
23 #include <sysdep.h>
25 /* If D is non-NULL, call all functions registered with `__cxa_atexit'
26 with the same dso handle. Otherwise, if D is NULL, call all of the
27 registered handlers. */
28 void
29 __cxa_finalize (void *d)
31 struct exit_function_list *funcs;
33 restart:
34 for (funcs = __exit_funcs; funcs; funcs = funcs->next)
36 struct exit_function *f;
38 for (f = &funcs->fns[funcs->idx - 1]; f >= &funcs->fns[0]; --f)
40 void (*cxafn) (void *arg, int status);
41 void *cxaarg;
43 if ((d == NULL || d == f->func.cxa.dso_handle)
44 /* We don't want to run this cleanup more than once. */
45 && (cxafn = f->func.cxa.fn,
46 cxaarg = f->func.cxa.arg,
47 ! catomic_compare_and_exchange_bool_acq (&f->flavor, ef_free,
48 ef_cxa)))
50 uint64_t check = __new_exitfn_called;
52 #ifdef PTR_DEMANGLE
53 PTR_DEMANGLE (cxafn);
54 #endif
55 cxafn (cxaarg, 0);
57 /* It is possible that that last exit function registered
58 more exit functions. Start the loop over. */
59 if (__builtin_expect (check != __new_exitfn_called, 0))
60 goto restart;
65 /* Also remove the quick_exit handlers, but do not call them. */
66 for (funcs = __quick_exit_funcs; funcs; funcs = funcs->next)
68 struct exit_function *f;
70 for (f = &funcs->fns[funcs->idx - 1]; f >= &funcs->fns[0]; --f)
71 if (d == NULL || d == f->func.cxa.dso_handle)
72 f->flavor = ef_free;
75 /* Remove the registered fork handlers. We do not have to
76 unregister anything if the program is going to terminate anyway. */
77 #ifdef UNREGISTER_ATFORK
78 if (d != NULL)
79 UNREGISTER_ATFORK (d);
80 #endif