|  |  | @ -1,205 +1,618 @@ | 
			
		
	
		
			
				
					|  |  |  | //! This module implements the permutation check protocol.
 | 
			
		
	
		
			
				
					|  |  |  | //! Main module for the Permutation Check protocol
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | use crate::{utils::get_index, PolyIOPErrors};
 | 
			
		
	
		
			
				
					|  |  |  | use crate::{
 | 
			
		
	
		
			
				
					|  |  |  |     errors::PolyIOPErrors, perm_check::util::compute_prod_0, structs::IOPProof,
 | 
			
		
	
		
			
				
					|  |  |  |     transcript::IOPTranscript, utils::get_index, PolyIOP, VirtualPolynomial, ZeroCheck,
 | 
			
		
	
		
			
				
					|  |  |  | };
 | 
			
		
	
		
			
				
					|  |  |  | use ark_ff::PrimeField;
 | 
			
		
	
		
			
				
					|  |  |  | use ark_poly::DenseMultilinearExtension;
 | 
			
		
	
		
			
				
					|  |  |  | use ark_std::{end_timer, start_timer};
 | 
			
		
	
		
			
				
					|  |  |  | use std::rc::Rc;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | /// Compute `prod(0,x) := prod(0, x1, …, xn)` which is the MLE over the
 | 
			
		
	
		
			
				
					|  |  |  | /// evaluations of the following polynomial on the boolean hypercube {0,1}^n:
 | 
			
		
	
		
			
				
					|  |  |  | pub mod util;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | /// A PermutationCheck is derived from ZeroCheck.
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | ///  (w(x) + \beta s_id(x) + \gamma)/(w(x) + \beta s_perm(x) + \gamma)
 | 
			
		
	
		
			
				
					|  |  |  | /// A Permutation Check IOP takes the following steps:
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// where
 | 
			
		
	
		
			
				
					|  |  |  | /// - beta and gamma are challenges
 | 
			
		
	
		
			
				
					|  |  |  | /// - w(x), s_id(x), s_perm(x) are mle-s
 | 
			
		
	
		
			
				
					|  |  |  | /// Inputs:
 | 
			
		
	
		
			
				
					|  |  |  | /// - f(x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - g(x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - permutation s_perm(x)
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// The caller needs to check num_vars matches in w/s_id/s_perm
 | 
			
		
	
		
			
				
					|  |  |  | /// Cost: linear in N.
 | 
			
		
	
		
			
				
					|  |  |  | #[allow(dead_code)]
 | 
			
		
	
		
			
				
					|  |  |  | // TODO: remove
 | 
			
		
	
		
			
				
					|  |  |  | fn compute_prod_0<F: PrimeField>(
 | 
			
		
	
		
			
				
					|  |  |  |     beta: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     gamma: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     w: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     s_id: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     s_perm: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  | ) -> Result<DenseMultilinearExtension<F>, PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |     let start = start_timer!(|| "compute prod(1,x)");
 | 
			
		
	
		
			
				
					|  |  |  |     let num_vars = w.num_vars;
 | 
			
		
	
		
			
				
					|  |  |  |     let eval: Vec<F> = w
 | 
			
		
	
		
			
				
					|  |  |  |         .iter()
 | 
			
		
	
		
			
				
					|  |  |  |         .zip(s_id.iter().zip(s_perm.iter()))
 | 
			
		
	
		
			
				
					|  |  |  |         .map(|(wi, (s_id_i, s_perm_i))| {
 | 
			
		
	
		
			
				
					|  |  |  |             let tmp = *wi + *gamma;
 | 
			
		
	
		
			
				
					|  |  |  |             (tmp + *beta * *s_id_i) / (tmp + *beta * *s_perm_i)
 | 
			
		
	
		
			
				
					|  |  |  |         })
 | 
			
		
	
		
			
				
					|  |  |  |         .collect();
 | 
			
		
	
		
			
				
					|  |  |  | /// Steps:
 | 
			
		
	
		
			
				
					|  |  |  | /// 1. `generate_challenge` from current transcript (generate beta, gamma)
 | 
			
		
	
		
			
				
					|  |  |  | /// 2. `compute_product` to build `prod(x)` etc. from f, g and s_perm
 | 
			
		
	
		
			
				
					|  |  |  | /// 3. push a commitment of `prod(x)` to the transcript (done by the snark
 | 
			
		
	
		
			
				
					|  |  |  | /// caller)
 | 
			
		
	
		
			
				
					|  |  |  | /// 4. `update_challenge` with the updated transcript (generate alpha)
 | 
			
		
	
		
			
				
					|  |  |  | /// 5. `prove` to generate the proof
 | 
			
		
	
		
			
				
					|  |  |  | pub trait PermutationCheck<F: PrimeField>: ZeroCheck<F> {
 | 
			
		
	
		
			
				
					|  |  |  |     type PermutationCheckSubClaim;
 | 
			
		
	
		
			
				
					|  |  |  |     type PermutationChallenge;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Initialize the system with a transcript
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// This function is optional -- in the case where a PermutationCheck is
 | 
			
		
	
		
			
				
					|  |  |  |     /// an building block for a more complex protocol, the transcript
 | 
			
		
	
		
			
				
					|  |  |  |     /// may be initialized by this complex protocol, and passed to the
 | 
			
		
	
		
			
				
					|  |  |  |     /// PermutationCheck prover/verifier.
 | 
			
		
	
		
			
				
					|  |  |  |     fn init_transcript() -> Self::Transcript;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 1 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Generate challenge beta and gamma from a transcript.
 | 
			
		
	
		
			
				
					|  |  |  |     fn generate_challenge(
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut Self::Transcript,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<Self::PermutationChallenge, PolyIOPErrors>;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 4 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Update the challenge with alpha; returns an error if
 | 
			
		
	
		
			
				
					|  |  |  |     /// alpha already exists.
 | 
			
		
	
		
			
				
					|  |  |  |     fn update_challenge(
 | 
			
		
	
		
			
				
					|  |  |  |         challenge: &mut Self::PermutationChallenge,
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut Self::Transcript,
 | 
			
		
	
		
			
				
					|  |  |  |         prod_x_binding: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<(), PolyIOPErrors>;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 2 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Compute the following 7 polynomials
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(x)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(1, x)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(x, 0)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - numerator
 | 
			
		
	
		
			
				
					|  |  |  |     /// - denominator
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// where
 | 
			
		
	
		
			
				
					|  |  |  |     /// - `prod(0,x) := prod(0, x0, x1, …, xn)` which is the MLE over the
 | 
			
		
	
		
			
				
					|  |  |  |     /// evaluations of the following polynomial on the boolean hypercube
 | 
			
		
	
		
			
				
					|  |  |  |     /// {0,1}^n:
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     ///  (f(x) + \beta s_id(x) + \gamma)/(g(x) + \beta s_perm(x) + \gamma)
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     ///   where
 | 
			
		
	
		
			
				
					|  |  |  |     ///   - beta and gamma are challenges
 | 
			
		
	
		
			
				
					|  |  |  |     ///   - f(x), g(x), s_id(x), s_perm(x) are mle-s
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// - `prod(1,x) := prod(x, 0) * prod(x, 1)`
 | 
			
		
	
		
			
				
					|  |  |  |     /// - numerator is the MLE for `f(x) + \beta s_id(x) + \gamma`
 | 
			
		
	
		
			
				
					|  |  |  |     /// - denominator is the MLE for `g(x) + \beta s_perm(x) + \gamma`
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// The caller needs to check num_vars matches in f/g/s_id/s_perm
 | 
			
		
	
		
			
				
					|  |  |  |     /// Cost: linear in N.
 | 
			
		
	
		
			
				
					|  |  |  |     fn compute_products(
 | 
			
		
	
		
			
				
					|  |  |  |         challenge: &Self::PermutationChallenge,
 | 
			
		
	
		
			
				
					|  |  |  |         fx: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |         gx: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |         s_perm: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<[DenseMultilinearExtension<F>; 7], PolyIOPErrors>;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 5 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// Initialize the prover to argue that an MLE g(x) is a permutation of
 | 
			
		
	
		
			
				
					|  |  |  |     /// MLE f(x) over a permutation given by s_perm.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Inputs:
 | 
			
		
	
		
			
				
					|  |  |  |     /// - 7 MLEs from `Self::compute_products`
 | 
			
		
	
		
			
				
					|  |  |  |     /// - challenge: `Self::Challenge` that has been updated
 | 
			
		
	
		
			
				
					|  |  |  |     /// - transcript: a transcript that is used to generate the challenges beta
 | 
			
		
	
		
			
				
					|  |  |  |     ///   and gamma
 | 
			
		
	
		
			
				
					|  |  |  |     /// Cost: O(N)
 | 
			
		
	
		
			
				
					|  |  |  |     fn prove(
 | 
			
		
	
		
			
				
					|  |  |  |         prod_x_and_aux_info: &[DenseMultilinearExtension<F>; 7],
 | 
			
		
	
		
			
				
					|  |  |  |         challenge: &Self::PermutationChallenge,
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut IOPTranscript<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<Self::Proof, PolyIOPErrors>;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Verify that an MLE g(x) is a permutation of
 | 
			
		
	
		
			
				
					|  |  |  |     /// MLE f(x) over a permutation given by s_perm.
 | 
			
		
	
		
			
				
					|  |  |  |     fn verify(
 | 
			
		
	
		
			
				
					|  |  |  |         proof: &Self::Proof,
 | 
			
		
	
		
			
				
					|  |  |  |         aux_info: &Self::VPAuxInfo,
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut Self::Transcript,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<Self::PermutationCheckSubClaim, PolyIOPErrors>;
 | 
			
		
	
		
			
				
					|  |  |  | }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     let res = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval);
 | 
			
		
	
		
			
				
					|  |  |  |     end_timer!(start);
 | 
			
		
	
		
			
				
					|  |  |  |     Ok(res)
 | 
			
		
	
		
			
				
					|  |  |  | /// A permutation subclaim consists of
 | 
			
		
	
		
			
				
					|  |  |  | /// - A zero check IOP subclaim for Q(x) is 0, consists of the following:
 | 
			
		
	
		
			
				
					|  |  |  | ///  (See `build_qx` for definition of Q(x).)
 | 
			
		
	
		
			
				
					|  |  |  | ///   - the SubClaim from the SumCheck
 | 
			
		
	
		
			
				
					|  |  |  | ///   - the initial challenge r which is used to build eq(x, r) in ZeroCheck
 | 
			
		
	
		
			
				
					|  |  |  | /// - A final query for `prod(1, ..., 1, 0) = 1`.
 | 
			
		
	
		
			
				
					|  |  |  | // Note that this final query is in fact a constant that
 | 
			
		
	
		
			
				
					|  |  |  | // is independent from the proof. So we should avoid
 | 
			
		
	
		
			
				
					|  |  |  | // (de)serialize it.
 | 
			
		
	
		
			
				
					|  |  |  | #[derive(Clone, Debug, Default, PartialEq)]
 | 
			
		
	
		
			
				
					|  |  |  | pub struct PermutationCheckSubClaim<F: PrimeField, ZC: ZeroCheck<F>> {
 | 
			
		
	
		
			
				
					|  |  |  |     // the SubClaim from the ZeroCheck
 | 
			
		
	
		
			
				
					|  |  |  |     zero_check_sub_claim: ZC::ZeroCheckSubClaim,
 | 
			
		
	
		
			
				
					|  |  |  |     // final query which consists of
 | 
			
		
	
		
			
				
					|  |  |  |     // - the vector `(1, ..., 1, 0)`
 | 
			
		
	
		
			
				
					|  |  |  |     // - the evaluation `1`
 | 
			
		
	
		
			
				
					|  |  |  |     final_query: (Vec<F>, F),
 | 
			
		
	
		
			
				
					|  |  |  | }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | /// Compute the following 5 polynomials
 | 
			
		
	
		
			
				
					|  |  |  | /// - prod(x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - prod(1, x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - prod(x, 0)
 | 
			
		
	
		
			
				
					|  |  |  | /// - prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// where
 | 
			
		
	
		
			
				
					|  |  |  | /// - `prod(0,x) := prod(0, x0, x1, …, xn)` which is the MLE over the
 | 
			
		
	
		
			
				
					|  |  |  | /// evaluations of the following polynomial on the boolean hypercube {0,1}^n:
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | ///  (w(x) + \beta s_id(x) + \gamma)/(w(x) + \beta s_perm(x) + \gamma)
 | 
			
		
	
		
			
				
					|  |  |  | pub struct PermutationChallenge<F: PrimeField> {
 | 
			
		
	
		
			
				
					|  |  |  |     alpha: Option<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     beta: F,
 | 
			
		
	
		
			
				
					|  |  |  |     gamma: F,
 | 
			
		
	
		
			
				
					|  |  |  | }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | /// A PermutationCheck is derived from ZeroCheck.
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | ///   where
 | 
			
		
	
		
			
				
					|  |  |  | ///   - beta and gamma are challenges
 | 
			
		
	
		
			
				
					|  |  |  | ///   - w(x), s_id(x), s_perm(x) are mle-s
 | 
			
		
	
		
			
				
					|  |  |  | /// A Permutation Check IOP takes the following steps:
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// - `prod(1,x) := prod(x, 0) * prod(x, 1)`
 | 
			
		
	
		
			
				
					|  |  |  | /// Inputs:
 | 
			
		
	
		
			
				
					|  |  |  | /// - f(x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - g(x)
 | 
			
		
	
		
			
				
					|  |  |  | /// - permutation s_perm(x)
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// Returns an error when the num_vars in w/s_id/s_perm does not match
 | 
			
		
	
		
			
				
					|  |  |  | /// Cost: linear in N.
 | 
			
		
	
		
			
				
					|  |  |  | #[allow(dead_code)]
 | 
			
		
	
		
			
				
					|  |  |  | // TODO: remove
 | 
			
		
	
		
			
				
					|  |  |  | fn compute_products<F: PrimeField>(
 | 
			
		
	
		
			
				
					|  |  |  |     beta: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     gamma: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     w: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     s_id: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     s_perm: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  | ) -> Result<[DenseMultilinearExtension<F>; 5], PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |     let start = start_timer!(|| "compute all prod polynomial");
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     let num_vars = w.num_vars;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     if num_vars != s_id.num_vars || num_vars != s_perm.num_vars {
 | 
			
		
	
		
			
				
					|  |  |  |         return Err(PolyIOPErrors::InvalidParameters(
 | 
			
		
	
		
			
				
					|  |  |  |             "num of variables do not match".to_string(),
 | 
			
		
	
		
			
				
					|  |  |  |         ));
 | 
			
		
	
		
			
				
					|  |  |  | /// Steps:
 | 
			
		
	
		
			
				
					|  |  |  | /// 1. `generate_challenge` from current transcript (generate beta, gamma)
 | 
			
		
	
		
			
				
					|  |  |  | /// 2. `compute_product` to build `prod(x)` etc. from f, g and s_perm
 | 
			
		
	
		
			
				
					|  |  |  | /// 3. push a commitment of `prod(x)` to the transcript (done by the snark
 | 
			
		
	
		
			
				
					|  |  |  | /// caller)
 | 
			
		
	
		
			
				
					|  |  |  | /// 4. `update_challenge` with the updated transcript (generate alpha)
 | 
			
		
	
		
			
				
					|  |  |  | /// 5. `prove` to generate the proof
 | 
			
		
	
		
			
				
					|  |  |  | impl<F: PrimeField> PermutationCheck<F> for PolyIOP<F> {
 | 
			
		
	
		
			
				
					|  |  |  |     /// A Permutation SubClaim is indeed a ZeroCheck SubClaim that consists of
 | 
			
		
	
		
			
				
					|  |  |  |     /// - the SubClaim from the SumCheck
 | 
			
		
	
		
			
				
					|  |  |  |     /// - the initial challenge r which is used to build eq(x, r)
 | 
			
		
	
		
			
				
					|  |  |  |     type PermutationCheckSubClaim = PermutationCheckSubClaim<F, Self>;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     type PermutationChallenge = PermutationChallenge<F>;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Initialize the system with a transcript
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// This function is optional -- in the case where a PermutationCheck is
 | 
			
		
	
		
			
				
					|  |  |  |     /// an building block for a more complex protocol, the transcript
 | 
			
		
	
		
			
				
					|  |  |  |     /// may be initialized by this complex protocol, and passed to the
 | 
			
		
	
		
			
				
					|  |  |  |     /// PermutationCheck prover/verifier.
 | 
			
		
	
		
			
				
					|  |  |  |     fn init_transcript() -> Self::Transcript {
 | 
			
		
	
		
			
				
					|  |  |  |         IOPTranscript::<F>::new(b"Initializing PermutationCheck transcript")
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     // prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_0x = compute_prod_0(beta, gamma, w, s_id, s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     // prod(1, x)
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     //
 | 
			
		
	
		
			
				
					|  |  |  |     // `prod(1, x)` can be computed via recursing the following formula for 2^n-1
 | 
			
		
	
		
			
				
					|  |  |  |     // times
 | 
			
		
	
		
			
				
					|  |  |  |     //
 | 
			
		
	
		
			
				
					|  |  |  |     // `prod(1, x_1, ..., x_n) :=
 | 
			
		
	
		
			
				
					|  |  |  |     //      prod(x_1, x_2, ..., x_n, 0) * prod(x_1, x_2, ..., x_n, 1)`
 | 
			
		
	
		
			
				
					|  |  |  |     //
 | 
			
		
	
		
			
				
					|  |  |  |     // At any given step, the right hand side of the equation
 | 
			
		
	
		
			
				
					|  |  |  |     // is available via either eval_0x or the current view of eval_1x
 | 
			
		
	
		
			
				
					|  |  |  |     let eval_0x = &prod_0x.evaluations;
 | 
			
		
	
		
			
				
					|  |  |  |     let mut eval_1x = vec![];
 | 
			
		
	
		
			
				
					|  |  |  |     for x in 0..(1 << num_vars) - 1 {
 | 
			
		
	
		
			
				
					|  |  |  |         // sign will decide if the evaluation should be looked up from eval_0x or
 | 
			
		
	
		
			
				
					|  |  |  |         // eval_1x; x_zero_index is the index for the evaluation (x_2, ..., x_n,
 | 
			
		
	
		
			
				
					|  |  |  |         // 0); x_one_index is the index for the evaluation (x_2, ..., x_n, 1);
 | 
			
		
	
		
			
				
					|  |  |  |         let (x_zero_index, x_one_index, sign) = get_index(x, num_vars);
 | 
			
		
	
		
			
				
					|  |  |  |         if !sign {
 | 
			
		
	
		
			
				
					|  |  |  |             eval_1x.push(eval_0x[x_zero_index] * eval_0x[x_one_index]);
 | 
			
		
	
		
			
				
					|  |  |  |         } else {
 | 
			
		
	
		
			
				
					|  |  |  |             // sanity check: if we are trying to look up from the eval_1x table,
 | 
			
		
	
		
			
				
					|  |  |  |             // then the target index must already exist
 | 
			
		
	
		
			
				
					|  |  |  |             if x_zero_index >= eval_1x.len() || x_one_index >= eval_1x.len() {
 | 
			
		
	
		
			
				
					|  |  |  |                 return Err(PolyIOPErrors::ShouldNotArrive);
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
		
			
				
					|  |  |  |             eval_1x.push(eval_1x[x_zero_index] * eval_1x[x_one_index]);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 1 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Generate challenge beta and gamma from a transcript.
 | 
			
		
	
		
			
				
					|  |  |  |     fn generate_challenge(
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut Self::Transcript,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<Self::PermutationChallenge, PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         Ok(Self::PermutationChallenge {
 | 
			
		
	
		
			
				
					|  |  |  |             beta: transcript.get_and_append_challenge(b"beta")?,
 | 
			
		
	
		
			
				
					|  |  |  |             gamma: transcript.get_and_append_challenge(b"gamma")?,
 | 
			
		
	
		
			
				
					|  |  |  |             alpha: None,
 | 
			
		
	
		
			
				
					|  |  |  |         })
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 4 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Update the challenge with alpha; returns an error if
 | 
			
		
	
		
			
				
					|  |  |  |     /// alpha already exists.
 | 
			
		
	
		
			
				
					|  |  |  |     fn update_challenge(
 | 
			
		
	
		
			
				
					|  |  |  |         challenge: &mut Self::PermutationChallenge,
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut Self::Transcript,
 | 
			
		
	
		
			
				
					|  |  |  |         prod_x_binding: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<(), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         if challenge.alpha.is_some() {
 | 
			
		
	
		
			
				
					|  |  |  |             return Err(PolyIOPErrors::InvalidTranscript(
 | 
			
		
	
		
			
				
					|  |  |  |                 "alpha should not be sampled at the current stage".to_string(),
 | 
			
		
	
		
			
				
					|  |  |  |             ));
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  |         transcript.append_field_element(b"prod(x)", prod_x_binding)?;
 | 
			
		
	
		
			
				
					|  |  |  |         challenge.alpha = Some(transcript.get_and_append_challenge(b"alpha")?);
 | 
			
		
	
		
			
				
					|  |  |  |         Ok(())
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  |     // prod(1, 1, ..., 1) := 0
 | 
			
		
	
		
			
				
					|  |  |  |     eval_1x.push(F::zero());
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     // prod(x)
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     // prod(x)'s evaluation is indeed `e := [eval_0x[..], eval_1x[..]].concat()`
 | 
			
		
	
		
			
				
					|  |  |  |     let eval = [eval_0x.as_slice(), eval_1x.as_slice()].concat();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     // prod(x, 0) and prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  |     // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |     //
 | 
			
		
	
		
			
				
					|  |  |  |     // now we compute eval_x0 and eval_x1
 | 
			
		
	
		
			
				
					|  |  |  |     // eval_0x will be the odd coefficients of eval
 | 
			
		
	
		
			
				
					|  |  |  |     // and eval_1x will be the even coefficients of eval
 | 
			
		
	
		
			
				
					|  |  |  |     let mut eval_x0 = vec![];
 | 
			
		
	
		
			
				
					|  |  |  |     let mut eval_x1 = vec![];
 | 
			
		
	
		
			
				
					|  |  |  |     for (x, &prod_x) in eval.iter().enumerate() {
 | 
			
		
	
		
			
				
					|  |  |  |         if x & 1 == 0 {
 | 
			
		
	
		
			
				
					|  |  |  |             eval_x0.push(prod_x);
 | 
			
		
	
		
			
				
					|  |  |  |         } else {
 | 
			
		
	
		
			
				
					|  |  |  |             eval_x1.push(prod_x);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 2 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Compute the following 7 polynomials
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(x)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(1, x)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(x, 0)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  |     /// - numerator
 | 
			
		
	
		
			
				
					|  |  |  |     /// - denominator
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// where
 | 
			
		
	
		
			
				
					|  |  |  |     /// - `prod(0,x) := prod(0, x0, x1, …, xn)` which is the MLE over the
 | 
			
		
	
		
			
				
					|  |  |  |     /// evaluations of the following polynomial on the boolean hypercube
 | 
			
		
	
		
			
				
					|  |  |  |     /// {0,1}^n:
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     ///  (f(x) + \beta s_id(x) + \gamma)/(g(x) + \beta s_perm(x) + \gamma)
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     ///   where
 | 
			
		
	
		
			
				
					|  |  |  |     ///   - beta and gamma are challenges
 | 
			
		
	
		
			
				
					|  |  |  |     ///   - f(x), g(x), s_id(x), s_perm(x) are mle-s
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// - `prod(1,x) := prod(x, 0) * prod(x, 1)`
 | 
			
		
	
		
			
				
					|  |  |  |     /// - numerator is the MLE for `f(x) + \beta s_id(x) + \gamma`
 | 
			
		
	
		
			
				
					|  |  |  |     /// - denominator is the MLE for `g(x) + \beta s_perm(x) + \gamma`
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// The caller needs to check num_vars matches in f/g/s_id/s_perm
 | 
			
		
	
		
			
				
					|  |  |  |     /// Cost: linear in N.
 | 
			
		
	
		
			
				
					|  |  |  |     fn compute_products(
 | 
			
		
	
		
			
				
					|  |  |  |         challenge: &Self::PermutationChallenge,
 | 
			
		
	
		
			
				
					|  |  |  |         fx: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |         gx: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |         s_perm: &DenseMultilinearExtension<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<[DenseMultilinearExtension<F>; 7], PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         let start = start_timer!(|| "compute all prod polynomial");
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         if challenge.alpha.is_some() {
 | 
			
		
	
		
			
				
					|  |  |  |             return Err(PolyIOPErrors::InvalidTranscript(
 | 
			
		
	
		
			
				
					|  |  |  |                 "alpha is already sampled".to_string(),
 | 
			
		
	
		
			
				
					|  |  |  |             ));
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         let num_vars = fx.num_vars;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         // prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         let (prod_0x, numerator, denominator) =
 | 
			
		
	
		
			
				
					|  |  |  |             compute_prod_0(&challenge.beta, &challenge.gamma, fx, gx, s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         // prod(1, x)
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         //
 | 
			
		
	
		
			
				
					|  |  |  |         // `prod(1, x)` can be computed via recursing the following formula for 2^n-1
 | 
			
		
	
		
			
				
					|  |  |  |         // times
 | 
			
		
	
		
			
				
					|  |  |  |         //
 | 
			
		
	
		
			
				
					|  |  |  |         // `prod(1, x_1, ..., x_n) :=
 | 
			
		
	
		
			
				
					|  |  |  |         //      prod(x_1, x_2, ..., x_n, 0) * prod(x_1, x_2, ..., x_n, 1)`
 | 
			
		
	
		
			
				
					|  |  |  |         //
 | 
			
		
	
		
			
				
					|  |  |  |         // At any given step, the right hand side of the equation
 | 
			
		
	
		
			
				
					|  |  |  |         // is available via either eval_0x or the current view of eval_1x
 | 
			
		
	
		
			
				
					|  |  |  |         let eval_0x = &prod_0x.evaluations;
 | 
			
		
	
		
			
				
					|  |  |  |         let mut eval_1x = vec![];
 | 
			
		
	
		
			
				
					|  |  |  |         for x in 0..(1 << num_vars) - 1 {
 | 
			
		
	
		
			
				
					|  |  |  |             // sign will decide if the evaluation should be looked up from eval_0x or
 | 
			
		
	
		
			
				
					|  |  |  |             // eval_1x; x_zero_index is the index for the evaluation (x_2, ..., x_n,
 | 
			
		
	
		
			
				
					|  |  |  |             // 0); x_one_index is the index for the evaluation (x_2, ..., x_n, 1);
 | 
			
		
	
		
			
				
					|  |  |  |             let (x_zero_index, x_one_index, sign) = get_index(x, num_vars);
 | 
			
		
	
		
			
				
					|  |  |  |             if !sign {
 | 
			
		
	
		
			
				
					|  |  |  |                 eval_1x.push(eval_0x[x_zero_index] * eval_0x[x_one_index]);
 | 
			
		
	
		
			
				
					|  |  |  |             } else {
 | 
			
		
	
		
			
				
					|  |  |  |                 // sanity check: if we are trying to look up from the eval_1x table,
 | 
			
		
	
		
			
				
					|  |  |  |                 // then the target index must already exist
 | 
			
		
	
		
			
				
					|  |  |  |                 if x_zero_index >= eval_1x.len() || x_one_index >= eval_1x.len() {
 | 
			
		
	
		
			
				
					|  |  |  |                     return Err(PolyIOPErrors::ShouldNotArrive);
 | 
			
		
	
		
			
				
					|  |  |  |                 }
 | 
			
		
	
		
			
				
					|  |  |  |                 eval_1x.push(eval_1x[x_zero_index] * eval_1x[x_one_index]);
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  |         // prod(1, 1, ..., 1) := 0
 | 
			
		
	
		
			
				
					|  |  |  |         eval_1x.push(F::zero());
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         // prod(x)
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         // prod(x)'s evaluation is indeed `e := [eval_0x[..], eval_1x[..]].concat()`
 | 
			
		
	
		
			
				
					|  |  |  |         let eval = [eval_0x.as_slice(), eval_1x.as_slice()].concat();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         // prod(x, 0) and prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  |         // ===================================
 | 
			
		
	
		
			
				
					|  |  |  |         //
 | 
			
		
	
		
			
				
					|  |  |  |         // now we compute eval_x0 and eval_x1
 | 
			
		
	
		
			
				
					|  |  |  |         // eval_0x will be the odd coefficients of eval
 | 
			
		
	
		
			
				
					|  |  |  |         // and eval_1x will be the even coefficients of eval
 | 
			
		
	
		
			
				
					|  |  |  |         let mut eval_x0 = vec![];
 | 
			
		
	
		
			
				
					|  |  |  |         let mut eval_x1 = vec![];
 | 
			
		
	
		
			
				
					|  |  |  |         for (x, &prod_x) in eval.iter().enumerate() {
 | 
			
		
	
		
			
				
					|  |  |  |             if x & 1 == 0 {
 | 
			
		
	
		
			
				
					|  |  |  |                 eval_x0.push(prod_x);
 | 
			
		
	
		
			
				
					|  |  |  |             } else {
 | 
			
		
	
		
			
				
					|  |  |  |                 eval_x1.push(prod_x);
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         let prod_1x = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval_1x);
 | 
			
		
	
		
			
				
					|  |  |  |         let prod_x0 = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval_x0);
 | 
			
		
	
		
			
				
					|  |  |  |         let prod_x1 = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval_x1);
 | 
			
		
	
		
			
				
					|  |  |  |         let prod = DenseMultilinearExtension::from_evaluations_vec(num_vars + 1, eval);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         end_timer!(start);
 | 
			
		
	
		
			
				
					|  |  |  |         Ok([
 | 
			
		
	
		
			
				
					|  |  |  |             prod,
 | 
			
		
	
		
			
				
					|  |  |  |             prod_0x,
 | 
			
		
	
		
			
				
					|  |  |  |             prod_1x,
 | 
			
		
	
		
			
				
					|  |  |  |             prod_x0,
 | 
			
		
	
		
			
				
					|  |  |  |             prod_x1,
 | 
			
		
	
		
			
				
					|  |  |  |             numerator,
 | 
			
		
	
		
			
				
					|  |  |  |             denominator,
 | 
			
		
	
		
			
				
					|  |  |  |         ])
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_1x = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval_1x);
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_x0 = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval_x0);
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_x1 = DenseMultilinearExtension::from_evaluations_vec(num_vars, eval_x1);
 | 
			
		
	
		
			
				
					|  |  |  |     let prod = DenseMultilinearExtension::from_evaluations_vec(num_vars + 1, eval);
 | 
			
		
	
		
			
				
					|  |  |  |     /// Step 5 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  |     ///
 | 
			
		
	
		
			
				
					|  |  |  |     /// Generate a proof to argue that an MLE g(x) is a permutation of
 | 
			
		
	
		
			
				
					|  |  |  |     /// MLE f(x) over a permutation given by s_perm.
 | 
			
		
	
		
			
				
					|  |  |  |     /// Inputs:
 | 
			
		
	
		
			
				
					|  |  |  |     /// - 7 MLEs from `Self::compute_products(*, f, g, s_perm)`
 | 
			
		
	
		
			
				
					|  |  |  |     /// - challenge: `Self::Challenge` that has been updated
 | 
			
		
	
		
			
				
					|  |  |  |     /// - transcript: a transcript that is used to generate the challenges beta
 | 
			
		
	
		
			
				
					|  |  |  |     ///   and gamma
 | 
			
		
	
		
			
				
					|  |  |  |     /// Cost: O(N)
 | 
			
		
	
		
			
				
					|  |  |  |     fn prove(
 | 
			
		
	
		
			
				
					|  |  |  |         prod_x_and_aux_info: &[DenseMultilinearExtension<F>; 7],
 | 
			
		
	
		
			
				
					|  |  |  |         challenge: &Self::PermutationChallenge,
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut IOPTranscript<F>,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<Self::Proof, PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         let alpha = match challenge.alpha {
 | 
			
		
	
		
			
				
					|  |  |  |             Some(p) => p,
 | 
			
		
	
		
			
				
					|  |  |  |             None => {
 | 
			
		
	
		
			
				
					|  |  |  |                 return Err(PolyIOPErrors::InvalidTranscript(
 | 
			
		
	
		
			
				
					|  |  |  |                     "alpha is not sampled yet".to_string(),
 | 
			
		
	
		
			
				
					|  |  |  |                 ))
 | 
			
		
	
		
			
				
					|  |  |  |             },
 | 
			
		
	
		
			
				
					|  |  |  |         };
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         let (proof, _q_x) = prove_internal(prod_x_and_aux_info, &alpha, transcript)?;
 | 
			
		
	
		
			
				
					|  |  |  |         Ok(proof)
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// Verify that an MLE g(x) is a permutation of an
 | 
			
		
	
		
			
				
					|  |  |  |     /// MLE f(x) over a permutation given by s_perm.
 | 
			
		
	
		
			
				
					|  |  |  |     fn verify(
 | 
			
		
	
		
			
				
					|  |  |  |         proof: &Self::Proof,
 | 
			
		
	
		
			
				
					|  |  |  |         aux_info: &Self::VPAuxInfo,
 | 
			
		
	
		
			
				
					|  |  |  |         transcript: &mut Self::Transcript,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<Self::PermutationCheckSubClaim, PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         let start = start_timer!(|| "Permutation check verify");
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         // invoke the zero check on the iop_proof
 | 
			
		
	
		
			
				
					|  |  |  |         let zero_check_sub_claim = <Self as ZeroCheck<F>>::verify(proof, aux_info, transcript)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         let mut final_query = vec![F::one(); aux_info.num_variables];
 | 
			
		
	
		
			
				
					|  |  |  |         final_query[aux_info.num_variables - 1] = F::zero();
 | 
			
		
	
		
			
				
					|  |  |  |         let final_eval = F::one();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         end_timer!(start);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         Ok(PermutationCheckSubClaim {
 | 
			
		
	
		
			
				
					|  |  |  |             zero_check_sub_claim,
 | 
			
		
	
		
			
				
					|  |  |  |             final_query: (final_query, final_eval),
 | 
			
		
	
		
			
				
					|  |  |  |         })
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | /// Step 5 of the IOP.
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// Generate a proof to argue that an MLE g(x) is a permutation of
 | 
			
		
	
		
			
				
					|  |  |  | /// MLE f(x) over a permutation given by s_perm.
 | 
			
		
	
		
			
				
					|  |  |  | /// Inputs:
 | 
			
		
	
		
			
				
					|  |  |  | /// - 7 MLEs from `Self::compute_products(*, f, g, s_perm)`
 | 
			
		
	
		
			
				
					|  |  |  | /// - challenge: `Self::Challenge` that has been updated
 | 
			
		
	
		
			
				
					|  |  |  | /// - transcript: a transcript that is used to generate the challenges beta and
 | 
			
		
	
		
			
				
					|  |  |  | ///   gamma
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// Returns proof and Q(x) for testing purpose.
 | 
			
		
	
		
			
				
					|  |  |  | ///
 | 
			
		
	
		
			
				
					|  |  |  | /// Cost: O(N)
 | 
			
		
	
		
			
				
					|  |  |  | fn prove_internal<F: PrimeField>(
 | 
			
		
	
		
			
				
					|  |  |  |     prod_x_and_aux_info: &[DenseMultilinearExtension<F>; 7],
 | 
			
		
	
		
			
				
					|  |  |  |     alpha: &F,
 | 
			
		
	
		
			
				
					|  |  |  |     transcript: &mut IOPTranscript<F>,
 | 
			
		
	
		
			
				
					|  |  |  | ) -> Result<(IOPProof<F>, VirtualPolynomial<F>), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |     let start = start_timer!(|| "Permutation check prove");
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     // prods consists of the following:
 | 
			
		
	
		
			
				
					|  |  |  |     // - prod(x)
 | 
			
		
	
		
			
				
					|  |  |  |     // - prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  |     // - prod(1, x)
 | 
			
		
	
		
			
				
					|  |  |  |     // - prod(x, 0)
 | 
			
		
	
		
			
				
					|  |  |  |     // - prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  |     // - numerator
 | 
			
		
	
		
			
				
					|  |  |  |     // - denominator
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_0x = Rc::new(prod_x_and_aux_info[1].clone());
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_1x = Rc::new(prod_x_and_aux_info[2].clone());
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_x1 = Rc::new(prod_x_and_aux_info[3].clone());
 | 
			
		
	
		
			
				
					|  |  |  |     let prod_x0 = Rc::new(prod_x_and_aux_info[4].clone());
 | 
			
		
	
		
			
				
					|  |  |  |     let numerator = Rc::new(prod_x_and_aux_info[5].clone());
 | 
			
		
	
		
			
				
					|  |  |  |     let denominator = Rc::new(prod_x_and_aux_info[6].clone());
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     // compute (g(x) + beta * s_perm(x) + gamma) * prod(0, x) * alpha
 | 
			
		
	
		
			
				
					|  |  |  |     // which is prods[6] * prod[1] * alpha
 | 
			
		
	
		
			
				
					|  |  |  |     let mut q_x = VirtualPolynomial::new_from_mle(denominator, F::one());
 | 
			
		
	
		
			
				
					|  |  |  |     q_x.mul_by_mle(prod_0x, *alpha)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     //   (g(x) + beta * s_perm(x) + gamma) * prod(0, x) * alpha
 | 
			
		
	
		
			
				
					|  |  |  |     // - (f(x) + beta * s_id(x)   + gamma) * alpha
 | 
			
		
	
		
			
				
					|  |  |  |     q_x.add_mle_list([numerator], -*alpha)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     // Q(x) := prod(1,x) - prod(x, 0) * prod(x, 1)
 | 
			
		
	
		
			
				
					|  |  |  |     //       + alpha * (
 | 
			
		
	
		
			
				
					|  |  |  |     //             (g(x) + beta * s_perm(x) + gamma) * prod(0, x)
 | 
			
		
	
		
			
				
					|  |  |  |     //           - (f(x) + beta * s_id(x)   + gamma))
 | 
			
		
	
		
			
				
					|  |  |  |     q_x.add_mle_list([prod_x0, prod_x1], -F::one())?;
 | 
			
		
	
		
			
				
					|  |  |  |     q_x.add_mle_list([prod_1x], F::one())?;
 | 
			
		
	
		
			
				
					|  |  |  |     let iop_proof = <PolyIOP<F> as ZeroCheck<F>>::prove(&q_x, transcript)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     end_timer!(start);
 | 
			
		
	
		
			
				
					|  |  |  |     Ok([prod, prod_0x, prod_1x, prod_x0, prod_x1])
 | 
			
		
	
		
			
				
					|  |  |  |     Ok((iop_proof, q_x))
 | 
			
		
	
		
			
				
					|  |  |  | }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | #[cfg(test)]
 | 
			
		
	
		
			
				
					|  |  |  | mod test {
 | 
			
		
	
		
			
				
					|  |  |  |     use super::*;
 | 
			
		
	
		
			
				
					|  |  |  |     use crate::utils::bit_decompose;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     use super::PermutationCheck;
 | 
			
		
	
		
			
				
					|  |  |  |     use crate::{
 | 
			
		
	
		
			
				
					|  |  |  |         errors::PolyIOPErrors,
 | 
			
		
	
		
			
				
					|  |  |  |         perm_check::{prove_internal, util::identity_permutation_mle},
 | 
			
		
	
		
			
				
					|  |  |  |         random_permutation_mle,
 | 
			
		
	
		
			
				
					|  |  |  |         structs::IOPProof,
 | 
			
		
	
		
			
				
					|  |  |  |         utils::bit_decompose,
 | 
			
		
	
		
			
				
					|  |  |  |         virtual_poly::VPAuxInfo,
 | 
			
		
	
		
			
				
					|  |  |  |         PolyIOP, VirtualPolynomial,
 | 
			
		
	
		
			
				
					|  |  |  |     };
 | 
			
		
	
		
			
				
					|  |  |  |     use ark_bls12_381::Fr;
 | 
			
		
	
		
			
				
					|  |  |  |     use ark_ff::UniformRand;
 | 
			
		
	
		
			
				
					|  |  |  |     use ark_poly::MultilinearExtension;
 | 
			
		
	
		
			
				
					|  |  |  |     use ark_ff::{PrimeField, Zero};
 | 
			
		
	
		
			
				
					|  |  |  |     use ark_poly::{DenseMultilinearExtension, MultilinearExtension};
 | 
			
		
	
		
			
				
					|  |  |  |     use ark_std::test_rng;
 | 
			
		
	
		
			
				
					|  |  |  |     use std::marker::PhantomData;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     /// An MLE that represent an identity permutation: `f(index) \mapto index`
 | 
			
		
	
		
			
				
					|  |  |  |     fn identity_permutation_mle<F: PrimeField>(num_vars: usize) -> DenseMultilinearExtension<F> {
 | 
			
		
	
		
			
				
					|  |  |  |         let s_id_vec = (0..1u64 << num_vars).map(|index| F::from(index)).collect();
 | 
			
		
	
		
			
				
					|  |  |  |         DenseMultilinearExtension::from_evaluations_vec(num_vars, s_id_vec)
 | 
			
		
	
		
			
				
					|  |  |  |     fn mock_commit<F: PrimeField>(_f: &DenseMultilinearExtension<F>) -> F {
 | 
			
		
	
		
			
				
					|  |  |  |         let mut rng = test_rng();
 | 
			
		
	
		
			
				
					|  |  |  |         F::rand(&mut rng)
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     #[test]
 | 
			
		
	
		
			
				
					|  |  |  |     fn test_compute_prod_0() -> Result<(), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         let mut rng = test_rng();
 | 
			
		
	
		
			
				
					|  |  |  |     fn test_permutation_check_helper(
 | 
			
		
	
		
			
				
					|  |  |  |         f: &DenseMultilinearExtension<Fr>,
 | 
			
		
	
		
			
				
					|  |  |  |         g: &DenseMultilinearExtension<Fr>,
 | 
			
		
	
		
			
				
					|  |  |  |         s_perm: &DenseMultilinearExtension<Fr>,
 | 
			
		
	
		
			
				
					|  |  |  |     ) -> Result<(IOPProof<Fr>, VirtualPolynomial<Fr>), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         let mut transcript = <PolyIOP<Fr> as PermutationCheck<Fr>>::init_transcript();
 | 
			
		
	
		
			
				
					|  |  |  |         transcript.append_message(b"testing", b"initializing transcript for testing")?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         for num_vars in 2..6 {
 | 
			
		
	
		
			
				
					|  |  |  |             let w_vec: Vec<Fr> = (0..(1 << num_vars)).map(|_| Fr::rand(&mut rng)).collect();
 | 
			
		
	
		
			
				
					|  |  |  |             let w = DenseMultilinearExtension::from_evaluations_vec(num_vars, w_vec);
 | 
			
		
	
		
			
				
					|  |  |  |         let mut challenge =
 | 
			
		
	
		
			
				
					|  |  |  |             <PolyIOP<Fr> as PermutationCheck<Fr>>::generate_challenge(&mut transcript)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let s_id = identity_permutation_mle::<Fr>(num_vars);
 | 
			
		
	
		
			
				
					|  |  |  |         let prod_x_and_aux =
 | 
			
		
	
		
			
				
					|  |  |  |             <PolyIOP<Fr> as PermutationCheck<Fr>>::compute_products(&challenge, f, g, s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm_vec: Vec<Fr> = (0..(1 << num_vars)).map(|_| Fr::rand(&mut rng)).collect();
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm = DenseMultilinearExtension::from_evaluations_vec(num_vars, s_perm_vec);
 | 
			
		
	
		
			
				
					|  |  |  |         let prod_x_binding = mock_commit(&prod_x_and_aux[0]);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let beta = Fr::rand(&mut rng);
 | 
			
		
	
		
			
				
					|  |  |  |             let gamma = Fr::rand(&mut rng);
 | 
			
		
	
		
			
				
					|  |  |  |         <PolyIOP<Fr> as PermutationCheck<Fr>>::update_challenge(
 | 
			
		
	
		
			
				
					|  |  |  |             &mut challenge,
 | 
			
		
	
		
			
				
					|  |  |  |             &mut transcript,
 | 
			
		
	
		
			
				
					|  |  |  |             &prod_x_binding,
 | 
			
		
	
		
			
				
					|  |  |  |         )?;
 | 
			
		
	
		
			
				
					|  |  |  |         let alpha = challenge.alpha.unwrap();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let prod_0 = compute_prod_0(&beta, &gamma, &w, &s_id, &s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  |         prove_internal(&prod_x_and_aux, &alpha, &mut transcript)
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             for i in 0..1 << num_vars {
 | 
			
		
	
		
			
				
					|  |  |  |                 let r: Vec<Fr> = bit_decompose(i, num_vars)
 | 
			
		
	
		
			
				
					|  |  |  |                     .iter()
 | 
			
		
	
		
			
				
					|  |  |  |                     .map(|&x| Fr::from(x))
 | 
			
		
	
		
			
				
					|  |  |  |                     .collect();
 | 
			
		
	
		
			
				
					|  |  |  |     fn test_permutation_check(nv: usize) -> Result<(), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         let mut rng = test_rng();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 let eval = prod_0.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |         let poly_info = VPAuxInfo {
 | 
			
		
	
		
			
				
					|  |  |  |             max_degree: 2,
 | 
			
		
	
		
			
				
					|  |  |  |             num_variables: nv,
 | 
			
		
	
		
			
				
					|  |  |  |             phantom: PhantomData::default(),
 | 
			
		
	
		
			
				
					|  |  |  |         };
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         {
 | 
			
		
	
		
			
				
					|  |  |  |             // good path: w is a permutation of w itself under the identify map
 | 
			
		
	
		
			
				
					|  |  |  |             let w = DenseMultilinearExtension::rand(nv, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             // s_perm is the identity map
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm = identity_permutation_mle(nv);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let (proof, q_x) = test_permutation_check_helper(&w, &w, &s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let mut transcript = <PolyIOP<Fr> as PermutationCheck<Fr>>::init_transcript();
 | 
			
		
	
		
			
				
					|  |  |  |             transcript.append_message(b"testing", b"initializing transcript for testing")?;
 | 
			
		
	
		
			
				
					|  |  |  |             let subclaim =
 | 
			
		
	
		
			
				
					|  |  |  |                 <PolyIOP<Fr> as PermutationCheck<Fr>>::verify(&proof, &poly_info, &mut transcript)?
 | 
			
		
	
		
			
				
					|  |  |  |                     .zero_check_sub_claim;
 | 
			
		
	
		
			
				
					|  |  |  |             assert_eq!(
 | 
			
		
	
		
			
				
					|  |  |  |                 q_x.evaluate(&subclaim.sum_check_sub_claim.point).unwrap(),
 | 
			
		
	
		
			
				
					|  |  |  |                 subclaim.sum_check_sub_claim.expected_evaluation,
 | 
			
		
	
		
			
				
					|  |  |  |                 "wrong subclaim"
 | 
			
		
	
		
			
				
					|  |  |  |             );
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             // test q_x is a 0 over boolean hypercube
 | 
			
		
	
		
			
				
					|  |  |  |             for i in 0..1 << nv {
 | 
			
		
	
		
			
				
					|  |  |  |                 let bit_sequence = bit_decompose(i, nv);
 | 
			
		
	
		
			
				
					|  |  |  |                 let eval: Vec<Fr> = bit_sequence.iter().map(|x| Fr::from(*x as u64)).collect();
 | 
			
		
	
		
			
				
					|  |  |  |                 let res = q_x.evaluate(&eval)?;
 | 
			
		
	
		
			
				
					|  |  |  |                 assert!(res.is_zero())
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 let w_eval = w.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let s_id_eval = s_id.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let s_perm_eval = s_perm.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let eval_rec =
 | 
			
		
	
		
			
				
					|  |  |  |                     (w_eval + beta * s_id_eval + gamma) / (w_eval + beta * s_perm_eval + gamma);
 | 
			
		
	
		
			
				
					|  |  |  |         {
 | 
			
		
	
		
			
				
					|  |  |  |             // bad path 1: w is a not permutation of w itself under a random map
 | 
			
		
	
		
			
				
					|  |  |  |             let w = DenseMultilinearExtension::rand(nv, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             // s_perm is a random map
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm = random_permutation_mle(nv, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let (proof, q_x) = test_permutation_check_helper(&w, &w, &s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let mut transcript = <PolyIOP<Fr> as PermutationCheck<Fr>>::init_transcript();
 | 
			
		
	
		
			
				
					|  |  |  |             transcript.append_message(b"testing", b"initializing transcript for testing")?;
 | 
			
		
	
		
			
				
					|  |  |  |             if nv != 1 {
 | 
			
		
	
		
			
				
					|  |  |  |                 assert!(<PolyIOP<Fr> as PermutationCheck<Fr>>::verify(
 | 
			
		
	
		
			
				
					|  |  |  |                     &proof,
 | 
			
		
	
		
			
				
					|  |  |  |                     &poly_info,
 | 
			
		
	
		
			
				
					|  |  |  |                     &mut transcript
 | 
			
		
	
		
			
				
					|  |  |  |                 )
 | 
			
		
	
		
			
				
					|  |  |  |                 .is_err());
 | 
			
		
	
		
			
				
					|  |  |  |             } else {
 | 
			
		
	
		
			
				
					|  |  |  |                 // a trivial poly is always a permutation of itself, so the zero check should
 | 
			
		
	
		
			
				
					|  |  |  |                 // pass
 | 
			
		
	
		
			
				
					|  |  |  |                 let subclaim = <PolyIOP<Fr> as PermutationCheck<Fr>>::verify(
 | 
			
		
	
		
			
				
					|  |  |  |                     &proof,
 | 
			
		
	
		
			
				
					|  |  |  |                     &poly_info,
 | 
			
		
	
		
			
				
					|  |  |  |                     &mut transcript,
 | 
			
		
	
		
			
				
					|  |  |  |                 )?
 | 
			
		
	
		
			
				
					|  |  |  |                 .zero_check_sub_claim;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 // the evaluation should fail because a different s_perm is used for proof and
 | 
			
		
	
		
			
				
					|  |  |  |                 // for w |-> w mapping
 | 
			
		
	
		
			
				
					|  |  |  |                 assert_ne!(
 | 
			
		
	
		
			
				
					|  |  |  |                     q_x.evaluate(&subclaim.sum_check_sub_claim.point).unwrap(),
 | 
			
		
	
		
			
				
					|  |  |  |                     subclaim.sum_check_sub_claim.expected_evaluation,
 | 
			
		
	
		
			
				
					|  |  |  |                     "wrong subclaim"
 | 
			
		
	
		
			
				
					|  |  |  |                 );
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 assert_eq!(eval, eval_rec);
 | 
			
		
	
		
			
				
					|  |  |  |         {
 | 
			
		
	
		
			
				
					|  |  |  |             // bad path 2: f is a not permutation of g under a identity map
 | 
			
		
	
		
			
				
					|  |  |  |             let f = DenseMultilinearExtension::rand(nv, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  |             let g = DenseMultilinearExtension::rand(nv, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             // s_perm is the identity map
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm = identity_permutation_mle(nv);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let (proof, q_x) = test_permutation_check_helper(&f, &g, &s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let mut transcript = <PolyIOP<Fr> as PermutationCheck<Fr>>::init_transcript();
 | 
			
		
	
		
			
				
					|  |  |  |             transcript.append_message(b"testing", b"initializing transcript for testing")?;
 | 
			
		
	
		
			
				
					|  |  |  |             if nv != 1 {
 | 
			
		
	
		
			
				
					|  |  |  |                 assert!(<PolyIOP<Fr> as PermutationCheck<Fr>>::verify(
 | 
			
		
	
		
			
				
					|  |  |  |                     &proof,
 | 
			
		
	
		
			
				
					|  |  |  |                     &poly_info,
 | 
			
		
	
		
			
				
					|  |  |  |                     &mut transcript
 | 
			
		
	
		
			
				
					|  |  |  |                 )
 | 
			
		
	
		
			
				
					|  |  |  |                 .is_err());
 | 
			
		
	
		
			
				
					|  |  |  |             } else {
 | 
			
		
	
		
			
				
					|  |  |  |                 // a trivial poly is always a permutation of itself, so the zero check should
 | 
			
		
	
		
			
				
					|  |  |  |                 // pass
 | 
			
		
	
		
			
				
					|  |  |  |                 let subclaim = <PolyIOP<Fr> as PermutationCheck<Fr>>::verify(
 | 
			
		
	
		
			
				
					|  |  |  |                     &proof,
 | 
			
		
	
		
			
				
					|  |  |  |                     &poly_info,
 | 
			
		
	
		
			
				
					|  |  |  |                     &mut transcript,
 | 
			
		
	
		
			
				
					|  |  |  |                 )?
 | 
			
		
	
		
			
				
					|  |  |  |                 .zero_check_sub_claim;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 // the evaluation should fail because a different s_perm is used for proof and
 | 
			
		
	
		
			
				
					|  |  |  |                 // for f |-> g mapping
 | 
			
		
	
		
			
				
					|  |  |  |                 assert_ne!(
 | 
			
		
	
		
			
				
					|  |  |  |                     q_x.evaluate(&subclaim.sum_check_sub_claim.point).unwrap(),
 | 
			
		
	
		
			
				
					|  |  |  |                     subclaim.sum_check_sub_claim.expected_evaluation,
 | 
			
		
	
		
			
				
					|  |  |  |                     "wrong subclaim"
 | 
			
		
	
		
			
				
					|  |  |  |                 );
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
		
			
				
					|  |  |  |         }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         Ok(())
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     #[test]
 | 
			
		
	
		
			
				
					|  |  |  |     fn test_trivial_polynomial() -> Result<(), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         test_permutation_check(1)
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  |     #[test]
 | 
			
		
	
		
			
				
					|  |  |  |     fn test_normal_polynomial() -> Result<(), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         test_permutation_check(5)
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |     #[test]
 | 
			
		
	
		
			
				
					|  |  |  |     fn zero_polynomial_should_error() -> Result<(), PolyIOPErrors> {
 | 
			
		
	
		
			
				
					|  |  |  |         assert!(test_permutation_check(0).is_err());
 | 
			
		
	
		
			
				
					|  |  |  |         Ok(())
 | 
			
		
	
		
			
				
					|  |  |  |     }
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  | @ -208,19 +621,20 @@ mod test { | 
			
		
	
		
			
				
					|  |  |  |         let mut rng = test_rng();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |         for num_vars in 2..6 {
 | 
			
		
	
		
			
				
					|  |  |  |             let w_vec: Vec<Fr> = (0..(1 << num_vars)).map(|_| Fr::rand(&mut rng)).collect();
 | 
			
		
	
		
			
				
					|  |  |  |             let w = DenseMultilinearExtension::from_evaluations_vec(num_vars, w_vec);
 | 
			
		
	
		
			
				
					|  |  |  |             let f = DenseMultilinearExtension::rand(num_vars, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  |             let g = DenseMultilinearExtension::rand(num_vars, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let s_id = identity_permutation_mle::<Fr>(num_vars);
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm = random_permutation_mle(num_vars, &mut rng);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm_vec: Vec<Fr> = (0..(1 << num_vars)).map(|_| Fr::rand(&mut rng)).collect();
 | 
			
		
	
		
			
				
					|  |  |  |             let s_perm = DenseMultilinearExtension::from_evaluations_vec(num_vars, s_perm_vec);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             let beta = Fr::rand(&mut rng);
 | 
			
		
	
		
			
				
					|  |  |  |             let gamma = Fr::rand(&mut rng);
 | 
			
		
	
		
			
				
					|  |  |  |             let mut transcript = <PolyIOP<Fr> as PermutationCheck<Fr>>::init_transcript();
 | 
			
		
	
		
			
				
					|  |  |  |             transcript.append_message(b"testing", b"initializing transcript for testing")?;
 | 
			
		
	
		
			
				
					|  |  |  |             let challenge =
 | 
			
		
	
		
			
				
					|  |  |  |                 <PolyIOP<Fr> as PermutationCheck<Fr>>::generate_challenge(&mut transcript)?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             // TODO: also test other 4 polynomials
 | 
			
		
	
		
			
				
					|  |  |  |             let res = compute_products(&beta, &gamma, &w, &s_id, &s_perm)?;
 | 
			
		
	
		
			
				
					|  |  |  |             let res = <PolyIOP<Fr> as PermutationCheck<Fr>>::compute_products(
 | 
			
		
	
		
			
				
					|  |  |  |                 &challenge, &f, &g, &s_perm,
 | 
			
		
	
		
			
				
					|  |  |  |             )?;
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |             for i in 0..1 << num_vars {
 | 
			
		
	
		
			
				
					|  |  |  |                 let r: Vec<Fr> = bit_decompose(i, num_vars)
 | 
			
		
	
	
		
			
				
					|  |  | @ -230,11 +644,12 @@ mod test { | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 let eval = res[1].evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 let w_eval = w.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let f_eval = f.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let g_eval = g.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let s_id_eval = s_id.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let s_perm_eval = s_perm.evaluate(&r).unwrap();
 | 
			
		
	
		
			
				
					|  |  |  |                 let eval_rec =
 | 
			
		
	
		
			
				
					|  |  |  |                     (w_eval + beta * s_id_eval + gamma) / (w_eval + beta * s_perm_eval + gamma);
 | 
			
		
	
		
			
				
					|  |  |  |                 let eval_rec = (f_eval + challenge.beta * s_id_eval + challenge.gamma)
 | 
			
		
	
		
			
				
					|  |  |  |                     / (g_eval + challenge.beta * s_perm_eval + challenge.gamma);
 | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |                 assert_eq!(eval, eval_rec);
 | 
			
		
	
		
			
				
					|  |  |  |             }
 | 
			
		
	
	
		
			
				
					|  |  | 
 |