xref: /freebsd/contrib/llvm-project/compiler-rt/lib/builtins/int_math.h (revision 415efcecd8b80f68e76376ef2b854cb6f5c84b5a)
1 //===-- int_math.h - internal math inlines --------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file is not part of the interface of this library.
10 //
11 // This file defines substitutes for the libm functions used in some of the
12 // compiler-rt implementations, defined in such a way that there is not a direct
13 // dependency on libm or math.h. Instead, we use the compiler builtin versions
14 // where available. This reduces our dependencies on the system SDK by foisting
15 // the responsibility onto the compiler.
16 //
17 //===----------------------------------------------------------------------===//
18 
19 #ifndef INT_MATH_H
20 #define INT_MATH_H
21 
22 #ifndef __has_builtin
23 #define __has_builtin(x) 0
24 #endif
25 
26 #if defined(_MSC_VER) && !defined(__clang__)
27 #include <math.h>
28 #include <stdlib.h>
29 #endif
30 
31 #if defined(_MSC_VER) && !defined(__clang__)
32 #define CRT_INFINITY INFINITY
33 #else
34 #define CRT_INFINITY __builtin_huge_valf()
35 #endif
36 
37 #if defined(_MSC_VER) && !defined(__clang__)
38 #define crt_isfinite(x) _finite((x))
39 #define crt_isinf(x) !_finite((x))
40 #define crt_isnan(x) _isnan((x))
41 #else
42 // Define crt_isfinite in terms of the builtin if available, otherwise provide
43 // an alternate version in terms of our other functions. This supports some
44 // versions of GCC which didn't have __builtin_isfinite.
45 #if __has_builtin(__builtin_isfinite)
46 #define crt_isfinite(x) __builtin_isfinite((x))
47 #elif defined(__GNUC__)
48 #define crt_isfinite(x)                                                        \
49   __extension__(({                                                             \
50     __typeof((x)) x_ = (x);                                                    \
51     !crt_isinf(x_) && !crt_isnan(x_);                                          \
52   }))
53 #else
54 #error "Do not know how to check for infinity"
55 #endif // __has_builtin(__builtin_isfinite)
56 #define crt_isinf(x) __builtin_isinf((x))
57 #define crt_isnan(x) __builtin_isnan((x))
58 #endif // _MSC_VER
59 
60 #if defined(_MSC_VER) && !defined(__clang__)
61 #define crt_copysign(x, y) copysign((x), (y))
62 #define crt_copysignf(x, y) copysignf((x), (y))
63 #define crt_copysignl(x, y) copysignl((x), (y))
64 #else
65 #define crt_copysign(x, y) __builtin_copysign((x), (y))
66 #define crt_copysignf(x, y) __builtin_copysignf((x), (y))
67 #define crt_copysignl(x, y) __builtin_copysignl((x), (y))
68 // We define __has_builtin to always return 0 for GCC versions below 10,
69 // but __builtin_copysignf128 is available since version 7.
70 #if __has_builtin(__builtin_copysignf128) ||                                   \
71     (defined(__GNUC__) && __GNUC__ >= 7)
72 #define crt_copysignf128(x, y) __builtin_copysignf128((x), (y))
73 #elif __has_builtin(__builtin_copysignq)
74 #define crt_copysignf128(x, y) __builtin_copysignq((x), (y))
75 #endif
76 #endif
77 
78 #if defined(_MSC_VER) && !defined(__clang__)
79 #define crt_fabs(x) fabs((x))
80 #define crt_fabsf(x) fabsf((x))
81 #define crt_fabsl(x) fabs((x))
82 #else
83 #define crt_fabs(x) __builtin_fabs((x))
84 #define crt_fabsf(x) __builtin_fabsf((x))
85 #define crt_fabsl(x) __builtin_fabsl((x))
86 // We define __has_builtin to always return 0 for GCC versions below 10,
87 // but __builtin_fabsf128 is available since version 7.
88 #if __has_builtin(__builtin_fabsf128) || (defined(__GNUC__) && __GNUC__ >= 7)
89 #define crt_fabsf128(x) __builtin_fabsf128((x))
90 #elif __has_builtin(__builtin_fabsq)
91 #define crt_fabsf128(x) __builtin_fabsq((x))
92 #endif
93 #endif
94 
95 #if defined(_MSC_VER) && !defined(__clang__)
96 #define crt_fmaxl(x, y) __max((x), (y))
97 #else
98 #define crt_fmaxl(x, y) __builtin_fmaxl((x), (y))
99 #endif
100 
101 #if defined(_MSC_VER) && !defined(__clang__)
102 #define crt_logbl(x) logbl((x))
103 #else
104 #define crt_logbl(x) __builtin_logbl((x))
105 #endif
106 
107 #if defined(_MSC_VER) && !defined(__clang__)
108 #define crt_scalbnl(x, y) scalbnl((x), (y))
109 #else
110 #define crt_scalbnl(x, y) __builtin_scalbnl((x), (y))
111 #endif
112 
113 #endif // INT_MATH_H
114