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.10 christos __RCSID("$NetBSD: arm_initfini.c,v 1.10 2025/03/03 17:00:22 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.10 christos return; 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