xref: /openbsd-src/gnu/llvm/compiler-rt/lib/crt/crtbegin.c (revision 810390e339a5425391477d5d41c78d7cab2424ac)
13cab2bb3Spatrick //===-- crtbegin.c - Start of constructors and destructors ----------------===//
23cab2bb3Spatrick //
33cab2bb3Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
43cab2bb3Spatrick // See https://llvm.org/LICENSE.txt for license information.
53cab2bb3Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
63cab2bb3Spatrick //
73cab2bb3Spatrick //===----------------------------------------------------------------------===//
83cab2bb3Spatrick 
93cab2bb3Spatrick #include <stddef.h>
103cab2bb3Spatrick 
113cab2bb3Spatrick __attribute__((visibility("hidden"))) void *__dso_handle = &__dso_handle;
123cab2bb3Spatrick 
133cab2bb3Spatrick #ifdef EH_USE_FRAME_REGISTRY
143cab2bb3Spatrick __extension__ static void *__EH_FRAME_LIST__[]
153cab2bb3Spatrick     __attribute__((section(".eh_frame"), aligned(sizeof(void *)))) = {};
163cab2bb3Spatrick 
173cab2bb3Spatrick extern void __register_frame_info(const void *, void *) __attribute__((weak));
183cab2bb3Spatrick extern void *__deregister_frame_info(const void *) __attribute__((weak));
193cab2bb3Spatrick #endif
203cab2bb3Spatrick 
213cab2bb3Spatrick #ifndef CRT_HAS_INITFINI_ARRAY
223cab2bb3Spatrick typedef void (*fp)(void);
233cab2bb3Spatrick 
243cab2bb3Spatrick static fp __CTOR_LIST__[]
253cab2bb3Spatrick     __attribute__((section(".ctors"), aligned(sizeof(fp)))) = {(fp)-1};
263cab2bb3Spatrick extern fp __CTOR_LIST_END__[];
273cab2bb3Spatrick #endif
283cab2bb3Spatrick 
293cab2bb3Spatrick extern void __cxa_finalize(void *) __attribute__((weak));
303cab2bb3Spatrick 
__do_init(void)31*810390e3Srobert static void __attribute__((used)) __do_init(void) {
323cab2bb3Spatrick   static _Bool __initialized;
333cab2bb3Spatrick   if (__builtin_expect(__initialized, 0))
343cab2bb3Spatrick     return;
353cab2bb3Spatrick   __initialized = 1;
363cab2bb3Spatrick 
373cab2bb3Spatrick #ifdef EH_USE_FRAME_REGISTRY
383cab2bb3Spatrick   static struct { void *p[8]; } __object;
393cab2bb3Spatrick   if (__register_frame_info)
403cab2bb3Spatrick     __register_frame_info(__EH_FRAME_LIST__, &__object);
413cab2bb3Spatrick #endif
423cab2bb3Spatrick #ifndef CRT_HAS_INITFINI_ARRAY
433cab2bb3Spatrick   const size_t n = __CTOR_LIST_END__ - __CTOR_LIST__ - 1;
443cab2bb3Spatrick   for (size_t i = n; i >= 1; i--) __CTOR_LIST__[i]();
453cab2bb3Spatrick #endif
463cab2bb3Spatrick }
473cab2bb3Spatrick 
483cab2bb3Spatrick #ifdef CRT_HAS_INITFINI_ARRAY
493cab2bb3Spatrick __attribute__((section(".init_array"),
503cab2bb3Spatrick                used)) static void (*__init)(void) = __do_init;
513cab2bb3Spatrick #elif defined(__i386__) || defined(__x86_64__)
523cab2bb3Spatrick __asm__(".pushsection .init,\"ax\",@progbits\n\t"
533cab2bb3Spatrick     "call " __USER_LABEL_PREFIX__ "__do_init\n\t"
543cab2bb3Spatrick     ".popsection");
55d89ec533Spatrick #elif defined(__riscv)
56d89ec533Spatrick __asm__(".pushsection .init,\"ax\",%progbits\n\t"
57d89ec533Spatrick         "call " __USER_LABEL_PREFIX__ "__do_init\n\t"
58d89ec533Spatrick         ".popsection");
593cab2bb3Spatrick #elif defined(__arm__) || defined(__aarch64__)
603cab2bb3Spatrick __asm__(".pushsection .init,\"ax\",%progbits\n\t"
613cab2bb3Spatrick     "bl " __USER_LABEL_PREFIX__ "__do_init\n\t"
623cab2bb3Spatrick     ".popsection");
633cab2bb3Spatrick #elif defined(__powerpc__) || defined(__powerpc64__)
643cab2bb3Spatrick __asm__(".pushsection .init,\"ax\",@progbits\n\t"
653cab2bb3Spatrick     "bl " __USER_LABEL_PREFIX__ "__do_init\n\t"
663cab2bb3Spatrick     "nop\n\t"
673cab2bb3Spatrick     ".popsection");
683cab2bb3Spatrick #elif defined(__sparc__)
693cab2bb3Spatrick __asm__(".pushsection .init,\"ax\",@progbits\n\t"
703cab2bb3Spatrick     "call " __USER_LABEL_PREFIX__ "__do_init\n\t"
713cab2bb3Spatrick     ".popsection");
723cab2bb3Spatrick #else
733cab2bb3Spatrick #error "crtbegin without .init_fini array unimplemented for this architecture"
743cab2bb3Spatrick #endif // CRT_HAS_INITFINI_ARRAY
753cab2bb3Spatrick 
763cab2bb3Spatrick #ifndef CRT_HAS_INITFINI_ARRAY
773cab2bb3Spatrick static fp __DTOR_LIST__[]
783cab2bb3Spatrick     __attribute__((section(".dtors"), aligned(sizeof(fp)))) = {(fp)-1};
793cab2bb3Spatrick extern fp __DTOR_LIST_END__[];
803cab2bb3Spatrick #endif
813cab2bb3Spatrick 
__do_fini(void)82*810390e3Srobert static void __attribute__((used)) __do_fini(void) {
833cab2bb3Spatrick   static _Bool __finalized;
843cab2bb3Spatrick   if (__builtin_expect(__finalized, 0))
853cab2bb3Spatrick     return;
863cab2bb3Spatrick   __finalized = 1;
873cab2bb3Spatrick 
883cab2bb3Spatrick   if (__cxa_finalize)
893cab2bb3Spatrick     __cxa_finalize(__dso_handle);
903cab2bb3Spatrick 
913cab2bb3Spatrick #ifndef CRT_HAS_INITFINI_ARRAY
923cab2bb3Spatrick   const size_t n = __DTOR_LIST_END__ - __DTOR_LIST__ - 1;
933cab2bb3Spatrick   for (size_t i = 1; i <= n; i++) __DTOR_LIST__[i]();
943cab2bb3Spatrick #endif
953cab2bb3Spatrick #ifdef EH_USE_FRAME_REGISTRY
963cab2bb3Spatrick   if (__deregister_frame_info)
973cab2bb3Spatrick     __deregister_frame_info(__EH_FRAME_LIST__);
983cab2bb3Spatrick #endif
993cab2bb3Spatrick }
1003cab2bb3Spatrick 
1013cab2bb3Spatrick #ifdef CRT_HAS_INITFINI_ARRAY
1023cab2bb3Spatrick __attribute__((section(".fini_array"),
1033cab2bb3Spatrick                used)) static void (*__fini)(void) = __do_fini;
1043cab2bb3Spatrick #elif defined(__i386__) || defined(__x86_64__)
1053cab2bb3Spatrick __asm__(".pushsection .fini,\"ax\",@progbits\n\t"
1063cab2bb3Spatrick     "call " __USER_LABEL_PREFIX__ "__do_fini\n\t"
1073cab2bb3Spatrick     ".popsection");
1083cab2bb3Spatrick #elif defined(__arm__) || defined(__aarch64__)
1093cab2bb3Spatrick __asm__(".pushsection .fini,\"ax\",%progbits\n\t"
1103cab2bb3Spatrick     "bl " __USER_LABEL_PREFIX__ "__do_fini\n\t"
1113cab2bb3Spatrick     ".popsection");
1123cab2bb3Spatrick #elif defined(__powerpc__) || defined(__powerpc64__)
1133cab2bb3Spatrick __asm__(".pushsection .fini,\"ax\",@progbits\n\t"
1143cab2bb3Spatrick     "bl " __USER_LABEL_PREFIX__ "__do_fini\n\t"
1153cab2bb3Spatrick     "nop\n\t"
1163cab2bb3Spatrick     ".popsection");
117d89ec533Spatrick #elif defined(__riscv)
118d89ec533Spatrick __asm__(".pushsection .fini,\"ax\",@progbits\n\t"
119d89ec533Spatrick         "call " __USER_LABEL_PREFIX__ "__do_fini\n\t"
120d89ec533Spatrick         ".popsection");
1213cab2bb3Spatrick #elif defined(__sparc__)
1223cab2bb3Spatrick __asm__(".pushsection .fini,\"ax\",@progbits\n\t"
1233cab2bb3Spatrick     "call " __USER_LABEL_PREFIX__ "__do_fini\n\t"
1243cab2bb3Spatrick     ".popsection");
1253cab2bb3Spatrick #else
1263cab2bb3Spatrick #error "crtbegin without .init_fini array unimplemented for this architecture"
1273cab2bb3Spatrick #endif  // CRT_HAS_INIT_FINI_ARRAY
128