refactor(tazjin/rlox): Represent VM values as enums
Introduces a new enum which represents the different types of possible values, and modifies the rest of the existing code to wrap/unwrap these enum variants correctly. Notably in the vm module, a new macro has been introduced that makes it possible to encode a type expectation and return a runtime error in case of a type mismatch. Change-Id: I325b5e31e395c62d8819ab2af6d398e1277333c0 Reviewed-on: https://cl.tvl.fyi/c/depot/+/2570 Reviewed-by: tazjin <mail@tazj.in> Tested-by: BuildkiteCI
This commit is contained in:
parent
6b990a7571
commit
127ef98486
6 changed files with 59 additions and 25 deletions
|
@ -49,7 +49,7 @@ impl Chunk {
|
|||
}
|
||||
}
|
||||
|
||||
fn get_line(&self, offset: usize) -> usize {
|
||||
pub fn get_line(&self, offset: usize) -> usize {
|
||||
let mut pos = 0;
|
||||
for span in &self.lines {
|
||||
pos += span.count;
|
||||
|
|
|
@ -128,7 +128,7 @@ impl<T: Iterator<Item = Token>> Compiler<T> {
|
|||
|
||||
fn number(&mut self) -> LoxResult<()> {
|
||||
if let TokenKind::Number(num) = self.previous().kind {
|
||||
self.emit_constant(num);
|
||||
self.emit_constant(Value::Number(num));
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
|
|
|
@ -8,6 +8,7 @@ pub enum ErrorKind {
|
|||
UnterminatedString,
|
||||
ExpectedToken(&'static str),
|
||||
InternalError(&'static str),
|
||||
TypeError(String),
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
|
|
|
@ -2,47 +2,47 @@ use super::*;
|
|||
|
||||
use crate::Lox;
|
||||
|
||||
fn expect(code: &str, value: value::Value) {
|
||||
fn expect_num(code: &str, value: f64) {
|
||||
let result = Interpreter::create()
|
||||
.interpret(code.into())
|
||||
.expect("evaluation failed");
|
||||
assert_eq!(result, value);
|
||||
assert_eq!(result, value::Value::Number(value));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn numbers() {
|
||||
expect("1", 1.0);
|
||||
expect("13.37", 13.37);
|
||||
expect_num("1", 1.0);
|
||||
expect_num("13.37", 13.37);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn negative_numbers() {
|
||||
// Note: This technically tests unary operators.
|
||||
expect("-1", -1.0);
|
||||
expect("-13.37", -13.37);
|
||||
expect_num("-1", -1.0);
|
||||
expect_num("-13.37", -13.37);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn terms() {
|
||||
expect("1 + 2", 3.0);
|
||||
expect("3 - 1", 2.0);
|
||||
expect("0.7 + 0.3", 1.0);
|
||||
expect("1 + -3", -2.0);
|
||||
expect("-1 - -1", 0.0);
|
||||
expect("10 - -10 + 10", 30.0);
|
||||
expect_num("1 + 2", 3.0);
|
||||
expect_num("3 - 1", 2.0);
|
||||
expect_num("0.7 + 0.3", 1.0);
|
||||
expect_num("1 + -3", -2.0);
|
||||
expect_num("-1 - -1", 0.0);
|
||||
expect_num("10 - -10 + 10", 30.0);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn factors() {
|
||||
expect("1 * 2", 2.0);
|
||||
expect("10 / 5", 2.0);
|
||||
expect("0.7 * 4 / 1.4", 2.0);
|
||||
expect("10 * -10 / 10", -10.0);
|
||||
expect_num("1 * 2", 2.0);
|
||||
expect_num("10 / 5", 2.0);
|
||||
expect_num("0.7 * 4 / 1.4", 2.0);
|
||||
expect_num("10 * -10 / 10", -10.0);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn arithmetic() {
|
||||
expect("10 - 3 * 2", 4.0);
|
||||
expect("-4 * -4 + (14 - 5)", 25.0);
|
||||
expect("(702 + 408) - ((239 - 734) / -5) + -4", 1007.0);
|
||||
expect_num("10 - 3 * 2", 4.0);
|
||||
expect_num("-4 * -4 + (14 - 5)", 25.0);
|
||||
expect_num("(702 + 408) - ((239 - 734) / -5) + -4", 1007.0);
|
||||
}
|
||||
|
|
|
@ -1 +1,6 @@
|
|||
pub type Value = f64;
|
||||
#[derive(Clone, Debug, PartialEq)]
|
||||
pub enum Value {
|
||||
Nil,
|
||||
Bool(bool),
|
||||
Number(f64),
|
||||
}
|
||||
|
|
|
@ -23,11 +23,34 @@ impl VM {
|
|||
}
|
||||
}
|
||||
|
||||
macro_rules! with_type {
|
||||
( $self:ident, $val:ident, $type:pat, $body:expr ) => {
|
||||
match $val {
|
||||
$type => $body,
|
||||
_ => {
|
||||
return Err(Error {
|
||||
line: $self.chunk.get_line($self.ip - 1),
|
||||
kind: ErrorKind::TypeError(format!(
|
||||
"Expected type {}, but found value: {:?}",
|
||||
stringify!($type),
|
||||
$val,
|
||||
)),
|
||||
})
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
macro_rules! binary_op {
|
||||
( $vm:ident, $op:tt ) => {{
|
||||
let b = $vm.pop();
|
||||
let a = $vm.pop();
|
||||
$vm.push(a $op b);
|
||||
|
||||
with_type!($vm, b, Value::Number(num_b), {
|
||||
with_type!($vm, a, Value::Number(num_a), {
|
||||
$vm.push(Value::Number(num_a $op num_b))
|
||||
})
|
||||
})
|
||||
}}
|
||||
}
|
||||
|
||||
|
@ -45,13 +68,18 @@ impl VM {
|
|||
OpCode::OpReturn => return Ok(self.pop()),
|
||||
|
||||
OpCode::OpConstant(idx) => {
|
||||
let c = *self.chunk.constant(*idx);
|
||||
let c = self.chunk.constant(*idx).clone();
|
||||
self.push(c);
|
||||
}
|
||||
|
||||
OpCode::OpNegate => {
|
||||
let v = self.pop();
|
||||
self.push(-v)
|
||||
with_type!(
|
||||
self,
|
||||
v,
|
||||
Value::Number(num),
|
||||
self.push(Value::Number(-num))
|
||||
);
|
||||
}
|
||||
|
||||
OpCode::OpAdd => binary_op!(self, +),
|
||||
|
|
Loading…
Reference in a new issue