2021-06-29 12:53:55 -07:00
|
|
|
//! Benchmarks for batch verifiication of RedPallas signatures.
|
|
|
|
|
2022-01-24 08:25:06 -08:00
|
|
|
// Disabled due to warnings in criterion macros
|
|
|
|
#![allow(missing_docs)]
|
2021-06-12 00:11:44 -07:00
|
|
|
|
|
|
|
use criterion::{criterion_group, criterion_main, BenchmarkId, Criterion, Throughput};
|
|
|
|
use rand::{thread_rng, Rng};
|
2023-02-01 15:27:28 -08:00
|
|
|
use reddsa::{
|
|
|
|
batch,
|
|
|
|
orchard::{Binding, SpendAuth},
|
|
|
|
Signature, SigningKey, VerificationKey, VerificationKeyBytes,
|
|
|
|
};
|
2021-06-12 00:11:44 -07:00
|
|
|
|
2021-06-29 14:33:42 -07:00
|
|
|
const MESSAGE_BYTES: &[u8; 0] = b"";
|
|
|
|
|
2021-06-29 12:53:55 -07:00
|
|
|
/// A batch verification item of a RedPallas signature variant.
|
|
|
|
///
|
|
|
|
/// This struct exists to allow batch processing to be decoupled from the
|
|
|
|
/// lifetime of the message. This is useful when using the batch verification
|
|
|
|
/// API in an async context. The different enum variants are for the different
|
|
|
|
/// signature types which use different Pallas basepoints for computation.
|
2021-06-12 00:11:44 -07:00
|
|
|
enum Item {
|
|
|
|
SpendAuth {
|
|
|
|
vk_bytes: VerificationKeyBytes<SpendAuth>,
|
|
|
|
sig: Signature<SpendAuth>,
|
|
|
|
},
|
|
|
|
Binding {
|
|
|
|
vk_bytes: VerificationKeyBytes<Binding>,
|
|
|
|
sig: Signature<Binding>,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-06-29 12:53:55 -07:00
|
|
|
/// Generates an iterator of random [Item]s
|
|
|
|
///
|
2021-11-12 11:30:22 -08:00
|
|
|
/// Each [Item] has a unique [SigningKey], randomly chosen [SigType] variant,
|
2021-06-29 12:53:55 -07:00
|
|
|
/// and signature over the empty message, "".
|
2021-06-12 00:11:44 -07:00
|
|
|
fn sigs_with_distinct_keys() -> impl Iterator<Item = Item> {
|
|
|
|
std::iter::repeat_with(|| {
|
|
|
|
let mut rng = thread_rng();
|
2021-06-29 14:33:42 -07:00
|
|
|
// let msg = b"";
|
2021-06-12 00:11:44 -07:00
|
|
|
match rng.gen::<u8>() % 2 {
|
|
|
|
0 => {
|
|
|
|
let sk = SigningKey::<SpendAuth>::new(thread_rng());
|
|
|
|
let vk_bytes = VerificationKey::from(&sk).into();
|
2021-06-29 14:33:42 -07:00
|
|
|
let sig = sk.sign(thread_rng(), &MESSAGE_BYTES[..]);
|
2021-06-12 00:11:44 -07:00
|
|
|
Item::SpendAuth { vk_bytes, sig }
|
|
|
|
}
|
|
|
|
1 => {
|
|
|
|
let sk = SigningKey::<Binding>::new(thread_rng());
|
|
|
|
let vk_bytes = VerificationKey::from(&sk).into();
|
2021-06-29 14:33:42 -07:00
|
|
|
let sig = sk.sign(thread_rng(), &MESSAGE_BYTES[..]);
|
2021-06-12 00:11:44 -07:00
|
|
|
Item::Binding { vk_bytes, sig }
|
|
|
|
}
|
|
|
|
_ => panic!(),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-06-29 12:53:55 -07:00
|
|
|
/// Benchmarks batched vs unbatched RedPallas signature verification.
|
|
|
|
///
|
|
|
|
/// Includes heterogeneous groups across [SigType], [SigningKey]s, and messages.
|
2021-06-12 00:11:44 -07:00
|
|
|
fn bench_batch_verify(c: &mut Criterion) {
|
|
|
|
let mut group = c.benchmark_group("Batch Verification");
|
|
|
|
for &n in [8usize, 16, 24, 32, 40, 48, 56, 64].iter() {
|
|
|
|
group.throughput(Throughput::Elements(n as u64));
|
|
|
|
|
|
|
|
let sigs = sigs_with_distinct_keys().take(n).collect::<Vec<_>>();
|
|
|
|
|
|
|
|
group.bench_with_input(
|
|
|
|
BenchmarkId::new("Unbatched verification", n),
|
|
|
|
&sigs,
|
|
|
|
|b, sigs| {
|
|
|
|
b.iter(|| {
|
|
|
|
for item in sigs.iter() {
|
|
|
|
match item {
|
|
|
|
Item::SpendAuth { vk_bytes, sig } => {
|
2021-06-29 14:56:12 -07:00
|
|
|
assert!(VerificationKey::try_from(*vk_bytes)
|
|
|
|
.and_then(|vk| vk.verify(MESSAGE_BYTES, sig))
|
|
|
|
.is_ok());
|
2021-06-12 00:11:44 -07:00
|
|
|
}
|
|
|
|
Item::Binding { vk_bytes, sig } => {
|
2021-06-29 14:56:12 -07:00
|
|
|
assert!(VerificationKey::try_from(*vk_bytes)
|
|
|
|
.and_then(|vk| vk.verify(MESSAGE_BYTES, sig))
|
|
|
|
.is_ok());
|
2021-06-12 00:11:44 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
group.bench_with_input(
|
|
|
|
BenchmarkId::new("Batched verification", n),
|
|
|
|
&sigs,
|
|
|
|
|b, sigs| {
|
|
|
|
b.iter(|| {
|
|
|
|
let mut batch = batch::Verifier::new();
|
|
|
|
for item in sigs.iter() {
|
|
|
|
match item {
|
|
|
|
Item::SpendAuth { vk_bytes, sig } => {
|
2023-02-01 15:27:28 -08:00
|
|
|
batch.queue(batch::Item::from_spendauth(
|
|
|
|
*vk_bytes,
|
|
|
|
*sig,
|
|
|
|
MESSAGE_BYTES,
|
|
|
|
));
|
2021-06-12 00:11:44 -07:00
|
|
|
}
|
|
|
|
Item::Binding { vk_bytes, sig } => {
|
2023-02-01 15:27:28 -08:00
|
|
|
batch.queue(batch::Item::from_binding(
|
|
|
|
*vk_bytes,
|
|
|
|
*sig,
|
|
|
|
MESSAGE_BYTES,
|
|
|
|
));
|
2021-06-12 00:11:44 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-06-29 14:56:12 -07:00
|
|
|
assert!(batch.verify(thread_rng()).is_ok())
|
2021-06-12 00:11:44 -07:00
|
|
|
})
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
group.finish();
|
|
|
|
}
|
|
|
|
|
|
|
|
criterion_group!(benches, bench_batch_verify);
|
|
|
|
criterion_main!(benches);
|