You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

260 lines
7.8 KiB

  1. use ark_ec::AffineRepr;
  2. use ark_std::{One, Zero};
  3. use std::marker::PhantomData;
  4. use crate::pedersen::{Commitment, Params as PedersenParams, Pedersen};
  5. use crate::r1cs::*;
  6. use crate::transcript::Transcript;
  7. use crate::utils::*;
  8. use ark_std::{
  9. rand::{Rng, RngCore},
  10. UniformRand,
  11. };
  12. // Phi: φ in the paper (later 𝖴), a folded instance
  13. pub struct Phi<C: AffineRepr> {
  14. cmE: Commitment<C>, // TODO do not contain Commitment, only Commitment.cm (without the randomness)
  15. u: C::ScalarField,
  16. cmW: Commitment<C>, // TODO same as in cmE
  17. x: Vec<C::ScalarField>,
  18. }
  19. // FWit: Folded Witness
  20. pub struct FWit<C: AffineRepr> {
  21. E: Vec<C::ScalarField>,
  22. rE: C::ScalarField,
  23. W: Vec<C::ScalarField>,
  24. rW: C::ScalarField,
  25. }
  26. impl<C: AffineRepr> FWit<C> {
  27. pub fn commit(&self, params: &PedersenParams<C>, x: &Vec<C::ScalarField>) -> Phi<C> {
  28. let cmE = Pedersen::commit(&params, &self.E, &self.rE);
  29. let cmW = Pedersen::commit(&params, &self.W, &self.rW);
  30. Phi {
  31. cmE,
  32. u: C::ScalarField::one(),
  33. cmW,
  34. x: x.clone(),
  35. }
  36. }
  37. }
  38. pub struct NIFS<C: AffineRepr> {
  39. _phantom: PhantomData<C>,
  40. }
  41. impl<C: AffineRepr> NIFS<C> {
  42. pub fn comp_T(
  43. cs1: RelaxedR1CS<C::ScalarField>,
  44. cs2: RelaxedR1CS<C::ScalarField>,
  45. z1: &Vec<C::ScalarField>,
  46. z2: &Vec<C::ScalarField>,
  47. ) -> Vec<C::ScalarField> {
  48. // assuming cs1.R1CS == cs2.R1CS
  49. let (A, B, C) = (cs1.ABC.A, cs1.ABC.B, cs1.ABC.C);
  50. // this is parallelizable (for the future)
  51. let Az1 = matrix_vector_product(&A, &z1);
  52. let Bz1 = matrix_vector_product(&B, &z1);
  53. let Cz1 = matrix_vector_product(&C, &z1);
  54. let Az2 = matrix_vector_product(&A, &z2);
  55. let Bz2 = matrix_vector_product(&B, &z2);
  56. let Cz2 = matrix_vector_product(&C, &z2);
  57. let Az1_Bz2 = hadamard_product(Az1, Bz2);
  58. let Az2_Bz1 = hadamard_product(Az2, Bz1);
  59. let u1Cz2 = vector_elem_product(&Cz2, &cs1.u);
  60. let u2Cz1 = vector_elem_product(&Cz1, &cs2.u);
  61. // let T = vec_sub(vec_sub(vec_add(Az1_Bz2, Az2_Bz1), u1Cz2), u2Cz1);
  62. let T = ((Ve(Az1_Bz2) + Ve(Az2_Bz1)) - Ve(u1Cz2)) - Ve(u2Cz1);
  63. T.0
  64. }
  65. pub fn fold_witness(
  66. r: C::ScalarField,
  67. fw1: &FWit<C>,
  68. fw2: &FWit<C>,
  69. T: Vec<C::ScalarField>,
  70. rT: C::ScalarField,
  71. ) -> FWit<C> {
  72. let r2 = r * r;
  73. let E: Vec<C::ScalarField> = vec_add(
  74. // this syntax will be simplified with future operators impl (or at least a method
  75. // for r-lin)
  76. &vec_add(&fw1.E, &vector_elem_product(&T, &r)),
  77. &vector_elem_product(&fw2.E, &r2),
  78. );
  79. let rE = fw1.rE + r * rT + r2 * fw2.rE;
  80. let W = vec_add(&fw1.W, &vector_elem_product(&fw2.W, &r));
  81. let rW = fw1.rW + r * fw2.rW;
  82. FWit::<C> {
  83. E: E.into(),
  84. rE,
  85. W: W.into(),
  86. rW,
  87. }
  88. }
  89. pub fn fold_instance(
  90. r: C::ScalarField,
  91. phi1: Phi<C>,
  92. phi2: Phi<C>,
  93. cmT: Commitment<C>,
  94. ) -> Phi<C> {
  95. let r2 = r * r;
  96. let cmE = phi1.cmE.0 + cmT.0.mul(r) + phi2.cmE.0.mul(r2);
  97. let u = phi1.u + r * phi2.u;
  98. let cmW = phi1.cmW.0 + phi2.cmW.0.mul(r);
  99. let x = vec_add(&phi1.x, &vector_elem_product(&phi2.x, &r));
  100. // let x = rlin(phi1.x, phi2.x, r);
  101. Phi::<C> {
  102. cmE: Commitment(cmE.into()),
  103. u,
  104. cmW: Commitment(cmW.into()),
  105. x,
  106. }
  107. }
  108. }
  109. #[cfg(test)]
  110. mod tests {
  111. use super::*;
  112. use crate::pedersen::Pedersen;
  113. use ark_bn254::{g1::G1Affine, Fr};
  114. use ark_ec::CurveGroup;
  115. use ark_std::{
  116. rand::{Rng, RngCore},
  117. UniformRand,
  118. };
  119. use ark_std::{One, Zero};
  120. use std::ops::Mul;
  121. #[test]
  122. fn test_simple_folding() {
  123. let mut rng = ark_std::test_rng();
  124. // R1CS for: x^3 + x + 5 = y (example from article
  125. // https://www.vitalik.ca/general/2016/12/10/qap.html )
  126. let A = to_F_matrix::<Fr>(vec![
  127. vec![0, 1, 0, 0, 0, 0],
  128. vec![0, 0, 0, 1, 0, 0],
  129. vec![0, 1, 0, 0, 1, 0],
  130. vec![5, 0, 0, 0, 0, 1],
  131. ]);
  132. let B = to_F_matrix::<Fr>(vec![
  133. vec![0, 1, 0, 0, 0, 0],
  134. vec![0, 1, 0, 0, 0, 0],
  135. vec![1, 0, 0, 0, 0, 0],
  136. vec![1, 0, 0, 0, 0, 0],
  137. ]);
  138. let C = to_F_matrix::<Fr>(vec![
  139. vec![0, 0, 0, 1, 0, 0],
  140. vec![0, 0, 0, 0, 1, 0],
  141. vec![0, 0, 0, 0, 0, 1],
  142. vec![0, 0, 1, 0, 0, 0],
  143. ]);
  144. let z1 = to_F_vec::<Fr>(vec![1, 3, 35, 9, 27, 30]);
  145. let x1 = to_F_vec::<Fr>(vec![35]);
  146. let z2 = to_F_vec::<Fr>(vec![1, 4, 73, 16, 64, 68]);
  147. let x2 = to_F_vec::<Fr>(vec![73]);
  148. let relaxed_r1cs_1 = R1CS::<Fr> {
  149. A: A.clone(),
  150. B: B.clone(),
  151. C: C.clone(),
  152. }
  153. .relax();
  154. let relaxed_r1cs_2 = R1CS::<Fr> {
  155. A: A.clone(),
  156. B: B.clone(),
  157. C: C.clone(),
  158. }
  159. .relax();
  160. let T = NIFS::<G1Affine>::comp_T(relaxed_r1cs_1, relaxed_r1cs_2, &z1, &z2);
  161. let pedersen_params = Pedersen::<G1Affine>::new_params(&mut rng, 100); // 100 is wip, will get it from actual vec
  162. let rT: Fr = Fr::rand(&mut rng);
  163. let cmT = Pedersen::commit(&pedersen_params, &T, &rT);
  164. let r = Fr::rand(&mut rng); // this would come from the transcript
  165. // WIP TMP
  166. let fw1 = FWit::<G1Affine> {
  167. E: vec![Fr::zero(); T.len()],
  168. rE: Fr::one(),
  169. W: z1,
  170. rW: Fr::one(),
  171. };
  172. let fw2 = FWit::<G1Affine> {
  173. E: vec![Fr::zero(); T.len()],
  174. rE: Fr::one(),
  175. W: z2,
  176. rW: Fr::one(),
  177. };
  178. // fold witness
  179. let fw3 = NIFS::<G1Affine>::fold_witness(r, &fw1, &fw2, T, rT);
  180. // get committed instances
  181. let phi1 = fw1.commit(&pedersen_params, &x1); // wip
  182. let phi2 = fw2.commit(&pedersen_params, &x2);
  183. // fold instance
  184. let phi3 = NIFS::<G1Affine>::fold_instance(r, phi1, phi2, cmT);
  185. // naive check that the folded witness satisfies the relaxed r1cs
  186. let Az = matrix_vector_product(&A, &fw3.W);
  187. let Bz = matrix_vector_product(&B, &fw3.W);
  188. let Cz = matrix_vector_product(&C, &fw3.W);
  189. assert_eq!(
  190. hadamard_product(Az, Bz),
  191. vec_add(&vector_elem_product(&Cz, &phi3.u), &fw3.E)
  192. );
  193. // check that folded commitments from folded instance (phi) are equal to folding the
  194. // witnesses and committing into it
  195. let x3 = vec_add(&x1, &vector_elem_product(&x2, &r));
  196. // use folded rE, rW to commit fw3
  197. let phi3_expected = fw3.commit(&pedersen_params, &x3);
  198. assert_eq!(phi3_expected.cmE.0, phi3.cmE.0);
  199. assert_eq!(phi3_expected.cmW.0, phi3.cmW.0);
  200. // init Prover's transcript
  201. let mut transcript_p: Transcript<Fr> = Transcript::<Fr>::new();
  202. // init Verifier's transcript
  203. let mut transcript_v: Transcript<Fr> = Transcript::<Fr>::new();
  204. // check openings of phi3.cmE and phi3.cmW
  205. let phi3_cmE_proof = Pedersen::prove(
  206. &pedersen_params,
  207. &mut transcript_p,
  208. &phi3.cmE,
  209. fw3.E,
  210. fw3.rE,
  211. );
  212. let v = Pedersen::verify(
  213. &pedersen_params,
  214. &mut transcript_v,
  215. phi3.cmE,
  216. phi3_cmE_proof,
  217. );
  218. let phi3_cmW_proof = Pedersen::prove(
  219. &pedersen_params,
  220. &mut transcript_p,
  221. &phi3.cmW,
  222. fw3.W,
  223. fw3.rW,
  224. );
  225. let v = Pedersen::verify(
  226. &pedersen_params,
  227. &mut transcript_v,
  228. phi3.cmW,
  229. phi3_cmW_proof,
  230. );
  231. }
  232. }