aboutsummaryrefslogtreecommitdiffstats
path: root/src/lib/math/bigint
diff options
context:
space:
mode:
authorJack Lloyd <[email protected]>2018-12-01 13:06:31 -0500
committerJack Lloyd <[email protected]>2018-12-01 13:06:31 -0500
commitbe796a3f30d4bec1af7c0e6770a0397dca6ccd4a (patch)
tree0b2607f4193a899c4012e3072f4a71e9d17128e4 /src/lib/math/bigint
parent5cc1b897edf52a2037004bae28291c67c966c320 (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/bigint')
-rw-r--r--src/lib/math/bigint/big_ops2.cpp30
-rw-r--r--src/lib/math/bigint/big_ops3.cpp17
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;