use crate::fft::{EvaluationDomain, Evaluations, Polynomial};
use snarkvm_fields::{Field, PrimeField};
use snarkvm_utilities::serialize::*;
use std::{collections::BTreeMap, fmt};
#[derive(Clone, PartialEq, Eq, Hash, Default, CanonicalSerialize, CanonicalDeserialize)]
#[must_use]
pub struct SparsePolynomial<F: Field> {
coeffs: BTreeMap<usize, F>,
}
impl<F: Field> fmt::Debug for SparsePolynomial<F> {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
for (i, coeff) in self.coeffs.iter().filter(|(_, c)| !c.is_zero()) {
if *i == 0 {
write!(f, "\n{coeff:?}")?;
} else if *i == 1 {
write!(f, " + \n{coeff:?} * x")?;
} else {
write!(f, " + \n{coeff:?} * x^{i}")?;
}
}
Ok(())
}
}
impl<F: Field> SparsePolynomial<F> {
pub fn zero() -> Self {
Self { coeffs: BTreeMap::new() }
}
pub fn is_zero(&self) -> bool {
self.coeffs.is_empty() || self.coeffs.iter().all(|(_, c)| c.is_zero())
}
pub fn from_coefficients_slice(coeffs: &[(usize, F)]) -> Self {
Self::from_coefficients(coeffs.iter().copied())
}
pub fn from_coefficients(coeffs: impl IntoIterator<Item = (usize, F)>) -> Self {
let coeffs: BTreeMap<_, _> = coeffs.into_iter().filter(|(_, c)| !c.is_zero()).collect();
Self { coeffs }
}
pub fn coeffs(&self) -> impl Iterator<Item = (&usize, &F)> {
self.coeffs.iter()
}
pub fn degree(&self) -> usize {
if self.is_zero() {
0
} else {
let last = self.coeffs.iter().max();
assert!(last.map_or(false, |(_, c)| !c.is_zero()));
*last.unwrap().0
}
}
pub fn evaluate(&self, point: F) -> F {
if self.is_zero() {
return F::zero();
}
let mut total = F::zero();
for (i, c) in &self.coeffs {
total += *c * point.pow([*i as u64]);
}
total
}
pub fn mul(&self, other: &Self) -> Self {
if self.is_zero() || other.is_zero() {
SparsePolynomial::zero()
} else {
let mut result = std::collections::BTreeMap::new();
for (i, self_coeff) in self.coeffs.iter() {
for (j, other_coeff) in other.coeffs.iter() {
let cur_coeff = result.entry(i + j).or_insert_with(F::zero);
*cur_coeff += *self_coeff * other_coeff;
}
}
SparsePolynomial::from_coefficients(result)
}
}
}
impl<F: PrimeField> SparsePolynomial<F> {
pub fn evaluate_over_domain_by_ref(&self, domain: EvaluationDomain<F>) -> Evaluations<F> {
let poly: Polynomial<'_, F> = self.into();
Polynomial::<F>::evaluate_over_domain(poly, domain)
}
pub fn evaluate_over_domain(self, domain: EvaluationDomain<F>) -> Evaluations<F> {
let poly: Polynomial<'_, F> = self.into();
Polynomial::<F>::evaluate_over_domain(poly, domain)
}
}
impl<F: PrimeField> core::ops::MulAssign<F> for SparsePolynomial<F> {
fn mul_assign(&mut self, other: F) {
if other.is_zero() {
*self = Self::zero()
} else {
for coeff in self.coeffs.values_mut() {
*coeff *= other;
}
}
}
}
impl<'a, F: PrimeField> core::ops::Mul<F> for &'a SparsePolynomial<F> {
type Output = SparsePolynomial<F>;
fn mul(self, other: F) -> Self::Output {
let mut result = self.clone();
result *= other;
result
}
}
impl<'a, F: PrimeField> core::ops::AddAssign<&'a Self> for SparsePolynomial<F> {
fn add_assign(&mut self, other: &'a Self) {
let mut result = other.clone();
for (i, coeff) in self.coeffs.iter() {
let cur_coeff = result.coeffs.entry(*i).or_insert_with(F::zero);
*cur_coeff += coeff;
}
*self = Self::from_coefficients(result.coeffs.into_iter().filter(|(_, f)| !f.is_zero()));
}
}
impl<'a, F: PrimeField> core::ops::AddAssign<(F, &'a Self)> for SparsePolynomial<F> {
fn add_assign(&mut self, (f, other): (F, &'a Self)) {
let mut result = other.clone();
for (i, coeff) in self.coeffs.iter() {
let cur_coeff = result.coeffs.entry(*i).or_insert_with(F::zero);
*cur_coeff += f * coeff;
}
*self = Self::from_coefficients(result.coeffs.into_iter().filter(|(_, f)| !f.is_zero()))
}
}
#[cfg(test)]
mod tests {
use crate::fft::{DensePolynomial, EvaluationDomain, SparsePolynomial};
use snarkvm_curves::bls12_377::Fr;
use snarkvm_fields::One;
#[test]
fn evaluate_over_domain() {
for size in 2..10 {
let domain_size = 1 << size;
let domain = EvaluationDomain::new(domain_size).unwrap();
let two = Fr::one() + Fr::one();
let sparse_poly = SparsePolynomial::from_coefficients(vec![(0, two), (1, two)]);
let evals1 = sparse_poly.evaluate_over_domain_by_ref(domain);
let dense_poly: DensePolynomial<Fr> = sparse_poly.into();
let evals2 = dense_poly.clone().evaluate_over_domain(domain);
assert_eq!(evals1.clone().interpolate(), evals2.clone().interpolate());
assert_eq!(evals1.interpolate(), dense_poly);
assert_eq!(evals2.interpolate(), dense_poly);
}
}
}