sp1-core-machine 6.0.0-beta.1

SP1 is a performant, 100% open-source, contributor-friendly zkVM.
Documentation
use crate::air::WordAirBuilder;
use serde::{Deserialize, Serialize};
use slop_algebra::{AbstractField, Field};
use sp1_core_executor::{events::ByteRecord, ByteOpcode};
use sp1_derive::{AlignedBorrow, InputExpr, InputParams, IntoShape, SP1OperationBuilder};
use sp1_hypercube::air::SP1AirBuilder;

use crate::air::SP1Operation;

/// A set of columns to describe the state of the CPU.
/// The state is composed of the shard, clock, and the program counter.
/// The clock is split into 24 bits, 8 bits, 16 bits limbs.
#[derive(
    AlignedBorrow,
    Default,
    Debug,
    Clone,
    Copy,
    Serialize,
    Deserialize,
    IntoShape,
    SP1OperationBuilder,
)]
#[repr(C)]
pub struct CPUState<T> {
    pub clk_high: T,
    pub clk_16_24: T,
    pub clk_0_16: T,
    pub pc: [T; 3],
}

impl<T: Copy> CPUState<T> {
    pub fn clk_high<AB>(&self) -> AB::Expr
    where
        AB: SP1AirBuilder<Var = T>,
        T: Into<AB::Expr>,
    {
        self.clk_high.into()
    }
    pub fn clk_low<AB>(&self) -> AB::Expr
    where
        AB: SP1AirBuilder<Var = T>,
        T: Into<AB::Expr>,
    {
        self.clk_0_16.into() + self.clk_16_24.into() * AB::Expr::from_canonical_u32(1 << 16)
    }
}

impl<F: Field> CPUState<F> {
    #[allow(clippy::too_many_arguments)]
    pub fn populate(&mut self, blu_events: &mut impl ByteRecord, clk: u64, pc: u64) {
        let clk_high = (clk >> 24) as u32;
        let clk_16_24 = ((clk >> 16) & 0xFF) as u8;
        let clk_0_16 = (clk & 0xFFFF) as u16;
        self.clk_high = F::from_canonical_u32(clk_high);
        self.clk_16_24 = F::from_canonical_u8(clk_16_24);
        self.clk_0_16 = F::from_canonical_u16(clk_0_16);
        self.pc = [
            F::from_canonical_u16((pc & 0xFFFF) as u16),
            F::from_canonical_u16(((pc >> 16) & 0xFFFF) as u16),
            F::from_canonical_u16(((pc >> 32) & 0xFFFF) as u16),
        ];

        // 0 <= (clk_0_16 - 1) / 8 < 2^13 shows clk == 1 (mod 8) and clk_0_16 is 16 bits.
        blu_events.add_bit_range_check((clk_0_16 - 1) / 8, 13);
        blu_events.add_u8_range_checks(&[clk_16_24, 0]);
    }

    #[allow(clippy::too_many_arguments)]
    pub fn eval<AB: SP1AirBuilder>(
        builder: &mut AB,
        cols: CPUState<AB::Var>,
        next_pc: [AB::Expr; 3],
        clk_increment: AB::Expr,
        is_real: AB::Expr,
    ) {
        let clk_high = cols.clk_high::<AB>();
        let clk_low = cols.clk_low::<AB>();
        builder.assert_bool(is_real.clone());
        builder.receive_state(clk_high.clone(), clk_low.clone(), cols.pc, is_real.clone());
        builder.send_state(
            clk_high.clone(),
            clk_low.clone() + clk_increment,
            next_pc,
            is_real.clone(),
        );

        builder.send_byte(
            AB::Expr::from_canonical_u32(ByteOpcode::Range as u32),
            (cols.clk_0_16 - AB::Expr::one()) * AB::F::from_canonical_u8(8).inverse(),
            AB::Expr::from_canonical_u32(13),
            AB::Expr::zero(),
            is_real.clone(),
        );

        builder.slice_range_check_u8(&[cols.clk_16_24.into(), AB::Expr::zero()], is_real.clone());
    }
}

#[derive(Clone, InputParams, InputExpr)]
pub struct CPUStateInput<AB: SP1AirBuilder> {
    pub cols: CPUState<AB::Var>,
    pub next_pc: [AB::Expr; 3],
    pub clk_increment: AB::Expr,
    pub is_real: AB::Expr,
}

impl<AB: SP1AirBuilder> SP1Operation<AB> for CPUState<AB::F> {
    type Input = CPUStateInput<AB>;
    type Output = ();

    fn lower(builder: &mut AB, input: Self::Input) -> Self::Output {
        Self::eval(builder, input.cols, input.next_pc, input.clk_increment, input.is_real);
    }
}