aboutsummaryrefslogtreecommitdiffstats
path: root/src/lib/math
diff options
context:
space:
mode:
authorJack Lloyd <[email protected]>2018-12-05 08:32:26 -0500
committerJack Lloyd <[email protected]>2018-12-05 08:32:26 -0500
commit340ee4f3e36ec37baa9748ad7107d90050b8af20 (patch)
tree89a74a624316f38596365cfd8ac3364a231b1b58 /src/lib/math
parent1b8163a7c465cf08f43f2b93db9c64dfb1ced901 (diff)
Remove some conditional branches from division
Diffstat (limited to 'src/lib/math')
-rw-r--r--src/lib/math/bigint/bigint.cpp7
-rw-r--r--src/lib/math/bigint/bigint.h2
-rw-r--r--src/lib/math/bigint/divide.cpp40
3 files changed, 27 insertions, 22 deletions
diff --git a/src/lib/math/bigint/bigint.cpp b/src/lib/math/bigint/bigint.cpp
index a4545e4a1..1a09a92f1 100644
--- a/src/lib/math/bigint/bigint.cpp
+++ b/src/lib/math/bigint/bigint.cpp
@@ -317,7 +317,7 @@ BigInt BigInt::operator-() const
return x;
}
-void BigInt::reduce_below(const BigInt& p, secure_vector<word>& ws)
+size_t BigInt::reduce_below(const BigInt& p, secure_vector<word>& ws)
{
if(p.is_negative())
throw Invalid_Argument("BigInt::reduce_below mod must be positive");
@@ -332,14 +332,19 @@ void BigInt::reduce_below(const BigInt& p, secure_vector<word>& ws)
clear_mem(ws.data(), ws.size());
+ size_t reductions = 0;
+
for(;;)
{
word borrow = bigint_sub3(ws.data(), data(), p_words + 1, p.data(), p_words);
if(borrow)
break;
+ ++reductions;
swap_reg(ws);
}
+
+ return reductions;
}
/*
diff --git a/src/lib/math/bigint/bigint.h b/src/lib/math/bigint/bigint.h
index f0aa04391..9b7af4169 100644
--- a/src/lib/math/bigint/bigint.h
+++ b/src/lib/math/bigint/bigint.h
@@ -344,7 +344,7 @@ class BOTAN_PUBLIC_API(2,0) BigInt final
* mod and performs repeated subtractions. It should not be used if
* *this is much larger than mod, instead use modulo operator.
*/
- void reduce_below(const BigInt& mod, secure_vector<word> &ws);
+ size_t reduce_below(const BigInt& mod, secure_vector<word> &ws);
/**
* Zeroize the BigInt. The size of the underlying register is not
diff --git a/src/lib/math/bigint/divide.cpp b/src/lib/math/bigint/divide.cpp
index 2dfb7405e..ed9a12112 100644
--- a/src/lib/math/bigint/divide.cpp
+++ b/src/lib/math/bigint/divide.cpp
@@ -154,6 +154,8 @@ BigInt ct_modulo(const BigInt& x, const BigInt& y)
/*
* Solve x = q * y + r
+*
+* See Handbook of Applied Cryptography section 14.2.5
*/
void divide(const BigInt& x, const BigInt& y_arg, BigInt& q_out, BigInt& r_out)
{
@@ -166,11 +168,12 @@ void divide(const BigInt& x, const BigInt& y_arg, BigInt& q_out, BigInt& r_out)
BigInt r = x;
BigInt q = 0;
+ secure_vector<word> ws;
r.set_sign(BigInt::Positive);
y.set_sign(BigInt::Positive);
- if(r >= y)
+ if(r.is_nonzero())
{
// Calculate shifts needed to normalize y with high bit set
const size_t shifts = BOTAN_MP_WORD_BITS - high_bit(y.word_at(y_words-1));
@@ -180,7 +183,7 @@ void divide(const BigInt& x, const BigInt& y_arg, BigInt& q_out, BigInt& r_out)
// we know y has not changed size, since we only shifted up to set high bit
const size_t t = y_words - 1;
- const size_t n = r.sig_words() - 1; // r may have changed size however
+ const size_t n = std::max(t, r.sig_words() - 1); // r may have changed size however
BOTAN_ASSERT_NOMSG(n >= t);
@@ -190,38 +193,35 @@ void divide(const BigInt& x, const BigInt& y_arg, BigInt& q_out, BigInt& r_out)
BigInt shifted_y = y << (BOTAN_MP_WORD_BITS * (n-t));
- while(r >= shifted_y)
- {
- r -= shifted_y;
- q_words[n-t] += 1;
- }
+ // Set q_{n-t} to number of times r > shifted_y
+ q_words[n-t] = r.reduce_below(shifted_y, ws);
+
+ const word y_t0 = y.word_at(t);
+ const word y_t1 = y.word_at(t-1);
for(size_t j = n; j != t; --j)
{
const word x_j0 = r.word_at(j);
const word x_j1 = r.word_at(j-1);
const word x_j2 = r.word_at(j-2);
- const word y_t0 = y.word_at(t);
- const word y_t1 = y.word_at(t-1);
- word qjt = (x_j0 == y_t0) ? MP_WORD_MAX : bigint_divop(x_j0, x_j1, y_t0);
+ word qjt = bigint_divop(x_j0, x_j1, y_t0);
+
+ qjt = CT::Mask<word>::is_equal(x_j0, y_t0).select(MP_WORD_MAX, qjt);
- while(division_check(qjt, y_t0, y_t1, x_j0, x_j1, x_j2))
- {
- qjt -= 1;
- }
+ // Per HAC 14.23, this operation is required at most twice
+ qjt -= division_check(qjt, y_t0, y_t1, x_j0, x_j1, x_j2);
+ qjt -= division_check(qjt, y_t0, y_t1, x_j0, x_j1, x_j2);
+ BOTAN_DEBUG_ASSERT(division_check(qjt, y_t0, y_t1, x_j0, x_j1, x_j2) == false);
shifted_y >>= BOTAN_MP_WORD_BITS;
// Now shifted_y == y << (BOTAN_MP_WORD_BITS * (j-t-1))
r -= qjt * shifted_y;
- if(r.is_negative())
- {
- // overcorrected
- qjt -= 1;
- r += shifted_y;
- }
+ // TODO this could be better
+ qjt -= r.is_negative();
+ r += static_cast<word>(r.is_negative()) * shifted_y;
q_words[j-t-1] = qjt;
}