match: Improve handling of double convert [PR117776]
[gcc.git] / libgcc / config / rs6000 / cxa_finalize.c
blob6773ee0777a3908e8fe7e2f01588fb857f7efa05
1 /* Copyright (C) 1999-2024 Free Software Foundation, Inc.
3 NOTE: This source is derived from an old version taken from the GNU C
4 Library (glibc).
6 This file is part of GCC.
8 GCC is free software; you can redistribute it and/or modify it under
9 the terms of the GNU General Public License as published by the Free
10 Software Foundation; either version 3, or (at your option) any later
11 version.
13 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
14 WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16 for more details.
18 Under Section 7 of GPL version 3, you are granted additional
19 permissions described in the GCC Runtime Library Exception, version
20 3.1, as published by the Free Software Foundation.
22 You should have received a copy of the GNU General Public License and
23 a copy of the GCC Runtime Library Exception along with this program;
24 see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
25 <http://www.gnu.org/licenses/>. */
27 #include <assert.h>
28 #include <stdlib.h>
29 #include "exit.h"
32 static boolean_t
33 catomic_compare_and_exchange_bool_acq (long *mem, long newval, long oldval)
35 return ! __atomic_compare_exchange (mem, &oldval, &newval, 0,
36 __ATOMIC_ACQUIRE, __ATOMIC_ACQUIRE);
39 /* If D is non-NULL, call all functions registered with `__cxa_atexit'
40 with the same dso handle. Otherwise, if D is NULL, call all of the
41 registered handlers. */
42 void
43 __cxa_finalize (void *d)
45 struct exit_function_list *funcs;
47 restart:
48 for (funcs = __exit_funcs; funcs; funcs = funcs->next)
50 struct exit_function *f;
52 for (f = &funcs->fns[funcs->idx - 1]; f >= &funcs->fns[0]; --f)
54 void (*cxafn) (void *arg, int status);
55 void *cxaarg;
57 if ((d == NULL || d == f->func.cxa.dso_handle)
58 /* We don't want to run this cleanup more than once. */
59 && (cxafn = f->func.cxa.fn,
60 cxaarg = f->func.cxa.arg,
61 ! catomic_compare_and_exchange_bool_acq (&f->flavor, ef_free,
62 ef_cxa)))
64 uint64_t check = __new_exitfn_called;
66 #ifdef PTR_DEMANGLE
67 PTR_DEMANGLE (cxafn);
68 #endif
69 cxafn (cxaarg, 0);
71 /* It is possible that that last exit function registered
72 more exit functions. Start the loop over. */
73 if (__builtin_expect (check != __new_exitfn_called, 0))
74 goto restart;
79 /* Remove the registered fork handlers. We do not have to
80 unregister anything if the program is going to terminate anyway. */
81 #ifdef UNREGISTER_ATFORK
82 if (d != NULL)
83 UNREGISTER_ATFORK (d);
84 #endif