feat(tazjin/rlox): Implement unary negation operator

Change-Id: I9a5bd3581d4ed05371651697ec496341eb7971ae
Reviewed-on: https://cl.tvl.fyi/c/depot/+/2572
Reviewed-by: tazjin <mail@tazj.in>
Tested-by: BuildkiteCI
This commit is contained in:
Vincent Ambo 2021-02-28 15:28:04 +02:00 committed by tazjin
parent 47ffa80711
commit 2d9456d247
5 changed files with 32 additions and 0 deletions

View file

@ -116,6 +116,10 @@ fn rule_for<T: Iterator<Item = Token>>(token: &TokenKind) -> ParseRule<T> {
ParseRule::new(Some(Compiler::literal), None, Precedence::None)
}
TokenKind::Bang => {
ParseRule::new(Some(Compiler::unary), None, Precedence::None)
}
_ => ParseRule::new(None, None, Precedence::None),
}
}
@ -168,6 +172,7 @@ impl<T: Iterator<Item = Token>> Compiler<T> {
// Emit operator instruction
match kind {
TokenKind::Bang => self.emit_op(OpCode::OpNot),
TokenKind::Minus => self.emit_op(OpCode::OpNegate),
_ => unreachable!("only called for unary operator tokens"),
}

View file

@ -11,6 +11,9 @@ pub enum OpCode {
/// Return from the current function.
OpReturn,
// Boolean operators
OpNot,
/// Unary negation
OpNegate,

View file

@ -58,3 +58,12 @@ fn trivial_literals() {
expect("false", Value::Bool(false));
expect("nil", Value::Nil);
}
#[test]
fn negation() {
expect("!true", Value::Bool(false));
expect("!false", Value::Bool(true));
expect("!nil", Value::Bool(true));
expect("!13.5", Value::Bool(false));
expect("!-42", Value::Bool(false));
}

View file

@ -4,3 +4,13 @@ pub enum Value {
Bool(bool),
Number(f64),
}
impl Value {
pub fn is_falsey(&self) -> bool {
match self {
Value::Nil => true,
Value::Bool(false) => true,
_ => false,
}
}
}

View file

@ -76,6 +76,11 @@ impl VM {
OpCode::OpTrue => self.push(Value::Bool(true)),
OpCode::OpFalse => self.push(Value::Bool(false)),
OpCode::OpNot => {
let v = self.pop();
self.push(Value::Bool(v.is_falsey()));
}
OpCode::OpNegate => {
let v = self.pop();
with_type!(