Compare commits
7 Commits
debcf6ad6c
...
323d3342a3
Author | SHA1 | Date | |
---|---|---|---|
323d3342a3 | |||
9d243e33b6 | |||
77738f7e20 | |||
bf7f44e776 | |||
857471c6a9 | |||
6c01265f0b | |||
8bb0e28d80 |
|
@ -2,7 +2,7 @@
|
||||||
name = "mclangc"
|
name = "mclangc"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
default-run = "mclangc"
|
||||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
use std::{collections::HashMap, ffi::OsStr, io::Write, os::unix::ffi::OsStrExt, path::{Path, PathBuf}, process::ExitCode};
|
use std::{collections::HashMap, ffi::OsStr, io::Write, os::unix::ffi::OsStrExt, path::{Path, PathBuf}, process::ExitCode};
|
||||||
use anyhow::bail;
|
|
||||||
use camino::Utf8PathBuf;
|
use camino::Utf8PathBuf;
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use mclangc;
|
use mclangc;
|
||||||
|
|
36
src/cli.rs
36
src/cli.rs
|
@ -1,3 +1,39 @@
|
||||||
|
use clap::{error::ErrorKind, CommandFactory};
|
||||||
|
|
||||||
|
use crate::logger::Level;
|
||||||
|
|
||||||
|
|
||||||
|
#[derive(Debug, clap::Parser)]
|
||||||
|
pub struct CliArgs {
|
||||||
|
/// Output more info, will get overwritten if -q|--quiet is specified
|
||||||
|
#[arg(long, short)]
|
||||||
|
verbose: bool,
|
||||||
|
/// Output nothing, except errors, will overwrite -v|--verbose
|
||||||
|
#[arg(long, short)]
|
||||||
|
quiet: bool,
|
||||||
|
/// Output file
|
||||||
|
#[arg(long, short, default_value="a.out")]
|
||||||
|
pub output: String,
|
||||||
|
/// All input files
|
||||||
|
#[clap(num_args = 1..)]
|
||||||
|
pub input: Vec<String>
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CliArgs {
|
||||||
|
pub fn set_log_level(&self) {
|
||||||
|
if self.quiet {
|
||||||
|
unsafe {
|
||||||
|
crate::logger::LEVEL = Level::Error;
|
||||||
|
}
|
||||||
|
} else if self.verbose {
|
||||||
|
unsafe {
|
||||||
|
crate::logger::LEVEL = Level::Debug;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn validate(&self) {
|
||||||
|
if self.input.len() < 1 {
|
||||||
|
CliArgs::command().error(ErrorKind::TooFewValues, "at least one value is required for '<INPUT>' but none was supplied").exit();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
112
src/logger.rs
112
src/logger.rs
|
@ -1,8 +1,8 @@
|
||||||
use crate::common::Loc;
|
pub static mut LEVEL: Level = Level::Info;
|
||||||
|
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
#[repr(u8)]
|
#[repr(u8)]
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
pub enum Level {
|
pub enum Level {
|
||||||
Off = 0,
|
Off = 0,
|
||||||
Error,
|
Error,
|
||||||
|
@ -20,26 +20,38 @@ const C_INFO: &'static str = "\x1B[1;32m";
|
||||||
const C_DEBUG: &'static str = "\x1B[1;35m";
|
const C_DEBUG: &'static str = "\x1B[1;35m";
|
||||||
const C_HELP: &'static str = "\x1B[1;36m";
|
const C_HELP: &'static str = "\x1B[1;36m";
|
||||||
|
|
||||||
pub fn _log(level: Level, str: &str) {
|
pub fn _log(dbg_loc: Option<(&'static str, u32, u32)>, level: Level, str: &str) {
|
||||||
match level {
|
if level > unsafe { LEVEL }{
|
||||||
Level::Off => return,
|
return;
|
||||||
Level::Error => println!("{C_ERROR}error{C_RESET}: {str}"),
|
}
|
||||||
Level::Warn => println!("{C_WARN}warn{C_RESET}: {str}"),
|
let dbg = if let Some((file, line, col)) = dbg_loc {
|
||||||
Level::Info => println!("{C_INFO}info{C_RESET}: {str}"),
|
format!("{file}:{line}:{col}: ")
|
||||||
Level::Help => println!("{C_HELP}help{C_RESET}: {str}"),
|
} else {String::new()};
|
||||||
Level::Debug => println!("{C_DEBUG}debug{C_RESET}: {str}"),
|
match level {
|
||||||
}
|
Level::Off => return,
|
||||||
|
Level::Error => println!("{dbg}{C_ERROR}error{C_RESET}: {str}"),
|
||||||
|
Level::Warn => println!("{dbg}{C_WARN}warn{C_RESET}: {str}"),
|
||||||
|
Level::Info => println!("{dbg}{C_INFO}info{C_RESET}: {str}"),
|
||||||
|
Level::Help => println!("{dbg}{C_HELP}help{C_RESET}: {str}"),
|
||||||
|
Level::Debug => println!("{dbg}{C_DEBUG}debug{C_RESET}: {str}"),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn _log_with_loc(loc: &Loc, level: Level, str: &str) {
|
pub fn _log_with_loc(dbg_loc: Option<(&'static str, u32, u32)>, loc: String, level: Level, str: &str) {
|
||||||
match level {
|
if level > unsafe { LEVEL }{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let dbg = if let Some((file, line, col)) = dbg_loc {
|
||||||
|
format!("{file}:{line}:{col}: ")
|
||||||
|
} else {String::new()};
|
||||||
|
match level {
|
||||||
Level::Off => return,
|
Level::Off => return,
|
||||||
Level::Error => println!("{loc}: {C_ERROR}error{C_RESET}: {str}"),
|
Level::Error => println!("{dbg}{loc}: {C_ERROR}error{C_RESET}: {str}"),
|
||||||
Level::Warn => println!("{loc}: {C_WARN}warn{C_RESET}: {str}"),
|
Level::Warn => println!("{dbg}{loc}: {C_WARN}warn{C_RESET}: {str}"),
|
||||||
Level::Info => println!("{loc}: {C_INFO}info{C_RESET}: {str}"),
|
Level::Info => println!("{dbg}{loc}: {C_INFO}info{C_RESET}: {str}"),
|
||||||
Level::Help => println!("{loc}: {C_HELP}help{C_RESET}: {str}"),
|
Level::Help => println!("{dbg}{loc}: {C_HELP}help{C_RESET}: {str}"),
|
||||||
Level::Debug => println!("{loc}: {C_DEBUG}debug{C_RESET}: {str}"),
|
Level::Debug => println!("{dbg}{loc}: {C_DEBUG}debug{C_RESET}: {str}"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
|
@ -47,31 +59,51 @@ pub mod log {
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! error {
|
macro_rules! error {
|
||||||
($($arg:tt)*) => {
|
($($arg:tt)*) => {
|
||||||
crate::logger::_log(crate::logger::Level::Error, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log(Some((file!(), line!(), column!())), crate::logger::Level::Error, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log(None, crate::logger::Level::Error, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! warn {
|
macro_rules! warn {
|
||||||
($($arg:tt)*) => {
|
($($arg:tt)*) => {
|
||||||
crate::logger::_log(crate::logger::Level::Warn, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log(Some((file!(), line!(), column!())), crate::logger::Level::Warn, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log(None, crate::logger::Level::Warn, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! info {
|
macro_rules! info {
|
||||||
($($arg:tt)*) => {
|
($($arg:tt)*) => {
|
||||||
crate::logger::_log(crate::logger::Level::Info, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log(Some((file!(), line!(), column!())), crate::logger::Level::Info, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log(None, crate::logger::Level::Info, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! help {
|
macro_rules! help {
|
||||||
($($arg:tt)*) => {
|
($($arg:tt)*) => {
|
||||||
crate::logger::_log(crate::logger::Level::Help, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log(Some((file!(), line!(), column!())), crate::logger::Level::Help, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log(None, crate::logger::Level::Help, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! debug {
|
macro_rules! debug {
|
||||||
($($arg:tt)*) => {
|
($($arg:tt)*) => {
|
||||||
crate::logger::_log(crate::logger::Level::Debug, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log(Some((file!(), line!(), column!())), crate::logger::Level::Debug, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log(None, crate::logger::Level::Debug, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -79,31 +111,51 @@ pub mod log {
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! lerror {
|
macro_rules! lerror {
|
||||||
($loc:expr, $($arg:tt)*) => {
|
($loc:expr, $($arg:tt)*) => {
|
||||||
crate::logger::_log_with_loc($loc, crate::logger::Level::Error, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log_with_loc(Some((file!(), line!(), column!())), format!("{}", $loc), crate::logger::Level::Error, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log_with_loc(None, format!("{}", $loc), crate::logger::Level::Error, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! lwarn {
|
macro_rules! lwarn {
|
||||||
($loc:expr, $($arg:tt)*) => {
|
($loc:expr, $($arg:tt)*) => {
|
||||||
crate::logger::_log_with_loc($loc, crate::logger::Level::Warn, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log_with_loc(Some((file!(), line!(), column!())), format!("{}", $loc), crate::logger::Level::Warn, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log_with_loc(None, format!("{}", $loc), crate::logger::Level::Warn, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! linfo {
|
macro_rules! linfo {
|
||||||
($loc:expr, $($arg:tt)*) => {
|
($loc:expr, $($arg:tt)*) => {
|
||||||
crate::logger::_log_with_loc($loc, crate::logger::Level::Info, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log_with_loc(Some((file!(), line!(), column!())), format!("{}", $loc), crate::logger::Level::Info, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log_with_loc(None, format!("{}", $loc), crate::logger::Level::Info, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! lhelp {
|
macro_rules! lhelp {
|
||||||
($loc:expr, $($arg:tt)*) => {
|
($loc:expr, $($arg:tt)*) => {
|
||||||
crate::logger::_log_with_loc($loc, crate::logger::Level::Help, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log_with_loc(Some((file!(), line!(), column!())), format!("{}", $loc), crate::logger::Level::Help, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log_with_loc(None, format!("{}", $loc), crate::logger::Level::Help, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! ldebug {
|
macro_rules! ldebug {
|
||||||
($loc:expr, $($arg:tt)*) => {
|
($loc:expr, $($arg:tt)*) => {
|
||||||
crate::logger::_log_with_loc($loc, crate::logger::Level::Debug, &format!($($arg)*))
|
if cfg!(debug_assertions) {
|
||||||
|
crate::logger::_log_with_loc(Some((file!(), line!(), column!())), format!("{}", $loc), crate::logger::Level::Debug, &format!($($arg)*))
|
||||||
|
} else {
|
||||||
|
crate::logger::_log_with_loc(None, format!("{}", $loc), crate::logger::Level::Debug, &format!($($arg)*))
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
40
src/main.rs
40
src/main.rs
|
@ -1,11 +1,39 @@
|
||||||
|
use std::{path::PathBuf, process::ExitCode};
|
||||||
|
|
||||||
|
use clap::Parser;
|
||||||
|
// Importing logger here too cause the logger macros dont work outside the mclanc lib
|
||||||
|
mod logger;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
fn main() -> anyhow::Result<()> {
|
|
||||||
let data = std::fs::read_to_string("test.mcl").unwrap();
|
|
||||||
|
|
||||||
let tokens = mclangc::tokeniser::tokenise(&data, "test.mcl")?;
|
fn main() -> ExitCode {
|
||||||
let prog = mclangc::parser::parse_program(tokens)?;
|
let cli = mclangc::cli::CliArgs::parse();
|
||||||
mclangc::validator::validate_code(&prog);
|
cli.set_log_level();
|
||||||
Ok(())
|
cli.validate();
|
||||||
|
for file in &cli.input {
|
||||||
|
let fp = PathBuf::from(file);
|
||||||
|
if !fp.exists() {
|
||||||
|
error!("File {fp:?} doesnt exits, exiting");
|
||||||
|
return ExitCode::FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
let data = std::fs::read_to_string(fp).unwrap();
|
||||||
|
info!("Tokenising {file}");
|
||||||
|
let Ok(tokens) = mclangc::tokeniser::tokenise(&data, &file) else {
|
||||||
|
error!("Failed to tokenise file, exiting");
|
||||||
|
return ExitCode::FAILURE;
|
||||||
|
};
|
||||||
|
info!("Parsing {file}");
|
||||||
|
let Ok(prog) = mclangc::parser::parse_program(tokens) else {
|
||||||
|
error!("Failed to parse file, exiting");
|
||||||
|
return ExitCode::FAILURE;
|
||||||
|
};
|
||||||
|
info!("Validating {file}");
|
||||||
|
let Ok(validated) = mclangc::validator::validate_code(&prog) else {
|
||||||
|
error!("Failed to validate file, exiting");
|
||||||
|
return ExitCode::FAILURE;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
ExitCode::SUCCESS
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ use std::collections::HashMap;
|
||||||
|
|
||||||
use anyhow::{bail, Result};
|
use anyhow::{bail, Result};
|
||||||
|
|
||||||
use crate::{debug, lerror, parser::{typ::parse_type, Punctuation}, tokeniser::Token};
|
use crate::{debug, error, lerror, parser::{typ::parse_type, Punctuation}, tokeniser::Token};
|
||||||
|
|
||||||
use super::{ast::{expr::{Block, CallParams, Expr, IfBranchExpr, IfExpr, Path}, literal::Literal, TokenType}, parse_item, utils, Delimiter, Keyword};
|
use super::{ast::{expr::{Block, CallParams, Expr, IfBranchExpr, IfExpr, Path}, literal::Literal, TokenType}, parse_item, utils, Delimiter, Keyword};
|
||||||
|
|
||||||
|
@ -108,12 +108,11 @@ pub fn parse_expr(tokens: &mut Vec<Token>, precedence: usize, consume_semi: bool
|
||||||
if let Some(_) = utils::check_from_many(tokens, BINOP_LIST) {
|
if let Some(_) = utils::check_from_many(tokens, BINOP_LIST) {
|
||||||
return Ok(Some(parse_binop(tokens, res, precedence)?));
|
return Ok(Some(parse_binop(tokens, res, precedence)?));
|
||||||
} else {
|
} else {
|
||||||
|
if consume_semi {
|
||||||
|
_ = utils::check_consume_or_err(tokens, TokenType::Punct(Punctuation::Semi), "Expected ; at the end of the expression")?;
|
||||||
|
}
|
||||||
return Ok(Some(res));
|
return Ok(Some(res));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
|
||||||
if consume_semi {
|
|
||||||
_ = utils::check_consume_or_err(tokens, TokenType::Punct(Punctuation::Semi), "Expected ; at the end of the expression")?;
|
|
||||||
}
|
}
|
||||||
Ok(res)
|
Ok(res)
|
||||||
}
|
}
|
||||||
|
@ -138,7 +137,17 @@ fn parse_if(tokens: &mut Vec<Token>) -> Result<IfExpr> {
|
||||||
lerror!(loc.loc(), "Expected test for if statement, got nothing");
|
lerror!(loc.loc(), "Expected test for if statement, got nothing");
|
||||||
bail!("")
|
bail!("")
|
||||||
};
|
};
|
||||||
let block = parse_block(tokens)?;
|
let block = if let Some(_) = utils::check(tokens, TokenType::Delim(Delimiter::CurlyL)) {
|
||||||
|
if let Some(_) = utils::check_2_last(tokens, TokenType::Delim(Delimiter::CurlyR)) {
|
||||||
|
_ = utils::check_consume(tokens, TokenType::Delim(Delimiter::CurlyR));
|
||||||
|
Block(Vec::new())
|
||||||
|
} else {
|
||||||
|
parse_block(tokens)?
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
lerror!(loc.loc(), "Expected '{{'");
|
||||||
|
bail!("")
|
||||||
|
};
|
||||||
if let Some(_) = utils::check_consume(tokens, TokenType::Keyword(Keyword::Else)) {
|
if let Some(_) = utils::check_consume(tokens, TokenType::Keyword(Keyword::Else)) {
|
||||||
if let Some(_) = utils::check(tokens, TokenType::Keyword(Keyword::If)) {
|
if let Some(_) = utils::check(tokens, TokenType::Keyword(Keyword::If)) {
|
||||||
let branch = IfBranchExpr::ElseIf(Box::new(parse_if(tokens)?));
|
let branch = IfBranchExpr::ElseIf(Box::new(parse_if(tokens)?));
|
||||||
|
@ -181,7 +190,7 @@ fn parse_for_loop(tokens: &mut Vec<Token>) -> Result<Expr> {
|
||||||
lerror!(loc.loc(), "Expected init stat for a for loop, got nothing");
|
lerror!(loc.loc(), "Expected init stat for a for loop, got nothing");
|
||||||
bail!("")
|
bail!("")
|
||||||
};
|
};
|
||||||
_ = utils::check_consume_or_err(tokens, TokenType::Punct(Punctuation::Semi), "");
|
// Semicolon parsed out by parse_item above
|
||||||
let Some(test) = parse_expr(tokens, 0, false)? else {
|
let Some(test) = parse_expr(tokens, 0, false)? else {
|
||||||
lerror!(loc.loc(), "Expected test comparrison for a for loop, got nothing");
|
lerror!(loc.loc(), "Expected test comparrison for a for loop, got nothing");
|
||||||
bail!("")
|
bail!("")
|
||||||
|
@ -440,14 +449,14 @@ pub fn parse_block(tokens: &mut Vec<Token>) -> Result<Block> {
|
||||||
utils::check_consume_or_err(tokens, TokenType::Delim(Delimiter::CurlyL), "")?;
|
utils::check_consume_or_err(tokens, TokenType::Delim(Delimiter::CurlyL), "")?;
|
||||||
let mut items = Vec::new();
|
let mut items = Vec::new();
|
||||||
while !tokens.is_empty() {
|
while !tokens.is_empty() {
|
||||||
|
if let Some(_) = utils::check(tokens, TokenType::Delim(Delimiter::CurlyR)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
if let Some(item) = parse_item(tokens)? {
|
if let Some(item) = parse_item(tokens)? {
|
||||||
items.push(item);
|
items.push(item);
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if let Some(_) = utils::check(tokens, TokenType::Delim(Delimiter::CurlyR)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
utils::check_consume_or_err(tokens, TokenType::Delim(Delimiter::CurlyR), "")?;
|
utils::check_consume_or_err(tokens, TokenType::Delim(Delimiter::CurlyR), "")?;
|
||||||
Ok(Block(items))
|
Ok(Block(items))
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
use crate::tokeniser::Token;
|
use crate::{parser::Delimiter, tokeniser::Token};
|
||||||
|
|
||||||
use super::{ast::{typ::Type, TokenType}, expr::parse_expr, utils, Keyword, Punctuation};
|
use super::{ast::{typ::Type, TokenType}, expr::parse_expr, utils, Keyword, Punctuation};
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ pub fn parse_type(tokens: &mut Vec<Token>) -> Result<Type> {
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut typ;
|
let mut typ;
|
||||||
if let Some(_) = utils::check(tokens, TokenType::Delim(super::Delimiter::SquareL)) {
|
if let Some(_) = utils::check_consume(tokens, TokenType::Delim(super::Delimiter::SquareL)) {
|
||||||
let itm_typ = parse_type(tokens)?;
|
let itm_typ = parse_type(tokens)?;
|
||||||
if let Some(_) = utils::check_consume(tokens, TokenType::Punct(Punctuation::Semi)) {
|
if let Some(_) = utils::check_consume(tokens, TokenType::Punct(Punctuation::Semi)) {
|
||||||
let count = parse_expr(tokens, 0, false)?.unwrap();
|
let count = parse_expr(tokens, 0, false)?.unwrap();
|
||||||
|
@ -28,6 +28,7 @@ pub fn parse_type(tokens: &mut Vec<Token>) -> Result<Type> {
|
||||||
inner: Box::new(itm_typ),
|
inner: Box::new(itm_typ),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_ = utils::check_consume_or_err(tokens, TokenType::Delim(Delimiter::SquareR), "")?;
|
||||||
} else {
|
} else {
|
||||||
let ident = utils::check_consume_or_err(tokens, TokenType::ident(""), "a")?;
|
let ident = utils::check_consume_or_err(tokens, TokenType::ident(""), "a")?;
|
||||||
typ = Type::Owned(ident.tt().unwrap_ident());
|
typ = Type::Owned(ident.tt().unwrap_ident());
|
||||||
|
|
|
@ -1,6 +1,33 @@
|
||||||
Program {
|
Program {
|
||||||
ast: Block(
|
ast: Block(
|
||||||
[],
|
[
|
||||||
|
Statement(
|
||||||
|
Enum {
|
||||||
|
name: Ident(
|
||||||
|
"Foo",
|
||||||
|
),
|
||||||
|
fields: [],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Statement(
|
||||||
|
Enum {
|
||||||
|
name: Ident(
|
||||||
|
"Bar",
|
||||||
|
),
|
||||||
|
fields: [
|
||||||
|
Ident(
|
||||||
|
"A",
|
||||||
|
),
|
||||||
|
Ident(
|
||||||
|
"B",
|
||||||
|
),
|
||||||
|
Ident(
|
||||||
|
"C",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
structs: {},
|
structs: {},
|
||||||
enums: {},
|
enums: {},
|
||||||
|
|
|
@ -0,0 +1,7 @@
|
||||||
|
enum Foo {}
|
||||||
|
|
||||||
|
enum Bar {
|
||||||
|
A,
|
||||||
|
B,
|
||||||
|
C,
|
||||||
|
}
|
|
@ -1,6 +1,57 @@
|
||||||
Program {
|
Program {
|
||||||
ast: Block(
|
ast: Block(
|
||||||
[],
|
[
|
||||||
|
Expr(
|
||||||
|
BinOp {
|
||||||
|
typ: Eq,
|
||||||
|
left: UnOp {
|
||||||
|
typ: Star,
|
||||||
|
right: Path(
|
||||||
|
Path(
|
||||||
|
[
|
||||||
|
Ident(
|
||||||
|
"a",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
right: BinOp {
|
||||||
|
typ: EqEq,
|
||||||
|
left: BinOp {
|
||||||
|
typ: Star,
|
||||||
|
left: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 1,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
right: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 3,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
right: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 4,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
structs: {},
|
structs: {},
|
||||||
enums: {},
|
enums: {},
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
*a = 1 * 3 == 4;
|
||||||
|
b.c = 3/4 == *a;
|
||||||
|
c->d = (a->b.c->d) / 2;
|
||||||
|
*d->e.f = 2 / a->b.c->d;
|
||||||
|
e = a->b.c->d / 2;
|
||||||
|
f = a.b.c.d / 2;
|
||||||
|
g = a.b[*a.c] * 5;
|
||||||
|
|
|
@ -1,6 +1,191 @@
|
||||||
Program {
|
Program {
|
||||||
ast: Block(
|
ast: Block(
|
||||||
[],
|
[
|
||||||
|
Statement(
|
||||||
|
Fn {
|
||||||
|
struct_name: None,
|
||||||
|
name: Ident(
|
||||||
|
"main",
|
||||||
|
),
|
||||||
|
params: [
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"argc",
|
||||||
|
),
|
||||||
|
Owned(
|
||||||
|
Ident(
|
||||||
|
"i32",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"argv",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Array {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Str",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
mutable: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
ret_type: Some(
|
||||||
|
Owned(
|
||||||
|
Ident(
|
||||||
|
"i32",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
qual_const: false,
|
||||||
|
qual_extern: None,
|
||||||
|
body: Some(
|
||||||
|
Block(
|
||||||
|
[
|
||||||
|
Expr(
|
||||||
|
Return(
|
||||||
|
Some(
|
||||||
|
Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 0,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Statement(
|
||||||
|
Fn {
|
||||||
|
struct_name: Some(
|
||||||
|
Ident(
|
||||||
|
"Baz",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
name: Ident(
|
||||||
|
"main",
|
||||||
|
),
|
||||||
|
params: [
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"self",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Baz",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: true,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"a",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Foo",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"b",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Bar",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: true,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
ret_type: Some(
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Nya",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
qual_const: false,
|
||||||
|
qual_extern: None,
|
||||||
|
body: None,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Statement(
|
||||||
|
Fn {
|
||||||
|
struct_name: Some(
|
||||||
|
Ident(
|
||||||
|
"Baz",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
name: Ident(
|
||||||
|
"main",
|
||||||
|
),
|
||||||
|
params: [
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"a",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Foo",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"b",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Bar",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: true,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
ret_type: Some(
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"Nya",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
qual_const: false,
|
||||||
|
qual_extern: None,
|
||||||
|
body: None,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
structs: {},
|
structs: {},
|
||||||
enums: {},
|
enums: {},
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
fn main(argc: i32, argv: &[Str]) -> i32 {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
fn Baz.main(self: &mut Baz, a: &Foo, b: &mut Bar) -> &Nya;
|
||||||
|
fn Baz.main(a: &Foo, b: &mut Bar) -> &Nya;
|
|
@ -1,6 +1,38 @@
|
||||||
Program {
|
Program {
|
||||||
ast: Block(
|
ast: Block(
|
||||||
[],
|
[
|
||||||
|
Expr(
|
||||||
|
If(
|
||||||
|
IfExpr {
|
||||||
|
test: BinOp {
|
||||||
|
typ: Gt,
|
||||||
|
left: Path(
|
||||||
|
Path(
|
||||||
|
[
|
||||||
|
Ident(
|
||||||
|
"i",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
right: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 3,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
body: Block(
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
else_if: None,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
structs: {},
|
structs: {},
|
||||||
enums: {},
|
enums: {},
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
|
||||||
|
if i > 3 {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if 1 {
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if 1 {
|
||||||
|
|
||||||
|
} else if a > 3 {
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
}
|
|
@ -1,6 +1,110 @@
|
||||||
Program {
|
Program {
|
||||||
ast: Block(
|
ast: Block(
|
||||||
[],
|
[
|
||||||
|
Expr(
|
||||||
|
ForLoop {
|
||||||
|
init: Statement(
|
||||||
|
Let {
|
||||||
|
name: Ident(
|
||||||
|
"i",
|
||||||
|
),
|
||||||
|
typ: None,
|
||||||
|
val: Some(
|
||||||
|
Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 0,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
test: BinOp {
|
||||||
|
typ: Lt,
|
||||||
|
left: Path(
|
||||||
|
Path(
|
||||||
|
[
|
||||||
|
Ident(
|
||||||
|
"i",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
right: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 10,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
on_loop: BinOp {
|
||||||
|
typ: AddEq,
|
||||||
|
left: Path(
|
||||||
|
Path(
|
||||||
|
[
|
||||||
|
Ident(
|
||||||
|
"i",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
right: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 1,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
body: Block(
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Expr(
|
||||||
|
WhileLoop {
|
||||||
|
test: BinOp {
|
||||||
|
typ: Gt,
|
||||||
|
left: Path(
|
||||||
|
Path(
|
||||||
|
[
|
||||||
|
Ident(
|
||||||
|
"i",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
right: Literal(
|
||||||
|
Number(
|
||||||
|
Number {
|
||||||
|
val: 3,
|
||||||
|
base: 10,
|
||||||
|
signed: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
body: Block(
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Expr(
|
||||||
|
InfLoop {
|
||||||
|
body: Block(
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
structs: {},
|
structs: {},
|
||||||
enums: {},
|
enums: {},
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
for let i = 0; i < 10; i += 1 {}
|
||||||
|
while i > 3 {}
|
||||||
|
loop {}
|
|
@ -1,6 +1,37 @@
|
||||||
Program {
|
Program {
|
||||||
ast: Block(
|
ast: Block(
|
||||||
[],
|
[
|
||||||
|
Statement(
|
||||||
|
Struct {
|
||||||
|
name: Ident(
|
||||||
|
"foo_t",
|
||||||
|
),
|
||||||
|
fields: [],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Statement(
|
||||||
|
Struct {
|
||||||
|
name: Ident(
|
||||||
|
"bar_t",
|
||||||
|
),
|
||||||
|
fields: [
|
||||||
|
(
|
||||||
|
Ident(
|
||||||
|
"a",
|
||||||
|
),
|
||||||
|
Ref {
|
||||||
|
inner: Owned(
|
||||||
|
Ident(
|
||||||
|
"bar_t",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
mutable: false,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
structs: {},
|
structs: {},
|
||||||
enums: {},
|
enums: {},
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
|
||||||
|
struct foo_t {}
|
||||||
|
|
||||||
|
struct bar_t {
|
||||||
|
a: &bar_t
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user