Skip to content

Commit

Permalink
feat: add shift right instruction component
Browse files Browse the repository at this point in the history
  • Loading branch information
zmalatrax committed Jan 9, 2025
1 parent 01bb7ef commit 2499c33
Show file tree
Hide file tree
Showing 2 changed files with 193 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
pub mod input_component;
pub mod output_component;
pub mod plus_component;
pub mod right_component;
pub mod table;
Original file line number Diff line number Diff line change
@@ -0,0 +1,192 @@
use crate::components::{processor::table::ProcessorElements, ProcessorInstructionClaim};
use brainfuck_vm::instruction::InstructionType;
use num_traits::One;
use stwo_prover::constraint_framework::{
EvalAtRow, FrameworkComponent, FrameworkEval, RelationEntry,
};

/// Implementation of `Component` and `ComponentProver` for the Right `>` instruction component.
/// It targets the `SimdBackend` from the Stwo constraint framework, with a fallback
/// on `CpuBakend` for small traces.
pub type RightInstructionComponent = FrameworkComponent<RightInstructionEval>;

/// The AIR for the [`RightInstructionComponent`].
///
/// Constraints are defined through the [`FrameworkEval`]
/// provided by the constraint framework of Stwo.
pub struct RightInstructionEval {
/// The log size of the component's main trace height.
log_size: u32,
/// The random elements used for the lookup protocol linking the instruction logic components
/// to the processor component.
processor_lookup_elements: ProcessorElements,
}

impl RightInstructionEval {
pub const fn new(
claim: &ProcessorInstructionClaim,
processor_lookup_elements: ProcessorElements,
) -> Self {
Self { log_size: claim.log_size, processor_lookup_elements }
}
}

impl FrameworkEval for RightInstructionEval {
/// Returns the log size from the main claim.
fn log_size(&self) -> u32 {
self.log_size
}

/// The degree of the constraints is bounded by the size of the trace.
///
/// Returns the ilog2 (upper) bound of the constraint degree for the component.
fn max_constraint_log_degree_bound(&self) -> u32 {
self.log_size + 1
}

/// Defines the AIR for the [`RightInstructionComponent`].
///
/// Registers' values from the current row are obtained through masks.
/// When you apply a mask, you target the current column and then pass to the next
/// one: the register order matters to correctly fetch them.
///
/// - Use `eval.next_trace_mask()` to get the current register from the main trace
/// (`ORIGINAL_TRACE_IDX`)
///
/// Use `eval.add_constraint` to define a local constraint (boundary, transition).
/// Use `eval.add_to_relation` to define a global constraint for the logUp protocol.
///
/// The logUp must be finalized with `eval.finalize_logup()`.
#[allow(clippy::similar_names)]
fn evaluate<E: EvalAtRow>(&self, mut eval: E) -> E {
let clk = eval.next_trace_mask();
let ip = eval.next_trace_mask();
let ci = eval.next_trace_mask();
let ni = eval.next_trace_mask();
let mp = eval.next_trace_mask();
let mv = eval.next_trace_mask();
let mvi = eval.next_trace_mask();
let d = eval.next_trace_mask();
let _next_clk = eval.next_trace_mask();
let next_ip = eval.next_trace_mask();
let _next_ci = eval.next_trace_mask();
let _next_ni = eval.next_trace_mask();
let next_mp = eval.next_trace_mask();
let _next_mv = eval.next_trace_mask();
let _next_mvi = eval.next_trace_mask();
let _next_d = eval.next_trace_mask();

// ┌─────────────────────────────┐
// │ Consistency Constraints │
// └─────────────────────────────┘

// `ci` is either 0 (dummy row), or equal to the Right (memory shift right) instruction `>`.
let right_instruction = ci.clone() - InstructionType::Right.to_base_field().into();

eval.add_constraint(ci.clone() * right_instruction);

// The dummy `d` is either 0 or 1
eval.add_constraint(d.clone() * (d.clone() - E::F::one()));

// If `d` is set, then `mv` equals 0
eval.add_constraint(d.clone() * mv.clone());

// If `d` is set, then `ci` equals 0
eval.add_constraint(d.clone() * ci.clone());

// ┌────────────────────────────┐
// │ Transition Constraints │
// └────────────────────────────┘

// `ip` increases by one
eval.add_constraint((E::F::one() - d.clone()) * (next_ip - ip.clone() - E::F::one()));

// `mp` increases by one
eval.add_constraint((E::F::one() - d.clone()) * (next_mp - mp.clone() - E::F::one()));

// ┌─────────────────────────────┐
// │ Interaction Constraints │
// └─────────────────────────────┘

let num = d - E::F::one();

eval.add_to_relation(&[RelationEntry::new(
&self.processor_lookup_elements,
num.into(),
&[clk, ip, ci, ni, mp, mv, mvi],
)]);

eval.finalize_logup();

eval
}
}

#[cfg(test)]
mod test {
use crate::components::processor::{
instructions::{
right_component::RightInstructionEval,
table::{interaction_trace_evaluation, RightInstructionTable},
},
table::ProcessorElements,
};
use brainfuck_vm::{compiler::Compiler, test_helper::create_test_machine};
use stwo_prover::{
constraint_framework::{
assert_constraints, preprocessed_columns::gen_is_first, FrameworkEval,
},
core::{
pcs::TreeVec,
poly::circle::{CanonicCoset, CircleEvaluation},
},
};

#[test]
fn test_right_instruction_constraints() {
const LOG_SIZE: u32 = 6;

// Get an execution trace from a valid Brainfuck program
let code = "+++>,<[>+.<-]";
let mut compiler = Compiler::new(code);
let instructions = compiler.compile();
let (mut machine, _) = create_test_machine(&instructions, &[1]);
let () = machine.execute().expect("Failed to execute machine");

let trace_vm = machine.trace();

// Construct the IsFirst preprocessed column
let is_first_col = gen_is_first(LOG_SIZE);
let preprocessed_trace = vec![is_first_col];

// Construct the main trace from the execution trace
let table = RightInstructionTable::from(&trace_vm);
let (main_trace, claim) = table.trace_evaluation();

// Draw Interaction elements
let processor_lookup_elements = ProcessorElements::dummy();

// Generate interaction trace
let (interaction_trace, interaction_claim) =
interaction_trace_evaluation(&main_trace, &processor_lookup_elements).unwrap();

// Create the trace evaluation TreeVec
let trace = TreeVec::new(vec![preprocessed_trace, main_trace, interaction_trace]);

// Interpolate the trace for the evaluation
let trace_polys = trace.map_cols(CircleEvaluation::interpolate);

// Get the Plus `+` AIR evaluator
let right_instruction_eval = RightInstructionEval::new(&claim, processor_lookup_elements);

// Assert that the constraints are valid for a valid Brainfuck program.
assert_constraints(
&trace_polys,
CanonicCoset::new(LOG_SIZE),
|eval| {
right_instruction_eval.evaluate(eval);
},
(interaction_claim.claimed_sum, None),
);
}
}

0 comments on commit 2499c33

Please sign in to comment.