openvm_circuit_primitives/sub_air.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
use openvm_stark_backend::p3_air::AirBuilder;
/// Trait with associated types intended to allow re-use of constraint logic
/// inside other AIRs.
///
/// A `SubAir` is **not** an [`Air`](p3_air::Air) itself.
/// A `SubAir` is a struct that holds the means to generate a particular set of constraints,
/// meant to be re-usable within other AIRs.
///
/// The trait is designed to be maximally flexible, but typical implementations will separate
/// the `AirContext` into two parts: `Io` and `AuxCols`. The `Io` part will consist of
/// expressions (built using `AB::Expr`) that the `SubAir` does not own, while the `AuxCols`
/// are any internal columns that the `SubAir` requires to generate its constraints. The
/// `AuxCols` are columns that the `SubAir` fully owns and should be internally determined by
/// the `SubAir` from the `Io` part. These `AuxCols` are typically just slices of `AB::Var`.
///
/// This trait only owns the constraints, but it is expected that the [TraceSubRowGenerator] trait
/// or some analogous functionality is also implemented so that the trace generation of the `AuxCols`
/// of each row can be done purely in terms of the `Io` part.
pub trait SubAir<AB: AirBuilder> {
/// Type to define the context, typically in terms of `AB::Expr` that are needed
/// to define the SubAir's constraints.
type AirContext<'a>
where
Self: 'a,
AB: 'a,
AB::Var: 'a,
AB::Expr: 'a;
fn eval<'a>(&'a self, builder: &'a mut AB, ctx: Self::AirContext<'a>)
where
AB::Var: 'a,
AB::Expr: 'a;
}
/// This is a helper for generation of the trace on a subset of the columns in a single row
/// of the trace matrix.
// [jpw] This could be part of SubAir, but I want to keep SubAir to be constraints only
pub trait TraceSubRowGenerator<F> {
/// The minimal amount of information needed to generate the sub-row of the trace matrix.
/// This type has a lifetime so other context, such as references to other chips, can be provided.
type TraceContext<'a>
where
Self: 'a;
/// The type for the columns to mutate. Often this can be `&'a mut Cols<F>` if `Cols` is on the stack.
/// For structs that use the heap, this should be a struct that contains mutable slices.
type ColsMut<'a>
where
Self: 'a;
fn generate_subrow<'a>(&'a self, ctx: Self::TraceContext<'a>, sub_row: Self::ColsMut<'a>);
}