use crate::{septic_curve::SepticCurve, septic_extension::SepticExtension};
use deepsize2::DeepSizeOf;
use serde::{Deserialize, Serialize};
use slop_algebra::{AbstractExtensionField, AbstractField, Field};
use std::{iter::Sum, ops::Add};
pub const CURVE_CUMULATIVE_SUM_START_X: [u32; 7] =
[0x1414213, 0x5623730, 0x9504880, 0x1688724, 0x2096980, 0x7856967, 0x1875376];
pub const CURVE_CUMULATIVE_SUM_START_Y: [u32; 7] =
[2020310104, 1513506566, 1843922297, 2003644209, 805967281, 1882435203, 1623804682];
pub const DIGEST_SUM_START_X: [u32; 7] =
[0x1732050, 0x8075688, 0x7729352, 0x7446341, 0x5058723, 0x6694280, 0x5253810];
pub const DIGEST_SUM_START_Y: [u32; 7] =
[1095433104, 7540207, 1124564165, 2035506693, 11121645, 102781365, 398772161];
#[derive(Debug, Clone, Copy, Default, Serialize, Deserialize, PartialEq, Eq, Hash, DeepSizeOf)]
#[repr(C)]
pub struct SepticDigest<F>(pub SepticCurve<F>);
impl<F: AbstractField> SepticDigest<F> {
#[must_use]
pub fn zero() -> Self {
SepticDigest(SepticCurve {
x: SepticExtension::<F>::from_base_fn(|i| {
F::from_canonical_u32(CURVE_CUMULATIVE_SUM_START_X[i])
}),
y: SepticExtension::<F>::from_base_fn(|i| {
F::from_canonical_u32(CURVE_CUMULATIVE_SUM_START_Y[i])
}),
})
}
#[must_use]
pub fn starting_digest() -> Self {
SepticDigest(SepticCurve {
x: SepticExtension::<F>::from_base_fn(|i| F::from_canonical_u32(DIGEST_SUM_START_X[i])),
y: SepticExtension::<F>::from_base_fn(|i| F::from_canonical_u32(DIGEST_SUM_START_Y[i])),
})
}
}
impl<F: Field> SepticDigest<F> {
pub fn is_zero(&self) -> bool {
*self == SepticDigest::<F>::zero()
}
}
impl<F: Field> Add for SepticDigest<F> {
type Output = Self;
fn add(self, rhs: Self) -> Self {
let start = Self::starting_digest().0;
let sum_a = start.add_incomplete(self.0).sub_incomplete(Self::zero().0);
let sum_b = sum_a.add_incomplete(rhs.0).sub_incomplete(Self::zero().0);
let mut result = sum_b;
result.add_assign(SepticDigest::<F>::zero().0);
result.sub_assign(start);
SepticDigest(result)
}
}
impl<F: Field> Sum for SepticDigest<F> {
fn sum<I: Iterator<Item = Self>>(iter: I) -> Self {
let start = SepticDigest::<F>::starting_digest().0;
let mut ret = iter.fold(start, |acc, x| {
let sum_offset = acc.add_incomplete(x.0);
sum_offset.sub_incomplete(SepticDigest::<F>::zero().0)
});
ret.add_assign(SepticDigest::<F>::zero().0);
ret.sub_assign(start);
SepticDigest(ret)
}
}
#[cfg(test)]
mod test {
use crate::septic_curve::{CURVE_WITNESS_DUMMY_POINT_X, CURVE_WITNESS_DUMMY_POINT_Y};
use super::*;
use sp1_primitives::SP1Field;
#[test]
fn test_const_points() {
let x: SepticExtension<SP1Field> = SepticExtension::from_base_fn(|i| {
SP1Field::from_canonical_u32(CURVE_CUMULATIVE_SUM_START_X[i])
});
let y: SepticExtension<SP1Field> = SepticExtension::from_base_fn(|i| {
SP1Field::from_canonical_u32(CURVE_CUMULATIVE_SUM_START_Y[i])
});
let point = SepticCurve { x, y };
assert!(point.check_on_point());
let x: SepticExtension<SP1Field> =
SepticExtension::from_base_fn(|i| SP1Field::from_canonical_u32(DIGEST_SUM_START_X[i]));
let y: SepticExtension<SP1Field> =
SepticExtension::from_base_fn(|i| SP1Field::from_canonical_u32(DIGEST_SUM_START_Y[i]));
let point = SepticCurve { x, y };
assert!(point.check_on_point());
let x: SepticExtension<SP1Field> = SepticExtension::from_base_fn(|i| {
SP1Field::from_canonical_u32(CURVE_WITNESS_DUMMY_POINT_X[i])
});
let y: SepticExtension<SP1Field> = SepticExtension::from_base_fn(|i| {
SP1Field::from_canonical_u32(CURVE_WITNESS_DUMMY_POINT_Y[i])
});
let point = SepticCurve { x, y };
assert!(point.check_on_point());
}
}