Skip to content

Commit

Permalink
LinePoly backend
Browse files Browse the repository at this point in the history
  • Loading branch information
spapinistarkware committed Feb 15, 2024
1 parent b875fc8 commit a6e6f51
Show file tree
Hide file tree
Showing 8 changed files with 243 additions and 243 deletions.
2 changes: 1 addition & 1 deletion src/core/air/evaluation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
//! Given a random alpha, the combined polynomial is defined as
//! f(p) = sum_i alpha^{N-1-i} u_i (P).
use crate::core::backend::{Backend, CPUBackend, Column, VecLike};
use crate::core::backend::{Backend, CPUBackend, Column, ColumnTrait};
use crate::core::fields::m31::BaseField;
use crate::core::fields::qm31::SecureField;
use crate::core::fields::Field;
Expand Down
12 changes: 0 additions & 12 deletions src/core/backend/cpu/poly.rs → src/core/backend/cpu/circle.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,6 @@ use crate::core::poly::circle::{
CanonicCoset, CircleDomain, CircleEvaluation, CirclePoly, PolyOps,
};
use crate::core::poly::utils::fold;
use crate::core::poly::{BitReversedOrder, NaturalOrder};
use crate::core::utils::bit_reverse;

type B = CPUBackend;
impl<F: ExtensionOf<BaseField>> PolyOps<F> for CPUBackend {
Expand Down Expand Up @@ -53,16 +51,6 @@ impl<F: ExtensionOf<BaseField>> PolyOps<F> for CPUBackend {

CirclePoly::new(values)
}
fn bit_reverse_natural(
eval: CircleEvaluation<B, F, NaturalOrder>,
) -> CircleEvaluation<B, F, BitReversedOrder> {
CircleEvaluation::new(eval.domain, bit_reverse(eval.values))
}
fn bit_reverse_reversed(
eval: CircleEvaluation<B, F, BitReversedOrder>,
) -> CircleEvaluation<B, F, NaturalOrder> {
CircleEvaluation::new(eval.domain, bit_reverse(eval.values))
}

fn eval_at_point<E: ExtensionOf<F>>(poly: &CirclePoly<B, F>, point: CirclePoint<E>) -> E {
// TODO(Andrew): Allocation here expensive for small polynomials.
Expand Down
119 changes: 119 additions & 0 deletions src/core/backend/cpu/line.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
use super::CPUBackend;
use crate::core::circle::CirclePoint;
use crate::core::fft::{butterfly, ibutterfly};
use crate::core::fields::m31::BaseField;
use crate::core::fields::{ExtensionOf, Field};
use crate::core::poly::line::{LineDomain, LineEvaluation, LinePoly, LinePolyOps};
use crate::core::poly::utils::{fold, repeat_value};

impl<F: ExtensionOf<BaseField>> LinePolyOps<F> for CPUBackend {
fn eval_at_point<E: ExtensionOf<F>>(poly: &LinePoly<Self, F>, mut x: E) -> E {
// TODO(Andrew): Allocation here expensive for small polynomials.
let mut doublings = vec![x];
for _ in 1..poly.log_size {
x = CirclePoint::double_x(x);
doublings.push(x);
}
fold(&poly.coeffs, &doublings)
}

fn evaluate(poly: LinePoly<Self, F>, domain: LineDomain) -> LineEvaluation<Self, F> {
assert!(domain.size() >= poly.coeffs.len());

// The first few FFT layers may just copy coefficients so we do it directly.
// See the docs for `n_skipped_layers` in [line_fft].
let log_degree_bound = poly.log_size;
let n_skipped_layers = (domain.log_size() - log_degree_bound) as usize;
let duplicity = 1 << n_skipped_layers;
let mut coeffs = repeat_value(&poly.coeffs, duplicity);

line_fft(&mut coeffs, domain, n_skipped_layers);
LineEvaluation::new(domain, coeffs)
}

fn interpolate(eval: LineEvaluation<Self, F>) -> LinePoly<Self, F> {
let domain = eval.domain();
let mut values = eval.values;
line_ifft(&mut values, domain);
// Normalize the coefficients.
let len_inv = BaseField::from(values.len()).inverse();
values.iter_mut().for_each(|v| *v *= len_inv);
LinePoly::new(values)
}
}

/// Performs a univariate FFT of a polynomial over a [LineDomain].
///
/// The transform happens in-place. `values` consist of coefficients in [line_ifft] algorithm's
/// basis need to be stored in bit-reversed order. After the transformation `values` becomes
/// evaluations of the polynomial over `domain` stored in natural order.
///
/// The `n_skipped_layers` argument allows specifying how many of the initial butterfly layers of
/// the FFT to skip. This is useful when doing more efficient degree aware FFTs as the butterflies
/// in the first layers of the FFT only involve copying coefficients to different locations (because
/// one or more of the coefficients is zero). This new algorithm is `O(n log d)` vs `O(n log n)`
/// where `n` is the domain size and `d` is the number of coefficients.
///
/// # Panics
///
/// Panics if the number of values doesn't match the size of the domain.
fn line_fft<F: ExtensionOf<BaseField>>(
values: &mut [F],
mut domain: LineDomain,
n_skipped_layers: usize,
) {
assert_eq!(values.len(), domain.size());

// Construct the domains we need.
let mut domains = vec![];
while domain.size() > 1 << n_skipped_layers {
domains.push(domain);
domain = domain.double();
}

// Execute the butterfly layers.
for domain in domains.iter().rev() {
for chunk in values.chunks_exact_mut(domain.size()) {
let (l, r) = chunk.split_at_mut(domain.size() / 2);
for (i, x) in domain.iter().take(domain.size() / 2).enumerate() {
butterfly(&mut l[i], &mut r[i], x);
}
}
}
}

/// Performs a univariate IFFT on a polynomial's evaluation over a [LineDomain].
///
/// This is not the standard univariate IFFT, because [LineDomain] is not a cyclic group.
///
/// The transform happens in-place. `values` should be the evaluations of a polynomial over `domain`
/// in their natural order. After the transformation `values` becomes the coefficients of the
/// polynomial stored in bit-reversed order.
///
/// For performance reasons and flexibility the normalization of the coefficients is omitted. The
/// normalized coefficients can be obtained by scaling all coefficients by `1 / len(values)`.
///
/// This algorithm does not return coefficients in the standard monomial basis but rather returns
/// coefficients in a basis relating to the circle's x-coordinate doubling map `pi(x) = 2x^2 - 1`
/// i.e.
///
/// ```text
/// B = { 1 } * { x } * { pi(x) } * { pi(pi(x)) } * ...
/// = { 1, x, pi(x), pi(x) * x, pi(pi(x)), pi(pi(x)) * x, pi(pi(x)) * pi(x), ... }
/// ```
///
/// # Panics
///
/// Panics if the number of values doesn't match the size of the domain.
fn line_ifft<F: ExtensionOf<BaseField>>(values: &mut [F], mut domain: LineDomain) {
assert_eq!(values.len(), domain.size());
while domain.size() > 1 {
for chunk in values.chunks_exact_mut(domain.size()) {
let (l, r) = chunk.split_at_mut(domain.size() / 2);
for (i, x) in domain.iter().take(domain.size() / 2).enumerate() {
ibutterfly(&mut l[i], &mut r[i], x.inverse());
}
}
domain = domain.double();
}
}
18 changes: 15 additions & 3 deletions src/core/backend/cpu/mod.rs
Original file line number Diff line number Diff line change
@@ -1,20 +1,32 @@
mod poly;
mod circle;
mod line;

use super::{Backend, FieldOps, VecLike};
use super::{Backend, ColumnTrait, FieldOps};
use crate::core::fields::Field;
use crate::core::utils::bit_reverse;

#[derive(Copy, Clone, Debug)]
pub struct CPUBackend;
impl Backend for CPUBackend {}

impl<F: Field> FieldOps<F> for CPUBackend {
type Column = Vec<F>;

fn bit_reverse_column(column: Self::Column) -> Self::Column {
bit_reverse(column)
}
}

impl<F> VecLike<F> for Vec<F> {
impl<F: Field> ColumnTrait<F> for Vec<F> {
fn zeros(len: usize) -> Self {
vec![F::zero(); len]
}
fn from_vec(vec: Vec<F>) -> Self {
vec
}
fn to_vec(&self) -> Vec<F> {
self.clone()
}
fn len(&self) -> usize {
self.len()
}
Expand Down
7 changes: 5 additions & 2 deletions src/core/backend/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,16 @@ pub trait Backend:
}

pub trait FieldOps<F: Field> {
type Column: Clone + std::fmt::Debug + VecLike<F> + Index<usize, Output = F>;
type Column: Clone + std::fmt::Debug + ColumnTrait<F> + Index<usize, Output = F>;
fn bit_reverse_column(column: Self::Column) -> Self::Column;
}

pub type Column<B, F> = <B as FieldOps<F>>::Column;

pub trait VecLike<F> {
pub trait ColumnTrait<F> {
fn zeros(len: usize) -> Self;
fn from_vec(vec: Vec<F>) -> Self;
fn to_vec(&self) -> Vec<F>;
fn len(&self) -> usize;
fn is_empty(&self) -> bool {
self.len() == 0
Expand Down
Loading

0 comments on commit a6e6f51

Please sign in to comment.