Line data Source code
1 : /* Definitions for simple data type for real numbers.
2 : Copyright (C) 2002-2026 Free Software Foundation, Inc.
3 :
4 : This file is part of GCC.
5 :
6 : GCC is free software; you can redistribute it and/or modify it under
7 : the terms of the GNU General Public License as published by the Free
8 : Software Foundation; either version 3, or (at your option) any later
9 : version.
10 :
11 : GCC is distributed in the hope that it will be useful, but WITHOUT ANY
12 : WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 : FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 : for more details.
15 :
16 : You should have received a copy of the GNU General Public License
17 : along with GCC; see the file COPYING3. If not see
18 : <http://www.gnu.org/licenses/>. */
19 :
20 : #ifndef GCC_SREAL_H
21 : #define GCC_SREAL_H
22 :
23 : #define SREAL_PART_BITS 31
24 :
25 : #define UINT64_BITS 64
26 :
27 : #define SREAL_MIN_SIG ((int64_t) 1 << (SREAL_PART_BITS - 2))
28 : #define SREAL_MAX_SIG (((int64_t) 1 << (SREAL_PART_BITS - 1)) - 1)
29 : #define SREAL_MAX_EXP (INT_MAX / 4)
30 :
31 : #define SREAL_BITS SREAL_PART_BITS
32 :
33 : #define SREAL_SIGN(v) (v < 0 ? -1 : 1)
34 : #define SREAL_ABS(v) (v < 0 ? -v : v)
35 :
36 : struct output_block;
37 : class lto_input_block;
38 :
39 : /* Structure for holding a simple real number. */
40 : class sreal
41 : {
42 : public:
43 : /* Construct an uninitialized sreal. */
44 1673317676 : sreal () : m_sig (-1), m_exp (-1) {}
45 :
46 : /* Construct a sreal. */
47 3147784990 : sreal (int64_t sig, int exp = 0)
48 3147784990 : {
49 2701716488 : normalize (sig, exp);
50 1334685 : }
51 :
52 : void dump (FILE *) const;
53 : int64_t to_int () const;
54 : int64_t to_nearest_int () const;
55 : double to_double () const;
56 : void stream_out (struct output_block *);
57 : static sreal stream_in (class lto_input_block *);
58 : sreal operator+ (const sreal &other) const;
59 : sreal operator- (const sreal &other) const;
60 : sreal operator* (const sreal &other) const;
61 : sreal operator/ (const sreal &other) const;
62 :
63 592812855 : bool operator< (const sreal &other) const
64 : {
65 592812855 : if (m_exp == other.m_exp)
66 34435117 : return m_sig < other.m_sig;
67 : else
68 : {
69 558377738 : bool negative = m_sig < 0;
70 558377738 : bool other_negative = other.m_sig < 0;
71 :
72 558377738 : if (negative != other_negative)
73 37276459 : return negative > other_negative;
74 :
75 521101279 : bool r = m_exp < other.m_exp;
76 521101279 : return negative ? !r : r;
77 : }
78 : }
79 :
80 1074038103 : bool operator== (const sreal &other) const
81 : {
82 250377306 : return m_exp == other.m_exp && m_sig == other.m_sig;
83 : }
84 :
85 10063867 : sreal operator- () const
86 : {
87 10063867 : sreal tmp = *this;
88 10063867 : tmp.m_sig *= -1;
89 :
90 10063867 : return tmp;
91 : }
92 :
93 60972772 : sreal shift (int s) const
94 : {
95 : /* Zero needs no shifting. */
96 60972772 : if (!m_sig)
97 6440447 : return *this;
98 54532325 : gcc_checking_assert (s <= SREAL_MAX_EXP);
99 54532325 : gcc_checking_assert (s >= -SREAL_MAX_EXP);
100 :
101 : /* Overflows/drop to 0 could be handled gracefully, but hopefully we do not
102 : need to do so. */
103 54532325 : gcc_checking_assert (m_exp + s <= SREAL_MAX_EXP);
104 54532325 : gcc_checking_assert (m_exp + s >= -SREAL_MAX_EXP);
105 :
106 54532325 : sreal tmp = *this;
107 54532325 : tmp.m_exp += s;
108 :
109 54532325 : return tmp;
110 : }
111 :
112 : /* Global minimum sreal can hold. */
113 1268014 : inline static sreal min ()
114 : {
115 1268014 : sreal min;
116 : /* This never needs normalization. */
117 1268014 : min.m_sig = -SREAL_MAX_SIG;
118 1268014 : min.m_exp = SREAL_MAX_EXP;
119 1268014 : return min;
120 : }
121 :
122 : /* Global minimum sreal can hold. */
123 2714833 : inline static sreal max ()
124 : {
125 2714833 : sreal max;
126 : /* This never needs normalization. */
127 2714833 : max.m_sig = SREAL_MAX_SIG;
128 2714833 : max.m_exp = SREAL_MAX_EXP;
129 475784 : return max;
130 : }
131 :
132 : private:
133 : inline void normalize (int64_t new_sig, signed int new_exp);
134 : inline void normalize_up (int64_t new_sig, signed int new_exp);
135 : inline void normalize_down (int64_t new_sig, signed int new_exp);
136 : void shift_right (int amount);
137 : static sreal signedless_plus (const sreal &a, const sreal &b, bool negative);
138 : static sreal signedless_minus (const sreal &a, const sreal &b, bool negative);
139 :
140 : int32_t m_sig; /* Significant. */
141 : signed int m_exp; /* Exponent. */
142 : };
143 :
144 : extern void debug (const sreal &ref);
145 : extern void debug (const sreal *ptr);
146 :
147 657077936 : inline sreal &operator+= (sreal &a, const sreal &b)
148 : {
149 655756221 : return a = a + b;
150 : }
151 :
152 504574 : inline sreal &operator-= (sreal &a, const sreal &b)
153 : {
154 504574 : return a = a - b;
155 : }
156 :
157 507182 : inline sreal &operator/= (sreal &a, const sreal &b)
158 : {
159 507182 : return a = a / b;
160 : }
161 :
162 18164380 : inline sreal &operator*= (sreal &a, const sreal &b)
163 : {
164 18164380 : return a = a * b;
165 : }
166 :
167 65277311 : inline bool operator!= (const sreal &a, const sreal &b)
168 : {
169 113762316 : return !(a == b);
170 : }
171 :
172 438777649 : inline bool operator> (const sreal &a, const sreal &b)
173 : {
174 451906094 : return !(a == b || a < b);
175 : }
176 :
177 12333970 : inline bool operator<= (const sreal &a, const sreal &b)
178 : {
179 12333970 : return a < b || a == b;
180 : }
181 :
182 482703677 : inline bool operator>= (const sreal &a, const sreal &b)
183 : {
184 484521855 : return a == b || a > b;
185 : }
186 :
187 2971955 : inline sreal operator<< (const sreal &a, int exp)
188 : {
189 2971955 : return a.shift (exp);
190 : }
191 :
192 48256952 : inline sreal operator>> (const sreal &a, int exp)
193 : {
194 48256952 : return a.shift (-exp);
195 : }
196 :
197 : /* Make significant to be >= SREAL_MIN_SIG.
198 :
199 : Make this separate method so inliner can handle hot path better. */
200 :
201 : inline void
202 1156028202 : sreal::normalize_up (int64_t new_sig, signed int new_exp)
203 : {
204 1156028202 : unsigned HOST_WIDE_INT sig = absu_hwi (new_sig);
205 1156028202 : int shift = SREAL_PART_BITS - 2 - floor_log2 (sig);
206 :
207 1156028202 : gcc_checking_assert (shift > 0);
208 1156028202 : sig <<= shift;
209 1156028202 : new_exp -= shift;
210 1156028202 : gcc_checking_assert (sig <= SREAL_MAX_SIG && sig >= SREAL_MIN_SIG);
211 :
212 : /* Check underflow. */
213 1156028202 : if (new_exp < -SREAL_MAX_EXP)
214 : {
215 0 : new_exp = -SREAL_MAX_EXP;
216 0 : sig = 0;
217 : }
218 1156028202 : m_exp = new_exp;
219 1156028202 : if (SREAL_SIGN (new_sig) == -1)
220 43368235 : m_sig = -sig;
221 : else
222 1112659967 : m_sig = sig;
223 1156028202 : }
224 :
225 : /* Make significant to be <= SREAL_MAX_SIG.
226 :
227 : Make this separate method so inliner can handle hot path better. */
228 :
229 : inline void
230 955317272 : sreal::normalize_down (int64_t new_sig, signed int new_exp)
231 : {
232 955317272 : int last_bit;
233 955317272 : unsigned HOST_WIDE_INT sig = absu_hwi (new_sig);
234 955317272 : int shift = floor_log2 (sig) - SREAL_PART_BITS + 2;
235 :
236 955317272 : gcc_checking_assert (shift > 0);
237 955317272 : last_bit = (sig >> (shift-1)) & 1;
238 955317272 : sig >>= shift;
239 955317272 : new_exp += shift;
240 955317272 : gcc_checking_assert (sig <= SREAL_MAX_SIG && sig >= SREAL_MIN_SIG);
241 :
242 : /* Round the number. */
243 955317272 : sig += last_bit;
244 955317272 : if (sig > SREAL_MAX_SIG)
245 : {
246 3536959 : sig >>= 1;
247 3536959 : new_exp++;
248 : }
249 :
250 : /* Check overflow. */
251 955317272 : if (new_exp > SREAL_MAX_EXP)
252 : {
253 0 : new_exp = SREAL_MAX_EXP;
254 0 : sig = SREAL_MAX_SIG;
255 : }
256 955317272 : m_exp = new_exp;
257 955317272 : if (SREAL_SIGN (new_sig) == -1)
258 16529246 : m_sig = -sig;
259 : else
260 938788026 : m_sig = sig;
261 955317272 : }
262 :
263 : /* Normalize *this; the hot path. */
264 :
265 : inline void
266 3637487881 : sreal::normalize (int64_t new_sig, signed int new_exp)
267 : {
268 3637487881 : unsigned HOST_WIDE_INT sig = absu_hwi (new_sig);
269 :
270 3637487881 : if (sig == 0)
271 : {
272 1116117655 : m_sig = 0;
273 1116117655 : m_exp = -SREAL_MAX_EXP;
274 : }
275 2521370226 : else if (sig > SREAL_MAX_SIG)
276 955317272 : normalize_down (new_sig, new_exp);
277 1566052954 : else if (sig < SREAL_MIN_SIG)
278 1156028202 : normalize_up (new_sig, new_exp);
279 : else
280 : {
281 410024752 : m_sig = new_sig;
282 410024752 : m_exp = new_exp;
283 : }
284 3637487881 : }
285 :
286 : #endif
|