🔊 improve error message for this error with the missing math lib

main
_ 2023-09-30 23:41:12 -05:00
parent bccd5fc3a7
commit 06638574f7
5 changed files with 84 additions and 29 deletions

View File

@ -1,4 +1,4 @@
#[derive (Debug, PartialEq)] #[derive (Clone, Copy, Debug, PartialEq)]
pub enum Instruction { pub enum Instruction {
Add (u8, u8, u8), Add (u8, u8, u8),
AddI (u8, u8, i8), AddI (u8, u8, i8),

View File

@ -221,8 +221,6 @@ pub fn parse_block <R: Read> (rdr: &mut R, blocks: &mut Vec <Block>)
{ {
// Ignore things I haven't implemented yet // Ignore things I haven't implemented yet
use crate::value::Value;
parse_string (rdr)?; // function name parse_string (rdr)?; // function name
parse_int (rdr).unwrap (); // start line in source code parse_int (rdr).unwrap (); // start line in source code
parse_int (rdr).unwrap (); // last line in source code parse_int (rdr).unwrap (); // last line in source code

View File

@ -8,7 +8,7 @@ mod value;
#[cfg (test)] #[cfg (test)]
mod tests; mod tests;
fn main () { fn main () -> Result <(), state::StepError> {
use state::State; use state::State;
let mut list_bytecode = false; let mut list_bytecode = false;
@ -86,17 +86,17 @@ fn main () {
match input.as_str ().trim_end () { match input.as_str ().trim_end () {
"c" => in_break = false, "c" => in_break = false,
"q" => return, "q" => return Ok (()),
"registers" => { "registers" => {
dbg! (&vm.registers); dbg! (&vm.registers);
continue; continue;
} }
"s" => { "s" => {
match vm.step () { match vm.step ()? {
None => (), None => (),
Some (state::StepOutput::ChunkReturned (x)) => { Some (state::StepOutput::ChunkReturned (x)) => {
dbg! (x); dbg! (x);
return; return Ok (());
}, },
} }
continue; continue;
@ -105,11 +105,11 @@ fn main () {
} }
} }
match vm.step () { match vm.step ()? {
None => (), None => (),
Some (state::StepOutput::ChunkReturned (x)) => { Some (state::StepOutput::ChunkReturned (x)) => {
dbg! (x); dbg! (x);
return; return Ok (());
}, },
} }
} }

View File

@ -79,6 +79,13 @@ pub enum StepOutput {
ChunkReturned (Vec <Value>), ChunkReturned (Vec <Value>),
} }
#[derive (Debug)]
pub struct StepError {
frame: StackFrame,
inst: Instruction,
msg: &'static str,
}
impl <'a> State <'a> { impl <'a> State <'a> {
pub fn new (chunk: &'a Chunk, upvalues: &'a [Value]) -> Self { pub fn new (chunk: &'a Chunk, upvalues: &'a [Value]) -> Self {
Self { Self {
@ -140,7 +147,17 @@ impl <'a> State <'a> {
self.top - self.stack.last ().unwrap ().register_offset self.top - self.stack.last ().unwrap ().register_offset
} }
pub fn step (&mut self) -> Option <StepOutput> { fn make_step_error (&self, msg: &'static str, inst: &Instruction) -> StepError
{
StepError {
frame: self.stack.last ().unwrap ().clone (),
inst: inst.clone (),
msg,
}
}
pub fn step (&mut self) -> Result <Option <StepOutput>, StepError>
{
let chunk = self.chunk; let chunk = self.chunk;
self.step_count += 1; self.step_count += 1;
@ -161,6 +178,10 @@ impl <'a> State <'a> {
// let r = &mut self.registers [frame.register_offset..]; // let r = &mut self.registers [frame.register_offset..];
let k = &block.constants; let k = &block.constants;
let make_step_error = |msg| {
Err (self.make_step_error (msg, instruction))
};
match instruction { match instruction {
Instruction::Add (a, b, c) => { Instruction::Add (a, b, c) => {
let v_b = self.reg (*b); let v_b = self.reg (*b);
@ -230,7 +251,7 @@ impl <'a> State <'a> {
} }
// Skip the PC increment at the bottom of the loop // Skip the PC increment at the bottom of the loop
return None; return Ok (None);
}, },
Value::RsFunc (x) => { Value::RsFunc (x) => {
let current_frame = self.stack.last ().unwrap (); let current_frame = self.stack.last ().unwrap ();
@ -341,9 +362,9 @@ impl <'a> State <'a> {
}, },
Instruction::GetField (a, b, c) => { Instruction::GetField (a, b, c) => {
let t = match self.reg (*b) { let t = match self.reg (*b) {
Value::Nil => panic! ("R[B] must not be nil {}:{}", frame.block_idx, frame.program_counter), Value::Nil => make_step_error ("R[B] must not be nil")?,
Value::Table (t) => t, Value::Table (t) => t,
_ => panic! ("R[B] must be a table"), _ => make_step_error ("R[B] must be a table")?,
}; };
let key = match &k [usize::from (*c)] { let key = match &k [usize::from (*c)] {
@ -573,7 +594,7 @@ impl <'a> State <'a> {
} }
else { else {
// Return from the entire program // Return from the entire program
return Some (StepOutput::ChunkReturned (self.registers [a..(a + b - 1)].to_vec())); return Ok (Some (StepOutput::ChunkReturned (self.registers [a..(a + b - 1)].to_vec())));
} }
}, },
Instruction::Return0 => { Instruction::Return0 => {
@ -697,7 +718,7 @@ impl <'a> State <'a> {
frame.block_idx = closure.idx; frame.block_idx = closure.idx;
// Skip the PC increment // Skip the PC increment
return None; return Ok (None);
}, },
Instruction::Test (a, _k) => { Instruction::Test (a, _k) => {
if self.reg (*a).is_truthy() { if self.reg (*a).is_truthy() {
@ -727,11 +748,11 @@ impl <'a> State <'a> {
frame.program_counter = next_pc; frame.program_counter = next_pc;
} }
None Ok (None)
} }
pub fn execute_chunk (&mut self, breakpoints: &[Breakpoint]) pub fn execute_chunk (&mut self, breakpoints: &[Breakpoint])
-> Vec <Value> { -> Result <Vec <Value>, StepError> {
let max_iters = 2000; let max_iters = 2000;
for _ in 0..max_iters { for _ in 0..max_iters {
@ -739,9 +760,9 @@ impl <'a> State <'a> {
dbg! (&self); dbg! (&self);
} }
match self.step () { match self.step ()? {
None => (), None => (),
Some (StepOutput::ChunkReturned (x)) => return x, Some (StepOutput::ChunkReturned (x)) => return Ok (x),
} }
} }

View File

@ -25,7 +25,7 @@ fn calculate_hash<T: Hash>(t: &T) -> u64 {
fn run_chunk (args: &[&str], chunk: &Chunk) -> Vec <Value> { fn run_chunk (args: &[&str], chunk: &Chunk) -> Vec <Value> {
let upvalues = State::upvalues_from_args (args.into_iter ().map (|s| s.to_string ())); let upvalues = State::upvalues_from_args (args.into_iter ().map (|s| s.to_string ()));
let mut vm = State::new (chunk, &upvalues); let mut vm = State::new (chunk, &upvalues);
vm.execute_chunk (&[]) vm.execute_chunk (&[]).unwrap ()
} }
/// Takes arguments and Lua bytecode, loads it, runs it, /// Takes arguments and Lua bytecode, loads it, runs it,
@ -120,7 +120,7 @@ fn bools () {
let upvalues = State::upvalues_from_args (arg.into_iter ().map (|s| s.to_string ())); let upvalues = State::upvalues_from_args (arg.into_iter ().map (|s| s.to_string ()));
let mut vm = State::new (&chunk, &upvalues); let mut vm = State::new (&chunk, &upvalues);
let actual = vm.execute_chunk (&[]); let actual = vm.execute_chunk (&[]).unwrap ();
assert_eq! (actual, expected); assert_eq! (actual, expected);
} }
} }
@ -176,7 +176,7 @@ fn floats () {
let expected: Vec <Value> = expected; let expected: Vec <Value> = expected;
let upvalues = State::upvalues_from_args (arg.into_iter ().map (|s| s.to_string ())); let upvalues = State::upvalues_from_args (arg.into_iter ().map (|s| s.to_string ()));
let mut vm = State::new (&chunk, &upvalues); let mut vm = State::new (&chunk, &upvalues);
let actual = vm.execute_chunk (&[]); let actual = vm.execute_chunk (&[]).unwrap ();
assert_eq! (actual, expected); assert_eq! (actual, expected);
} }
@ -198,7 +198,7 @@ fn fma () {
let expected: Vec <Value> = expected; let expected: Vec <Value> = expected;
let upvalues = State::upvalues_from_args (arg.into_iter ().map (|s| s.to_string ())); let upvalues = State::upvalues_from_args (arg.into_iter ().map (|s| s.to_string ()));
let mut vm = State::new (&chunk, &upvalues); let mut vm = State::new (&chunk, &upvalues);
let actual = vm.execute_chunk (&[]); let actual = vm.execute_chunk (&[]).unwrap ();
assert_eq! (actual, expected); assert_eq! (actual, expected);
} }
@ -368,14 +368,50 @@ fn value_size () {
// with 64-bit floats // with 64-bit floats
// //
// It would be nice if LunarWaveVM is the same or better. // It would be nice if LunarWaveVM is the same or better.
// There are some exploratory things in this test, too // I'm also just checking a bunch of my assumptions about how
// Rust organizes and sizes different types
use std::mem::size_of; use std::mem::size_of;
assert! (size_of::<Box <()>> () <= 8); {
assert! (size_of::<std::rc::Rc <()>> () <= 8); // Make sure that Rx / Box are both pointers that hide the size
// of big types
assert! (size_of::<Box <()>> () <= 8);
assert! (size_of::<std::rc::Rc <()>> () <= 8);
}
let sz = size_of::<crate::value::Value> (); {
let expected = 16; // Make sure LWVM's Values are 16 bytes or smaller.
assert! (sz <= expected, "{sz} > {expected}"); // Because types are usually aligned to their size, f64s
// are supposed to be aligned to 8 bytes. So even an `Option <f64>`
// uses 8 bytes to say "Some" or "None".
// I could _maybe_ fudge this somehow but it's fine to start with.
let sz = size_of::<crate::value::Value> ();
let expected = 16;
assert! (sz <= expected, "{sz} > {expected}");
}
{
// All these are 8 bytes for the same reason Value is 16 bytes.
// Luckily Rust doesn't seem to stack the 4-byte overhead
// of Result and Option.
let sz = size_of::<(i32, i32)> ();
let expected = 8;
assert! (sz == expected, "{sz} != {expected}");
let sz = size_of::<Option <i32>> ();
let expected = 8;
assert! (sz == expected, "{sz} != {expected}");
let sz = size_of::<Result <i32, i32>> ();
let expected = 8;
assert! (sz == expected, "{sz} != {expected}");
let sz = size_of::<Result <Option <i32>, i32>> ();
let expected = 8;
assert! (sz == expected, "{sz} != {expected}");
}
} }