@@ -27,7 +27,7 @@ void sign_fixup(const BigInt& x, const BigInt& y, BigInt& q, BigInt& r) {
27
27
}
28
28
}
29
29
30
- inline bool division_check (word q, word y2, word y1, word x3, word x2, word x1) {
30
+ inline bool division_check_vartime (word q, word y2, word y1, word x3, word x2, word x1) {
31
31
/*
32
32
Compute (y3,y2,y1) = (y2,y1) * q
33
33
and return true if (y3,y2,y1) > (x3,x2,x1)
@@ -37,10 +37,13 @@ inline bool division_check(word q, word y2, word y1, word x3, word x2, word x1)
37
37
y1 = word_madd2 (q, y1 , &y3);
38
38
y2 = word_madd2 (q, y2, &y3);
39
39
40
- const word x[3 ] = {x1, x2, x3};
41
- const word y[3 ] = {y1 , y2, y3};
42
-
43
- return bigint_ct_is_lt (x, 3 , y, 3 ).as_bool ();
40
+ if (x3 != y3) {
41
+ return (y3 > x3);
42
+ }
43
+ if (x2 != y2) {
44
+ return (y2 > x2);
45
+ }
46
+ return (y1 > x1);
44
47
}
45
48
46
49
} // namespace
@@ -245,8 +248,10 @@ void vartime_divide(const BigInt& x, const BigInt& y_arg, BigInt& q_out, BigInt&
245
248
// Calculate shifts needed to normalize y with high bit set
246
249
const size_t shifts = y.top_bits_free ();
247
250
248
- y <<= shifts;
249
- r <<= shifts;
251
+ if (shifts > 0 ) {
252
+ y <<= shifts;
253
+ r <<= shifts;
254
+ }
250
255
251
256
// we know y has not changed size, since we only shifted up to set high bit
252
257
const size_t t = y_words - 1 ;
@@ -272,27 +277,30 @@ void vartime_divide(const BigInt& x, const BigInt& y_arg, BigInt& q_out, BigInt&
272
277
const word x_j1 = r.word_at (j - 1 );
273
278
const word x_j2 = r.word_at (j - 2 );
274
279
275
- word qjt = bigint_divop_vartime (x_j0, x_j1, y_t0);
276
-
277
- qjt = CT::Mask<word>::is_equal (x_j0, y_t0).select (WordInfo<word>::max, qjt);
280
+ word qjt = (x_j0 == y_t0) ? WordInfo<word>::max : bigint_divop_vartime (x_j0, x_j1, y_t0);
278
281
279
282
// Per HAC 14.23, this operation is required at most twice
280
- qjt -= division_check ( qjt, y_t0, y_t1, x_j0, x_j1, x_j2);
281
- qjt -= division_check (qjt, y_t0, y_t1, x_j0, x_j1, x_j2) ;
282
- BOTAN_DEBUG_ASSERT ( division_check (qjt, y_t0, y_t1, x_j0, x_j1, x_j2) == false );
283
+ while ( division_check_vartime ( qjt, y_t0, y_t1, x_j0, x_j1, x_j2)) {
284
+ qjt-- ;
285
+ }
283
286
284
287
shifted_y >>= WordInfo<word>::bits;
285
288
// Now shifted_y == y << (WordInfo<word>::bits * (j-t-1))
286
289
287
290
// TODO this sequence could be better
288
291
r -= qjt * shifted_y;
289
- qjt -= r.is_negative ();
290
- r += static_cast <word>(r.is_negative ()) * shifted_y;
292
+ if (r.is_negative ()) {
293
+ qjt--;
294
+ r += shifted_y;
295
+ BOTAN_DEBUG_ASSERT (r.is_positive ());
296
+ }
291
297
292
298
q_words[j - t - 1 ] = qjt;
293
299
}
294
300
295
- r >>= shifts;
301
+ if (shifts > 0 ) {
302
+ r >>= shifts;
303
+ }
296
304
297
305
sign_fixup (x, y_arg, q, r);
298
306
0 commit comments