Changes: - get rid of `extract_r1cs_and_z` and `extract_z` - move `extract_r1cs` and `extract_w_x` from `frontend/arkworks` into `r1cs.rs` The reasoning: they are not methods needed for the Frontend interface, but only needed internally for the folding scheme to extract values from the AugmentedF circuit and similar. - set the `FCircuit` as the trait for the `src/frontend` - remove the `frontend/arkworks` since the `FCircuit` trait can be directly implemented without a middle layer - reorganize test circuits into `src/frontend/mod.rs`, updating them into `CubicFCircuit`: the typical x^3+x+5=y circuit `CustomFCircuit`: a circuit in which you can specify the number of constraints that it will take where both fulfill the `FCircuit` trait, and they are used for different tests being folded.main
@ -1,122 +0,0 @@ |
|||
/// arkworks frontend
|
|||
use ark_ff::PrimeField;
|
|||
use ark_relations::r1cs::ConstraintSystem;
|
|||
|
|||
use crate::ccs::r1cs::R1CS;
|
|||
use crate::utils::vec::SparseMatrix;
|
|||
|
|||
/// extracts arkworks ConstraintSystem matrices into crate::utils::vec::SparseMatrix format, and
|
|||
/// extracts public inputs and witness into z vector. Returns a tuple containing (R1CS, z).
|
|||
pub fn extract_r1cs_and_z<F: PrimeField>(cs: &ConstraintSystem<F>) -> (R1CS<F>, Vec<F>) {
|
|||
let r1cs = extract_r1cs(cs);
|
|||
|
|||
// z = (1, x, w)
|
|||
let z = extract_z(cs);
|
|||
|
|||
(r1cs, z)
|
|||
}
|
|||
|
|||
/// extracts arkworks ConstraintSystem matrices into crate::utils::vec::SparseMatrix format as R1CS
|
|||
/// struct.
|
|||
pub fn extract_r1cs<F: PrimeField>(cs: &ConstraintSystem<F>) -> R1CS<F> {
|
|||
let m = cs.to_matrices().unwrap();
|
|||
|
|||
let n_rows = cs.num_constraints;
|
|||
let n_cols = cs.num_instance_variables + cs.num_witness_variables; // cs.num_instance_variables already counts the 1
|
|||
|
|||
let A = SparseMatrix::<F> {
|
|||
n_rows,
|
|||
n_cols,
|
|||
coeffs: m.a,
|
|||
};
|
|||
let B = SparseMatrix::<F> {
|
|||
n_rows,
|
|||
n_cols,
|
|||
coeffs: m.b,
|
|||
};
|
|||
let C = SparseMatrix::<F> {
|
|||
n_rows,
|
|||
n_cols,
|
|||
coeffs: m.c,
|
|||
};
|
|||
|
|||
R1CS::<F> {
|
|||
l: cs.num_instance_variables - 1, // -1 to substract the first '1'
|
|||
A,
|
|||
B,
|
|||
C,
|
|||
}
|
|||
}
|
|||
|
|||
/// extracts public inputs and witness into z vector from arkworks ConstraintSystem.
|
|||
pub fn extract_z<F: PrimeField>(cs: &ConstraintSystem<F>) -> Vec<F> {
|
|||
// z = (1, x, w)
|
|||
[
|
|||
// 1 is already included in cs.instance_assignment
|
|||
cs.instance_assignment.clone(),
|
|||
cs.witness_assignment.clone(),
|
|||
]
|
|||
.concat()
|
|||
}
|
|||
|
|||
/// extracts the witness and the public inputs from arkworks ConstraintSystem.
|
|||
pub fn extract_w_x<F: PrimeField>(cs: &ConstraintSystem<F>) -> (Vec<F>, Vec<F>) {
|
|||
(
|
|||
cs.witness_assignment.clone(),
|
|||
// skip the first element which is '1'
|
|||
cs.instance_assignment[1..].to_vec(),
|
|||
)
|
|||
}
|
|||
|
|||
#[cfg(test)]
|
|||
pub mod tests {
|
|||
use super::*;
|
|||
use ark_ff::PrimeField;
|
|||
use ark_pallas::Fr;
|
|||
use ark_r1cs_std::{alloc::AllocVar, eq::EqGadget, fields::fp::FpVar};
|
|||
use ark_relations::r1cs::{
|
|||
ConstraintSynthesizer, ConstraintSystem, ConstraintSystemRef, SynthesisError,
|
|||
};
|
|||
|
|||
// TestCircuit implements the R1CS for: x^3 + x + 5 = y (example from article
|
|||
// https://www.vitalik.ca/general/2016/12/10/qap.html )
|
|||
#[derive(Clone, Copy, Debug)]
|
|||
pub struct TestCircuit<F: PrimeField> {
|
|||
pub x: F,
|
|||
pub y: F,
|
|||
}
|
|||
impl<F: PrimeField> ConstraintSynthesizer<F> for TestCircuit<F> {
|
|||
fn generate_constraints(self, cs: ConstraintSystemRef<F>) -> Result<(), SynthesisError> {
|
|||
let x = FpVar::<F>::new_input(cs.clone(), || Ok(self.x))?;
|
|||
let y = FpVar::<F>::new_witness(cs.clone(), || Ok(self.y))?;
|
|||
let five = FpVar::<F>::new_constant(cs.clone(), F::from(5u32))?;
|
|||
|
|||
let comp_y = (&x * &x * &x) + &x + &five;
|
|||
comp_y.enforce_equal(&y)?;
|
|||
Ok(())
|
|||
}
|
|||
}
|
|||
|
|||
#[test]
|
|||
fn test_cs_to_matrices() {
|
|||
let cs = ConstraintSystem::<Fr>::new_ref();
|
|||
|
|||
let x = Fr::from(5_u32);
|
|||
let y = x * x * x + x + Fr::from(5_u32);
|
|||
|
|||
let test_circuit = TestCircuit::<Fr> { x, y };
|
|||
test_circuit.generate_constraints(cs.clone()).unwrap();
|
|||
cs.finalize();
|
|||
assert!(cs.is_satisfied().unwrap());
|
|||
|
|||
let cs = cs.into_inner().unwrap();
|
|||
|
|||
// ensure that num_instance_variables is 2: 1 + 1 public input
|
|||
assert_eq!(cs.num_instance_variables, 2);
|
|||
|
|||
let (r1cs, z) = extract_r1cs_and_z::<Fr>(&cs);
|
|||
r1cs.check_relation(&z).unwrap();
|
|||
// ensure that number of public inputs (l) is 1
|
|||
assert_eq!(r1cs.l, 1);
|
|||
}
|
|||
}
|
@ -1,2 +1,169 @@ |
|||
pub mod arkworks;
|
|||
pub mod circom;
|
|||
|
|||
use crate::Error;
|
|||
use ark_ff::PrimeField;
|
|||
use ark_r1cs_std::fields::fp::FpVar;
|
|||
use ark_relations::r1cs::{ConstraintSystemRef, SynthesisError};
|
|||
use ark_std::fmt::Debug;
|
|||
|
|||
/// FCircuit defines the trait of the circuit of the F function, which is the one being folded (ie.
|
|||
/// inside the agmented F' function).
|
|||
pub trait FCircuit<F: PrimeField>: Clone + Copy + Debug {
|
|||
type Params: Debug;
|
|||
|
|||
/// returns a new FCircuit instance
|
|||
fn new(params: Self::Params) -> Self;
|
|||
|
|||
/// computes the next state values in place, assigning z_{i+1} into z_i, and computing the new
|
|||
/// z_{i+1}
|
|||
fn step_native(
|
|||
// this method uses self, so that each FCircuit implementation (and different frontends)
|
|||
// can hold a state if needed to store data to compute the next state.
|
|||
self,
|
|||
z_i: Vec<F>,
|
|||
) -> Result<Vec<F>, Error>;
|
|||
|
|||
/// generates the constraints for the step of F for the given z_i
|
|||
fn generate_step_constraints(
|
|||
// this method uses self, so that each FCircuit implementation (and different frontends)
|
|||
// can hold a state if needed to store data to generate the constraints.
|
|||
self,
|
|||
cs: ConstraintSystemRef<F>,
|
|||
z_i: Vec<FpVar<F>>,
|
|||
) -> Result<Vec<FpVar<F>>, SynthesisError>;
|
|||
}
|
|||
|
|||
#[cfg(test)]
|
|||
pub mod tests {
|
|||
use super::*;
|
|||
use ark_pallas::Fr;
|
|||
use ark_r1cs_std::{alloc::AllocVar, eq::EqGadget};
|
|||
use ark_relations::r1cs::{
|
|||
ConstraintSynthesizer, ConstraintSystem, ConstraintSystemRef, SynthesisError,
|
|||
};
|
|||
use core::marker::PhantomData;
|
|||
|
|||
/// CubicFCircuit is a struct that implements the FCircuit trait, for the R1CS example circuit
|
|||
/// from https://www.vitalik.ca/general/2016/12/10/qap.html, which checks `x^3 + x + 5 = y`. It
|
|||
/// has 2 public inputs which are used as the state. `z_i` is used as `x`, and `z_{i+1}` is
|
|||
/// used as `y`, and at the next step, `z_{i+1}` will be assigned to `z_i`, and a new `z+{i+1}`
|
|||
/// will be computted.
|
|||
#[derive(Clone, Copy, Debug)]
|
|||
pub struct CubicFCircuit<F: PrimeField> {
|
|||
_f: PhantomData<F>,
|
|||
}
|
|||
impl<F: PrimeField> FCircuit<F> for CubicFCircuit<F> {
|
|||
type Params = ();
|
|||
fn new(_params: Self::Params) -> Self {
|
|||
Self { _f: PhantomData }
|
|||
}
|
|||
fn step_native(self, z_i: Vec<F>) -> Result<Vec<F>, Error> {
|
|||
Ok(vec![z_i[0] * z_i[0] * z_i[0] + z_i[0] + F::from(5_u32)])
|
|||
}
|
|||
fn generate_step_constraints(
|
|||
self,
|
|||
cs: ConstraintSystemRef<F>,
|
|||
z_i: Vec<FpVar<F>>,
|
|||
) -> Result<Vec<FpVar<F>>, SynthesisError> {
|
|||
let five = FpVar::<F>::new_constant(cs.clone(), F::from(5u32))?;
|
|||
let z_i = z_i[0].clone();
|
|||
|
|||
Ok(vec![&z_i * &z_i * &z_i + &z_i + &five])
|
|||
}
|
|||
}
|
|||
|
|||
/// CustomFCircuit is a circuit that has the number of constraints specified in the
|
|||
/// `n_constraints` parameter. Note that the generated circuit will have very sparse matrices.
|
|||
#[derive(Clone, Copy, Debug)]
|
|||
pub struct CustomFCircuit<F: PrimeField> {
|
|||
_f: PhantomData<F>,
|
|||
pub n_constraints: usize,
|
|||
}
|
|||
impl<F: PrimeField> FCircuit<F> for CustomFCircuit<F> {
|
|||
type Params = usize;
|
|||
|
|||
fn new(params: Self::Params) -> Self {
|
|||
Self {
|
|||
_f: PhantomData,
|
|||
n_constraints: params,
|
|||
}
|
|||
}
|
|||
fn step_native(self, z_i: Vec<F>) -> Result<Vec<F>, Error> {
|
|||
let mut z_i1 = F::one();
|
|||
for _ in 0..self.n_constraints - 1 {
|
|||
z_i1 *= z_i[0];
|
|||
}
|
|||
Ok(vec![z_i1])
|
|||
}
|
|||
fn generate_step_constraints(
|
|||
self,
|
|||
cs: ConstraintSystemRef<F>,
|
|||
z_i: Vec<FpVar<F>>,
|
|||
) -> Result<Vec<FpVar<F>>, SynthesisError> {
|
|||
let mut z_i1 = FpVar::<F>::new_witness(cs.clone(), || Ok(F::one()))?;
|
|||
for _ in 0..self.n_constraints - 1 {
|
|||
z_i1 *= z_i[0].clone();
|
|||
}
|
|||
|
|||
Ok(vec![z_i1])
|
|||
}
|
|||
}
|
|||
|
|||
/// WrapperCircuit is a circuit that wraps any circuit that implements the FCircuit trait. This
|
|||
/// is used to test the `FCircuit.generate_step_constraints` method. This is a similar wrapping
|
|||
/// than the one done in the `AugmentedFCircuit`, but without adding all the extra constraints
|
|||
/// of the AugmentedF circuit logic, in order to run lighter tests when we're not interested in
|
|||
/// the the AugmentedF logic but in the wrapping of the circuits.
|
|||
pub struct WrapperCircuit<F: PrimeField, FC: FCircuit<F>> {
|
|||
pub FC: FC, // F circuit
|
|||
pub z_i: Option<Vec<F>>,
|
|||
pub z_i1: Option<Vec<F>>,
|
|||
}
|
|||
impl<F, FC> ConstraintSynthesizer<F> for WrapperCircuit<F, FC>
|
|||
where
|
|||
F: PrimeField,
|
|||
FC: FCircuit<F>,
|
|||
{
|
|||
fn generate_constraints(self, cs: ConstraintSystemRef<F>) -> Result<(), SynthesisError> {
|
|||
let z_i = Vec::<FpVar<F>>::new_witness(cs.clone(), || {
|
|||
Ok(self.z_i.unwrap_or(vec![F::zero()]))
|
|||
})?;
|
|||
let z_i1 = Vec::<FpVar<F>>::new_input(cs.clone(), || {
|
|||
Ok(self.z_i1.unwrap_or(vec![F::zero()]))
|
|||
})?;
|
|||
let computed_z_i1 = self.FC.generate_step_constraints(cs.clone(), z_i.clone())?;
|
|||
|
|||
computed_z_i1.enforce_equal(&z_i1)?;
|
|||
Ok(())
|
|||
}
|
|||
}
|
|||
|
|||
#[test]
|
|||
fn test_testfcircuit() {
|
|||
let cs = ConstraintSystem::<Fr>::new_ref();
|
|||
let F_circuit = CubicFCircuit::<Fr>::new(());
|
|||
|
|||
let wrapper_circuit = WrapperCircuit::<Fr, CubicFCircuit<Fr>> {
|
|||
FC: F_circuit,
|
|||
z_i: Some(vec![Fr::from(3_u32)]),
|
|||
z_i1: Some(vec![Fr::from(35_u32)]),
|
|||
};
|
|||
wrapper_circuit.generate_constraints(cs.clone()).unwrap();
|
|||
assert_eq!(cs.num_constraints(), 3);
|
|||
}
|
|||
|
|||
#[test]
|
|||
fn test_customtestfcircuit() {
|
|||
let cs = ConstraintSystem::<Fr>::new_ref();
|
|||
let n_constraints = 1000;
|
|||
let custom_circuit = CustomFCircuit::<Fr>::new(n_constraints);
|
|||
let z_i = vec![Fr::from(5_u32)];
|
|||
let wrapper_circuit = WrapperCircuit::<Fr, CustomFCircuit<Fr>> {
|
|||
FC: custom_circuit,
|
|||
z_i: Some(z_i.clone()),
|
|||
z_i1: Some(custom_circuit.step_native(z_i).unwrap()),
|
|||
};
|
|||
wrapper_circuit.generate_constraints(cs.clone()).unwrap();
|
|||
assert_eq!(cs.num_constraints(), n_constraints);
|
|||
}
|
|||
}
|