use slop_air::AirBuilder;
use slop_algebra::{AbstractField, Field, PrimeField32};
use sp1_derive::AlignedBorrow;
use sp1_core_executor::{events::ByteRecord, ByteOpcode};
use sp1_hypercube::air::SP1AirBuilder;
use sp1_primitives::consts::u64_to_u16_limbs;
use super::{IsZeroOperation, IsZeroOperationInput};
use crate::air::{SP1Operation, SP1OperationBuilder};
#[derive(AlignedBorrow, Default, Debug, Clone, Copy)]
#[repr(C)]
pub struct SyscallAddrOperation<T> {
pub addr: [T; 3],
pub top_two_limb_min: T,
pub top_two_limb_max: IsZeroOperation<T>,
}
impl<F: PrimeField32> SyscallAddrOperation<F> {
pub fn populate(&mut self, record: &mut impl ByteRecord, addr: u64, len: u64) {
let addr_limbs = u64_to_u16_limbs(addr);
let sum_top_two_limb =
F::from_canonical_u16(addr_limbs[1]) + F::from_canonical_u16(addr_limbs[2]);
self.addr[0] = F::from_canonical_u16(addr_limbs[0]);
self.addr[1] = F::from_canonical_u16(addr_limbs[1]);
self.addr[2] = F::from_canonical_u16(addr_limbs[2]);
self.top_two_limb_min = sum_top_two_limb.inverse();
let is_max = self.top_two_limb_max.populate_from_field_element(
sum_top_two_limb - F::from_canonical_u16(u16::MAX) * F::two(),
);
if is_max == 1 {
record.add_bit_range_check((addr_limbs[0] + (len as u16)) / 8, 13);
} else {
record.add_bit_range_check(addr_limbs[0] / 8, 13);
}
}
}
impl<F: Field> SyscallAddrOperation<F> {
#[allow(clippy::too_many_arguments)]
pub fn eval<AB: SP1AirBuilder + SP1OperationBuilder<IsZeroOperation<<AB as AirBuilder>::F>>>(
builder: &mut AB,
len: u32,
cols: SyscallAddrOperation<AB::Var>,
is_real: AB::Expr,
) -> [AB::Var; 3] {
assert!(len.is_multiple_of(8));
builder.assert_bool(is_real.clone());
let sum_top_two_limb = cols.addr[1] + cols.addr[2];
builder.assert_eq(cols.top_two_limb_min * sum_top_two_limb.clone(), is_real.clone());
IsZeroOperation::<AB::F>::eval(
builder,
IsZeroOperationInput::new(
sum_top_two_limb.clone() - AB::Expr::from_canonical_u16(u16::MAX) * AB::Expr::two(),
cols.top_two_limb_max,
is_real.clone(),
),
);
builder.send_byte(
AB::Expr::from_canonical_u32(ByteOpcode::Range as u32),
(cols.addr[0] + cols.top_two_limb_max.result * AB::Expr::from_canonical_u32(len))
* AB::F::from_canonical_u32(8).inverse(),
AB::Expr::from_canonical_u32(13),
AB::Expr::zero(),
is_real.clone(),
);
[cols.addr[0], cols.addr[1], cols.addr[2]]
}
}