halo2_proofs/poly/commitment/
verifier.rs
1use group::{
2 ff::{BatchInvert, Field},
3 Curve,
4};
5
6use super::super::Error;
7use super::{Params, MSM};
8use crate::transcript::{EncodedChallenge, TranscriptRead};
9
10use crate::arithmetic::{best_multiexp, CurveAffine};
11
12#[derive(Debug, Clone)]
14pub struct Guard<'a, C: CurveAffine, E: EncodedChallenge<C>> {
15 msm: MSM<'a, C>,
16 neg_c: C::Scalar,
17 u: Vec<C::Scalar>,
18 u_packed: Vec<E>,
19}
20
21#[derive(Debug, Clone)]
23pub struct Accumulator<C: CurveAffine, E: EncodedChallenge<C>> {
24 pub g: C,
26
27 pub u_packed: Vec<E>,
30}
31
32impl<'a, C: CurveAffine, E: EncodedChallenge<C>> Guard<'a, C, E> {
33 pub fn use_challenges(mut self) -> MSM<'a, C> {
36 let s = compute_s(&self.u, self.neg_c);
37 self.msm.add_to_g_scalars(&s);
38
39 self.msm
40 }
41
42 pub fn use_g(mut self, g: C) -> (MSM<'a, C>, Accumulator<C, E>) {
45 self.msm.append_term(self.neg_c, g);
46
47 let accumulator = Accumulator {
48 g,
49 u_packed: self.u_packed,
50 };
51
52 (self.msm, accumulator)
53 }
54
55 pub fn compute_g(&self) -> C {
57 let s = compute_s(&self.u, C::Scalar::one());
58
59 best_multiexp(&s, &self.msm.params.g).to_affine()
60 }
61}
62
63pub fn verify_proof<'a, C: CurveAffine, E: EncodedChallenge<C>, T: TranscriptRead<C, E>>(
67 params: &'a Params<C>,
68 mut msm: MSM<'a, C>,
69 transcript: &mut T,
70 x: C::Scalar,
71 v: C::Scalar,
72) -> Result<Guard<'a, C, E>, Error> {
73 let k = params.k as usize;
74
75 msm.add_constant_term(-v); let s_poly_commitment = transcript.read_point().map_err(|_| Error::OpeningError)?;
78 let xi = *transcript.squeeze_challenge_scalar::<()>();
79 msm.append_term(xi, s_poly_commitment);
80
81 let z = *transcript.squeeze_challenge_scalar::<()>();
82
83 let mut rounds = vec![];
84 for _ in 0..k {
85 let l = transcript.read_point().map_err(|_| Error::OpeningError)?;
87 let r = transcript.read_point().map_err(|_| Error::OpeningError)?;
88
89 let u_j_packed = transcript.squeeze_challenge();
90 let u_j = *u_j_packed.as_challenge_scalar::<()>();
91
92 rounds.push((l, r, u_j, u_j, u_j_packed));
93 }
94
95 rounds
96 .iter_mut()
97 .map(|&mut (_, _, _, ref mut u_j, _)| u_j)
98 .batch_invert();
99
100 let mut u = Vec::with_capacity(k);
103 let mut u_packed: Vec<E> = Vec::with_capacity(k);
104 for (l, r, u_j, u_j_inv, u_j_packed) in rounds {
105 msm.append_term(u_j_inv, l);
106 msm.append_term(u_j, r);
107
108 u.push(u_j);
109 u_packed.push(u_j_packed);
110 }
111
112 let c = transcript.read_scalar().map_err(|_| Error::SamplingError)?;
124 let neg_c = -c;
125 let f = transcript.read_scalar().map_err(|_| Error::SamplingError)?;
126 let b = compute_b(x, &u);
127
128 msm.add_to_u_scalar(neg_c * &b * &z);
129 msm.add_to_w_scalar(-f);
130
131 let guard = Guard {
132 msm,
133 neg_c,
134 u,
135 u_packed,
136 };
137
138 Ok(guard)
139}
140
141fn compute_b<F: Field>(x: F, u: &[F]) -> F {
143 let mut tmp = F::one();
144 let mut cur = x;
145 for u_j in u.iter().rev() {
146 tmp *= F::one() + &(*u_j * &cur);
147 cur *= cur;
148 }
149 tmp
150}
151
152fn compute_s<F: Field>(u: &[F], init: F) -> Vec<F> {
154 assert!(!u.is_empty());
155 let mut v = vec![F::zero(); 1 << u.len()];
156 v[0] = init;
157
158 for (len, u_j) in u.iter().rev().enumerate().map(|(i, u_j)| (1 << i, u_j)) {
159 let (left, right) = v.split_at_mut(len);
160 let right = &mut right[0..len];
161 right.copy_from_slice(left);
162 for v in right {
163 *v *= u_j;
164 }
165 }
166
167 v
168}