Skip to content

Commit 8bcf660

Browse files
authoredFeb 13, 2025··
Certificate benchmark as a function of number of voters (#192)
* Added committee-size benchmarks for certificate operations * Documented certificate benchmarks as a function of committee size * Updated logbook * Relocated function for realistic stake distribution * Removed lint
1 parent d9af90d commit 8bcf660

File tree

6 files changed

+232
-43
lines changed

6 files changed

+232
-43
lines changed
 

‎Logbook.md

+19
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,24 @@
11
# Leios logbook
22

3+
## 2025-02-13
4+
5+
### Certificate CPU benchmarks as a function of number of voters
6+
7+
In support of the Haskell and Rust simulations, we've benchmarked certificate operations as a function of the number of voters. A realistic distribution of stake is used in these measurements.
8+
9+
| Number of pools | Number of committee seats | Generate certificate | Verify certificate | Weigh certificate |
10+
|----------------:|--------------------------:|---------------------:|-------------------:|------------------:|
11+
| 2500 | 500 | 63.4 ms | 104.8 ms | 10.6 ms |
12+
| 2500 | 600 | 71.1 ms | 116.9 ms | 12.0 ms |
13+
| 2500 | 700 | 77.4 ms | 125.5 ms | 12.3 ms |
14+
| 2500 | 800 | 83.5 ms | 134.4 ms | 12.8 ms |
15+
| 2500 | 900 | 88.2 ms | 141.1 ms | 12.4 ms |
16+
| 2500 | 1000 | 92.5 ms | 144.9 ms | 12.3 ms |
17+
18+
Serialization and deserialization likely also exhibit the same trend.
19+
20+
A recipe for parallelizing parts of the certificate operations has been added to the [Specification for BLS certificates](crypto-benchmarks.rs/Specification.md).
21+
322
## 2025-02-12
423

524
### Added BLS crypto to CI

‎crypto-benchmarks.rs/Specification.md

+11
Original file line numberDiff line numberDiff line change
@@ -142,6 +142,17 @@ but not including any minor overhead arising from CBOR serialization. As noted p
142142

143143
As a general rule of thumb, assume that 80% of votes are persistent and 20% are non-persistent.
144144

145+
Here are details for how certificate operations vary with committee size.
146+
147+
| Number of pools | Number of committee seats | Generate certificate | Verify certificate | Weigh certificate |
148+
|----------------:|--------------------------:|---------------------:|-------------------:|------------------:|
149+
| 2500 | 500 | 63.4 ms | 104.8 ms | 10.6 ms |
150+
| 2500 | 600 | 71.1 ms | 116.9 ms | 12.0 ms |
151+
| 2500 | 700 | 77.4 ms | 125.5 ms | 12.3 ms |
152+
| 2500 | 800 | 83.5 ms | 134.4 ms | 12.8 ms |
153+
| 2500 | 900 | 88.2 ms | 141.1 ms | 12.4 ms |
154+
| 2500 | 1000 | 92.5 ms | 144.9 ms | 12.3 ms |
155+
145156

146157
## Certificate size for realistic stake distributions
147158

‎crypto-benchmarks.rs/benches/vote_bench.rs

+172-12
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ use criterion::{criterion_group, criterion_main, Criterion};
22
use leios_crypto_benchmarks::cert::*;
33
use leios_crypto_benchmarks::registry::{arbitrary_pools, PersistentId, Registry};
44
use quickcheck::{Arbitrary, Gen};
5+
use std::env;
56

67
use leios_crypto_benchmarks::key::{check_pop, key_gen, SecKey};
78
use leios_crypto_benchmarks::primitive::{
@@ -101,14 +102,22 @@ fn benchmark_verify_vote_nonpersistent(c: &mut Criterion) {
101102
});
102103
}
103104

