Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Always prove to produce proofs for BatchProofs #1540

Open
wants to merge 10 commits into
base: nightly
Choose a base branch
from
11 changes: 6 additions & 5 deletions bin/citrea/src/rollup/bitcoin.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,8 @@ use sov_rollup_interface::spec::SpecId;
use sov_rollup_interface::zk::Zkvm;
use sov_state::ZkStorage;
use sov_stf_runner::ProverGuestRunConfig;
use tokio::sync::broadcast;
use tokio::sync::mpsc::unbounded_channel;
use tokio::sync::{broadcast, Mutex};
use tracing::instrument;

use crate::CitreaRollupBlueprint;
Expand Down Expand Up @@ -201,10 +201,10 @@ impl RollupBlueprint for BitcoinRollup {
ProverGuestRunConfig::Skip => ProofGenMode::Skip,
ProverGuestRunConfig::Simulate => {
let stf_verifier = StateTransitionVerifier::new(zk_stf, da_verifier);
ProofGenMode::Simulate(stf_verifier)
ProofGenMode::Simulate(Arc::new(Mutex::new(stf_verifier)))
}
ProverGuestRunConfig::Execute => ProofGenMode::Execute,
ProverGuestRunConfig::Prove => ProofGenMode::Prove,
ProverGuestRunConfig::Prove => ProofGenMode::Prove(prover_config.proof_sampling_number),
};

ParallelProverService::new_from_env(
Expand Down Expand Up @@ -241,10 +241,11 @@ impl RollupBlueprint for BitcoinRollup {
ProverGuestRunConfig::Skip => ProofGenMode::Skip,
ProverGuestRunConfig::Simulate => {
let stf_verifier = StateTransitionVerifier::new(zk_stf, da_verifier);
ProofGenMode::Simulate(stf_verifier)

ProofGenMode::Simulate(Arc::new(Mutex::new(stf_verifier)))
}
ProverGuestRunConfig::Execute => ProofGenMode::Execute,
ProverGuestRunConfig::Prove => ProofGenMode::Prove,
ProverGuestRunConfig::Prove => ProofGenMode::Prove(prover_config.proof_sampling_number),
};

ParallelProverService::new(da_service.clone(), vm, proof_mode, zk_storage, 1, ledger_db)
Expand Down
10 changes: 5 additions & 5 deletions bin/citrea/src/rollup/mock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ use sov_rollup_interface::spec::SpecId;
use sov_rollup_interface::zk::Zkvm;
use sov_state::ZkStorage;
use sov_stf_runner::ProverGuestRunConfig;
use tokio::sync::broadcast;
use tokio::sync::{broadcast, Mutex};

use crate::CitreaRollupBlueprint;

Expand Down Expand Up @@ -132,10 +132,10 @@ impl RollupBlueprint for MockDemoRollup {
ProverGuestRunConfig::Skip => ProofGenMode::Skip,
ProverGuestRunConfig::Simulate => {
let stf_verifier = StateTransitionVerifier::new(zk_stf, da_verifier);
ProofGenMode::Simulate(stf_verifier)
ProofGenMode::Simulate(Arc::new(Mutex::new(stf_verifier)))
}
ProverGuestRunConfig::Execute => ProofGenMode::Execute,
ProverGuestRunConfig::Prove => ProofGenMode::Prove,
ProverGuestRunConfig::Prove => ProofGenMode::Prove(prover_config.proof_sampling_number),
};

ParallelProverService::new(da_service.clone(), vm, proof_mode, zk_storage, 1, ledger_db)
Expand All @@ -158,10 +158,10 @@ impl RollupBlueprint for MockDemoRollup {
ProverGuestRunConfig::Skip => ProofGenMode::Skip,
ProverGuestRunConfig::Simulate => {
let stf_verifier = StateTransitionVerifier::new(zk_stf, da_verifier);
ProofGenMode::Simulate(stf_verifier)
ProofGenMode::Simulate(Arc::new(Mutex::new(stf_verifier)))
}
ProverGuestRunConfig::Execute => ProofGenMode::Execute,
ProverGuestRunConfig::Prove => ProofGenMode::Prove,
ProverGuestRunConfig::Prove => ProofGenMode::Prove(prover_config.proof_sampling_number),
};

ParallelProverService::new(da_service.clone(), vm, proof_mode, zk_storage, 1, ledger_db)
Expand Down
32 changes: 12 additions & 20 deletions crates/batch-prover/src/da_block_handler.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,6 @@ use citrea_common::utils::merge_state_diffs;
use citrea_common::BatchProverConfig;
use citrea_primitives::compression::compress_blob;
use citrea_primitives::MAX_TXBODY_SIZE;
use rand::Rng;
use serde::de::DeserializeOwned;
use serde::Serialize;
use sov_db::ledger_db::BatchProverLedgerOps;
Expand Down Expand Up @@ -233,25 +232,18 @@ where
l1_block.header().height(),
);

