use std::ops::Index;
use crate::plonkish_lang::{CombinableEvals, PlonkishChallenge, PlonkishWitnessGeneric};
use ark_ec::AffineRepr;
use ark_ff::Field;
use ark_poly::{Evaluations, Radix2EvaluationDomain as R2D};
use folding::{
columns::ExtendedFoldingColumn,
eval_leaf::EvalLeaf,
expressions::{ExpExtension, FoldingCompatibleExprInner, FoldingExp},
instance_witness::ExtendedWitness,
Alphas, FoldingCompatibleExpr, FoldingConfig,
};
use kimchi::{
self,
circuits::{expr::Variable, gate::CurrOrNext},
curve::KimchiCurve,
};
use kimchi_msm::columns::Column as GenericColumn;
use strum::EnumCount;
#[derive(Clone)]
pub struct GenericVecStructure<G: KimchiCurve>(pub Vec<Vec<G::ScalarField>>);
impl<G: KimchiCurve> Index<GenericColumn<usize>> for GenericVecStructure<G> {
type Output = [G::ScalarField];
fn index(&self, index: GenericColumn<usize>) -> &Self::Output {
match index {
GenericColumn::FixedSelector(i) => &self.0[i],
_ => panic!("should not happen"),
}
}
}
pub struct GenericEvalEnv<
Curve: KimchiCurve,
const N_COL: usize,
const N_FSEL: usize,
Eval: CombinableEvals<Curve::ScalarField>,
> {
pub ext_witness:
ExtendedWitness<Curve, PlonkishWitnessGeneric<N_COL, N_FSEL, Curve::ScalarField, Eval>>,
pub alphas: Alphas<Curve::ScalarField>,
pub challenges: [Curve::ScalarField; PlonkishChallenge::COUNT],
pub error_vec: Eval,
pub u: Curve::ScalarField,
}
pub type SimpleEvalEnv<Curve, const N_COL: usize, const N_FSEL: usize> = GenericEvalEnv<
Curve,
N_COL,
N_FSEL,
Evaluations<<Curve as AffineRepr>::ScalarField, R2D<<Curve as AffineRepr>::ScalarField>>,
>;
impl<
Curve: KimchiCurve,
const N_COL: usize,
const N_FSEL: usize,
Evals: CombinableEvals<Curve::ScalarField>,
> GenericEvalEnv<Curve, N_COL, N_FSEL, Evals>
{
fn challenge(&self, challenge: PlonkishChallenge) -> Curve::ScalarField {
match challenge {
PlonkishChallenge::Beta => self.challenges[0],
PlonkishChallenge::Gamma => self.challenges[1],
PlonkishChallenge::JointCombiner => self.challenges[2],
}
}
pub fn process_extended_folding_column<
FC: FoldingConfig<Column = GenericColumn<usize>, Curve = Curve, Challenge = PlonkishChallenge>,
>(
&self,
col: &ExtendedFoldingColumn<FC>,
) -> EvalLeaf<Curve::ScalarField> {
use EvalLeaf::Col;
use ExtendedFoldingColumn::*;
match col {
Inner(Variable { col, row }) => {
let wit = match row {
CurrOrNext::Curr => &self.ext_witness.witness,
CurrOrNext::Next => panic!("not implemented"),
};
Col(&wit[*col])
},
WitnessExtended(i) => Col(&self.ext_witness.extended.get(i).unwrap().evals),
Error => panic!("shouldn't happen"),
Constant(c) => EvalLeaf::Const(*c),
Challenge(chall) => EvalLeaf::Const(self.challenge(*chall)),
Alpha(i) => {
let alpha = self.alphas.get(*i).expect("alpha not present");
EvalLeaf::Const(alpha)
}
Selector(_s) => unimplemented!("Selector not implemented for FoldingEnvironment. No selectors are supposed to be used when it is Plonkish relations."),
}
}
pub fn eval_naive_fexpr<
'a,
FC: FoldingConfig<Column = GenericColumn<usize>, Curve = Curve, Challenge = PlonkishChallenge>,
>(
&'a self,
exp: &FoldingExp<FC>,
) -> EvalLeaf<'a, Curve::ScalarField> {
use FoldingExp::*;
match exp {
Atom(column) => self.process_extended_folding_column(column),
Double(e) => {
let col = self.eval_naive_fexpr(e);
col.map(Field::double, |f| {
Field::double_in_place(f);
})
}
Square(e) => {
let col = self.eval_naive_fexpr(e);
col.map(Field::square, |f| {
Field::square_in_place(f);
})
}
Add(e1, e2) => self.eval_naive_fexpr(e1) + self.eval_naive_fexpr(e2),
Sub(e1, e2) => self.eval_naive_fexpr(e1) - self.eval_naive_fexpr(e2),
Mul(e1, e2) => self.eval_naive_fexpr(e1) * self.eval_naive_fexpr(e2),
Pow(_e, _i) => panic!("We're not supposed to use this"),
}
}
pub fn eval_naive_fcompat<
'a,
FC: FoldingConfig<Column = GenericColumn<usize>, Curve = Curve, Challenge = PlonkishChallenge>,
>(
&'a self,
exp: &FoldingCompatibleExpr<FC>,
) -> EvalLeaf<'a, Curve::ScalarField> where {
use FoldingCompatibleExpr::*;
match exp {
Atom(column) => {
use FoldingCompatibleExprInner::*;
match column {
Cell(Variable { col, row }) => {
let wit = match row {
CurrOrNext::Curr => &self.ext_witness.witness,
CurrOrNext::Next => panic!("not implemented"),
};
EvalLeaf::Col(&wit[*col])
}
Challenge(chal) => EvalLeaf::Const(self.challenge(*chal)),
Constant(c) => EvalLeaf::Const(*c),
Extensions(ext) => {
use ExpExtension::*;
match ext {
U => EvalLeaf::Const(self.u),
Error => EvalLeaf::Col(self.error_vec.e_as_slice()),
ExtendedWitness(i) => {
EvalLeaf::Col(&self.ext_witness.extended.get(i).unwrap().evals)
}
Alpha(i) => EvalLeaf::Const(self.alphas.get(*i).unwrap()),
Selector(_sel) => panic!("No selectors supported yet"),
}
}
}
}
Double(e) => {
let col = self.eval_naive_fcompat(e);
col.map(Field::double, |f| {
Field::double_in_place(f);
})
}
Square(e) => {
let col = self.eval_naive_fcompat(e);
col.map(Field::square, |f| {
Field::square_in_place(f);
})
}
Add(e1, e2) => self.eval_naive_fcompat(e1) + self.eval_naive_fcompat(e2),
Sub(e1, e2) => self.eval_naive_fcompat(e1) - self.eval_naive_fcompat(e2),
Mul(e1, e2) => self.eval_naive_fcompat(e1) * self.eval_naive_fcompat(e2),
Pow(_e, _i) => panic!("We're not supposed to use this"),
}
}
}