arm_initfini.c revision 1.9 1 1.1 matt /*-
2 1.1 matt * Copyright (c) 2013 The NetBSD Foundation, Inc.
3 1.1 matt * All rights reserved.
4 1.1 matt *
5 1.1 matt * This code is derived from software contributed to The NetBSD Foundation
6 1.1 matt * by Matt Thomas of 3am Software Foundry.
7 1.1 matt *
8 1.1 matt * Redistribution and use in source and binary forms, with or without
9 1.1 matt * modification, are permitted provided that the following conditions
10 1.1 matt * are met:
11 1.1 matt * 1. Redistributions of source code must retain the above copyright
12 1.1 matt * notice, this list of conditions and the following disclaimer.
13 1.1 matt * 2. Redistributions in binary form must reproduce the above copyright
14 1.1 matt * notice, this list of conditions and the following disclaimer in the
15 1.1 matt * documentation and/or other materials provided with the distribution.
16 1.1 matt *
17 1.1 matt * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
18 1.1 matt * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19 1.1 matt * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20 1.1 matt * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
21 1.1 matt * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 1.1 matt * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 1.1 matt * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24 1.1 matt * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25 1.1 matt * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26 1.1 matt * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 1.1 matt * POSSIBILITY OF SUCH DAMAGE.
28 1.1 matt */
29 1.1 matt
30 1.1 matt #include <sys/cdefs.h>
31 1.1 matt
32 1.8 christos __RCSID("$NetBSD: arm_initfini.c,v 1.9 2025/02/27 10:55:27 christos Exp $");
33 1.3 skrll
34 1.3 skrll #include "namespace.h"
35 1.1 matt
36 1.1 matt /*
37 1.1 matt * To properly implement setjmp/longjmp for the ARM AAPCS ABI, it has to be
38 1.1 matt * aware of whether there is a FPU is present or not. Regardless of whether
39 1.1 matt * the hard-float ABI is being used, setjmp needs to save D8-D15. But it can
40 1.1 matt * only do this if those instructions won't cause an exception.
41 1.1 matt */
42 1.1 matt
43 1.1 matt #include <sys/param.h>
44 1.1 matt #include <sys/sysctl.h>
45 1.1 matt
46 1.1 matt #include <stdbool.h>
47 1.8 christos #include <string.h>
48 1.2 matt #include <stddef.h>
49 1.9 christos #include <stdlib.h>
50 1.1 matt
51 1.4 matt __dso_hidden int _libc_arm_fpu_present;
52 1.7 matt #ifndef __ARM_ARCH_EXT_IDIV__
53 1.6 matt __dso_hidden int _libc_arm_hwdiv_present;
54 1.8 christos # define NENTRIES 2
55 1.8 christos #else
56 1.8 christos # define NENTRIES 1
57 1.7 matt #endif
58 1.8 christos
59 1.1 matt static bool _libc_aapcs_initialized;
60 1.1 matt
61 1.1 matt void _libc_aapcs_init(void) __attribute__((__constructor__, __used__));
62 1.1 matt
63 1.5 matt void __section(".text.startup")
64 1.1 matt _libc_aapcs_init(void)
65 1.1 matt {
66 1.8 christos if (_libc_aapcs_initialized)
67 1.8 christos return;
68 1.8 christos
69 1.8 christos struct sysctlnode query, md[64];
70 1.8 christos int mib[2];
71 1.8 christos size_t len, mlen = sizeof(_libc_arm_fpu_present), nentries = 0;
72 1.8 christos
73 1.8 christos _libc_aapcs_initialized = true;
74 1.8 christos mib[0] = CTL_MACHDEP;
75 1.8 christos mib[1] = CTL_QUERY;
76 1.8 christos memset(&query, 0, sizeof(query));
77 1.8 christos query.sysctl_flags = SYSCTL_VERSION;
78 1.8 christos len = sizeof(md);
79 1.8 christos if (sysctl(mib, 2, md, &len, &query, sizeof(query)) == -1)
80 1.9 christos abort();
81 1.8 christos
82 1.8 christos for (size_t i = 0; i < len / sizeof(md[0]); i++) {
83 1.8 christos if (strcmp(md[i].sysctl_name, "fpu_present") == 0) {
84 1.8 christos mib[1] = md[i].sysctl_num;
85 1.8 christos (void)sysctl(mib, 2, &_libc_arm_fpu_present, &mlen,
86 1.8 christos NULL, 0);
87 1.8 christos nentries++;
88 1.8 christos
89 1.8 christos }
90 1.7 matt #ifndef __ARM_ARCH_EXT_IDIV__
91 1.8 christos else if (strcmp(md[i].sysctl_name, "hwdiv_present") == 0) {
92 1.8 christos mib[1] = md[i].sysctl_num;
93 1.8 christos (void)sysctl(mib, 2, &_libc_arm_hwdiv_present, &mlen,
94 1.8 christos NULL, 0);
95 1.8 christos nentries++;
96 1.8 christos }
97 1.7 matt #endif
98 1.8 christos if (nentries >= NENTRIES)
99 1.8 christos return;
100 1.2 matt }
101 1.1 matt }
102