diff options
author | vlefevre <vlefevre@280ebfd0-de03-0410-8827-d642c229c3f4> | 2016-05-20 14:17:57 +0000 |
---|---|---|
committer | vlefevre <vlefevre@280ebfd0-de03-0410-8827-d642c229c3f4> | 2016-05-20 14:17:57 +0000 |
commit | a6a3a797f8f1911f45428cca0bd659a9d8cd2e1c (patch) | |
tree | 84e264142417840814b8975b6cd0ed9a78e811e5 /src/sub_ui.c | |
parent | 829e8e990c5fc951351c0b4948fd6f62f41ca231 (diff) | |
download | mpfr-a6a3a797f8f1911f45428cca0bd659a9d8cd2e1c.tar.gz |
Partly rewrote mpfr_add_ui and mpfr_sub_ui (more similar to the
mpfr_ui_sub code), solving the failures.
git-svn-id: svn://scm.gforge.inria.fr/svn/mpfr/trunk@10300 280ebfd0-de03-0410-8827-d642c229c3f4
Diffstat (limited to 'src/sub_ui.c')
-rw-r--r-- | src/sub_ui.c | 77 |
1 files changed, 49 insertions, 28 deletions
diff --git a/src/sub_ui.c b/src/sub_ui.c index 50de94181..0b6d60ff1 100644 --- a/src/sub_ui.c +++ b/src/sub_ui.c @@ -27,34 +27,55 @@ http://www.gnu.org/licenses/ or write to the Free Software Foundation, Inc., int mpfr_sub_ui (mpfr_ptr y, mpfr_srcptr x, unsigned long int u, mpfr_rnd_t rnd_mode) { - if (MPFR_LIKELY (u != 0)) /* if u=0, do nothing */ + MPFR_LOG_FUNC + (("x[%Pu]=%.*Rg u=%lu rnd=%d", + mpfr_get_prec(x), mpfr_log_prec, x, u, rnd_mode), + ("y[%Pu]=%.*Rg", mpfr_get_prec (y), mpfr_log_prec, y)); + + /* (unsigned long) 0 is assumed to be a real 0 (unsigned) */ + if (MPFR_UNLIKELY (u == 0)) + return mpfr_set (y, x, rnd_mode); + + /* This block is actually useless, but this is a minor optimization. */ + if (MPFR_UNLIKELY (MPFR_IS_SINGULAR (x))) { - mpfr_t uu; - mp_limb_t up[1]; - int cnt; - int inex; - - MPFR_SAVE_EXPO_DECL (expo); - - MPFR_LOG_FUNC - (("x[%Pu]=%.*Rg u=%lu rnd=%d", - mpfr_get_prec(x), mpfr_log_prec, x, u, rnd_mode), - ("y[%Pu]=%.*Rg inexact=%d", - mpfr_get_prec(y), mpfr_log_prec, y, inex)); - - MPFR_TMP_INIT1 (up, uu, GMP_NUMB_BITS); - MPFR_ASSERTN (u == (mp_limb_t) u); - count_leading_zeros (cnt, (mp_limb_t) u); - *up = (mp_limb_t) u << cnt; - - /* Optimization note: Exponent save/restore operations may be - removed if mpfr_sub works even when uu is out-of-range. */ - MPFR_SAVE_EXPO_MARK (expo); - MPFR_SET_EXP (uu, GMP_NUMB_BITS - cnt); - inex = mpfr_sub (y, x, uu, rnd_mode); - MPFR_SAVE_EXPO_FREE (expo); - return mpfr_check_range (y, inex, rnd_mode); + if (MPFR_IS_NAN (x)) + { + MPFR_SET_NAN (y); + MPFR_RET_NAN; + } + if (MPFR_IS_INF (x)) + { + MPFR_SET_INF (y); + MPFR_SET_SAME_SIGN (y, x); + MPFR_RET (0); /* +/-infinity is exact */ + } + /* The case x being zero is handled below: we can't use mpfr_set_si + as the opposite of u does not necessarily fit in a long. */ } - else - return mpfr_set (y, x, rnd_mode); + + /* Main code */ + { + mpfr_t uu; + mp_limb_t up[1]; + int cnt; + int inex; + MPFR_SAVE_EXPO_DECL (expo); + + MPFR_TMP_INIT1 (up, uu, GMP_NUMB_BITS); + MPFR_STAT_STATIC_ASSERT ((mp_limb_t) -1 >= (unsigned long) -1); + /* So, u fits in a mp_limb_t, which justifies the casts below. */ + MPFR_ASSERTD (u != 0); + count_leading_zeros (cnt, (mp_limb_t) u); + *up = (mp_limb_t) u << cnt; + + /* Optimization note: Exponent save/restore operations may be + removed if mpfr_sub works even when uu is out-of-range. */ + MPFR_SAVE_EXPO_MARK (expo); + MPFR_SET_EXP (uu, GMP_NUMB_BITS - cnt); + inex = mpfr_sub (y, x, uu, rnd_mode); + MPFR_SAVE_EXPO_UPDATE_FLAGS (expo, __gmpfr_flags); + MPFR_SAVE_EXPO_FREE (expo); + return mpfr_check_range (y, inex, rnd_mode); + } } |