<T>LAPACK 0.1.1
C++ Template Linear Algebra PACKage
Loading...
Searching...
No Matches
gnuquad.hpp
Go to the documentation of this file.
1
3//
4// Copyright (c) 2025, University of Colorado Denver. All rights reserved.
5//
6// This file is part of <T>LAPACK.
7// <T>LAPACK is free software: you can redistribute it and/or modify it under
8// the terms of the BSD 3-Clause license. See the accompanying LICENSE file.
9
10#ifndef TLAPACK_GNUQUAD_HH
11#define TLAPACK_GNUQUAD_HH
12
13#include <quadmath.h>
14
15#include <complex>
16#include <limits>
17#include <ostream>
18
20
21namespace std {
22
23#if !defined(__STRICT_ANSI__) && defined(_GLIBCXX_USE_FLOAT128)
24constexpr __float128 abs(__float128 x); // See include/bits/std_abs.h
25#else
26inline __float128 abs(__float128 x) noexcept { return fabsq(x); }
27#endif
28inline __float128 ceil(__float128 x) noexcept { return ceilq(x); }
29inline __float128 floor(__float128 x) noexcept { return floorq(x); }
30inline bool isinf(__float128 x) noexcept { return isinfq(x); }
31inline bool isnan(__float128 x) noexcept { return isnanq(x); }
32inline __float128 log2(__float128 x) noexcept { return log2q(x); }
33inline __float128 max(__float128 x, __float128 y) noexcept
34{
35 return fmaxq(x, y);
36}
37inline __float128 min(__float128 x, __float128 y) noexcept
38{
39 return fminq(x, y);
40}
41inline __float128 pow(int x, __float128 y) noexcept { return powq(x, y); }
42inline __float128 sqrt(__float128 x) noexcept { return sqrtq(x); }
43
44inline __float128 abs(complex<__float128> z)
45{
46 __float128 x = z.real();
47 __float128 y = z.imag();
48 const __float128 __s = max(abs(x), abs(y));
49 if (__s == __float128()) // well ...
50 return __s;
51 x /= __s;
52 y /= __s;
53 return __s * sqrt(x * x + y * y);
54}
55inline complex<__float128> sqrt(complex<__float128> z)
56{
57 const __float128 x = real(z);
58 const __float128 y = imag(z);
59 const __float128 zero(0);
60 const __float128 two(2);
61 const __float128 half(0.5);
62
63 if (x == zero) {
64 const __float128 t = sqrt(half * abs(y));
65 return complex<__float128>(t, (y < zero) ? -t : t);
66 }
67 else {
68 const __float128 t = sqrt(two * (abs(z) + abs(x)));
69 const __float128 u = half * t;
70 return (x > zero)
71 ? complex<__float128>(u, y / t)
72 : complex<__float128>(abs(y) / t, (y < zero) ? -u : u);
73 }
74}
75
76template <>
77struct numeric_limits<__float128> {
78 static constexpr bool is_specialized = true;
79
80 static constexpr __float128 min() noexcept;
81 static constexpr __float128 max() noexcept;
82 static constexpr __float128 lowest() noexcept;
83
84 static constexpr int digits = FLT128_MANT_DIG;
85 static constexpr int digits10 = FLT128_DIG;
86 static constexpr int max_digits10 = (2 + FLT128_MANT_DIG * 643L / 2136);
87
88 static constexpr bool is_signed = true;
89 static constexpr bool is_integer = false;
90 static constexpr bool is_exact = false;
91
92 static constexpr int radix = 2;
93 static constexpr __float128 epsilon() noexcept;
94 static constexpr __float128 round_error() noexcept { return 0.5F; }
95
96 static constexpr int min_exponent = FLT128_MIN_EXP;
97 static constexpr int min_exponent10 = FLT128_MIN_10_EXP;
98 static constexpr int max_exponent = FLT128_MAX_EXP;
99 static constexpr int max_exponent10 = FLT128_MAX_10_EXP;
100
101 static constexpr bool has_infinity = true;
102 static constexpr bool has_quiet_NaN = true;
103 // static constexpr bool has_signaling_NaN = ;
104 static constexpr float_denorm_style has_denorm = denorm_present;
105 static constexpr bool has_denorm_loss = false;
106
107 static constexpr __float128 infinity() noexcept
108 {
109 return __builtin_huge_valq();
110 }
111 static __float128 quiet_NaN() noexcept { return nanq(""); }
112 // static constexpr __float128 signaling_NaN() noexcept {}
113 static constexpr __float128 denorm_min() noexcept;
114
115 // static constexpr bool is_iec559 = ;
116 static constexpr bool is_bounded = true;
117 static constexpr bool is_modulo = false;
118
119 // static constexpr bool traps = ;
120 // static constexpr bool tinyness_before = ;
121 static constexpr float_round_style round_style = round_to_nearest;
122};
123
124inline ostream& operator<<(ostream& out, const __float128& x)
125{
126 char buf[128];
127 quadmath_snprintf(buf, sizeof buf, "%+-#*.20Qe", x);
128 out << buf;
129 return out;
130}
131
132inline istream& operator>>(istream& in, __float128& x)
133{
134 char buf[128];
135 in >> buf;
136 x = strtoflt128(buf, nullptr);
137 return in;
138}
139
140#ifdef __GNUC__
141 #pragma GCC system_header
142#endif
143
144constexpr __float128 numeric_limits<__float128>::min() noexcept
145{
146 return FLT128_MIN;
147}
148constexpr __float128 numeric_limits<__float128>::max() noexcept
149{
150 return FLT128_MAX;
151}
152constexpr __float128 numeric_limits<__float128>::lowest() noexcept
153{
154 return -FLT128_MAX;
155}
156constexpr __float128 numeric_limits<__float128>::epsilon() noexcept
157{
158 return FLT128_EPSILON;
159}
160constexpr __float128 numeric_limits<__float128>::denorm_min() noexcept
161{
162 return FLT128_DENORM_MIN;
163}
164
165} // namespace std
166
167namespace tlapack {
168
169namespace traits {
170 // __float128 is a real type that satisfies tlapack::concepts::Real
171 template <>
173 using type = __float128;
174 constexpr static bool is_real = true;
175 };
176 // The complex type of __float128 is std::complex<__float128>
177 template <>
179 using type = std::complex<__float128>;
180 constexpr static bool is_complex = false;
181 };
182} // namespace traits
183
184} // namespace tlapack
185
186#endif
constexpr bool isnan(const T &x) noexcept
Extends std::isnan() to complex numbers.
Definition utils.hpp:125
constexpr real_type< T > real(const T &x) noexcept
Extends std::real() to real datatypes.
Definition utils.hpp:71
constexpr real_type< T > imag(const T &x) noexcept
Extends std::imag() to real datatypes.
Definition utils.hpp:86
constexpr bool isinf(const T &x) noexcept
Extends std::isinf() to complex numbers.
Definition utils.hpp:117
typename traits::real_type_traits< Types..., int >::type real_type
The common real type of the list of types.
Definition scalar_type_traits.hpp:113
constexpr bool is_complex
True if T is a complex scalar type.
Definition scalar_type_traits.hpp:192
constexpr bool is_real
True if T is a real scalar type.
Definition scalar_type_traits.hpp:117
Complex type traits for the list of types Types.
Definition scalar_type_traits.hpp:145
Real type traits for the list of types Types.
Definition scalar_type_traits.hpp:71