Skip to content

Commit 08dbb45

Browse files
peterdettmansipa
authored andcommitted
Optimization: use 9x32 representation for recoded bits
1 parent 34c19aa commit 08dbb45

File tree

1 file changed

+18
-10
lines changed

1 file changed

+18
-10
lines changed

src/ecmult_gen_impl.h

+18-10
Original file line numberDiff line numberDiff line change
@@ -34,8 +34,9 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
3434
secp256k1_ge add;
3535
secp256k1_fe neg;
3636
secp256k1_ge_storage adds;
37-
secp256k1_scalar recoded;
38-
int first = 1;
37+
secp256k1_scalar tmp;
38+
uint32_t recoded[9];
39+
int first = 1, i;
3940

4041
memset(&adds, 0, sizeof(adds));
4142

@@ -44,17 +45,24 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
4445
* To blind the scalar used in the computation, we rewrite this to be R = (gn-b)*G + b*G.
4546
*
4647
* Next, we write (gn-b)*G as a sum of values (2*bit_i-1) * 2^i * G, for i=0..COMB_BITS-1.
47-
* The values bit_i can be found as the binary representation of recoded =
48-
* (gn + 2^COMB_BITS - 1 - b)/2 (mod order).
48+
* The values bit_i can be found as the binary representation of
49+
* (gn + 2^COMB_BITS - 1 - b)/2 (mod order), stored in recoded.
4950
*
5051
* The value (2^COMB_BITS - 1 - b) is precomputed as ctx->scalar_offset, and bG is
5152
* precomputed as ctx->final_point_add. Thus recoded can be written as
5253
* recoded = (gn + scalar_offset)/2, and R becomes the sum of (2*bit_i-1)*2^i*G
5354
* values plus final_point_add. */
5455

55-
/* Compute the recoded value as a scalar. */
56-
secp256k1_scalar_add(&recoded, gn, &ctx->scalar_offset);
57-
secp256k1_scalar_half(&recoded, &recoded);
56+
/* Compute (gn + 2^COMB_BITS - 1 - 1)/2 value as a scalar. */
57+
secp256k1_scalar_add(&tmp, gn, &ctx->scalar_offset);
58+
secp256k1_scalar_half(&tmp, &tmp);
59+
/* Convert to 9x32 bit representation (the additional limb avoids the need to deal
60+
* with out-of-bounds reads. */
61+
for (i = 0; i < 8; ++i) {
62+
recoded[i] = secp256k1_scalar_get_bits(&tmp, 32 * i, 32);
63+
}
64+
recoded[8] = 0;
65+
secp256k1_scalar_clear(&tmp);
5866

5967
/* In secp256k1_ecmult_gen_prec_table we have precomputed sums of the
6068
* (2*bit_i-1) * 2^i * G points, for various combinations of i positions.
@@ -122,8 +130,8 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
122130
* together: bit (tooth) of bits = bit
123131
* ((block*COMB_TEETH + tooth)*COMB_SPACING + comb_off) of recoded. */
124132
uint32_t bits = 0, sign, abs, index, tooth;
125-
for (tooth = 0; tooth < COMB_TEETH && bit_pos < 256; ++tooth) {
126-
uint32_t bit = secp256k1_scalar_get_bits(&recoded, bit_pos, 1);
133+
for (tooth = 0; tooth < COMB_TEETH; ++tooth) {
134+
uint32_t bit = (recoded[bit_pos >> 5] >> (bit_pos & 0x1f)) & 1;
127135
bits |= bit << tooth;
128136
bit_pos += COMB_SPACING;
129137
}
@@ -178,7 +186,7 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
178186
secp256k1_fe_clear(&neg);
179187
secp256k1_ge_clear(&add);
180188
memset(&adds, 0, sizeof(adds));
181-
secp256k1_scalar_clear(&recoded);
189+
memset(&recoded, 0, sizeof(recoded));
182190
}
183191

184192
/* Setup blinding values for secp256k1_ecmult_gen. */

0 commit comments

Comments
 (0)