summaryrefslogtreecommitdiff
path: root/src/sub_ui.c
diff options
context:
space:
mode:
authorvlefevre <vlefevre@280ebfd0-de03-0410-8827-d642c229c3f4>2016-05-20 14:17:57 +0000
committervlefevre <vlefevre@280ebfd0-de03-0410-8827-d642c229c3f4>2016-05-20 14:17:57 +0000
commita6a3a797f8f1911f45428cca0bd659a9d8cd2e1c (patch)
tree84e264142417840814b8975b6cd0ed9a78e811e5 /src/sub_ui.c
parent829e8e990c5fc951351c0b4948fd6f62f41ca231 (diff)
downloadmpfr-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.c77
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);
+ }
}