work
This commit is contained in:
Generated
+1
-1
@@ -3,5 +3,5 @@
|
|||||||
version = 4
|
version = 4
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "lang2"
|
name = "lang"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
|
|||||||
+1
-1
@@ -1,5 +1,5 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "lang2"
|
name = "lang"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
edition = "2024"
|
edition = "2024"
|
||||||
|
|
||||||
|
|||||||
+30
-22
@@ -1,5 +1,4 @@
|
|||||||
use crate::io::{CompilerMsg, Span, Spanned};
|
use crate::io::{CompilerMsg, Span, Spanned};
|
||||||
use std::iter::Peekable;
|
|
||||||
|
|
||||||
mod lit;
|
mod lit;
|
||||||
mod token;
|
mod token;
|
||||||
@@ -8,37 +7,52 @@ pub use token::*;
|
|||||||
|
|
||||||
pub struct Cursor<'a> {
|
pub struct Cursor<'a> {
|
||||||
pub span: Span,
|
pub span: Span,
|
||||||
tokens: Peekable<Tokens<'a>>,
|
next: Option<TokenInst>,
|
||||||
|
tokens: Tokens<'a>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Cursor<'a> {
|
impl<'a> Cursor<'a> {
|
||||||
pub fn new(text: &'a str, file: usize) -> Self {
|
pub fn new(text: &'a str, file: usize) -> Self {
|
||||||
Self {
|
let mut s = Self {
|
||||||
span: Span {
|
span: Span {
|
||||||
start: 0,
|
start: 0,
|
||||||
end: 0,
|
end: 0,
|
||||||
file,
|
file,
|
||||||
},
|
},
|
||||||
tokens: Tokens::new(text, file).peekable(),
|
next: None,
|
||||||
}
|
tokens: Tokens::new(text, file),
|
||||||
|
};
|
||||||
|
s.next();
|
||||||
|
s
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn next(&mut self) -> Option<Token> {
|
pub fn next(&mut self) -> Option<Token> {
|
||||||
self.tokens.next().map(|inst| {
|
let mut next = self.tokens.next();
|
||||||
|
std::mem::swap(&mut self.next, &mut next);
|
||||||
|
next.map(|inst| {
|
||||||
self.span = inst.span;
|
self.span = inst.span;
|
||||||
inst.inner
|
inst.inner
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn peek(&mut self) -> Option<&Token> {
|
pub fn next_if(&mut self, token: Token) -> bool {
|
||||||
self.tokens.peek().map(|inst| &inst.inner)
|
if self.peek().is_some_and(|t| *t == token) {
|
||||||
|
self.next();
|
||||||
|
true
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn peek(&self) -> Option<&Token> {
|
||||||
|
self.next.as_ref().map(|i| &i.inner)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn expect_next(&mut self) -> Result<Token, CompilerMsg> {
|
pub fn expect_next(&mut self) -> Result<Token, CompilerMsg> {
|
||||||
self.next().ok_or_else(CompilerMsg::unexpected_eof)
|
self.next().ok_or_else(CompilerMsg::unexpected_eof)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn expect_peek(&mut self) -> Result<&Token, CompilerMsg> {
|
pub fn expect_peek(&self) -> Result<&Token, CompilerMsg> {
|
||||||
self.peek().ok_or_else(CompilerMsg::unexpected_eof)
|
self.peek().ok_or_else(CompilerMsg::unexpected_eof)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -47,22 +61,16 @@ impl<'a> Cursor<'a> {
|
|||||||
if next == token {
|
if next == token {
|
||||||
Ok(next)
|
Ok(next)
|
||||||
} else {
|
} else {
|
||||||
self.unexpected(next, &format!("'{token}'"))
|
self.unexpected(&next, &format!("'{token}'"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn unexpected<T>(&self, token: Token, expected: &str) -> Result<T, CompilerMsg> {
|
pub fn unexpected<T>(&self, token: &Token, expected: &str) -> Result<T, CompilerMsg> {
|
||||||
Err(CompilerMsg::unexpected_token(
|
Err(CompilerMsg::unexpected_token(token, self.span, expected))
|
||||||
Spanned {
|
|
||||||
inner: token,
|
|
||||||
span: self.span,
|
|
||||||
},
|
|
||||||
expected,
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn peek_start(&mut self) -> usize {
|
pub fn peek_start(&mut self) -> usize {
|
||||||
self.tokens.peek().map(|i| i.span.start).unwrap_or(0)
|
self.next.as_ref().map(|i| i.span.start).unwrap_or(0)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn cur_end(&mut self) -> usize {
|
pub fn cur_end(&mut self) -> usize {
|
||||||
@@ -75,10 +83,10 @@ impl<'a> Cursor<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl CompilerMsg {
|
impl CompilerMsg {
|
||||||
pub fn unexpected_token(inst: TokenInst, expected: &str) -> Self {
|
pub fn unexpected_token(token: &Token, span: Span, expected: &str) -> Self {
|
||||||
Self {
|
Self {
|
||||||
spans: vec![inst.span],
|
spans: vec![span],
|
||||||
msg: format!("Unexpected token '{}'; expected {expected}", inst.inner),
|
msg: format!("Unexpected token '{}'; expected {expected}", token),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+69
-37
@@ -3,17 +3,26 @@ use std::{iter::Peekable, str::CharIndices};
|
|||||||
|
|
||||||
def_tokens! {
|
def_tokens! {
|
||||||
symbol {
|
symbol {
|
||||||
'=' => Equal,
|
Equal: "=",
|
||||||
':' => Colon,
|
Colon: ":",
|
||||||
';' => Semicolon,
|
Semicolon: ";",
|
||||||
'{' => OpenCurly,
|
Plus: "+",
|
||||||
'}' => CloseCurly,
|
Dash: "-",
|
||||||
'(' => OpenParen,
|
Asterisk: "*",
|
||||||
')' => CloseParen,
|
Slash: "/",
|
||||||
'-' => Dash,
|
OpenParen: "(",
|
||||||
'+' => Plus,
|
CloseParen: ")",
|
||||||
'*' => Asterisk,
|
OpenSquare: "[",
|
||||||
'/' => Slash,
|
CloseSquare: "]",
|
||||||
|
OpenCurly: "{",
|
||||||
|
CloseCurly: "}",
|
||||||
|
Arrow: "->",
|
||||||
|
DoubleArrow: "=>",
|
||||||
|
PlusEqual: "+=",
|
||||||
|
DashEqual: "-=",
|
||||||
|
AsteriskEqual: "*=",
|
||||||
|
SlashEqual: "/=",
|
||||||
|
DoubleColon: "::",
|
||||||
}
|
}
|
||||||
keyword {
|
keyword {
|
||||||
Let: "let",
|
Let: "let",
|
||||||
@@ -22,6 +31,7 @@ def_tokens! {
|
|||||||
Loop: "loop",
|
Loop: "loop",
|
||||||
While: "while",
|
While: "while",
|
||||||
For: "for",
|
For: "for",
|
||||||
|
Match: "match",
|
||||||
}
|
}
|
||||||
other {
|
other {
|
||||||
Ident(String),
|
Ident(String),
|
||||||
@@ -55,13 +65,53 @@ impl Iterator for Tokens<'_> {
|
|||||||
end: i,
|
end: i,
|
||||||
file: self.file,
|
file: self.file,
|
||||||
};
|
};
|
||||||
if let Some(inner) = from_char(c) {
|
|
||||||
return Some(Spanned { inner, span });
|
|
||||||
}
|
|
||||||
if c.is_whitespace() {
|
if c.is_whitespace() {
|
||||||
return self.next();
|
return self.next();
|
||||||
}
|
}
|
||||||
|
macro_rules! then {
|
||||||
|
(_ => $def:expr, $($char:expr => $to:expr,)*) => {
|
||||||
|
match self.chars.peek() {
|
||||||
|
$(Some((_, $char)) => {
|
||||||
|
self.chars.next();
|
||||||
|
$to
|
||||||
|
},)*
|
||||||
|
_ => $def,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
let inner = match c {
|
let inner = match c {
|
||||||
|
'(' => Token::OpenParen,
|
||||||
|
')' => Token::CloseParen,
|
||||||
|
'[' => Token::OpenSquare,
|
||||||
|
']' => Token::CloseSquare,
|
||||||
|
'{' => Token::OpenCurly,
|
||||||
|
'}' => Token::CloseCurly,
|
||||||
|
'+' => then! {
|
||||||
|
_ => Token::Plus,
|
||||||
|
'=' => Token::PlusEqual,
|
||||||
|
},
|
||||||
|
'-' => then! {
|
||||||
|
_ => Token::Dash,
|
||||||
|
'=' => Token::DashEqual,
|
||||||
|
'>' => Token::Arrow,
|
||||||
|
},
|
||||||
|
'*' => then! {
|
||||||
|
_ => Token::Asterisk,
|
||||||
|
'=' => Token::AsteriskEqual,
|
||||||
|
},
|
||||||
|
'/' => then! {
|
||||||
|
_ => Token::Slash,
|
||||||
|
'=' => Token::SlashEqual,
|
||||||
|
},
|
||||||
|
':' => then! {
|
||||||
|
_ => Token::Colon,
|
||||||
|
':' => Token::DoubleColon,
|
||||||
|
},
|
||||||
|
';' => Token::Semicolon,
|
||||||
|
'=' => then! {
|
||||||
|
_ => Token::Equal,
|
||||||
|
'>' => Token::DoubleArrow,
|
||||||
|
},
|
||||||
'0'..='9' => {
|
'0'..='9' => {
|
||||||
let mut s = c.to_string();
|
let mut s = c.to_string();
|
||||||
while let Some((i, c)) = self.chars.peek()
|
while let Some((i, c)) = self.chars.peek()
|
||||||
@@ -105,23 +155,11 @@ impl Iterator for Tokens<'_> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! expand_sym_names {
|
|
||||||
({
|
|
||||||
$($sym_char:expr => $sym_name:ident,)*
|
|
||||||
}) => {
|
|
||||||
$($sym_name,)*
|
|
||||||
};
|
|
||||||
({
|
|
||||||
$($sym_char:expr => $sym_res:tt,)*
|
|
||||||
}) => {
|
|
||||||
expand_sym_names!($sym_res)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
use expand_sym_names;
|
|
||||||
|
|
||||||
macro_rules! def_tokens {
|
macro_rules! def_tokens {
|
||||||
{
|
{
|
||||||
symbol $syms:tt
|
symbol {
|
||||||
|
$($sym_name:ident: $sym_str:expr,)*
|
||||||
|
}
|
||||||
keyword {
|
keyword {
|
||||||
$($kw_name:ident: $kw_str:expr,)*
|
$($kw_name:ident: $kw_str:expr,)*
|
||||||
}
|
}
|
||||||
@@ -131,16 +169,10 @@ macro_rules! def_tokens {
|
|||||||
} => {
|
} => {
|
||||||
#[derive(PartialEq)]
|
#[derive(PartialEq)]
|
||||||
pub enum Token {
|
pub enum Token {
|
||||||
$($syms,)*
|
$($sym_name,)*
|
||||||
$($kw_name,)*
|
$($kw_name,)*
|
||||||
$($other_name($data),)*
|
$($other_name($data),)*
|
||||||
}
|
}
|
||||||
fn from_char(c: char) -> Option<Token> {
|
|
||||||
match c {
|
|
||||||
$($sym_char => Some(Token::$sym_res),)*
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fn from_str(s: String) -> Token {
|
fn from_str(s: String) -> Token {
|
||||||
match s.as_str() {
|
match s.as_str() {
|
||||||
$($kw_str => Token::$kw_name,)*
|
$($kw_str => Token::$kw_name,)*
|
||||||
@@ -150,7 +182,7 @@ macro_rules! def_tokens {
|
|||||||
impl std::fmt::Display for Token {
|
impl std::fmt::Display for Token {
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
match self {
|
match self {
|
||||||
$(Token::$sym_res => write!(f, "{}", $sym_char),)*
|
$(Token::$sym_name => write!(f, "{}", $sym_str),)*
|
||||||
$(Token::$kw_name => write!(f, $kw_str),)*
|
$(Token::$kw_name => write!(f, $kw_str),)*
|
||||||
$(Token::$other_name(v) => write!(f, "{v}"),)*
|
$(Token::$other_name(v) => write!(f, "{v}"),)*
|
||||||
}
|
}
|
||||||
|
|||||||
+2
-29
@@ -1,6 +1,6 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
io::{CompilerOutput, Span},
|
io::{CompilerOutput, Span},
|
||||||
parser::{Cursor, Lit, nodes::*},
|
parser::{Cursor, nodes::*},
|
||||||
};
|
};
|
||||||
|
|
||||||
mod id;
|
mod id;
|
||||||
@@ -8,16 +8,8 @@ mod parse;
|
|||||||
pub use id::*;
|
pub use id::*;
|
||||||
pub use parse::*;
|
pub use parse::*;
|
||||||
|
|
||||||
def_nodes! {
|
|
||||||
exprs: Expr,
|
|
||||||
idents: Ident,
|
|
||||||
statements: Statement,
|
|
||||||
blocks: Block,
|
|
||||||
lits: Lit,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Nodes {
|
impl Nodes {
|
||||||
pub fn parse_root(path: &str, output: &mut CompilerOutput) -> Option<(Self, Id<Block>)> {
|
pub fn parse_root(path: &str, output: &mut CompilerOutput) -> Option<(Self, Id<Module>)> {
|
||||||
let root_code = match std::fs::read_to_string(path) {
|
let root_code = match std::fs::read_to_string(path) {
|
||||||
Ok(code) => code,
|
Ok(code) => code,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
@@ -66,22 +58,3 @@ pub trait Node: Sized {
|
|||||||
fn vec(nodes: &Nodes) -> &NodeVec<Self>;
|
fn vec(nodes: &Nodes) -> &NodeVec<Self>;
|
||||||
fn vec_mut(nodes: &mut Nodes) -> &mut NodeVec<Self>;
|
fn vec_mut(nodes: &mut Nodes) -> &mut NodeVec<Self>;
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! def_nodes {
|
|
||||||
{$($field:ident: $ty:ident,)*} => {
|
|
||||||
#[derive(Default)]
|
|
||||||
pub struct Nodes {
|
|
||||||
$($field: NodeVec<$ty>,)*
|
|
||||||
}
|
|
||||||
|
|
||||||
$(impl Node for $ty {
|
|
||||||
fn vec(nodes: &Nodes) -> &NodeVec<Self> {
|
|
||||||
&nodes.$field
|
|
||||||
}
|
|
||||||
fn vec_mut(nodes: &mut Nodes) -> &mut NodeVec<Self> {
|
|
||||||
&mut nodes.$field
|
|
||||||
}
|
|
||||||
})*
|
|
||||||
};
|
|
||||||
}
|
|
||||||
use def_nodes;
|
|
||||||
|
|||||||
@@ -10,6 +10,16 @@ pub trait Parsable: Sized + Node {
|
|||||||
fn parse(ctx: &mut ParseCtx) -> Result<Self, CompilerMsg>;
|
fn parse(ctx: &mut ParseCtx) -> Result<Self, CompilerMsg>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait ParsableWith<Input>: Sized + Node {
|
||||||
|
fn parse_with(ctx: &mut ParseCtx, input: Input) -> Result<Self, CompilerMsg>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<P: Parsable> ParsableWith<()> for P {
|
||||||
|
fn parse_with(ctx: &mut ParseCtx, _: ()) -> Result<Self, CompilerMsg> {
|
||||||
|
P::parse(ctx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub struct ParseCtx<'a> {
|
pub struct ParseCtx<'a> {
|
||||||
start: usize,
|
start: usize,
|
||||||
cursor: Cursor<'a>,
|
cursor: Cursor<'a>,
|
||||||
@@ -24,10 +34,18 @@ impl<'a> ParseCtx<'a> {
|
|||||||
cursor,
|
cursor,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn parse<P: Parsable>(&mut self) -> Result<Id<P>, CompilerMsg> {
|
pub fn parse<P: Parsable>(&mut self) -> Result<Id<P>, CompilerMsg> {
|
||||||
|
self.parse_with(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn parse_with<P: ParsableWith<Input>, Input>(
|
||||||
|
&mut self,
|
||||||
|
input: Input,
|
||||||
|
) -> Result<Id<P>, CompilerMsg> {
|
||||||
let old_start = self.start;
|
let old_start = self.start;
|
||||||
self.start = self.cursor.peek_start();
|
self.start = self.cursor.peek_start();
|
||||||
let res = P::parse(self).map(|r| self.push(r));
|
let res = P::parse_with(self, input).map(|r| self.push(r));
|
||||||
self.start = old_start;
|
self.start = old_start;
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,34 +0,0 @@
|
|||||||
use super::*;
|
|
||||||
|
|
||||||
pub struct Block {
|
|
||||||
statements: Vec<Id<Statement>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Parsable for Block {
|
|
||||||
fn parse(ctx: &mut ParseCtx) -> Result<Self, CompilerMsg> {
|
|
||||||
let mut statements = Vec::new();
|
|
||||||
if *ctx.expect_peek()? != Token::CloseCurly {
|
|
||||||
statements.push(ctx.parse()?);
|
|
||||||
while *ctx.expect_peek()? == Token::Semicolon {
|
|
||||||
ctx.next();
|
|
||||||
statements.push(ctx.parse()?);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(Self { statements })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl FmtNode for Block {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter, mut ctx: DisplayCtx) -> std::fmt::Result {
|
|
||||||
ctx.indent += 3;
|
|
||||||
write!(f, "{{")?;
|
|
||||||
if !self.statements.is_empty() {
|
|
||||||
writeln!(f)?;
|
|
||||||
}
|
|
||||||
for &s in &self.statements {
|
|
||||||
writeln!(f, "{}{};", " ".repeat(ctx.indent), s.dsp(ctx))?;
|
|
||||||
}
|
|
||||||
write!(f, "}}")?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -13,7 +13,7 @@ impl Parsable for Expr {
|
|||||||
Token::Dash => Self::Negate(ctx.parse()?),
|
Token::Dash => Self::Negate(ctx.parse()?),
|
||||||
Token::Ident(s) => Self::Ident(ctx.ident(s)),
|
Token::Ident(s) => Self::Ident(ctx.ident(s)),
|
||||||
Token::Lit(l) => Self::Lit(ctx.lit(l)),
|
Token::Lit(l) => Self::Lit(ctx.lit(l)),
|
||||||
other => return ctx.unexpected(other, "an expression"),
|
other => return ctx.unexpected(&other, "an expression"),
|
||||||
};
|
};
|
||||||
let Some(next) = ctx.peek() else {
|
let Some(next) = ctx.peek() else {
|
||||||
return Ok(e1);
|
return Ok(e1);
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ impl Parsable for Ident {
|
|||||||
fn parse(ctx: &mut super::ParseCtx) -> Result<Self, crate::io::CompilerMsg> {
|
fn parse(ctx: &mut super::ParseCtx) -> Result<Self, crate::io::CompilerMsg> {
|
||||||
match ctx.expect_next()? {
|
match ctx.expect_next()? {
|
||||||
Token::Ident(ident) => Ok(Self { inner: ident }),
|
Token::Ident(ident) => Ok(Self { inner: ident }),
|
||||||
t => ctx.unexpected(t, "an identifier"),
|
t => ctx.unexpected(&t, "an identifier"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
use super::*;
|
||||||
|
|
||||||
|
pub enum Item {
|
||||||
|
Module(Id<Module>),
|
||||||
|
Statement(Id<Statement>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Parsable for Item {
|
||||||
|
fn parse(ctx: &mut ParseCtx) -> Result<Self, CompilerMsg> {
|
||||||
|
Ok(match ctx.expect_peek()? {
|
||||||
|
Token::Fn => Self::Module(ctx.parse()?),
|
||||||
|
_ => Self::Statement(ctx.parse()?),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FmtNode for Item {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter, ctx: DisplayCtx) -> std::fmt::Result {
|
||||||
|
match self {
|
||||||
|
Item::Module(id) => write!(f, "{}", id.dsp(ctx)),
|
||||||
|
Item::Statement(id) => write!(f, "{}", id.dsp(ctx)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
+36
-3
@@ -1,11 +1,44 @@
|
|||||||
mod block;
|
|
||||||
mod expr;
|
mod expr;
|
||||||
mod ident;
|
mod ident;
|
||||||
|
mod item;
|
||||||
|
mod module;
|
||||||
mod statement;
|
mod statement;
|
||||||
pub use block::*;
|
mod ty;
|
||||||
pub use expr::*;
|
pub use expr::*;
|
||||||
pub use ident::*;
|
pub use ident::*;
|
||||||
|
pub use item::*;
|
||||||
|
pub use module::*;
|
||||||
pub use statement::*;
|
pub use statement::*;
|
||||||
|
pub use ty::*;
|
||||||
|
|
||||||
use super::{DisplayCtx, FmtNode, Id, Lit, Parsable, ParseCtx, Token};
|
use super::{DisplayCtx, FmtNode, Id, Lit, Node, NodeVec, Parsable, ParseCtx, Token};
|
||||||
use crate::io::CompilerMsg;
|
use crate::io::CompilerMsg;
|
||||||
|
|
||||||
|
def_nodes! {
|
||||||
|
exprs: Expr,
|
||||||
|
idents: Ident,
|
||||||
|
statements: Statement,
|
||||||
|
blocks: Module,
|
||||||
|
lits: Lit,
|
||||||
|
types: Type,
|
||||||
|
items: Item,
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! def_nodes {
|
||||||
|
{$($field:ident: $ty:ident,)*} => {
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct Nodes {
|
||||||
|
$(pub $field: NodeVec<$ty>,)*
|
||||||
|
}
|
||||||
|
|
||||||
|
$(impl Node for $ty {
|
||||||
|
fn vec(nodes: &Nodes) -> &NodeVec<Self> {
|
||||||
|
&nodes.$field
|
||||||
|
}
|
||||||
|
fn vec_mut(nodes: &mut Nodes) -> &mut NodeVec<Self> {
|
||||||
|
&mut nodes.$field
|
||||||
|
}
|
||||||
|
})*
|
||||||
|
};
|
||||||
|
}
|
||||||
|
use def_nodes;
|
||||||
|
|||||||
@@ -0,0 +1,35 @@
|
|||||||
|
use super::*;
|
||||||
|
|
||||||
|
pub struct Module {
|
||||||
|
items: Vec<Id<Item>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Parsable for Module {
|
||||||
|
fn parse(ctx: &mut ParseCtx) -> Result<Self, CompilerMsg> {
|
||||||
|
let mut items = Vec::new();
|
||||||
|
if ctx.peek().is_none() {
|
||||||
|
return Ok(Self { items });
|
||||||
|
}
|
||||||
|
items.push(ctx.parse()?);
|
||||||
|
while *ctx.expect_peek()? == Token::Semicolon {
|
||||||
|
ctx.next();
|
||||||
|
items.push(ctx.parse()?);
|
||||||
|
}
|
||||||
|
Ok(Self { items })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FmtNode for Module {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter, mut ctx: DisplayCtx) -> std::fmt::Result {
|
||||||
|
ctx.indent += 3;
|
||||||
|
write!(f, "{{")?;
|
||||||
|
if !self.items.is_empty() {
|
||||||
|
writeln!(f)?;
|
||||||
|
}
|
||||||
|
for &i in &self.items {
|
||||||
|
writeln!(f, "{}{};", " ".repeat(ctx.indent), i.dsp(ctx))?;
|
||||||
|
}
|
||||||
|
write!(f, "}}")?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,8 +1,15 @@
|
|||||||
pub use super::*;
|
pub use super::*;
|
||||||
|
|
||||||
pub enum Statement {
|
pub enum Statement {
|
||||||
Let(Id<Ident>, Id<Expr>),
|
Let {
|
||||||
If { cond: Id<Expr>, body: Id<Expr> },
|
name: Id<Ident>,
|
||||||
|
ty: Option<Id<Type>>,
|
||||||
|
val: Id<Expr>,
|
||||||
|
},
|
||||||
|
If {
|
||||||
|
cond: Id<Expr>,
|
||||||
|
body: Id<Expr>,
|
||||||
|
},
|
||||||
Expr(Id<Expr>),
|
Expr(Id<Expr>),
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -12,8 +19,16 @@ impl Parsable for Statement {
|
|||||||
Token::Let => {
|
Token::Let => {
|
||||||
ctx.next();
|
ctx.next();
|
||||||
let name = ctx.parse()?;
|
let name = ctx.parse()?;
|
||||||
|
let mut ty = None;
|
||||||
|
if ctx.next_if(Token::Colon) {
|
||||||
|
ty = Some(ctx.parse()?);
|
||||||
|
}
|
||||||
ctx.expect(Token::Equal)?;
|
ctx.expect(Token::Equal)?;
|
||||||
Self::Let(name, ctx.parse()?)
|
Self::Let {
|
||||||
|
name,
|
||||||
|
ty,
|
||||||
|
val: ctx.parse()?,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Token::If => {
|
Token::If => {
|
||||||
ctx.next();
|
ctx.next();
|
||||||
@@ -32,8 +47,12 @@ impl FmtNode for Statement {
|
|||||||
Self::If { cond, body } => {
|
Self::If { cond, body } => {
|
||||||
write!(f, "if {} {}", cond.dsp(ctx), body.dsp(ctx))
|
write!(f, "if {} {}", cond.dsp(ctx), body.dsp(ctx))
|
||||||
}
|
}
|
||||||
Self::Let(name, expr) => {
|
Self::Let { name, ty, val } => {
|
||||||
write!(f, "let {} = {}", name.dsp(ctx), expr.dsp(ctx))
|
write!(f, "let {}", name.dsp(ctx))?;
|
||||||
|
if let Some(ty) = ty {
|
||||||
|
write!(f, ": {}", ty.dsp(ctx))?;
|
||||||
|
}
|
||||||
|
write!(f, " = {}", val.dsp(ctx))
|
||||||
}
|
}
|
||||||
Self::Expr(expr) => expr.fmt(f, ctx),
|
Self::Expr(expr) => expr.fmt(f, ctx),
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
use super::*;
|
||||||
|
|
||||||
|
pub enum Type {
|
||||||
|
Ident(Id<Ident>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Parsable for Type {
|
||||||
|
fn parse(ctx: &mut ParseCtx) -> Result<Self, CompilerMsg> {
|
||||||
|
Ok(match ctx.expect_next()? {
|
||||||
|
Token::Ident(s) => Self::Ident(ctx.ident(s)),
|
||||||
|
t => ctx.unexpected(&t, "a type")?,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FmtNode for Type {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter, ctx: DisplayCtx) -> std::fmt::Result {
|
||||||
|
match self {
|
||||||
|
Type::Ident(id) => id.fmt(f, ctx),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
+1
-1
@@ -1 +1 @@
|
|||||||
let x = arst -3
|
let x: i32 = arst -3
|
||||||
|
|||||||
Reference in New Issue
Block a user