#![allow(clippy::type_complexity)]
#![allow(clippy::boxed_local)]
use crate::{
column_env::ColumnEnvironment,
expr::E,
logup,
logup::{prover::Env, LookupProof, LookupTableID},
proof::{Proof, ProofCommitments, ProofEvaluations, ProofInputs},
witness::Witness,
MAX_SUPPORTED_DEGREE,
};
use ark_ff::{Field, One, Zero};
use ark_poly::{
univariate::DensePolynomial, EvaluationDomain, Evaluations, Polynomial,
Radix2EvaluationDomain as R2D,
};
use kimchi::{
circuits::{
berkeley_columns::BerkeleyChallenges,
domains::EvaluationDomains,
expr::{l0_1, Constants, Expr},
},
curve::KimchiCurve,
groupmap::GroupMap,
plonk_sponge::FrSponge,
proof::PointEvaluations,
};
use mina_poseidon::{sponge::ScalarChallenge, FqSponge};
use o1_utils::ExtendedDensePolynomial;
use poly_commitment::{
commitment::{absorb_commitment, PolyComm},
utils::DensePolynomialOrEvaluations,
OpenProof, SRS,
};
use rand::{CryptoRng, RngCore};
use rayon::iter::{IntoParallelIterator, ParallelIterator};
use thiserror::Error;
#[derive(Error, Debug, Clone)]
pub enum ProverError {
#[error("the proof could not be constructed: {0}")]
Generic(&'static str),
#[error("the provided (witness) constraints was not satisfied: {0}")]
ConstraintNotSatisfied(String),
#[error("the provided (witness) constraint has degree {0} > allowed {1}; expr: {2}")]
ConstraintDegreeTooHigh(u64, u64, String),
}
pub fn prove<
G: KimchiCurve,
OpeningProof: OpenProof<G>,
EFqSponge: Clone + FqSponge<G::BaseField, G, G::ScalarField>,
EFrSponge: FrSponge<G::ScalarField>,
RNG,
const N_WIT: usize,
const N_REL: usize,
const N_DSEL: usize,
const N_FSEL: usize,
ID: LookupTableID,
>(
domain: EvaluationDomains<G::ScalarField>,
srs: &OpeningProof::SRS,
constraints: &[E<G::ScalarField>],
fixed_selectors: Box<[Vec<G::ScalarField>; N_FSEL]>,
inputs: ProofInputs<N_WIT, G::ScalarField, ID>,
rng: &mut RNG,
) -> Result<Proof<N_WIT, N_REL, N_DSEL, N_FSEL, G, OpeningProof, ID>, ProverError>
where
OpeningProof::SRS: Sync,
RNG: RngCore + CryptoRng,
{
let group_map = G::Map::setup();
let mut fq_sponge = EFqSponge::new(G::other_curve_sponge_params());
let fixed_selectors_evals_d1: Box<[Evaluations<G::ScalarField, R2D<G::ScalarField>>; N_FSEL]> =
o1_utils::array::vec_to_boxed_array(
fixed_selectors
.into_par_iter()
.map(|evals| Evaluations::from_vec_and_domain(evals, domain.d1))
.collect(),
);
let fixed_selectors_polys: Box<[DensePolynomial<G::ScalarField>; N_FSEL]> =
o1_utils::array::vec_to_boxed_array(
fixed_selectors_evals_d1
.into_par_iter()
.map(|evals| evals.interpolate())
.collect(),
);
let fixed_selectors_comms: Box<[PolyComm<G>; N_FSEL]> = {
let comm = |poly: &DensePolynomial<G::ScalarField>| srs.commit_non_hiding(poly, 1);
o1_utils::array::vec_to_boxed_array(
fixed_selectors_polys
.as_ref()
.into_par_iter()
.map(comm)
.collect(),
)
};
(fixed_selectors_comms)
.into_iter()
.for_each(|comm| absorb_commitment(&mut fq_sponge, &comm));
let witness_evals_d1: Witness<N_WIT, Evaluations<G::ScalarField, R2D<G::ScalarField>>> = inputs
.evaluations
.into_par_iter()
.map(|evals| {
Evaluations::<G::ScalarField, R2D<G::ScalarField>>::from_vec_and_domain(
evals, domain.d1,
)
})
.collect::<Witness<N_WIT, Evaluations<G::ScalarField, R2D<G::ScalarField>>>>();
let witness_polys: Witness<N_WIT, DensePolynomial<G::ScalarField>> = {
let interpolate =
|evals: Evaluations<G::ScalarField, R2D<G::ScalarField>>| evals.interpolate();
witness_evals_d1
.into_par_iter()
.map(interpolate)
.collect::<Witness<N_WIT, DensePolynomial<G::ScalarField>>>()
};
let witness_comms: Witness<N_WIT, PolyComm<G>> = {
let blinders = PolyComm {
chunks: vec![G::ScalarField::one()],
};
let comm = {
|poly: &DensePolynomial<G::ScalarField>| {
let comm = srs.commit_custom(poly, 1, &blinders).unwrap();
comm.commitment
}
};
(&witness_polys)
.into_par_iter()
.map(comm)
.collect::<Witness<N_WIT, PolyComm<G>>>()
};
(&witness_comms)
.into_iter()
.for_each(|comm| absorb_commitment(&mut fq_sponge, comm));
let lookup_env = if !inputs.logups.is_empty() {
Some(Env::create::<OpeningProof, EFqSponge>(
inputs.logups,
domain,
&mut fq_sponge,
srs,
))
} else {
None
};
let max_degree = {
if lookup_env.is_none() {
constraints
.iter()
.map(|expr| expr.degree(1, 0))
.max()
.unwrap_or(0)
} else {
8
}
};
let logup_comms = Option::map(lookup_env.as_ref(), |lookup_env| LookupProof {
m: lookup_env.lookup_counters_comm_d1.clone(),
h: lookup_env.lookup_terms_comms_d1.clone(),
sum: lookup_env.lookup_aggregation_comm_d1.clone(),
fixed_tables: lookup_env.fixed_lookup_tables_comms_d1.clone(),
});
let domain_eval = if max_degree <= 4 {
domain.d4
} else if max_degree as usize <= MAX_SUPPORTED_DEGREE {
domain.d8
} else {
panic!("We do support constraints up to {:?}", MAX_SUPPORTED_DEGREE)
};
let witness_evals: Witness<N_WIT, Evaluations<G::ScalarField, R2D<G::ScalarField>>> = {
(&witness_polys)
.into_par_iter()
.map(|evals| evals.evaluate_over_domain_by_ref(domain_eval))
.collect::<Witness<N_WIT, Evaluations<G::ScalarField, R2D<G::ScalarField>>>>()
};
let fixed_selectors_evals: Box<[Evaluations<G::ScalarField, R2D<G::ScalarField>>; N_FSEL]> = {
o1_utils::array::vec_to_boxed_array(
(fixed_selectors_polys.as_ref())
.into_par_iter()
.map(|evals| evals.evaluate_over_domain_by_ref(domain_eval))
.collect(),
)
};
let (_, endo_r) = G::endos();
let alpha: G::ScalarField = fq_sponge.challenge();
let zk_rows = 0;
let column_env: ColumnEnvironment<'_, N_WIT, N_REL, N_DSEL, N_FSEL, _, _> = {
let challenges = BerkeleyChallenges {
alpha,
beta: Option::map(lookup_env.as_ref(), |x| x.beta).unwrap_or(G::ScalarField::zero()),
gamma: G::ScalarField::zero(),
joint_combiner: Option::map(lookup_env.as_ref(), |x| x.joint_combiner)
.unwrap_or(G::ScalarField::zero()),
};
ColumnEnvironment {
constants: Constants {
endo_coefficient: *endo_r,
mds: &G::sponge_params().mds,
zk_rows,
},
challenges,
witness: &witness_evals,
fixed_selectors: &fixed_selectors_evals,
l0_1: l0_1(domain.d1),
lookup: Option::map(lookup_env.as_ref(), |lookup_env| {
logup::prover::QuotientPolynomialEnvironment {
lookup_terms_evals_d8: &lookup_env.lookup_terms_evals_d8,
lookup_aggregation_evals_d8: &lookup_env.lookup_aggregation_evals_d8,
lookup_counters_evals_d8: &lookup_env.lookup_counters_evals_d8,
fixed_tables_evals_d8: &lookup_env.fixed_lookup_tables_evals_d8,
}
}),
domain,
}
};
let quotient_poly: DensePolynomial<G::ScalarField> = {
let mut last_constraint_failed = None;
for expr in constraints.iter() {
let fail_q_division =
ProverError::ConstraintNotSatisfied(format!("Unsatisfied expression: {:}", expr));
let (_, res) = expr
.evaluations(&column_env)
.interpolate_by_ref()
.divide_by_vanishing_poly(domain.d1)
.ok_or(fail_q_division.clone())?;
if !res.is_zero() {
eprintln!("Unsatisfied expression: {}", expr);
last_constraint_failed = Some(expr.clone());
}
}
if let Some(expr) = last_constraint_failed {
return Err(ProverError::ConstraintNotSatisfied(format!(
"Unsatisfied expression: {:}",
expr
)));
}
let combined_expr =
Expr::combine_constraints(0..(constraints.len() as u32), constraints.to_vec());
let expr_evaluation: Evaluations<G::ScalarField, R2D<G::ScalarField>> =
combined_expr.evaluations(&column_env);
let expr_evaluation_interpolated = expr_evaluation.interpolate();
let fail_final_q_division = || {
panic!("Division by vanishing poly must not fail at this point, we checked it before")
};
let (quotient, res) = expr_evaluation_interpolated
.divide_by_vanishing_poly(domain.d1)
.unwrap_or_else(fail_final_q_division);
if !res.is_zero() {
fail_final_q_division();
}
quotient
};
let num_chunks: usize = if max_degree == 1 {
1
} else {
(max_degree - 1) as usize
};
let t_comm = srs.commit_non_hiding("ient_poly, num_chunks);
absorb_commitment(&mut fq_sponge, &t_comm);
let zeta_chal = ScalarChallenge(fq_sponge.challenge());
let zeta = zeta_chal.to_field(endo_r);
let omega = domain.d1.group_gen;
let zeta_omega = zeta * omega;
let witness_evals: Witness<N_WIT, PointEvaluations<_>> = {
let eval = |p: &DensePolynomial<_>| PointEvaluations {
zeta: p.evaluate(&zeta),
zeta_omega: p.evaluate(&zeta_omega),
};
(&witness_polys)
.into_par_iter()
.map(eval)
.collect::<Witness<N_WIT, PointEvaluations<_>>>()
};
let fixed_selectors_evals: Box<[PointEvaluations<_>; N_FSEL]> = {
let eval = |p: &DensePolynomial<_>| PointEvaluations {
zeta: p.evaluate(&zeta),
zeta_omega: p.evaluate(&zeta_omega),
};
o1_utils::array::vec_to_boxed_array(
fixed_selectors_polys
.as_ref()
.into_par_iter()
.map(eval)
.collect::<_>(),
)
};
let logup_evals = lookup_env.as_ref().map(|lookup_env| LookupProof {
m: lookup_env
.lookup_counters_poly_d1
.iter()
.map(|(id, polys)| {
(
*id,
polys
.iter()
.map(|poly| {
let zeta = poly.evaluate(&zeta);
let zeta_omega = poly.evaluate(&zeta_omega);
PointEvaluations { zeta, zeta_omega }
})
.collect(),
)
})
.collect(),
h: lookup_env
.lookup_terms_poly_d1
.iter()
.map(|(id, polys)| {
let polys_evals: Vec<_> = polys
.iter()
.map(|poly| PointEvaluations {
zeta: poly.evaluate(&zeta),
zeta_omega: poly.evaluate(&zeta_omega),
})
.collect();
(*id, polys_evals)
})
.collect(),
sum: PointEvaluations {
zeta: lookup_env.lookup_aggregation_poly_d1.evaluate(&zeta),
zeta_omega: lookup_env.lookup_aggregation_poly_d1.evaluate(&zeta_omega),
},
fixed_tables: {
lookup_env
.fixed_lookup_tables_poly_d1
.iter()
.map(|(id, poly)| {
let zeta = poly.evaluate(&zeta);
let zeta_omega = poly.evaluate(&zeta_omega);
(*id, PointEvaluations { zeta, zeta_omega })
})
.collect()
},
});
let fq_sponge_before_evaluations = fq_sponge.clone();
let mut fr_sponge = EFrSponge::new(G::sponge_params());
fr_sponge.absorb(&fq_sponge.digest());
for PointEvaluations { zeta, zeta_omega } in (&witness_evals).into_iter() {
fr_sponge.absorb(zeta);
fr_sponge.absorb(zeta_omega);
}
for PointEvaluations { zeta, zeta_omega } in fixed_selectors_evals.as_ref().iter() {
fr_sponge.absorb(zeta);
fr_sponge.absorb(zeta_omega);
}
if lookup_env.is_some() {
for PointEvaluations { zeta, zeta_omega } in logup_evals.as_ref().unwrap().into_iter() {
fr_sponge.absorb(zeta);
fr_sponge.absorb(zeta_omega);
}
}
let ft: DensePolynomial<G::ScalarField> = {
let evaluation_point_to_domain_size = zeta.pow([domain.d1.size]);
let t_chunked: DensePolynomial<G::ScalarField> = quotient_poly
.to_chunked_polynomial(num_chunks, domain.d1.size as usize)
.linearize(evaluation_point_to_domain_size);
let minus_vanishing_poly_at_zeta = -domain.d1.vanishing_polynomial().evaluate(&zeta);
t_chunked.scale(minus_vanishing_poly_at_zeta)
};
let ft_eval1 = ft.evaluate(&zeta_omega);
fr_sponge.absorb(&ft_eval1);
let v_chal = fr_sponge.challenge();
let v = v_chal.to_field(endo_r);
let u_chal = fr_sponge.challenge();
let u = u_chal.to_field(endo_r);
let coefficients_form = DensePolynomialOrEvaluations::DensePolynomial;
let non_hiding = |n_chunks| PolyComm {
chunks: vec![G::ScalarField::zero(); n_chunks],
};
let hiding = |n_chunks| PolyComm {
chunks: vec![G::ScalarField::one(); n_chunks],
};
let mut polynomials: Vec<_> = (&witness_polys)
.into_par_iter()
.map(|poly| (coefficients_form(poly), hiding(1)))
.collect();
polynomials.extend(
fixed_selectors_polys
.as_ref()
.into_par_iter()
.map(|poly| (coefficients_form(poly), non_hiding(1)))
.collect::<Vec<_>>(),
);
if let Some(ref lookup_env) = lookup_env {
polynomials.extend(
lookup_env
.lookup_counters_poly_d1
.values()
.flat_map(|polys| {
polys
.iter()
.map(|poly| (coefficients_form(poly), non_hiding(1)))
.collect::<Vec<_>>()
})
.collect::<Vec<_>>(),
);
polynomials.extend({
let polys = lookup_env.lookup_terms_poly_d1.values().map(|polys| {
polys
.iter()
.map(|poly| (coefficients_form(poly), non_hiding(1)))
.collect::<Vec<_>>()
});
let polys: Vec<_> = polys.flatten().collect();
polys
});
polynomials.push((
coefficients_form(&lookup_env.lookup_aggregation_poly_d1),
non_hiding(1),
));
polynomials.extend(
lookup_env
.fixed_lookup_tables_poly_d1
.values()
.map(|poly| (coefficients_form(poly), non_hiding(1)))
.collect::<Vec<_>>(),
);
}
polynomials.push((coefficients_form(&ft), non_hiding(1)));
let opening_proof = OpenProof::open::<_, _, R2D<G::ScalarField>>(
srs,
&group_map,
polynomials.as_slice(),
&[zeta, zeta_omega],
v,
u,
fq_sponge_before_evaluations,
rng,
);
let proof_evals: ProofEvaluations<N_WIT, N_REL, N_DSEL, N_FSEL, G::ScalarField, ID> = {
ProofEvaluations {
witness_evals,
fixed_selectors_evals,
logup_evals,
ft_eval1,
}
};
Ok(Proof {
proof_comms: ProofCommitments {
witness_comms,
logup_comms,
t_comm,
},
proof_evals,
opening_proof,
})
}