@@ -34,8 +34,9 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
34
34
secp256k1_ge add ;
35
35
secp256k1_fe neg ;
36
36
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 ;
39
40
40
41
memset (& adds , 0 , sizeof (adds ));
41
42
@@ -44,17 +45,24 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
44
45
* To blind the scalar used in the computation, we rewrite this to be R = (gn-b)*G + b*G.
45
46
*
46
47
* 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 .
49
50
*
50
51
* The value (2^COMB_BITS - 1 - b) is precomputed as ctx->scalar_offset, and bG is
51
52
* precomputed as ctx->final_point_add. Thus recoded can be written as
52
53
* recoded = (gn + scalar_offset)/2, and R becomes the sum of (2*bit_i-1)*2^i*G
53
54
* values plus final_point_add. */
54
55
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 );
58
66
59
67
/* In secp256k1_ecmult_gen_prec_table we have precomputed sums of the
60
68
* (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
122
130
* together: bit (tooth) of bits = bit
123
131
* ((block*COMB_TEETH + tooth)*COMB_SPACING + comb_off) of recoded. */
124
132
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 ;
127
135
bits |= bit << tooth ;
128
136
bit_pos += COMB_SPACING ;
129
137
}
@@ -178,7 +186,7 @@ static void secp256k1_ecmult_gen(const secp256k1_ecmult_gen_context *ctx, secp25
178
186
secp256k1_fe_clear (& neg );
179
187
secp256k1_ge_clear (& add );
180
188
memset (& adds , 0 , sizeof (adds ));
181
- secp256k1_scalar_clear (& recoded );
189
+ memset (& recoded , 0 , sizeof ( recoded ) );
182
190
}
183
191
184
192
/* Setup blinding values for secp256k1_ecmult_gen. */
0 commit comments