2024-11-05 19:38:26 -05:00
|
|
|
//! Virtual machine and errors
|
2024-10-30 09:32:46 -04:00
|
|
|
use std::{cmp::Ordering, mem::replace};
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-09-12 00:39:31 -04:00
|
|
|
use crate::{
|
2024-11-06 13:50:49 -05:00
|
|
|
parse, value::ConcreteValue, AnnotatedError, Chunk, ChunkError, DustError, FunctionType,
|
|
|
|
Instruction, Local, NativeFunction, NativeFunctionError, Operation, Span, Type, Value,
|
|
|
|
ValueError,
|
2024-09-12 00:39:31 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
pub fn run(source: &str) -> Result<Option<Value>, DustError> {
|
|
|
|
let chunk = parse(source)?;
|
2024-10-18 23:34:48 -04:00
|
|
|
let vm = Vm::new(chunk);
|
2024-09-12 00:39:31 -04:00
|
|
|
|
|
|
|
vm.run()
|
|
|
|
.map_err(|error| DustError::Runtime { error, source })
|
|
|
|
}
|
|
|
|
|
2024-11-05 19:38:26 -05:00
|
|
|
/// Dust virtual machine.
|
|
|
|
///
|
|
|
|
/// See the [module-level documentation](index.html) for more information.
|
2024-09-12 00:39:31 -04:00
|
|
|
#[derive(Debug, Eq, PartialEq)]
|
|
|
|
pub struct Vm {
|
|
|
|
ip: usize,
|
2024-10-12 03:06:44 -04:00
|
|
|
chunk: Chunk,
|
|
|
|
stack: Vec<Register>,
|
2024-10-29 23:11:55 -04:00
|
|
|
last_assigned_register: Option<u8>,
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Vm {
|
|
|
|
const STACK_LIMIT: usize = u16::MAX as usize;
|
|
|
|
|
|
|
|
pub fn new(chunk: Chunk) -> Self {
|
|
|
|
Self {
|
|
|
|
ip: 0,
|
2024-10-12 03:06:44 -04:00
|
|
|
chunk,
|
|
|
|
stack: Vec::new(),
|
2024-10-29 23:11:55 -04:00
|
|
|
last_assigned_register: None,
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-18 23:34:48 -04:00
|
|
|
pub fn run(mut self) -> Result<Option<Value>, VmError> {
|
2024-09-23 23:38:49 -04:00
|
|
|
// DRY helper to get constant or register values for binary operations
|
2024-09-23 19:16:15 -04:00
|
|
|
fn get_arguments(
|
2024-09-18 23:02:28 -04:00
|
|
|
vm: &mut Vm,
|
|
|
|
instruction: Instruction,
|
|
|
|
position: Span,
|
2024-09-23 19:16:15 -04:00
|
|
|
) -> Result<(&Value, &Value), VmError> {
|
2024-09-24 16:49:17 -04:00
|
|
|
let left = if instruction.b_is_constant() {
|
2024-11-06 13:50:49 -05:00
|
|
|
vm.get_constant(instruction.b(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
vm.get_register(instruction.b(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
};
|
2024-09-24 16:49:17 -04:00
|
|
|
let right = if instruction.c_is_constant() {
|
2024-11-06 13:50:49 -05:00
|
|
|
vm.get_constant(instruction.c(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
vm.get_register(instruction.c(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok((left, right))
|
2024-09-18 23:02:28 -04:00
|
|
|
}
|
2024-09-17 21:10:44 -04:00
|
|
|
|
2024-09-12 00:39:31 -04:00
|
|
|
while let Ok((instruction, position)) = self.read(Span(0, 0)).copied() {
|
2024-10-12 03:06:44 -04:00
|
|
|
log::info!(
|
|
|
|
"{} | {} | {} | {}",
|
|
|
|
self.ip - 1,
|
|
|
|
position,
|
|
|
|
instruction.operation(),
|
2024-11-05 23:20:58 -05:00
|
|
|
instruction.disassembly_info(&self.chunk)
|
2024-10-12 03:06:44 -04:00
|
|
|
);
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-09-15 01:24:04 -04:00
|
|
|
match instruction.operation() {
|
2024-11-01 02:51:18 -04:00
|
|
|
Operation::Move => {
|
|
|
|
let to_register = instruction.a();
|
|
|
|
let from_register = instruction.b();
|
2024-11-06 15:08:51 -05:00
|
|
|
let from_register_has_value = self
|
|
|
|
.stack
|
|
|
|
.get(from_register as usize)
|
|
|
|
.is_some_and(|register| !matches!(register, Register::Empty));
|
2024-11-01 02:51:18 -04:00
|
|
|
|
2024-11-06 15:08:51 -05:00
|
|
|
if from_register_has_value {
|
|
|
|
self.set_pointer(to_register, from_register, position)?;
|
|
|
|
}
|
2024-11-01 02:51:18 -04:00
|
|
|
}
|
2024-09-17 23:06:58 -04:00
|
|
|
Operation::Close => {
|
2024-10-12 03:06:44 -04:00
|
|
|
let from_register = instruction.b();
|
|
|
|
let to_register = instruction.c();
|
|
|
|
|
|
|
|
if self.stack.len() < to_register as usize {
|
|
|
|
return Err(VmError::StackUnderflow { position });
|
|
|
|
}
|
2024-09-17 23:06:58 -04:00
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
for register_index in from_register..to_register {
|
|
|
|
self.stack[register_index as usize] = Register::Empty;
|
2024-09-17 23:06:58 -04:00
|
|
|
}
|
|
|
|
}
|
2024-09-18 13:42:32 -04:00
|
|
|
Operation::LoadBoolean => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let to_register = instruction.a();
|
|
|
|
let boolean = instruction.b_as_boolean();
|
2024-10-30 14:48:30 -04:00
|
|
|
let jump = instruction.c_as_boolean();
|
2024-09-18 13:42:32 -04:00
|
|
|
let value = Value::boolean(boolean);
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(to_register, value, position)?;
|
2024-09-18 13:42:32 -04:00
|
|
|
|
2024-10-30 14:48:30 -04:00
|
|
|
if jump {
|
2024-11-01 09:55:15 -04:00
|
|
|
self.ip += 1;
|
2024-09-18 13:42:32 -04:00
|
|
|
}
|
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
Operation::LoadConstant => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let to_register = instruction.a();
|
|
|
|
let from_constant = instruction.b();
|
2024-09-25 09:55:10 -04:00
|
|
|
let jump = instruction.c_as_boolean();
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
self.set_constant(to_register, from_constant, position)?;
|
2024-09-25 09:55:10 -04:00
|
|
|
|
|
|
|
if jump {
|
2024-11-01 09:55:15 -04:00
|
|
|
self.ip += 1
|
2024-09-25 09:55:10 -04:00
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
2024-09-17 19:35:33 -04:00
|
|
|
Operation::LoadList => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let to_register = instruction.a();
|
2024-11-06 03:36:46 -05:00
|
|
|
let start_register = instruction.b();
|
|
|
|
let item_type = (start_register..to_register)
|
|
|
|
.find_map(|register_index| {
|
|
|
|
if let Ok(value) = self.get_register(register_index, position) {
|
|
|
|
Some(value.r#type())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.unwrap_or(Type::Any);
|
|
|
|
let value = Value::abstract_list(start_register, to_register, item_type);
|
2024-09-17 19:35:33 -04:00
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(to_register, value, position)?;
|
2024-09-17 19:35:33 -04:00
|
|
|
}
|
2024-10-20 02:30:22 -04:00
|
|
|
Operation::LoadSelf => {
|
|
|
|
let to_register = instruction.a();
|
|
|
|
let value = Value::function(
|
|
|
|
self.chunk.clone(),
|
|
|
|
FunctionType {
|
|
|
|
type_parameters: None,
|
|
|
|
value_parameters: None,
|
|
|
|
return_type: None,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(to_register, value, position)?;
|
2024-10-20 02:30:22 -04:00
|
|
|
}
|
2024-09-18 11:27:41 -04:00
|
|
|
Operation::DefineLocal => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let from_register = instruction.a();
|
|
|
|
let to_local = instruction.b();
|
2024-09-12 05:08:55 -04:00
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.define_local(to_local, from_register, position)?;
|
2024-09-12 05:08:55 -04:00
|
|
|
}
|
2024-09-12 13:03:24 -04:00
|
|
|
Operation::GetLocal => {
|
2024-10-12 03:06:44 -04:00
|
|
|
let to_register = instruction.a();
|
2024-09-24 16:49:17 -04:00
|
|
|
let local_index = instruction.b();
|
2024-11-04 15:38:58 -05:00
|
|
|
let local = self.get_local(local_index, position)?;
|
2024-09-12 14:16:26 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
self.set_pointer(to_register, local.register_index, position)?;
|
2024-09-12 05:08:55 -04:00
|
|
|
}
|
2024-09-14 21:05:03 -04:00
|
|
|
Operation::SetLocal => {
|
2024-10-12 03:06:44 -04:00
|
|
|
let register = instruction.a();
|
2024-09-24 16:49:17 -04:00
|
|
|
let local_index = instruction.b();
|
2024-09-24 22:58:14 -04:00
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.define_local(local_index, register, position)?;
|
2024-09-14 21:05:03 -04:00
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
Operation::Add => {
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
let sum = left
|
2024-09-23 08:57:49 -04:00
|
|
|
.add(right)
|
2024-09-12 00:39:31 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), sum, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
Operation::Subtract => {
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
let difference = left
|
2024-09-23 08:57:49 -04:00
|
|
|
.subtract(right)
|
2024-09-12 00:39:31 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), difference, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
Operation::Multiply => {
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
let product = left
|
2024-09-23 08:57:49 -04:00
|
|
|
.multiply(right)
|
2024-09-12 00:39:31 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), product, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
Operation::Divide => {
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
let quotient = left
|
2024-09-23 08:57:49 -04:00
|
|
|
.divide(right)
|
2024-09-12 00:39:31 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), quotient, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
2024-09-17 17:51:39 -04:00
|
|
|
Operation::Modulo => {
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-09-17 17:51:39 -04:00
|
|
|
let remainder = left
|
2024-09-23 08:57:49 -04:00
|
|
|
.modulo(right)
|
2024-09-17 17:51:39 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), remainder, position)?;
|
2024-09-17 17:51:39 -04:00
|
|
|
}
|
2024-09-19 11:41:18 -04:00
|
|
|
Operation::Test => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let register = instruction.a();
|
|
|
|
let test_value = instruction.c_as_boolean();
|
2024-11-04 15:38:58 -05:00
|
|
|
let value = self.get_register(register, position)?;
|
2024-11-06 03:36:46 -05:00
|
|
|
let boolean = if let Value::Concrete(ConcreteValue::Boolean(boolean)) = value {
|
2024-10-12 04:50:30 -04:00
|
|
|
*boolean
|
|
|
|
} else {
|
|
|
|
return Err(VmError::ExpectedBoolean {
|
|
|
|
found: value.clone(),
|
|
|
|
position,
|
|
|
|
});
|
|
|
|
};
|
2024-09-19 11:41:18 -04:00
|
|
|
|
2024-09-24 16:49:17 -04:00
|
|
|
if boolean != test_value {
|
2024-09-19 11:41:18 -04:00
|
|
|
self.ip += 1;
|
|
|
|
}
|
2024-09-17 17:51:39 -04:00
|
|
|
}
|
2024-10-19 03:06:14 -04:00
|
|
|
Operation::TestSet => todo!(),
|
2024-09-18 16:43:34 -04:00
|
|
|
Operation::Equal => {
|
2024-10-12 03:06:44 -04:00
|
|
|
debug_assert_eq!(
|
2024-11-04 15:38:58 -05:00
|
|
|
self.get_instruction(self.ip, position)?.0.operation(),
|
2024-10-12 03:06:44 -04:00
|
|
|
Operation::Jump
|
|
|
|
);
|
2024-09-24 10:16:19 -04:00
|
|
|
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-10-12 04:50:30 -04:00
|
|
|
let equal_result = left
|
2024-09-23 08:57:49 -04:00
|
|
|
.equal(right)
|
2024-10-12 04:50:30 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
let boolean =
|
2024-11-06 03:36:46 -05:00
|
|
|
if let Value::Concrete(ConcreteValue::Boolean(boolean)) = equal_result {
|
2024-10-12 04:50:30 -04:00
|
|
|
boolean
|
|
|
|
} else {
|
|
|
|
return Err(VmError::ExpectedBoolean {
|
|
|
|
found: equal_result.clone(),
|
|
|
|
position,
|
|
|
|
});
|
|
|
|
};
|
2024-09-24 16:49:17 -04:00
|
|
|
let compare_to = instruction.a_as_boolean();
|
2024-09-18 16:43:34 -04:00
|
|
|
|
2024-09-24 10:16:19 -04:00
|
|
|
if boolean == compare_to {
|
|
|
|
self.ip += 1;
|
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
let (jump, _) = self.get_instruction(self.ip, position)?;
|
2024-09-24 16:49:17 -04:00
|
|
|
let jump_distance = jump.a();
|
2024-10-05 21:30:48 -04:00
|
|
|
let is_positive = jump.b_as_boolean();
|
2024-09-24 10:16:19 -04:00
|
|
|
let new_ip = if is_positive {
|
|
|
|
self.ip + jump_distance as usize
|
|
|
|
} else {
|
|
|
|
self.ip - jump_distance as usize
|
|
|
|
};
|
|
|
|
|
2024-11-01 09:55:15 -04:00
|
|
|
self.ip = new_ip;
|
2024-09-18 16:43:34 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Operation::Less => {
|
2024-10-12 03:06:44 -04:00
|
|
|
debug_assert_eq!(
|
2024-11-04 15:38:58 -05:00
|
|
|
self.get_instruction(self.ip, position)?.0.operation(),
|
2024-10-12 03:06:44 -04:00
|
|
|
Operation::Jump
|
|
|
|
);
|
2024-09-24 10:16:19 -04:00
|
|
|
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-10-12 04:50:30 -04:00
|
|
|
let less_result = left
|
2024-09-24 11:40:12 -04:00
|
|
|
.less_than(right)
|
2024-10-12 04:50:30 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
2024-11-06 03:36:46 -05:00
|
|
|
let boolean =
|
|
|
|
if let Value::Concrete(ConcreteValue::Boolean(boolean)) = less_result {
|
|
|
|
boolean
|
|
|
|
} else {
|
|
|
|
return Err(VmError::ExpectedBoolean {
|
|
|
|
found: less_result.clone(),
|
|
|
|
position,
|
|
|
|
});
|
|
|
|
};
|
2024-09-24 16:49:17 -04:00
|
|
|
let compare_to = instruction.a_as_boolean();
|
2024-09-18 16:43:34 -04:00
|
|
|
|
2024-09-24 10:16:19 -04:00
|
|
|
if boolean == compare_to {
|
|
|
|
self.ip += 1;
|
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
let jump = self.get_instruction(self.ip, position)?.0;
|
2024-09-24 16:49:17 -04:00
|
|
|
let jump_distance = jump.a();
|
2024-10-05 21:30:48 -04:00
|
|
|
let is_positive = jump.b_as_boolean();
|
2024-09-24 10:16:19 -04:00
|
|
|
let new_ip = if is_positive {
|
|
|
|
self.ip + jump_distance as usize
|
|
|
|
} else {
|
|
|
|
self.ip - jump_distance as usize
|
|
|
|
};
|
|
|
|
|
2024-11-01 09:55:15 -04:00
|
|
|
self.ip = new_ip;
|
2024-09-18 16:43:34 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Operation::LessEqual => {
|
2024-10-12 03:06:44 -04:00
|
|
|
debug_assert_eq!(
|
2024-11-04 15:38:58 -05:00
|
|
|
self.get_instruction(self.ip, position)?.0.operation(),
|
2024-10-12 03:06:44 -04:00
|
|
|
Operation::Jump
|
|
|
|
);
|
2024-09-24 11:40:12 -04:00
|
|
|
|
2024-10-18 23:34:48 -04:00
|
|
|
let (left, right) = get_arguments(&mut self, instruction, position)?;
|
2024-10-12 04:50:30 -04:00
|
|
|
let less_or_equal_result = left
|
2024-09-24 11:40:12 -04:00
|
|
|
.less_than_or_equal(right)
|
2024-10-12 04:50:30 -04:00
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
2024-11-06 03:36:46 -05:00
|
|
|
let boolean = if let Value::Concrete(ConcreteValue::Boolean(boolean)) =
|
2024-10-12 04:50:30 -04:00
|
|
|
less_or_equal_result
|
|
|
|
{
|
|
|
|
boolean
|
|
|
|
} else {
|
|
|
|
return Err(VmError::ExpectedBoolean {
|
|
|
|
found: less_or_equal_result.clone(),
|
2024-09-24 10:16:19 -04:00
|
|
|
position,
|
2024-10-12 04:50:30 -04:00
|
|
|
});
|
|
|
|
};
|
2024-09-24 16:49:17 -04:00
|
|
|
let compare_to = instruction.a_as_boolean();
|
2024-09-18 16:43:34 -04:00
|
|
|
|
2024-09-24 10:16:19 -04:00
|
|
|
if boolean == compare_to {
|
|
|
|
self.ip += 1;
|
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
let jump = self.get_instruction(self.ip, position)?.0;
|
2024-09-24 16:49:17 -04:00
|
|
|
let jump_distance = jump.a();
|
2024-10-05 21:30:48 -04:00
|
|
|
let is_positive = jump.b_as_boolean();
|
2024-09-24 10:16:19 -04:00
|
|
|
let new_ip = if is_positive {
|
|
|
|
self.ip + jump_distance as usize
|
|
|
|
} else {
|
|
|
|
self.ip - jump_distance as usize
|
|
|
|
};
|
|
|
|
|
2024-11-01 09:55:15 -04:00
|
|
|
self.ip = new_ip;
|
2024-09-18 16:43:34 -04:00
|
|
|
}
|
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
Operation::Negate => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let value = if instruction.b_is_constant() {
|
2024-11-06 13:50:49 -05:00
|
|
|
self.get_constant(instruction.b(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
self.get_register(instruction.b(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
};
|
2024-09-12 00:39:31 -04:00
|
|
|
let negated = value
|
|
|
|
.negate()
|
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), negated, position)?;
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
2024-09-17 17:51:39 -04:00
|
|
|
Operation::Not => {
|
2024-09-24 16:49:17 -04:00
|
|
|
let value = if instruction.b_is_constant() {
|
2024-11-06 13:50:49 -05:00
|
|
|
self.get_constant(instruction.b(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
} else {
|
2024-11-04 15:38:58 -05:00
|
|
|
self.get_register(instruction.b(), position)?
|
2024-09-17 21:10:44 -04:00
|
|
|
};
|
2024-09-18 07:58:31 -04:00
|
|
|
let not = value
|
2024-09-17 17:51:39 -04:00
|
|
|
.not()
|
|
|
|
.map_err(|error| VmError::Value { error, position })?;
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(instruction.a(), not, position)?;
|
2024-09-17 17:51:39 -04:00
|
|
|
}
|
2024-09-18 16:43:34 -04:00
|
|
|
Operation::Jump => {
|
2024-10-30 14:48:30 -04:00
|
|
|
let jump_distance = instruction.b();
|
|
|
|
let is_positive = instruction.c_as_boolean();
|
|
|
|
let new_ip = if is_positive {
|
|
|
|
self.ip + jump_distance as usize
|
|
|
|
} else {
|
2024-10-31 20:33:46 -04:00
|
|
|
self.ip - jump_distance as usize - 1
|
2024-10-30 14:48:30 -04:00
|
|
|
};
|
2024-11-01 09:55:15 -04:00
|
|
|
self.ip = new_ip;
|
2024-09-18 16:43:34 -04:00
|
|
|
}
|
2024-10-12 10:55:34 -04:00
|
|
|
Operation::Call => {
|
2024-10-13 07:14:12 -04:00
|
|
|
let to_register = instruction.a();
|
2024-10-19 17:24:22 -04:00
|
|
|
let function_register = instruction.b();
|
|
|
|
let argument_count = instruction.c();
|
2024-11-04 15:38:58 -05:00
|
|
|
let value = self.get_register(function_register, position)?.clone();
|
2024-11-06 03:36:46 -05:00
|
|
|
let function = if let Value::Concrete(ConcreteValue::Function(function)) = value
|
|
|
|
{
|
2024-10-12 10:55:34 -04:00
|
|
|
function
|
|
|
|
} else {
|
2024-10-13 02:33:58 -04:00
|
|
|
return Err(VmError::ExpectedFunction {
|
|
|
|
found: value,
|
|
|
|
position,
|
|
|
|
});
|
2024-10-12 10:55:34 -04:00
|
|
|
};
|
2024-10-12 20:19:21 -04:00
|
|
|
let mut function_vm = Vm::new(function.take_chunk());
|
2024-10-19 17:24:22 -04:00
|
|
|
let first_argument_index = function_register + 1;
|
2024-10-12 10:55:34 -04:00
|
|
|
|
2024-10-19 17:24:22 -04:00
|
|
|
for argument_index in
|
|
|
|
first_argument_index..first_argument_index + argument_count
|
|
|
|
{
|
2024-11-04 15:38:58 -05:00
|
|
|
let argument = match self.get_register(argument_index, position) {
|
2024-10-20 00:06:22 -04:00
|
|
|
Ok(value) => value.clone(),
|
|
|
|
Err(VmError::EmptyRegister { .. }) => continue,
|
|
|
|
Err(error) => return Err(error),
|
|
|
|
};
|
2024-10-13 07:14:12 -04:00
|
|
|
let top_of_stack = function_vm.stack.len() as u8;
|
2024-10-12 10:55:34 -04:00
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
function_vm.set_register(top_of_stack, argument, position)?;
|
2024-10-12 10:55:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
let return_value = function_vm.run()?;
|
|
|
|
|
|
|
|
if let Some(value) = return_value {
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(to_register, value, position)?;
|
2024-10-12 10:55:34 -04:00
|
|
|
}
|
|
|
|
}
|
2024-10-30 03:08:25 -04:00
|
|
|
Operation::CallNative => {
|
|
|
|
let native_function = NativeFunction::from(instruction.b());
|
2024-10-30 08:02:22 -04:00
|
|
|
let return_value = native_function.call(instruction, &self, position)?;
|
2024-10-30 03:08:25 -04:00
|
|
|
|
|
|
|
if let Some(value) = return_value {
|
2024-10-30 08:02:22 -04:00
|
|
|
let to_register = instruction.a();
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
self.set_register(to_register, value, position)?;
|
2024-10-30 03:08:25 -04:00
|
|
|
}
|
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
Operation::Return => {
|
2024-10-12 03:06:44 -04:00
|
|
|
let should_return_value = instruction.b_as_boolean();
|
|
|
|
|
2024-10-29 23:11:55 -04:00
|
|
|
if !should_return_value {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(register) = self.last_assigned_register {
|
2024-11-06 03:36:46 -05:00
|
|
|
let value = self
|
|
|
|
.empty_register(register, position)?
|
|
|
|
.to_concrete(&mut self, position)?;
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-29 23:11:55 -04:00
|
|
|
return Ok(Some(value));
|
2024-10-04 22:33:47 -04:00
|
|
|
} else {
|
2024-10-29 23:11:55 -04:00
|
|
|
return Err(VmError::StackUnderflow { position });
|
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-04 22:21:17 -04:00
|
|
|
Ok(None)
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
fn set_register(
|
|
|
|
&mut self,
|
|
|
|
to_register: u8,
|
|
|
|
value: Value,
|
|
|
|
position: Span,
|
|
|
|
) -> Result<(), VmError> {
|
2024-10-29 23:11:55 -04:00
|
|
|
self.last_assigned_register = Some(to_register);
|
2024-11-06 15:08:51 -05:00
|
|
|
|
|
|
|
let length = self.stack.len();
|
2024-10-13 04:21:07 -04:00
|
|
|
let to_register = to_register as usize;
|
2024-09-14 21:05:03 -04:00
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
if length == Self::STACK_LIMIT {
|
|
|
|
return Err(VmError::StackOverflow { position });
|
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
match to_register.cmp(&length) {
|
|
|
|
Ordering::Less => {
|
2024-10-19 03:06:14 -04:00
|
|
|
log::trace!("Change R{to_register} to {value}");
|
2024-09-12 09:11:49 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
self.stack[to_register] = Register::Value(value);
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
Ordering::Equal => {
|
2024-10-19 03:06:14 -04:00
|
|
|
log::trace!("Set R{to_register} to {value}");
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
self.stack.push(Register::Value(value));
|
2024-09-19 11:41:18 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
Ok(())
|
|
|
|
}
|
2024-10-20 22:18:58 -04:00
|
|
|
Ordering::Greater => {
|
|
|
|
let difference = to_register - length;
|
|
|
|
|
|
|
|
for index in 0..difference {
|
|
|
|
log::trace!("Set R{index} to empty");
|
|
|
|
|
|
|
|
self.stack.push(Register::Empty);
|
|
|
|
}
|
|
|
|
|
|
|
|
log::trace!("Set R{to_register} to {value}");
|
|
|
|
|
|
|
|
self.stack.push(Register::Value(value));
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-10-13 04:21:07 -04:00
|
|
|
}
|
|
|
|
}
|
2024-09-19 11:41:18 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
fn set_pointer(
|
|
|
|
&mut self,
|
|
|
|
to_register: u8,
|
|
|
|
from_register: u8,
|
|
|
|
position: Span,
|
|
|
|
) -> Result<(), VmError> {
|
2024-10-29 23:11:55 -04:00
|
|
|
self.last_assigned_register = Some(to_register);
|
2024-11-06 15:08:51 -05:00
|
|
|
|
|
|
|
let length = self.stack.len();
|
2024-10-13 04:21:07 -04:00
|
|
|
let to_register = to_register as usize;
|
|
|
|
|
|
|
|
if length == Self::STACK_LIMIT {
|
|
|
|
return Err(VmError::StackOverflow { position });
|
2024-09-19 11:41:18 -04:00
|
|
|
}
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-13 04:21:07 -04:00
|
|
|
match to_register.cmp(&length) {
|
|
|
|
Ordering::Less => {
|
2024-10-19 03:06:14 -04:00
|
|
|
log::trace!("Change R{to_register} to R{from_register}");
|
2024-10-13 04:21:07 -04:00
|
|
|
|
|
|
|
self.stack[to_register] = Register::Pointer(from_register);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
Ordering::Equal => {
|
2024-10-19 03:06:14 -04:00
|
|
|
log::trace!("Set R{to_register} to R{from_register}");
|
2024-10-13 04:21:07 -04:00
|
|
|
|
|
|
|
self.stack.push(Register::Pointer(from_register));
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-10-20 22:18:58 -04:00
|
|
|
Ordering::Greater => {
|
|
|
|
let difference = to_register - length;
|
|
|
|
|
|
|
|
for index in 0..difference {
|
|
|
|
log::trace!("Set R{index} to empty");
|
|
|
|
|
|
|
|
self.stack.push(Register::Empty);
|
|
|
|
}
|
|
|
|
|
|
|
|
log::trace!("Set R{to_register} to R{from_register}");
|
|
|
|
|
|
|
|
self.stack.push(Register::Pointer(from_register));
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-10-13 04:21:07 -04:00
|
|
|
}
|
2024-09-19 11:41:18 -04:00
|
|
|
}
|
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
fn set_constant(
|
|
|
|
&mut self,
|
2024-10-20 22:18:58 -04:00
|
|
|
to_register: u8,
|
2024-10-12 03:06:44 -04:00
|
|
|
constant_index: u8,
|
|
|
|
position: Span,
|
|
|
|
) -> Result<(), VmError> {
|
2024-10-29 23:11:55 -04:00
|
|
|
self.last_assigned_register = Some(to_register);
|
2024-11-06 15:08:51 -05:00
|
|
|
|
|
|
|
let length = self.stack.len();
|
2024-10-20 22:18:58 -04:00
|
|
|
let to_register = to_register as usize;
|
2024-09-23 08:57:49 -04:00
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
if length == Self::STACK_LIMIT {
|
|
|
|
return Err(VmError::StackOverflow { position });
|
|
|
|
}
|
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
match to_register.cmp(&length) {
|
|
|
|
Ordering::Less => {
|
|
|
|
log::trace!("Change R{to_register} to C{constant_index}");
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
self.stack[to_register] = Register::Constant(constant_index);
|
2024-09-23 08:57:49 -04:00
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
Ordering::Equal => {
|
|
|
|
log::trace!("Set R{to_register} to C{constant_index}");
|
|
|
|
|
|
|
|
self.stack.push(Register::Constant(constant_index));
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
Ordering::Greater => {
|
|
|
|
let difference = to_register - length;
|
|
|
|
|
|
|
|
for index in 0..difference {
|
|
|
|
log::trace!("Set R{index} to empty");
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
self.stack.push(Register::Empty);
|
|
|
|
}
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
log::trace!("Set R{to_register} to C{constant_index}");
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
self.stack.push(Register::Constant(constant_index));
|
2024-10-12 03:06:44 -04:00
|
|
|
|
2024-10-20 22:18:58 -04:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
2024-09-23 08:57:49 -04:00
|
|
|
}
|
|
|
|
|
2024-11-06 13:50:49 -05:00
|
|
|
fn get_constant(&self, index: u8, position: Span) -> Result<&Value, VmError> {
|
|
|
|
self.chunk
|
|
|
|
.get_constant(index)
|
|
|
|
.map_err(|error| VmError::Chunk { error, position })
|
|
|
|
}
|
|
|
|
|
2024-11-04 15:38:58 -05:00
|
|
|
pub fn get_register(&self, index: u8, position: Span) -> Result<&Value, VmError> {
|
2024-09-14 21:05:03 -04:00
|
|
|
let index = index as usize;
|
2024-10-12 03:06:44 -04:00
|
|
|
let register = self
|
|
|
|
.stack
|
|
|
|
.get(index)
|
2024-10-13 04:21:07 -04:00
|
|
|
.ok_or_else(|| VmError::RegisterIndexOutOfBounds { index, position })?;
|
2024-09-14 21:05:03 -04:00
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
match register {
|
|
|
|
Register::Value(value) => Ok(value),
|
2024-11-04 15:38:58 -05:00
|
|
|
Register::Pointer(register_index) => self.get_register(*register_index, position),
|
2024-11-06 13:50:49 -05:00
|
|
|
Register::Constant(constant_index) => self.get_constant(*constant_index, position),
|
2024-10-12 03:06:44 -04:00
|
|
|
Register::Empty => Err(VmError::EmptyRegister { index, position }),
|
2024-09-12 21:14:15 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-06 03:36:46 -05:00
|
|
|
pub fn empty_register(&mut self, index: u8, position: Span) -> Result<Value, VmError> {
|
2024-10-12 10:55:34 -04:00
|
|
|
let index = index as usize;
|
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
if index >= self.stack.len() {
|
2024-10-13 04:21:07 -04:00
|
|
|
return Err(VmError::RegisterIndexOutOfBounds { index, position });
|
2024-09-18 07:58:31 -04:00
|
|
|
}
|
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
let register = replace(&mut self.stack[index], Register::Empty);
|
2024-11-06 15:08:51 -05:00
|
|
|
let value = match register {
|
|
|
|
Register::Value(value) => value,
|
|
|
|
Register::Pointer(register_index) => self.empty_register(register_index, position)?,
|
2024-10-12 03:06:44 -04:00
|
|
|
Register::Constant(constant_index) => {
|
2024-11-06 15:08:51 -05:00
|
|
|
let constant_index = constant_index as usize;
|
2024-11-06 03:36:46 -05:00
|
|
|
|
2024-11-06 15:08:51 -05:00
|
|
|
if constant_index >= self.chunk.constants().len() {
|
|
|
|
return Err(VmError::Chunk {
|
|
|
|
error: ChunkError::ConstantIndexOutOfBounds {
|
|
|
|
index: constant_index,
|
|
|
|
},
|
|
|
|
position,
|
|
|
|
});
|
|
|
|
}
|
2024-11-06 03:36:46 -05:00
|
|
|
|
2024-11-06 15:08:51 -05:00
|
|
|
let constant = &mut self.chunk.constants_mut()[constant_index];
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-11-06 15:08:51 -05:00
|
|
|
replace(constant, Value::integer(0))
|
2024-10-12 03:06:44 -04:00
|
|
|
}
|
2024-11-06 15:08:51 -05:00
|
|
|
Register::Empty => return Err(VmError::EmptyRegister { index, position }),
|
|
|
|
};
|
|
|
|
|
|
|
|
self.chunk.is_poisoned = true;
|
|
|
|
|
|
|
|
Ok(value)
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fn read(&mut self, position: Span) -> Result<&(Instruction, Span), VmError> {
|
2024-11-06 13:50:49 -05:00
|
|
|
self.chunk
|
|
|
|
.expect_not_poisoned()
|
|
|
|
.map_err(|error| VmError::Chunk { error, position })?;
|
2024-11-01 09:55:15 -04:00
|
|
|
|
2024-11-06 13:50:49 -05:00
|
|
|
let max_ip = self.chunk.len() - 1;
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-11-06 13:50:49 -05:00
|
|
|
if self.ip > max_ip {
|
|
|
|
return self.get_instruction(max_ip, position);
|
|
|
|
} else {
|
|
|
|
self.ip += 1;
|
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
|
2024-11-06 13:50:49 -05:00
|
|
|
self.get_instruction(self.ip - 1, position)
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
2024-11-04 15:38:58 -05:00
|
|
|
|
|
|
|
fn define_local(
|
|
|
|
&mut self,
|
|
|
|
local_index: u8,
|
|
|
|
register_index: u8,
|
|
|
|
position: Span,
|
|
|
|
) -> Result<(), VmError> {
|
|
|
|
let local = self
|
|
|
|
.chunk
|
2024-11-06 13:50:49 -05:00
|
|
|
.get_local_mut(local_index)
|
|
|
|
.map_err(|error| VmError::Chunk { error, position })?;
|
2024-11-04 15:38:58 -05:00
|
|
|
|
|
|
|
log::debug!("Define local L{}", local_index);
|
|
|
|
|
|
|
|
local.register_index = register_index;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_local(&self, local_index: u8, position: Span) -> Result<&Local, VmError> {
|
|
|
|
self.chunk
|
2024-11-06 13:50:49 -05:00
|
|
|
.get_local(local_index)
|
|
|
|
.map_err(|error| VmError::Chunk { error, position })
|
2024-11-04 15:38:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn get_instruction(
|
|
|
|
&self,
|
|
|
|
index: usize,
|
|
|
|
position: Span,
|
|
|
|
) -> Result<&(Instruction, Span), VmError> {
|
|
|
|
self.chunk
|
2024-11-06 13:50:49 -05:00
|
|
|
.get_instruction(index)
|
|
|
|
.map_err(|error| VmError::Chunk { error, position })
|
2024-11-04 15:38:58 -05:00
|
|
|
}
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
|
2024-10-12 03:06:44 -04:00
|
|
|
#[derive(Debug, Eq, PartialEq)]
|
|
|
|
enum Register {
|
|
|
|
Empty,
|
|
|
|
Value(Value),
|
|
|
|
Pointer(u8),
|
|
|
|
Constant(u8),
|
|
|
|
}
|
|
|
|
|
2024-11-06 13:50:49 -05:00
|
|
|
#[derive(Clone, Debug, PartialEq)]
|
2024-09-12 00:39:31 -04:00
|
|
|
pub enum VmError {
|
2024-11-06 03:36:46 -05:00
|
|
|
// Stack errors
|
|
|
|
StackOverflow { position: Span },
|
|
|
|
StackUnderflow { position: Span },
|
|
|
|
|
|
|
|
// Register errors
|
|
|
|
EmptyRegister { index: usize, position: Span },
|
|
|
|
RegisterIndexOutOfBounds { index: usize, position: Span },
|
|
|
|
|
|
|
|
// Execution errors
|
|
|
|
ExpectedBoolean { found: Value, position: Span },
|
|
|
|
ExpectedFunction { found: Value, position: Span },
|
2024-09-12 00:39:31 -04:00
|
|
|
|
|
|
|
// Wrappers for foreign errors
|
2024-11-06 13:50:49 -05:00
|
|
|
Chunk { error: ChunkError, position: Span },
|
2024-10-30 09:32:46 -04:00
|
|
|
NativeFunction(NativeFunctionError),
|
2024-11-06 03:36:46 -05:00
|
|
|
Value { error: ValueError, position: Span },
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl AnnotatedError for VmError {
|
|
|
|
fn title() -> &'static str {
|
|
|
|
"Runtime Error"
|
|
|
|
}
|
|
|
|
|
|
|
|
fn description(&self) -> &'static str {
|
|
|
|
match self {
|
2024-11-06 13:50:49 -05:00
|
|
|
Self::Chunk { .. } => "Chunk error",
|
2024-09-12 00:39:31 -04:00
|
|
|
Self::EmptyRegister { .. } => "Empty register",
|
2024-09-19 11:41:18 -04:00
|
|
|
Self::ExpectedBoolean { .. } => "Expected boolean",
|
2024-10-13 02:33:58 -04:00
|
|
|
Self::ExpectedFunction { .. } => "Expected function",
|
2024-11-06 03:36:46 -05:00
|
|
|
Self::NativeFunction(error) => error.description(),
|
|
|
|
Self::RegisterIndexOutOfBounds { .. } => "Register index out of bounds",
|
2024-09-12 00:39:31 -04:00
|
|
|
Self::StackOverflow { .. } => "Stack overflow",
|
|
|
|
Self::StackUnderflow { .. } => "Stack underflow",
|
2024-10-12 03:06:44 -04:00
|
|
|
Self::Value { .. } => "Value error",
|
2024-09-12 00:39:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn details(&self) -> Option<String> {
|
|
|
|
match self {
|
2024-11-06 13:50:49 -05:00
|
|
|
Self::Chunk { error, .. } => Some(error.to_string()),
|
2024-11-06 03:36:46 -05:00
|
|
|
Self::EmptyRegister { index, .. } => Some(format!("Register R{index} is empty")),
|
2024-10-13 02:33:58 -04:00
|
|
|
Self::ExpectedFunction { found, .. } => Some(format!("{found} is not a function")),
|
2024-10-13 04:21:07 -04:00
|
|
|
Self::RegisterIndexOutOfBounds { index, .. } => {
|
2024-10-13 16:46:45 -04:00
|
|
|
Some(format!("Register {index} does not exist"))
|
2024-10-13 04:21:07 -04:00
|
|
|
}
|
2024-10-30 09:32:46 -04:00
|
|
|
Self::NativeFunction(error) => error.details(),
|
2024-09-12 00:39:31 -04:00
|
|
|
Self::Value { error, .. } => Some(error.to_string()),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn position(&self) -> Span {
|
|
|
|
match self {
|
2024-11-06 13:50:49 -05:00
|
|
|
Self::Chunk { position, .. } => *position,
|
2024-09-12 00:39:31 -04:00
|
|
|
Self::EmptyRegister { position, .. } => *position,
|
2024-09-19 11:41:18 -04:00
|
|
|
Self::ExpectedBoolean { position, .. } => *position,
|
2024-10-13 02:33:58 -04:00
|
|
|
Self::ExpectedFunction { position, .. } => *position,
|
2024-10-30 09:32:46 -04:00
|
|
|
Self::NativeFunction(error) => error.position(),
|
2024-11-06 03:36:46 -05:00
|
|
|
Self::RegisterIndexOutOfBounds { position, .. } => *position,
|
|
|
|
Self::StackOverflow { position } => *position,
|
|
|
|
Self::StackUnderflow { position } => *position,
|
2024-09-12 00:39:31 -04:00
|
|
|
Self::Value { position, .. } => *position,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|