104-
fn benchmark_gen_cert(c: &mut Criterion) {
105+
fn benchmark_gen_cert_impl(c: &mut Criterion, name: &'static str, n_pools: usize, n_voters: usize) {
105106
let g = &mut Gen::new(10);
106-
c.bench_function("cert::gen_cert", |b| {
107+
c.bench_function(name, |b| {
107108
b.iter_batched(
108109
|| {
109110
let total = realistic_total_stake(g);
110-
let n = realistic_pool_count(g);
111-
let voters = realistic_voters(g, n);
111+
let n = if n_pools > 0 {
112+
n_pools
113+
} else {
114+
realistic_pool_count(g)
115+
};
116+
let voters = if n_voters > 0 {
117+
n_voters
118+
} else {
119+
realistic_voters(g, n)
120+
};
112121
let stake = arbitrary_stake_distribution(g, total, n, 11., 1.);
113122
let pools = arbitrary_pools(g, &stake);
114123
let reg = Registry::make(&pools, voters);
@@ -121,14 +130,69 @@ fn benchmark_gen_cert(c: &mut Criterion) {
121130
});
122131
}
123132

124-
fn benchmark_verify_cert(c: &mut Criterion) {
133+
fn benchmark_gen_cert(c: &mut Criterion) {
134+
benchmark_gen_cert_impl(c, "cert::gen_cert", 0, 0);
135+
if env::var("BENCHMARK_CERT_EXTRA").map(|x| x == "1") == Result::Ok(true) {
136+
benchmark_gen_cert_impl(
137+
c,
138+
"cert::gen_cert, n_pools = 2500, n_voters = 500",
139+
2500,
140+
500,
141+
);
142+
benchmark_gen_cert_impl(
143+
c,
144+
"cert::gen_cert, n_pools = 2500, n_voters = 600",
145+
2500,
146+
600,
147+
);
148+
benchmark_gen_cert_impl(
149+
c,
150+
"cert::gen_cert, n_pools = 2500, n_voters = 700",
151+
2500,
152+
700,
153+
);
154+
benchmark_gen_cert_impl(
155+
c,
156+
"cert::gen_cert, n_pools = 2500, n_voters = 800",
157+
2500,
158+
800,
159+
);
160+
benchmark_gen_cert_impl(
161+
c,
162+
"cert::gen_cert, n_pools = 2500, n_voters = 900",
163+
2500,
164+
900,
165+
);
166+
benchmark_gen_cert_impl(
167+
c,
168+
"cert::gen_cert, n_pools = 2500, n_voters = 1000",
169+
2500,
170+
1000,
171+
);
172+
}
173+
}
174+
175+
fn benchmark_verify_cert_impl(
176+
c: &mut Criterion,
177+
name: &'static str,
178+
n_pools: usize,
179+
n_voters: usize,
180+
) {
125181
let g = &mut Gen::new(10);
126-
c.bench_function("cert::verify_cert", |b| {
182+
c.bench_function(name, |b| {
127183
b.iter_batched(
128184
|| {
129185
let total = realistic_total_stake(g);
130-
let n = realistic_pool_count(g);
131-
let voters = realistic_voters(g, n);
186+
let n = if n_pools > 0 {
187+
n_pools
188+
} else {
189+
realistic_pool_count(g)
190+
};
191+
let voters = if n_voters > 0 {
192+
n_voters
193+
} else {
194+
realistic_voters(g, n)
195+
};
132196
let stake = arbitrary_stake_distribution(g, total, n, 11., 1.);
133197
let pools = arbitrary_pools(g, &stake);
134198
let reg = Registry::make(&pools, voters);
@@ -142,14 +206,69 @@ fn benchmark_verify_cert(c: &mut Criterion) {
142206
});
143207
}
144208

145-
fn benchmark_weigh_cert(c: &mut Criterion) {
209+
fn benchmark_verify_cert(c: &mut Criterion) {
210+
benchmark_verify_cert_impl(c, "cert::verify_cert", 0, 0);
211+
if env::var("BENCHMARK_CERT_EXTRA").map(|x| x == "1") == Result::Ok(true) {
212+
benchmark_verify_cert_impl(
213+
c,
214+
"cert::verify_cert, n_pools = 2500, n_voters = 500",
215+
2500,
216+
500,
217+
);
218+
benchmark_verify_cert_impl(
219+
c,
220+
"cert::verify_cert, n_pools = 2500, n_voters = 600",
221+
2500,
222+
600,
223+
);
224+
benchmark_verify_cert_impl(
225+
c,
226+
"cert::verify_cert, n_pools = 2500, n_voters = 700",
227+
2500,
228+
700,
229+
);
230+
benchmark_verify_cert_impl(
231+
c,
232+
"cert::verify_cert, n_pools = 2500, n_voters = 800",
233+
2500,
234+
800,
235+
);
236+
benchmark_verify_cert_impl(
237+
c,
238+
"cert::verify_cert, n_pools = 2500, n_voters = 900",
239+
2500,
240+
900,
241+
);
242+
benchmark_verify_cert_impl(
243+
c,
244+
"cert::verify_cert, n_pools = 2500, n_voters = 1000",
245+
2500,
246+
1000,
247+
);
248+
}
249+
}
250+
251+
fn benchmark_weigh_cert_impl(
252+
c: &mut Criterion,
253+
name: &'static str,
254+
n_pools: usize,
255+
n_voters: usize,
256+
) {
146257
let g = &mut Gen::new(10);
147-
c.bench_function("cert::weigh_cert", |b| {
258+
c.bench_function(name, |b| {
148259
b.iter_batched(
149260
|| {
150261
let total = realistic_total_stake(g);
151-
let n = realistic_pool_count(g);
152-
let voters = realistic_voters(g, n);
262+
let n = if n_pools > 0 {
263+
n_pools
264+
} else {
265+
realistic_pool_count(g)
266+
};
267+
let voters = if n_voters > 0 {
268+
n_voters
269+
} else {
270+
realistic_voters(g, n)
271+
};
153272
let stake = arbitrary_stake_distribution(g, total, n, 11., 1.);
154273
let pools = arbitrary_pools(g, &stake);
155274
let reg = Registry::make(&pools, voters);
@@ -162,6 +281,47 @@ fn benchmark_weigh_cert(c: &mut Criterion) {
162281
)
163282
});
164283
}
284+
fn benchmark_weigh_cert(c: &mut Criterion) {
285+
benchmark_weigh_cert_impl(c, "cert::weigh_cert", 0, 0);
286+
if env::var("BENCHMARK_CERT_EXTRA").map(|x| x == "1") == Result::Ok(true) {
287+
benchmark_weigh_cert_impl(
288+
c,
289+
"cert::weigh_cert, n_pools = 2500, n_voters = 500",
290+
2500,
291+
500,
292+
);
293+
benchmark_weigh_cert_impl(
294+
c,
295+
"cert::weigh_cert, n_pools = 2500, n_voters = 600",
296+
2500,
297+
600,
298+
);
299+
benchmark_weigh_cert_impl(
300+
c,
301+
"cert::weigh_cert, n_pools = 2500, n_voters = 700",
302+
2500,
303+
700,
304+
);
305+
benchmark_weigh_cert_impl(
306+
c,
307+
"cert::weigh_cert, n_pools = 2500, n_voters = 800",
308+
2500,
309+
800,
310+
);
311+
benchmark_weigh_cert_impl(
312+
c,
313+
"cert::weigh_cert, n_pools = 2500, n_voters = 900",
314+
2500,
315+
900,
316+
);
317+
benchmark_weigh_cert_impl(
318+
c,
319+
"cert::weigh_cert, n_pools = 2500, n_voters = 1000",
320+
2500,
321+
1000,
322+
);
323+
}
324+
}
165325

166326
criterion_group!(
167327
benches,

‎crypto-benchmarks.rs/src/main.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,7 @@ use leios_crypto_benchmarks::key::{
1515
check_pop, key_gen, sign_message, verify_message, PoP, PubKey, Reg, SecKey, Sig,
1616
};
1717
use leios_crypto_benchmarks::primitive::{
18-
arbitrary_poolkeyhash, arbitrary_stake_distribution, Coin, EbHash, Eid, KesSig,
19-
PoolKeyhash,
18+
arbitrary_poolkeyhash, arbitrary_stake_distribution, Coin, EbHash, Eid, KesSig, PoolKeyhash,
2019
};
2120
use leios_crypto_benchmarks::registry::{arbitrary_pools, PoolInfo, Registry};
2221
use leios_crypto_benchmarks::sortition::voter_check;

‎crypto-benchmarks.rs/src/primitive.rs

+1-27
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,10 @@ use num_traits::FromPrimitive;
55
use pallas::ledger::primitives::{byron::Blake2b256, Hash};
66
use pallas::ledger::traverse::time::Slot;
77
use quickcheck::{Arbitrary, Gen};
8-
use rand::prelude::Distribution;
9-
use rand::rngs::StdRng;
10-
use rand::SeedableRng;
118
use serde::{Deserialize, Deserializer, Serialize, Serializer};
12-
use statrs::distribution::ContinuousCDF;
13-
use statrs::distribution::{Beta, Uniform};
149
use std::collections::BTreeMap;
1510

11+
use crate::realism::realistic_stake_dist;
1612
use crate::util::{arbitrary_fixed_bytes, deserialize_fixed_bytes, serialize_fixed_bytes};
1713

1814
pub use pallas::ledger::primitives::PoolKeyhash;
@@ -27,28 +23,6 @@ pub fn arbitrary_coin(g: &mut Gen) -> Coin {
2723
u64::arbitrary(g) % 999999 + 1
2824
}
2925

30-
pub(crate) fn realistic_stake_dist(
31-
g: &mut Gen,
32-
total: u64,
33-
n: usize,
34-
alpha: f64,
35-
beta: f64,
36-
) -> Vec<Coin> {
37-
let rng = &mut StdRng::seed_from_u64(u64::arbitrary(g));
38-
let noise = Uniform::new(0.75, 1.25).unwrap();
39-
let curve = Beta::new(alpha, beta).unwrap();
40-
let cum: Vec<f64> = (0..n)
41-
.map(|i| curve.cdf((i as f64) / (total as f64)))
42-
.collect();
43-
let dif: Vec<f64> = (1..n)
44-
.map(|i| (cum[i] - cum[i - 1]) * noise.sample(rng))
45-
.collect();
46-
let scale: f64 = (total as f64) / dif.iter().sum::<f64>();
47-
dif.iter()
48-
.map(|coin| (scale * *coin).round() as Coin)
49-
.collect()
50-
}
51-
5226
pub fn arbitrary_stake_distribution(
5327
g: &mut Gen,
5428
total: u64,

‎crypto-benchmarks.rs/src/realism.rs

+28-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,10 @@
1+
use pallas::ledger::primitives::Coin;
12
use quickcheck::{Arbitrary, Gen};
2-
3-
use crate::primitive::Coin;
3+
use rand::prelude::Distribution;
4+
use rand::rngs::StdRng;
5+
use rand::SeedableRng;
6+
use statrs::distribution::ContinuousCDF;
7+
use statrs::distribution::{Beta, Uniform};
48

59
pub fn realistic_pool_count(g: &mut Gen) -> usize {
610
usize::arbitrary(g) % 1500 + 1500
@@ -13,3 +17,25 @@ pub fn realistic_total_stake(g: &mut Gen) -> Coin {
1317
pub fn realistic_voters(g: &mut Gen, pools: usize) -> usize {
1418
pools * (usize::arbitrary(g) % 500 + 500) / 1000
1519
}
20+
21+
pub(crate) fn realistic_stake_dist(
22+
g: &mut Gen,
23+
total: u64,
24+
n: usize,
25+
alpha: f64,
26+
beta: f64,
27+
) -> Vec<Coin> {
28+
let rng = &mut StdRng::seed_from_u64(u64::arbitrary(g));
29+
let noise = Uniform::new(0.75, 1.25).unwrap();
30+
let curve = Beta::new(alpha, beta).unwrap();
31+
let cum: Vec<f64> = (0..n)
32+
.map(|i| curve.cdf((i as f64) / (total as f64)))
33+
.collect();
34+
let dif: Vec<f64> = (1..n)
35+
.map(|i| (cum[i] - cum[i - 1]) * noise.sample(rng))
36+
.collect();
37+
let scale: f64 = (total as f64) / dif.iter().sum::<f64>();
38+
dif.iter()
39+
.map(|coin| (scale * *coin).round() as Coin)
40+
.collect()
41+
}

0 commit comments

Comments
 (0)
Please sign in to comment.