Skip to content

Commit aa15154

Browse files
committedFeb 25, 2019
Merge #568: Fix integer overflow in ecmult_multi_var when n is large
2277af5 Fix integer overflow in ecmult_multi_var when n is large (Jonas Nick) Pull request description: Without this PR ecmult_multi could return wrong results. If the number of points `n` is large enough then some or all multiplications could be skipped or the function could end up in an infinite loop. This PR adds two checks to prevent `n` from wrapping around. Tree-SHA512: 342944369b24776fa3ec0694eee159259ff67e94d2d8176c1d3159875f387d943d5bfdff7cde59f058e13f07fd09bde1cbc609426e63c8a5b8040e382dd865d8
2 parents 85d0e1b + 2277af5 commit aa15154

File tree

2 files changed

+69
-12
lines changed

2 files changed

+69
-12
lines changed
 

‎src/ecmult_impl.h

+24-12
Original file line numberDiff line numberDiff line change
@@ -1111,12 +1111,31 @@ static int secp256k1_ecmult_multi_simple_var(const secp256k1_ecmult_context *ctx
11111111
return 1;
11121112
}
11131113

1114+
/* Compute the number of batches and the batch size given the maximum batch size and the
1115+
* total number of points */
1116+
static int secp256k1_ecmult_multi_batch_size_helper(size_t *n_batches, size_t *n_batch_points, size_t max_n_batch_points, size_t n) {
1117+
if (max_n_batch_points == 0) {
1118+
return 0;
1119+
}
1120+
if (max_n_batch_points > ECMULT_MAX_POINTS_PER_BATCH) {
1121+
max_n_batch_points = ECMULT_MAX_POINTS_PER_BATCH;
1122+
}
1123+
if (n == 0) {
1124+
*n_batches = 0;
1125+
*n_batch_points = 0;
1126+
return 1;
1127+
}
1128+
/* Compute ceil(n/max_n_batch_points) and ceil(n/n_batches) */
1129+
*n_batches = 1 + (n - 1) / max_n_batch_points;
1130+
*n_batch_points = 1 + (n - 1) / *n_batches;
1131+
return 1;
1132+
}
1133+
11141134
typedef int (*secp256k1_ecmult_multi_func)(const secp256k1_ecmult_context*, secp256k1_scratch*, secp256k1_gej*, const secp256k1_scalar*, secp256k1_ecmult_multi_callback cb, void*, size_t);
11151135
static int secp256k1_ecmult_multi_var(const secp256k1_ecmult_context *ctx, secp256k1_scratch *scratch, secp256k1_gej *r, const secp256k1_scalar *inp_g_sc, secp256k1_ecmult_multi_callback cb, void *cbdata, size_t n) {
11161136
size_t i;
11171137

11181138
int (*f)(const secp256k1_ecmult_context*, secp256k1_scratch*, secp256k1_gej*, const secp256k1_scalar*, secp256k1_ecmult_multi_callback cb, void*, size_t, size_t);
1119-
size_t max_points;
11201139
size_t n_batches;
11211140
size_t n_batch_points;
11221141

@@ -1133,24 +1152,17 @@ static int secp256k1_ecmult_multi_var(const secp256k1_ecmult_context *ctx, secp2
11331152
return secp256k1_ecmult_multi_simple_var(ctx, r, inp_g_sc, cb, cbdata, n);
11341153
}
11351154

1136-
max_points = secp256k1_pippenger_max_points(scratch);
1137-
if (max_points == 0) {
1155+
/* Compute the batch sizes for pippenger given a scratch space. If it's greater than a threshold
1156+
* use pippenger. Otherwise use strauss */
1157+
if (!secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, secp256k1_pippenger_max_points(scratch), n)) {
11381158
return 0;
1139-
} else if (max_points > ECMULT_MAX_POINTS_PER_BATCH) {
1140-
max_points = ECMULT_MAX_POINTS_PER_BATCH;
11411159
}
1142-
n_batches = (n+max_points-1)/max_points;
1143-
n_batch_points = (n+n_batches-1)/n_batches;
1144-
11451160
if (n_batch_points >= ECMULT_PIPPENGER_THRESHOLD) {
11461161
f = secp256k1_ecmult_pippenger_batch;
11471162
} else {
1148-
max_points = secp256k1_strauss_max_points(scratch);
1149-
if (max_points == 0) {
1163+
if (!secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, secp256k1_strauss_max_points(scratch), n)) {
11501164
return 0;
11511165
}
1152-
n_batches = (n+max_points-1)/max_points;
1153-
n_batch_points = (n+n_batches-1)/n_batches;
11541166
f = secp256k1_ecmult_strauss_batch;
11551167
}
11561168
for(i = 0; i < n_batches; i++) {

‎src/tests.c

+45
Original file line numberDiff line numberDiff line change
@@ -2854,6 +2854,50 @@ void test_ecmult_multi_pippenger_max_points(void) {
28542854
CHECK(bucket_window == PIPPENGER_MAX_BUCKET_WINDOW);
28552855
}
28562856

2857+
void test_ecmult_multi_batch_size_helper(void) {
2858+
size_t n_batches, n_batch_points, max_n_batch_points, n;
2859+
2860+
max_n_batch_points = 0;
2861+
n = 1;
2862+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 0);
2863+
2864+
max_n_batch_points = 1;
2865+
n = 0;
2866+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1);
2867+
CHECK(n_batches == 0);
2868+
CHECK(n_batch_points == 0);
2869+
2870+
max_n_batch_points = 2;
2871+
n = 5;
2872+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1);
2873+
CHECK(n_batches == 3);
2874+
CHECK(n_batch_points == 2);
2875+
2876+
max_n_batch_points = ECMULT_MAX_POINTS_PER_BATCH;
2877+
n = ECMULT_MAX_POINTS_PER_BATCH;
2878+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1);
2879+
CHECK(n_batches == 1);
2880+
CHECK(n_batch_points == ECMULT_MAX_POINTS_PER_BATCH);
2881+
2882+
max_n_batch_points = ECMULT_MAX_POINTS_PER_BATCH + 1;
2883+
n = ECMULT_MAX_POINTS_PER_BATCH + 1;
2884+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1);
2885+
CHECK(n_batches == 2);
2886+
CHECK(n_batch_points == ECMULT_MAX_POINTS_PER_BATCH/2 + 1);
2887+
2888+
max_n_batch_points = 1;
2889+
n = SIZE_MAX;
2890+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1);
2891+
CHECK(n_batches == SIZE_MAX);
2892+
CHECK(n_batch_points == 1);
2893+
2894+
max_n_batch_points = 2;
2895+
n = SIZE_MAX;
2896+
CHECK(secp256k1_ecmult_multi_batch_size_helper(&n_batches, &n_batch_points, max_n_batch_points, n) == 1);
2897+
CHECK(n_batches == SIZE_MAX/2 + 1);
2898+
CHECK(n_batch_points == 2);
2899+
}
2900+
28572901
/**
28582902
* Run secp256k1_ecmult_multi_var with num points and a scratch space restricted to
28592903
* 1 <= i <= num points.
@@ -2936,6 +2980,7 @@ void run_ecmult_multi_tests(void) {
29362980
test_ecmult_multi(scratch, secp256k1_ecmult_multi_var);
29372981
secp256k1_scratch_destroy(scratch);
29382982

2983+
test_ecmult_multi_batch_size_helper();
29392984
test_ecmult_multi_batching();
29402985
}
29412986

0 commit comments

Comments
 (0)
Please sign in to comment.