diff options
author | Mark Dickinson <dickinsm@gmail.com> | 2010-01-12 22:55:51 (GMT) |
---|---|---|
committer | Mark Dickinson <dickinsm@gmail.com> | 2010-01-12 22:55:51 (GMT) |
commit | 5ff4f279e6342bf142de22c91662b8eb70ce722a (patch) | |
tree | b5e9d55c8edd6b1d2c0082707d9a243e53d5d91e /Python/dtoa.c | |
parent | 8efef5ce9f6c0dfd2cb4720b61a37bbdfdb5e30e (diff) | |
download | cpython-5ff4f279e6342bf142de22c91662b8eb70ce722a.zip cpython-5ff4f279e6342bf142de22c91662b8eb70ce722a.tar.gz cpython-5ff4f279e6342bf142de22c91662b8eb70ce722a.tar.bz2 |
Issue #7632: Fix a bug in dtoa.c that could lead to incorrectly-rounded results.
Diffstat (limited to 'Python/dtoa.c')
-rw-r--r-- | Python/dtoa.c | 24 |
1 files changed, 22 insertions, 2 deletions
diff --git a/Python/dtoa.c b/Python/dtoa.c index 4d64be5..1fe20f4 100644 --- a/Python/dtoa.c +++ b/Python/dtoa.c @@ -1130,6 +1130,26 @@ quorem(Bigint *b, Bigint *S) return q; } +/* version of ulp(x) that takes bc.scale into account. + + Assuming that x is finite and nonzero, and x / 2^bc.scale is exactly + representable as a double, sulp(x) is equivalent to 2^bc.scale * ulp(x / + 2^bc.scale). */ + +static double +sulp(U *x, BCinfo *bc) +{ + U u; + + if (bc->scale && 2*P + 1 - ((word0(x) & Exp_mask) >> Exp_shift) > 0) { + /* rv/2^bc->scale is subnormal */ + word0(&u) = (P+2)*Exp_msk1; + word1(&u) = 0; + return u.d; + } + else + return ulp(x); +} /* return 0 on success, -1 on failure */ @@ -1289,12 +1309,12 @@ bigcomp(U *rv, const char *s0, BCinfo *bc) else if (dd < 0) { if (!dsign) /* does not happen for round-near */ retlow1: - dval(rv) -= ulp(rv); + dval(rv) -= sulp(rv, bc); } else if (dd > 0) { if (dsign) { rethi1: - dval(rv) += ulp(rv); + dval(rv) += sulp(rv, bc); } } else { |