binius_ntt/additive_ntt.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
// Copyright 2024-2025 Irreducible Inc.
use binius_field::{ExtensionField, PackedField, RepackedExtension};
use binius_utils::checked_arithmetics::log2_strict_usize;
use super::error::Error;
/// The additive NTT defined in [LCH14].
///
/// [LCH14]: <https://arxiv.org/abs/1404.3458>
pub trait AdditiveNTT<P: PackedField> {
/// Base-2 logarithm of the size of the NTT domain.
fn log_domain_size(&self) -> usize;
/// Get the normalized subspace polynomial evaluation $\hat{W}_i(\beta_j)$.
///
/// ## Preconditions
///
/// * `i` must be less than `self.log_domain_size()`
/// * `j` must be less than `self.log_domain_size() - i`
fn get_subspace_eval(&self, i: usize, j: usize) -> P::Scalar;
/// Forward transformation defined in [LCH14] on a batch of inputs.
///
/// Input is the vector of polynomial coefficients in novel basis, output is in Lagrange basis.
/// The batched inputs are interleaved, which improves the cache-efficiency of the computation.
///
/// [LCH14]: <https://arxiv.org/abs/1404.3458>
fn forward_transform(
&self,
data: &mut [P],
coset: u32,
log_batch_size: usize,
) -> Result<(), Error>;
/// Inverse transformation defined in [LCH14] on a batch of inputs.
///
/// Input is the vector of polynomial coefficients in Lagrange basis, output is in novel basis.
/// The batched inputs are interleaved, which improves the cache-efficiency of the computation.
///
/// [LCH14]: https://arxiv.org/abs/1404.3458
fn inverse_transform(
&self,
data: &mut [P],
coset: u32,
log_batch_size: usize,
) -> Result<(), Error>;
fn forward_transform_ext<PE>(&self, data: &mut [PE], coset: u32) -> Result<(), Error>
where
PE: RepackedExtension<P>,
PE::Scalar: ExtensionField<P::Scalar>,
{
if !PE::Scalar::DEGREE.is_power_of_two() {
return Err(Error::PowerOfTwoExtensionDegreeRequired);
}
let log_batch_size = log2_strict_usize(PE::Scalar::DEGREE);
self.forward_transform(PE::cast_bases_mut(data), coset, log_batch_size)
}
fn inverse_transform_ext<PE>(&self, data: &mut [PE], coset: u32) -> Result<(), Error>
where
PE: RepackedExtension<P>,
PE::Scalar: ExtensionField<P::Scalar>,
{
if !PE::Scalar::DEGREE.is_power_of_two() {
return Err(Error::PowerOfTwoExtensionDegreeRequired);
}
let log_batch_size = log2_strict_usize(PE::Scalar::DEGREE);
self.inverse_transform(PE::cast_bases_mut(data), coset, log_batch_size)
}
}