1*8d67ca89SAndroid Build Coastguard Worker /*
2*8d67ca89SAndroid Build Coastguard Worker * Copyright (C) 2019 The Android Open Source Project
3*8d67ca89SAndroid Build Coastguard Worker * All rights reserved.
4*8d67ca89SAndroid Build Coastguard Worker *
5*8d67ca89SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
6*8d67ca89SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
7*8d67ca89SAndroid Build Coastguard Worker * are met:
8*8d67ca89SAndroid Build Coastguard Worker * * Redistributions of source code must retain the above copyright
9*8d67ca89SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
10*8d67ca89SAndroid Build Coastguard Worker * * Redistributions in binary form must reproduce the above copyright
11*8d67ca89SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in
12*8d67ca89SAndroid Build Coastguard Worker * the documentation and/or other materials provided with the
13*8d67ca89SAndroid Build Coastguard Worker * distribution.
14*8d67ca89SAndroid Build Coastguard Worker *
15*8d67ca89SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16*8d67ca89SAndroid Build Coastguard Worker * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17*8d67ca89SAndroid Build Coastguard Worker * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18*8d67ca89SAndroid Build Coastguard Worker * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19*8d67ca89SAndroid Build Coastguard Worker * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20*8d67ca89SAndroid Build Coastguard Worker * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21*8d67ca89SAndroid Build Coastguard Worker * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22*8d67ca89SAndroid Build Coastguard Worker * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23*8d67ca89SAndroid Build Coastguard Worker * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24*8d67ca89SAndroid Build Coastguard Worker * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25*8d67ca89SAndroid Build Coastguard Worker * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26*8d67ca89SAndroid Build Coastguard Worker * SUCH DAMAGE.
27*8d67ca89SAndroid Build Coastguard Worker */
28*8d67ca89SAndroid Build Coastguard Worker
29*8d67ca89SAndroid Build Coastguard Worker #include <private/bionic_ifuncs.h>
30*8d67ca89SAndroid Build Coastguard Worker #include <stddef.h>
31*8d67ca89SAndroid Build Coastguard Worker
__bionic_is_oryon(unsigned long hwcap)32*8d67ca89SAndroid Build Coastguard Worker static inline bool __bionic_is_oryon(unsigned long hwcap) {
33*8d67ca89SAndroid Build Coastguard Worker if (!(hwcap & HWCAP_CPUID)) return false;
34*8d67ca89SAndroid Build Coastguard Worker
35*8d67ca89SAndroid Build Coastguard Worker // Extract the implementor and variant bits from MIDR_EL1.
36*8d67ca89SAndroid Build Coastguard Worker // https://www.kernel.org/doc/html/latest/arch/arm64/cpu-feature-registers.html#list-of-registers-with-visible-features
37*8d67ca89SAndroid Build Coastguard Worker unsigned long midr;
38*8d67ca89SAndroid Build Coastguard Worker __asm__ __volatile__("mrs %0, MIDR_EL1" : "=r"(midr));
39*8d67ca89SAndroid Build Coastguard Worker uint16_t cpu = (midr >> 20) & 0xfff;
40*8d67ca89SAndroid Build Coastguard Worker
41*8d67ca89SAndroid Build Coastguard Worker auto make_cpu = [](unsigned implementor, unsigned variant) {
42*8d67ca89SAndroid Build Coastguard Worker return (implementor << 4) | variant;
43*8d67ca89SAndroid Build Coastguard Worker };
44*8d67ca89SAndroid Build Coastguard Worker
45*8d67ca89SAndroid Build Coastguard Worker // Check for implementor Qualcomm's variants 0x1..0x5 (Oryon).
46*8d67ca89SAndroid Build Coastguard Worker return cpu >= make_cpu('Q', 0x1) && cpu <= make_cpu('Q', 0x5);
47*8d67ca89SAndroid Build Coastguard Worker }
48*8d67ca89SAndroid Build Coastguard Worker
49*8d67ca89SAndroid Build Coastguard Worker extern "C" {
50*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(memchr)51*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(memchr) {
52*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MTE) {
53*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memchr_func_t, __memchr_aarch64_mte);
54*8d67ca89SAndroid Build Coastguard Worker } else {
55*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memchr_func_t, __memchr_aarch64);
56*8d67ca89SAndroid Build Coastguard Worker }
57*8d67ca89SAndroid Build Coastguard Worker }
58*8d67ca89SAndroid Build Coastguard Worker MEMCHR_SHIM()
59*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(memcmp)60*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(memcmp) {
61*8d67ca89SAndroid Build Coastguard Worker // TODO: enable the SVE version.
62*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memcmp_func_t, __memcmp_aarch64);
63*8d67ca89SAndroid Build Coastguard Worker }
64*8d67ca89SAndroid Build Coastguard Worker MEMCMP_SHIM()
65*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(memcpy)66*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(memcpy) {
67*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MOPS) {
68*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memcpy_func_t, __memmove_aarch64_mops);
69*8d67ca89SAndroid Build Coastguard Worker } else if (__bionic_is_oryon(arg->_hwcap)) {
70*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memcpy_func_t, __memcpy_aarch64_nt);
71*8d67ca89SAndroid Build Coastguard Worker } else if (arg->_hwcap & HWCAP_ASIMD) {
72*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memcpy_func_t, __memcpy_aarch64_simd);
73*8d67ca89SAndroid Build Coastguard Worker } else {
74*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memcpy_func_t, __memcpy_aarch64);
75*8d67ca89SAndroid Build Coastguard Worker }
76*8d67ca89SAndroid Build Coastguard Worker }
77*8d67ca89SAndroid Build Coastguard Worker MEMCPY_SHIM()
78*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(memmove)79*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(memmove) {
80*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MOPS) {
81*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memmove_func_t, __memmove_aarch64_mops);
82*8d67ca89SAndroid Build Coastguard Worker } else if (__bionic_is_oryon(arg->_hwcap)) {
83*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memmove_func_t, __memmove_aarch64_nt);
84*8d67ca89SAndroid Build Coastguard Worker } else if (arg->_hwcap & HWCAP_ASIMD) {
85*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memmove_func_t, __memmove_aarch64_simd);
86*8d67ca89SAndroid Build Coastguard Worker } else {
87*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memmove_func_t, __memmove_aarch64);
88*8d67ca89SAndroid Build Coastguard Worker }
89*8d67ca89SAndroid Build Coastguard Worker }
90*8d67ca89SAndroid Build Coastguard Worker MEMMOVE_SHIM()
91*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(memrchr)92*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(memrchr) {
93*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memrchr_func_t, __memrchr_aarch64);
94*8d67ca89SAndroid Build Coastguard Worker }
95*8d67ca89SAndroid Build Coastguard Worker MEMRCHR_SHIM()
96*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(memset)97*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(memset) {
98*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MOPS) {
99*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memset_func_t, __memset_aarch64_mops);
100*8d67ca89SAndroid Build Coastguard Worker } else if (__bionic_is_oryon(arg->_hwcap)) {
101*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memset_func_t, __memset_aarch64_nt);
102*8d67ca89SAndroid Build Coastguard Worker } else {
103*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(memset_func_t, __memset_aarch64);
104*8d67ca89SAndroid Build Coastguard Worker }
105*8d67ca89SAndroid Build Coastguard Worker }
106*8d67ca89SAndroid Build Coastguard Worker MEMSET_SHIM()
107*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(stpcpy)108*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(stpcpy) {
109*8d67ca89SAndroid Build Coastguard Worker // TODO: enable the SVE version.
110*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(stpcpy_func_t, __stpcpy_aarch64);
111*8d67ca89SAndroid Build Coastguard Worker }
112*8d67ca89SAndroid Build Coastguard Worker STPCPY_SHIM()
113*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strchr)114*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strchr) {
115*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MTE) {
116*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strchr_func_t, __strchr_aarch64_mte);
117*8d67ca89SAndroid Build Coastguard Worker } else {
118*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strchr_func_t, __strchr_aarch64);
119*8d67ca89SAndroid Build Coastguard Worker }
120*8d67ca89SAndroid Build Coastguard Worker }
121*8d67ca89SAndroid Build Coastguard Worker STRCHR_SHIM()
122*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strchrnul)123*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strchrnul) {
124*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MTE) {
125*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strchrnul_func_t, __strchrnul_aarch64_mte);
126*8d67ca89SAndroid Build Coastguard Worker } else {
127*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strchrnul_func_t, __strchrnul_aarch64);
128*8d67ca89SAndroid Build Coastguard Worker }
129*8d67ca89SAndroid Build Coastguard Worker }
130*8d67ca89SAndroid Build Coastguard Worker STRCHRNUL_SHIM()
131*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strcmp)132*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strcmp) {
133*8d67ca89SAndroid Build Coastguard Worker // TODO: enable the SVE version.
134*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strcmp_func_t, __strcmp_aarch64);
135*8d67ca89SAndroid Build Coastguard Worker }
136*8d67ca89SAndroid Build Coastguard Worker STRCMP_SHIM()
137*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strcpy)138*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strcpy) {
139*8d67ca89SAndroid Build Coastguard Worker // TODO: enable the SVE version.
140*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strcpy_func_t, __strcpy_aarch64);
141*8d67ca89SAndroid Build Coastguard Worker }
142*8d67ca89SAndroid Build Coastguard Worker STRCPY_SHIM()
143*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strlen)144*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strlen) {
145*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MTE) {
146*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strlen_func_t, __strlen_aarch64_mte);
147*8d67ca89SAndroid Build Coastguard Worker } else {
148*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strlen_func_t, __strlen_aarch64);
149*8d67ca89SAndroid Build Coastguard Worker }
150*8d67ca89SAndroid Build Coastguard Worker }
151*8d67ca89SAndroid Build Coastguard Worker STRLEN_SHIM()
152*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strncmp)153*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strncmp) {
154*8d67ca89SAndroid Build Coastguard Worker // TODO: enable the SVE version.
155*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strncmp_func_t, __strncmp_aarch64);
156*8d67ca89SAndroid Build Coastguard Worker }
157*8d67ca89SAndroid Build Coastguard Worker STRNCMP_SHIM()
158*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strnlen)159*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strnlen) {
160*8d67ca89SAndroid Build Coastguard Worker // TODO: enable the SVE version.
161*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strnlen_func_t, __strnlen_aarch64);
162*8d67ca89SAndroid Build Coastguard Worker }
163*8d67ca89SAndroid Build Coastguard Worker STRNLEN_SHIM()
164*8d67ca89SAndroid Build Coastguard Worker
DEFINE_IFUNC_FOR(strrchr)165*8d67ca89SAndroid Build Coastguard Worker DEFINE_IFUNC_FOR(strrchr) {
166*8d67ca89SAndroid Build Coastguard Worker if (arg->_hwcap2 & HWCAP2_MTE) {
167*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strrchr_func_t, __strrchr_aarch64_mte);
168*8d67ca89SAndroid Build Coastguard Worker } else {
169*8d67ca89SAndroid Build Coastguard Worker RETURN_FUNC(strrchr_func_t, __strrchr_aarch64);
170*8d67ca89SAndroid Build Coastguard Worker }
171*8d67ca89SAndroid Build Coastguard Worker }
172*8d67ca89SAndroid Build Coastguard Worker STRRCHR_SHIM()
173*8d67ca89SAndroid Build Coastguard Worker
174*8d67ca89SAndroid Build Coastguard Worker } // extern "C"
175