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:
parent
47ffa80711
commit
2d9456d247
5 changed files with 32 additions and 0 deletions
|
@ -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"),
|
||||
}
|
||||
|
|
|
@ -11,6 +11,9 @@ pub enum OpCode {
|
|||
/// Return from the current function.
|
||||
OpReturn,
|
||||
|
||||
// Boolean operators
|
||||
OpNot,
|
||||
|
||||
/// Unary negation
|
||||
OpNegate,
|
||||
|
||||
|
|
|
@ -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));
|
||||
}
|
||||
|
|
|
@ -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,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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!(
|
||||
|
|
Loading…
Reference in a new issue