diff options
author | Kevin Ryde <user42@zip.com.au> | 2002-04-09 02:10:20 +0200 |
---|---|---|
committer | Kevin Ryde <user42@zip.com.au> | 2002-04-09 02:10:20 +0200 |
commit | af5c898efe6d9c24434cb196b6d53dc52611aac7 (patch) | |
tree | b0bb85f345ef57985def7f699b594fdb783250df /mpfr/exp2.c | |
parent | 6478d5b3d18cef5a016730d9921f0d2d970ef600 (diff) | |
download | gmp-af5c898efe6d9c24434cb196b6d53dc52611aac7.tar.gz |
* mpfr/*: Update to 2.0.1.
Diffstat (limited to 'mpfr/exp2.c')
-rw-r--r-- | mpfr/exp2.c | 42 |
1 files changed, 24 insertions, 18 deletions
diff --git a/mpfr/exp2.c b/mpfr/exp2.c index 8370510a1..bb733b99d 100644 --- a/mpfr/exp2.c +++ b/mpfr/exp2.c @@ -1,6 +1,6 @@ /* mpfr_exp2 -- power of 2 function 2^y -Copyright (C) 2001-2002 Free Software Foundation. +Copyright 2001, 2002 Free Software Foundation. This file is part of the MPFR Library. @@ -15,7 +15,7 @@ or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License -along with the MPFR Library; see the file COPYING.LIB. If not, write to +along with the MPFR Library; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ @@ -62,6 +62,11 @@ mpfr_exp2 (mpfr_ptr y, mpfr_srcptr x, mp_rnd_t rnd_mode) if (MPFR_IS_ZERO(x)) return mpfr_set_ui (y, 1, rnd_mode); + /* since the smallest representable non-zero float is 1/2*2^__mpfr_emin, + if x < __mpfr_emin - 1, the result is either 1/2*2^__mpfr_emin or 0 */ + if (mpfr_cmp_si_2exp (x, __mpfr_emin - 1, 0) < 0) + return mpfr_set_underflow (y, rnd_mode, 1); + /* General case */ { /* Declaration of the intermediary variable */ @@ -75,38 +80,39 @@ mpfr_exp2 (mpfr_ptr y, mpfr_srcptr x, mp_rnd_t rnd_mode) long int err; /* Precision of error */ /* compute the precision of intermediary variable */ - Nt=MAX(Nx,Ny); + Nt = MAX(Nx, Ny); /* the optimal number of bits : see algorithms.ps */ - Nt=Nt+5+_mpfr_ceil_log2(Nt); + Nt = Nt + 5 + _mpfr_ceil_log2 (Nt); /* initialise of intermediary variable */ - mpfr_init(t); - mpfr_init(te); + mpfr_init (t); + mpfr_init (te); - /* First computation of cosh */ + /* First computation */ do { /* reactualisation of the precision */ - mpfr_set_prec(t,Nt); - mpfr_set_prec(te,Nt); + mpfr_set_prec (t, Nt); + mpfr_set_prec (te, Nt); /* compute exp(x*ln(2))*/ - mpfr_const_log2(t,GMP_RNDU); /* ln(2) */ - mpfr_mul(te,x,t,GMP_RNDU); /* x*ln(2) */ - mpfr_exp(t,te,GMP_RNDN); /* exp(x*ln(2))*/ + mpfr_const_log2 (t, GMP_RNDU); /* ln(2) */ + mpfr_mul (te, x, t, GMP_RNDU); /* x*ln(2) */ + mpfr_exp (t, te, GMP_RNDN); /* exp(x*ln(2))*/ /* estimate of the error -- see pow function in algorithms.ps*/ - err=Nt-(MPFR_EXP(te)+2); + err = Nt - (MPFR_EXP(te) + 2); /* actualisation of the precision */ - Nt += 10; + Nt += _mpfr_isqrt (Nt) + 10; - } while ((err<0) || !mpfr_can_round(t,err,GMP_RNDN,rnd_mode,Ny)); + } while ((err < 0) || !mpfr_can_round (t, err, GMP_RNDN, rnd_mode, Ny)); - inexact = mpfr_set(y,t,rnd_mode); - mpfr_clear(t); - mpfr_clear(te); + inexact = mpfr_set (y, t, rnd_mode); + + mpfr_clear (t); + mpfr_clear (te); } return inexact; |