sp1-core-machine 6.0.0-beta.1

SP1 is a performant, 100% open-source, contributor-friendly zkVM.
Documentation
use sp1_core_executor::{events::ByteRecord, ByteOpcode};
use sp1_hypercube::air::SP1AirBuilder;

use slop_air::AirBuilder;
use slop_algebra::{AbstractField, Field};
use sp1_derive::AlignedBorrow;

use crate::air::WordAirBuilder;

/// A set of columns needed to increment the clk and handle the carry.
#[derive(AlignedBorrow, Default, Debug, Clone, Copy)]
#[repr(C)]
pub struct ClkOperation<T> {
    next_clk_16_24: T,
    next_clk_0_16: T,
    is_overflow: T,
}

impl<T: Copy> ClkOperation<T> {
    pub fn next_clk_high<AB>(&self, clk_high: AB::Var) -> AB::Expr
    where
        AB: SP1AirBuilder<Var = T>,
        T: Into<AB::Expr>,
    {
        clk_high.into() + self.is_overflow
    }

    pub fn next_clk_low<AB>(&self) -> AB::Expr
    where
        AB: SP1AirBuilder<Var = T>,
        T: Into<AB::Expr>,
    {
        self.next_clk_0_16.into()
            + self.next_clk_16_24.into() * AB::Expr::from_canonical_u32(1 << 16)
    }
}

impl<F: Field> ClkOperation<F> {
    pub fn populate(&mut self, record: &mut impl ByteRecord, clk: u64, increment: u64) {
        let next_clk = clk + increment;
        let next_clk_16_24 = ((next_clk >> 16) & 0xFF) as u8;
        let next_clk_0_16 = (next_clk & 0xFFFF) as u16;

        let is_overflow = (next_clk >> 24) != (clk >> 24);
        self.is_overflow = F::from_canonical_u8(is_overflow as u8);
        self.next_clk_16_24 = F::from_canonical_u8(next_clk_16_24);
        self.next_clk_0_16 = F::from_canonical_u16(next_clk_0_16);

        record.add_bit_range_check(next_clk_0_16, 16);
        record.add_u8_range_checks(&[next_clk_16_24, 0]);
    }

    // Check that `clk_low + increment` overflows 24 bits.
    // Checks that `is_real` is boolean. If `is_real` is true, `next_clk` limbs are correct
    // low 24 bits of `clk_low + increment`, and `is_overflow` is the carry.
    // This function assumes that `clk_low` and `increment` is within 24 bits.
    pub fn eval<AB: SP1AirBuilder>(
        builder: &mut AB,
        clk_low: AB::Expr,
        increment: AB::Expr,
        cols: ClkOperation<AB::Var>,
        is_real: AB::Expr,
    ) {
        // Check that `is_real` is boolean.
        builder.assert_bool(is_real.clone());

        // Check that `is_overflow` is boolean.
        builder.assert_bool(cols.is_overflow);

        // Constrain the `next_clk_low` value.
        // If `is_overflow` is false, then it's equal to `clk_low + increment`.
        // If `is_overflow` is true, then it's equal to `clk_low + increment - 2^24`.
        builder.when(is_real.clone()).assert_eq(
            clk_low.clone() + increment.clone()
                - cols.is_overflow.into() * AB::Expr::from_canonical_u32(1 << 24),
            cols.next_clk_low::<AB>(),
        );

        // Constrain that `next_clk_low` is a valid 24 bit value by decomposing into two limbs.
        builder.send_byte(
            AB::Expr::from_canonical_u32(ByteOpcode::Range as u32),
            cols.next_clk_0_16.into(),
            AB::Expr::from_canonical_u32(16),
            AB::Expr::zero(),
            is_real.clone(),
        );
        builder
            .slice_range_check_u8(&[cols.next_clk_16_24.into(), AB::Expr::zero()], is_real.clone());
    }
}