diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2021-10-09 22:53:29 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2021-10-09 22:53:29 -0700 |
commit | c9d98cf5b72dc5408687c27d9ce5873ddf6aa695 (patch) | |
tree | e0718c7ab59283f014bf99a910ac23aea9d14146 /mpi | |
parent | 60b367cddd49f3d387851746ad1699eeb654da49 (diff) | |
download | txr-c9d98cf5b72dc5408687c27d9ce5873ddf6aa695.tar.gz txr-c9d98cf5b72dc5408687c27d9ce5873ddf6aa695.tar.bz2 txr-c9d98cf5b72dc5408687c27d9ce5873ddf6aa695.zip |
math: two bad edge cases in double_uintptr_t conversion.
This fixes two failing test cases introduced in the
parent commit.
* arith.c (c_dbl_unum): Here, what is wrong that if the
incoming value is a CHR or NUM, we just convert it to a signed
cnum, and return that value. The problem with this is that
negative values are supposed to be out of range for
double_uintptr_t. We now check for negative and route to
the out-of-range error.
* mpi/mpi.c (s_mp_in_big_range): Here, the edge case of
handling the most negative two's complement value is
incorrectly coded. We replace the logic by a simple test for
that exact special case. If a negative bignum being tested
whether it fits into the signed double_intptr_t, then
we check whether its mantissa has the 0x80..00 bit pattern.
That is the only value greater than 0x7F..FF that is
still in range, so we return 1 for that case. We remove the
bogus subtraction (top - neg). After handling the above
special value, we just need to look whether the most
significant word of the bignum is 0x7F...FF or lower.
Diffstat (limited to 'mpi')
-rw-r--r-- | mpi/mpi.c | 22 |
1 files changed, 21 insertions, 1 deletions
@@ -582,10 +582,30 @@ static int s_mp_in_big_range(mp_int *mp, double_uintptr_t lim, int unsig) if (nd > ptrnd) return 0; + if (neg) { + mp_digit *dp = DIGITS(mp); + const mp_digit Ox8__0 = MP_DIGIT_MAX ^ (MP_DIGIT_MAX >> 1); + + switch (ptrnd) { + case 1: + if (dp[0] == Ox8__0) + return 1; + break; + case 2: + if (dp[0] == 0 && dp[1] == Ox8__0) + return 1; + break; + case 4: + if (dp[0] == 0 && dp[1] == 0 && dp[2] == 0 && dp[3] == Ox8__0) + return 1; + break; + } + } + { mp_digit top = DIGITS(mp)[ptrnd - 1]; lim >>= ((ptrnd - 1) * MP_DIGIT_BIT); - return (top - neg) <= lim; + return top <= lim; } } |