8 #include <botan/curve_nistp.h>
9 #include <botan/internal/mp_core.h>
10 #include <botan/internal/mp_asmi.h>
11 #include <botan/internal/ct_utils.h>
17 static const BigInt p521(
"0x1FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF"
18 "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF");
25 const size_t p_full_words = 521 / BOTAN_MP_WORD_BITS;
26 const size_t p_top_bits = 521 % BOTAN_MP_WORD_BITS;
27 const size_t p_words = p_full_words + 1;
29 #if (BOTAN_MP_WORD_BITS == 64)
30 static const word p521_words[p_words] = {
31 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF,
32 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF,
35 static const word p521_words[p_words] = {
36 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF,
37 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF,
38 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF,
39 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF,
43 if(ws.size() < p_words + 1)
44 ws.resize(p_words + 1);
56 const word top_word = x.
word_at(p_full_words);
67 for(
size_t i = 0; i != p_full_words; ++i)
71 const auto is_p521 = all_512_low_bits_set & has_p521_top_word;
73 const auto needs_reduction = is_p521 | bit_522_set;
84 inline uint32_t get_uint32(
const BigInt& x,
size_t i)
86 #if (BOTAN_MP_WORD_BITS == 32)
89 return static_cast<uint32_t
>(x.word_at(i/2) >> ((i % 2)*32));
93 inline void set_words(word x[],
size_t i, uint32_t R0, uint32_t R1)
95 #if (BOTAN_MP_WORD_BITS == 32)
99 x[i/2] = (
static_cast<uint64_t
>(R1) << 32) | R0;
107 static const BigInt p192(
"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF");
115 static const size_t p192_limbs = 192 / BOTAN_MP_WORD_BITS;
117 const uint64_t X00 = get_uint32(x, 0);
118 const uint64_t X01 = get_uint32(x, 1);
119 const uint64_t X02 = get_uint32(x, 2);
120 const uint64_t X03 = get_uint32(x, 3);
121 const uint64_t X04 = get_uint32(x, 4);
122 const uint64_t X05 = get_uint32(x, 5);
123 const uint64_t X06 = get_uint32(x, 6);
124 const uint64_t X07 = get_uint32(x, 7);
125 const uint64_t X08 = get_uint32(x, 8);
126 const uint64_t X09 = get_uint32(x, 9);
127 const uint64_t X10 = get_uint32(x, 10);
128 const uint64_t X11 = get_uint32(x, 11);
130 const uint64_t S0 = X00 + X06 + X10;
131 const uint64_t S1 = X01 + X07 + X11;
132 const uint64_t S2 = X02 + X06 + X08 + X10;
133 const uint64_t S3 = X03 + X07 + X09 + X11;
134 const uint64_t S4 = X04 + X08 + X10;
135 const uint64_t S5 = X05 + X09 + X11;
143 uint32_t R0 = 0, R1 = 0;
146 R0 =
static_cast<uint32_t
>(S);
150 R1 =
static_cast<uint32_t
>(S);
153 set_words(xw, 0, R0, R1);
156 R0 =
static_cast<uint32_t
>(S);
160 R1 =
static_cast<uint32_t
>(S);
163 set_words(xw, 2, R0, R1);
166 R0 =
static_cast<uint32_t
>(S);
170 R1 =
static_cast<uint32_t
>(S);
173 set_words(xw, 4, R0, R1);
180 static const word p192_mults[3][p192_limbs] = {
181 #if (BOTAN_MP_WORD_BITS == 64)
182 {0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFE, 0xFFFFFFFFFFFFFFFF},
183 {0xFFFFFFFFFFFFFFFE, 0xFFFFFFFFFFFFFFFD, 0xFFFFFFFFFFFFFFFF},
184 {0xFFFFFFFFFFFFFFFD, 0xFFFFFFFFFFFFFFFC, 0xFFFFFFFFFFFFFFFF},
186 {0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFE, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
187 {0xFFFFFFFE, 0xFFFFFFFF, 0xFFFFFFFD, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
188 {0xFFFFFFFD, 0xFFFFFFFF, 0xFFFFFFFC, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
203 static const BigInt p224(
"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF000000000000000000000001");
209 static const size_t p224_limbs = (BOTAN_MP_WORD_BITS == 32) ? 7 : 4;
213 const int64_t X00 = get_uint32(x, 0);
214 const int64_t X01 = get_uint32(x, 1);
215 const int64_t X02 = get_uint32(x, 2);
216 const int64_t X03 = get_uint32(x, 3);
217 const int64_t X04 = get_uint32(x, 4);
218 const int64_t X05 = get_uint32(x, 5);
219 const int64_t X06 = get_uint32(x, 6);
220 const int64_t X07 = get_uint32(x, 7);
221 const int64_t X08 = get_uint32(x, 8);
222 const int64_t X09 = get_uint32(x, 9);
223 const int64_t X10 = get_uint32(x, 10);
224 const int64_t X11 = get_uint32(x, 11);
225 const int64_t X12 = get_uint32(x, 12);
226 const int64_t X13 = get_uint32(x, 13);
230 const int64_t S0 = 0x00000001 + X00 - X07 - X11;
231 const int64_t S1 = 0x00000000 + X01 - X08 - X12;
232 const int64_t S2 = 0x00000000 + X02 - X09 - X13;
233 const int64_t S3 = 0xFFFFFFFF + X03 + X07 + X11 - X10;
234 const int64_t S4 = 0xFFFFFFFF + X04 + X08 + X12 - X11;
235 const int64_t S5 = 0xFFFFFFFF + X05 + X09 + X13 - X12;
236 const int64_t S6 = 0xFFFFFFFF + X06 + X10 - X13;
244 uint32_t R0 = 0, R1 = 0;
247 R0 =
static_cast<uint32_t
>(S);
251 R1 =
static_cast<uint32_t
>(S);
254 set_words(xw, 0, R0, R1);
257 R0 =
static_cast<uint32_t
>(S);
261 R1 =
static_cast<uint32_t
>(S);
264 set_words(xw, 2, R0, R1);
267 R0 =
static_cast<uint32_t
>(S);
271 R1 =
static_cast<uint32_t
>(S);
274 set_words(xw, 4, R0, R1);
277 R0 =
static_cast<uint32_t
>(S);
280 set_words(xw, 6, R0, 0);
282 static const word p224_mults[3][p224_limbs] = {
283 #if (BOTAN_MP_WORD_BITS == 64)
284 {0x0000000000000001, 0xFFFFFFFF00000000, 0xFFFFFFFFFFFFFFFF, 0x00000000FFFFFFFF},
285 {0x0000000000000002, 0xFFFFFFFE00000000, 0xFFFFFFFFFFFFFFFF, 0x00000000FFFFFFFF},
286 {0x0000000000000003, 0xFFFFFFFD00000000, 0xFFFFFFFFFFFFFFFF, 0x00000000FFFFFFFF},
288 {0x00000001, 0x00000000, 0x00000000, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
289 {0x00000002, 0x00000000, 0x00000000, 0xFFFFFFFE, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
290 {0x00000003, 0x00000000, 0x00000000, 0xFFFFFFFD, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF}
306 static const BigInt p256(
"0xFFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFF");
312 static const size_t p256_limbs = (BOTAN_MP_WORD_BITS == 32) ? 8 : 4;
316 const int64_t X00 = get_uint32(x, 0);
317 const int64_t X01 = get_uint32(x, 1);
318 const int64_t X02 = get_uint32(x, 2);
319 const int64_t X03 = get_uint32(x, 3);
320 const int64_t X04 = get_uint32(x, 4);
321 const int64_t X05 = get_uint32(x, 5);
322 const int64_t X06 = get_uint32(x, 6);
323 const int64_t X07 = get_uint32(x, 7);
324 const int64_t X08 = get_uint32(x, 8);
325 const int64_t X09 = get_uint32(x, 9);
326 const int64_t X10 = get_uint32(x, 10);
327 const int64_t X11 = get_uint32(x, 11);
328 const int64_t X12 = get_uint32(x, 12);
329 const int64_t X13 = get_uint32(x, 13);
330 const int64_t X14 = get_uint32(x, 14);
331 const int64_t X15 = get_uint32(x, 15);
334 const int64_t S0 = 0xFFFFFFFA + X00 + X08 + X09 - (X11 + X12 + X13) - X14;
335 const int64_t S1 = 0xFFFFFFFF + X01 + X09 + X10 - X12 - (X13 + X14 + X15);
336 const int64_t S2 = 0xFFFFFFFF + X02 + X10 + X11 - (X13 + X14 + X15);
337 const int64_t S3 = 0x00000005 + X03 + (X11 + X12)*2 + X13 - X15 - X08 - X09;
338 const int64_t S4 = 0x00000000 + X04 + (X12 + X13)*2 + X14 - X09 - X10;
339 const int64_t S5 = 0x00000000 + X05 + (X13 + X14)*2 + X15 - X10 - X11;
340 const int64_t S6 = 0x00000006 + X06 + X13 + X14*3 + X15*2 - X08 - X09;
341 const int64_t S7 = 0xFFFFFFFA + X07 + X15*3 + X08 - X10 - (X11 + X12 + X13);
350 uint32_t R0 = 0, R1 = 0;
353 R0 =
static_cast<uint32_t
>(S);
357 R1 =
static_cast<uint32_t
>(S);
360 set_words(xw, 0, R0, R1);
363 R0 =
static_cast<uint32_t
>(S);
367 R1 =
static_cast<uint32_t
>(S);
370 set_words(xw, 2, R0, R1);
373 R0 =
static_cast<uint32_t
>(S);
377 R1 =
static_cast<uint32_t
>(S);
380 set_words(xw, 4, R0, R1);
383 R0 =
static_cast<uint32_t
>(S);
387 R1 =
static_cast<uint32_t
>(S);
389 set_words(xw, 6, R0, R1);
396 static const word p256_mults[11][p256_limbs] = {
397 #if (BOTAN_MP_WORD_BITS == 64)
398 {0xFFFFFFFFFFFFFFFF, 0x00000000FFFFFFFF, 0x0000000000000000, 0xFFFFFFFF00000001},
399 {0xFFFFFFFFFFFFFFFE, 0x00000001FFFFFFFF, 0x0000000000000000, 0xFFFFFFFE00000002},
400 {0xFFFFFFFFFFFFFFFD, 0x00000002FFFFFFFF, 0x0000000000000000, 0xFFFFFFFD00000003},
401 {0xFFFFFFFFFFFFFFFC, 0x00000003FFFFFFFF, 0x0000000000000000, 0xFFFFFFFC00000004},
402 {0xFFFFFFFFFFFFFFFB, 0x00000004FFFFFFFF, 0x0000000000000000, 0xFFFFFFFB00000005},
403 {0xFFFFFFFFFFFFFFFA, 0x00000005FFFFFFFF, 0x0000000000000000, 0xFFFFFFFA00000006},
404 {0xFFFFFFFFFFFFFFF9, 0x00000006FFFFFFFF, 0x0000000000000000, 0xFFFFFFF900000007},
405 {0xFFFFFFFFFFFFFFF8, 0x00000007FFFFFFFF, 0x0000000000000000, 0xFFFFFFF800000008},
406 {0xFFFFFFFFFFFFFFF7, 0x00000008FFFFFFFF, 0x0000000000000000, 0xFFFFFFF700000009},
407 {0xFFFFFFFFFFFFFFF6, 0x00000009FFFFFFFF, 0x0000000000000000, 0xFFFFFFF60000000A},
408 {0xFFFFFFFFFFFFFFF5, 0x0000000AFFFFFFFF, 0x0000000000000000, 0xFFFFFFF50000000B},
410 {0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000000, 0x00000000, 0x00000000, 0x00000001, 0xFFFFFFFF},
411 {0xFFFFFFFE, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000001, 0x00000000, 0x00000000, 0x00000002, 0xFFFFFFFE},
412 {0xFFFFFFFD, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000002, 0x00000000, 0x00000000, 0x00000003, 0xFFFFFFFD},
413 {0xFFFFFFFC, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000003, 0x00000000, 0x00000000, 0x00000004, 0xFFFFFFFC},
414 {0xFFFFFFFB, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000004, 0x00000000, 0x00000000, 0x00000005, 0xFFFFFFFB},
415 {0xFFFFFFFA, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000005, 0x00000000, 0x00000000, 0x00000006, 0xFFFFFFFA},
416 {0xFFFFFFF9, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000006, 0x00000000, 0x00000000, 0x00000007, 0xFFFFFFF9},
417 {0xFFFFFFF8, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000007, 0x00000000, 0x00000000, 0x00000008, 0xFFFFFFF8},
418 {0xFFFFFFF7, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000008, 0x00000000, 0x00000000, 0x00000009, 0xFFFFFFF7},
419 {0xFFFFFFF6, 0xFFFFFFFF, 0xFFFFFFFF, 0x00000009, 0x00000000, 0x00000000, 0x0000000A, 0xFFFFFFF6},
420 {0xFFFFFFF5, 0xFFFFFFFF, 0xFFFFFFFF, 0x0000000A, 0x00000000, 0x00000000, 0x0000000B, 0xFFFFFFF5},
435 static const BigInt p384(
"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFF0000000000000000FFFFFFFF");
443 static const size_t p384_limbs = (BOTAN_MP_WORD_BITS == 32) ? 12 : 6;
445 const int64_t X00 = get_uint32(x, 0);
446 const int64_t X01 = get_uint32(x, 1);
447 const int64_t X02 = get_uint32(x, 2);
448 const int64_t X03 = get_uint32(x, 3);
449 const int64_t X04 = get_uint32(x, 4);
450 const int64_t X05 = get_uint32(x, 5);
451 const int64_t X06 = get_uint32(x, 6);
452 const int64_t X07 = get_uint32(x, 7);
453 const int64_t X08 = get_uint32(x, 8);
454 const int64_t X09 = get_uint32(x, 9);
455 const int64_t X10 = get_uint32(x, 10);
456 const int64_t X11 = get_uint32(x, 11);
457 const int64_t X12 = get_uint32(x, 12);
458 const int64_t X13 = get_uint32(x, 13);
459 const int64_t X14 = get_uint32(x, 14);
460 const int64_t X15 = get_uint32(x, 15);
461 const int64_t X16 = get_uint32(x, 16);
462 const int64_t X17 = get_uint32(x, 17);
463 const int64_t X18 = get_uint32(x, 18);
464 const int64_t X19 = get_uint32(x, 19);
465 const int64_t X20 = get_uint32(x, 20);
466 const int64_t X21 = get_uint32(x, 21);
467 const int64_t X22 = get_uint32(x, 22);
468 const int64_t X23 = get_uint32(x, 23);
471 const int64_t S0 = 0xFFFFFFFF + X00 + X12 + X20 + X21 - X23;
472 const int64_t S1 = 0x00000000 + X01 + X13 + X22 + X23 - X12 - X20;
473 const int64_t S2 = 0x00000000 + X02 + X14 + X23 - X13 - X21;
474 const int64_t S3 = 0xFFFFFFFF + X03 + X12 + X15 + X20 + X21 - X14 - X22 - X23;
475 const int64_t S4 = 0xFFFFFFFE + X04 + X12 + X13 + X16 + X20 + X21*2 + X22 - X15 - X23*2;
476 const int64_t S5 = 0xFFFFFFFF + X05 + X13 + X14 + X17 + X21 + X22*2 + X23 - X16;
477 const int64_t S6 = 0xFFFFFFFF + X06 + X14 + X15 + X18 + X22 + X23*2 - X17;
478 const int64_t S7 = 0xFFFFFFFF + X07 + X15 + X16 + X19 + X23 - X18;
479 const int64_t S8 = 0xFFFFFFFF + X08 + X16 + X17 + X20 - X19;
480 const int64_t S9 = 0xFFFFFFFF + X09 + X17 + X18 + X21 - X20;
481 const int64_t SA = 0xFFFFFFFF + X10 + X18 + X19 + X22 - X21;
482 const int64_t SB = 0xFFFFFFFF + X11 + X19 + X20 + X23 - X22;
491 uint32_t R0 = 0, R1 = 0;
494 R0 =
static_cast<uint32_t
>(S);
498 R1 =
static_cast<uint32_t
>(S);
501 set_words(xw, 0, R0, R1);
504 R0 =
static_cast<uint32_t
>(S);
508 R1 =
static_cast<uint32_t
>(S);
511 set_words(xw, 2, R0, R1);
514 R0 =
static_cast<uint32_t
>(S);
518 R1 =
static_cast<uint32_t
>(S);
521 set_words(xw, 4, R0, R1);
524 R0 =
static_cast<uint32_t
>(S);
528 R1 =
static_cast<uint32_t
>(S);
531 set_words(xw, 6, R0, R1);
534 R0 =
static_cast<uint32_t
>(S);
538 R1 =
static_cast<uint32_t
>(S);
541 set_words(xw, 8, R0, R1);
544 R0 =
static_cast<uint32_t
>(S);
548 R1 =
static_cast<uint32_t
>(S);
551 set_words(xw, 10, R0, R1);
556 static const word p384_mults[5][p384_limbs] = {
557 #if (BOTAN_MP_WORD_BITS == 64)
558 {0x00000000FFFFFFFF, 0xFFFFFFFF00000000, 0xFFFFFFFFFFFFFFFE, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF},
559 {0x00000001FFFFFFFE, 0xFFFFFFFE00000000, 0xFFFFFFFFFFFFFFFD, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF},
560 {0x00000002FFFFFFFD, 0xFFFFFFFD00000000, 0xFFFFFFFFFFFFFFFC, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF},
561 {0x00000003FFFFFFFC, 0xFFFFFFFC00000000, 0xFFFFFFFFFFFFFFFB, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF},
562 {0x00000004FFFFFFFB, 0xFFFFFFFB00000000, 0xFFFFFFFFFFFFFFFA, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF, 0xFFFFFFFFFFFFFFFF},
565 {0xFFFFFFFF, 0x00000000, 0x00000000, 0xFFFFFFFF, 0xFFFFFFFE, 0xFFFFFFFF,
566 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
567 {0xFFFFFFFE, 0x00000001, 0x00000000, 0xFFFFFFFE, 0xFFFFFFFD, 0xFFFFFFFF,
568 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
569 {0xFFFFFFFD, 0x00000002, 0x00000000, 0xFFFFFFFD, 0xFFFFFFFC, 0xFFFFFFFF,
570 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
571 {0xFFFFFFFC, 0x00000003, 0x00000000, 0xFFFFFFFC, 0xFFFFFFFB, 0xFFFFFFFF,
572 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
573 {0xFFFFFFFB, 0x00000004, 0x00000000, 0xFFFFFFFB, 0xFFFFFFFA, 0xFFFFFFFF,
574 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF},
word word_at(size_t n) const
void bigint_shr2(word y[], const word x[], size_t x_size, size_t word_shift, size_t bit_shift)
void carry(int64_t &h0, int64_t &h1)
void redc_p256(BigInt &x, secure_vector< word > &ws)
void clear_mem(T *ptr, size_t n)
word bigint_sub2(word x[], size_t x_size, const word y[], size_t y_size)
const BigInt & prime_p256()
word bigint_add3_nc(word z[], const word x[], size_t x_size, const word y[], size_t y_size)
void redc_p384(BigInt &x, secure_vector< word > &ws)
#define BOTAN_ASSERT_NOMSG(expr)
const BigInt & prime_p224()
const BigInt & prime_p192()
#define BOTAN_ASSERT(expr, assertion_made)
static Mask< T > expand(T v)
word bigint_cnd_sub(word cnd, word x[], size_t x_size, const word y[], size_t y_size)
std::vector< T, secure_allocator< T >> secure_vector
#define BOTAN_DEBUG_ASSERT(expr)
#define BOTAN_ASSERT_EQUAL(expr1, expr2, assertion_made)
word bigint_cnd_add(word cnd, word x[], word x_size, const word y[], size_t y_size)
const word * data() const
void redc_p521(BigInt &x, secure_vector< word > &ws)
#define BOTAN_UNUSED(...)
void redc_p224(BigInt &x, secure_vector< word > &ws)
const BigInt & prime_p521()
void grow_to(size_t n) const
const BigInt & prime_p384()
void unpoison(const T *p, size_t n)
void redc_p192(BigInt &x, secure_vector< word > &ws)
static Mask< T > is_equal(T x, T y)