#![no_std]
extern crate alloc;
#[cfg(feature = "std")]
extern crate std;
use alloc::{string::ToString, vec, vec::Vec};
use ::serde::Serialize;
use miden_air::{MidenMultiAir, ProverStatement, Statement};
use miden_core::{Felt, field::QuadFelt, utils::RowMajorMatrix};
use miden_crypto::stark::{
ProverInstance, StarkConfig,
lmcs::Lmcs,
proof::{StarkOutput, StarkProofData},
};
use miden_processor::{
FastProcessor, Program,
trace::{ExecutionTrace, build_trace},
};
use serde_wincode::SerdeCompat;
use tracing::instrument;
mod proving_options;
pub use miden_air::{DeserializationError, MidenAir, PublicInputs, config};
pub use miden_core::proof::{ExecutionProof, HashFunction};
pub use miden_processor::{
ExecutionError, ExecutionOptions, ExecutionOutput, FutureMaybeSend, Host, InputError,
ProgramInfo, StackInputs, StackOutputs, SyncHost, TraceBuildInputs, TraceGenerationContext,
Word, advice::AdviceInputs, crypto, field, serde, utils,
};
pub use proving_options::ProvingOptions;
#[derive(Debug)]
pub struct TraceProvingInputs {
trace_inputs: TraceBuildInputs,
options: ProvingOptions,
}
impl TraceProvingInputs {
pub fn new(trace_inputs: TraceBuildInputs, options: ProvingOptions) -> Self {
Self { trace_inputs, options }
}
pub fn into_parts(self) -> (TraceBuildInputs, ProvingOptions) {
(self.trace_inputs, self.options)
}
}
#[instrument("prove_program", skip_all)]
pub async fn prove(
program: &Program,
stack_inputs: StackInputs,
advice_inputs: AdviceInputs,
host: &mut impl Host,
execution_options: ExecutionOptions,
proving_options: ProvingOptions,
) -> Result<(StackOutputs, ExecutionProof), ExecutionError> {
let processor = FastProcessor::new_with_options(stack_inputs, advice_inputs, execution_options)
.map_err(ExecutionError::advice_error_no_context)?;
let trace_inputs = processor.execute_trace_inputs(program, host).await?;
prove_from_trace_sync(TraceProvingInputs::new(trace_inputs, proving_options))
}
#[instrument("prove_program_sync", skip_all)]
pub fn prove_sync(
program: &Program,
stack_inputs: StackInputs,
advice_inputs: AdviceInputs,
host: &mut impl SyncHost,
execution_options: ExecutionOptions,
proving_options: ProvingOptions,
) -> Result<(StackOutputs, ExecutionProof), ExecutionError> {
let processor = FastProcessor::new_with_options(stack_inputs, advice_inputs, execution_options)
.map_err(ExecutionError::advice_error_no_context)?;
let trace_inputs = processor.execute_trace_inputs_sync(program, host)?;
prove_from_trace_sync(TraceProvingInputs::new(trace_inputs, proving_options))
}
#[instrument("prove_trace_sync", skip_all)]
pub fn prove_from_trace_sync(
inputs: TraceProvingInputs,
) -> Result<(StackOutputs, ExecutionProof), ExecutionError> {
let (trace_inputs, options) = inputs.into_parts();
let trace = build_trace(trace_inputs)?;
prove_execution_trace(trace, options)
}
fn prove_execution_trace(
trace: ExecutionTrace,
options: ProvingOptions,
) -> Result<(StackOutputs, ExecutionProof), ExecutionError> {
tracing::event!(
tracing::Level::INFO,
"Generated execution trace of {} columns and {} steps (padded from {})",
miden_air::trace::TRACE_WIDTH,
trace.trace_len_summary().padded_trace_len(),
trace.trace_len_summary().trace_len()
);
let stack_outputs = *trace.stack_outputs();
let precompile_requests = trace.precompile_requests().to_vec();
let hash_fn = options.hash_fn();
let (public_values, kernel_felts) = trace.public_inputs().to_air_inputs();
let (core_matrix, chiplets_matrix) = {
let _span = tracing::info_span!("to_core_chiplets_matrices").entered();
trace.into_core_chiplets_matrices()
};
let params = config::pcs_params();
let proof_bytes = match hash_fn {
HashFunction::Blake3_256 => {
let config = config::blake3_256_config(params);
prove_stark(&config, core_matrix, chiplets_matrix, &public_values, &kernel_felts)
},
HashFunction::Keccak => {
let config = config::keccak_config(params);
prove_stark(&config, core_matrix, chiplets_matrix, &public_values, &kernel_felts)
},
HashFunction::Rpo256 => {
let config = config::rpo_config(params);
prove_stark(&config, core_matrix, chiplets_matrix, &public_values, &kernel_felts)
},
HashFunction::Poseidon2 => {
let config = config::poseidon2_config(params);
prove_stark(&config, core_matrix, chiplets_matrix, &public_values, &kernel_felts)
},
HashFunction::Rpx256 => {
let config = config::rpx_config(params);
prove_stark(&config, core_matrix, chiplets_matrix, &public_values, &kernel_felts)
},
}?;
let proof = ExecutionProof::new(proof_bytes, hash_fn, precompile_requests);
Ok((stack_outputs, proof))
}
pub fn prove_stark<SC>(
config: &SC,
core_trace: RowMajorMatrix<Felt>,
chiplets_trace: RowMajorMatrix<Felt>,
public_values: &[Felt],
kernel_felts: &[Felt],
) -> Result<Vec<u8>, ExecutionError>
where
SC: StarkConfig<Felt, QuadFelt>,
<SC::Lmcs as Lmcs>::Commitment: Serialize,
{
let mut challenger = config.challenger();
config::observe_protocol_params(&mut challenger);
let statement =
Statement::new(MidenMultiAir::new(), public_values.to_vec(), kernel_felts.to_vec())
.map_err(|e| ExecutionError::ProvingError(e.to_string()))?;
let prover_statement = ProverStatement::new(statement, vec![core_trace, chiplets_trace])
.map_err(|e| ExecutionError::ProvingError(e.to_string()))?;
let output: StarkOutput<Felt, QuadFelt, SC> =
ProverInstance::new(config, &prover_statement, None)
.map_err(|e| ExecutionError::ProvingError(e.to_string()))?
.prove(challenger)
.map_err(|e| ExecutionError::ProvingError(e.to_string()))?;
let proof_encoding_config = wincode::config::Configuration::default();
let proof_bytes =
<SerdeCompat<StarkProofData<Felt, QuadFelt, SC>> as wincode::config::Serialize<_>>::serialize(
&output.proof,
proof_encoding_config,
)
.map_err(|e| ExecutionError::ProvingError(e.to_string()))?;
Ok(proof_bytes)
}