// if proof_sampling_number is 0, then we always prove and submit
// otherwise we submit and prove with a probability of 1/proof_sampling_number
let should_prove = self.prover_config.proof_sampling_number == 0
|| rand::thread_rng().gen_range(0..self.prover_config.proof_sampling_number) == 0;

if should_prove {
if l1_height >= self.skip_submission_until_l1 {
prove_l1::<Da, Ps, Vm, DB, StateRoot, Witness, Tx>(
self.prover_service.clone(),
self.ledger_db.clone(),
self.code_commitments_by_spec.clone(),
l1_block.clone(),
sequencer_commitments,
inputs,
)
.await?;
} else {
info!("Skipping proving for l1 height {}", l1_height);
}
if l1_height >= self.skip_submission_until_l1 {
prove_l1::<Da, Ps, Vm, DB, StateRoot, Witness, Tx>(
self.prover_service.clone(),
self.ledger_db.clone(),
self.code_commitments_by_spec.clone(),
l1_block.clone(),
sequencer_commitments,
inputs,
)
.await?;
} else {
info!("Skipping proving for l1 height {}", l1_height);
}

if let Err(e) = self
Expand Down
33 changes: 31 additions & 2 deletions crates/prover-services/src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,17 @@
use std::sync::Arc;

use citrea_stf::verifier::StateTransitionVerifier;
use sov_rollup_interface::services::da::DaService;
use sov_rollup_interface::stf::StateTransitionFunction;
use sov_rollup_interface::zk::ZkvmHost;
use tokio::sync::Mutex;

mod parallel;
pub use parallel::*;

type Simulator<Stf, DaVerifier, VmGuest> =
Arc<Mutex<StateTransitionVerifier<Stf, DaVerifier, VmGuest>>>;

pub enum ProofGenMode<Da, Vm, Stf>
where
Da: DaService,
Expand All @@ -15,10 +21,33 @@ where
/// Skips proving.
Skip,
/// The simulator runs the rollup verifier logic without even emulating the zkVM
Simulate(StateTransitionVerifier<Stf, Da::Verifier, Vm::Guest>),
Simulate(Simulator<Stf, Da::Verifier, Vm::Guest>),
yaziciahmet marked this conversation as resolved.
Show resolved Hide resolved
/// The executor runs the rollup verification logic in the zkVM, but does not actually
/// produce a zk proof
Execute,
/// The prover runs the rollup verification logic in the zkVM and produces a zk proof
Prove,
Prove(
/// Average number of commitments to prove
/// If proof_sampling_number is 0, then we always prove and submit
/// Otherwise we submit and prove with a probability of 1/proof_sampling_number
///
/// proof_sampling_number:
usize,
),
}

