From d99c22da6b733131c665b8bf8edcc33258d4f41d Mon Sep 17 00:00:00 2001 From: Gus Gutoski Date: Wed, 1 Nov 2023 13:28:14 -0400 Subject: [PATCH] test: update VID bench code (#392) * nix voodoo * bench panic with useful message if needed features are missing * ignore benchmark reports put in primitives/target * use power-of-2 node counts to accommodate advz vid scheme constraints * [no-ci] clarifying comment --- .envrc | 2 + .gitignore | 4 + flake.nix | 2 + primitives/benches/advz.rs | 242 ++++++++++++++++++++----------------- 4 files changed, 136 insertions(+), 114 deletions(-) diff --git a/.envrc b/.envrc index 1d953f4bd..79a1f884f 100644 --- a/.envrc +++ b/.envrc @@ -1 +1,3 @@ use nix +watch_file flake.nix +watch_file flake.lock \ No newline at end of file diff --git a/.gitignore b/.gitignore index 24c783915..c1e4d1003 100644 --- a/.gitignore +++ b/.gitignore @@ -11,3 +11,7 @@ Cargo.lock # Test coverage (grcov) default.profraw /.pre-commit-config.yaml + +# benchmark reports placed here for some reason: +# https://doc.rust-lang.org/cargo/commands/cargo-bench.html#working-directory-of-benchmarks +/primitives/target diff --git a/flake.nix b/flake.nix index c659ef0d4..5440c966c 100644 --- a/flake.nix +++ b/flake.nix @@ -81,6 +81,8 @@ llvm_15 ] ++ lib.optionals stdenv.isDarwin [ darwin.apple_sdk.frameworks.Security ]; + CARGO_TARGET_DIR = "target/nix_rustc"; + shellHook = '' export RUST_BACKTRACE=full export PATH="$PATH:$(pwd)/target/debug:$(pwd)/target/release" diff --git a/primitives/benches/advz.rs b/primitives/benches/advz.rs index 38a551deb..7017e7a17 100644 --- a/primitives/benches/advz.rs +++ b/primitives/benches/advz.rs @@ -4,130 +4,144 @@ // You should have received a copy of the MIT License // along with the Jellyfish library. If not, see . -#![cfg(feature = "test-srs")] -use ark_bls12_381::Bls12_381; -use ark_bn254::Bn254; -use ark_ec::pairing::Pairing; -use ark_serialize::Write; -use ark_std::rand::RngCore; -use criterion::{criterion_group, criterion_main, BenchmarkId, Criterion, Throughput}; -use digest::{crypto_common::generic_array::ArrayLength, Digest, DynDigest, OutputSizeUser}; -use jf_primitives::{ - pcs::{checked_fft_size, prelude::UnivariateKzgPCS, PolynomialCommitmentScheme}, - vid::{advz::Advz, VidScheme}, -}; -use sha2::Sha256; +#[cfg(not(feature = "test-srs"))] +fn main() { + panic!("need `test-srs` feature to run this benchmark"); +} -const KB: usize = 1 << 10; -const MB: usize = KB << 10; +#[cfg(feature = "test-srs")] +criterion::criterion_main!(feature_gated::benches); -fn advz(c: &mut Criterion, pairing_name: &str) -where - E: Pairing, - // TODO(Gus) clean up nasty trait bounds upstream - H: Digest + DynDigest + Default + Clone + Write, - <::OutputSize as ArrayLength>::ArrayType: Copy, -{ - // play with these items - const RATE: usize = 4; // ratio of num_storage_nodes : polynomial_degree - let storage_node_counts = [600, 700, 800, 900, 1000]; - let payload_byte_lens = [1 * MB]; +#[cfg(feature = "test-srs")] +mod feature_gated { + use ark_bls12_381::Bls12_381; + use ark_bn254::Bn254; + use ark_ec::pairing::Pairing; + use ark_serialize::Write; + use ark_std::rand::RngCore; + use criterion::{BenchmarkId, Criterion, Throughput}; + use digest::{crypto_common::generic_array::ArrayLength, Digest, DynDigest, OutputSizeUser}; + use jf_primitives::{ + pcs::{checked_fft_size, prelude::UnivariateKzgPCS, PolynomialCommitmentScheme}, + vid::{advz::Advz, VidScheme}, + }; + use sha2::Sha256; - // more items as a function of the above - let poly_degrees_iter = storage_node_counts.iter().map(|c| c / RATE); - let supported_degree = poly_degrees_iter.clone().max().unwrap(); - let vid_sizes_iter = poly_degrees_iter.zip(storage_node_counts); - let mut rng = jf_utils::test_rng(); - let srs = UnivariateKzgPCS::::gen_srs_for_testing( - &mut rng, - checked_fft_size(supported_degree).unwrap(), - ) - .unwrap(); + const KB: usize = 1 << 10; + const MB: usize = KB << 10; - // run all benches for each payload_byte_lens - for len in payload_byte_lens { - // random payload data - let mut payload_bytes = vec![0u8; len]; - rng.fill_bytes(&mut payload_bytes); + fn advz(c: &mut Criterion, pairing_name: &str) + where + E: Pairing, + // TODO(Gus) clean up nasty trait bounds upstream + H: Digest + DynDigest + Default + Clone + Write, + <::OutputSize as ArrayLength>::ArrayType: Copy, + { + // play with these items + // + // CODE_RATE is merely a convenient way to automatically choose polynomial + // degree as a function of storage node count. + // If desired, you could set polynomial degrees independent of storage node + // count. + const CODE_RATE: usize = 4; // ratio of num_storage_nodes : polynomial_degree + let storage_node_counts = [512, 1024]; + let payload_byte_lens = [1 * MB]; - let benchmark_group_name = - |op_name| format!("advz_{}_{}_{}KB", pairing_name, op_name, len / KB); + // more items as a function of the above + let poly_degrees_iter = storage_node_counts.iter().map(|c| c / CODE_RATE); + let supported_degree = poly_degrees_iter.clone().max().unwrap(); + let vid_sizes_iter = poly_degrees_iter.zip(storage_node_counts); + let mut rng = jf_utils::test_rng(); + let srs = UnivariateKzgPCS::::gen_srs_for_testing( + &mut rng, + checked_fft_size(supported_degree).unwrap(), + ) + .unwrap(); - // commit - let mut grp = c.benchmark_group(benchmark_group_name("commit")); - grp.throughput(Throughput::Bytes(len as u64)); - for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { - let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); - grp.bench_with_input( - BenchmarkId::from_parameter(num_storage_nodes), - &num_storage_nodes, - |b, _| { - b.iter(|| advz.commit_only(&payload_bytes).unwrap()); - }, - ); - } - grp.finish(); + // run all benches for each payload_byte_lens + for len in payload_byte_lens { + // random payload data + let mut payload_bytes = vec![0u8; len]; + rng.fill_bytes(&mut payload_bytes); - // disperse - let mut grp = c.benchmark_group(benchmark_group_name("disperse")); - grp.throughput(Throughput::Bytes(len as u64)); - for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { - let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); - grp.bench_with_input( - BenchmarkId::from_parameter(num_storage_nodes), - &num_storage_nodes, - |b, _| { - b.iter(|| advz.disperse(&payload_bytes).unwrap()); - }, - ); - } - grp.finish(); + let benchmark_group_name = + |op_name| format!("advz_{}_{}_{}KB", pairing_name, op_name, len / KB); - // verify - let mut grp = c.benchmark_group(benchmark_group_name("verify")); - grp.throughput(Throughput::Bytes(len as u64)); - for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { - let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); - let disperse = advz.disperse(&payload_bytes).unwrap(); - let (shares, common) = (disperse.shares, disperse.common); - grp.bench_with_input( - BenchmarkId::from_parameter(num_storage_nodes), - &num_storage_nodes, - |b, _| { - // verify only the 0th share - b.iter(|| advz.verify_share(&shares[0], &common).unwrap().unwrap()); - }, - ); - } - grp.finish(); + // commit + let mut grp = c.benchmark_group(benchmark_group_name("commit")); + grp.throughput(Throughput::Bytes(len as u64)); + for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { + let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); + grp.bench_with_input( + BenchmarkId::from_parameter(num_storage_nodes), + &num_storage_nodes, + |b, _| { + b.iter(|| advz.commit_only(&payload_bytes).unwrap()); + }, + ); + } + grp.finish(); + + // disperse + let mut grp = c.benchmark_group(benchmark_group_name("disperse")); + grp.throughput(Throughput::Bytes(len as u64)); + for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { + let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); + grp.bench_with_input( + BenchmarkId::from_parameter(num_storage_nodes), + &num_storage_nodes, + |b, _| { + b.iter(|| advz.disperse(&payload_bytes).unwrap()); + }, + ); + } + grp.finish(); - // recover - let mut grp = c.benchmark_group(benchmark_group_name("recover")); - grp.throughput(Throughput::Bytes(len as u64)); - for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { - let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); - let disperse = advz.disperse(&payload_bytes).unwrap(); - let (shares, common) = (disperse.shares, disperse.common); - grp.bench_with_input( - BenchmarkId::from_parameter(num_storage_nodes), - &num_storage_nodes, - |b, _| { - // recover from only the first poly_degree shares - b.iter(|| { - advz.recover_payload(&shares[..poly_degree], &common) - .unwrap() - }); - }, - ); + // verify + let mut grp = c.benchmark_group(benchmark_group_name("verify")); + grp.throughput(Throughput::Bytes(len as u64)); + for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { + let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); + let disperse = advz.disperse(&payload_bytes).unwrap(); + let (shares, common) = (disperse.shares, disperse.common); + grp.bench_with_input( + BenchmarkId::from_parameter(num_storage_nodes), + &num_storage_nodes, + |b, _| { + // verify only the 0th share + b.iter(|| advz.verify_share(&shares[0], &common).unwrap().unwrap()); + }, + ); + } + grp.finish(); + + // recover + let mut grp = c.benchmark_group(benchmark_group_name("recover")); + grp.throughput(Throughput::Bytes(len as u64)); + for (poly_degree, num_storage_nodes) in vid_sizes_iter.clone() { + let advz = Advz::::new(poly_degree, num_storage_nodes, &srs).unwrap(); + let disperse = advz.disperse(&payload_bytes).unwrap(); + let (shares, common) = (disperse.shares, disperse.common); + grp.bench_with_input( + BenchmarkId::from_parameter(num_storage_nodes), + &num_storage_nodes, + |b, _| { + // recover from only the first poly_degree shares + b.iter(|| { + advz.recover_payload(&shares[..poly_degree], &common) + .unwrap() + }); + }, + ); + } + grp.finish(); } - grp.finish(); } -} -fn advz_main(c: &mut Criterion) { - advz::(c, "Bls381"); - advz::(c, "Bn254"); -} + fn advz_main(c: &mut Criterion) { + advz::(c, "Bls381"); + advz::(c, "Bn254"); + } -criterion_group!(name = benches; config = Criterion::default().sample_size(10); targets = advz_main); -criterion_main!(benches); + criterion::criterion_group!(name = benches; config = Criterion::default().sample_size(10); targets = advz_main); +}