Implement blocks with scopes
This commit is contained in:
parent
4ba3a47ae5
commit
c3790e90bf
@ -129,6 +129,14 @@ impl Chunk {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn begin_scope(&mut self) {
|
||||||
|
self.identifiers.begin_scope();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn end_scope(&mut self) {
|
||||||
|
self.identifiers.end_scope();
|
||||||
|
}
|
||||||
|
|
||||||
pub fn clear(&mut self) {
|
pub fn clear(&mut self) {
|
||||||
self.code.clear();
|
self.code.clear();
|
||||||
self.constants.clear();
|
self.constants.clear();
|
||||||
@ -138,9 +146,13 @@ impl Chunk {
|
|||||||
pub fn disassemble(&self, name: &str) -> String {
|
pub fn disassemble(&self, name: &str) -> String {
|
||||||
let mut output = String::new();
|
let mut output = String::new();
|
||||||
|
|
||||||
output.push_str("# ");
|
let name_length = name.len();
|
||||||
output.push_str(name);
|
let buffer_length = 32_usize.saturating_sub(name_length + 2);
|
||||||
output.push_str("\n\n## Code\n");
|
let name_buffer = " ".repeat(buffer_length / 2);
|
||||||
|
let name_line = format!("{name_buffer} {name} {name_buffer}\n");
|
||||||
|
|
||||||
|
output.push_str(&name_line);
|
||||||
|
output.push_str("\n Code \n");
|
||||||
output.push_str("------ ------------ ------------\n");
|
output.push_str("------ ------------ ------------\n");
|
||||||
output.push_str("OFFSET POSITION INSTRUCTION\n");
|
output.push_str("OFFSET POSITION INSTRUCTION\n");
|
||||||
output.push_str("------ ------------ ------------\n");
|
output.push_str("------ ------------ ------------\n");
|
||||||
@ -172,7 +184,7 @@ impl Chunk {
|
|||||||
output.push_str(&display);
|
output.push_str(&display);
|
||||||
}
|
}
|
||||||
|
|
||||||
output.push_str("\n## Constants\n");
|
output.push_str("\n Constants \n");
|
||||||
output.push_str("----- ---- -----\n");
|
output.push_str("----- ---- -----\n");
|
||||||
output.push_str("INDEX KIND VALUE\n");
|
output.push_str("INDEX KIND VALUE\n");
|
||||||
output.push_str("----- ---- -----\n");
|
output.push_str("----- ---- -----\n");
|
||||||
@ -188,7 +200,7 @@ impl Chunk {
|
|||||||
output.push_str(&display);
|
output.push_str(&display);
|
||||||
}
|
}
|
||||||
|
|
||||||
output.push_str("\n## Identifiers\n");
|
output.push_str("\n Identifiers \n");
|
||||||
output.push_str("----- ---------- -------- -----\n");
|
output.push_str("----- ---------- -------- -----\n");
|
||||||
output.push_str("INDEX IDENTIFIER LOCATION DEPTH\n");
|
output.push_str("INDEX IDENTIFIER LOCATION DEPTH\n");
|
||||||
output.push_str("----- ---------- -------- -----\n");
|
output.push_str("----- ---------- -------- -----\n");
|
||||||
|
@ -134,3 +134,9 @@ impl Display for Instruction {
|
|||||||
write!(f, "{self:?}")
|
write!(f, "{self:?}")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<Instruction> for u8 {
|
||||||
|
fn from(instruction: Instruction) -> Self {
|
||||||
|
instruction as u8
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -90,8 +90,8 @@ impl<'src> Parser<'src> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn emit_byte(&mut self, byte: u8, position: Span) {
|
fn emit_byte<T: Into<u8>>(&mut self, into_byte: T, position: Span) {
|
||||||
self.chunk.push_code(byte, position);
|
self.chunk.push_code(into_byte.into(), position);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn emit_constant(&mut self, value: Value) -> Result<(), ParseError> {
|
fn emit_constant(&mut self, value: Value) -> Result<(), ParseError> {
|
||||||
@ -101,7 +101,7 @@ impl<'src> Parser<'src> {
|
|||||||
.push_constant(value)
|
.push_constant(value)
|
||||||
.map_err(|error| ParseError::Chunk { error, position })?;
|
.map_err(|error| ParseError::Chunk { error, position })?;
|
||||||
|
|
||||||
self.emit_byte(Instruction::Constant as u8, position);
|
self.emit_byte(Instruction::Constant, position);
|
||||||
self.emit_byte(constant_index, position);
|
self.emit_byte(constant_index, position);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -183,7 +183,7 @@ impl<'src> Parser<'src> {
|
|||||||
fn parse_unary(&mut self, _allow_assignment: bool) -> Result<(), ParseError> {
|
fn parse_unary(&mut self, _allow_assignment: bool) -> Result<(), ParseError> {
|
||||||
let operator_position = self.previous_position;
|
let operator_position = self.previous_position;
|
||||||
let byte = match self.previous_token.kind() {
|
let byte = match self.previous_token.kind() {
|
||||||
TokenKind::Minus => Instruction::Negate as u8,
|
TokenKind::Minus => Instruction::Negate,
|
||||||
_ => {
|
_ => {
|
||||||
return Err(ParseError::ExpectedTokenMultiple {
|
return Err(ParseError::ExpectedTokenMultiple {
|
||||||
expected: vec![TokenKind::Minus],
|
expected: vec![TokenKind::Minus],
|
||||||
@ -209,11 +209,11 @@ impl<'src> Parser<'src> {
|
|||||||
self.parse(rule.precedence.increment())?;
|
self.parse(rule.precedence.increment())?;
|
||||||
|
|
||||||
let byte = match operator {
|
let byte = match operator {
|
||||||
TokenKind::Plus => Instruction::Add as u8,
|
TokenKind::Plus => Instruction::Add,
|
||||||
TokenKind::Minus => Instruction::Subtract as u8,
|
TokenKind::Minus => Instruction::Subtract,
|
||||||
TokenKind::Star => Instruction::Multiply as u8,
|
TokenKind::Star => Instruction::Multiply,
|
||||||
TokenKind::Slash => Instruction::Divide as u8,
|
TokenKind::Slash => Instruction::Divide,
|
||||||
TokenKind::DoubleAmpersand => Instruction::And as u8,
|
TokenKind::DoubleAmpersand => Instruction::And,
|
||||||
_ => {
|
_ => {
|
||||||
return Err(ParseError::ExpectedTokenMultiple {
|
return Err(ParseError::ExpectedTokenMultiple {
|
||||||
expected: vec![
|
expected: vec![
|
||||||
@ -243,10 +243,10 @@ impl<'src> Parser<'src> {
|
|||||||
|
|
||||||
if allow_assignment && self.allow(TokenKind::Equal)? {
|
if allow_assignment && self.allow(TokenKind::Equal)? {
|
||||||
self.parse_expression()?;
|
self.parse_expression()?;
|
||||||
self.emit_byte(Instruction::SetVariable as u8, self.previous_position);
|
self.emit_byte(Instruction::SetVariable, self.previous_position);
|
||||||
self.emit_byte(identifier_index, self.previous_position);
|
self.emit_byte(identifier_index, self.previous_position);
|
||||||
} else {
|
} else {
|
||||||
self.emit_byte(Instruction::GetVariable as u8, self.previous_position);
|
self.emit_byte(Instruction::GetVariable, self.previous_position);
|
||||||
self.emit_byte(identifier_index, self.previous_position);
|
self.emit_byte(identifier_index, self.previous_position);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -274,34 +274,47 @@ impl<'src> Parser<'src> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn parse_block(&mut self, _allow_assignment: bool) -> Result<(), ParseError> {
|
||||||
|
self.chunk.begin_scope();
|
||||||
|
|
||||||
|
while !self.allow(TokenKind::RightCurlyBrace)? && !self.is_eof() {
|
||||||
|
self.parse_statement()?;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.chunk.end_scope();
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
fn parse_expression(&mut self) -> Result<(), ParseError> {
|
fn parse_expression(&mut self) -> Result<(), ParseError> {
|
||||||
self.parse(Precedence::None)
|
self.parse(Precedence::None)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_statement(&mut self) -> Result<(), ParseError> {
|
fn parse_statement(&mut self) -> Result<(), ParseError> {
|
||||||
let start = self.current_position.0;
|
let start = self.current_position.0;
|
||||||
let is_expression_statement = match self.current_token {
|
let (is_expression_statement, contains_block) = match self.current_token {
|
||||||
Token::Let => {
|
Token::Let => {
|
||||||
self.parse_let_assignment(true)?;
|
self.parse_let_assignment(true)?;
|
||||||
|
|
||||||
false
|
(false, false)
|
||||||
|
}
|
||||||
|
Token::LeftCurlyBrace => {
|
||||||
|
self.parse_expression()?;
|
||||||
|
|
||||||
|
(true, true)
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
self.parse_expression()?;
|
self.parse_expression()?;
|
||||||
|
|
||||||
true
|
(true, false)
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
let has_semicolon = self.allow(TokenKind::Semicolon)?;
|
let has_semicolon = self.allow(TokenKind::Semicolon)?;
|
||||||
|
|
||||||
if is_expression_statement {
|
if is_expression_statement && !contains_block && !has_semicolon {
|
||||||
let end = self.previous_position.1;
|
let end = self.previous_position.1;
|
||||||
|
|
||||||
if has_semicolon {
|
self.emit_byte(Instruction::Return, Span(start, end))
|
||||||
self.emit_byte(Instruction::Pop as u8, Span(start, end));
|
|
||||||
} else {
|
|
||||||
self.emit_byte(Instruction::Return as u8, Span(start, end));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -343,14 +356,14 @@ impl<'src> Parser<'src> {
|
|||||||
.map_err(|error| ParseError::Chunk { error, position })?;
|
.map_err(|error| ParseError::Chunk { error, position })?;
|
||||||
|
|
||||||
if is_constant {
|
if is_constant {
|
||||||
self.emit_byte(Instruction::DefineVariableConstant as u8, position);
|
self.emit_byte(Instruction::DefineVariableConstant, position);
|
||||||
self.emit_byte(identifier_index, position);
|
self.emit_byte(identifier_index, position);
|
||||||
|
|
||||||
self.parse_expression()?;
|
self.parse_expression()?;
|
||||||
} else {
|
} else {
|
||||||
self.parse_expression()?;
|
self.parse_expression()?;
|
||||||
|
|
||||||
self.emit_byte(Instruction::DefineVariableRuntime as u8, position);
|
self.emit_byte(Instruction::DefineVariableRuntime, position);
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -533,7 +546,11 @@ impl From<&TokenKind> for ParseRule<'_> {
|
|||||||
TokenKind::Equal => todo!(),
|
TokenKind::Equal => todo!(),
|
||||||
TokenKind::Greater => todo!(),
|
TokenKind::Greater => todo!(),
|
||||||
TokenKind::GreaterOrEqual => todo!(),
|
TokenKind::GreaterOrEqual => todo!(),
|
||||||
TokenKind::LeftCurlyBrace => todo!(),
|
TokenKind::LeftCurlyBrace => ParseRule {
|
||||||
|
prefix: Some(Parser::parse_block),
|
||||||
|
infix: None,
|
||||||
|
precedence: Precedence::None,
|
||||||
|
},
|
||||||
TokenKind::LeftParenthesis => ParseRule {
|
TokenKind::LeftParenthesis => ParseRule {
|
||||||
prefix: Some(Parser::parse_grouped),
|
prefix: Some(Parser::parse_grouped),
|
||||||
infix: None,
|
infix: None,
|
||||||
@ -556,7 +573,11 @@ impl From<&TokenKind> for ParseRule<'_> {
|
|||||||
precedence: Precedence::Term,
|
precedence: Precedence::Term,
|
||||||
},
|
},
|
||||||
TokenKind::PlusEqual => todo!(),
|
TokenKind::PlusEqual => todo!(),
|
||||||
TokenKind::RightCurlyBrace => todo!(),
|
TokenKind::RightCurlyBrace => ParseRule {
|
||||||
|
prefix: None,
|
||||||
|
infix: None,
|
||||||
|
precedence: Precedence::None,
|
||||||
|
},
|
||||||
TokenKind::RightParenthesis => ParseRule {
|
TokenKind::RightParenthesis => ParseRule {
|
||||||
prefix: None,
|
prefix: None,
|
||||||
infix: None,
|
infix: None,
|
||||||
@ -678,6 +699,27 @@ mod tests {
|
|||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn block() {
|
||||||
|
let source = "{ 42; 42 }";
|
||||||
|
let test_chunk = parse(source);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
test_chunk,
|
||||||
|
Ok(Chunk::with_data(
|
||||||
|
vec![
|
||||||
|
(Instruction::Constant as u8, Span(2, 4)),
|
||||||
|
(0, Span(2, 4)),
|
||||||
|
(Instruction::Constant as u8, Span(6, 8)),
|
||||||
|
(1, Span(6, 8)),
|
||||||
|
(Instruction::Return as u8, Span(6, 8)),
|
||||||
|
],
|
||||||
|
vec![Value::integer(42), Value::integer(42)],
|
||||||
|
vec![]
|
||||||
|
))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn add_variables() {
|
fn add_variables() {
|
||||||
let source = "let x = 42; let y = 42; x + y";
|
let source = "let x = 42; let y = 42; x + y";
|
||||||
|
Loading…
Reference in New Issue
Block a user