From 2cd77ea26d76b20ff820f1ebe5e77f1360f5d1f5 Mon Sep 17 00:00:00 2001 From: Vincent Ambo Date: Tue, 2 Mar 2021 22:15:08 +0200 Subject: feat(tazjin/rlox): Add support for print statement Change-Id: Ic3e7e722325c8784b848c0bcd573c2e51e123c40 Reviewed-on: https://cl.tvl.fyi/c/depot/+/2583 Reviewed-by: tazjin Tested-by: BuildkiteCI --- users/tazjin/rlox/src/bytecode/compiler.rs | 44 ++++++++++++++++++++++++++---- users/tazjin/rlox/src/bytecode/opcode.rs | 3 ++ users/tazjin/rlox/src/bytecode/vm.rs | 19 +++++++++++++ 3 files changed, 61 insertions(+), 5 deletions(-) (limited to 'users') diff --git a/users/tazjin/rlox/src/bytecode/compiler.rs b/users/tazjin/rlox/src/bytecode/compiler.rs index ae38654c5d38..1408f519bb2d 100644 --- a/users/tazjin/rlox/src/bytecode/compiler.rs +++ b/users/tazjin/rlox/src/bytecode/compiler.rs @@ -156,11 +156,10 @@ fn rule_for>(token: &TokenKind) -> ParseRule { impl> Compiler { fn compile(&mut self) -> LoxResult<()> { self.advance(); - self.expression()?; - self.consume( - &TokenKind::Eof, - ErrorKind::ExpectedToken("Expected end of expression"), - ); + + while !self.match_token(&TokenKind::Eof) { + self.declaration()?; + } self.end_compiler() } @@ -174,6 +173,28 @@ impl> Compiler { self.parse_precedence(Precedence::Assignment) } + fn declaration(&mut self) -> LoxResult<()> { + self.statement() + } + + fn statement(&mut self) -> LoxResult<()> { + if self.match_token(&TokenKind::Print) { + return self.print_statement(); + } + + Ok(()) + } + + fn print_statement(&mut self) -> LoxResult<()> { + self.expression()?; + self.consume( + &TokenKind::Semicolon, + ErrorKind::ExpectedToken("Expected ';' after value"), + ); + self.emit_op(OpCode::OpPrint); + Ok(()) + } + fn number(&mut self) -> LoxResult<()> { if let TokenKind::Number(num) = self.previous().kind { self.emit_constant(Value::Number(num)); @@ -352,6 +373,19 @@ impl> Compiler { self.panic = true; self.errors.push(Error { kind, line }) } + + fn match_token(&mut self, token: &TokenKind) -> bool { + if !self.check(token) { + return false; + } + + self.advance(); + true + } + + fn check(&self, token: &TokenKind) -> bool { + return self.current().kind == *token; + } } pub fn compile(code: &str) -> Result<(Interner, Chunk), Vec> { diff --git a/users/tazjin/rlox/src/bytecode/opcode.rs b/users/tazjin/rlox/src/bytecode/opcode.rs index 78c1b2d8e3a5..7a3355460473 100644 --- a/users/tazjin/rlox/src/bytecode/opcode.rs +++ b/users/tazjin/rlox/src/bytecode/opcode.rs @@ -25,4 +25,7 @@ pub enum OpCode { OpSubtract, OpMultiply, OpDivide, + + // Built in operations + OpPrint, } diff --git a/users/tazjin/rlox/src/bytecode/vm.rs b/users/tazjin/rlox/src/bytecode/vm.rs index 02db30de58a9..c43b197279be 100644 --- a/users/tazjin/rlox/src/bytecode/vm.rs +++ b/users/tazjin/rlox/src/bytecode/vm.rs @@ -72,6 +72,10 @@ impl VM { match op { OpCode::OpReturn => { + if self.stack.is_empty() { + return Ok(Value::Nil); + } + let val = self.pop(); return Ok(self.return_value(val)); } @@ -135,6 +139,11 @@ impl VM { }) } } + + OpCode::OpPrint => { + let val = self.pop(); + println!("{}", self.print_value(val)); + } } #[cfg(feature = "disassemble")] @@ -159,6 +168,16 @@ impl VM { LoxString::Interned(id) => self.strings.lookup(*id), } } + + fn print_value(&self, val: Value) -> String { + match val { + Value::String(LoxString::Heap(s)) => s, + Value::String(LoxString::Interned(id)) => { + self.strings.lookup(id).into() + } + _ => format!("{:?}", val), + } + } } pub fn interpret(strings: Interner, chunk: chunk::Chunk) -> LoxResult { -- cgit 1.4.1