1use crate::{
17 AlgebraicSponge,
18 fft::{DensePolynomial, EvaluationDomain},
19};
20use snarkvm_curves::{AffineCurve, PairingCurve, PairingEngine, ProjectiveCurve};
21use snarkvm_fields::{ConstraintFieldError, ToConstraintField, Zero};
22use snarkvm_parameters::mainnet::PowersOfG;
23use snarkvm_utilities::{
24 FromBytes,
25 ToBytes,
26 error,
27 into_io_error,
28 serialize::{CanonicalDeserialize, CanonicalSerialize, Compress, SerializationError, Valid, Validate},
29};
30
31use crate::srs::{UniversalProver, UniversalVerifier};
32use anyhow::Result;
33use core::ops::{Add, AddAssign};
34use rand::RngCore;
35use std::{
36 borrow::Cow,
37 collections::BTreeMap,
38 io,
39 io::{Read, Write},
40 ops::Range,
41 sync::Arc,
42};
43
44#[derive(Clone, Debug)]
46pub struct UniversalParams<E: PairingEngine> {
47 powers: Arc<PowersOfG<E>>,
53 pub h: E::G2Affine,
55 pub prepared_h: <E::G2Affine as PairingCurve>::Prepared,
57 pub prepared_beta_h: <E::G2Affine as PairingCurve>::Prepared,
59}
60
61impl<E: PairingEngine> UniversalParams<E> {
62 pub fn load() -> Result<Self> {
63 let powers = Arc::new(PowersOfG::<E>::load()?);
64 let h = E::G2Affine::prime_subgroup_generator();
65 let prepared_h = h.prepare();
66 let prepared_beta_h = powers.beta_h().prepare();
67
68 Ok(Self { powers, h, prepared_h, prepared_beta_h })
69 }
70
71 pub fn download_powers_for(&self, range: Range<usize>) -> Result<()> {
72 self.powers.download_powers_for(range)
73 }
74
75 pub fn lagrange_basis(&self, domain: EvaluationDomain<E::Fr>) -> Result<Vec<E::G1Affine>> {
76 let basis = domain
77 .ifft(&self.powers_of_beta_g(0, domain.size())?.iter().map(|e| (*e).to_projective()).collect::<Vec<_>>());
78 Ok(E::G1Projective::batch_normalization_into_affine(basis))
79 }
80
81 pub fn power_of_beta_g(&self, index: usize) -> Result<E::G1Affine> {
82 self.powers.power_of_beta_g(index)
83 }
84
85 pub fn powers_of_beta_g(&self, lower: usize, upper: usize) -> Result<Vec<E::G1Affine>> {
86 self.powers.powers_of_beta_g(lower..upper)
87 }
88
89 pub fn powers_of_beta_times_gamma_g(&self) -> &BTreeMap<usize, E::G1Affine> {
90 self.powers.powers_of_beta_gamma_g()
91 }
92
93 pub fn beta_h(&self) -> E::G2Affine {
94 self.powers.beta_h()
95 }
96
97 pub fn max_degree(&self) -> usize {
98 self.powers.max_num_powers() - 1
99 }
100
101 pub fn to_universal_prover(&self) -> Result<UniversalProver<E>> {
102 Ok(UniversalProver::<E> { max_degree: self.max_degree(), _unused: None })
103 }
104
105 pub fn to_universal_verifier(&self) -> Result<UniversalVerifier<E>> {
106 let g = self.power_of_beta_g(0)?;
107 let h = self.h;
108 let beta_h = self.beta_h();
109 let gamma_g = self.powers_of_beta_times_gamma_g()[&0];
110 let prepared_h = self.prepared_h.clone();
111 let prepared_beta_h = self.prepared_beta_h.clone();
112
113 Ok(UniversalVerifier {
114 vk: VerifierKey::<E> { g, gamma_g, h, beta_h, prepared_h, prepared_beta_h },
115 prepared_negative_powers_of_beta_h: self.powers.prepared_negative_powers_of_beta_h(),
116 })
117 }
118}
119
120impl<E: PairingEngine> FromBytes for UniversalParams<E> {
121 fn read_le<R: Read>(mut reader: R) -> io::Result<Self> {
122 let powers = Arc::new(PowersOfG::read_le(&mut reader)?);
124
125 let h: E::G2Affine = FromBytes::read_le(&mut reader)?;
127
128 let prepared_h: <E::G2Affine as PairingCurve>::Prepared = FromBytes::read_le(&mut reader)?;
130
131 let prepared_beta_h: <E::G2Affine as PairingCurve>::Prepared = FromBytes::read_le(&mut reader)?;
133
134 Ok(Self { powers, h, prepared_h, prepared_beta_h })
135 }
136}
137
138impl<E: PairingEngine> ToBytes for UniversalParams<E> {
139 fn write_le<W: Write>(&self, mut writer: W) -> io::Result<()> {
140 self.powers.write_le(&mut writer)?;
142
143 self.h.write_le(&mut writer)?;
145
146 self.prepared_h.write_le(&mut writer)?;
148
149 self.prepared_beta_h.write_le(&mut writer)?;
151
152 Ok(())
153 }
154}
155
156#[derive(Clone, Debug, Default, Hash)]
159pub struct Powers<'a, E: PairingEngine> {
160 pub powers_of_beta_g: Cow<'a, [E::G1Affine]>,
162 pub powers_of_beta_times_gamma_g: Cow<'a, [E::G1Affine]>,
164}
165
166impl<E: PairingEngine> Powers<'_, E> {
167 pub fn size(&self) -> usize {
169 self.powers_of_beta_g.len()
170 }
171}
172#[derive(Clone, Debug, Hash)]
175pub struct LagrangeBasis<'a, E: PairingEngine> {
176 pub lagrange_basis_at_beta_g: Cow<'a, [E::G1Affine]>,
178 pub powers_of_beta_times_gamma_g: Cow<'a, [E::G1Affine]>,
180 pub domain: EvaluationDomain<E::Fr>,
183}
184
185impl<E: PairingEngine> LagrangeBasis<'_, E> {
186 pub fn size(&self) -> usize {
188 self.lagrange_basis_at_beta_g.len()
189 }
190}
191
192#[derive(Clone, Debug, Default, PartialEq, Eq)]
194pub struct VerifierKey<E: PairingEngine> {
195 pub g: E::G1Affine,
197 pub gamma_g: E::G1Affine,
199 pub h: E::G2Affine,
201 pub beta_h: E::G2Affine,
203 pub prepared_h: <E::G2Affine as PairingCurve>::Prepared,
205 pub prepared_beta_h: <E::G2Affine as PairingCurve>::Prepared,
207}
208
209impl<E: PairingEngine> CanonicalSerialize for VerifierKey<E> {
210 fn serialize_with_mode<W: Write>(&self, mut writer: W, compress: Compress) -> Result<(), SerializationError> {
211 self.g.serialize_with_mode(&mut writer, compress)?;
212 self.gamma_g.serialize_with_mode(&mut writer, compress)?;
213 self.h.serialize_with_mode(&mut writer, compress)?;
214 self.beta_h.serialize_with_mode(&mut writer, compress)?;
215 Ok(())
216 }
217
218 fn serialized_size(&self, compress: Compress) -> usize {
219 self.g.serialized_size(compress)
220 + self.gamma_g.serialized_size(compress)
221 + self.h.serialized_size(compress)
222 + self.beta_h.serialized_size(compress)
223 }
224}
225
226impl<E: PairingEngine> CanonicalDeserialize for VerifierKey<E> {
227 fn deserialize_with_mode<R: Read>(
228 mut reader: R,
229 compress: Compress,
230 validate: Validate,
231 ) -> Result<Self, SerializationError> {
232 let g = CanonicalDeserialize::deserialize_with_mode(&mut reader, compress, validate)?;
233 let gamma_g = CanonicalDeserialize::deserialize_with_mode(&mut reader, compress, validate)?;
234 let h: E::G2Affine = CanonicalDeserialize::deserialize_with_mode(&mut reader, compress, validate)?;
235 let beta_h: E::G2Affine = CanonicalDeserialize::deserialize_with_mode(&mut reader, compress, validate)?;
236 let prepared_h = h.prepare();
237 let prepared_beta_h = beta_h.prepare();
238 Ok(VerifierKey { g, gamma_g, h, beta_h, prepared_h, prepared_beta_h })
239 }
240}
241
242impl<E: PairingEngine> Valid for VerifierKey<E> {
243 fn check(&self) -> Result<(), SerializationError> {
244 Valid::check(&self.g)?;
245 Valid::check(&self.gamma_g)?;
246 Valid::check(&self.h)?;
247 Valid::check(&self.beta_h)?;
248 Ok(())
249 }
250
251 fn batch_check<'a>(batch: impl Iterator<Item = &'a Self> + Send) -> Result<(), SerializationError>
252 where
253 Self: 'a,
254 {
255 let batch: Vec<_> = batch.collect();
256 Valid::batch_check(batch.iter().map(|v| &v.g))?;
257 Valid::batch_check(batch.iter().map(|v| &v.gamma_g))?;
258 Valid::batch_check(batch.iter().map(|v| &v.h))?;
259 Valid::batch_check(batch.iter().map(|v| &v.beta_h))?;
260 Ok(())
261 }
262}
263
264impl<E: PairingEngine> FromBytes for VerifierKey<E> {
265 fn read_le<R: Read>(mut reader: R) -> io::Result<Self> {
266 CanonicalDeserialize::deserialize_compressed(&mut reader)
267 .map_err(|_| error("could not deserialize VerifierKey"))
268 }
269}
270
271impl<E: PairingEngine> ToBytes for VerifierKey<E> {
272 fn write_le<W: Write>(&self, mut writer: W) -> io::Result<()> {
273 CanonicalSerialize::serialize_compressed(self, &mut writer)
274 .map_err(|_| error("could not serialize VerifierKey"))
275 }
276}
277
278#[derive(Copy, Clone, Debug, Default, PartialEq, Eq, Hash, CanonicalSerialize, CanonicalDeserialize)]
280pub struct KZGCommitment<E: PairingEngine>(
281 pub E::G1Affine,
283);
284
285impl<E: PairingEngine> FromBytes for KZGCommitment<E> {
286 fn read_le<R: Read>(mut reader: R) -> io::Result<Self> {
287 CanonicalDeserialize::deserialize_compressed(&mut reader)
288 .map_err(|_| error("could not deserialize KZGCommitment"))
289 }
290}
291
292impl<E: PairingEngine> ToBytes for KZGCommitment<E> {
293 fn write_le<W: Write>(&self, mut writer: W) -> io::Result<()> {
294 CanonicalSerialize::serialize_compressed(self, &mut writer)
295 .map_err(|_| error("could not serialize KZGCommitment"))
296 }
297}
298
299impl<E: PairingEngine> KZGCommitment<E> {
300 #[inline]
301 pub fn empty() -> Self {
302 KZGCommitment(E::G1Affine::zero())
303 }
304
305 pub fn has_degree_bound(&self) -> bool {
306 false
307 }
308
309 pub fn is_in_correct_subgroup_assuming_on_curve(&self) -> bool {
310 self.0.is_in_correct_subgroup_assuming_on_curve()
311 }
312}
313
314impl<E: PairingEngine> ToConstraintField<E::Fq> for KZGCommitment<E> {
315 fn to_field_elements(&self) -> Result<Vec<E::Fq>, ConstraintFieldError> {
316 self.0.to_field_elements()
317 }
318}
319
320#[derive(Clone, Debug, Default, PartialEq, Eq, Hash, CanonicalSerialize, CanonicalDeserialize)]
323pub struct KZGRandomness<E: PairingEngine> {
324 pub blinding_polynomial: DensePolynomial<E::Fr>,
326}
327impl<E: PairingEngine> FromBytes for KZGRandomness<E> {
328 fn read_le<R: Read>(mut reader: R) -> io::Result<Self> {
329 CanonicalDeserialize::deserialize_compressed(&mut reader)
330 .map_err(|_| error("could not deserialize KZGRandomness"))
331 }
332}
333
334impl<E: PairingEngine> ToBytes for KZGRandomness<E> {
335 fn write_le<W: Write>(&self, mut writer: W) -> io::Result<()> {
336 CanonicalSerialize::serialize_compressed(self, &mut writer)
337 .map_err(|_| error("could not serialize KZGRandomness"))
338 }
339}
340
341impl<E: PairingEngine> KZGRandomness<E> {
342 #[inline]
346 pub fn is_hiding(&self) -> bool {
347 !self.blinding_polynomial.is_zero()
348 }
349
350 #[inline]
352 pub fn calculate_hiding_polynomial_degree(hiding_bound: usize) -> usize {
353 hiding_bound + 1
354 }
355}
356
357impl<E: PairingEngine> KZGRandomness<E> {
358 pub fn empty() -> Self {
359 Self { blinding_polynomial: DensePolynomial::zero() }
360 }
361
362 pub fn rand<R: RngCore>(hiding_bound: usize, _: bool, rng: &mut R) -> Self {
363 let mut randomness = KZGRandomness::empty();
364 let hiding_poly_degree = Self::calculate_hiding_polynomial_degree(hiding_bound);
365 randomness.blinding_polynomial = DensePolynomial::rand(hiding_poly_degree, rng);
366 randomness
367 }
368}
369
370impl<'a, E: PairingEngine> Add<&'a KZGRandomness<E>> for KZGRandomness<E> {
371 type Output = Self;
372
373 #[inline]
374 fn add(mut self, other: &'a Self) -> Self {
375 self.blinding_polynomial += &other.blinding_polynomial;
376 self
377 }
378}
379
380impl<'a, E: PairingEngine> Add<(E::Fr, &'a KZGRandomness<E>)> for KZGRandomness<E> {
381 type Output = Self;
382
383 #[inline]
384 fn add(mut self, other: (E::Fr, &'a KZGRandomness<E>)) -> Self {
385 self += other;
386 self
387 }
388}
389
390impl<'a, E: PairingEngine> AddAssign<&'a KZGRandomness<E>> for KZGRandomness<E> {
391 #[inline]
392 fn add_assign(&mut self, other: &'a Self) {
393 self.blinding_polynomial += &other.blinding_polynomial;
394 }
395}
396
397impl<'a, E: PairingEngine> AddAssign<(E::Fr, &'a KZGRandomness<E>)> for KZGRandomness<E> {
398 #[inline]
399 fn add_assign(&mut self, (f, other): (E::Fr, &'a KZGRandomness<E>)) {
400 self.blinding_polynomial += (f, &other.blinding_polynomial);
401 }
402}
403
404#[derive(Copy, Clone, Debug, Default, PartialEq, Eq, Hash, CanonicalSerialize, CanonicalDeserialize)]
406pub struct KZGProof<E: PairingEngine> {
407 pub w: E::G1Affine,
412 pub random_v: Option<E::Fr>,
415}
416
417impl<E: PairingEngine> KZGProof<E> {
418 pub fn absorb_into_sponge(&self, sponge: &mut impl AlgebraicSponge<E::Fq, 2>) {
419 sponge.absorb_native_field_elements(&self.w.to_field_elements().unwrap());
420 if let Some(random_v) = self.random_v {
421 sponge.absorb_nonnative_field_elements([random_v]);
422 }
423 }
424}
425
426impl<E: PairingEngine> FromBytes for KZGProof<E> {
427 fn read_le<R: Read>(mut reader: R) -> io::Result<Self> {
428 CanonicalDeserialize::deserialize_compressed(&mut reader)
429 .map_err(|err| into_io_error(anyhow::Error::from(err).context("could not deserialize KZG proof")))
430 }
431}
432
433impl<E: PairingEngine> ToBytes for KZGProof<E> {
434 fn write_le<W: Write>(&self, mut writer: W) -> io::Result<()> {
435 CanonicalSerialize::serialize_compressed(self, &mut writer)
436 .map_err(|err| into_io_error(anyhow::Error::from(err).context("could not serialize KZG proof")))
437 }
438}
439
440impl<E: PairingEngine> KZGProof<E> {
441 pub fn is_hiding(&self) -> bool {
442 self.random_v.is_some()
443 }
444}