diff options
author | Vincent Ambo <mail@tazj.in> | 2021-01-14T19·49+0300 |
---|---|---|
committer | tazjin <mail@tazj.in> | 2021-01-14T20·26+0000 |
commit | 39439d59e8e9ddb1e2b7802f3aff092d77de7acf (patch) | |
tree | e7c808364ec2078faa42739877d4ffdba5bd4ec9 /users/tazjin/rlox/src/parser.rs | |
parent | 20a6cfeee233bde9ba1f482fa4545f5e395d6235 (diff) |
feat(tazjin/rlox): Implement early return from functions r/2108
In the book this is implemented via exceptions as control flow, and I'm sticking somewhat closely to that by doing it via an error variant. Change-Id: I9c7b84d6bb28265ab94021ea681df84f16a53da2 Reviewed-on: https://cl.tvl.fyi/c/depot/+/2395 Reviewed-by: tazjin <mail@tazj.in> Tested-by: BuildkiteCI
Diffstat (limited to 'users/tazjin/rlox/src/parser.rs')
-rw-r--r-- | users/tazjin/rlox/src/parser.rs | 17 |
1 files changed, 17 insertions, 0 deletions
diff --git a/users/tazjin/rlox/src/parser.rs b/users/tazjin/rlox/src/parser.rs index ed5e670ecb79..495304686b2f 100644 --- a/users/tazjin/rlox/src/parser.rs +++ b/users/tazjin/rlox/src/parser.rs @@ -80,6 +80,11 @@ pub struct Var { } #[derive(Debug)] +pub struct Return { + pub value: Expr, +} + +#[derive(Debug)] pub struct If { pub condition: Expr, pub then_branch: Box<Statement>, @@ -110,6 +115,7 @@ pub enum Statement { If(If), While(While), Function(Rc<Function>), + Return(Return), } // Parser @@ -130,6 +136,7 @@ statement → exprStmt | forStmt | ifStmt | printStmt + | returnStmt | whileStmt | block ; @@ -137,6 +144,8 @@ forStmt → "for" "(" ( varDecl | exprStmt | ";" ) expression? ";" expression? ")" statement ; +returnStmt → "return" expression? ";" ; + whileStmt → "while" "(" expression ")" statement ; exprStmt → expression ";" ; @@ -256,6 +265,8 @@ impl Parser { self.while_statement() } else if self.match_token(&TokenKind::For) { self.for_statement() + } else if self.match_token(&TokenKind::Return) { + self.return_statement() } else { self.expr_statement() } @@ -379,6 +390,12 @@ impl Parser { Ok(body) } + fn return_statement(&mut self) -> StmtResult { + let value = self.expression()?; + self.consume(&TokenKind::Semicolon, ErrorKind::ExpectedSemicolon)?; + Ok(Statement::Return(Return { value })) + } + fn expr_statement(&mut self) -> StmtResult { let expr = self.expression()?; self.consume(&TokenKind::Semicolon, ErrorKind::ExpectedSemicolon)?; |