diff options
| author | Edvard Thörnros <edvard.thornros@gmail.com> | 2021-01-31 22:28:48 +0100 |
|---|---|---|
| committer | Edvard Thörnros <edvard.thornros@gmail.com> | 2021-02-01 10:22:56 +0100 |
| commit | afc4a3fe1618239017e15cabc664dd70abc1d75b (patch) | |
| tree | 6f086d71b24a230fe7f656a9780536808b56644b /src/compiler.rs | |
| parent | cc4bacf33c98e9bd186a2a3da7335577df879a9d (diff) | |
| download | sylt-afc4a3fe1618239017e15cabc664dd70abc1d75b.tar.gz | |
add assignment ops
Diffstat (limited to 'src/compiler.rs')
| -rw-r--r-- | src/compiler.rs | 82 |
1 files changed, 66 insertions, 16 deletions
diff --git a/src/compiler.rs b/src/compiler.rs index b7fdce1..69bf6eb 100644 --- a/src/compiler.rs +++ b/src/compiler.rs @@ -621,9 +621,38 @@ impl Compiler { } } - fn assign(&mut self, name: &str, block: &mut Block) { + fn assign(&mut self, block: &mut Block) { + let name = match self.eat() { + Token::Identifier(name) => name, + _ => { + error!(self, format!("Expected identifier in assignment")); + return; + } + }; + + let op = match self.eat() { + Token::Equal => None, + + Token::PlusEqual => Some(Op::Add), + Token::MinusEqual => Some(Op::Sub), + Token::StarEqual => Some(Op::Mul), + Token::SlashEqual => Some(Op::Div), + + _ => { + error!(self, format!("Expected '=' in assignment")); + return; + } + }; + if let Some(var) = self.find_variable(&name) { - self.expression(block); + if let Some(op) = op { + block.add(Op::Copy, self.line()); + self.expression(block); + block.add(op, self.line()); + } else { + self.expression(block); + } + if var.upvalue { block.add(Op::AssignUpvalue(var.slot), self.line()); } else { @@ -832,14 +861,31 @@ impl Compiler { return Err(()); }; - if self.peek() == Token::Equal { - self.eat(); - self.expression(block); - block.add(Op::Set(field), self.line()); - return Ok(()); - } else { - block.add(Op::Get(field), self.line()); - } + let op = match self.peek() { + Token::Equal => { + self.eat(); + self.expression(block); + block.add(Op::Set(field), self.line()); + return Ok(()); + } + + Token::PlusEqual => Op::Add, + Token::MinusEqual => Op::Sub, + Token::StarEqual => Op::Mul, + Token::SlashEqual => Op::Div, + + _ => { + block.add(Op::Get(field), self.line()); + continue; + } + }; + block.add(Op::Copy, self.line()); + block.add(Op::Get(field.clone()), self.line()); + self.eat(); + self.expression(block); + block.add(op, self.line()); + block.add(Op::Set(field), self.line()); + return Ok(()); } Token::LeftParen => { self.call(block); @@ -867,6 +913,16 @@ impl Compiler { block.add(Op::Print, self.line()); } + (Token::Identifier(_), Token::Equal, ..) | + (Token::Identifier(_), Token::PlusEqual, ..) | + (Token::Identifier(_), Token::MinusEqual, ..) | + (Token::Identifier(_), Token::SlashEqual, ..) | + (Token::Identifier(_), Token::StarEqual, ..) + + => { + self.assign(block); + } + (Token::Identifier(_), Token::Dot, ..) => { let block_length = block.ops.len(); let token_length = self.curr; @@ -900,12 +956,6 @@ impl Compiler { self.definition_statement(&name, Type::UnknownType, block); } - (Token::Identifier(name), Token::Equal, ..) => { - self.eat(); - self.eat(); - self.assign(&name, block); - } - (Token::Blob, Token::Identifier(_), ..) => { self.blob_statement(block); } |
