use super::column::{N_MIPS_SEL_COLS, SCRATCH_SIZE, SCRATCH_SIZE_INVERSE};
use crate::{
cannon::{
Hint, Meta, Page, Start, State, StepFrequency, VmConfiguration, PAGE_ADDRESS_MASK,
PAGE_ADDRESS_SIZE, PAGE_SIZE,
},
interpreters::{
keccak::environment::KeccakEnv,
mips::{
column::{
ColumnAlias as Column, MIPS_BYTE_COUNTER_OFF, MIPS_CHUNK_BYTES_LEN,
MIPS_END_OF_PREIMAGE_OFF, MIPS_HASH_COUNTER_OFF, MIPS_HAS_N_BYTES_OFF,
MIPS_LENGTH_BYTES_OFF, MIPS_NUM_BYTES_READ_OFF, MIPS_PREIMAGE_BYTES_OFF,
MIPS_PREIMAGE_CHUNK_OFF, MIPS_PREIMAGE_KEY,
},
interpreter::{
self, ITypeInstruction, Instruction, InterpreterEnv, JTypeInstruction,
RTypeInstruction,
},
registers::Registers,
},
},
lookups::Lookup,
preimage_oracle::PreImageOracleT,
utils::memory_size,
};
use ark_ff::Field;
use core::panic;
use kimchi::o1_utils::Two;
use log::{debug, info};
use std::{
array,
fs::File,
io::{BufWriter, Write},
};
pub const MAX_NB_REG_ACC: u64 = 7;
pub const MAX_NB_MEM_ACC: u64 = 12;
pub const MAX_ACC: u64 = MAX_NB_REG_ACC + MAX_NB_MEM_ACC;
pub const NUM_GLOBAL_LOOKUP_TERMS: usize = 1;
pub const NUM_DECODING_LOOKUP_TERMS: usize = 2;
pub const NUM_INSTRUCTION_LOOKUP_TERMS: usize = 5;
pub const NUM_LOOKUP_TERMS: usize =
NUM_GLOBAL_LOOKUP_TERMS + NUM_DECODING_LOOKUP_TERMS + NUM_INSTRUCTION_LOOKUP_TERMS;
#[derive(Clone, Default)]
pub struct SyscallEnv {
pub last_hint: Option<Vec<u8>>,
}
impl SyscallEnv {
pub fn create(state: &State) -> Self {
SyscallEnv {
last_hint: state.last_hint.clone(),
}
}
}
pub struct Env<Fp, PreImageOracle: PreImageOracleT> {
pub instruction_counter: u64,
pub memory: Vec<(u32, Vec<u8>)>,
pub last_memory_accesses: [usize; 3],
pub memory_write_index: Vec<(u32, Vec<u64>)>,
pub last_memory_write_index_accesses: [usize; 3],
pub registers: Registers<u32>,
pub registers_write_index: Registers<u64>,
pub scratch_state_idx: usize,
pub scratch_state_idx_inverse: usize,
pub scratch_state: [Fp; SCRATCH_SIZE],
pub scratch_state_inverse: [Fp; SCRATCH_SIZE_INVERSE],
pub halt: bool,
pub syscall_env: SyscallEnv,
pub selector: usize,
pub preimage_oracle: PreImageOracle,
pub preimage: Option<Vec<u8>>,
pub preimage_bytes_read: u64,
pub preimage_key: Option<[u8; 32]>,
pub keccak_env: Option<KeccakEnv<Fp>>,
pub hash_counter: u64,
}
fn fresh_scratch_state<Fp: Field, const N: usize>() -> [Fp; N] {
array::from_fn(|_| Fp::zero())
}
impl<Fp: Field, PreImageOracle: PreImageOracleT> InterpreterEnv for Env<Fp, PreImageOracle> {
type Position = Column;
fn alloc_scratch(&mut self) -> Self::Position {
let scratch_idx = self.scratch_state_idx;
self.scratch_state_idx += 1;
Column::ScratchState(scratch_idx)
}
fn alloc_scratch_inverse(&mut self) -> Self::Position {
let scratch_idx = self.scratch_state_idx_inverse;
self.scratch_state_idx_inverse += 1;
Column::ScratchStateInverse(scratch_idx)
}
type Variable = u64;
fn variable(&self, _column: Self::Position) -> Self::Variable {
todo!()
}
fn add_constraint(&mut self, _assert_equals_zero: Self::Variable) {
}
fn activate_selector(&mut self, instruction: Instruction) {
self.selector = instruction.into();
}
fn check_is_zero(assert_equals_zero: &Self::Variable) {
assert_eq!(*assert_equals_zero, 0);
}
fn check_equal(x: &Self::Variable, y: &Self::Variable) {
assert_eq!(*x, *y);
}
fn check_boolean(x: &Self::Variable) {
if !(*x == 0 || *x == 1) {
panic!("The value {} is not a boolean", *x);
}
}
fn add_lookup(&mut self, _lookup: Lookup<Self::Variable>) {
}
fn instruction_counter(&self) -> Self::Variable {
self.instruction_counter
}
fn increase_instruction_counter(&mut self) {
self.instruction_counter += 1;
}
unsafe fn fetch_register(
&mut self,
idx: &Self::Variable,
output: Self::Position,
) -> Self::Variable {
let res = self.registers[*idx as usize] as u64;
self.write_column(output, res);
res
}
unsafe fn push_register_if(
&mut self,
idx: &Self::Variable,
value: Self::Variable,
if_is_true: &Self::Variable,
) {
let value: u32 = value.try_into().unwrap();
if *if_is_true == 1 {
self.registers[*idx as usize] = value
} else if *if_is_true == 0 {
} else {
panic!("Bad value for flag in push_register: {}", *if_is_true);
}
}
unsafe fn fetch_register_access(
&mut self,
idx: &Self::Variable,
output: Self::Position,
) -> Self::Variable {
let res = self.registers_write_index[*idx as usize];
self.write_column(output, res);
res
}
unsafe fn push_register_access_if(
&mut self,
idx: &Self::Variable,
value: Self::Variable,
if_is_true: &Self::Variable,
) {
if *if_is_true == 1 {
self.registers_write_index[*idx as usize] = value
} else if *if_is_true == 0 {
} else {
panic!("Bad value for flag in push_register: {}", *if_is_true);
}
}
unsafe fn fetch_memory(
&mut self,
addr: &Self::Variable,
output: Self::Position,
) -> Self::Variable {
let addr: u32 = (*addr).try_into().unwrap();
let page = addr >> PAGE_ADDRESS_SIZE;
let page_address = (addr & PAGE_ADDRESS_MASK) as usize;
let memory_page_idx = self.get_memory_page_index(page);
let value = self.memory[memory_page_idx].1[page_address];
self.write_column(output, value.into());
value.into()
}
unsafe fn push_memory(&mut self, addr: &Self::Variable, value: Self::Variable) {
let addr: u32 = (*addr).try_into().unwrap();
let page = addr >> PAGE_ADDRESS_SIZE;
let page_address = (addr & PAGE_ADDRESS_MASK) as usize;
let memory_page_idx = self.get_memory_page_index(page);
self.memory[memory_page_idx].1[page_address] =
value.try_into().expect("push_memory values fit in a u8");
}
unsafe fn fetch_memory_access(
&mut self,
addr: &Self::Variable,
output: Self::Position,
) -> Self::Variable {
let addr: u32 = (*addr).try_into().unwrap();
let page = addr >> PAGE_ADDRESS_SIZE;
let page_address = (addr & PAGE_ADDRESS_MASK) as usize;
let memory_write_index_page_idx = self.get_memory_access_page_index(page);
let value = self.memory_write_index[memory_write_index_page_idx].1[page_address];
self.write_column(output, value);
value
}
unsafe fn push_memory_access(&mut self, addr: &Self::Variable, value: Self::Variable) {
let addr = *addr as u32;
let page = addr >> PAGE_ADDRESS_SIZE;
let page_address = (addr & PAGE_ADDRESS_MASK) as usize;
let memory_write_index_page_idx = self.get_memory_access_page_index(page);
self.memory_write_index[memory_write_index_page_idx].1[page_address] = value;
}
fn constant(x: u32) -> Self::Variable {
x as u64
}
unsafe fn bitmask(
&mut self,
x: &Self::Variable,
highest_bit: u32,
lowest_bit: u32,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let res = (x >> lowest_bit) & ((1 << (highest_bit - lowest_bit)) - 1);
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn shift_left(
&mut self,
x: &Self::Variable,
by: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let by: u32 = (*by).try_into().unwrap();
let res = x << by;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn shift_right(
&mut self,
x: &Self::Variable,
by: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let by: u32 = (*by).try_into().unwrap();
let res = x >> by;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn shift_right_arithmetic(
&mut self,
x: &Self::Variable,
by: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let by: u32 = (*by).try_into().unwrap();
let res = ((x as i32) >> by) as u32;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn test_zero(&mut self, x: &Self::Variable, position: Self::Position) -> Self::Variable {
let res = if *x == 0 { 1 } else { 0 };
self.write_column(position, res);
res
}
fn is_zero(&mut self, x: &Self::Variable) -> Self::Variable {
let res = {
let pos = self.alloc_scratch();
unsafe { self.test_zero(x, pos) }
};
let pos = self.alloc_scratch_inverse();
if *x == 0 {
self.write_field_column(pos, Fp::zero());
} else {
self.write_field_column(pos, Fp::from(*x));
};
res
}
fn equal(&mut self, x: &Self::Variable, y: &Self::Variable) -> Self::Variable {
let to_zero_test = Fp::from(*x) - Fp::from(*y);
let res = {
let pos = self.alloc_scratch();
let is_zero: u64 = if to_zero_test == Fp::zero() { 1 } else { 0 };
self.write_column(pos, is_zero);
is_zero
};
let pos = self.alloc_scratch_inverse();
if to_zero_test == Fp::zero() {
self.write_field_column(pos, Fp::zero());
} else {
self.write_field_column(pos, to_zero_test);
};
res
}
unsafe fn test_less_than(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = if x < y { 1 } else { 0 };
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn test_less_than_signed(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = if (x as i32) < (y as i32) { 1 } else { 0 };
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn and_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = x & y;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn nor_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = !(x | y);
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn or_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = x | y;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn xor_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = x ^ y;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn add_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
out_position: Self::Position,
overflow_position: Self::Position,
) -> (Self::Variable, Self::Variable) {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = x.overflowing_add(y);
let (res_, overflow) = (res.0 as u64, res.1 as u64);
self.write_column(out_position, res_);
self.write_column(overflow_position, overflow);
(res_, overflow)
}
unsafe fn sub_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
out_position: Self::Position,
underflow_position: Self::Position,
) -> (Self::Variable, Self::Variable) {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = x.overflowing_sub(y);
let (res_, underflow) = (res.0 as u64, res.1 as u64);
self.write_column(out_position, res_);
self.write_column(underflow_position, underflow);
(res_, underflow)
}
unsafe fn mul_signed_witness(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let res = ((x as i32) * (y as i32)) as u32;
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn mul_hi_lo_signed(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position_hi: Self::Position,
position_lo: Self::Position,
) -> (Self::Variable, Self::Variable) {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let mul = (((x as i32) as i64) * ((y as i32) as i64)) as u64;
let hi = (mul >> 32) as u32;
let lo = (mul & ((1 << 32) - 1)) as u32;
let hi = hi as u64;
let lo = lo as u64;
self.write_column(position_hi, hi);
self.write_column(position_lo, lo);
(hi, lo)
}
unsafe fn mul_hi_lo(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position_hi: Self::Position,
position_lo: Self::Position,
) -> (Self::Variable, Self::Variable) {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let mul = (x as u64) * (y as u64);
let hi = (mul >> 32) as u32;
let lo = (mul & ((1 << 32) - 1)) as u32;
let hi = hi as u64;
let lo = lo as u64;
self.write_column(position_hi, hi);
self.write_column(position_lo, lo);
(hi, lo)
}
unsafe fn divmod_signed(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position_quotient: Self::Position,
position_remainder: Self::Position,
) -> (Self::Variable, Self::Variable) {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let q = ((x as i32) / (y as i32)) as u32;
let r = ((x as i32) % (y as i32)) as u32;
let q = q as u64;
let r = r as u64;
self.write_column(position_quotient, q);
self.write_column(position_remainder, r);
(q, r)
}
unsafe fn divmod(
&mut self,
x: &Self::Variable,
y: &Self::Variable,
position_quotient: Self::Position,
position_remainder: Self::Position,
) -> (Self::Variable, Self::Variable) {
let x: u32 = (*x).try_into().unwrap();
let y: u32 = (*y).try_into().unwrap();
let q = x / y;
let r = x % y;
let q = q as u64;
let r = r as u64;
self.write_column(position_quotient, q);
self.write_column(position_remainder, r);
(q, r)
}
unsafe fn count_leading_zeros(
&mut self,
x: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let res = x.leading_zeros();
let res = res as u64;
self.write_column(position, res);
res
}
unsafe fn count_leading_ones(
&mut self,
x: &Self::Variable,
position: Self::Position,
) -> Self::Variable {
let x: u32 = (*x).try_into().unwrap();
let res = x.leading_ones();
let res = res as u64;
self.write_column(position, res);
res
}
fn copy(&mut self, x: &Self::Variable, position: Self::Position) -> Self::Variable {
self.write_column(position, *x);
*x
}
fn set_halted(&mut self, flag: Self::Variable) {
if flag == 0 {
self.halt = false
} else if flag == 1 {
self.halt = true
} else {
panic!("Bad value for flag in set_halted: {}", flag);
}
}
fn report_exit(&mut self, exit_code: &Self::Variable) {
println!(
"Exited with code {} at step {}",
*exit_code,
self.normalized_instruction_counter()
);
}
fn request_preimage_write(
&mut self,
addr: &Self::Variable,
len: &Self::Variable,
pos: Self::Position,
) -> Self::Variable {
if self.registers.preimage_offset == 0 {
let mut preimage_key = [0u8; 32];
for i in 0..8 {
let bytes = u32::to_be_bytes(self.registers.preimage_key[i]);
for j in 0..4 {
preimage_key[4 * i + j] = bytes[j]
}
}
let preimage = self.preimage_oracle.get_preimage(preimage_key).get();
self.preimage = Some(preimage.clone());
self.preimage_key = Some(preimage_key);
}
const LENGTH_SIZE: usize = 8;
let preimage = self
.preimage
.as_ref()
.expect("to have a preimage if we're requesting it at a non-zero offset");
let preimage_len = preimage.len();
let preimage_offset = self.registers.preimage_offset as u64;
let max_read_len =
std::cmp::min(preimage_offset + len, (preimage_len + LENGTH_SIZE) as u64)
- preimage_offset;
let actual_read_len = std::cmp::min(max_read_len, 4 - (addr & 3));
let mut preimage_read_len = 0;
let mut chunk = 0;
for i in 0..actual_read_len {
let idx = (preimage_offset + i) as usize;
if idx < LENGTH_SIZE {
let len_i = idx % MIPS_CHUNK_BYTES_LEN;
let length_byte = u64::to_be_bytes(preimage_len as u64)[idx];
self.write_column(
Column::ScratchState(MIPS_LENGTH_BYTES_OFF + len_i),
length_byte as u64,
);
unsafe {
self.push_memory(&(*addr + i), length_byte as u64);
self.push_memory_access(&(*addr + i), self.next_instruction_counter());
}
} else {
let byte_i = (idx - LENGTH_SIZE) % MIPS_CHUNK_BYTES_LEN;
let preimage_byte = self.preimage.as_ref().unwrap()[idx - LENGTH_SIZE];
self.write_column(
Column::ScratchState(MIPS_PREIMAGE_BYTES_OFF + byte_i),
preimage_byte as u64,
);
chunk = chunk << 8 | preimage_byte as u64;
preimage_read_len += 1;
unsafe {
self.push_memory(&(*addr + i), preimage_byte as u64);
self.push_memory_access(&(*addr + i), self.next_instruction_counter());
}
}
}
self.write_column(Column::ScratchState(MIPS_PREIMAGE_CHUNK_OFF), chunk);
self.write_column(pos, actual_read_len);
self.write_column(
Column::ScratchState(MIPS_NUM_BYTES_READ_OFF),
preimage_read_len,
);
for i in 0..MIPS_CHUNK_BYTES_LEN {
if preimage_read_len > i as u64 {
self.write_column(Column::ScratchState(MIPS_HAS_N_BYTES_OFF + i), 1);
}
}
self.preimage_bytes_read += preimage_read_len;
self.write_column(
Column::ScratchState(MIPS_BYTE_COUNTER_OFF),
self.preimage_bytes_read,
);
if self.preimage_bytes_read == preimage_len as u64 {
self.write_column(Column::ScratchState(MIPS_END_OF_PREIMAGE_OFF), 1);
let bytes31 = (1..32).fold(Fp::zero(), |acc, i| {
acc * Fp::two_pow(8) + Fp::from(self.preimage_key.unwrap()[i])
});
self.write_field_column(Self::Position::ScratchState(MIPS_PREIMAGE_KEY), bytes31);
debug!("Preimage has been read entirely, triggering Keccak process");
self.keccak_env = Some(KeccakEnv::<Fp>::new(
self.hash_counter,
self.preimage.as_ref().unwrap(),
));
self.write_column(
Column::ScratchState(MIPS_HASH_COUNTER_OFF),
self.hash_counter,
);
self.preimage_bytes_read = 0;
self.preimage_key = None;
self.hash_counter += 1;
}
actual_read_len
}
fn request_hint_write(&mut self, addr: &Self::Variable, len: &Self::Variable) {
let mut last_hint = match std::mem::take(&mut self.syscall_env.last_hint) {
Some(mut last_hint) => {
last_hint.reserve(*len as usize);
last_hint
}
None => Vec::with_capacity(*len as usize),
};
for i in 0..*len {
unsafe { self.push_memory_access(&(*addr + i), self.next_instruction_counter()) };
let value = {
let addr: u32 = (*addr).try_into().unwrap();
let page = addr >> PAGE_ADDRESS_SIZE;
let page_address = (addr & PAGE_ADDRESS_MASK) as usize;
let memory_page_idx = self.get_memory_page_index(page);
self.memory[memory_page_idx].1[page_address]
};
last_hint.push(value);
}
let len = last_hint.len();
let mut idx = 0;
while idx + 4 <= len {
let hint_len = u32::from_be_bytes(last_hint[idx..idx + 4].try_into().unwrap()) as usize;
idx += 4;
if idx + hint_len <= len {
let hint = last_hint[idx..idx + hint_len].to_vec();
idx += hint_len;
self.preimage_oracle.hint(Hint::create(hint));
}
}
let remaining = last_hint[idx..len].to_vec();
self.syscall_env.last_hint = Some(remaining);
}
fn reset(&mut self) {
self.scratch_state_idx = 0;
self.scratch_state = fresh_scratch_state();
self.selector = N_MIPS_SEL_COLS;
}
}
impl<Fp: Field, PreImageOracle: PreImageOracleT> Env<Fp, PreImageOracle> {
pub fn create(page_size: usize, state: State, preimage_oracle: PreImageOracle) -> Self {
let initial_instruction_pointer = state.pc;
let next_instruction_pointer = state.next_pc;
let selector = N_MIPS_SEL_COLS;
let syscall_env = SyscallEnv::create(&state);
let mut initial_memory: Vec<(u32, Vec<u8>)> = state
.memory
.into_iter()
.map(|page| (page.index, page.data))
.collect();
for (_address, initial_memory) in initial_memory.iter_mut() {
initial_memory.extend((0..(page_size - initial_memory.len())).map(|_| 0u8));
assert_eq!(initial_memory.len(), page_size);
}
let memory_offsets = initial_memory
.iter()
.map(|(offset, _)| *offset)
.collect::<Vec<_>>();
let initial_registers = {
let preimage_key = {
let mut preimage_key = [0u32; 8];
for (i, preimage_key_word) in preimage_key.iter_mut().enumerate() {
*preimage_key_word = u32::from_be_bytes(
state.preimage_key[i * 4..(i + 1) * 4].try_into().unwrap(),
)
}
preimage_key
};
Registers {
lo: state.lo,
hi: state.hi,
general_purpose: state.registers,
current_instruction_pointer: initial_instruction_pointer,
next_instruction_pointer,
heap_pointer: state.heap,
preimage_key,
preimage_offset: state.preimage_offset,
}
};
Env {
instruction_counter: state.step,
memory: initial_memory.clone(),
last_memory_accesses: [0usize; 3],
memory_write_index: memory_offsets
.iter()
.map(|offset| (*offset, vec![0u64; page_size]))
.collect(),
last_memory_write_index_accesses: [0usize; 3],
registers: initial_registers.clone(),
registers_write_index: Registers::default(),
scratch_state_idx: 0,
scratch_state_idx_inverse: 0,
scratch_state: fresh_scratch_state(),
scratch_state_inverse: fresh_scratch_state(),
halt: state.exited,
syscall_env,
selector,
preimage_oracle,
preimage: state.preimage,
preimage_bytes_read: 0,
preimage_key: None,
keccak_env: None,
hash_counter: 0,
}
}
pub fn reset_scratch_state(&mut self) {
self.scratch_state_idx = 0;
self.scratch_state = fresh_scratch_state();
self.selector = N_MIPS_SEL_COLS;
}
pub fn reset_scratch_state_inverse(&mut self) {
self.scratch_state_idx_inverse = 0;
self.scratch_state_inverse = fresh_scratch_state();
}
pub fn write_column(&mut self, column: Column, value: u64) {
self.write_field_column(column, value.into())
}
pub fn write_field_column(&mut self, column: Column, value: Fp) {
match column {
Column::ScratchState(idx) => self.scratch_state[idx] = value,
Column::ScratchStateInverse(idx) => self.scratch_state_inverse[idx] = value,
Column::InstructionCounter => panic!("Cannot overwrite the column {:?}", column),
Column::Selector(s) => self.selector = s,
}
}
pub fn update_last_memory_access(&mut self, i: usize) {
let [i_0, i_1, _] = self.last_memory_accesses;
self.last_memory_accesses = [i, i_0, i_1]
}
pub fn get_memory_page_index(&mut self, page: u32) -> usize {
for &i in self.last_memory_accesses.iter() {
if self.memory_write_index[i].0 == page {
return i;
}
}
for (i, (page_index, _memory)) in self.memory.iter_mut().enumerate() {
if *page_index == page {
self.update_last_memory_access(i);
return i;
}
}
let memory = vec![0u8; PAGE_SIZE as usize];
self.memory.push((page, memory));
let i = self.memory.len() - 1;
self.update_last_memory_access(i);
i
}
pub fn update_last_memory_write_index_access(&mut self, i: usize) {
let [i_0, i_1, _] = self.last_memory_write_index_accesses;
self.last_memory_write_index_accesses = [i, i_0, i_1]
}
pub fn get_memory_access_page_index(&mut self, page: u32) -> usize {
for &i in self.last_memory_write_index_accesses.iter() {
if self.memory_write_index[i].0 == page {
return i;
}
}
for (i, (page_index, _memory_write_index)) in self.memory_write_index.iter_mut().enumerate()
{
if *page_index == page {
self.update_last_memory_write_index_access(i);
return i;
}
}
let memory_write_index = vec![0u64; PAGE_SIZE as usize];
self.memory_write_index.push((page, memory_write_index));
let i = self.memory_write_index.len() - 1;
self.update_last_memory_write_index_access(i);
i
}
pub fn get_memory_direct(&mut self, addr: u32) -> u8 {
let page = addr >> PAGE_ADDRESS_SIZE;
let page_address = (addr & PAGE_ADDRESS_MASK) as usize;
let memory_idx = self.get_memory_page_index(page);
self.memory[memory_idx].1[page_address]
}
pub fn decode_instruction(&mut self) -> (Instruction, u32) {
let instruction =
((self.get_memory_direct(self.registers.current_instruction_pointer) as u32) << 24)
| ((self.get_memory_direct(self.registers.current_instruction_pointer + 1) as u32)
<< 16)
| ((self.get_memory_direct(self.registers.current_instruction_pointer + 2) as u32)
<< 8)
| (self.get_memory_direct(self.registers.current_instruction_pointer + 3) as u32);
let opcode = {
match instruction >> 26 {
0x00 => match instruction & 0x3F {
0x00 => Instruction::RType(RTypeInstruction::ShiftLeftLogical),
0x02 => Instruction::RType(RTypeInstruction::ShiftRightLogical),
0x03 => Instruction::RType(RTypeInstruction::ShiftRightArithmetic),
0x04 => Instruction::RType(RTypeInstruction::ShiftLeftLogicalVariable),
0x06 => Instruction::RType(RTypeInstruction::ShiftRightLogicalVariable),
0x07 => Instruction::RType(RTypeInstruction::ShiftRightArithmeticVariable),
0x08 => Instruction::RType(RTypeInstruction::JumpRegister),
0x09 => Instruction::RType(RTypeInstruction::JumpAndLinkRegister),
0x0a => Instruction::RType(RTypeInstruction::MoveZero),
0x0b => Instruction::RType(RTypeInstruction::MoveNonZero),
0x0c => match self.registers.general_purpose[2] {
4090 => Instruction::RType(RTypeInstruction::SyscallMmap),
4045 => {
Instruction::RType(RTypeInstruction::SyscallOther)
}
4120 => {
Instruction::RType(RTypeInstruction::SyscallOther)
}
4246 => Instruction::RType(RTypeInstruction::SyscallExitGroup),
4003 => match self.registers.general_purpose[4] {
interpreter::FD_HINT_READ => {
Instruction::RType(RTypeInstruction::SyscallReadHint)
}
interpreter::FD_PREIMAGE_READ => {
Instruction::RType(RTypeInstruction::SyscallReadPreimage)
}
_ => Instruction::RType(RTypeInstruction::SyscallReadOther),
},
4004 => match self.registers.general_purpose[4] {
interpreter::FD_PREIMAGE_WRITE => {
Instruction::RType(RTypeInstruction::SyscallWritePreimage)
}
interpreter::FD_HINT_WRITE => {
Instruction::RType(RTypeInstruction::SyscallWriteHint)
}
_ => Instruction::RType(RTypeInstruction::SyscallWriteOther),
},
4055 => Instruction::RType(RTypeInstruction::SyscallFcntl),
_ => {
Instruction::RType(RTypeInstruction::SyscallOther)
}
},
0x0f => Instruction::RType(RTypeInstruction::Sync),
0x10 => Instruction::RType(RTypeInstruction::MoveFromHi),
0x11 => Instruction::RType(RTypeInstruction::MoveToHi),
0x12 => Instruction::RType(RTypeInstruction::MoveFromLo),
0x13 => Instruction::RType(RTypeInstruction::MoveToLo),
0x18 => Instruction::RType(RTypeInstruction::Multiply),
0x19 => Instruction::RType(RTypeInstruction::MultiplyUnsigned),
0x1a => Instruction::RType(RTypeInstruction::Div),
0x1b => Instruction::RType(RTypeInstruction::DivUnsigned),
0x20 => Instruction::RType(RTypeInstruction::Add),
0x21 => Instruction::RType(RTypeInstruction::AddUnsigned),
0x22 => Instruction::RType(RTypeInstruction::Sub),
0x23 => Instruction::RType(RTypeInstruction::SubUnsigned),
0x24 => Instruction::RType(RTypeInstruction::And),
0x25 => Instruction::RType(RTypeInstruction::Or),
0x26 => Instruction::RType(RTypeInstruction::Xor),
0x27 => Instruction::RType(RTypeInstruction::Nor),
0x2a => Instruction::RType(RTypeInstruction::SetLessThan),
0x2b => Instruction::RType(RTypeInstruction::SetLessThanUnsigned),
_ => {
panic!("Unhandled instruction {:#X}", instruction)
}
},
0x01 => {
match (instruction >> 16) & 0x1F {
0x0 => Instruction::IType(ITypeInstruction::BranchLtZero),
0x1 => Instruction::IType(ITypeInstruction::BranchGeqZero),
_ => panic!("Unhandled instruction {:#X}", instruction),
}
}
0x02 => Instruction::JType(JTypeInstruction::Jump),
0x03 => Instruction::JType(JTypeInstruction::JumpAndLink),
0x04 => Instruction::IType(ITypeInstruction::BranchEq),
0x05 => Instruction::IType(ITypeInstruction::BranchNeq),
0x06 => Instruction::IType(ITypeInstruction::BranchLeqZero),
0x07 => Instruction::IType(ITypeInstruction::BranchGtZero),
0x08 => Instruction::IType(ITypeInstruction::AddImmediate),
0x09 => Instruction::IType(ITypeInstruction::AddImmediateUnsigned),
0x0A => Instruction::IType(ITypeInstruction::SetLessThanImmediate),
0x0B => Instruction::IType(ITypeInstruction::SetLessThanImmediateUnsigned),
0x0C => Instruction::IType(ITypeInstruction::AndImmediate),
0x0D => Instruction::IType(ITypeInstruction::OrImmediate),
0x0E => Instruction::IType(ITypeInstruction::XorImmediate),
0x0F => Instruction::IType(ITypeInstruction::LoadUpperImmediate),
0x1C => match instruction & 0x3F {
0x02 => Instruction::RType(RTypeInstruction::MultiplyToRegister),
0x20 => Instruction::RType(RTypeInstruction::CountLeadingZeros),
0x21 => Instruction::RType(RTypeInstruction::CountLeadingOnes),
_ => panic!("Unhandled instruction {:#X}", instruction),
},
0x20 => Instruction::IType(ITypeInstruction::Load8),
0x21 => Instruction::IType(ITypeInstruction::Load16),
0x22 => Instruction::IType(ITypeInstruction::LoadWordLeft),
0x23 => Instruction::IType(ITypeInstruction::Load32),
0x24 => Instruction::IType(ITypeInstruction::Load8Unsigned),
0x25 => Instruction::IType(ITypeInstruction::Load16Unsigned),
0x26 => Instruction::IType(ITypeInstruction::LoadWordRight),
0x28 => Instruction::IType(ITypeInstruction::Store8),
0x29 => Instruction::IType(ITypeInstruction::Store16),
0x2a => Instruction::IType(ITypeInstruction::StoreWordLeft),
0x2b => Instruction::IType(ITypeInstruction::Store32),
0x2e => Instruction::IType(ITypeInstruction::StoreWordRight),
0x30 => {
Instruction::IType(ITypeInstruction::Load32)
}
0x38 => {
Instruction::IType(ITypeInstruction::Store32Conditional)
}
_ => {
panic!("Unhandled instruction {:#X}", instruction)
}
}
};
(opcode, instruction)
}
pub fn normalized_instruction_counter(&self) -> u64 {
self.instruction_counter / MAX_ACC
}
pub fn next_instruction_counter(&self) -> u64 {
(self.normalized_instruction_counter() + 1) * MAX_ACC
}
pub fn step(
&mut self,
config: &VmConfiguration,
metadata: &Option<Meta>,
start: &Start,
) -> Instruction {
self.reset_scratch_state();
self.reset_scratch_state_inverse();
let (opcode, _instruction) = self.decode_instruction();
self.pp_info(&config.info_at, metadata, start);
self.snapshot_state_at(&config.snapshot_state_at);
if self.should_trigger_at(&config.stop_at) {
self.halt = true;
println!(
"Halted as requested at step={} instruction={:?}",
self.normalized_instruction_counter(),
opcode
);
return opcode;
}
interpreter::interpret_instruction(self, opcode);
self.instruction_counter = self.next_instruction_counter();
if self.halt {
println!(
"Halted at step={} instruction={:?}",
self.normalized_instruction_counter(),
opcode
);
}
opcode
}
fn should_trigger_at(&self, at: &StepFrequency) -> bool {
let m: u64 = self.normalized_instruction_counter();
match at {
StepFrequency::Never => false,
StepFrequency::Always => true,
StepFrequency::Exactly(n) => *n == m,
StepFrequency::Every(n) => m % *n == 0,
StepFrequency::Range(lo, hi_opt) => {
m >= *lo && (hi_opt.is_none() || m < hi_opt.unwrap())
}
}
}
fn memory_usage(&self) -> String {
let total = self.memory.len() * PAGE_SIZE as usize;
memory_size(total)
}
fn page_address(&self) -> (u32, usize) {
let address = self.registers.current_instruction_pointer;
let page = address >> PAGE_ADDRESS_SIZE;
let page_address = (address & PAGE_ADDRESS_MASK) as usize;
(page, page_address)
}
fn get_opcode(&mut self) -> Option<u32> {
let (page_id, page_address) = self.page_address();
for (page_index, memory) in self.memory.iter() {
if page_id == *page_index {
let memory_slice: [u8; 4] = memory[page_address..page_address + 4]
.try_into()
.expect("Couldn't read 4 bytes at given address");
return Some(u32::from_be_bytes(memory_slice));
}
}
None
}
fn snapshot_state_at(&mut self, at: &StepFrequency) {
if self.should_trigger_at(at) {
let filename = format!(
"snapshot-state-{}.json",
self.normalized_instruction_counter()
);
let file = File::create(filename.clone()).expect("Impossible to open file");
let mut writer = BufWriter::new(file);
let mut preimage_key = [0u8; 32];
for i in 0..8 {
let bytes = u32::to_be_bytes(self.registers.preimage_key[i]);
for j in 0..4 {
preimage_key[4 * i + j] = bytes[j]
}
}
let memory = self
.memory
.clone()
.into_iter()
.map(|(idx, data)| Page { index: idx, data })
.collect();
let s: State = State {
pc: self.registers.current_instruction_pointer,
next_pc: self.registers.next_instruction_pointer,
step: self.normalized_instruction_counter(),
registers: self.registers.general_purpose,
lo: self.registers.lo,
hi: self.registers.hi,
heap: self.registers.heap_pointer,
exit: if self.halt { 1 } else { 0 },
last_hint: self.syscall_env.last_hint.clone(),
exited: self.halt,
preimage_offset: self.registers.preimage_offset,
preimage_key,
memory,
preimage: self.preimage.clone(),
};
let _ = serde_json::to_writer(&mut writer, &s);
info!(
"Snapshot state in {}, step {}",
filename,
self.normalized_instruction_counter()
);
writer.flush().expect("Flush writer failing")
}
}
fn pp_info(&mut self, at: &StepFrequency, meta: &Option<Meta>, start: &Start) {
if self.should_trigger_at(at) {
let elapsed = start.time.elapsed();
let step = self.normalized_instruction_counter();
let pc = self.registers.current_instruction_pointer;
let insn = self.get_opcode().unwrap();
let how_many_steps = step as usize - start.step;
let ips = how_many_steps as f64 / elapsed.as_secs() as f64;
let pages = self.memory.len();
let mem = self.memory_usage();
let name = meta
.as_ref()
.and_then(|m| m.find_address_symbol(pc))
.unwrap_or("n/a".to_string());
info!(
"processing step={} pc={:010x} insn={:010x} ips={:.2} pages={} mem={} name={}",
step, pc, insn, ips, pages, mem, name
);
}
}
}