binius_math/multilinear.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 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
// Copyright 2023-2025 Irreducible Inc.
use std::fmt::Debug;
use binius_field::PackedField;
use either::Either;
use crate::{Error, MultilinearExtension, MultilinearQueryRef};
/// Represents a multilinear polynomial.
///
/// This interface includes no generic methods, in order to support the creation of trait objects.
#[auto_impl::auto_impl(&, &mut, Box, Rc, Arc)]
pub trait MultilinearPoly<P: PackedField>: Debug {
/// Number of variables.
fn n_vars(&self) -> usize;
/// The number of coefficients required to specify the polynomial.
fn size(&self) -> usize {
1 << self.n_vars()
}
/// Binary logarithm of the extension degree (always exists because we only support power-of-two extension degrees)
fn log_extension_degree(&self) -> usize;
/// Get the evaluations of the polynomial at a vertex of the hypercube.
///
/// # Arguments
///
/// * `index` - The index of the point, in lexicographic order
fn evaluate_on_hypercube(&self, index: usize) -> Result<P::Scalar, Error>;
/// Get the evaluations of the polynomial at a vertex of the hypercube and scale the value.
///
/// This can be more efficient than calling `evaluate_on_hypercube` followed by a
/// multiplication when the trait implementation can use a subfield multiplication.
///
/// # Arguments
///
/// * `index` - The index of the point, in lexicographic order
/// * `scalar` - The scaling coefficient
fn evaluate_on_hypercube_and_scale(
&self,
index: usize,
scalar: P::Scalar,
) -> Result<P::Scalar, Error>;
fn evaluate(&self, query: MultilinearQueryRef<P>) -> Result<P::Scalar, Error>;
fn evaluate_partial_low(
&self,
query: MultilinearQueryRef<P>,
) -> Result<MultilinearExtension<P>, Error>;
fn evaluate_partial_high(
&self,
query: MultilinearQueryRef<P>,
) -> Result<MultilinearExtension<P>, Error>;
/// Compute inner products of a multilinear query inside a subcube.
///
/// Equivalent computation is `evaluate_partial_low(query)` followed by a `subcube_evals`
/// on a result. This method is more efficient due to handling it as a special case.
fn subcube_inner_products(
&self,
query: MultilinearQueryRef<P>,
subcube_vars: usize,
subcube_index: usize,
inner_products: &mut [P],
) -> Result<(), Error>;
/// Get a subcube of the boolean hypercube of a given size.
///
/// Subcube of a multilinear is a set of evaluations $M(\beta_i\Vert x_j)$ , where
/// $\beta_i \in \mathcal{B}_k$ iterates over `subcube_vars`-sized hypercube and $x_j$ is a binary
/// representation of the `subcube_index`.
///
/// The result slice `evals` holds subcube evaluations in lexicographic order of $\beta_i$, with the
/// fastest stride corresponding to the first variable. Each scalar of the packed field `P` is assumed
/// to be a `2^log_embedding_degree` extension field, where subcube evaluations are assigned to bases
/// in lexicographic order of the lowest `log_embedding_degree` variables.
///
/// Note that too large `log_embedding_degree` values may cause this method to fail.
fn subcube_evals(
&self,
subcube_vars: usize,
subcube_index: usize,
log_embedding_degree: usize,
evals: &mut [P],
) -> Result<(), Error>;
/// Returns the hypercube evaluations, embedded into packed extension field elements, if the
/// data is already available.
///
/// This method is primarily used to access the raw evaluation data underlying a
/// [`MultilinearExtension`] that is type-erased as a [`MultilinearPoly`] trait object. The
/// evaluation data is useful for cases where the caller needs to dynamically re-interpret it
/// as subfield coefficients while avoiding copying, like for the small-field polynomial
/// commitment scheme or to provide directly to a hardware accelerator.
///
/// If the data is not available, this method returns `None`. If the data is available, it
/// should be interpreted not actually as a list of evaluations points given by iterating the
/// packed slice, but rather by iterating coefficients from a subfield with an embedding degree
/// given by [`Self::log_extension_degree`].
///
/// The data returned, if `Some`, should be the same as the data that is written by
/// [`Self::subcube_evals`].
fn packed_evals(&self) -> Option<&[P]>;
}
impl<P, L, R> MultilinearPoly<P> for Either<L, R>
where
P: PackedField,
L: MultilinearPoly<P>,
R: MultilinearPoly<P>,
{
fn n_vars(&self) -> usize {
either::for_both!(self, inner => inner.n_vars())
}
fn size(&self) -> usize {
either::for_both!(self, inner => inner.size())
}
fn log_extension_degree(&self) -> usize {
either::for_both!(self, inner => inner.log_extension_degree())
}
fn evaluate_on_hypercube(&self, index: usize) -> Result<P::Scalar, Error> {
either::for_both!(self, inner => inner.evaluate_on_hypercube(index))
}
fn evaluate_on_hypercube_and_scale(
&self,
index: usize,
scalar: P::Scalar,
) -> Result<P::Scalar, Error> {
either::for_both!(self, inner => inner.evaluate_on_hypercube_and_scale(index, scalar))
}
fn evaluate(&self, query: MultilinearQueryRef<P>) -> Result<P::Scalar, Error> {
either::for_both!(self, inner => inner.evaluate(query))
}
fn evaluate_partial_low(
&self,
query: MultilinearQueryRef<P>,
) -> Result<MultilinearExtension<P>, Error> {
either::for_both!(self, inner => inner.evaluate_partial_low(query))
}
fn evaluate_partial_high(
&self,
query: MultilinearQueryRef<P>,
) -> Result<MultilinearExtension<P>, Error> {
either::for_both!(self, inner => inner.evaluate_partial_high(query))
}
fn subcube_inner_products(
&self,
query: MultilinearQueryRef<P>,
subcube_vars: usize,
subcube_index: usize,
inner_products: &mut [P],
) -> Result<(), Error> {
either::for_both!(
self,
inner => {
inner.subcube_inner_products(query, subcube_vars, subcube_index, inner_products)
}
)
}
fn subcube_evals(
&self,
subcube_vars: usize,
subcube_index: usize,
log_embedding_degree: usize,
evals: &mut [P],
) -> Result<(), Error> {
either::for_both!(
self,
inner => inner.subcube_evals(subcube_vars, subcube_index, log_embedding_degree, evals)
)
}
fn packed_evals(&self) -> Option<&[P]> {
either::for_both!(self, inner => inner.packed_evals())
}
}