From 4266b151841e4f10e4245856fc465ae08494bda6 Mon Sep 17 00:00:00 2001 From: Arthur Paulino Date: Tue, 19 Mar 2024 09:20:25 -0300 Subject: [PATCH] feat: simple NIVC benchmark with the Trie coprocessors (#1220) Add a straightforward NIVC benchmark that uses the Trie coprocessors to create proofs with the `SuperNovaProver`. This should be enough for a first implementation to drive design/optimization decisions for our NIVC pipeline. Closes #1219 --- Cargo.toml | 4 ++ benches/trie_nivc.rs | 114 +++++++++++++++++++++++++++++++++++++++++++ src/lem/mod.rs | 2 +- 3 files changed, 119 insertions(+), 1 deletion(-) create mode 100644 benches/trie_nivc.rs diff --git a/Cargo.toml b/Cargo.toml index 3475b210f..e91055dae 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -187,3 +187,7 @@ harness = false [[bench]] name = "public_params" harness = false + +[[bench]] +name = "trie_nivc" +harness = false diff --git a/benches/trie_nivc.rs b/benches/trie_nivc.rs new file mode 100644 index 000000000..2b718f0f1 --- /dev/null +++ b/benches/trie_nivc.rs @@ -0,0 +1,114 @@ +use criterion::{ + black_box, criterion_group, criterion_main, measurement, BenchmarkGroup, BenchmarkId, + Criterion, SamplingMode, +}; +use halo2curves::bn256::Fr; +use std::{sync::Arc, time::Duration}; + +use lurk::{ + coprocessor::trie::{install, TrieCoproc}, + dual_channel::dummy_terminal, + lang::Lang, + lem::{ + eval::{evaluate, make_cprocs_funcs_from_lang, make_eval_step_from_config, EvalConfig}, + interpreter::Frame, + store::Store, + }, + proof::supernova::{public_params, SuperNovaProver}, + state::State, +}; + +const CODE: &str = " +(let ((fib (letrec ((next (lambda (a b n target) + (if (eq n target) + a + (next b + (+ a b) + (+ 1 n) + target)))) + (fib (next 0 1 0))) + fib)) + (fib-trie (.lurk.trie.new)) + (fib-trie (.lurk.trie.insert fib-trie 40 (fib 40))) + (fib-trie (.lurk.trie.insert fib-trie 50 (fib 50)))) + (+ (num (.lurk.trie.lookup fib-trie 40)) (num (.lurk.trie.lookup fib-trie 50))))"; + +fn prove( + name: &str, + reduction_count: usize, + lang: &Arc>>, + store: &Store, + frames: &[Frame], + c: &mut BenchmarkGroup<'_, M>, +) { + c.bench_with_input( + BenchmarkId::new(name.to_string(), reduction_count), + &reduction_count, + |b, reduction_count| { + let rc = *reduction_count; + let prover = SuperNovaProver::new(rc, lang.clone()); + let pp = public_params(rc, lang.clone()); + b.iter(|| { + let (proof, ..) = prover.prove_from_frames(&pp, frames, store, None).unwrap(); + let _ = black_box(proof); + }) + }, + ); +} + +fn trie_nivc(c: &mut Criterion) { + let batch_sizes = [5, 10, 100, 200]; + let mut group: BenchmarkGroup<'_, _> = c.benchmark_group("trie-nivc"); + group.sampling_mode(SamplingMode::Flat); // This can take a *while* + group.sample_size(10); + + let state = State::init_lurk_state().rccell(); + let mut lang = Lang::new(); + install(&state, &mut lang); + let lang = Arc::new(lang); + + let store = Store::::default(); + let expr = store.read(state, CODE).unwrap(); + + let lurk_step = make_eval_step_from_config(&EvalConfig::new_nivc(&lang)); + let cprocs = make_cprocs_funcs_from_lang(&lang); + let frames = evaluate( + Some((&lurk_step, &cprocs, &lang)), + expr, + &store, + 1_000_000, + &dummy_terminal(), + ) + .unwrap(); + + assert_eq!(frames.last().unwrap().output[0], store.num_u64(12688603180)); + + for size in batch_sizes { + prove("rc", size, &lang, &store, &frames, &mut group); + } +} + +cfg_if::cfg_if! { + if #[cfg(feature = "flamegraph")] { + criterion_group! { + name = benches; + config = Criterion::default() + .measurement_time(Duration::from_secs(120)) + .sample_size(10) + .with_profiler(pprof::criterion::PProfProfiler::new(100, pprof::criterion::Output::Flamegraph(None))); + targets = + trie_nivc, + } + } else { + criterion_group! { + name = benches; + config = Criterion::default() + .measurement_time(Duration::from_secs(120)) + .sample_size(10); + targets = + trie_nivc, + } + } +} + +criterion_main!(benches); diff --git a/src/lem/mod.rs b/src/lem/mod.rs index bccdb2dbd..25186d7fe 100644 --- a/src/lem/mod.rs +++ b/src/lem/mod.rs @@ -62,7 +62,7 @@ pub mod circuit; pub mod coroutine; pub mod eval; -pub(crate) mod interpreter; +pub mod interpreter; mod macros; pub mod multiframe; pub mod pointers;