1 /* Exception flags and utilities.
3 Copyright 2001-2016 Free Software Foundation, Inc.
4 Contributed by the AriC and Caramba projects, INRIA.
6 This file is part of the GNU MPFR Library.
8 The GNU MPFR Library is free software; you can redistribute it and/or modify
9 it under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or (at your
11 option) any later version.
13 The GNU MPFR Library is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
15 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
16 License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with the GNU MPFR Library; see the file COPYING.LESSER. If not, see
20 http://www.gnu.org/licenses/ or write to the Free Software Foundation, Inc.,
21 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. */
23 #include "mpfr-impl.h"
25 MPFR_THREAD_ATTR
unsigned int __gmpfr_flags
= 0;
27 MPFR_THREAD_ATTR mpfr_exp_t __gmpfr_emin
= MPFR_EMIN_DEFAULT
;
28 MPFR_THREAD_ATTR mpfr_exp_t __gmpfr_emax
= MPFR_EMAX_DEFAULT
;
30 #ifdef MPFR_WIN_THREAD_SAFE_DLL
31 unsigned int * __gmpfr_flags_f() { return &__gmpfr_flags
; }
32 mpfr_exp_t
* __gmpfr_emin_f() { return &__gmpfr_emin
; }
33 mpfr_exp_t
* __gmpfr_emax_f() { return &__gmpfr_emax
; }
47 mpfr_set_emin (mpfr_exp_t exponent
)
49 if (exponent
>= MPFR_EMIN_MIN
&& exponent
<= MPFR_EMIN_MAX
)
51 __gmpfr_emin
= exponent
;
61 mpfr_get_emin_min (void)
67 mpfr_get_emin_max (void)
83 mpfr_set_emax (mpfr_exp_t exponent
)
85 if (exponent
>= MPFR_EMAX_MIN
&& exponent
<= MPFR_EMAX_MAX
)
87 __gmpfr_emax
= exponent
;
97 mpfr_get_emax_min (void)
102 mpfr_get_emax_max (void)
104 return MPFR_EMAX_MAX
;
108 #undef mpfr_clear_flags
111 mpfr_clear_flags (void)
116 #undef mpfr_clear_underflow
119 mpfr_clear_underflow (void)
121 __gmpfr_flags
&= MPFR_FLAGS_ALL
^ MPFR_FLAGS_UNDERFLOW
;
124 #undef mpfr_clear_overflow
127 mpfr_clear_overflow (void)
129 __gmpfr_flags
&= MPFR_FLAGS_ALL
^ MPFR_FLAGS_OVERFLOW
;
132 #undef mpfr_clear_divby0
135 mpfr_clear_divby0 (void)
137 __gmpfr_flags
&= MPFR_FLAGS_ALL
^ MPFR_FLAGS_DIVBY0
;
140 #undef mpfr_clear_nanflag
143 mpfr_clear_nanflag (void)
145 __gmpfr_flags
&= MPFR_FLAGS_ALL
^ MPFR_FLAGS_NAN
;
148 #undef mpfr_clear_inexflag
151 mpfr_clear_inexflag (void)
153 __gmpfr_flags
&= MPFR_FLAGS_ALL
^ MPFR_FLAGS_INEXACT
;
156 #undef mpfr_clear_erangeflag
159 mpfr_clear_erangeflag (void)
161 __gmpfr_flags
&= MPFR_FLAGS_ALL
^ MPFR_FLAGS_ERANGE
;
164 #undef mpfr_set_underflow
167 mpfr_set_underflow (void)
169 __gmpfr_flags
|= MPFR_FLAGS_UNDERFLOW
;
172 #undef mpfr_set_overflow
175 mpfr_set_overflow (void)
177 __gmpfr_flags
|= MPFR_FLAGS_OVERFLOW
;
180 #undef mpfr_set_divby0
183 mpfr_set_divby0 (void)
185 __gmpfr_flags
|= MPFR_FLAGS_DIVBY0
;
188 #undef mpfr_set_nanflag
191 mpfr_set_nanflag (void)
193 __gmpfr_flags
|= MPFR_FLAGS_NAN
;
196 #undef mpfr_set_inexflag
199 mpfr_set_inexflag (void)
201 __gmpfr_flags
|= MPFR_FLAGS_INEXACT
;
204 #undef mpfr_set_erangeflag
207 mpfr_set_erangeflag (void)
209 __gmpfr_flags
|= MPFR_FLAGS_ERANGE
;
213 #undef mpfr_check_range
216 mpfr_check_range (mpfr_ptr x
, int t
, mpfr_rnd_t rnd_mode
)
218 if (MPFR_LIKELY( MPFR_IS_PURE_FP(x
)) )
219 { /* x is a non-zero FP */
220 mpfr_exp_t exp
= MPFR_EXP (x
); /* Do not use MPFR_GET_EXP */
221 if (MPFR_UNLIKELY( exp
< __gmpfr_emin
) )
223 /* The following test is necessary because in the rounding to the
224 * nearest mode, mpfr_underflow always rounds away from 0. In
225 * this rounding mode, we need to round to 0 if:
226 * _ |x| < 2^(emin-2), or
227 * _ |x| = 2^(emin-2) and the absolute value of the exact
228 * result is <= 2^(emin-2).
230 if (rnd_mode
== MPFR_RNDN
&&
231 (exp
+ 1 < __gmpfr_emin
||
232 (mpfr_powerof2_raw(x
) &&
233 (MPFR_IS_NEG(x
) ? t
<= 0 : t
>= 0))))
234 rnd_mode
= MPFR_RNDZ
;
235 return mpfr_underflow(x
, rnd_mode
, MPFR_SIGN(x
));
237 if (MPFR_UNLIKELY( exp
> __gmpfr_emax
) )
238 return mpfr_overflow (x
, rnd_mode
, MPFR_SIGN(x
));
240 else if (MPFR_UNLIKELY (t
!= 0 && MPFR_IS_INF (x
)))
242 /* We need to do the following because most MPFR functions are
243 * implemented in the following way:
245 * | Compute an approximation to the result and an error bound.
246 * | Possible underflow/overflow detection -> return.
247 * | If can_round, break (exit the loop).
248 * | Otherwise, increase the working precision and loop.
249 * Round the approximation in the target precision. <== See below
250 * Restore the flags (that could have been set due to underflows
251 * or overflows during the internal computations).
252 * Execute: return mpfr_check_range (...).
253 * The problem is that an overflow could be generated when rounding the
254 * approximation (in general, such an overflow could not be detected
255 * earlier), and the overflow flag is lost when the flags are restored.
256 * This can occur only when the rounding yields an exponent change
257 * and the new exponent is larger than the maximum exponent, so that
258 * an infinity is necessarily obtained.
259 * So, the simplest solution is to detect this overflow case here in
260 * mpfr_check_range, which is easy to do since the rounded result is
261 * necessarily an inexact infinity.
263 __gmpfr_flags
|= MPFR_FLAGS_OVERFLOW
;
265 MPFR_RET (t
); /* propagate inexact ternary value, unlike most functions */
268 #undef mpfr_underflow_p
271 mpfr_underflow_p (void)
273 return __gmpfr_flags
& MPFR_FLAGS_UNDERFLOW
;
276 #undef mpfr_overflow_p
279 mpfr_overflow_p (void)
281 return __gmpfr_flags
& MPFR_FLAGS_OVERFLOW
;
289 return __gmpfr_flags
& MPFR_FLAGS_DIVBY0
;
292 #undef mpfr_nanflag_p
295 mpfr_nanflag_p (void)
297 return __gmpfr_flags
& MPFR_FLAGS_NAN
;
300 #undef mpfr_inexflag_p
303 mpfr_inexflag_p (void)
305 return __gmpfr_flags
& MPFR_FLAGS_INEXACT
;
308 #undef mpfr_erangeflag_p
311 mpfr_erangeflag_p (void)
313 return __gmpfr_flags
& MPFR_FLAGS_ERANGE
;
316 /* #undef mpfr_underflow */
318 /* Note: In the rounding to the nearest mode, mpfr_underflow
319 always rounds away from 0. In this rounding mode, you must call
320 mpfr_underflow with rnd_mode = MPFR_RNDZ if the exact result
321 is <= 2^(emin-2) in absolute value. */
324 mpfr_underflow (mpfr_ptr x
, mpfr_rnd_t rnd_mode
, int sign
)
328 MPFR_ASSERT_SIGN (sign
);
330 if (MPFR_IS_LIKE_RNDZ(rnd_mode
, sign
< 0))
337 mpfr_setmin (x
, __gmpfr_emin
);
340 MPFR_SET_SIGN(x
, sign
);
341 __gmpfr_flags
|= MPFR_FLAGS_INEXACT
| MPFR_FLAGS_UNDERFLOW
;
342 return sign
> 0 ? inex
: -inex
;
345 /* #undef mpfr_overflow */
348 mpfr_overflow (mpfr_ptr x
, mpfr_rnd_t rnd_mode
, int sign
)
352 MPFR_ASSERT_SIGN(sign
);
353 if (MPFR_IS_LIKE_RNDZ(rnd_mode
, sign
< 0))
355 mpfr_setmax (x
, __gmpfr_emax
);
363 MPFR_SET_SIGN(x
,sign
);
364 __gmpfr_flags
|= MPFR_FLAGS_INEXACT
| MPFR_FLAGS_OVERFLOW
;
365 return sign
> 0 ? inex
: -inex
;