xref: /aosp_15_r20/external/vixl/src/compiler-intrinsics-vixl.h (revision f5c631da2f1efdd72b5fd1e20510e4042af13d77)
1*f5c631daSSadaf Ebrahimi // Copyright 2015, VIXL authors
2*f5c631daSSadaf Ebrahimi // All rights reserved.
3*f5c631daSSadaf Ebrahimi //
4*f5c631daSSadaf Ebrahimi // Redistribution and use in source and binary forms, with or without
5*f5c631daSSadaf Ebrahimi // modification, are permitted provided that the following conditions are met:
6*f5c631daSSadaf Ebrahimi //
7*f5c631daSSadaf Ebrahimi //   * Redistributions of source code must retain the above copyright notice,
8*f5c631daSSadaf Ebrahimi //     this list of conditions and the following disclaimer.
9*f5c631daSSadaf Ebrahimi //   * Redistributions in binary form must reproduce the above copyright notice,
10*f5c631daSSadaf Ebrahimi //     this list of conditions and the following disclaimer in the documentation
11*f5c631daSSadaf Ebrahimi //     and/or other materials provided with the distribution.
12*f5c631daSSadaf Ebrahimi //   * Neither the name of ARM Limited nor the names of its contributors may be
13*f5c631daSSadaf Ebrahimi //     used to endorse or promote products derived from this software without
14*f5c631daSSadaf Ebrahimi //     specific prior written permission.
15*f5c631daSSadaf Ebrahimi //
16*f5c631daSSadaf Ebrahimi // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS CONTRIBUTORS "AS IS" AND
17*f5c631daSSadaf Ebrahimi // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18*f5c631daSSadaf Ebrahimi // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19*f5c631daSSadaf Ebrahimi // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
20*f5c631daSSadaf Ebrahimi // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21*f5c631daSSadaf Ebrahimi // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22*f5c631daSSadaf Ebrahimi // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
23*f5c631daSSadaf Ebrahimi // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24*f5c631daSSadaf Ebrahimi // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25*f5c631daSSadaf Ebrahimi // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26*f5c631daSSadaf Ebrahimi 
27*f5c631daSSadaf Ebrahimi 
28*f5c631daSSadaf Ebrahimi #ifndef VIXL_COMPILER_INTRINSICS_H
29*f5c631daSSadaf Ebrahimi #define VIXL_COMPILER_INTRINSICS_H
30*f5c631daSSadaf Ebrahimi 
31*f5c631daSSadaf Ebrahimi #include <limits.h>
32*f5c631daSSadaf Ebrahimi #include "globals-vixl.h"
33*f5c631daSSadaf Ebrahimi 
34*f5c631daSSadaf Ebrahimi namespace vixl {
35*f5c631daSSadaf Ebrahimi 
36*f5c631daSSadaf Ebrahimi // Helper to check whether the version of GCC used is greater than the specified
37*f5c631daSSadaf Ebrahimi // requirement.
38*f5c631daSSadaf Ebrahimi #define MAJOR 1000000
39*f5c631daSSadaf Ebrahimi #define MINOR 1000
40*f5c631daSSadaf Ebrahimi #if defined(__GNUC__) && defined(__GNUC_MINOR__) && defined(__GNUC_PATCHLEVEL__)
41*f5c631daSSadaf Ebrahimi #define GCC_VERSION_OR_NEWER(major, minor, patchlevel)                      \
42*f5c631daSSadaf Ebrahimi   ((__GNUC__ * (MAJOR) + __GNUC_MINOR__ * (MINOR) + __GNUC_PATCHLEVEL__) >= \
43*f5c631daSSadaf Ebrahimi    ((major) * (MAJOR) + ((minor)) * (MINOR) + (patchlevel)))
44*f5c631daSSadaf Ebrahimi #elif defined(__GNUC__) && defined(__GNUC_MINOR__)
45*f5c631daSSadaf Ebrahimi #define GCC_VERSION_OR_NEWER(major, minor, patchlevel) \
46*f5c631daSSadaf Ebrahimi   ((__GNUC__ * (MAJOR) + __GNUC_MINOR__ * (MINOR)) >=  \
47*f5c631daSSadaf Ebrahimi    ((major) * (MAJOR) + ((minor)) * (MINOR) + (patchlevel)))
48*f5c631daSSadaf Ebrahimi #else
49*f5c631daSSadaf Ebrahimi #define GCC_VERSION_OR_NEWER(major, minor, patchlevel) 0
50*f5c631daSSadaf Ebrahimi #endif
51*f5c631daSSadaf Ebrahimi 
52*f5c631daSSadaf Ebrahimi 
53*f5c631daSSadaf Ebrahimi #if defined(__clang__) && !defined(VIXL_NO_COMPILER_BUILTINS)
54*f5c631daSSadaf Ebrahimi 
55*f5c631daSSadaf Ebrahimi // clang-format off
56*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_CLRSB    (__has_builtin(__builtin_clrsb))
57*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_CLZ      (__has_builtin(__builtin_clz))
58*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_CTZ      (__has_builtin(__builtin_ctz))
59*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_FFS      (__has_builtin(__builtin_ffs))
60*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_POPCOUNT (__has_builtin(__builtin_popcount))
61*f5c631daSSadaf Ebrahimi // clang-format on
62*f5c631daSSadaf Ebrahimi 
63*f5c631daSSadaf Ebrahimi #elif defined(__GNUC__) && !defined(VIXL_NO_COMPILER_BUILTINS)
64*f5c631daSSadaf Ebrahimi // The documentation for these builtins is available at:
65*f5c631daSSadaf Ebrahimi // https://gcc.gnu.org/onlinedocs/gcc-$MAJOR.$MINOR.$PATCHLEVEL/gcc//Other-Builtins.html
66*f5c631daSSadaf Ebrahimi 
67*f5c631daSSadaf Ebrahimi // clang-format off
68*f5c631daSSadaf Ebrahimi # define COMPILER_HAS_BUILTIN_CLRSB    (GCC_VERSION_OR_NEWER(4, 7, 0))
69*f5c631daSSadaf Ebrahimi # define COMPILER_HAS_BUILTIN_CLZ      (GCC_VERSION_OR_NEWER(3, 4, 0))
70*f5c631daSSadaf Ebrahimi # define COMPILER_HAS_BUILTIN_CTZ      (GCC_VERSION_OR_NEWER(3, 4, 0))
71*f5c631daSSadaf Ebrahimi # define COMPILER_HAS_BUILTIN_FFS      (GCC_VERSION_OR_NEWER(3, 4, 0))
72*f5c631daSSadaf Ebrahimi # define COMPILER_HAS_BUILTIN_POPCOUNT (GCC_VERSION_OR_NEWER(3, 4, 0))
73*f5c631daSSadaf Ebrahimi // clang-format on
74*f5c631daSSadaf Ebrahimi 
75*f5c631daSSadaf Ebrahimi #else
76*f5c631daSSadaf Ebrahimi // One can define VIXL_NO_COMPILER_BUILTINS to force using the manually
77*f5c631daSSadaf Ebrahimi // implemented C++ methods.
78*f5c631daSSadaf Ebrahimi 
79*f5c631daSSadaf Ebrahimi // clang-format off
80*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_BSWAP    false
81*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_CLRSB    false
82*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_CLZ      false
83*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_CTZ      false
84*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_FFS      false
85*f5c631daSSadaf Ebrahimi #define COMPILER_HAS_BUILTIN_POPCOUNT false
86*f5c631daSSadaf Ebrahimi // clang-format on
87*f5c631daSSadaf Ebrahimi 
88*f5c631daSSadaf Ebrahimi #endif
89*f5c631daSSadaf Ebrahimi 
90*f5c631daSSadaf Ebrahimi 
91*f5c631daSSadaf Ebrahimi template <typename V>
IsPowerOf2(V value)92*f5c631daSSadaf Ebrahimi inline bool IsPowerOf2(V value) {
93*f5c631daSSadaf Ebrahimi   return (value != 0) && ((value & (value - 1)) == 0);
94*f5c631daSSadaf Ebrahimi }
95*f5c631daSSadaf Ebrahimi 
96*f5c631daSSadaf Ebrahimi 
97*f5c631daSSadaf Ebrahimi // Declaration of fallback functions.
98*f5c631daSSadaf Ebrahimi int CountLeadingSignBitsFallBack(int64_t value, int width);
99*f5c631daSSadaf Ebrahimi int CountLeadingZerosFallBack(uint64_t value, int width);
100*f5c631daSSadaf Ebrahimi int CountSetBitsFallBack(uint64_t value, int width);
101*f5c631daSSadaf Ebrahimi int CountTrailingZerosFallBack(uint64_t value, int width);
102*f5c631daSSadaf Ebrahimi 
103*f5c631daSSadaf Ebrahimi 
104*f5c631daSSadaf Ebrahimi // Implementation of intrinsics functions.
105*f5c631daSSadaf Ebrahimi // TODO: The implementations could be improved for sizes different from 32bit
106*f5c631daSSadaf Ebrahimi // and 64bit: we could mask the values and call the appropriate builtin.
107*f5c631daSSadaf Ebrahimi 
108*f5c631daSSadaf Ebrahimi // Return the number of leading bits that match the topmost (sign) bit,
109*f5c631daSSadaf Ebrahimi // excluding the topmost bit itself.
110*f5c631daSSadaf Ebrahimi template <typename V>
111*f5c631daSSadaf Ebrahimi inline int CountLeadingSignBits(V value, int width = (sizeof(V) * 8)) {
112*f5c631daSSadaf Ebrahimi   VIXL_ASSERT(IsPowerOf2(width) && (width <= 64));
113*f5c631daSSadaf Ebrahimi #if COMPILER_HAS_BUILTIN_CLRSB
114*f5c631daSSadaf Ebrahimi   VIXL_ASSERT((LLONG_MIN <= value) && (value <= LLONG_MAX));
115*f5c631daSSadaf Ebrahimi   int ll_width = sizeof(long long) * kBitsPerByte;  // NOLINT(runtime/int)
116*f5c631daSSadaf Ebrahimi   int result = __builtin_clrsbll(value) - (ll_width - width);
117*f5c631daSSadaf Ebrahimi   // Check that the value fits in the specified width.
118*f5c631daSSadaf Ebrahimi   VIXL_ASSERT(result >= 0);
119*f5c631daSSadaf Ebrahimi   return result;
120*f5c631daSSadaf Ebrahimi #else
121*f5c631daSSadaf Ebrahimi   VIXL_ASSERT((INT64_MIN <= value) && (value <= INT64_MAX));
122*f5c631daSSadaf Ebrahimi   return CountLeadingSignBitsFallBack(value, width);
123*f5c631daSSadaf Ebrahimi #endif
124*f5c631daSSadaf Ebrahimi }
125*f5c631daSSadaf Ebrahimi 
126*f5c631daSSadaf Ebrahimi 
127*f5c631daSSadaf Ebrahimi template <typename V>
128*f5c631daSSadaf Ebrahimi inline int CountLeadingZeros(V value, int width = (sizeof(V) * 8)) {
129*f5c631daSSadaf Ebrahimi #if COMPILER_HAS_BUILTIN_CLZ
130*f5c631daSSadaf Ebrahimi   if (width == 32) {
131*f5c631daSSadaf Ebrahimi     return (value == 0) ? 32 : __builtin_clz(static_cast<unsigned>(value));
132*f5c631daSSadaf Ebrahimi   } else if (width == 64) {
133*f5c631daSSadaf Ebrahimi     return (value == 0) ? 64 : __builtin_clzll(value);
134*f5c631daSSadaf Ebrahimi   }
135*f5c631daSSadaf Ebrahimi #endif
136*f5c631daSSadaf Ebrahimi   return CountLeadingZerosFallBack(value, width);
137*f5c631daSSadaf Ebrahimi }
138*f5c631daSSadaf Ebrahimi 
139*f5c631daSSadaf Ebrahimi 
140*f5c631daSSadaf Ebrahimi template <typename V>
141*f5c631daSSadaf Ebrahimi inline int CountSetBits(V value, int width = (sizeof(V) * 8)) {
142*f5c631daSSadaf Ebrahimi #if COMPILER_HAS_BUILTIN_POPCOUNT
143*f5c631daSSadaf Ebrahimi   if (width == 32) {
144*f5c631daSSadaf Ebrahimi     return __builtin_popcount(static_cast<unsigned>(value));
145*f5c631daSSadaf Ebrahimi   } else if (width == 64) {
146*f5c631daSSadaf Ebrahimi     return __builtin_popcountll(value);
147*f5c631daSSadaf Ebrahimi   }
148*f5c631daSSadaf Ebrahimi #endif
149*f5c631daSSadaf Ebrahimi   return CountSetBitsFallBack(value, width);
150*f5c631daSSadaf Ebrahimi }
151*f5c631daSSadaf Ebrahimi 
152*f5c631daSSadaf Ebrahimi 
153*f5c631daSSadaf Ebrahimi template <typename V>
154*f5c631daSSadaf Ebrahimi inline int CountTrailingZeros(V value, int width = (sizeof(V) * 8)) {
155*f5c631daSSadaf Ebrahimi #if COMPILER_HAS_BUILTIN_CTZ
156*f5c631daSSadaf Ebrahimi   if (width == 32) {
157*f5c631daSSadaf Ebrahimi     return (value == 0) ? 32 : __builtin_ctz(static_cast<unsigned>(value));
158*f5c631daSSadaf Ebrahimi   } else if (width == 64) {
159*f5c631daSSadaf Ebrahimi     return (value == 0) ? 64 : __builtin_ctzll(value);
160*f5c631daSSadaf Ebrahimi   }
161*f5c631daSSadaf Ebrahimi #endif
162*f5c631daSSadaf Ebrahimi   return CountTrailingZerosFallBack(value, width);
163*f5c631daSSadaf Ebrahimi }
164*f5c631daSSadaf Ebrahimi 
165*f5c631daSSadaf Ebrahimi }  // namespace vixl
166*f5c631daSSadaf Ebrahimi 
167*f5c631daSSadaf Ebrahimi #endif  // VIXL_COMPILER_INTRINSICS_H
168