openvm_stark_sdk/dummy_airs/interaction/
dummy_interaction_air.rs
use std::{iter, sync::Arc};
use itertools::izip;
use openvm_stark_backend::{
air_builders::PartitionedAirBuilder,
config::{StarkGenericConfig, Val},
interaction::{InteractionBuilder, InteractionType},
p3_air::{Air, BaseAir},
p3_field::{Field, FieldAlgebra},
p3_matrix::{dense::RowMajorMatrix, Matrix},
prover::types::{AirProofInput, AirProofRawInput, CommittedTraceData, TraceCommitter},
rap::{AnyRap, BaseAirWithPublicValues, PartitionedBaseAir},
Chip, ChipUsageGetter,
};
pub struct DummyInteractionCols;
impl DummyInteractionCols {
pub fn count_col() -> usize {
0
}
pub fn field_col(field_idx: usize) -> usize {
field_idx + 1
}
}
#[derive(Clone, Copy)]
pub struct DummyInteractionAir {
field_width: usize,
pub is_send: bool,
bus_index: usize,
pub partition: bool,
}
impl DummyInteractionAir {
pub fn new(field_width: usize, is_send: bool, bus_index: usize) -> Self {
Self {
field_width,
is_send,
bus_index,
partition: false,
}
}
pub fn partition(self) -> Self {
Self {
partition: true,
..self
}
}
pub fn field_width(&self) -> usize {
self.field_width
}
}
impl<F: Field> BaseAirWithPublicValues<F> for DummyInteractionAir {}
impl<F: Field> PartitionedBaseAir<F> for DummyInteractionAir {
fn cached_main_widths(&self) -> Vec<usize> {
if self.partition {
vec![self.field_width]
} else {
vec![]
}
}
fn common_main_width(&self) -> usize {
if self.partition {
1
} else {
1 + self.field_width
}
}
}
impl<F: Field> BaseAir<F> for DummyInteractionAir {
fn width(&self) -> usize {
1 + self.field_width
}
fn preprocessed_trace(&self) -> Option<RowMajorMatrix<F>> {
None
}
}
impl<AB: InteractionBuilder + PartitionedAirBuilder> Air<AB> for DummyInteractionAir {
fn eval(&self, builder: &mut AB) {
let (fields, count) = if self.partition {
let local_0 = builder.common_main().row_slice(0);
let local_1 = builder.cached_mains()[0].row_slice(0);
let count = local_0[0];
let fields = local_1.to_vec();
(fields, count)
} else {
let main = builder.main();
let local = main.row_slice(0);
let count = local[DummyInteractionCols::count_col()];
let fields: Vec<_> = (0..self.field_width)
.map(|i| local[DummyInteractionCols::field_col(i)])
.collect();
(fields, count)
};
let interaction_type = if self.is_send {
InteractionType::Send
} else {
InteractionType::Receive
};
builder.push_interaction(self.bus_index, fields, count, interaction_type)
}
}
pub struct DummyInteractionChip<'a, SC: StarkGenericConfig> {
trace_committer: Option<TraceCommitter<'a, SC>>,
data: Option<DummyInteractionData>,
pub air: DummyInteractionAir,
}
impl<SC: StarkGenericConfig> Clone for DummyInteractionChip<'_, SC> {
fn clone(&self) -> Self {
Self {
trace_committer: self.trace_committer.clone(),
data: self.data.clone(),
air: self.air,
}
}
}
#[derive(Debug, Clone)]
pub struct DummyInteractionData {
pub count: Vec<u32>,
pub fields: Vec<Vec<u32>>,
}
impl<'a, SC: StarkGenericConfig> DummyInteractionChip<'a, SC>
where
Val<SC>: FieldAlgebra,
{
pub fn new_without_partition(field_width: usize, is_send: bool, bus_index: usize) -> Self {
let air = DummyInteractionAir::new(field_width, is_send, bus_index);
Self {
trace_committer: None,
data: None,
air,
}
}
pub fn new_with_partition(
pcs: &'a SC::Pcs,
field_width: usize,
is_send: bool,
bus_index: usize,
) -> Self {
let air = DummyInteractionAir::new(field_width, is_send, bus_index).partition();
Self {
trace_committer: Some(TraceCommitter::new(pcs)),
data: None,
air,
}
}
pub fn load_data(&mut self, data: DummyInteractionData) {
let DummyInteractionData { count, fields } = &data;
let h = count.len();
assert_eq!(fields.len(), h);
let w = fields[0].len();
assert_eq!(self.air.field_width, w);
assert!(fields.iter().all(|r| r.len() == w));
self.data = Some(data);
}
fn generate_traces_with_partition(
&self,
data: DummyInteractionData,
) -> (RowMajorMatrix<Val<SC>>, CommittedTraceData<SC>) {
let DummyInteractionData {
mut count,
mut fields,
} = data;
let h = count.len();
assert_eq!(fields.len(), h);
let w = fields[0].len();
assert_eq!(self.air.field_width, w);
assert!(fields.iter().all(|r| r.len() == w));
let h = h.next_power_of_two();
count.resize(h, 0);
fields.resize(h, vec![0; w]);
let common_main_val: Vec<_> = count
.into_iter()
.map(Val::<SC>::from_canonical_u32)
.collect();
let cached_trace_val: Vec<_> = fields
.into_iter()
.flatten()
.map(Val::<SC>::from_canonical_u32)
.collect();
let cached_trace = RowMajorMatrix::new(cached_trace_val, w);
let prover_data = self
.trace_committer
.as_ref()
.unwrap()
.commit(vec![cached_trace.clone()]);
(
RowMajorMatrix::new(common_main_val, 1),
CommittedTraceData {
raw_data: Arc::new(cached_trace),
prover_data,
},
)
}
fn generate_traces_without_partition(
&self,
data: DummyInteractionData,
) -> RowMajorMatrix<Val<SC>> {
let DummyInteractionData { count, fields } = data;
let h = count.len();
assert_eq!(fields.len(), h);
let w = fields[0].len();
assert_eq!(self.air.field_width, w);
assert!(fields.iter().all(|r| r.len() == w));
let common_main_val: Vec<_> = izip!(count, fields)
.flat_map(|(count, fields)| iter::once(count).chain(fields))
.chain(iter::repeat(0))
.take((w + 1) * h.next_power_of_two())
.map(Val::<SC>::from_canonical_u32)
.collect();
RowMajorMatrix::new(common_main_val, w + 1)
}
}
impl<SC: StarkGenericConfig> Chip<SC> for DummyInteractionChip<'_, SC> {
fn air(&self) -> Arc<dyn AnyRap<SC>> {
Arc::new(self.air)
}
fn generate_air_proof_input(self) -> AirProofInput<SC> {
assert!(self.data.is_some());
let data = self.data.clone().unwrap();
if self.trace_committer.is_some() {
let (common_main, cached_main) = self.generate_traces_with_partition(data);
AirProofInput {
air: self.air(),
cached_mains_pdata: vec![cached_main.prover_data],
raw: AirProofRawInput {
cached_mains: vec![cached_main.raw_data],
common_main: Some(common_main),
public_values: vec![],
},
}
} else {
let common_main = self.generate_traces_without_partition(data);
AirProofInput {
air: self.air(),
cached_mains_pdata: vec![],
raw: AirProofRawInput {
cached_mains: vec![],
common_main: Some(common_main),
public_values: vec![],
},
}
}
}
}
impl<SC: StarkGenericConfig> ChipUsageGetter for DummyInteractionChip<'_, SC> {
fn air_name(&self) -> String {
"DummyInteractionAir".to_string()
}
fn current_trace_height(&self) -> usize {
if let Some(data) = &self.data {
data.count.len()
} else {
0
}
}
fn trace_width(&self) -> usize {
self.air.field_width + 1
}
}