-
Notifications
You must be signed in to change notification settings - Fork 90
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Witness multiplication optimization #70
base: master
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -35,7 +35,8 @@ pub struct ProverState<'a, F: PrimeField> { | |
zk_bound: usize, | ||
|
||
w_poly: Option<LabeledPolynomial<F>>, | ||
mz_polys: Option<(LabeledPolynomial<F>, LabeledPolynomial<F>)>, | ||
mz_rands: Option<(F, F)>, | ||
mz_rand_polys: Option<(LabeledPolynomial<F>, LabeledPolynomial<F>)>, | ||
|
||
index: &'a Index<F>, | ||
|
||
|
@@ -294,7 +295,8 @@ impl<F: PrimeField> AHPForR1CS<F> { | |
z_a: Some(z_a), | ||
z_b: Some(z_b), | ||
w_poly: None, | ||
mz_polys: None, | ||
mz_rands: None, | ||
mz_rand_polys: None, | ||
zk_bound, | ||
index, | ||
verifier_first_msg: None, | ||
|
@@ -356,14 +358,20 @@ impl<F: PrimeField> AHPForR1CS<F> { | |
|
||
let z_a_poly_time = start_timer!(|| "Computing z_A polynomial"); | ||
let z_a = state.z_a.clone().unwrap(); | ||
let z_a_poly = &EvaluationsOnDomain::from_vec_and_domain(z_a, domain_h).interpolate() | ||
+ &(&DensePolynomial::from_coefficients_slice(&[F::rand(rng)]) * &v_H); | ||
// Generate a randomized polynomial that agrees with z_a on H | ||
let r_a = F::rand(rng); | ||
let randomized_z_a_poly = &EvaluationsOnDomain::from_vec_and_domain(z_a, domain_h) | ||
.interpolate() | ||
+ &(&DensePolynomial::from_coefficients_slice(&[r_a]) * &v_H); | ||
end_timer!(z_a_poly_time); | ||
|
||
let z_b_poly_time = start_timer!(|| "Computing z_B polynomial"); | ||
let z_b = state.z_b.clone().unwrap(); | ||
let z_b_poly = &EvaluationsOnDomain::from_vec_and_domain(z_b, domain_h).interpolate() | ||
+ &(&DensePolynomial::from_coefficients_slice(&[F::rand(rng)]) * &v_H); | ||
// Generate a randomized polynomial that agrees with z_b on H | ||
let r_b = F::rand(rng); | ||
let randomized_z_b_poly = &EvaluationsOnDomain::from_vec_and_domain(z_b, domain_h) | ||
.interpolate() | ||
+ &(&DensePolynomial::from_coefficients_slice(&[r_b]) * &v_H); | ||
end_timer!(z_b_poly_time); | ||
|
||
let mask_poly_time = start_timer!(|| "Computing mask polynomial"); | ||
|
@@ -376,13 +384,13 @@ impl<F: PrimeField> AHPForR1CS<F> { | |
let msg = ProverMsg::EmptyMessage; | ||
|
||
assert!(w_poly.degree() < domain_h.size() - domain_x.size() + zk_bound); | ||
assert!(z_a_poly.degree() < domain_h.size() + zk_bound); | ||
assert!(z_b_poly.degree() < domain_h.size() + zk_bound); | ||
assert!(randomized_z_a_poly.degree() < domain_h.size() + zk_bound); | ||
assert!(randomized_z_b_poly.degree() < domain_h.size() + zk_bound); | ||
assert!(mask_poly.degree() <= 3 * domain_h.size() + 2 * zk_bound - 3); | ||
|
||
let w = LabeledPolynomial::new("w".to_string(), w_poly, None, Some(1)); | ||
let z_a = LabeledPolynomial::new("z_a".to_string(), z_a_poly, None, Some(1)); | ||
let z_b = LabeledPolynomial::new("z_b".to_string(), z_b_poly, None, Some(1)); | ||
let z_a = LabeledPolynomial::new("z_a".to_string(), randomized_z_a_poly, None, Some(1)); | ||
let z_b = LabeledPolynomial::new("z_b".to_string(), randomized_z_b_poly, None, Some(1)); | ||
let mask_poly = | ||
LabeledPolynomial::new("mask_poly".to_string(), mask_poly.clone(), None, None); | ||
|
||
|
@@ -394,7 +402,8 @@ impl<F: PrimeField> AHPForR1CS<F> { | |
}; | ||
|
||
state.w_poly = Some(w); | ||
state.mz_polys = Some((z_a, z_b)); | ||
state.mz_rands = Some((r_a, r_b)); | ||
state.mz_rand_polys = Some((z_a, z_b)); | ||
state.mask_poly = Some(mask_poly); | ||
end_timer!(round_time); | ||
|
||
|
@@ -456,17 +465,32 @@ impl<F: PrimeField> AHPForR1CS<F> { | |
} = *ver_message; | ||
|
||
let summed_z_m_poly_time = start_timer!(|| "Compute z_m poly"); | ||
let (z_a_poly, z_b_poly) = state.mz_polys.as_ref().unwrap(); | ||
let z_c_poly = z_a_poly.polynomial() * z_b_poly.polynomial(); | ||
|
||
let (r_a, r_b) = state.mz_rands.clone().unwrap(); | ||
let (randomized_z_a_poly, randomized_z_b_poly) = state.mz_rand_polys.as_ref().unwrap(); | ||
|
||
// Recover the non-randomized interpolation of z_a and z_b over H | ||
let v_H: DensePolynomial<F> = domain_h.vanishing_polynomial().into(); | ||
let z_a_poly = randomized_z_a_poly.polynomial() - &(&v_H * r_a); | ||
Comment on lines
+473
to
+474
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Oh hm I meant that we should instead mutate There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. (and also a way to multiply a sparse polynomial by a scalar, mirroring the one for There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Gotcha, so the problem with that is I'll make the other changes mentioned for now though! There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hmm that’s annoying... |
||
let z_b_poly = randomized_z_b_poly.polynomial() - &(&v_H * r_b); | ||
|
||
// Calculate the terms of z_c = (z_a_poly + r_A * v_H) * (z_b_poly + r_B * v_H) | ||
let t1 = &z_a_poly * &z_b_poly; | ||
let t2 = (&z_b_poly * r_a).mul_by_vanishing_poly(domain_h); | ||
let t3 = (&z_a_poly * r_b).mul_by_vanishing_poly(domain_h); | ||
let t4 = DensePolynomial::from_coefficients_slice(&[r_a * r_b]) | ||
.mul_by_vanishing_poly(domain_h) | ||
.mul_by_vanishing_poly(domain_h); | ||
let z_c_poly = t1 + t2 + t3 + t4; | ||
|
||
let mut summed_z_m_coeffs = z_c_poly.coeffs; | ||
// Note: Can't combine these two loops, because z_c_poly has 2x the degree | ||
// of z_a_poly and z_b_poly, so the second loop gets truncated due to | ||
// the `zip`s. | ||
cfg_iter_mut!(summed_z_m_coeffs).for_each(|c| *c *= &eta_c); | ||
cfg_iter_mut!(summed_z_m_coeffs) | ||
.zip(&z_a_poly.polynomial().coeffs) | ||
.zip(&z_b_poly.polynomial().coeffs) | ||
.zip(&randomized_z_a_poly.polynomial().coeffs) | ||
.zip(&randomized_z_b_poly.polynomial().coeffs) | ||
.for_each(|((c, a), b)| *c += &(eta_a * a + &(eta_b * b))); | ||
|
||
let summed_z_m = DensePolynomial::from_coefficients_vec(summed_z_m_coeffs); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can there be a comment here explaining the operation these 3 lines are doing? (Ditto for z_b)
I believe its "go from z_a evaluations to z_a_poly as
interpolate(z_a_evaluations over H) * (Z_I)) + rand_poly
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's computed as
interpolate(z_a_evaluations_over_H) + v_H * r_a