summaryrefslogtreecommitdiff
path: root/mpfr/exp2.c
diff options
context:
space:
mode:
authorKevin Ryde <user42@zip.com.au>2002-04-09 02:10:20 +0200
committerKevin Ryde <user42@zip.com.au>2002-04-09 02:10:20 +0200
commitaf5c898efe6d9c24434cb196b6d53dc52611aac7 (patch)
treeb0bb85f345ef57985def7f699b594fdb783250df /mpfr/exp2.c
parent6478d5b3d18cef5a016730d9921f0d2d970ef600 (diff)
downloadgmp-af5c898efe6d9c24434cb196b6d53dc52611aac7.tar.gz
* mpfr/*: Update to 2.0.1.
Diffstat (limited to 'mpfr/exp2.c')
-rw-r--r--mpfr/exp2.c42
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;