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.

175 lines
5.3 KiB

use crate::poseidon_transcript::{PoseidonTranscript, TranscriptWriter};
use ark_crypto_primitives::sponge::Absorb;
use ark_ff::{Field, PrimeField};
use ark_serialize::*;
// ax^2 + bx + c stored as vec![c,b,a]
// ax^3 + bx^2 + cx + d stored as vec![d,c,b,a]
#[derive(Debug, CanonicalDeserialize, CanonicalSerialize, Clone)]
pub struct UniPoly<F: Field> {
pub coeffs: Vec<F>,
// pub coeffs_fq: Vec<Fq>,
}
// ax^2 + bx + c stored as vec![c,a]
// ax^3 + bx^2 + cx + d stored as vec![d,b,a]
#[derive(CanonicalSerialize, CanonicalDeserialize, Debug)]
pub struct CompressedUniPoly<F: Field> {
pub coeffs_except_linear_term: Vec<F>,
}
impl<F: Field> UniPoly<F> {
pub fn from_evals(evals: &[F]) -> Self {
// we only support degree-2 or degree-3 univariate polynomials
assert!(evals.len() == 3 || evals.len() == 4);
let coeffs = if evals.len() == 3 {
// ax^2 + bx + c
let two_inv = F::from(2 as u8).inverse().unwrap();
let c = evals[0];
let a = two_inv * (evals[2] - evals[1] - evals[1] + c);
let b = evals[1] - c - a;
vec![c, b, a]
} else {
// ax^3 + bx^2 + cx + d
let two_inv = F::from(2 as u8).inverse().unwrap();
let six_inv = F::from(6 as u8).inverse().unwrap();
let d = evals[0];
let a = six_inv
* (evals[3] - evals[2] - evals[2] - evals[2] + evals[1] + evals[1] + evals[1] - evals[0]);
let b = two_inv
* (evals[0] + evals[0] - evals[1] - evals[1] - evals[1] - evals[1] - evals[1]
+ evals[2]
+ evals[2]
+ evals[2]
+ evals[2]
- evals[3]);
let c = evals[1] - d - a - b;
vec![d, c, b, a]
};
UniPoly { coeffs }
}
pub fn degree(&self) -> usize {
self.coeffs.len() - 1
}
pub fn eval_at_zero(&self) -> F {
self.coeffs[0]
}
pub fn eval_at_one(&self) -> F {
(0..self.coeffs.len()).map(|i| self.coeffs[i]).sum()
}
pub fn evaluate(&self, r: &F) -> F {
let mut eval = self.coeffs[0];
let mut power = *r;
for i in 1..self.coeffs.len() {
eval += power * self.coeffs[i];
power *= r;
}
eval
}
// pub fn compress(&self) -> CompressedUniPoly<F> {
// let coeffs_except_linear_term = [&self.coeffs[..1], &self.coeffs[2..]].concat();
// assert_eq!(coeffs_except_linear_term.len() + 1, self.coeffs.len());
// CompressedUniPoly {
// coeffs_except_linear_term,
// }
// }
}
// impl<F: PrimeField> CompressedUniPoly<F> {
// // we require eval(0) + eval(1) = hint, so we can solve for the linear term as:
// // linear_term = hint - 2 * constant_term - deg2 term - deg3 term
// pub fn decompress(&self, hint: &F) -> UniPoly<F> {
// let mut linear_term =
// (*hint) - self.coeffs_except_linear_term[0] - self.coeffs_except_linear_term[0];
// for i in 1..self.coeffs_except_linear_term.len() {
// linear_term -= self.coeffs_except_linear_term[i];
// }
// let mut coeffs = vec![self.coeffs_except_linear_term[0], linear_term];
// coeffs.extend(&self.coeffs_except_linear_term[1..]);
// assert_eq!(self.coeffs_except_linear_term.len() + 1, coeffs.len());
// UniPoly { coeffs }
// }
// }
impl<F: PrimeField + Absorb> TranscriptWriter<F> for UniPoly<F> {
fn write_to_transcript(&self, transcript: &mut PoseidonTranscript<F>) {
// transcript.append_message(label, b"UniPoly_begin");
for i in 0..self.coeffs.len() {
transcript.append_scalar(b"", &self.coeffs[i]);
}
// transcript.append_message(label, b"UniPoly_end");
}
}
#[cfg(test)]
mod tests {
use ark_ff::One;
use super::*;
type F = ark_bls12_377::Fr;
#[test]
fn test_from_evals_quad() {
// polynomial is 2x^2 + 3x + 1
let e0 = F::one();
let e1 = F::from(6 as u8);
let e2 = F::from(15 as u8);
let evals = vec![e0, e1, e2];
let poly = UniPoly::from_evals(&evals);
assert_eq!(poly.eval_at_zero(), e0);
assert_eq!(poly.eval_at_one(), e1);
assert_eq!(poly.coeffs.len(), 3);
assert_eq!(poly.coeffs[0], F::one());
assert_eq!(poly.coeffs[1], F::from(3 as u8));
assert_eq!(poly.coeffs[2], F::from(2 as u8));
// let hint = e0 + e1;
// // let compressed_poly = poly.compress();
// // let decompressed_poly = compressed_poly.decompress(&hint);
// for i in 0..poly.coeffs.len() {
// assert_eq!(poly.coeffs[i], poly.coeffs[i]);
// }
let e3 = F::from(28 as u8);
assert_eq!(poly.evaluate(&F::from(3 as u8)), e3);
}
#[test]
fn test_from_evals_cubic() {
// polynomial is x^3 + 2x^2 + 3x + 1
let e0 = F::one();
let e1 = F::from(7);
let e2 = F::from(23);
let e3 = F::from(55);
let evals = vec![e0, e1, e2, e3];
let poly = UniPoly::from_evals(&evals);
assert_eq!(poly.eval_at_zero(), e0);
assert_eq!(poly.eval_at_one(), e1);
assert_eq!(poly.coeffs.len(), 4);
assert_eq!(poly.coeffs[0], F::one());
assert_eq!(poly.coeffs[1], F::from(3));
assert_eq!(poly.coeffs[2], F::from(2));
assert_eq!(poly.coeffs[3], F::from(1));
// let hint = e0 + e1;
// let compressed_poly = poly.compress();
// let decompressed_poly = compressed_poly.decompress(&hint);
// for i in 0..decompressed_poly.coeffs.len() {
// assert_eq!(decompressed_poly.coeffs[i], poly.coeffs[i]);
// }
let e4 = F::from(109);
assert_eq!(poly.evaluate(&F::from(4)), e4);
}
}