diff options
author | Jack Lloyd <[email protected]> | 2018-12-01 13:06:31 -0500 |
---|---|---|
committer | Jack Lloyd <[email protected]> | 2018-12-01 13:06:31 -0500 |
commit | be796a3f30d4bec1af7c0e6770a0397dca6ccd4a (patch) | |
tree | 0b2607f4193a899c4012e3072f4a71e9d17128e4 /src/lib/math | |
parent | 5cc1b897edf52a2037004bae28291c67c966c320 (diff) |
Correct a bug in BigInt::operator%(word)
If reducing a negative number modulo a power of 2, an incorrect
result would be returned. This only affected the versions taking
a single word as the modulo.
Diffstat (limited to 'src/lib/math')
-rw-r--r-- | src/lib/math/bigint/big_ops2.cpp | 30 | ||||
-rw-r--r-- | src/lib/math/bigint/big_ops3.cpp | 17 |
2 files changed, 26 insertions, 21 deletions
diff --git a/src/lib/math/bigint/big_ops2.cpp b/src/lib/math/bigint/big_ops2.cpp index 815771294..d35dc082e 100644 --- a/src/lib/math/bigint/big_ops2.cpp +++ b/src/lib/math/bigint/big_ops2.cpp @@ -293,18 +293,18 @@ word BigInt::operator%=(word mod) if(mod == 0) throw BigInt::DivideByZero(); + word remainder = 0; + if(is_power_of_2(mod)) { - const word remainder = (word_at(0) & (mod - 1)); - m_data.set_to_zero(); - m_data.set_word_at(0, remainder); - return remainder; + remainder = (word_at(0) & (mod - 1)); } - - word remainder = 0; - - for(size_t j = sig_words(); j > 0; --j) - remainder = bigint_modop(remainder, word_at(j-1), mod); + else + { + const size_t sw = sig_words(); + for(size_t i = sw; i > 0; --i) + remainder = bigint_modop(remainder, word_at(i-1), mod); + } if(remainder && sign() == BigInt::Negative) remainder = mod - remainder; @@ -312,8 +312,7 @@ word BigInt::operator%=(word mod) m_data.set_to_zero(); m_data.set_word_at(0, remainder); set_sign(BigInt::Positive); - - return word_at(0); + return remainder; } /* @@ -323,9 +322,9 @@ BigInt& BigInt::operator<<=(size_t shift) { if(shift) { - const size_t shift_words = shift / BOTAN_MP_WORD_BITS, - shift_bits = shift % BOTAN_MP_WORD_BITS, - words = sig_words(); + const size_t shift_words = shift / BOTAN_MP_WORD_BITS; + const size_t shift_bits = shift % BOTAN_MP_WORD_BITS; + const size_t words = size(); /* * FIXME - if shift_words == 0 && the top shift_bits of the top word @@ -351,9 +350,8 @@ BigInt& BigInt::operator>>=(size_t shift) { const size_t shift_words = shift / BOTAN_MP_WORD_BITS; const size_t shift_bits = shift % BOTAN_MP_WORD_BITS; - const size_t sw = sig_words(); - bigint_shr1(m_data.mutable_data(), sw, shift_words, shift_bits); + bigint_shr1(m_data.mutable_data(), m_data.size(), shift_words, shift_bits); if(is_negative() && is_zero()) set_sign(Positive); diff --git a/src/lib/math/bigint/big_ops3.cpp b/src/lib/math/bigint/big_ops3.cpp index a38448050..7a4c8f215 100644 --- a/src/lib/math/bigint/big_ops3.cpp +++ b/src/lib/math/bigint/big_ops3.cpp @@ -127,13 +127,20 @@ word operator%(const BigInt& n, word mod) if(mod == 1) return 0; - if(is_power_of_2(mod)) - return (n.word_at(0) & (mod - 1)); - word remainder = 0; - for(size_t j = n.sig_words(); j > 0; --j) - remainder = bigint_modop(remainder, n.word_at(j-1), mod); + if(is_power_of_2(mod)) + { + remainder = (n.word_at(0) & (mod - 1)); + } + else + { + const size_t sw = n.sig_words(); + for(size_t i = sw; i > 0; --i) + { + remainder = bigint_modop(remainder, n.word_at(i-1), mod); + } + } if(remainder && n.sign() == BigInt::Negative) return mod - remainder; |