summaryrefslogtreecommitdiff
path: root/src/asin.c
diff options
context:
space:
mode:
authorvlefevre <vlefevre@280ebfd0-de03-0410-8827-d642c229c3f4>2011-04-12 08:17:39 +0000
committervlefevre <vlefevre@280ebfd0-de03-0410-8827-d642c229c3f4>2011-04-12 08:17:39 +0000
commite1ccffa82aed0a4c68551a06c494532bec7b54c9 (patch)
tree3615ebfe6ac8c5407e84bfba51d5e65def34c53a /src/asin.c
parent0cf1dfa88c88fc2a0c0049f61f4e2b699a7dc6ae (diff)
downloadmpfr-e1ccffa82aed0a4c68551a06c494532bec7b54c9.tar.gz
[src/asin.c] Fixed bug in mpfr_asin for x = 1 or -1 in extremely reduced
exponent range (when pi is not representable, but pi/2 is). [tests/tasin.c] Added testcase. git-svn-id: svn://scm.gforge.inria.fr/svn/mpfr/trunk@7625 280ebfd0-de03-0410-8827-d642c229c3f4
Diffstat (limited to 'src/asin.c')
-rw-r--r--src/asin.c64
1 files changed, 33 insertions, 31 deletions
diff --git a/src/asin.c b/src/asin.c
index dafc48108..3b9f3797c 100644
--- a/src/asin.c
+++ b/src/asin.c
@@ -65,11 +65,14 @@ mpfr_asin (mpfr_ptr asin, mpfr_srcptr x, mpfr_rnd_t rnd_mode)
compared = mpfr_cmp_ui (xp, 1);
+ MPFR_SAVE_EXPO_MARK (expo);
+
if (MPFR_UNLIKELY (compared >= 0))
{
mpfr_clear (xp);
if (compared > 0) /* asin(x) = NaN for |x| > 1 */
{
+ MPFR_SAVE_EXPO_FREE (expo);
MPFR_SET_NAN (asin);
MPFR_RET_NAN;
}
@@ -82,41 +85,40 @@ mpfr_asin (mpfr_ptr asin, mpfr_srcptr x, mpfr_rnd_t rnd_mode)
inexact = -mpfr_const_pi (asin, MPFR_INVERT_RND(rnd_mode));
MPFR_CHANGE_SIGN (asin);
}
- mpfr_div_2ui (asin, asin, 1, rnd_mode); /* May underflow */
- return inexact;
+ mpfr_div_2ui (asin, asin, 1, rnd_mode);
}
}
-
- MPFR_SAVE_EXPO_MARK (expo);
-
- /* Compute exponent of 1 - ABS(x) */
- mpfr_ui_sub (xp, 1, xp, MPFR_RNDD);
- MPFR_ASSERTD (MPFR_GET_EXP (xp) <= 0);
- MPFR_ASSERTD (MPFR_GET_EXP (x) <= 0);
- xp_exp = 2 - MPFR_GET_EXP (xp);
-
- /* Set up initial prec */
- prec = MPFR_PREC (asin) + 10 + xp_exp;
-
- /* use asin(x) = atan(x/sqrt(1-x^2)) */
- MPFR_ZIV_INIT (loop, prec);
- for (;;)
+ else
{
- mpfr_set_prec (xp, prec);
- mpfr_sqr (xp, x, MPFR_RNDN);
- mpfr_ui_sub (xp, 1, xp, MPFR_RNDN);
- mpfr_sqrt (xp, xp, MPFR_RNDN);
- mpfr_div (xp, x, xp, MPFR_RNDN);
- mpfr_atan (xp, xp, MPFR_RNDN);
- if (MPFR_LIKELY (MPFR_CAN_ROUND (xp, prec - xp_exp,
- MPFR_PREC (asin), rnd_mode)))
- break;
- MPFR_ZIV_NEXT (loop, prec);
- }
- MPFR_ZIV_FREE (loop);
- inexact = mpfr_set (asin, xp, rnd_mode);
+ /* Compute exponent of 1 - ABS(x) */
+ mpfr_ui_sub (xp, 1, xp, MPFR_RNDD);
+ MPFR_ASSERTD (MPFR_GET_EXP (xp) <= 0);
+ MPFR_ASSERTD (MPFR_GET_EXP (x) <= 0);
+ xp_exp = 2 - MPFR_GET_EXP (xp);
+
+ /* Set up initial prec */
+ prec = MPFR_PREC (asin) + 10 + xp_exp;
+
+ /* use asin(x) = atan(x/sqrt(1-x^2)) */
+ MPFR_ZIV_INIT (loop, prec);
+ for (;;)
+ {
+ mpfr_set_prec (xp, prec);
+ mpfr_sqr (xp, x, MPFR_RNDN);
+ mpfr_ui_sub (xp, 1, xp, MPFR_RNDN);
+ mpfr_sqrt (xp, xp, MPFR_RNDN);
+ mpfr_div (xp, x, xp, MPFR_RNDN);
+ mpfr_atan (xp, xp, MPFR_RNDN);
+ if (MPFR_LIKELY (MPFR_CAN_ROUND (xp, prec - xp_exp,
+ MPFR_PREC (asin), rnd_mode)))
+ break;
+ MPFR_ZIV_NEXT (loop, prec);
+ }
+ MPFR_ZIV_FREE (loop);
+ inexact = mpfr_set (asin, xp, rnd_mode);
- mpfr_clear (xp);
+ mpfr_clear (xp);
+ }
MPFR_SAVE_EXPO_FREE (expo);
return mpfr_check_range (asin, inexact, rnd_mode);