Skip to content

Commit

Permalink
update BLSCache::aggregate_verify() function to take an iterator of p…
Browse files Browse the repository at this point in the history
…airs rather than two separate iterators.
  • Loading branch information
arvidn committed Aug 2, 2024
1 parent ce4fff3 commit fceddb7
Show file tree
Hide file tree
Showing 2 changed files with 45 additions and 28 deletions.
40 changes: 30 additions & 10 deletions crates/chia-bls/benches/cache.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ fn cache_benchmark(c: &mut Criterion) {

let mut agg_sig = Signature::default();
for i in 0..1000 {
let derived = sk.derive_hardened(i as u32);
let derived = sk.derive_hardened(i);
let pk = derived.public_key();
let sig = sign(&derived, msg);
agg_sig.aggregate(&sig);
Expand All @@ -28,43 +28,63 @@ fn cache_benchmark(c: &mut Criterion) {
c.bench_function("bls_cache.aggregate_verify, 0% cache hits", |b| {
let mut cache = bls_cache.clone();
b.iter(|| {
assert!(cache.aggregate_verify(&pks, [&msg].iter().cycle(), &agg_sig));
assert!(
cache
.aggregate_verify(pks.iter().zip([&msg].iter().cycle()), &agg_sig)
.0
);
});
});

// populate 10% of keys
bls_cache.aggregate_verify(&pks[0..100], [&msg].iter().cycle(), &agg_sig);
bls_cache.aggregate_verify(pks[0..100].iter().zip([&msg].iter().cycle()), &agg_sig);
c.bench_function("bls_cache.aggregate_verify, 10% cache hits", |b| {
let mut cache = bls_cache.clone();
b.iter(|| {
assert!(cache.aggregate_verify(&pks, [&msg].iter().cycle(), &agg_sig));
assert!(
cache
.aggregate_verify(pks.iter().zip([&msg].iter().cycle()), &agg_sig)
.0
);
});
});

// populate another 10% of keys
bls_cache.aggregate_verify(&pks[100..200], [&msg].iter().cycle(), &agg_sig);
bls_cache.aggregate_verify(pks[100..200].iter().zip([&msg].iter().cycle()), &agg_sig);
c.bench_function("bls_cache.aggregate_verify, 20% cache hits", |b| {
let mut cache = bls_cache.clone();
b.iter(|| {
assert!(cache.aggregate_verify(&pks, [&msg].iter().cycle(), &agg_sig));
assert!(
cache
.aggregate_verify(pks.iter().zip([&msg].iter().cycle()), &agg_sig)
.0
);
});
});

// populate another 30% of keys
bls_cache.aggregate_verify(&pks[200..500], [&msg].iter().cycle(), &agg_sig);
bls_cache.aggregate_verify(pks[200..500].iter().zip([&msg].iter().cycle()), &agg_sig);
c.bench_function("bls_cache.aggregate_verify, 50% cache hits", |b| {
let mut cache = bls_cache.clone();
b.iter(|| {
assert!(cache.aggregate_verify(&pks, [&msg].iter().cycle(), &agg_sig));
assert!(
cache
.aggregate_verify(pks.iter().zip([&msg].iter().cycle()), &agg_sig)
.0
);
});
});

// populate all other keys
bls_cache.aggregate_verify(&pks[500..1000], [&msg].iter().cycle(), &agg_sig);
bls_cache.aggregate_verify(pks[500..1000].iter().zip([&msg].iter().cycle()), &agg_sig);
c.bench_function("bls_cache.aggregate_verify, 100% cache hits", |b| {
let mut cache = bls_cache.clone();
b.iter(|| {
assert!(cache.aggregate_verify(&pks, [&msg].iter().cycle(), &agg_sig));
assert!(
cache
.aggregate_verify(pks.iter().zip([&msg].iter().cycle()), &agg_sig)
.0
);
});
});

Expand Down
33 changes: 15 additions & 18 deletions crates/chia-bls/src/bls_cache.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,9 @@
use std::borrow::Borrow;
use std::num::NonZeroUsize;

use lru::LruCache;
use sha2::{Digest, Sha256};

use crate::{aggregate_verify_gt, hash_to_g2};
use crate::{GTElement, PublicKey, Signature};
use lru::LruCache;
use sha2::{Digest, Sha256};
use std::borrow::Borrow;
use std::num::NonZeroUsize;

/// This is a cache of pairings of public keys and their corresponding message.
/// It accelerates aggregate verification when some public keys have already
Expand Down Expand Up @@ -44,13 +42,12 @@ impl BlsCache {
self.cache.is_empty()
}

pub fn aggregate_verify(
pub fn aggregate_verify<Pk: Borrow<PublicKey>, Msg: AsRef<[u8]>>(
&mut self,
pks: impl IntoIterator<Item = impl Borrow<PublicKey>>,
msgs: impl IntoIterator<Item = impl AsRef<[u8]>>,
pks_msgs: impl IntoIterator<Item = (Pk, Msg)>,
sig: &Signature,
) -> bool {
let iter = pks.into_iter().zip(msgs).map(|(pk, msg)| -> GTElement {
let iter = pks_msgs.into_iter().map(|(pk, msg)| -> GTElement {
// Hash pubkey + message
let mut hasher = Sha256::new();
hasher.update(pk.borrow().to_bytes());
Expand Down Expand Up @@ -124,7 +121,7 @@ impl BlsCache {
.map(|item| item?.extract())
.collect::<PyResult<Vec<PyBackedBytes>>>()?;

Ok(self.aggregate_verify(pks, msgs, sig))
Ok(self.aggregate_verify(pks.into_iter().zip(msgs), sig))
}

#[pyo3(name = "len")]
Expand Down Expand Up @@ -187,11 +184,11 @@ pub mod tests {
assert!(bls_cache.is_empty());

// Verify the signature and add to the cache.
assert!(bls_cache.aggregate_verify(pk_list, msg_list, &sig));
assert!(bls_cache.aggregate_verify(pk_list.into_iter().zip(msg_list), &sig));
assert_eq!(bls_cache.len(), 1);

// Now that it's cached, it shouldn't cache it again.
assert!(bls_cache.aggregate_verify(pk_list, msg_list, &sig));
assert!(bls_cache.aggregate_verify(pk_list.into_iter().zip(msg_list), &sig));
assert_eq!(bls_cache.len(), 1);
}

Expand All @@ -211,7 +208,7 @@ pub mod tests {
assert!(bls_cache.is_empty());

// Add the first signature to cache.
assert!(bls_cache.aggregate_verify(&pk_list, &msg_list, &agg_sig));
assert!(bls_cache.aggregate_verify(pk_list.iter().zip(msg_list.iter()), &agg_sig));
assert_eq!(bls_cache.len(), 1);

// Try with the first key message pair in the cache but not the second.
Expand All @@ -223,7 +220,7 @@ pub mod tests {
pk_list.push(pk2);
msg_list.push(msg2);

assert!(bls_cache.aggregate_verify(&pk_list, &msg_list, &agg_sig));
assert!(bls_cache.aggregate_verify(pk_list.iter().zip(msg_list.iter()), &agg_sig));
assert_eq!(bls_cache.len(), 2);

// Try reusing a public key.
Expand All @@ -234,7 +231,7 @@ pub mod tests {
msg_list.push(msg3);

// Verify this signature and add to the cache as well (since it's still a different aggregate).
assert!(bls_cache.aggregate_verify(pk_list, msg_list, &agg_sig));
assert!(bls_cache.aggregate_verify(pk_list.iter().zip(msg_list), &agg_sig));
assert_eq!(bls_cache.len(), 3);
}

Expand All @@ -257,7 +254,7 @@ pub mod tests {
let msg_list = [msg];

// Add to cache by validating them one at a time.
assert!(bls_cache.aggregate_verify(pk_list.iter(), msg_list.iter(), &sig));
assert!(bls_cache.aggregate_verify(pk_list.into_iter().zip(msg_list), &sig));
}

// The cache should be full now.
Expand Down Expand Up @@ -285,6 +282,6 @@ pub mod tests {
let pks: [&PublicKey; 0] = [];
let msgs: [&[u8]; 0] = [];

assert!(bls_cache.aggregate_verify(pks, msgs, &Signature::default()));
assert!(bls_cache.aggregate_verify(pks.into_iter().zip(msgs), &Signature::default()));
}
}

0 comments on commit fceddb7

Please sign in to comment.