impl<Da, Vm, Stf> Clone for ProofGenMode<Da, Vm, Stf>
where
Da: DaService,
Vm: ZkvmHost,
Stf: StateTransitionFunction<Da::Spec>,
{
fn clone(&self) -> Self {
match self {
Self::Skip => Self::Skip,
Self::Execute => Self::Execute,
Self::Prove(proof_sampling_number) => Self::Prove(*proof_sampling_number),
Self::Simulate(simulate) => Self::Simulate(Arc::clone(simulate)),
}
}
}
43 changes: 29 additions & 14 deletions crates/prover-services/src/parallel/mod.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
use std::ops::DerefMut;
use std::sync::Arc;

use async_trait::async_trait;
use futures::future;
use rand::Rng;
use sov_db::ledger_db::LedgerDB;
use sov_rollup_interface::da::DaData;
use sov_rollup_interface::services::da::DaService;
Expand All @@ -27,7 +27,7 @@ where
{
thread_pool: rayon::ThreadPool,

proof_mode: Arc<Mutex<ProofGenMode<Da, Vm, Stf>>>,
proof_mode: ProofGenMode<Da, Vm, Stf>,

da_service: Arc<Da>,
vm: Vm,
Expand Down Expand Up @@ -68,8 +68,14 @@ where
ProofGenMode::Execute => {
tracing::info!("Prover is configured to execute proving");
}
ProofGenMode::Prove => {
tracing::info!("Prover is configured to prove");
ProofGenMode::Prove(proof_sampling_number) => {
if proof_sampling_number == 0 {
tracing::info!("Prover is configured to prove");
} else {
tracing::info!(
"Prover is configured to prove with 1/{proof_sampling_number} sampling"
);
}
}
};

Expand All @@ -80,7 +86,7 @@ where

Ok(Self {
thread_pool,
proof_mode: Arc::new(Mutex::new(proof_mode)),
proof_mode,
da_service,
vm,
zk_storage,
Expand Down Expand Up @@ -190,7 +196,7 @@ where

async fn prove(&self) -> anyhow::Result<Vec<Proof>> {
let mut proof_queue = self.proof_queue.lock().await;
if let ProofGenMode::Skip = *self.proof_mode.lock().await {
if let ProofGenMode::Skip = self.proof_mode {
tracing::debug!("Skipped proving {} proofs", proof_queue.len());
proof_queue.clear();
return Ok(vec![]);
Expand Down Expand Up @@ -233,22 +239,31 @@ where
fn make_proof<Da, Vm, Stf>(
mut vm: Vm,
zk_storage: Stf::PreState,
proof_mode: Arc<Mutex<ProofGenMode<Da, Vm, Stf>>>,
proof_mode: ProofGenMode<Da, Vm, Stf>,
) -> Result<Proof, anyhow::Error>
where
Da: DaService,
Vm: ZkvmHost,
Stf: StateTransitionFunction<Da::Spec> + Send + Sync,
Stf::PreState: Send + Sync,
{
let mut proof_mode = proof_mode.blocking_lock();
match proof_mode.deref_mut() {
let proof_mode = proof_mode;
match proof_mode {
kpp marked this conversation as resolved.
Show resolved Hide resolved
ProofGenMode::Skip => Ok(Vec::default()),
ProofGenMode::Simulate(ref mut verifier) => verifier
.run_sequencer_commitments_in_da_slot(vm.simulate_with_hints(), zk_storage)
.map(|_| Vec::default())
.map_err(|e| anyhow::anyhow!("Guest execution must succeed but failed with {:?}", e)),
ProofGenMode::Simulate(verifier) => {
let mut verifier = verifier.blocking_lock();
verifier
.run_sequencer_commitments_in_da_slot(vm.simulate_with_hints(), zk_storage)
.map(|_| Vec::default())
.map_err(|e| {
anyhow::anyhow!("Guest execution must succeed but failed with {:?}", e)
})
}
ProofGenMode::Execute => vm.run(false),
ProofGenMode::Prove => vm.run(true),
ProofGenMode::Prove(proof_sampling_number) => {
let with_prove = proof_sampling_number == 0
|| rand::thread_rng().gen_range(0..proof_sampling_number) == 0;
vm.run(with_prove)
}
}
}
Loading