Started primitives
This commit is contained in:
parent
7a33804ffe
commit
b3bdc41a3e
11
demo.lang
11
demo.lang
|
@ -1,6 +1,9 @@
|
||||||
i := 0;
|
i := 0x0;
|
||||||
|
|
||||||
while i < 10 {
|
if i > 10 {
|
||||||
i = i + 1;
|
print "a";
|
||||||
print i;
|
} else if i > 5 {
|
||||||
|
print "b";
|
||||||
|
} else {
|
||||||
|
print "c";
|
||||||
}
|
}
|
||||||
|
|
35
src/err.rs
35
src/err.rs
|
@ -8,9 +8,9 @@ pub fn error<T>() -> Result<T> {
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct Diagnostic {
|
pub struct Diagnostic {
|
||||||
reason: String,
|
pub reason: String,
|
||||||
span: Option<Span>,
|
pub span: Option<Span>,
|
||||||
backtrace: Vec<String>,
|
pub backtrace: Vec<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Diagnostic {
|
impl Diagnostic {
|
||||||
|
@ -30,8 +30,8 @@ impl std::fmt::Display for Diagnostic {
|
||||||
} else {
|
} else {
|
||||||
write!(f, "(E) {}\n", self.reason)?;
|
write!(f, "(E) {}\n", self.reason)?;
|
||||||
}
|
}
|
||||||
for b in self.backtrace.iter().rev() {
|
for (_i, b) in self.backtrace.iter().enumerate() {
|
||||||
write!(f, "--> {}\n", b)?;
|
write!(f, "> {}\n", b)?;
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -75,11 +75,11 @@ impl From<std::num::ParseFloatError> for Diagnostic {
|
||||||
pub trait IntoDiagnostic<T, S: Into<String>> {
|
pub trait IntoDiagnostic<T, S: Into<String>> {
|
||||||
fn reason(self, s: S) -> Result<T>;
|
fn reason(self, s: S) -> Result<T>;
|
||||||
fn trace(self, s: S) -> Result<T>;
|
fn trace(self, s: S) -> Result<T>;
|
||||||
|
fn trace_span(self, span: Span, s: S) -> Result<T>;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait WithSpan<T> {
|
pub trait WithSpan<T> {
|
||||||
fn span(self, span: &Span) -> Result<T>;
|
fn span(self, span: &Span) -> Result<T>;
|
||||||
fn no_span(self) -> Result<T>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> WithSpan<T> for Result<T> {
|
impl<T> WithSpan<T> for Result<T> {
|
||||||
|
@ -89,17 +89,6 @@ impl<T> WithSpan<T> for Result<T> {
|
||||||
e
|
e
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn no_span(self) -> Result<T> {
|
|
||||||
self.map_err(|mut e| {
|
|
||||||
e.span = None;
|
|
||||||
e
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait CoerceDiagnostic<T> {
|
|
||||||
fn coerce(self) -> Result<T>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T, S: Into<String>> IntoDiagnostic<T, S> for Option<T> {
|
impl<T, S: Into<String>> IntoDiagnostic<T, S> for Option<T> {
|
||||||
|
@ -124,6 +113,10 @@ impl<T, S: Into<String>> IntoDiagnostic<T, S> for Option<T> {
|
||||||
}),
|
}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn trace_span(self, span: Span, s: S) -> Result<T> {
|
||||||
|
self.trace(format!("{} {}", span, s.into()))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T, E: Into<Diagnostic>, S: Into<String>> IntoDiagnostic<T, S>
|
impl<T, E: Into<Diagnostic>, S: Into<String>> IntoDiagnostic<T, S>
|
||||||
|
@ -131,7 +124,9 @@ impl<T, E: Into<Diagnostic>, S: Into<String>> IntoDiagnostic<T, S>
|
||||||
{
|
{
|
||||||
fn reason(self, s: S) -> Result<T> {
|
fn reason(self, s: S) -> Result<T> {
|
||||||
self.map_err(|e| e.into()).map_err(|mut e| {
|
self.map_err(|e| e.into()).map_err(|mut e| {
|
||||||
|
if e.reason == "" {
|
||||||
e.reason = s.into();
|
e.reason = s.into();
|
||||||
|
}
|
||||||
e
|
e
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -142,10 +137,8 @@ impl<T, E: Into<Diagnostic>, S: Into<String>> IntoDiagnostic<T, S>
|
||||||
e
|
e
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, E: Into<Diagnostic>> CoerceDiagnostic<T> for std::result::Result<T, E> {
|
fn trace_span(self, span: Span, s: S) -> Result<T> {
|
||||||
fn coerce(self) -> Result<T> {
|
self.trace(format!("{} {}", span, s.into())).span(&span)
|
||||||
self.map_err(|e| e.into())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
67
src/frontend.rs
Normal file
67
src/frontend.rs
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
err::*, treewalk::Interpreter, Parser, Statement, StatementKind, Token,
|
||||||
|
Tokenizer,
|
||||||
|
};
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct Module {
|
||||||
|
file_name: String,
|
||||||
|
source: String,
|
||||||
|
ast: Vec<Statement>,
|
||||||
|
errors: Vec<Diagnostic>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Module {
|
||||||
|
pub fn from_file(path: impl AsRef<Path>) -> Result<Self> {
|
||||||
|
let file = std::fs::read(&path).trace(format!(
|
||||||
|
"while attempting to open file '{}'",
|
||||||
|
&path.as_ref().display()
|
||||||
|
))?;
|
||||||
|
let source = String::from_utf8_lossy(&file);
|
||||||
|
Ok(Self::from_string(
|
||||||
|
format!("{}", path.as_ref().display()),
|
||||||
|
source.into(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_string(file_name: String, source: String) -> Self {
|
||||||
|
let tokens = Tokenizer::new(source.chars()).filter(|t| t.0.is_meaningful());
|
||||||
|
let mut ast = vec![];
|
||||||
|
let mut errors = vec![];
|
||||||
|
for statement in Parser::new(tokens) {
|
||||||
|
use StatementKind as s;
|
||||||
|
match statement.kind {
|
||||||
|
s::Error(e) => errors.push(e),
|
||||||
|
_ => ast.push(statement),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
file_name,
|
||||||
|
source: source.into(),
|
||||||
|
ast,
|
||||||
|
errors,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn errors(&self) -> &[Diagnostic] {
|
||||||
|
&self.errors
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn ok(&self) -> bool {
|
||||||
|
self.errors.len() == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn execute(&self) {
|
||||||
|
if !self.ok() {
|
||||||
|
for e in self.errors() {
|
||||||
|
eprintln!("{e}");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Interpreter::new(self.ast.clone().into_iter())
|
||||||
|
.run()
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
}
|
|
@ -34,10 +34,6 @@ where
|
||||||
s
|
s
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn inner(&self) -> &I {
|
|
||||||
&self.iterator
|
|
||||||
}
|
|
||||||
|
|
||||||
fn normalize(&mut self) {
|
fn normalize(&mut self) {
|
||||||
for item in &mut self.buffer {
|
for item in &mut self.buffer {
|
||||||
if self.exhausted {
|
if self.exhausted {
|
||||||
|
|
38
src/main.rs
38
src/main.rs
|
@ -1,14 +1,16 @@
|
||||||
mod err;
|
mod err;
|
||||||
|
mod frontend;
|
||||||
mod lookahead;
|
mod lookahead;
|
||||||
mod parse;
|
mod parse;
|
||||||
mod token;
|
mod token;
|
||||||
mod treewalk;
|
mod treewalk;
|
||||||
|
mod types;
|
||||||
use std::ops::Add;
|
use std::ops::Add;
|
||||||
|
|
||||||
|
use err::*;
|
||||||
use lookahead::*;
|
use lookahead::*;
|
||||||
use parse::*;
|
use parse::*;
|
||||||
use token::*;
|
use token::*;
|
||||||
use treewalk::Interpreter;
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
#[derive(Clone, Copy, Debug)]
|
||||||
pub struct Span {
|
pub struct Span {
|
||||||
|
@ -16,13 +18,20 @@ pub struct Span {
|
||||||
pub column: usize,
|
pub column: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Display for Span {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(f, "({}:{})", self.row, self.column)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Add<Span> for Span {
|
impl Add<Span> for Span {
|
||||||
type Output = Span;
|
type Output = Span;
|
||||||
|
|
||||||
fn add(self, rhs: Span) -> Self::Output {
|
fn add(self, rhs: Span) -> Self::Output {
|
||||||
|
let max = (self.row, self.column).max((rhs.row, rhs.column));
|
||||||
Span {
|
Span {
|
||||||
row: usize::min(self.row, rhs.row),
|
row: max.0,
|
||||||
column: usize::max(self.column, rhs.column),
|
column: max.1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -45,13 +54,18 @@ fn test_tokenization() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn test_expression(expr: &str) {
|
||||||
let src = include_str!("../demo.lang");
|
let source = expr.to_string();
|
||||||
println!("{src}");
|
let tokens = Tokenizer::new(source.chars()).filter(|t| t.0.is_meaningful());
|
||||||
let tokens: Vec<_> = Tokenizer::new(src.chars())
|
let mut parser = Parser::new(tokens);
|
||||||
.filter(|t| t.0.is_meaningful())
|
println!("{:?}", parser.expression(0).unwrap());
|
||||||
.collect();
|
}
|
||||||
let parsed = Parser::new(tokens.into_iter()).file().unwrap();
|
|
||||||
let mut interp = Interpreter::new(parsed.into_iter());
|
fn main() -> Result<()> {
|
||||||
interp.run().unwrap();
|
/*
|
||||||
|
let module = frontend::Module::from_file("./demo.lang")?;
|
||||||
|
module.execute();
|
||||||
|
*/
|
||||||
|
test_expression("(a: b) -> c {}");
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
453
src/parse.rs
453
src/parse.rs
|
@ -1,453 +0,0 @@
|
||||||
use crate::err::*;
|
|
||||||
use crate::{Span, Token, TokenKind};
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub enum ExpressionKind {
|
|
||||||
Integer(i64),
|
|
||||||
Real(f64),
|
|
||||||
String(String),
|
|
||||||
Boolean(bool),
|
|
||||||
Identifier(String),
|
|
||||||
Binary {
|
|
||||||
token: TokenKind,
|
|
||||||
left: Box<Expression>,
|
|
||||||
right: Box<Expression>,
|
|
||||||
},
|
|
||||||
Unary {
|
|
||||||
token: TokenKind,
|
|
||||||
child: Box<Expression>,
|
|
||||||
},
|
|
||||||
Parenthesis(Box<Expression>),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct Expression {
|
|
||||||
pub kind: ExpressionKind,
|
|
||||||
pub span: Span,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Expression {
|
|
||||||
pub fn new(kind: ExpressionKind, span: Span) -> Self {
|
|
||||||
Self { kind, span }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl std::fmt::Debug for ExpressionKind {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
use ExpressionKind as e;
|
|
||||||
match self {
|
|
||||||
e::Integer(i) => write!(f, "{i}"),
|
|
||||||
e::Binary { token, left, right } => {
|
|
||||||
write!(f, "({left:?} {token:?} {right:?})")
|
|
||||||
},
|
|
||||||
e::Parenthesis(inner) => write!(f, "{inner:?}"),
|
|
||||||
e::Unary { token, child } => {
|
|
||||||
write!(f, "({token:?} {child:?})")
|
|
||||||
},
|
|
||||||
e::Real(fp) => write!(f, "{fp}"),
|
|
||||||
e::String(s) => write!(f, r#""{s}""#),
|
|
||||||
e::Identifier(i) => write!(f, "{i}"),
|
|
||||||
e::Boolean(b) => write!(f, "{b}"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl std::fmt::Debug for Expression {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
write!(f, "{:?}", self.kind)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub enum StatementKind {
|
|
||||||
Mutable {
|
|
||||||
name: String,
|
|
||||||
type_: Option<String>,
|
|
||||||
value: Option<Expression>,
|
|
||||||
},
|
|
||||||
Immutable {
|
|
||||||
name: String,
|
|
||||||
type_: Option<String>,
|
|
||||||
value: Expression,
|
|
||||||
},
|
|
||||||
Assignment {
|
|
||||||
name: String,
|
|
||||||
value: Expression,
|
|
||||||
},
|
|
||||||
If {
|
|
||||||
predicate: Expression,
|
|
||||||
block: Vec<Statement>,
|
|
||||||
else_: Option<Box<Statement>>,
|
|
||||||
},
|
|
||||||
While {
|
|
||||||
predicate: Expression,
|
|
||||||
block: Vec<Statement>,
|
|
||||||
},
|
|
||||||
Print(Expression),
|
|
||||||
Expression(Expression),
|
|
||||||
Block(Vec<Statement>),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub struct Statement {
|
|
||||||
pub kind: StatementKind,
|
|
||||||
pub span: Span,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub type Precedence = usize;
|
|
||||||
|
|
||||||
fn binary_prec(tok: &TokenKind) -> Result<(Precedence, bool)> {
|
|
||||||
use TokenKind::*;
|
|
||||||
Ok(match tok {
|
|
||||||
Star | Slash | Percent => (10, false),
|
|
||||||
Plus | Minus => (9, false),
|
|
||||||
And | Nand => (8, false),
|
|
||||||
Xor | Xnor => (7, false),
|
|
||||||
Or | Nor => (6, false),
|
|
||||||
DoubleEqual | BangEqual | Less | LessEqual | Greater | GreaterEqual => {
|
|
||||||
(5, false)
|
|
||||||
},
|
|
||||||
//Colon => Some((5, false)),
|
|
||||||
_ => {
|
|
||||||
return error()
|
|
||||||
.reason(format!("{:?} is not a valid binary operator", tok));
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn unary_prefix_prec(tok: &TokenKind) -> Result<Precedence> {
|
|
||||||
use TokenKind::*;
|
|
||||||
Ok(match tok {
|
|
||||||
Minus | Not => 11,
|
|
||||||
Break => 3,
|
|
||||||
_ => {
|
|
||||||
return error()
|
|
||||||
.reason(format!("{tok:?} is not a valid prefix unary operator"));
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn unary_postfix_prec(tok: &TokenKind) -> Result<Precedence> {
|
|
||||||
use TokenKind::*;
|
|
||||||
Ok(match tok {
|
|
||||||
Question => 12,
|
|
||||||
Bang => 12,
|
|
||||||
_ => {
|
|
||||||
return error()
|
|
||||||
.reason(format!("{tok:?} is not a valid postfix unary operator"));
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const PARSER_LOOKAHEAD: usize = 3;
|
|
||||||
|
|
||||||
type TokenIter<I> = crate::Window<PARSER_LOOKAHEAD, Token, I>;
|
|
||||||
|
|
||||||
pub struct Parser<I: Iterator<Item = Token>> {
|
|
||||||
iter: TokenIter<I>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<I: Iterator<Item = Token>> Parser<I> {
|
|
||||||
pub fn new(iter: I) -> Self {
|
|
||||||
Self {
|
|
||||||
iter: TokenIter::new(iter),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn skip(&mut self, n: usize) {
|
|
||||||
for _ in 0..n {
|
|
||||||
let _ = self.next();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn next(&mut self) -> Result<Token> {
|
|
||||||
self.iter.next().reason("Unexpected end of file")
|
|
||||||
}
|
|
||||||
|
|
||||||
fn peek(&self, n: usize) -> Result<Token> {
|
|
||||||
self.iter.peek(n).clone().reason("Unexpected end of file")
|
|
||||||
}
|
|
||||||
|
|
||||||
fn eat(&mut self, expect: TokenKind) -> Result<Token> {
|
|
||||||
match self.look(expect) {
|
|
||||||
Ok(t) => {
|
|
||||||
self.skip(1);
|
|
||||||
Ok(t)
|
|
||||||
},
|
|
||||||
Err(e) => Err(e),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn look(&mut self, expect: TokenKind) -> Result<Token> {
|
|
||||||
let next = self.peek(0)?;
|
|
||||||
if next.0 == expect {
|
|
||||||
Ok(next)
|
|
||||||
} else {
|
|
||||||
error()
|
|
||||||
.reason(format!("Expected {expect:?}, found {:?}", next.0))
|
|
||||||
.span(&next.1)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn file(&mut self) -> Result<Vec<Statement>> {
|
|
||||||
use TokenKind as t;
|
|
||||||
let mut statements = vec![];
|
|
||||||
loop {
|
|
||||||
// Trim extra ;
|
|
||||||
while self.eat(t::Semicolon).is_ok() {}
|
|
||||||
if self.eat(t::EOF).is_ok() {
|
|
||||||
return Ok(statements);
|
|
||||||
}
|
|
||||||
match self.statement() {
|
|
||||||
Ok(s) => statements.push(s),
|
|
||||||
Err(e) => {
|
|
||||||
return Err(e);
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn statement(&mut self) -> Result<Statement> {
|
|
||||||
use StatementKind as s;
|
|
||||||
use TokenKind as t;
|
|
||||||
let next = self.peek(0);
|
|
||||||
let next2 = self.peek(1);
|
|
||||||
let statement = match (next, next2) {
|
|
||||||
// (im)mutable declaration
|
|
||||||
(Ok(Token(t::Identifier(name), span)), Ok(Token(t::Colon, _))) => {
|
|
||||||
self.skip(2);
|
|
||||||
let type_ = match self.eat(t::Identifier("".into())) {
|
|
||||||
Ok(Token(t::Identifier(s), _)) => Some(s),
|
|
||||||
_ => None,
|
|
||||||
};
|
|
||||||
match self.eat(t::Equal).or_else(|_| self.eat(t::Colon)) {
|
|
||||||
Ok(Token(t::Colon, _)) => Statement {
|
|
||||||
kind: s::Immutable {
|
|
||||||
name,
|
|
||||||
type_,
|
|
||||||
value: self
|
|
||||||
.expression(0)
|
|
||||||
.trace("while parsing immutable declaration")?,
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
},
|
|
||||||
Ok(Token(t::Equal, _)) => Statement {
|
|
||||||
kind: s::Mutable {
|
|
||||||
name,
|
|
||||||
type_,
|
|
||||||
value: Some(
|
|
||||||
self
|
|
||||||
.expression(0)
|
|
||||||
.trace("while parsing mutable declaration")?,
|
|
||||||
),
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
},
|
|
||||||
_ => return error().reason("Expected expression here"),
|
|
||||||
}
|
|
||||||
},
|
|
||||||
(Ok(Token(t::Identifier(name), span)), Ok(Token(t::Equal, _))) => {
|
|
||||||
self.skip(2);
|
|
||||||
let value = self
|
|
||||||
.expression(0)
|
|
||||||
.trace("while parsing assignment expression")?;
|
|
||||||
Statement {
|
|
||||||
kind: s::Assignment { name, value },
|
|
||||||
span,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// If
|
|
||||||
(Ok(Token(t::If, span)), _) => {
|
|
||||||
self.skip(1);
|
|
||||||
let predicate = self
|
|
||||||
.expression(0)
|
|
||||||
.reason("Expected predicate after 'if' keyword")
|
|
||||||
.span(&span)?;
|
|
||||||
let block = self.block().trace("while parsing if statement")?;
|
|
||||||
return Ok(Statement {
|
|
||||||
span,
|
|
||||||
kind: s::If {
|
|
||||||
predicate,
|
|
||||||
block: block.0,
|
|
||||||
else_: None,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
},
|
|
||||||
// While
|
|
||||||
(Ok(Token(t::While, span)), _) => {
|
|
||||||
self.skip(1);
|
|
||||||
let predicate = self
|
|
||||||
.expression(0)
|
|
||||||
.reason("Expected predicate after 'while' keyword")
|
|
||||||
.span(&span)?;
|
|
||||||
let block = self.block().trace("while parsing while statement")?;
|
|
||||||
return Ok(Statement {
|
|
||||||
span,
|
|
||||||
kind: s::While {
|
|
||||||
predicate,
|
|
||||||
block: block.0,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
},
|
|
||||||
// (DEBUG) print
|
|
||||||
(Ok(Token(t::Print, span)), _) => {
|
|
||||||
self.skip(1);
|
|
||||||
let expr = self.expression(0).trace("while parsing print statement")?;
|
|
||||||
Statement {
|
|
||||||
span: span + expr.span,
|
|
||||||
kind: s::Print(expr),
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// Block
|
|
||||||
(Ok(Token(t::LeftBrace, _)), _) => {
|
|
||||||
// Skip check for semicolon
|
|
||||||
let (block, span) =
|
|
||||||
self.block().trace("while parsing block statement")?;
|
|
||||||
return Ok(Statement {
|
|
||||||
kind: s::Block(block),
|
|
||||||
span,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
// Expression
|
|
||||||
_ => {
|
|
||||||
let expr = self
|
|
||||||
.expression(0)
|
|
||||||
.trace("while parsing expression statement")?;
|
|
||||||
Statement {
|
|
||||||
span: expr.span,
|
|
||||||
kind: s::Expression(expr),
|
|
||||||
}
|
|
||||||
},
|
|
||||||
};
|
|
||||||
// Check for semicolon
|
|
||||||
if self.eat(t::Semicolon).is_ok() {
|
|
||||||
Ok(statement)
|
|
||||||
} else {
|
|
||||||
error().reason("Expected ;")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn expression(
|
|
||||||
&mut self,
|
|
||||||
mut precedence: Precedence,
|
|
||||||
) -> Result<Expression> {
|
|
||||||
use ExpressionKind as e;
|
|
||||||
use TokenKind as t;
|
|
||||||
let next = self.peek(0)?;
|
|
||||||
// Unary prefix expression
|
|
||||||
let mut current = if let Ok(p) = unary_prefix_prec(&next.0) {
|
|
||||||
let operator = self.next().expect("unreachable");
|
|
||||||
let child = self
|
|
||||||
.expression(p)
|
|
||||||
.trace(format!("while parsing unary {:?}", operator.0))
|
|
||||||
.span(&operator.1)?;
|
|
||||||
let span = child.span + operator.1;
|
|
||||||
Expression::new(
|
|
||||||
e::Unary {
|
|
||||||
token: operator.0,
|
|
||||||
child: child.into(),
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
// Terminal or paren
|
|
||||||
else {
|
|
||||||
self.primary()?
|
|
||||||
};
|
|
||||||
// Precedence climbing loop
|
|
||||||
while let Ok(next) = self.peek(0) {
|
|
||||||
// Binary infix
|
|
||||||
if let Ok((new_precedence, left_assoc)) = binary_prec(&next.0) {
|
|
||||||
if (!left_assoc && new_precedence <= precedence)
|
|
||||||
|| (new_precedence < precedence)
|
|
||||||
{
|
|
||||||
return Ok(current);
|
|
||||||
}
|
|
||||||
let operator = self.next().expect("unreachable");
|
|
||||||
let rhs = self
|
|
||||||
.expression(new_precedence)
|
|
||||||
.trace(format!("while parsing binary {:?}", operator.0))
|
|
||||||
.span(&operator.1)?;
|
|
||||||
let span = next.1 + rhs.span;
|
|
||||||
current = Expression::new(
|
|
||||||
e::Binary {
|
|
||||||
token: operator.0,
|
|
||||||
left: current.into(),
|
|
||||||
right: rhs.into(),
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
// Unary postfix
|
|
||||||
else if let Ok(new_precedence) = unary_postfix_prec(&next.0) {
|
|
||||||
let operator = self.next().expect("unreachable");
|
|
||||||
let span = next.1 + operator.1;
|
|
||||||
precedence = new_precedence;
|
|
||||||
current = Expression::new(
|
|
||||||
e::Unary {
|
|
||||||
token: operator.0,
|
|
||||||
child: current.into(),
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(current)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn primary(&mut self) -> Result<Expression> {
|
|
||||||
use ExpressionKind as e;
|
|
||||||
use TokenKind as t;
|
|
||||||
let next = self.peek(0)?;
|
|
||||||
let span = next.1;
|
|
||||||
let kind = match next.0 {
|
|
||||||
t::IntegerLiteral(i) => e::Integer(i),
|
|
||||||
t::FloatLiteral(f) => e::Real(f),
|
|
||||||
t::StringLiteral(s) => e::String(s),
|
|
||||||
t::True => e::Boolean(true),
|
|
||||||
t::Identifier(i) => e::Identifier(i),
|
|
||||||
t::LeftParen => {
|
|
||||||
self.eat(t::LeftParen).expect("unreachable");
|
|
||||||
let expr = self
|
|
||||||
.expression(0)
|
|
||||||
.trace("while parsing parenthesized expression")?;
|
|
||||||
self
|
|
||||||
.look(t::RightParen)
|
|
||||||
.reason("Unclosed '('")
|
|
||||||
.span(&expr.span)?;
|
|
||||||
e::Parenthesis(expr.into())
|
|
||||||
},
|
|
||||||
_ => {
|
|
||||||
return error()
|
|
||||||
.span(&span)
|
|
||||||
.reason(format!("Expected primary, found {:?}", next.0));
|
|
||||||
},
|
|
||||||
};
|
|
||||||
self.skip(1);
|
|
||||||
Ok(Expression { kind, span })
|
|
||||||
}
|
|
||||||
|
|
||||||
fn block(&mut self) -> Result<(Vec<Statement>, Span)> {
|
|
||||||
use TokenKind as t;
|
|
||||||
let mut span = self.eat(t::LeftBrace).reason("Expected block")?.1;
|
|
||||||
let mut statements = vec![];
|
|
||||||
loop {
|
|
||||||
let next = self.peek(0)?;
|
|
||||||
span = span + next.1;
|
|
||||||
match self.eat(t::RightBrace) {
|
|
||||||
Ok(t) => {
|
|
||||||
span = span + t.1;
|
|
||||||
break;
|
|
||||||
},
|
|
||||||
_ => {
|
|
||||||
let statement = self.statement()?;
|
|
||||||
span = span + statement.span;
|
|
||||||
statements.push(statement);
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
Ok((statements, span))
|
|
||||||
}
|
|
||||||
}
|
|
304
src/parse/expression.rs
Normal file
304
src/parse/expression.rs
Normal file
|
@ -0,0 +1,304 @@
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Parameter {
|
||||||
|
name: String,
|
||||||
|
type_: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for Parameter {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(f, "{}: {}", self.name, self.type_)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub enum ExpressionKind {
|
||||||
|
Integer(i64),
|
||||||
|
Real(f64),
|
||||||
|
String(String),
|
||||||
|
Boolean(bool),
|
||||||
|
Identifier(String),
|
||||||
|
Binary {
|
||||||
|
token: TokenKind,
|
||||||
|
left: Box<Expression>,
|
||||||
|
right: Box<Expression>,
|
||||||
|
},
|
||||||
|
Unary {
|
||||||
|
token: TokenKind,
|
||||||
|
child: Box<Expression>,
|
||||||
|
},
|
||||||
|
Parenthesis(Box<Expression>),
|
||||||
|
Function {
|
||||||
|
params: Vec<Parameter>,
|
||||||
|
returns: Option<String>,
|
||||||
|
body: Vec<Statement>,
|
||||||
|
},
|
||||||
|
Call {
|
||||||
|
callee: Box<Expression>,
|
||||||
|
args: Vec<Expression>,
|
||||||
|
},
|
||||||
|
Field {
|
||||||
|
namespace: Box<Expression>,
|
||||||
|
field: Box<Expression>,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Expression {
|
||||||
|
pub kind: ExpressionKind,
|
||||||
|
pub span: Span,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Expression {
|
||||||
|
pub fn new(kind: ExpressionKind, span: Span) -> Self {
|
||||||
|
Self { kind, span }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for ExpressionKind {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
use ExpressionKind as e;
|
||||||
|
match self {
|
||||||
|
e::Integer(i) => write!(f, "{i}"),
|
||||||
|
e::Binary { token, left, right } => {
|
||||||
|
write!(f, "({left:?} {token:?} {right:?})")
|
||||||
|
},
|
||||||
|
e::Parenthesis(inner) => write!(f, "{inner:?}"),
|
||||||
|
e::Unary { token, child } => {
|
||||||
|
write!(f, "({token:?} {child:?})")
|
||||||
|
},
|
||||||
|
e::Real(fp) => write!(f, "{fp}"),
|
||||||
|
e::String(s) => write!(f, r#""{s}""#),
|
||||||
|
e::Identifier(i) => write!(f, "{i}"),
|
||||||
|
e::Boolean(b) => write!(f, "{b}"),
|
||||||
|
e::Call { callee, args } => write!(f, "({callee:?} call {args:?})"),
|
||||||
|
e::Field { namespace, field } => {
|
||||||
|
write!(f, "({namespace:?} . {field:?})")
|
||||||
|
},
|
||||||
|
e::Function {
|
||||||
|
params, returns, ..
|
||||||
|
} => {
|
||||||
|
write!(f, "(fn({params:?}) -> {returns:?})")
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for Expression {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(f, "{:?}", self.kind)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<I: Iterator<Item = Token>> Parser<I> {
|
||||||
|
pub fn expression(&mut self, precedence: Precedence) -> Result<Expression> {
|
||||||
|
use ExpressionKind as e;
|
||||||
|
use TokenKind as t;
|
||||||
|
let next = self.peek(0)?;
|
||||||
|
// Unary prefix expression
|
||||||
|
let mut current = if let Ok(p) = unary_prefix_prec(&next.0) {
|
||||||
|
let operator = self.next_tok().expect("unreachable");
|
||||||
|
let child = self
|
||||||
|
.expression(p)
|
||||||
|
.trace(format!("while parsing unary {}", operator.0))
|
||||||
|
.span(&operator.1)?;
|
||||||
|
let span = child.span + operator.1;
|
||||||
|
Expression::new(
|
||||||
|
e::Unary {
|
||||||
|
token: operator.0,
|
||||||
|
child: child.into(),
|
||||||
|
},
|
||||||
|
span,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// Terminal or paren
|
||||||
|
else {
|
||||||
|
self.primary().reason("Expected expression")?
|
||||||
|
};
|
||||||
|
|
||||||
|
// Precedence climbing loop
|
||||||
|
while let Ok(next) = self.peek(0) {
|
||||||
|
// Binary infix
|
||||||
|
if let Ok((new_precedence, left_assoc)) = binary_prec(&next.0) {
|
||||||
|
if (!left_assoc && new_precedence <= precedence)
|
||||||
|
|| (new_precedence < precedence)
|
||||||
|
{
|
||||||
|
return Ok(current);
|
||||||
|
}
|
||||||
|
let operator = self.next_tok().expect("unreachable");
|
||||||
|
let rhs = self
|
||||||
|
.expression(new_precedence)
|
||||||
|
.trace(format!("while parsing binary {}", operator.0))
|
||||||
|
.span(&operator.1)?;
|
||||||
|
let span = next.1 + rhs.span;
|
||||||
|
current = Expression::new(
|
||||||
|
e::Binary {
|
||||||
|
token: operator.0,
|
||||||
|
left: current.into(),
|
||||||
|
right: rhs.into(),
|
||||||
|
},
|
||||||
|
span,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// Field
|
||||||
|
else if let Token(t::Dot, span) = next {
|
||||||
|
if FIELD_PREC <= precedence {
|
||||||
|
return Ok(current);
|
||||||
|
}
|
||||||
|
self.skip(1);
|
||||||
|
let field = self
|
||||||
|
.expression(FIELD_PREC)
|
||||||
|
.trace_span(span, "in field expression")?;
|
||||||
|
current = Expression::new(
|
||||||
|
e::Field {
|
||||||
|
namespace: current.into(),
|
||||||
|
field: field.into(),
|
||||||
|
},
|
||||||
|
span,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// Function call
|
||||||
|
else if let Token(t::LeftParen, mut span) = next {
|
||||||
|
if CALL_PREC <= precedence {
|
||||||
|
return Ok(current);
|
||||||
|
}
|
||||||
|
self.skip(1);
|
||||||
|
let mut args = vec![];
|
||||||
|
loop {
|
||||||
|
match self.expression(0) {
|
||||||
|
Ok(a) => {
|
||||||
|
span = span + a.span;
|
||||||
|
args.push(a)
|
||||||
|
},
|
||||||
|
Err(_) => break,
|
||||||
|
};
|
||||||
|
if !self.eat(t::Comma).is_ok() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let Token(_, span2) = self.eat(t::RightParen).span(&span)?;
|
||||||
|
current = Expression::new(
|
||||||
|
e::Call {
|
||||||
|
callee: current.into(),
|
||||||
|
args,
|
||||||
|
},
|
||||||
|
span + span2,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// Unary postfix
|
||||||
|
else if let Ok(_) = unary_postfix_prec(&next.0) {
|
||||||
|
let operator = self.next_tok().expect("unreachable");
|
||||||
|
let span = next.1 + operator.1;
|
||||||
|
current = Expression::new(
|
||||||
|
e::Unary {
|
||||||
|
token: operator.0,
|
||||||
|
child: current.into(),
|
||||||
|
},
|
||||||
|
span,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(current)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn primary(&mut self) -> Result<Expression> {
|
||||||
|
use ExpressionKind as e;
|
||||||
|
use TokenKind as t;
|
||||||
|
let next = self.peek(0)?;
|
||||||
|
let mut span = next.1;
|
||||||
|
let kind = match next.0 {
|
||||||
|
t::IntegerLiteral(i) => e::Integer(i),
|
||||||
|
t::FloatLiteral(f) => e::Real(f),
|
||||||
|
t::StringLiteral(s) => e::String(s),
|
||||||
|
t::True => e::Boolean(true),
|
||||||
|
t::Identifier(i) => e::Identifier(i),
|
||||||
|
// function
|
||||||
|
t::LeftParen
|
||||||
|
if (self.look(1, t::Identifier("".into())).is_ok()
|
||||||
|
&& self.look(2, t::Colon).is_ok())
|
||||||
|
|| self.look(1, t::RightParen).is_ok() =>
|
||||||
|
{
|
||||||
|
self.skip(1);
|
||||||
|
let mut params = vec![];
|
||||||
|
loop {
|
||||||
|
let (name, span2) = match self.identifier() {
|
||||||
|
Ok((name, span)) => (name, span),
|
||||||
|
Err(_) => break,
|
||||||
|
};
|
||||||
|
span = span + span2;
|
||||||
|
self
|
||||||
|
.eat(t::Colon)
|
||||||
|
.trace_span(span, "while parsing function parameter type")?;
|
||||||
|
let (type_, span2) = self
|
||||||
|
.identifier()
|
||||||
|
.trace_span(span, "while parsing function parameter type")?;
|
||||||
|
span = span + span2;
|
||||||
|
params.push(Parameter { name, type_ });
|
||||||
|
if !self.eat(t::Comma).is_ok() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let Token(_, span2) = self
|
||||||
|
.eat(t::RightParen)
|
||||||
|
.span(&span)
|
||||||
|
.trace_span(span, "while parsing function definition")?;
|
||||||
|
let returns = if let Ok(Token(_, span2)) = self.eat(t::Arrow) {
|
||||||
|
span = span + span2;
|
||||||
|
let (identifier, span2) = self
|
||||||
|
.identifier()
|
||||||
|
.trace_span(span, "while parsing function return type")?;
|
||||||
|
span = span + span2;
|
||||||
|
Some(identifier)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
span = span + span2;
|
||||||
|
let (body, span2) = self
|
||||||
|
.block()
|
||||||
|
.trace_span(span, "while parsing function body")?;
|
||||||
|
span = span + span2;
|
||||||
|
e::Function {
|
||||||
|
params,
|
||||||
|
returns,
|
||||||
|
body,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// parenthetical
|
||||||
|
t::LeftParen => {
|
||||||
|
self.skip(1);
|
||||||
|
let expr = self
|
||||||
|
.expression(0)
|
||||||
|
.trace("while parsing parenthesized expression")?;
|
||||||
|
self
|
||||||
|
.look(0, t::RightParen)
|
||||||
|
.reason("Unclosed '('")
|
||||||
|
.span(&expr.span)?;
|
||||||
|
e::Parenthesis(expr.into())
|
||||||
|
},
|
||||||
|
_ => {
|
||||||
|
return error()
|
||||||
|
.span(&span)
|
||||||
|
.reason(format!("Expected expression, found {}", next.0));
|
||||||
|
},
|
||||||
|
};
|
||||||
|
self.skip(1);
|
||||||
|
Ok(Expression { kind, span })
|
||||||
|
}
|
||||||
|
|
||||||
|
fn identifier(&mut self) -> Result<(String, Span)> {
|
||||||
|
use TokenKind as t;
|
||||||
|
match self.peek(0) {
|
||||||
|
Ok(Token(t::Identifier(i), span)) => {
|
||||||
|
self.skip(1);
|
||||||
|
Ok((i, span))
|
||||||
|
},
|
||||||
|
Ok(t) => error()
|
||||||
|
.reason(format!("Expected identifier, found {}", t.0))
|
||||||
|
.span(&t.1),
|
||||||
|
Err(e) => Err(e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
165
src/parse/mod.rs
Normal file
165
src/parse/mod.rs
Normal file
|
@ -0,0 +1,165 @@
|
||||||
|
mod expression;
|
||||||
|
mod statement;
|
||||||
|
pub use expression::*;
|
||||||
|
pub use statement::*;
|
||||||
|
|
||||||
|
use crate::err::*;
|
||||||
|
use crate::{Span, Token, TokenKind};
|
||||||
|
|
||||||
|
pub type Precedence = usize;
|
||||||
|
|
||||||
|
const FIELD_PREC: Precedence = 13;
|
||||||
|
const CALL_PREC: Precedence = 12;
|
||||||
|
|
||||||
|
fn binary_prec(tok: &TokenKind) -> Result<(Precedence, bool)> {
|
||||||
|
use TokenKind::*;
|
||||||
|
Ok(match tok {
|
||||||
|
Star | Slash | Percent => (10, false),
|
||||||
|
Plus | Minus => (9, false),
|
||||||
|
And | Nand => (8, false),
|
||||||
|
Xor | Xnor => (7, false),
|
||||||
|
Or | Nor => (6, false),
|
||||||
|
DoubleEqual | BangEqual | Less | LessEqual | Greater | GreaterEqual => {
|
||||||
|
(5, false)
|
||||||
|
},
|
||||||
|
//Colon => Some((5, false)),
|
||||||
|
_ => {
|
||||||
|
return error().reason(format!("{} is not a valid binary operator", tok));
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unary_prefix_prec(tok: &TokenKind) -> Result<Precedence> {
|
||||||
|
use TokenKind::*;
|
||||||
|
Ok(match tok {
|
||||||
|
Minus | Not => 11,
|
||||||
|
Break => 3,
|
||||||
|
_ => {
|
||||||
|
return error()
|
||||||
|
.reason(format!("{tok} is not a valid prefix unary operator"));
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unary_postfix_prec(tok: &TokenKind) -> Result<Precedence> {
|
||||||
|
use TokenKind::*;
|
||||||
|
Ok(match tok {
|
||||||
|
Question => 12,
|
||||||
|
Bang => 12,
|
||||||
|
_ => {
|
||||||
|
return error()
|
||||||
|
.reason(format!("{tok} is not a valid postfix unary operator"));
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const PARSER_LOOKAHEAD: usize = 3;
|
||||||
|
|
||||||
|
type TokenIter<I> = crate::Window<PARSER_LOOKAHEAD, Token, I>;
|
||||||
|
|
||||||
|
pub struct Parser<I: Iterator<Item = Token>> {
|
||||||
|
iter: TokenIter<I>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<I: Iterator<Item = Token>> Iterator for Parser<I> {
|
||||||
|
type Item = Statement;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
use StatementKind as s;
|
||||||
|
use TokenKind as t;
|
||||||
|
// Trim extra ;
|
||||||
|
while self.eat(t::Semicolon).is_ok() {}
|
||||||
|
loop {
|
||||||
|
if self.eat(t::EOF).is_ok() || self.iter.finished {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
match self.statement() {
|
||||||
|
Ok(s) => return Some(s),
|
||||||
|
Err(e) => {
|
||||||
|
loop {
|
||||||
|
let next = self.next_tok();
|
||||||
|
match next {
|
||||||
|
Ok(Token(t::Semicolon | t::RightBrace | t::EOF, _)) => break,
|
||||||
|
_ => {},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Some(Statement {
|
||||||
|
span: e.span.unwrap_or(Span { row: 0, column: 0 }),
|
||||||
|
kind: s::Error(e),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<I: Iterator<Item = Token>> Parser<I> {
|
||||||
|
pub fn new(iter: I) -> Self {
|
||||||
|
Self {
|
||||||
|
iter: TokenIter::new(iter),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn skip(&mut self, n: usize) {
|
||||||
|
for _ in 0..n {
|
||||||
|
let _ = self.next_tok();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn next_tok(&mut self) -> Result<Token> {
|
||||||
|
match self.iter.next() {
|
||||||
|
Some(Token(TokenKind::Error(e), span)) => Err(e).span(&span),
|
||||||
|
r => r.reason("Unexpected end of file"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn peek(&self, n: usize) -> Result<Token> {
|
||||||
|
match self.iter.peek(n).clone() {
|
||||||
|
Some(Token(TokenKind::Error(e), span)) => Err(e).span(&span),
|
||||||
|
r => r.reason("Unexpected end of file"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn eat(&mut self, expect: TokenKind) -> Result<Token> {
|
||||||
|
match self.look(0, expect) {
|
||||||
|
Ok(t) => {
|
||||||
|
self.skip(1);
|
||||||
|
Ok(t)
|
||||||
|
},
|
||||||
|
Err(e) => Err(e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn look(&mut self, n: usize, expect: TokenKind) -> Result<Token> {
|
||||||
|
let next = self.peek(n)?;
|
||||||
|
if next.0 == expect {
|
||||||
|
Ok(next)
|
||||||
|
} else {
|
||||||
|
error()
|
||||||
|
.reason(format!("Expected {expect}, found {}", next.0))
|
||||||
|
.span(&next.1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn block(&mut self) -> Result<(Vec<Statement>, Span)> {
|
||||||
|
use TokenKind as t;
|
||||||
|
let mut span = self.eat(t::LeftBrace).reason("Expected block")?.1;
|
||||||
|
let mut statements = vec![];
|
||||||
|
loop {
|
||||||
|
let next = self.peek(0)?;
|
||||||
|
span = span + next.1;
|
||||||
|
match self.eat(t::RightBrace) {
|
||||||
|
Ok(t) => {
|
||||||
|
span = span + t.1;
|
||||||
|
break;
|
||||||
|
},
|
||||||
|
_ => {
|
||||||
|
let statement = self.statement()?;
|
||||||
|
span = span + statement.span;
|
||||||
|
statements.push(statement);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Ok((statements, span))
|
||||||
|
}
|
||||||
|
}
|
207
src/parse/statement.rs
Normal file
207
src/parse/statement.rs
Normal file
|
@ -0,0 +1,207 @@
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub enum StatementKind {
|
||||||
|
Mutable {
|
||||||
|
name: String,
|
||||||
|
type_: Option<String>,
|
||||||
|
value: Option<Expression>,
|
||||||
|
},
|
||||||
|
Immutable {
|
||||||
|
name: String,
|
||||||
|
type_: Option<String>,
|
||||||
|
value: Expression,
|
||||||
|
},
|
||||||
|
Assignment {
|
||||||
|
name: String,
|
||||||
|
value: Expression,
|
||||||
|
},
|
||||||
|
If {
|
||||||
|
predicate: Expression,
|
||||||
|
block: Vec<Statement>,
|
||||||
|
else_: Option<Box<Statement>>,
|
||||||
|
},
|
||||||
|
While {
|
||||||
|
predicate: Expression,
|
||||||
|
block: Vec<Statement>,
|
||||||
|
},
|
||||||
|
Print(Expression),
|
||||||
|
Expression(Expression),
|
||||||
|
Block(Vec<Statement>),
|
||||||
|
Error(Diagnostic),
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct Statement {
|
||||||
|
pub kind: StatementKind,
|
||||||
|
pub span: Span,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<I: Iterator<Item = Token>> Parser<I> {
|
||||||
|
pub fn statement(&mut self) -> Result<Statement> {
|
||||||
|
use StatementKind as s;
|
||||||
|
use TokenKind as t;
|
||||||
|
let next = self.peek(0)?;
|
||||||
|
let next2 = self.peek(1);
|
||||||
|
let mut span = next.1;
|
||||||
|
let statement = match (next, next2) {
|
||||||
|
// (im)mutable declaration
|
||||||
|
(Token(t::Identifier(name), span2), Ok(Token(t::Colon, span3))) => {
|
||||||
|
self.skip(2);
|
||||||
|
span = span + span2 + span3;
|
||||||
|
// type
|
||||||
|
let type_ = match self.eat(t::Identifier("".into())) {
|
||||||
|
Ok(Token(t::Identifier(s), span2)) => {
|
||||||
|
span = span + span2;
|
||||||
|
Some(s)
|
||||||
|
},
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
// value
|
||||||
|
match self.eat(t::Equal).or_else(|_| self.eat(t::Colon)) {
|
||||||
|
Ok(Token(t::Colon, span2)) => {
|
||||||
|
span = span + span2;
|
||||||
|
let value = self
|
||||||
|
.expression(0)
|
||||||
|
.trace_span(span, "while parsing mutable declaration")?;
|
||||||
|
span = span + value.span;
|
||||||
|
Statement {
|
||||||
|
kind: s::Immutable { name, type_, value },
|
||||||
|
span,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Ok(Token(t::Equal, span2)) => {
|
||||||
|
span = span + span2;
|
||||||
|
let value = self
|
||||||
|
.expression(0)
|
||||||
|
.trace_span(span, "while parsing mutable declaration")?;
|
||||||
|
span = span + value.span;
|
||||||
|
Statement {
|
||||||
|
kind: s::Mutable {
|
||||||
|
name,
|
||||||
|
type_,
|
||||||
|
value: Some(value),
|
||||||
|
},
|
||||||
|
span,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => return error().reason("Expected expression").span(&span),
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// Assignment
|
||||||
|
(Token(t::Identifier(name), span2), Ok(Token(t::Equal, span3))) => {
|
||||||
|
self.skip(2);
|
||||||
|
span = span + span2 + span3;
|
||||||
|
let value = self
|
||||||
|
.expression(0)
|
||||||
|
.trace_span(span, "while parsing assignment")?;
|
||||||
|
Statement {
|
||||||
|
span,
|
||||||
|
kind: s::Assignment { name, value },
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// If
|
||||||
|
(Token(t::If, _), _) => {
|
||||||
|
return self.if_else();
|
||||||
|
},
|
||||||
|
// While
|
||||||
|
(Token(t::While, span2), _) => {
|
||||||
|
self.skip(1);
|
||||||
|
span = span + span2;
|
||||||
|
let predicate = self
|
||||||
|
.expression(0)
|
||||||
|
.reason("Expected predicate after 'while' keyword")
|
||||||
|
.span(&span)?;
|
||||||
|
span = span + predicate.span;
|
||||||
|
let block = self
|
||||||
|
.block()
|
||||||
|
.trace_span(span, "while parsing while statement")?;
|
||||||
|
span = span + block.1;
|
||||||
|
return Ok(Statement {
|
||||||
|
span,
|
||||||
|
kind: s::While {
|
||||||
|
predicate,
|
||||||
|
block: block.0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
},
|
||||||
|
// (DEBUG) print
|
||||||
|
(Token(t::Print, span2), _) => {
|
||||||
|
self.skip(1);
|
||||||
|
span = span + span2;
|
||||||
|
let expr = self
|
||||||
|
.expression(0)
|
||||||
|
.trace_span(span, "while parsing print statement")?;
|
||||||
|
span = span + expr.span;
|
||||||
|
Statement {
|
||||||
|
span,
|
||||||
|
kind: s::Print(expr),
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// Block
|
||||||
|
(Token(t::LeftBrace, span2), _) => {
|
||||||
|
// Skip check for semicolon
|
||||||
|
span = span + span2;
|
||||||
|
let (block, span2) = self
|
||||||
|
.block()
|
||||||
|
.trace_span(span, "while parsing block statement")?;
|
||||||
|
span = span + span2;
|
||||||
|
return Ok(Statement {
|
||||||
|
kind: s::Block(block),
|
||||||
|
span,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
// Expression
|
||||||
|
(Token(_, span2), _) => {
|
||||||
|
span = span + span2;
|
||||||
|
let expr = self
|
||||||
|
.expression(0)
|
||||||
|
.trace_span(span, "while parsing expression statement")?;
|
||||||
|
span = span + expr.span;
|
||||||
|
Statement {
|
||||||
|
span,
|
||||||
|
kind: s::Expression(expr),
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
// Check for semicolon
|
||||||
|
if self.eat(t::Semicolon).is_ok() {
|
||||||
|
Ok(statement)
|
||||||
|
} else {
|
||||||
|
error().reason("Expected ;").span(&span)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn if_else(&mut self) -> Result<Statement> {
|
||||||
|
use TokenKind as t;
|
||||||
|
if let Ok(Token(_, span)) = self.eat(t::If) {
|
||||||
|
let predicate = self
|
||||||
|
.expression(0)
|
||||||
|
.trace_span(span, "in predicate of 'if' statement")?;
|
||||||
|
let span = span + predicate.span;
|
||||||
|
let block = self
|
||||||
|
.block()
|
||||||
|
.trace_span(span, "in block of 'if' statement")?;
|
||||||
|
let (block, span) = (block.0, span + block.1);
|
||||||
|
let else_ = if self.eat(t::Else).is_ok() {
|
||||||
|
Some(Box::new(self.if_else()?))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
Ok(Statement {
|
||||||
|
kind: StatementKind::If {
|
||||||
|
predicate,
|
||||||
|
block,
|
||||||
|
else_,
|
||||||
|
},
|
||||||
|
span,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
let (block, span) = self.block()?;
|
||||||
|
Ok(Statement {
|
||||||
|
kind: StatementKind::Block(block),
|
||||||
|
span,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
146
src/token.rs
146
src/token.rs
|
@ -49,7 +49,7 @@ pub enum TokenKind {
|
||||||
|
|
||||||
Identifier(String),
|
Identifier(String),
|
||||||
StringLiteral(String),
|
StringLiteral(String),
|
||||||
CharLiteral(char),
|
GlyphLiteral(char),
|
||||||
IntegerLiteral(i64),
|
IntegerLiteral(i64),
|
||||||
FloatLiteral(f64),
|
FloatLiteral(f64),
|
||||||
|
|
||||||
|
@ -78,6 +78,7 @@ pub enum TokenKind {
|
||||||
SmallComment(String),
|
SmallComment(String),
|
||||||
BigComment(String),
|
BigComment(String),
|
||||||
|
|
||||||
|
Error(Diagnostic),
|
||||||
Idk,
|
Idk,
|
||||||
EOF,
|
EOF,
|
||||||
}
|
}
|
||||||
|
@ -103,6 +104,87 @@ impl PartialEq for TokenKind {
|
||||||
impl Eq for TokenKind {
|
impl Eq for TokenKind {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Display for TokenKind {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
use TokenKind::*;
|
||||||
|
write!(
|
||||||
|
f,
|
||||||
|
"'{}'",
|
||||||
|
match self {
|
||||||
|
LeftParen => "(",
|
||||||
|
RightParen => ")",
|
||||||
|
LeftBrace => "{",
|
||||||
|
RightBrace => "}",
|
||||||
|
LeftSquare => "[",
|
||||||
|
RightSquare => "]",
|
||||||
|
Comma => ",",
|
||||||
|
Colon => ":",
|
||||||
|
Semicolon => ";",
|
||||||
|
Dot => ".",
|
||||||
|
DotDot => "..",
|
||||||
|
Plus => "+",
|
||||||
|
Minus => "-",
|
||||||
|
Slash => "/",
|
||||||
|
Star => "*",
|
||||||
|
Percent => "%",
|
||||||
|
Arrow => "->",
|
||||||
|
FatArrow => "=>",
|
||||||
|
PlusEqual => "+=",
|
||||||
|
MinusEqual => "-=",
|
||||||
|
SlashEqual => "/=",
|
||||||
|
StarEqual => "*=",
|
||||||
|
PercentEqual => "%=",
|
||||||
|
Bang => "!",
|
||||||
|
BangEqual => "!=",
|
||||||
|
Question => "?",
|
||||||
|
QuestionEqual => "?=",
|
||||||
|
Equal => "=",
|
||||||
|
DoubleEqual => "==",
|
||||||
|
Greater => ">",
|
||||||
|
GreaterEqual => ">=",
|
||||||
|
Less => "<",
|
||||||
|
LessEqual => "<=",
|
||||||
|
Pipe => "|",
|
||||||
|
Ampersand => "&",
|
||||||
|
Carrot => "^",
|
||||||
|
Hash => "#",
|
||||||
|
DotDotEqual => "..=",
|
||||||
|
Identifier(i) => i,
|
||||||
|
StringLiteral(s) => s.as_str(),
|
||||||
|
GlyphLiteral(_) => "<glyph>",
|
||||||
|
IntegerLiteral(_) => "<integer>",
|
||||||
|
FloatLiteral(_) => "<real>",
|
||||||
|
If => "if",
|
||||||
|
Else => "else",
|
||||||
|
And => "and",
|
||||||
|
Or => "or",
|
||||||
|
Xor => "xor",
|
||||||
|
Not => "not",
|
||||||
|
Nand => "nand",
|
||||||
|
Nor => "nor",
|
||||||
|
Xnor => "xnor",
|
||||||
|
Print => "print",
|
||||||
|
Break => "break",
|
||||||
|
Return => "return",
|
||||||
|
Continue => "continue",
|
||||||
|
For => "for",
|
||||||
|
While => "while",
|
||||||
|
True => "true",
|
||||||
|
False => "false",
|
||||||
|
Struct => "struct",
|
||||||
|
Enum => "enum",
|
||||||
|
Union => "union",
|
||||||
|
Whitespace(_) => "<whitespace>",
|
||||||
|
SmallComment(_) => "<comment>",
|
||||||
|
BigComment(_) => "<comment>",
|
||||||
|
Error(_) => "<error>",
|
||||||
|
Idk => unreachable!(),
|
||||||
|
EOF => "<end of file>",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Token(pub TokenKind, pub Span);
|
pub struct Token(pub TokenKind, pub Span);
|
||||||
|
|
||||||
|
@ -118,7 +200,6 @@ pub struct Tokenizer<I: Iterator<Item = char>> {
|
||||||
iter: CharIter<I>,
|
iter: CharIter<I>,
|
||||||
column: usize,
|
column: usize,
|
||||||
row: usize,
|
row: usize,
|
||||||
finished: bool,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<I: Iterator<Item = char>> Tokenizer<I> {
|
impl<I: Iterator<Item = char>> Tokenizer<I> {
|
||||||
|
@ -127,14 +208,6 @@ impl<I: Iterator<Item = char>> Tokenizer<I> {
|
||||||
iter: CharIter::new(iter),
|
iter: CharIter::new(iter),
|
||||||
column: 1,
|
column: 1,
|
||||||
row: 1,
|
row: 1,
|
||||||
finished: false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn span(&self) -> Span {
|
|
||||||
Span {
|
|
||||||
column: self.column,
|
|
||||||
row: self.row,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -187,6 +260,16 @@ impl<I: Iterator<Item = char>> Tokenizer<I> {
|
||||||
column: self.column,
|
column: self.column,
|
||||||
};
|
};
|
||||||
let current = match self.next_char() {
|
let current = match self.next_char() {
|
||||||
|
Some(std::char::REPLACEMENT_CHARACTER) => {
|
||||||
|
return t(
|
||||||
|
Error(Diagnostic {
|
||||||
|
reason: "Non-UTF8 encoded glyph".into(),
|
||||||
|
span: Some(position),
|
||||||
|
backtrace: vec![],
|
||||||
|
}),
|
||||||
|
position,
|
||||||
|
);
|
||||||
|
},
|
||||||
Some(c) => c,
|
Some(c) => c,
|
||||||
None => return t(EOF, position),
|
None => return t(EOF, position),
|
||||||
};
|
};
|
||||||
|
@ -327,7 +410,7 @@ impl<I: Iterator<Item = char>> Tokenizer<I> {
|
||||||
.reason("Single quote (') contains more than one character")
|
.reason("Single quote (') contains more than one character")
|
||||||
.span(&position);
|
.span(&position);
|
||||||
}
|
}
|
||||||
let kind = CharLiteral(
|
let kind = GlyphLiteral(
|
||||||
baked
|
baked
|
||||||
.chars()
|
.chars()
|
||||||
.next()
|
.next()
|
||||||
|
@ -351,8 +434,12 @@ impl<I: Iterator<Item = char>> Tokenizer<I> {
|
||||||
// Only one dot per number
|
// Only one dot per number
|
||||||
let mut encountered_dot = false;
|
let mut encountered_dot = false;
|
||||||
while let Some(c) = self.peek(0) {
|
while let Some(c) = self.peek(0) {
|
||||||
if c == '.' && !encountered_dot {
|
if c == '.' {
|
||||||
if let Some('.') = self.peek(1) {
|
if encountered_dot {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
let Some(next) = self.peek(1) else { break };
|
||||||
|
if !next.is_ascii_digit() {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
encountered_dot = true;
|
encountered_dot = true;
|
||||||
|
@ -407,19 +494,12 @@ impl<I: Iterator<Item = char>> Iterator for Tokenizer<I> {
|
||||||
type Item = Token;
|
type Item = Token;
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
loop {
|
|
||||||
match self._next() {
|
match self._next() {
|
||||||
Ok(Token(TokenKind::EOF, span)) => {
|
Ok(s) => Some(s),
|
||||||
if self.finished {
|
Err(e) => Some(Token(
|
||||||
return None;
|
TokenKind::Error(e.clone()),
|
||||||
} else {
|
e.span.expect("error without span in tokenizer"),
|
||||||
self.finished = true;
|
)),
|
||||||
return Some(Token(TokenKind::EOF, span));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Ok(r) => return Some(r),
|
|
||||||
_ => {},
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -429,29 +509,35 @@ fn parse_number(num: &str) -> Result<TokenKind> {
|
||||||
let num = num.replace('_', "");
|
let num = num.replace('_', "");
|
||||||
// Floating point (only decimal)
|
// Floating point (only decimal)
|
||||||
if num.contains('.') {
|
if num.contains('.') {
|
||||||
num.parse::<f64>().map(|f| FloatLiteral(f)).coerce()
|
num
|
||||||
|
.parse::<f64>()
|
||||||
|
.map(|f| FloatLiteral(f))
|
||||||
|
.reason("Could not parse real number")
|
||||||
}
|
}
|
||||||
// Hex integer
|
// Hex integer
|
||||||
else if let Some(hex) = num.strip_prefix("0x") {
|
else if let Some(hex) = num.strip_prefix("0x") {
|
||||||
i64::from_str_radix(hex, 16)
|
i64::from_str_radix(hex, 16)
|
||||||
.map(|i| IntegerLiteral(i))
|
.map(|i| IntegerLiteral(i))
|
||||||
.coerce()
|
.reason("Could not parse hex integer number")
|
||||||
}
|
}
|
||||||
// Octal integer
|
// Octal integer
|
||||||
else if let Some(oct) = num.strip_prefix("0o") {
|
else if let Some(oct) = num.strip_prefix("0o") {
|
||||||
i64::from_str_radix(oct, 8)
|
i64::from_str_radix(oct, 8)
|
||||||
.map(|i| IntegerLiteral(i))
|
.map(|i| IntegerLiteral(i))
|
||||||
.coerce()
|
.reason("Could not parse octal integer number")
|
||||||
}
|
}
|
||||||
// Binary integer
|
// Binary integer
|
||||||
else if let Some(bin) = num.strip_prefix("0b") {
|
else if let Some(bin) = num.strip_prefix("0b") {
|
||||||
i64::from_str_radix(bin, 2)
|
i64::from_str_radix(bin, 2)
|
||||||
.map(|i| IntegerLiteral(i))
|
.map(|i| IntegerLiteral(i))
|
||||||
.coerce()
|
.reason("Could not parse binary integer number")
|
||||||
}
|
}
|
||||||
// Decimal integer
|
// Decimal integer
|
||||||
else {
|
else {
|
||||||
num.parse::<i64>().map(|i| IntegerLiteral(i)).coerce()
|
num
|
||||||
|
.parse::<i64>()
|
||||||
|
.map(|i| IntegerLiteral(i))
|
||||||
|
.reason("Could not parse integer number")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,6 +14,19 @@ enum Value {
|
||||||
Undefined,
|
Undefined,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Display for Value {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
use Value::*;
|
||||||
|
match self {
|
||||||
|
Integer(i) => write!(f, "{i}"),
|
||||||
|
Real(r) => write!(f, "{r}"),
|
||||||
|
String(s) => write!(f, "{s}"),
|
||||||
|
Boolean(b) => write!(f, "{b}"),
|
||||||
|
Undefined => write!(f, "undefined"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
struct Scope {
|
struct Scope {
|
||||||
outer: Option<Box<Scope>>,
|
outer: Option<Box<Scope>>,
|
||||||
|
@ -189,6 +202,13 @@ impl<I: Iterator<Item = Statement>> Interpreter<I> {
|
||||||
},
|
},
|
||||||
e::Unary { token, child } => self.evaluate_unary(token, *child),
|
e::Unary { token, child } => self.evaluate_unary(token, *child),
|
||||||
e::Parenthesis(e) => self.evaluate(*e),
|
e::Parenthesis(e) => self.evaluate(*e),
|
||||||
|
e::Call { callee, args } => todo!(),
|
||||||
|
e::Field { namespace, field } => todo!(),
|
||||||
|
e::Function {
|
||||||
|
params,
|
||||||
|
returns,
|
||||||
|
body,
|
||||||
|
} => todo!(),
|
||||||
}
|
}
|
||||||
.span(&expr.span)
|
.span(&expr.span)
|
||||||
}
|
}
|
||||||
|
@ -215,7 +235,7 @@ impl<I: Iterator<Item = Statement>> Interpreter<I> {
|
||||||
},
|
},
|
||||||
s::Print(e) => {
|
s::Print(e) => {
|
||||||
let e = self.evaluate(e)?;
|
let e = self.evaluate(e)?;
|
||||||
println!("{e:?}");
|
println!("{e}");
|
||||||
},
|
},
|
||||||
s::Expression(e) => {
|
s::Expression(e) => {
|
||||||
self.evaluate(e)?;
|
self.evaluate(e)?;
|
||||||
|
@ -231,6 +251,8 @@ impl<I: Iterator<Item = Statement>> Interpreter<I> {
|
||||||
if let Value::Boolean(b) = value {
|
if let Value::Boolean(b) = value {
|
||||||
if b {
|
if b {
|
||||||
self.block(block)?;
|
self.block(block)?;
|
||||||
|
} else if let Some(else_) = else_ {
|
||||||
|
self.execute(*else_)?;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return error()
|
return error()
|
||||||
|
@ -252,6 +274,10 @@ impl<I: Iterator<Item = Statement>> Interpreter<I> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
s::Error(e) => {
|
||||||
|
eprintln!("{e}");
|
||||||
|
panic!();
|
||||||
|
},
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
91
src/types.rs
Normal file
91
src/types.rs
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
use super::err::*;
|
||||||
|
|
||||||
|
macro_rules! primitives {
|
||||||
|
( $($i:ident),* ) => {
|
||||||
|
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||||
|
#[allow(non_camel_case_types, dead_code)]
|
||||||
|
pub enum Primitive {
|
||||||
|
whole_ambiguous,
|
||||||
|
integer_ambiguous,
|
||||||
|
real_ambiguous,
|
||||||
|
$($i,)*
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Primitive {
|
||||||
|
pub fn from_string(string: &'static str) -> Option<Self> {
|
||||||
|
match string {
|
||||||
|
$(stringify!{$i} => Some(Self::$i),)*
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl std::fmt::Display for Primitive {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
match self {
|
||||||
|
Primitive::whole_ambiguous => write!(f, "<ambiguous whole>"),
|
||||||
|
Primitive::integer_ambiguous => write!(f, "<ambiguous integer>"),
|
||||||
|
Primitive::real_ambiguous => write!(f, "<ambiguous real>"),
|
||||||
|
$(Primitive::$i => write!(f, stringify!{$i}),)*
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
primitives! {
|
||||||
|
w8, w16, w32, w64, whole,
|
||||||
|
i8, i16, i32, i64, integer,
|
||||||
|
r32, r64, real,
|
||||||
|
boolean,
|
||||||
|
string, glyph
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Primitive {
|
||||||
|
pub fn coerce(a: Self, b: Self) -> Result<Self> {
|
||||||
|
use Primitive::*;
|
||||||
|
match (a, b) {
|
||||||
|
// Whole? coerces to any whole or integer
|
||||||
|
(whole_ambiguous, w @ (w8 | w16 | w32 | w64 | i8 | i16 | i32 | i64))
|
||||||
|
| (w @ (w8 | w16 | w32 | w64 | i8 | i16 | i32 | i64), whole_ambiguous) => {
|
||||||
|
w
|
||||||
|
},
|
||||||
|
// Integer? coerces to any integer
|
||||||
|
(integer_ambiguous, i @ (i8 | i16 | i32 | i64))
|
||||||
|
| (i @ (i8 | i16 | i32 | i64), integer_ambiguous) => i,
|
||||||
|
_ => whole_ambiguous,
|
||||||
|
};
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
use Primitive as p;
|
||||||
|
|
||||||
|
// Implement math operations for regular types
|
||||||
|
macro_rules! selfsame_op {
|
||||||
|
($trait:ident, $fn:ident, $($i:ident),* ) => {
|
||||||
|
impl std::ops::$trait for Primitive {
|
||||||
|
type Output = Result<Primitive>;
|
||||||
|
fn $fn(self, rhs: Self) -> Self::Output {
|
||||||
|
match (self, rhs) {
|
||||||
|
$((p::$i, p::$i) => Ok(p::$i),)*
|
||||||
|
_ => error()
|
||||||
|
.reason(format!("Operation not defined for primitives {} and {}", self, rhs))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Implement all regular math
|
||||||
|
macro_rules! all_selfsame {
|
||||||
|
($($i:ident),*) => {
|
||||||
|
selfsame_op!(Add, add, $($i),*);
|
||||||
|
selfsame_op!(Sub, sub, $($i),*);
|
||||||
|
selfsame_op!(Mul, mul, $($i),*);
|
||||||
|
selfsame_op!(Div, div, $($i),*);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
all_selfsame!(
|
||||||
|
w8, w16, w32, w64, whole, i8, i16, i32, i64, integer, r32, r64, real, string
|
||||||
|
);
|
Loading…
Reference in a new issue