interpreter: rewrite interpreter
- Remove interpreter struct - Replace with `Interpret` trait - This separates concerns dramatically! Yay! - Implement block scoping via `Frame` abstraction - TODO: is this the right abstraction? - TODO: Modules?? - TODO: What environment should be passed into a function call? ast: - rename Name.name to Name.symbol (name.name.name.name.name.name.name is very readable, yes yes)
This commit is contained in:
		| @@ -69,7 +69,7 @@ pub mod program { | |||||||
|  |  | ||||||
|     use conlang::{ |     use conlang::{ | ||||||
|         ast::preamble::{expression::Expr, *}, |         ast::preamble::{expression::Expr, *}, | ||||||
|         interpreter::{error::IResult, Interpreter}, |         interpreter::{env::Environment, error::IResult}, | ||||||
|         lexer::Lexer, |         lexer::Lexer, | ||||||
|         parser::{error::PResult, Parser}, |         parser::{error::PResult, Parser}, | ||||||
|         pretty_printer::{PrettyPrintable, Printer}, |         pretty_printer::{PrettyPrintable, Printer}, | ||||||
| @@ -137,18 +137,17 @@ pub mod program { | |||||||
|             .map(|ty| println!("{ty}")) |             .map(|ty| println!("{ty}")) | ||||||
|         } |         } | ||||||
|         /// Runs the [Program] in the specified [Interpreter] |         /// Runs the [Program] in the specified [Interpreter] | ||||||
|         pub fn run(&self, interpreter: &mut Interpreter) -> IResult<()> { |         pub fn run(&self, env: &mut Environment) -> IResult<()> { | ||||||
|  |             println!( | ||||||
|  |                 "{}", | ||||||
|                 match &self.data { |                 match &self.data { | ||||||
|                 Parsed::Program(start) => interpreter.interpret(start), |                     Parsed::Program(start) => env.eval(start)?, | ||||||
|                 Parsed::Expr(expr) => { |                     Parsed::Expr(expr) => env.eval(expr)?, | ||||||
|                     for value in interpreter.eval(expr)? { |  | ||||||
|                         println!("{value}") |  | ||||||
|                 } |                 } | ||||||
|  |             ); | ||||||
|             Ok(()) |             Ok(()) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     impl PrettyPrintable for Program<Parsed> { |     impl PrettyPrintable for Program<Parsed> { | ||||||
|         fn visit<W: Write>(&self, p: &mut Printer<W>) -> IOResult<()> { |         fn visit<W: Write>(&self, p: &mut Printer<W>) -> IOResult<()> { | ||||||
| @@ -162,7 +161,8 @@ pub mod program { | |||||||
|  |  | ||||||
| pub mod cli { | pub mod cli { | ||||||
|     use conlang::{ |     use conlang::{ | ||||||
|         interpreter::Interpreter, pretty_printer::PrettyPrintable, resolver::Resolver, token::Token, |         interpreter::env::Environment, pretty_printer::PrettyPrintable, resolver::Resolver, | ||||||
|  |         token::Token, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     use crate::{ |     use crate::{ | ||||||
| @@ -231,7 +231,7 @@ pub mod cli { | |||||||
|                 } |                 } | ||||||
|                 (Mode::Beautify, Ok(program)) => Self::beautify(program), |                 (Mode::Beautify, Ok(program)) => Self::beautify(program), | ||||||
|                 (Mode::Resolve, Ok(program)) => Self::resolve(program, Default::default()), |                 (Mode::Resolve, Ok(program)) => Self::resolve(program, Default::default()), | ||||||
|                 (Mode::Interpret, Ok(program)) => Self::interpret(program, Default::default()), |                 (Mode::Interpret, Ok(program)) => Self::interpret(program, Environment::new()), | ||||||
|                 (_, Err(errors)) => { |                 (_, Err(errors)) => { | ||||||
|                     for error in errors { |                     for error in errors { | ||||||
|                         if let Some(path) = path { |                         if let Some(path) = path { | ||||||
| @@ -260,7 +260,7 @@ pub mod cli { | |||||||
|                 eprintln!("{e}"); |                 eprintln!("{e}"); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         fn interpret(program: Program<Tokenized>, mut interpreter: Interpreter) { |         fn interpret(program: Program<Tokenized>, mut interpreter: Environment) { | ||||||
|             let program = match program.parse() { |             let program = match program.parse() { | ||||||
|                 Ok(program) => program, |                 Ok(program) => program, | ||||||
|                 Err(e) => { |                 Err(e) => { | ||||||
| @@ -316,7 +316,7 @@ pub mod cli { | |||||||
|         prompt_again: &'static str, // " ?>" |         prompt_again: &'static str, // " ?>" | ||||||
|         prompt_begin: &'static str, // "cl>" |         prompt_begin: &'static str, // "cl>" | ||||||
|         prompt_error: &'static str, // "! >" |         prompt_error: &'static str, // "! >" | ||||||
|         interpreter: Interpreter, |         env: Environment, | ||||||
|         resolver: Resolver, |         resolver: Resolver, | ||||||
|         mode: Mode, |         mode: Mode, | ||||||
|     } |     } | ||||||
| @@ -327,7 +327,7 @@ pub mod cli { | |||||||
|                 prompt_begin: "cl>", |                 prompt_begin: "cl>", | ||||||
|                 prompt_again: " ?>", |                 prompt_again: " ?>", | ||||||
|                 prompt_error: "! >", |                 prompt_error: "! >", | ||||||
|                 interpreter: Default::default(), |                 env: Default::default(), | ||||||
|                 resolver: Default::default(), |                 resolver: Default::default(), | ||||||
|                 mode: Default::default(), |                 mode: Default::default(), | ||||||
|             } |             } | ||||||
| @@ -457,7 +457,7 @@ pub mod cli { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         fn interpret(&mut self, code: &Program<Parsed>) { |         fn interpret(&mut self, code: &Program<Parsed>) { | ||||||
|             if let Err(e) = code.run(&mut self.interpreter) { |             if let Err(e) = code.run(&mut self.env) { | ||||||
|                 self.prompt_error(&e) |                 self.prompt_error(&e) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ pub mod preamble { | |||||||
|         path::*, |         path::*, | ||||||
|         statement::*, |         statement::*, | ||||||
|         types::*, |         types::*, | ||||||
|         visitor::Visitor, |  | ||||||
|         *, |         *, | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| @@ -156,7 +155,7 @@ pub mod statement { | |||||||
|     /// # Syntax |     /// # Syntax | ||||||
|     #[derive(Clone, Debug)] |     #[derive(Clone, Debug)] | ||||||
|     pub struct Name { |     pub struct Name { | ||||||
|         pub name: Identifier, |         pub symbol: Identifier, | ||||||
|         /// The mutability of the [Name]. Functions are never mutable. |         /// The mutability of the [Name]. Functions are never mutable. | ||||||
|         pub mutable: bool, |         pub mutable: bool, | ||||||
|         /// The [type](TypeExpr) |         /// The [type](TypeExpr) | ||||||
| @@ -743,166 +742,6 @@ pub mod expression { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub mod visitor { |  | ||||||
|     //! A [`Visitor`] visits every kind of node in the [Abstract Syntax Tree](super) |  | ||||||
|     //! |  | ||||||
|     //! This trait is mostly here to ensure that every branch of the tree is accounted for. |  | ||||||
|     //! |  | ||||||
|     //! Default implementations are provided for |  | ||||||
|     use super::{ |  | ||||||
|         expression::{call::*, control::*, math::*, tuple::*, Block, *}, |  | ||||||
|         literal::*, |  | ||||||
|         statement::*, |  | ||||||
|         *, |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     /// A Visitor traverses every kind of node in the [Abstract Syntax Tree](super) |  | ||||||
|     #[deprecated] |  | ||||||
|     pub trait Visitor<R> { |  | ||||||
|         /// Visit the start of an AST |  | ||||||
|         fn visit(&mut self, start: &Start) -> R { |  | ||||||
|             self.visit_program(&start.0) |  | ||||||
|         } |  | ||||||
|         /// Visit a [Program] |  | ||||||
|         fn visit_program(&mut self, prog: &Program) -> R; |  | ||||||
|  |  | ||||||
|         /// Visit a [Statement](Stmt) |  | ||||||
|         fn visit_statement(&mut self, stmt: &Stmt) -> R { |  | ||||||
|             match stmt { |  | ||||||
|                 Stmt::Let(stmt) => self.visit_let(stmt), |  | ||||||
|                 Stmt::Fn(function) => self.visit_fn_decl(function), |  | ||||||
|                 Stmt::Expr(expr) => self.visit_expr(expr), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         /// Visit a [Let statement](Let) |  | ||||||
|         fn visit_let(&mut self, decl: &Let) -> R; |  | ||||||
|         /// Visit a [Fn declaration](FnDecl) |  | ||||||
|         fn visit_fn_decl(&mut self, decl: &FnDecl) -> R; |  | ||||||
|  |  | ||||||
|         /// Visit an [Expression](Expr) |  | ||||||
|         fn visit_expr(&mut self, expr: &Expr) -> R { |  | ||||||
|             self.visit_operation(&expr.0) |  | ||||||
|         } |  | ||||||
|         // Block expression |  | ||||||
|         /// Visit a [Block] expression |  | ||||||
|         fn visit_block(&mut self, block: &Block) -> R; |  | ||||||
|         /// Visit a [Group] expression |  | ||||||
|         fn visit_group(&mut self, group: &Group) -> R { |  | ||||||
|             match group { |  | ||||||
|                 Group::Tuple(tuple) => self.visit_tuple(tuple), |  | ||||||
|                 Group::Single(expr) => self.visit_expr(expr), |  | ||||||
|                 Group::Empty => self.visit_empty(), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         /// Visit a [Tuple] expression |  | ||||||
|         fn visit_tuple(&mut self, tuple: &Tuple) -> R; |  | ||||||
|         /// Visit a [Call] expression |  | ||||||
|         fn visit_call(&mut self, call: &Call) -> R { |  | ||||||
|             match call { |  | ||||||
|                 Call::FnCall(call) => self.visit_fn_call(call), |  | ||||||
|                 Call::Primary(primary) => self.visit_primary(primary), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         /// Visit a [Function Call](FnCall) expression |  | ||||||
|         fn visit_fn_call(&mut self, call: &FnCall) -> R; |  | ||||||
|  |  | ||||||
|         // Math expression |  | ||||||
|         /// Visit an [Operation] |  | ||||||
|         fn visit_operation(&mut self, operation: &Operation) -> R { |  | ||||||
|             match operation { |  | ||||||
|                 Operation::Assign(assign) => self.visit_assign(assign), |  | ||||||
|                 Operation::Binary(binary) => self.visit_binary(binary), |  | ||||||
|                 Operation::Unary(unary) => self.visit_unary(unary), |  | ||||||
|                 Operation::Call(call) => self.visit_call(call), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         /// Visit an [Assignment](Assign) operation |  | ||||||
|         fn visit_assign(&mut self, assign: &Assign) -> R; |  | ||||||
|         /// Visit a [Binary] Operation |  | ||||||
|         fn visit_binary(&mut self, binary: &Binary) -> R; |  | ||||||
|         /// Visit a [Unary] Operation |  | ||||||
|         fn visit_unary(&mut self, unary: &Unary) -> R; |  | ||||||
|         // Math operators |  | ||||||
|         /// Visit an [Assignment](Assign) [operator](operator::Assign) |  | ||||||
|         fn visit_assign_op(&mut self, op: &operator::Assign) -> R; |  | ||||||
|         /// Visit a [Binary] [operator](operator::Binary) |  | ||||||
|         fn visit_binary_op(&mut self, op: &operator::Binary) -> R; |  | ||||||
|         /// Visit a [Unary] [operator](operator::Unary) |  | ||||||
|         fn visit_unary_op(&mut self, op: &operator::Unary) -> R; |  | ||||||
|  |  | ||||||
|         /// Visit a [Primary] expression |  | ||||||
|         /// |  | ||||||
|         /// [`Primary`]` := `[`Identifier`]` | `[`Literal`]` | `[`Block`]` | `[`Flow`] |  | ||||||
|         fn visit_primary(&mut self, primary: &Primary) -> R { |  | ||||||
|             match primary { |  | ||||||
|                 Primary::Identifier(v) => self.visit_identifier(v), |  | ||||||
|                 Primary::Literal(v) => self.visit_literal(v), |  | ||||||
|                 Primary::Block(v) => self.visit_block(v), |  | ||||||
|                 Primary::Group(v) => self.visit_group(v), |  | ||||||
|                 Primary::Branch(v) => self.visit_branch(v), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         /// Visit a [Flow] expression. |  | ||||||
|         /// |  | ||||||
|         /// [`Flow`]` := `[`While`]` | `[`If`]` | `[`For`]` |  | ||||||
|         /// | `[`Continue`]` | `[`Return`]` | `[`Break`] |  | ||||||
|         fn visit_branch(&mut self, flow: &Flow) -> R { |  | ||||||
|             match flow { |  | ||||||
|                 Flow::While(e) => self.visit_while(e), |  | ||||||
|                 Flow::If(e) => self.visit_if(e), |  | ||||||
|                 Flow::For(e) => self.visit_for(e), |  | ||||||
|                 Flow::Continue(e) => self.visit_continue(e), |  | ||||||
|                 Flow::Return(e) => self.visit_return(e), |  | ||||||
|                 Flow::Break(e) => self.visit_break(e), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         /// Visit an [If] expression |  | ||||||
|         fn visit_if(&mut self, expr: &If) -> R; |  | ||||||
|         /// Visit a [While] loop expression |  | ||||||
|         fn visit_while(&mut self, expr: &While) -> R; |  | ||||||
|         /// Visit a [For] loop expression |  | ||||||
|         fn visit_for(&mut self, expr: &For) -> R; |  | ||||||
|         /// Visit an [Else] expression |  | ||||||
|         fn visit_else(&mut self, expr: &Else) -> R; |  | ||||||
|         /// Visit a [Continue] expression |  | ||||||
|         fn visit_continue(&mut self, expr: &Continue) -> R; |  | ||||||
|         /// Visit a [Break] expression |  | ||||||
|         fn visit_break(&mut self, expr: &Break) -> R; |  | ||||||
|         /// Visit a [Return] expression |  | ||||||
|         fn visit_return(&mut self, expr: &Return) -> R; |  | ||||||
|  |  | ||||||
|         // primary symbols |  | ||||||
|         /// Visit an [Identifier] |  | ||||||
|         fn visit_identifier(&mut self, ident: &Identifier) -> R; |  | ||||||
|         /// Visit a [Literal] |  | ||||||
|         /// |  | ||||||
|         /// [`Literal`]` := `[`String`]` | `[`char`]` | `[`bool`]` | `[`Float`]` | `[`u128`] |  | ||||||
|         fn visit_literal(&mut self, literal: &Literal) -> R { |  | ||||||
|             match literal { |  | ||||||
|                 Literal::String(l) => self.visit_string_literal(l), |  | ||||||
|                 Literal::Char(l) => self.visit_char_literal(l), |  | ||||||
|                 Literal::Bool(l) => self.visit_bool_literal(l), |  | ||||||
|                 Literal::Float(l) => self.visit_float_literal(l), |  | ||||||
|                 Literal::Int(l) => self.visit_int_literal(l), |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         /// Visit a [string](str) literal |  | ||||||
|         fn visit_string_literal(&mut self, string: &str) -> R; |  | ||||||
|         /// Visit a [character](char) literal |  | ||||||
|         fn visit_char_literal(&mut self, char: &char) -> R; |  | ||||||
|         /// Visit a [boolean](bool) literal |  | ||||||
|         fn visit_bool_literal(&mut self, bool: &bool) -> R; |  | ||||||
|         /// Visit a [floating point](Float) literal |  | ||||||
|         fn visit_float_literal(&mut self, float: &Float) -> R; |  | ||||||
|         /// Visit an [integer](u128) literal |  | ||||||
|         fn visit_int_literal(&mut self, int: &u128) -> R; |  | ||||||
|  |  | ||||||
|         /// Visit an Empty |  | ||||||
|         fn visit_empty(&mut self) -> R; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub mod todo { | pub mod todo { | ||||||
|     //! temporary storage for pending expression work.  \ |     //! temporary storage for pending expression work.  \ | ||||||
|     //! when an item is in progress, remove it from todo. |     //! when an item is in progress, remove it from todo. | ||||||
|   | |||||||
| @@ -2,15 +2,15 @@ | |||||||
| #![allow(deprecated)] // TODO: REMOVE | #![allow(deprecated)] // TODO: REMOVE | ||||||
|  |  | ||||||
| use crate::ast::preamble::*; | use crate::ast::preamble::*; | ||||||
|  | use env::Environment; | ||||||
| use error::{Error, IResult}; | use error::{Error, IResult}; | ||||||
| use scope::Environment; |  | ||||||
| use temp_type_impl::ConValue; | use temp_type_impl::ConValue; | ||||||
|  |  | ||||||
| /// Callable types can be called from within a Conlang program | /// Callable types can be called from within a Conlang program | ||||||
| pub trait Callable: std::fmt::Debug { | pub trait Callable: std::fmt::Debug { | ||||||
|     /// Calls this [Callable] in the provided [Interpreter], with [ConValue] args  \ |     /// Calls this [Callable] in the provided [Interpreter], with [ConValue] args  \ | ||||||
|     /// The Callable is responsible for checking the argument count and validating types |     /// The Callable is responsible for checking the argument count and validating types | ||||||
|     fn call(&self, interpreter: &mut Interpreter, args: &[ConValue]) -> IResult<ConValue>; |     fn call(&self, interpreter: &mut Environment, args: &[ConValue]) -> IResult<ConValue>; | ||||||
|     /// Returns the common name of this identifier. |     /// Returns the common name of this identifier. | ||||||
|     fn name(&self) -> &str; |     fn name(&self) -> &str; | ||||||
| } | } | ||||||
| @@ -25,7 +25,7 @@ pub mod temp_type_impl { | |||||||
|     use super::{ |     use super::{ | ||||||
|         error::{Error, IResult}, |         error::{Error, IResult}, | ||||||
|         function::Function, |         function::Function, | ||||||
|         BuiltIn, Callable, Interpreter, |         BuiltIn, Callable, Environment, | ||||||
|     }; |     }; | ||||||
|     use std::ops::*; |     use std::ops::*; | ||||||
|     /// A Conlang value |     /// A Conlang value | ||||||
| @@ -106,7 +106,7 @@ pub mod temp_type_impl { | |||||||
|                 _ => "", |                 _ => "", | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         fn call(&self, interpreter: &mut Interpreter, args: &[ConValue]) -> IResult<ConValue> { |         fn call(&self, interpreter: &mut Environment, args: &[ConValue]) -> IResult<ConValue> { | ||||||
|             match self { |             match self { | ||||||
|                 Self::Function(func) => func.call(interpreter, args), |                 Self::Function(func) => func.call(interpreter, args), | ||||||
|                 Self::BuiltIn(func) => func.call(interpreter, args), |                 Self::BuiltIn(func) => func.call(interpreter, args), | ||||||
| @@ -284,129 +284,76 @@ pub mod temp_type_impl { | |||||||
| } | } | ||||||
|  |  | ||||||
| /// A work-in-progress tree walk interpreter for Conlang | /// A work-in-progress tree walk interpreter for Conlang | ||||||
| #[derive(Clone, Debug)] | pub trait Interpret { | ||||||
| pub struct Interpreter { |     /// Interprets this thing using the given [`Interpreter`]. | ||||||
|     scope: Box<Environment>, |     /// | ||||||
|     stack: Vec<ConValue>, |     /// Everything returns a value!™ | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue>; | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Interpreter { | impl Interpret for Start { | ||||||
|     /// Creates a new [Interpreter] |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|     pub fn new() -> Self { |         self.0.interpret(env) | ||||||
|         Default::default() |  | ||||||
|     } |  | ||||||
|     /// Interprets the [Start] of a syntax tree |  | ||||||
|     pub fn interpret(&mut self, start: &Start) -> IResult<()> { |  | ||||||
|         self.visit(start) |  | ||||||
|     } |  | ||||||
|     /// Calls a function inside the interpreter's scope, |  | ||||||
|     /// and returns the result |  | ||||||
|     pub fn call(&mut self, name: &str, args: &[ConValue]) -> IResult<ConValue> { |  | ||||||
|         let function = self.resolve(name)?; |  | ||||||
|         function.call(self, args)?; |  | ||||||
|         self.pop() |  | ||||||
|     } |  | ||||||
|     /// Evaluates a single [Expression](expression::Expr) and returns the value stack. |  | ||||||
|     pub fn eval(&mut self, expr: &expression::Expr) -> IResult<Vec<ConValue>> { |  | ||||||
|         self.visit_expr(expr)?; |  | ||||||
|         Ok(std::mem::take(&mut self.stack)) |  | ||||||
|     } |  | ||||||
|     fn push(&mut self, value: impl Into<ConValue>) { |  | ||||||
|         self.stack.push(value.into()) |  | ||||||
|     } |  | ||||||
|     fn peek(&mut self) -> IResult<&ConValue> { |  | ||||||
|         self.stack.last().ok_or(Error::StackUnderflow) |  | ||||||
|     } |  | ||||||
|     fn pop(&mut self) -> IResult<ConValue> { |  | ||||||
|         self.stack.pop().ok_or(Error::StackUnderflow) |  | ||||||
|     } |  | ||||||
|     fn pop_two(&mut self) -> IResult<(ConValue, ConValue)> { |  | ||||||
|         Ok((self.pop()?, self.pop()?)) |  | ||||||
|     } |  | ||||||
|     fn resolve(&mut self, value: &str) -> IResult<ConValue> { |  | ||||||
|         self.scope.get(value).cloned() |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | impl Interpret for Program { | ||||||
| impl Visitor<IResult<()>> for Interpreter { |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|     fn visit_program(&mut self, prog: &Program) -> IResult<()> { |         let mut out = ConValue::Empty; | ||||||
|         for stmt in &prog.0 { |         for stmt in &self.0 { | ||||||
|             self.visit_statement(stmt)?; |             out = stmt.interpret(env)?; | ||||||
|         } |         } | ||||||
|         Ok(()) |         Ok(out) | ||||||
|     } |     } | ||||||
|  | } | ||||||
|     fn visit_statement(&mut self, stmt: &Stmt) -> IResult<()> { | impl Interpret for Stmt { | ||||||
|         match stmt { |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|             Stmt::Let(l) => self.visit_let(l), |         match self { | ||||||
|             Stmt::Fn(f) => self.visit_fn_decl(f), |             Stmt::Let(l) => l.interpret(env), | ||||||
|             Stmt::Expr(e) => { |             Stmt::Fn(f) => f.interpret(env), | ||||||
|                 self.visit_expr(e)?; |             Stmt::Expr(e) => e.interpret(env), | ||||||
|                 self.pop().map(drop) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | impl Interpret for Let { | ||||||
|     fn visit_let(&mut self, stmt: &Let) -> IResult<()> { |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         let Let { name: Name { name: Identifier { name, .. }, .. }, init, .. } = stmt; |         let Let { name: Name { symbol: Identifier { name, .. }, .. }, init, .. } = self; | ||||||
|         if let Some(init) = init { |         if let Some(init) = init { | ||||||
|             self.visit_expr(init)?; |             let init = init.interpret(env)?; | ||||||
|             let init = self.pop()?; |             env.insert(name, Some(init)); | ||||||
|             self.scope.insert(name, Some(init)); |  | ||||||
|         } else { |         } else { | ||||||
|             self.scope.insert(name, None); |             env.insert(name, None); | ||||||
|         } |         } | ||||||
|         Ok(()) |         Ok(ConValue::Empty) | ||||||
|     } |     } | ||||||
|  | } | ||||||
|     fn visit_fn_decl(&mut self, function: &FnDecl) -> IResult<()> { | impl Interpret for FnDecl { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         // register the function in the current environment |         // register the function in the current environment | ||||||
|         self.scope.insert_fn(function); |         env.insert_fn(self); | ||||||
|         Ok(()) |         Ok(ConValue::Empty) | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_block(&mut self, block: &expression::Block) -> IResult<()> { |  | ||||||
|         for stmt in &block.statements { |  | ||||||
|             self.visit_statement(stmt)?; |  | ||||||
|         } |  | ||||||
|         if let Some(expr) = block.expr.as_ref() { |  | ||||||
|             self.visit_expr(expr) |  | ||||||
|         } else { |  | ||||||
|             self.push(ConValue::Empty); |  | ||||||
|             Ok(()) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | impl Interpret for Expr { | ||||||
|     fn visit_tuple(&mut self, tuple: &Tuple) -> IResult<()> { |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         let mut out = vec![]; |         self.0.interpret(env) | ||||||
|         for expr in &tuple.elements { |  | ||||||
|             self.visit_expr(expr)?; |  | ||||||
|             out.push(self.pop()?); |  | ||||||
|     } |     } | ||||||
|         self.push(out); |  | ||||||
|         Ok(()) |  | ||||||
| } | } | ||||||
|  | impl Interpret for Operation { | ||||||
|     fn visit_fn_call(&mut self, call: &FnCall) -> IResult<()> { |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         // evaluate the callee |         match self { | ||||||
|         self.visit_primary(&call.callee)?; |             Operation::Assign(op) => op.interpret(env), | ||||||
|         for args in &call.args { |             Operation::Binary(op) => op.interpret(env), | ||||||
|             self.visit_tuple(args)?; |             Operation::Unary(op) => op.interpret(env), | ||||||
|             let (ConValue::Tuple(args), callee) = self.pop_two()? else { |             Operation::Call(op) => op.interpret(env), | ||||||
|                 Err(Error::TypeError)? |  | ||||||
|             }; |  | ||||||
|             let return_value = callee.call(self, &args)?; |  | ||||||
|             self.push(return_value); |  | ||||||
|         } |         } | ||||||
|         Ok(()) |  | ||||||
|     } |     } | ||||||
|  | } | ||||||
|     fn visit_assign(&mut self, assign: &math::Assign) -> IResult<()> { | impl Interpret for Assign { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         use operator::Assign; |         use operator::Assign; | ||||||
|         let math::Assign { target, operator, init } = assign; |         let math::Assign { target, operator, init } = self; | ||||||
|         self.visit_operation(init)?; |         let init = init.interpret(env)?; | ||||||
|         let init = self.pop()?; |         let resolved = env.get_mut(&target.name)?; | ||||||
|         let resolved = self.scope.get_mut(&target.name)?; |  | ||||||
|  |  | ||||||
|         if let Assign::Assign = operator { |         if let Assign::Assign = operator { | ||||||
|             use std::mem::discriminant as variant; |             use std::mem::discriminant as variant; | ||||||
| @@ -418,8 +365,7 @@ impl Visitor<IResult<()>> for Interpreter { | |||||||
|                 None => *resolved = Some(init), |                 None => *resolved = Some(init), | ||||||
|                 _ => Err(Error::TypeError)?, |                 _ => Err(Error::TypeError)?, | ||||||
|             } |             } | ||||||
|             self.push(ConValue::Empty); |             return Ok(ConValue::Empty); | ||||||
|             return Ok(()); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let Some(target) = resolved.as_mut() else { |         let Some(target) = resolved.as_mut() else { | ||||||
| @@ -439,98 +385,65 @@ impl Visitor<IResult<()>> for Interpreter { | |||||||
|             Assign::ShrAssign => target.shr_assign(init)?, |             Assign::ShrAssign => target.shr_assign(init)?, | ||||||
|             _ => (), |             _ => (), | ||||||
|         } |         } | ||||||
|  |         Ok(ConValue::Empty) | ||||||
|         self.push(ConValue::Empty); |  | ||||||
|  |  | ||||||
|         Ok(()) |  | ||||||
|     } |     } | ||||||
|  | } | ||||||
|     fn visit_binary(&mut self, bin: &math::Binary) -> IResult<()> { | impl Interpret for Binary { | ||||||
|         let Binary { first, other } = bin; |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         let Binary { first, other } = self; | ||||||
|         self.visit_operation(first)?; |         let mut first = first.interpret(env)?; | ||||||
|         for (op, other) in other { |         for (op, other) in other { | ||||||
|             match op { |             first = match op { | ||||||
|                 operator::Binary::LogAnd => { |                 operator::Binary::LogAnd => { | ||||||
|                     if self.peek()?.truthy()? { |                     if first.truthy()? { | ||||||
|                         self.pop()?; |                         other.interpret(env) | ||||||
|                         self.visit_operation(other)?; |                     } else { | ||||||
|  |                         return Ok(first); // Short circuiting | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 operator::Binary::LogOr => { |                 operator::Binary::LogOr => { | ||||||
|                     if !self.peek()?.truthy()? { |                     if !first.truthy()? { | ||||||
|                         self.pop()?; |                         other.interpret(env) | ||||||
|                         self.visit_operation(other)?; |                     } else { | ||||||
|  |                         return Ok(first); // Short circuiting | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 operator::Binary::LogXor => { |                 operator::Binary::LogXor => { | ||||||
|                     let first = self.pop()?.truthy()?; |                     // TODO: It should be possible to assemble better error information from this | ||||||
|                     self.visit_operation(other)?; |                     let (lhs, rhs) = (first.truthy()?, other.interpret(env)?.truthy()?); | ||||||
|                     let second = self.pop()?.truthy()?; |                     Ok(ConValue::Bool(lhs ^ rhs)) | ||||||
|                     self.push(first ^ second); |  | ||||||
|                 } |                 } | ||||||
|                 _ => { |                 // TODO: For all overloadable operators, transmute into function call | ||||||
|                     self.visit_operation(other)?; |                 operator::Binary::Mul => first * other.interpret(env)?, | ||||||
|                     self.visit_binary_op(op)?; |                 operator::Binary::Div => first / other.interpret(env)?, | ||||||
|  |                 operator::Binary::Rem => first % other.interpret(env)?, | ||||||
|  |                 operator::Binary::Add => first + other.interpret(env)?, | ||||||
|  |                 operator::Binary::Sub => first - other.interpret(env)?, | ||||||
|  |                 operator::Binary::Lsh => first << other.interpret(env)?, | ||||||
|  |                 operator::Binary::Rsh => first >> other.interpret(env)?, | ||||||
|  |                 operator::Binary::BitAnd => first & other.interpret(env)?, | ||||||
|  |                 operator::Binary::BitOr => first | other.interpret(env)?, | ||||||
|  |                 operator::Binary::BitXor => first ^ other.interpret(env)?, | ||||||
|  |                 operator::Binary::RangeExc => first.range_exc(other.interpret(env)?), | ||||||
|  |                 operator::Binary::RangeInc => first.range_inc(other.interpret(env)?), | ||||||
|  |                 operator::Binary::Less => first.lt(&other.interpret(env)?), | ||||||
|  |                 operator::Binary::LessEq => first.lt_eq(&other.interpret(env)?), | ||||||
|  |                 operator::Binary::Equal => first.eq(&other.interpret(env)?), | ||||||
|  |                 operator::Binary::NotEq => first.neq(&other.interpret(env)?), | ||||||
|  |                 operator::Binary::GreaterEq => first.gt_eq(&other.interpret(env)?), | ||||||
|  |                 operator::Binary::Greater => first.gt(&other.interpret(env)?), | ||||||
|  |             }?; | ||||||
|  |         } | ||||||
|  |         Ok(first) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|         } | impl Interpret for Unary { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         Ok(()) |         let Unary { operand, operators } = self; | ||||||
|     } |         let mut operand = operand.interpret(env)?; | ||||||
|  |  | ||||||
|     fn visit_unary(&mut self, unary: &math::Unary) -> IResult<()> { |  | ||||||
|         let Unary { operand, operators } = unary; |  | ||||||
|  |  | ||||||
|         self.visit_operation(operand)?; |  | ||||||
|  |  | ||||||
|         for op in operators.iter().rev() { |         for op in operators.iter().rev() { | ||||||
|             self.visit_unary_op(op)?; |             operand = match op { | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_assign_op(&mut self, _: &operator::Assign) -> IResult<()> { |  | ||||||
|         unimplemented!("visit_assign_op is implemented in visit_operation") |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_binary_op(&mut self, op: &operator::Binary) -> IResult<()> { |  | ||||||
|         use operator::Binary; |  | ||||||
|         let (second, first) = self.pop_two()?; |  | ||||||
|  |  | ||||||
|         let out = match op { |  | ||||||
|             Binary::Mul => first * second, |  | ||||||
|             Binary::Div => first / second, |  | ||||||
|             Binary::Rem => first % second, |  | ||||||
|             Binary::Add => first + second, |  | ||||||
|             Binary::Sub => first - second, |  | ||||||
|             Binary::Lsh => first << second, |  | ||||||
|             Binary::Rsh => first >> second, |  | ||||||
|             Binary::BitAnd => first & second, |  | ||||||
|             Binary::BitOr => first | second, |  | ||||||
|             Binary::BitXor => first ^ second, |  | ||||||
|             Binary::LogAnd | Binary::LogOr | Binary::LogXor => { |  | ||||||
|                 unimplemented!("Implemented in visit_operation") |  | ||||||
|             } |  | ||||||
|             Binary::RangeExc => first.range_exc(second), |  | ||||||
|             Binary::RangeInc => first.range_inc(second), |  | ||||||
|             Binary::Less => first.lt(&second), |  | ||||||
|             Binary::LessEq => first.lt_eq(&second), |  | ||||||
|             Binary::Equal => first.eq(&second), |  | ||||||
|             Binary::NotEq => first.neq(&second), |  | ||||||
|             Binary::GreaterEq => first.gt_eq(&second), |  | ||||||
|             Binary::Greater => first.gt(&second), |  | ||||||
|         }?; |  | ||||||
|  |  | ||||||
|         self.push(out); |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_unary_op(&mut self, op: &operator::Unary) -> IResult<()> { |  | ||||||
|         let operand = self.pop()?; |  | ||||||
|  |  | ||||||
|         self.push(match op { |  | ||||||
|                 operator::Unary::RefRef => todo!(), |                 operator::Unary::RefRef => todo!(), | ||||||
|                 operator::Unary::Ref => todo!(), |                 operator::Unary::Ref => todo!(), | ||||||
|                 operator::Unary::Deref => todo!(), |                 operator::Unary::Deref => todo!(), | ||||||
| @@ -542,159 +455,181 @@ impl Visitor<IResult<()>> for Interpreter { | |||||||
|                     operand |                     operand | ||||||
|                 } |                 } | ||||||
|                 operator::Unary::Tilde => todo!(), |                 operator::Unary::Tilde => todo!(), | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_if(&mut self, expr: &If) -> IResult<()> { |  | ||||||
|         self.visit_expr(&expr.cond)?; |  | ||||||
|         if self.pop()?.truthy()? { |  | ||||||
|             self.visit_block(&expr.body)?; |  | ||||||
|         } else if let Some(block) = &expr.else_ { |  | ||||||
|             self.visit_else(block)?; |  | ||||||
|         } else { |  | ||||||
|             self.push(ConValue::Empty) |  | ||||||
|         } |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_while(&mut self, expr: &While) -> IResult<()> { |  | ||||||
|         while { |  | ||||||
|             self.visit_expr(&expr.cond)?; |  | ||||||
|             self.pop()?.truthy()? |  | ||||||
|         } { |  | ||||||
|             let Err(out) = self.visit_block(&expr.body) else { |  | ||||||
|                 // Every expression returns a value. If allowed to pile up, they'll overflow the |  | ||||||
|                 // stack. |  | ||||||
|                 self.pop()?; |  | ||||||
|                 continue; |  | ||||||
|             }; |             }; | ||||||
|             match out { |  | ||||||
|                 Error::Continue => continue, |  | ||||||
|                 Error::Break(value) => { |  | ||||||
|                     self.push(value); |  | ||||||
|                     return Ok(()); |  | ||||||
|         } |         } | ||||||
|                 r => Err(r)?, |         Ok(operand) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|         if let Some(r#else) = &expr.else_ { | impl Interpret for Call { | ||||||
|             self.visit_else(r#else)?; |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         } else { |         match self { | ||||||
|             self.push(ConValue::Empty); |             Call::FnCall(fncall) => fncall.interpret(env), | ||||||
|  |             Call::Primary(primary) => primary.interpret(env), | ||||||
|         } |         } | ||||||
|         Ok(()) |  | ||||||
|     } |     } | ||||||
|  | } | ||||||
|     fn visit_for(&mut self, expr: &control::For) -> IResult<()> { | impl Interpret for FnCall { | ||||||
|         self.scope.enter(); |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|         self.visit_expr(&expr.iter)?; |         // evaluate the callee | ||||||
|         let bounds = match self.pop()? { |         let mut callee = self.callee.interpret(env)?; | ||||||
|             ConValue::RangeExc(a, b) | ConValue::RangeInc(a, b) => (a, b), |         for args in &self.args { | ||||||
|             _ => Err(Error::NotIterable)?, |             let ConValue::Tuple(args) = args.interpret(env)? else { | ||||||
|  |                 Err(Error::TypeError)? | ||||||
|             }; |             }; | ||||||
|         for loop_var in bounds.0..=bounds.1 { |             callee = callee.call(env, &args)?; | ||||||
|             self.scope.insert(&expr.var.name, Some(loop_var.into())); |  | ||||||
|             let Err(out) = self.visit_block(&expr.body) else { |  | ||||||
|                 self.pop()?; |  | ||||||
|                 continue; |  | ||||||
|             }; |  | ||||||
|             match out { |  | ||||||
|                 Error::Continue => continue, |  | ||||||
|                 Error::Break(value) => { |  | ||||||
|                     self.push(value); |  | ||||||
|                     return Ok(()); |  | ||||||
|         } |         } | ||||||
|                 r => Err(r)?, |         Ok(callee) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|         if let Some(r#else) = &expr.else_ { | impl Interpret for Primary { | ||||||
|             self.visit_else(r#else)?; |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         match self { | ||||||
|  |             Primary::Identifier(prim) => prim.interpret(env), | ||||||
|  |             Primary::Literal(prim) => prim.interpret(env), | ||||||
|  |             Primary::Block(prim) => prim.interpret(env), | ||||||
|  |             Primary::Group(prim) => prim.interpret(env), | ||||||
|  |             Primary::Branch(prim) => prim.interpret(env), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Identifier { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         env.resolve(&self.name) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Literal { | ||||||
|  |     fn interpret(&self, _env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         Ok(match self { | ||||||
|  |             Literal::String(value) => ConValue::from(value.as_str()), | ||||||
|  |             Literal::Char(value) => ConValue::Char(*value), | ||||||
|  |             Literal::Bool(value) => ConValue::Bool(*value), | ||||||
|  |             Literal::Float(value) => todo!("Float values in interpreter: {value:?}"), | ||||||
|  |             Literal::Int(value) => ConValue::Int(*value as _), | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Block { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         let mut env = env.frame("block"); | ||||||
|  |         // TODO: this could TOTALLY be done with a binary operator. | ||||||
|  |         for stmt in &self.statements { | ||||||
|  |             stmt.interpret(&mut env)?; | ||||||
|  |         } | ||||||
|  |         if let Some(expr) = self.expr.as_ref() { | ||||||
|  |             expr.interpret(&mut env) | ||||||
|         } else { |         } else { | ||||||
|             self.push(ConValue::Empty) |             Ok(ConValue::Empty) | ||||||
|         } |         } | ||||||
|         self.scope.exit()?; |  | ||||||
|         Ok(()) |  | ||||||
|     } |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Group { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         match self { | ||||||
|  |             Group::Tuple(tuple) => tuple.interpret(env), | ||||||
|  |             Group::Single(value) => value.interpret(env), | ||||||
|  |             Group::Empty => Ok(ConValue::Empty), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Tuple { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         Ok(ConValue::Tuple(self.elements.iter().try_fold( | ||||||
|  |             vec![], | ||||||
|  |             |mut out, element| { | ||||||
|  |                 out.push(element.interpret(env)?); | ||||||
|  |                 Ok(out) | ||||||
|  |             }, | ||||||
|  |         )?)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Flow { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         match self { | ||||||
|  |             Flow::While(flow) => flow.interpret(env), | ||||||
|  |             Flow::If(flow) => flow.interpret(env), | ||||||
|  |             Flow::For(flow) => flow.interpret(env), | ||||||
|  |             Flow::Continue(flow) => flow.interpret(env), | ||||||
|  |             Flow::Return(flow) => flow.interpret(env), | ||||||
|  |             Flow::Break(flow) => flow.interpret(env), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for While { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         while self.cond.interpret(env)?.truthy()? { | ||||||
|  |             match self.body.interpret(env) { | ||||||
|  |                 Err(Error::Break(value)) => return Ok(value), | ||||||
|  |                 Err(Error::Continue) => continue, | ||||||
|  |                 e => e?, | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |         if let Some(other) = &self.else_ { | ||||||
|  |             other.interpret(env) | ||||||
|  |         } else { | ||||||
|  |             Ok(ConValue::Empty) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Else { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         self.expr.interpret(env) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for If { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         if self.cond.interpret(env)?.truthy()? { | ||||||
|  |             self.body.interpret(env) | ||||||
|  |         } else if let Some(other) = &self.else_ { | ||||||
|  |             other.interpret(env) | ||||||
|  |         } else { | ||||||
|  |             Ok(ConValue::Empty) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for For { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         let bounds = match self.iter.interpret(env)? { | ||||||
|  |             ConValue::RangeExc(a, b) => a..=b, | ||||||
|  |             ConValue::RangeInc(a, b) => a..=b, | ||||||
|  |             _ => Err(Error::TypeError)?, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|     fn visit_else(&mut self, else_: &control::Else) -> IResult<()> { |         let mut env = env.frame("loop variable"); | ||||||
|         self.visit_expr(&else_.expr) |         for loop_var in bounds { | ||||||
|  |             env.insert(&self.var.name, Some(loop_var.into())); | ||||||
|  |             match self.body.interpret(&mut env) { | ||||||
|  |                 Err(Error::Break(value)) => return Ok(value), | ||||||
|  |                 Err(Error::Continue) => continue, | ||||||
|  |                 result => result?, | ||||||
|  |             }; | ||||||
|         } |         } | ||||||
|  |         if let Some(other) = &self.else_ { | ||||||
|     fn visit_continue(&mut self, _: &control::Continue) -> IResult<()> { |             other.interpret(&mut env) | ||||||
|         Err(Error::cnt()) |         } else { | ||||||
|     } |             Ok(ConValue::Empty) | ||||||
|  |  | ||||||
|     fn visit_break(&mut self, brk: &control::Break) -> IResult<()> { |  | ||||||
|         Err(Error::brk({ |  | ||||||
|             self.visit_expr(&brk.expr)?; |  | ||||||
|             self.pop()? |  | ||||||
|         })) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_return(&mut self, ret: &control::Return) -> IResult<()> { |  | ||||||
|         Err(Error::ret({ |  | ||||||
|             self.visit_expr(&ret.expr)?; |  | ||||||
|             self.pop()? |  | ||||||
|         })) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_identifier(&mut self, ident: &Identifier) -> IResult<()> { |  | ||||||
|         let value = self.resolve(&ident.name)?; |  | ||||||
|         self.push(value); |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_string_literal(&mut self, string: &str) -> IResult<()> { |  | ||||||
|         self.push(string); |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_char_literal(&mut self, char: &char) -> IResult<()> { |  | ||||||
|         self.push(*char); |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_bool_literal(&mut self, bool: &bool) -> IResult<()> { |  | ||||||
|         self.push(*bool); |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_float_literal(&mut self, float: &literal::Float) -> IResult<()> { |  | ||||||
|         todo!("visit floats in interpreter: {float:?}") |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_int_literal(&mut self, int: &u128) -> IResult<()> { |  | ||||||
|         self.push((*int) as i128); |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn visit_empty(&mut self) -> IResult<()> { |  | ||||||
|         self.push(()); |  | ||||||
|         Ok(()) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | } | ||||||
| impl Default for Interpreter { | impl Interpret for Continue { | ||||||
|     fn default() -> Self { |     fn interpret(&self, _env: &mut Environment) -> IResult<ConValue> { | ||||||
|         Self { scope: Environment::new().into(), stack: Default::default() } |         Err(Error::Continue) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Return { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         Err(Error::Return(self.expr.interpret(env)?)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | impl Interpret for Break { | ||||||
|  |     fn interpret(&self, env: &mut Environment) -> IResult<ConValue> { | ||||||
|  |         Err(Error::Return(self.expr.interpret(env)?)) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub mod function { | pub mod function { | ||||||
|     //! Represents a block of code which lives inside the Interpreter |     //! Represents a block of code which lives inside the Interpreter | ||||||
|     use super::{ |     use super::{Callable, ConValue, Environment, Error, FnDecl, IResult, Identifier, Interpret}; | ||||||
|         // scope::Environment, |     use crate::ast::preamble::Name; | ||||||
|         Callable, |  | ||||||
|         ConValue, |  | ||||||
|         Error, |  | ||||||
|         FnDecl, |  | ||||||
|         IResult, |  | ||||||
|         Identifier, |  | ||||||
|         Interpreter, |  | ||||||
|     }; |  | ||||||
|     use crate::ast::{preamble::Name, visitor::Visitor}; |  | ||||||
|     /// Represents a block of code which persists inside the Interpreter |     /// Represents a block of code which persists inside the Interpreter | ||||||
|     #[derive(Clone, Debug)] |     #[derive(Clone, Debug)] | ||||||
|     pub struct Function { |     pub struct Function { | ||||||
| @@ -716,9 +651,9 @@ pub mod function { | |||||||
|  |  | ||||||
|     impl Callable for Function { |     impl Callable for Function { | ||||||
|         fn name(&self) -> &str { |         fn name(&self) -> &str { | ||||||
|             &self.declaration.name.name.name |             &self.declaration.name.symbol.name | ||||||
|         } |         } | ||||||
|         fn call(&self, interpreter: &mut Interpreter, args: &[ConValue]) -> IResult<ConValue> { |         fn call(&self, env: &mut Environment, args: &[ConValue]) -> IResult<ConValue> { | ||||||
|             // Check arg mapping |             // Check arg mapping | ||||||
|             if args.len() != self.declaration.args.len() { |             if args.len() != self.declaration.args.len() { | ||||||
|                 return Err(Error::ArgNumber { |                 return Err(Error::ArgNumber { | ||||||
| @@ -727,20 +662,18 @@ pub mod function { | |||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
|             // TODO: Isolate cross-function scopes! |             // TODO: Isolate cross-function scopes! | ||||||
|             interpreter.scope.enter(); |  | ||||||
|             for (Name { name: Identifier { name, .. }, .. }, value) in |             let mut frame = env.frame("function args"); | ||||||
|  |             for (Name { symbol: Identifier { name, .. }, .. }, value) in | ||||||
|                 self.declaration.args.iter().zip(args) |                 self.declaration.args.iter().zip(args) | ||||||
|             { |             { | ||||||
|                 interpreter.scope.insert(name, Some(value.clone())); |                 frame.insert(name, Some(value.clone())); | ||||||
|             } |             } | ||||||
|             match interpreter.visit_block(&self.declaration.body) { |             match self.declaration.body.interpret(&mut frame) { | ||||||
|                 Err(Error::Return(value)) => interpreter.push(value), |                 Err(Error::Return(value)) => Ok(value), | ||||||
|                 Err(Error::Break(value)) => Err(Error::BadBreak(value))?, |                 Err(Error::Break(value)) => Err(Error::BadBreak(value)), | ||||||
|                 Err(e) => Err(e)?, |                 result => result, | ||||||
|                 Ok(()) => (), |  | ||||||
|             } |             } | ||||||
|             interpreter.scope.exit()?; |  | ||||||
|             interpreter.pop() |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -748,7 +681,7 @@ pub mod function { | |||||||
| pub mod builtin { | pub mod builtin { | ||||||
|     mod builtin_imports { |     mod builtin_imports { | ||||||
|         pub use crate::interpreter::{ |         pub use crate::interpreter::{ | ||||||
|             error::IResult, temp_type_impl::ConValue, BuiltIn, Callable, Interpreter, |             env::Environment, error::IResult, temp_type_impl::ConValue, BuiltIn, Callable, | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|     use super::BuiltIn; |     use super::BuiltIn; | ||||||
| @@ -765,7 +698,7 @@ pub mod builtin { | |||||||
|         #[rustfmt::skip] |         #[rustfmt::skip] | ||||||
|         impl Callable for Print { |         impl Callable for Print { | ||||||
|             fn name(&self) -> &'static str { "print" } |             fn name(&self) -> &'static str { "print" } | ||||||
|             fn call(&self, _inter: &mut Interpreter, args: &[ConValue]) -> IResult<ConValue> { |             fn call(&self, _inter: &mut Environment, args: &[ConValue]) -> IResult<ConValue> { | ||||||
|                 for arg in args { |                 for arg in args { | ||||||
|                     print!("{arg}") |                     print!("{arg}") | ||||||
|                 } |                 } | ||||||
| @@ -783,22 +716,20 @@ pub mod builtin { | |||||||
|         #[rustfmt::skip] |         #[rustfmt::skip] | ||||||
|         impl Callable for Dbg { |         impl Callable for Dbg { | ||||||
|             fn name(&self) -> &str { "dbg" } |             fn name(&self) -> &str { "dbg" } | ||||||
|             fn call(&self, _inter: &mut Interpreter, args: &[ConValue]) -> IResult<ConValue> { |             fn call(&self, _inter: &mut Environment, args: &[ConValue]) -> IResult<ConValue> { | ||||||
|                 println!("{args:?}"); |                 println!("{args:?}"); | ||||||
|                 Ok(args.into()) |                 Ok(args.into()) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     mod dump { |     mod dump { | ||||||
|         use super::builtin_imports::*; |         use super::builtin_imports::*; | ||||||
|         #[derive(Clone, Debug)] |         #[derive(Clone, Debug)] | ||||||
|         pub struct Dump; |         pub struct Dump; | ||||||
|         impl BuiltIn for Dump {} |         impl BuiltIn for Dump {} | ||||||
|         impl Callable for Dump { |         impl Callable for Dump { | ||||||
|             fn call(&self, interpreter: &mut Interpreter, _args: &[ConValue]) -> IResult<ConValue> { |             fn call(&self, env: &mut Environment, _args: &[ConValue]) -> IResult<ConValue> { | ||||||
|                 println!("Scope:\n{}", interpreter.scope); |                 println!("{}", *env); | ||||||
|                 println!("Stack:{:#?}", interpreter.stack); |  | ||||||
|                 Ok(ConValue::Empty) |                 Ok(ConValue::Empty) | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -809,7 +740,7 @@ pub mod builtin { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub mod scope { | pub mod env { | ||||||
|     //! Lexical and non-lexical scoping for variables |     //! Lexical and non-lexical scoping for variables | ||||||
|  |  | ||||||
|     use super::{ |     use super::{ | ||||||
| @@ -817,52 +748,76 @@ pub mod scope { | |||||||
|         error::{Error, IResult}, |         error::{Error, IResult}, | ||||||
|         function::Function, |         function::Function, | ||||||
|         temp_type_impl::ConValue, |         temp_type_impl::ConValue, | ||||||
|         FnDecl, |         Callable, FnDecl, Interpret, | ||||||
|  |     }; | ||||||
|  |     use std::{ | ||||||
|  |         collections::HashMap, | ||||||
|  |         fmt::Display, | ||||||
|  |         ops::{Deref, DerefMut}, | ||||||
|     }; |     }; | ||||||
|     use std::{collections::HashMap, fmt::Display}; |  | ||||||
|  |  | ||||||
|     /// Implements a nested lexical scope |     /// Implements a nested lexical scope | ||||||
|     #[derive(Clone, Debug, Default)] |     #[derive(Clone, Debug)] | ||||||
|     pub struct Environment { |     pub struct Environment { | ||||||
|         outer: Option<Box<Self>>, |         outer: Option<Box<Self>>, | ||||||
|         vars: HashMap<String, Option<ConValue>>, |         vars: HashMap<String, Option<ConValue>>, | ||||||
|  |         name: &'static str, | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     impl Display for Environment { |     impl Display for Environment { | ||||||
|         fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { |         fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||||
|  |             writeln!(f, "--- '{}' ---", self.name)?; | ||||||
|             for (var, val) in &self.vars { |             for (var, val) in &self.vars { | ||||||
|                 writeln!(f, "{var}: {}", if val.is_some() { "..." } else { "None" })?; |                 write!(f, "{var}: ")?; | ||||||
|  |                 match val { | ||||||
|  |                     Some(value) => writeln!(f, "{value}"), | ||||||
|  |                     None => writeln!(f, "<undefined>"), | ||||||
|  |                 }? | ||||||
|             } |             } | ||||||
|             "--- Frame ---\n".fmt(f)?; |  | ||||||
|             if let Some(outer) = &self.outer { |             if let Some(outer) = &self.outer { | ||||||
|                 outer.fmt(f)?; |                 outer.fmt(f)?; | ||||||
|             } |             } | ||||||
|             Ok(()) |             Ok(()) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     impl Default for Environment { | ||||||
|  |         fn default() -> Self { | ||||||
|  |             let mut outer = Self::no_builtins("builtins"); | ||||||
|  |             for &builtin in DEFAULT_BUILTINS { | ||||||
|  |                 outer.insert(builtin.name(), Some(ConValue::BuiltIn(builtin))); | ||||||
|  |             } | ||||||
|  |             Self { outer: Some(Box::new(outer)), vars: Default::default(), name: "base" } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     impl Environment { |     impl Environment { | ||||||
|         pub fn new() -> Self { |         pub fn new() -> Self { | ||||||
|             let mut out = Self::default(); |             Self::default() | ||||||
|             for &builtin in DEFAULT_BUILTINS { |  | ||||||
|                 out.insert(builtin.name(), Some(ConValue::BuiltIn(builtin))) |  | ||||||
|         } |         } | ||||||
|             out |         fn no_builtins(name: &'static str) -> Self { | ||||||
|  |             Self { outer: None, vars: Default::default(), name } | ||||||
|         } |         } | ||||||
|         /// Enter a nested scope |         /// Temporary function | ||||||
|         pub fn enter(&mut self) { |         #[deprecated] | ||||||
|             let outer = std::mem::take(self); |         pub fn resolve(&mut self, name: &str) -> IResult<ConValue> { | ||||||
|             self.outer = Some(outer.into()); |             self.get(name).cloned() | ||||||
|         } |         } | ||||||
|         /// Exits the scope, destroying all local variables and |  | ||||||
|         /// returning the outer scope, if there is one |         pub fn eval(&mut self, node: &impl Interpret) -> IResult<ConValue> { | ||||||
|         pub fn exit(&mut self) -> IResult<()> { |             node.interpret(self) | ||||||
|             if let Some(outer) = std::mem::take(&mut self.outer) { |  | ||||||
|                 *self = *outer; |  | ||||||
|                 Ok(()) |  | ||||||
|             } else { |  | ||||||
|                 Err(Error::ScopeExit) |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         /// Calls a function inside the interpreter's scope, | ||||||
|  |         /// and returns the result | ||||||
|  |         pub fn call(&mut self, name: &str, args: &[ConValue]) -> IResult<ConValue> { | ||||||
|  |             let function = self.resolve(name)?; | ||||||
|  |             function.call(self, args) | ||||||
|  |         } | ||||||
|  |         /// Enters a nested scope, returning a [`Frame`] stack-guard. | ||||||
|  |         /// | ||||||
|  |         /// [`Frame`] implements Deref/DerefMut for [`Environment`]. | ||||||
|  |         pub fn frame(&mut self, name: &'static str) -> Frame { | ||||||
|  |             Frame::new(self, name) | ||||||
|         } |         } | ||||||
|         /// Resolves a variable mutably |         /// Resolves a variable mutably | ||||||
|         /// |         /// | ||||||
| @@ -894,11 +849,62 @@ pub mod scope { | |||||||
|         /// A convenience function for registering a [FnDecl] as a [Function] |         /// A convenience function for registering a [FnDecl] as a [Function] | ||||||
|         pub fn insert_fn(&mut self, decl: &FnDecl) { |         pub fn insert_fn(&mut self, decl: &FnDecl) { | ||||||
|             self.vars.insert( |             self.vars.insert( | ||||||
|                 decl.name.name.name.clone(), |                 decl.name.symbol.name.clone(), | ||||||
|                 Some(Function::new(decl).into()), |                 Some(Function::new(decl).into()), | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Functions which aid in the implementation of [`Frame`] | ||||||
|  |     impl Environment { | ||||||
|  |         /// Enters a scope, creating a new namespace for variables | ||||||
|  |         fn enter(&mut self, name: &'static str) -> &mut Self { | ||||||
|  |             let outer = std::mem::replace(self, Environment::no_builtins(name)); | ||||||
|  |             self.outer = Some(outer.into()); | ||||||
|  |             self | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /// Exits the scope, destroying all local variables and | ||||||
|  |         /// returning the outer scope, if there is one | ||||||
|  |         fn exit(&mut self) -> &mut Self { | ||||||
|  |             if let Some(outer) = std::mem::take(&mut self.outer) { | ||||||
|  |                 *self = *outer; | ||||||
|  |             } | ||||||
|  |             self | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Represents a stack frame | ||||||
|  |     #[derive(Debug)] | ||||||
|  |     pub struct Frame<'scope> { | ||||||
|  |         scope: &'scope mut Environment, | ||||||
|  |     } | ||||||
|  |     impl<'scope> Frame<'scope> { | ||||||
|  |         fn new(scope: &'scope mut Environment, name: &'static str) -> Self { | ||||||
|  |             Self { scope: scope.enter(name) } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     impl<'scope> Deref for Frame<'scope> { | ||||||
|  |         type Target = Environment; | ||||||
|  |         fn deref(&self) -> &Self::Target { | ||||||
|  |             self.scope | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     impl<'scope> DerefMut for Frame<'scope> { | ||||||
|  |         fn deref_mut(&mut self) -> &mut Self::Target { | ||||||
|  |             self.scope | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     impl<'scope> Drop for Frame<'scope> { | ||||||
|  |         fn drop(&mut self) { | ||||||
|  |             self.scope.exit(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub mod module { | ||||||
|  |     //! Implements non-lexical "global" scoping rules | ||||||
|  |      | ||||||
| } | } | ||||||
|  |  | ||||||
| pub mod error { | pub mod error { | ||||||
|   | |||||||
| @@ -425,7 +425,7 @@ impl Parser { | |||||||
|         } else { |         } else { | ||||||
|             None |             None | ||||||
|         }; |         }; | ||||||
|         Ok(FnDecl { name: Name { name, mutable: false, ty }, args, body: self.block()? }) |         Ok(FnDecl { name: Name { symbol: name, mutable: false, ty }, args, body: self.block()? }) | ||||||
|     } |     } | ||||||
|     /// Parses a [parameter](Name) list for [FnDecl] |     /// Parses a [parameter](Name) list for [FnDecl] | ||||||
|     fn params(&mut self) -> PResult<Vec<Name>> { |     fn params(&mut self) -> PResult<Vec<Name>> { | ||||||
| @@ -442,7 +442,7 @@ impl Parser { | |||||||
|     fn name(&mut self) -> PResult<Name> { |     fn name(&mut self) -> PResult<Name> { | ||||||
|         Ok(Name { |         Ok(Name { | ||||||
|             mutable: self.keyword(Keyword::Mut).is_ok(), |             mutable: self.keyword(Keyword::Mut).is_ok(), | ||||||
|             name: self.identifier()?, |             symbol: self.identifier()?, | ||||||
|             ty: self |             ty: self | ||||||
|                 .consume_type(Type::Colon) |                 .consume_type(Type::Colon) | ||||||
|                 .and_then(|this| this.type_expr()) |                 .and_then(|this| this.type_expr()) | ||||||
| @@ -473,7 +473,7 @@ impl Parser { | |||||||
|             Type::Keyword(Keyword::SelfKw) => { |             Type::Keyword(Keyword::SelfKw) => { | ||||||
|                 self.keyword(Keyword::SelfKw).map(|_| PathPart::PathSelf) |                 self.keyword(Keyword::SelfKw).map(|_| PathPart::PathSelf) | ||||||
|             } |             } | ||||||
|             e => Err(Error::not_path_segment(e)) |             e => Err(Error::not_path_segment(e)), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -108,7 +108,7 @@ impl PrettyPrintable for Stmt { | |||||||
| } | } | ||||||
| impl PrettyPrintable for Let { | impl PrettyPrintable for Let { | ||||||
|     fn visit<W: Write>(&self, p: &mut Printer<W>) -> IOResult<()> { |     fn visit<W: Write>(&self, p: &mut Printer<W>) -> IOResult<()> { | ||||||
|         let Let { name: Name { name, mutable, ty }, init } = self; |         let Let { name: Name { symbol: name, mutable, ty }, init } = self; | ||||||
|         p.put("let")?.space()?; |         p.put("let")?.space()?; | ||||||
|         if *mutable { |         if *mutable { | ||||||
|             p.put("mut")?.space()?; |             p.put("mut")?.space()?; | ||||||
| @@ -144,7 +144,7 @@ impl PrettyPrintable for Name { | |||||||
|         if self.mutable { |         if self.mutable { | ||||||
|             p.put("mut")?.space()?; |             p.put("mut")?.space()?; | ||||||
|         } |         } | ||||||
|         self.name.visit(p)?; |         self.symbol.visit(p)?; | ||||||
|         if let Some(ty) = &self.ty { |         if let Some(ty) = &self.ty { | ||||||
|             ty.visit(p.put(':')?.space()?)?; |             ty.visit(p.put(':')?.space()?)?; | ||||||
|         } |         } | ||||||
| @@ -168,7 +168,9 @@ impl PrettyPrintable for Path { | |||||||
|             p.put("::")?; |             p.put("::")?; | ||||||
|         } |         } | ||||||
|         for (idx, part) in parts.iter().enumerate() { |         for (idx, part) in parts.iter().enumerate() { | ||||||
|             if idx != 0 { p.put("::")?;} |             if idx != 0 { | ||||||
|  |                 p.put("::")?; | ||||||
|  |             } | ||||||
|             part.visit(p)?; |             part.visit(p)?; | ||||||
|         } |         } | ||||||
|         Ok(()) |         Ok(()) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user