use acir::{
AcirField,
brillig::{BitSize, IntegerBitSize, MemoryAddress},
};
pub const MEMORY_ADDRESSING_BIT_SIZE: IntegerBitSize = IntegerBitSize::U32;
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub enum MemoryValue<F> {
Field(F),
U1(bool),
U8(u8),
U16(u16),
U32(u32),
U64(u64),
U128(u128),
}
#[derive(Debug, thiserror::Error)]
pub enum MemoryTypeError {
#[error(
"Bit size for value {value_bit_size} does not match the expected bit size {expected_bit_size}"
)]
MismatchedBitSize { value_bit_size: u32, expected_bit_size: u32 },
#[error("Value is not an integer")]
NotAnInteger,
}
impl<F: std::fmt::Display> MemoryValue<F> {
pub fn new_field(value: F) -> Self {
MemoryValue::Field(value)
}
pub fn new_integer(value: u128, bit_size: IntegerBitSize) -> Self {
match bit_size {
IntegerBitSize::U1 => MemoryValue::U1(value != 0),
IntegerBitSize::U8 => MemoryValue::U8(value as u8),
IntegerBitSize::U16 => MemoryValue::U16(value as u16),
IntegerBitSize::U32 => MemoryValue::U32(value as u32),
IntegerBitSize::U64 => MemoryValue::U64(value as u64),
IntegerBitSize::U128 => MemoryValue::U128(value),
}
}
pub fn bit_size(&self) -> BitSize {
match self {
MemoryValue::Field(_) => BitSize::Field,
MemoryValue::U1(_) => BitSize::Integer(IntegerBitSize::U1),
MemoryValue::U8(_) => BitSize::Integer(IntegerBitSize::U8),
MemoryValue::U16(_) => BitSize::Integer(IntegerBitSize::U16),
MemoryValue::U32(_) => BitSize::Integer(IntegerBitSize::U32),
MemoryValue::U64(_) => BitSize::Integer(IntegerBitSize::U64),
MemoryValue::U128(_) => BitSize::Integer(IntegerBitSize::U128),
}
}
pub fn to_usize(&self) -> usize {
match self {
MemoryValue::U32(value) => (*value).try_into().unwrap(),
other => panic!("value is not typed as brillig usize: {other}"),
}
}
}
impl<F: AcirField> MemoryValue<F> {
pub fn new_from_field(value: F, bit_size: BitSize) -> Self {
if let BitSize::Integer(bit_size) = bit_size {
MemoryValue::new_integer(value.to_u128(), bit_size)
} else {
MemoryValue::new_field(value)
}
}
pub fn new_checked(value: F, bit_size: BitSize) -> Option<Self> {
if let BitSize::Integer(bit_size) = bit_size {
if value.num_bits() > bit_size.into() {
return None;
}
}
Some(MemoryValue::new_from_field(value, bit_size))
}
pub fn to_field(&self) -> F {
match self {
MemoryValue::Field(value) => *value,
MemoryValue::U1(value) => F::from(*value),
MemoryValue::U8(value) => F::from(*value as u128),
MemoryValue::U16(value) => F::from(*value as u128),
MemoryValue::U32(value) => F::from(*value as u128),
MemoryValue::U64(value) => F::from(*value as u128),
MemoryValue::U128(value) => F::from(*value),
}
}
pub fn to_u128(&self) -> Result<u128, MemoryTypeError> {
match self {
MemoryValue::Field(..) => Err(MemoryTypeError::NotAnInteger),
MemoryValue::U1(value) => Ok(*value as u8 as u128),
MemoryValue::U8(value) => Ok(*value as u128),
MemoryValue::U16(value) => Ok(*value as u128),
MemoryValue::U32(value) => Ok(*value as u128),
MemoryValue::U64(value) => Ok(*value as u128),
MemoryValue::U128(value) => Ok(*value),
}
}
pub fn expect_field(self) -> Result<F, MemoryTypeError> {
if let MemoryValue::Field(field) = self {
Ok(field)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: F::max_num_bits(),
})
}
}
pub(crate) fn expect_u1(self) -> Result<bool, MemoryTypeError> {
if let MemoryValue::U1(value) = self {
Ok(value)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: 1,
})
}
}
pub(crate) fn expect_u8(self) -> Result<u8, MemoryTypeError> {
if let MemoryValue::U8(value) = self {
Ok(value)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: 8,
})
}
}
pub(crate) fn expect_u16(self) -> Result<u16, MemoryTypeError> {
if let MemoryValue::U16(value) = self {
Ok(value)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: 16,
})
}
}
pub(crate) fn expect_u32(self) -> Result<u32, MemoryTypeError> {
if let MemoryValue::U32(value) = self {
Ok(value)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: 32,
})
}
}
pub(crate) fn expect_u64(self) -> Result<u64, MemoryTypeError> {
if let MemoryValue::U64(value) = self {
Ok(value)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: 64,
})
}
}
pub(crate) fn expect_u128(self) -> Result<u128, MemoryTypeError> {
if let MemoryValue::U128(value) = self {
Ok(value)
} else {
Err(MemoryTypeError::MismatchedBitSize {
value_bit_size: self.bit_size().to_u32::<F>(),
expected_bit_size: 128,
})
}
}
}
impl<F: std::fmt::Display> std::fmt::Display for MemoryValue<F> {
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> Result<(), ::std::fmt::Error> {
match self {
MemoryValue::Field(value) => write!(f, "{value}: field"),
MemoryValue::U1(value) => write!(f, "{value}: u1"),
MemoryValue::U8(value) => write!(f, "{value}: u8"),
MemoryValue::U16(value) => write!(f, "{value}: u16"),
MemoryValue::U32(value) => write!(f, "{value}: u32"),
MemoryValue::U64(value) => write!(f, "{value}: u64"),
MemoryValue::U128(value) => write!(f, "{value}: u128"),
}
}
}
impl<F: AcirField> Default for MemoryValue<F> {
fn default() -> Self {
MemoryValue::new_field(F::zero())
}
}
impl<F: AcirField> From<bool> for MemoryValue<F> {
fn from(value: bool) -> Self {
MemoryValue::U1(value)
}
}
impl<F: AcirField> From<u8> for MemoryValue<F> {
fn from(value: u8) -> Self {
MemoryValue::U8(value)
}
}
impl<F: AcirField> From<usize> for MemoryValue<F> {
fn from(value: usize) -> Self {
MemoryValue::U32(value as u32)
}
}
impl<F: AcirField> From<u32> for MemoryValue<F> {
fn from(value: u32) -> Self {
MemoryValue::U32(value)
}
}
impl<F: AcirField> From<u64> for MemoryValue<F> {
fn from(value: u64) -> Self {
MemoryValue::U64(value)
}
}
impl<F: AcirField> From<u128> for MemoryValue<F> {
fn from(value: u128) -> Self {
MemoryValue::U128(value)
}
}
impl<F: AcirField> TryFrom<MemoryValue<F>> for bool {
type Error = MemoryTypeError;
fn try_from(memory_value: MemoryValue<F>) -> Result<Self, Self::Error> {
memory_value.expect_u1()
}
}
impl<F: AcirField> TryFrom<MemoryValue<F>> for u8 {
type Error = MemoryTypeError;
fn try_from(memory_value: MemoryValue<F>) -> Result<Self, Self::Error> {
memory_value.expect_u8()
}
}
impl<F: AcirField> TryFrom<MemoryValue<F>> for u32 {
type Error = MemoryTypeError;
fn try_from(memory_value: MemoryValue<F>) -> Result<Self, Self::Error> {
memory_value.expect_u32()
}
}
impl<F: AcirField> TryFrom<MemoryValue<F>> for u64 {
type Error = MemoryTypeError;
fn try_from(memory_value: MemoryValue<F>) -> Result<Self, Self::Error> {
memory_value.expect_u64()
}
}
impl<F: AcirField> TryFrom<MemoryValue<F>> for u128 {
type Error = MemoryTypeError;
fn try_from(memory_value: MemoryValue<F>) -> Result<Self, Self::Error> {
memory_value.expect_u128()
}
}
#[derive(Debug, Clone, Default, PartialEq, Eq)]
pub struct Memory<F> {
inner: Vec<MemoryValue<F>>,
}
impl<F: AcirField> Memory<F> {
fn get_stack_pointer(&self) -> usize {
self.read(MemoryAddress::Direct(0)).to_usize()
}
fn resolve(&self, address: MemoryAddress) -> usize {
match address {
MemoryAddress::Direct(address) => address,
MemoryAddress::Relative(offset) => self.get_stack_pointer() + offset,
}
}
pub fn read(&self, address: MemoryAddress) -> MemoryValue<F> {
let resolved_addr = self.resolve(address);
self.inner.get(resolved_addr).copied().unwrap_or_default()
}
pub fn read_ref(&self, ptr: MemoryAddress) -> MemoryAddress {
MemoryAddress::direct(self.read(ptr).to_usize())
}
pub fn read_slice(&self, addr: MemoryAddress, len: usize) -> &[MemoryValue<F>] {
if len == 0 {
return &[];
}
let resolved_addr = self.resolve(addr);
&self.inner[resolved_addr..(resolved_addr + len)]
}
pub fn write(&mut self, address: MemoryAddress, value: MemoryValue<F>) {
let resolved_ptr = self.resolve(address);
self.resize_to_fit(resolved_ptr + 1);
self.inner[resolved_ptr] = value;
}
fn resize_to_fit(&mut self, size: usize) {
let new_size = std::cmp::max(self.inner.len(), size);
self.inner.resize(new_size, MemoryValue::default());
}
pub fn write_slice(&mut self, address: MemoryAddress, values: &[MemoryValue<F>]) {
let resolved_address = self.resolve(address);
self.resize_to_fit(resolved_address + values.len());
self.inner[resolved_address..(resolved_address + values.len())].copy_from_slice(values);
}
pub fn values(&self) -> &[MemoryValue<F>] {
&self.inner
}
}