|
|
|
@ -219,6 +219,44 @@ static int secp256k1_fe_normalizes_to_zero(secp256k1_fe_t *r) {
|
|
|
|
|
return (z0 == 0) | (z1 == 0x3FFFFFFUL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int secp256k1_fe_normalizes_to_zero_var(secp256k1_fe_t *r) {
|
|
|
|
|
uint32_t t0 = r->n[0], t9 = r->n[9];
|
|
|
|
|
|
|
|
|
|
/* Reduce t9 at the start so there will be at most a single carry from the first pass */
|
|
|
|
|
uint32_t x = t9 >> 22;
|
|
|
|
|
|
|
|
|
|
/* The first pass ensures the magnitude is 1, ... */
|
|
|
|
|
t0 += x * 0x3D1UL;
|
|
|
|
|
|
|
|
|
|
/* z0 tracks a possible raw value of 0, z1 tracks a possible raw value of P */
|
|
|
|
|
uint32_t z0 = t0 & 0x3FFFFFFUL, z1 = z0 ^ 0x3D0UL;
|
|
|
|
|
|
|
|
|
|
/* Fast return path should catch the majority of cases */
|
|
|
|
|
if ((z0 != 0UL) & (z1 != 0x3FFFFFFUL))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
uint32_t t1 = r->n[1], t2 = r->n[2], t3 = r->n[3], t4 = r->n[4],
|
|
|
|
|
t5 = r->n[5], t6 = r->n[6], t7 = r->n[7], t8 = r->n[8];
|
|
|
|
|
t9 &= 0x03FFFFFUL;
|
|
|
|
|
t1 += (x << 6);
|
|
|
|
|
|
|
|
|
|
t1 += (t0 >> 26); t0 = z0;
|
|
|
|
|
t2 += (t1 >> 26); t1 &= 0x3FFFFFFUL; z0 |= t1; z1 &= t1 ^ 0x40UL;
|
|
|
|
|
t3 += (t2 >> 26); t2 &= 0x3FFFFFFUL; z0 |= t2; z1 &= t2;
|
|
|
|
|
t4 += (t3 >> 26); t3 &= 0x3FFFFFFUL; z0 |= t3; z1 &= t3;
|
|
|
|
|
t5 += (t4 >> 26); t4 &= 0x3FFFFFFUL; z0 |= t4; z1 &= t4;
|
|
|
|
|
t6 += (t5 >> 26); t5 &= 0x3FFFFFFUL; z0 |= t5; z1 &= t5;
|
|
|
|
|
t7 += (t6 >> 26); t6 &= 0x3FFFFFFUL; z0 |= t6; z1 &= t6;
|
|
|
|
|
t8 += (t7 >> 26); t7 &= 0x3FFFFFFUL; z0 |= t7; z1 &= t7;
|
|
|
|
|
t9 += (t8 >> 26); t8 &= 0x3FFFFFFUL; z0 |= t8; z1 &= t8;
|
|
|
|
|
z0 |= t9; z1 &= t9 ^ 0x3C00000UL;
|
|
|
|
|
|
|
|
|
|
/* ... except for a possible carry at bit 22 of t9 (i.e. bit 256 of the field element) */
|
|
|
|
|
VERIFY_CHECK(t9 >> 23 == 0);
|
|
|
|
|
|
|
|
|
|
return (z0 == 0) | (z1 == 0x3FFFFFFUL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SECP256K1_INLINE static void secp256k1_fe_set_int(secp256k1_fe_t *r, int a) {
|
|
|
|
|
r->n[0] = a;
|
|
|
|
|
r->n[1] = r->n[2] = r->n[3] = r->n[4] = r->n[5] = r->n[6] = r->n[7] = r->n[8] = r->n[9] = 0;
|
|
|
|
|