complete first draft of validator

This commit is contained in:
Scott Richmond 2024-12-10 23:42:05 -05:00
parent 9542dcf5e2
commit eafe7a7fa9
2 changed files with 166 additions and 32 deletions

View File

@ -148,11 +148,11 @@ pub fn run(src: &'static str) {
let dummy_prelude = vec![]; let dummy_prelude = vec![];
let mut valxor = validator::Validator::new(&ast, span, &dummy_prelude); let mut v6or = validator::Validator::new(&ast, span, &dummy_prelude);
// valxor.validate(); v6or.validate();
// dbg!(valxor); dbg!(v6or);
let mut ctx = prelude(); let mut ctx = prelude();
ctx.ast = * ctx.ast = *
@ -167,8 +167,10 @@ pub fn run(src: &'static str) {
pub fn main() { pub fn main() {
let src = " let src = "
let #{:a (x, y), :b [1, 2, (a, b)]} = #{:a (1, 2), :b [1, 2, (7, 8)]} let bar = :bar
(x, y, a, b) match :foo with {
:foo -> bar
}
"; ";
run(src); run(src);
// struct_scalpel::print_dissection_info::<value::Value>() // struct_scalpel::print_dissection_info::<value::Value>()

View File

@ -101,6 +101,21 @@ impl<'a> Validator<'a> {
self.status.used_bindings.push(name); self.status.used_bindings.push(name);
} }
fn arity(&mut self) -> Arity {
let Ast::MatchClause(pattern, ..) = self.ast else {
unreachable!("internal Ludus error")
};
let (Ast::TuplePattern(members), _) = pattern.as_ref() else {
unreachable!("internal Ludus error");
};
let last_member = members.last();
match last_member {
None => Arity::Fixed(0),
Some((Ast::Splattern(..), _)) => Arity::Splat(members.len() as u8),
Some(_) => Arity::Fixed(members.len() as u8),
}
}
pub fn validate(&mut self) { pub fn validate(&mut self) {
use Ast::*; use Ast::*;
let root = self.ast; let root = self.ast;
@ -297,22 +312,27 @@ impl<'a> Validator<'a> {
self.validate(); self.validate();
} }
} }
WhenClause(cond, body) => todo!(), WhenClause(cond, body) => {
When(clauses) => { let tailpos = self.status.tail_position;
// let tailpos = self.status.tail_position; self.status.tail_position = false;
// for (clause, _) in clauses { let (expr, span) = cond.as_ref();
// self.status.tail_position = false; self.ast = expr;
// let (expr, span) = clause.cond.clone(); self.span = *span;
// self.ast = &expr; self.validate();
// self.span = span;
// self.validate();
// self.status.tail_position = tailpos; self.status.tail_position = tailpos;
// let (expr, span) = clause.body; let (expr, span) = body.as_ref();
// self.ast = &expr; self.ast = expr;
// self.span = span; self.span = *span;
// self.validate(); self.validate();
// } }
When(clauses) => {
for clause in clauses {
let (expr, span) = clause;
self.ast = expr;
self.span = *span;
self.validate();
}
} }
// binding forms // binding forms
@ -339,14 +359,40 @@ impl<'a> Validator<'a> {
self.span = *span; self.span = *span;
self.validate(); self.validate();
} }
MatchClause(pattern, guard, body) => todo!(), MatchClause(pattern, guard, body) => {
Match(scrutinee, clauses) => { let to = self.locals.len();
let (expr, span) = scrutinee.as_ref();
let (patt, span) = pattern.as_ref();
self.ast = patt;
self.span = *span;
self.validate();
if let Some((expr, span)) = guard.as_ref() {
self.ast = expr; self.ast = expr;
self.span = *span; self.span = *span;
self.validate(); self.validate();
} }
let (expr, span) = body.as_ref();
self.ast = expr;
self.span = *span;
self.validate();
self.locals.truncate(to);
}
Match(scrutinee, clauses) => {
let (expr, span) = scrutinee.as_ref();
self.ast = expr;
self.span = *span;
self.validate();
for clause in clauses {
let (expr, span) = clause;
self.ast = expr;
self.span = *span;
self.validate();
}
}
FnDeclaration(name) => { FnDeclaration(name) => {
let tailpos = self.status.tail_position; let tailpos = self.status.tail_position;
self.status.tail_position = false; self.status.tail_position = false;
@ -367,11 +413,16 @@ impl<'a> Validator<'a> {
} }
let from = self.status.used_bindings.len(); let from = self.status.used_bindings.len();
let arities = HashSet::new(); let mut arities = HashSet::new();
for clause in clauses { for clause in clauses {
// TODO: validate all parts of clauses // TODO: validate all parts of clauses
let (expr, span) = clause;
self.ast = expr;
self.span = *span;
// add clause arity to arities // add clause arity to arities
arities.insert(self.arity());
self.validate();
} }
// this should be right // this should be right
@ -389,7 +440,11 @@ impl<'a> Validator<'a> {
let info = FnInfo::Defined(arities, closed_over); let info = FnInfo::Defined(arities, closed_over);
self.define_fn(name.to_string(), info) let root_ptr: *const Ast = root;
self.fn_info.insert(root_ptr, info.clone());
self.define_fn(name.to_string(), info);
} }
Panic(msg) => { Panic(msg) => {
@ -448,7 +503,24 @@ impl<'a> Validator<'a> {
self.status.in_loop = true; self.status.in_loop = true;
self.status.loop_arity = input.len() as u8; self.status.loop_arity = input.len() as u8;
// for clause in body {} for clause in body {
let (expr, span) = clause;
self.ast = expr;
self.span = *span;
match self.arity() {
Arity::Fixed(clause_arity) => {
if clause_arity != arity {
self.err(format!("mismatched arity: expected {arity} arguments in `loop` clause; got {clause_arity}"))
}
}
Arity::Splat(clause_arity) => {
if clause_arity > arity {
self.err(format!("mismathced arity: expected {arity} arguments in `loop` clause; this clause takes {clause_arity} or more"))
}
}
};
self.validate();
}
self.status.in_loop = in_loop; self.status.in_loop = in_loop;
self.status.loop_arity = arity; self.status.loop_arity = arity;
@ -484,13 +556,73 @@ impl<'a> Validator<'a> {
self.bind(name.to_string()); self.bind(name.to_string());
} }
}, },
InterpolatedPattern(parts, _) => todo!(), InterpolatedPattern(parts, _) => {
AsPattern(name, r#type) => todo!(), for (part, span) in parts {
Splattern(splatted) => todo!(), if let StringPart::Word(name) = part {
TuplePattern(tuple) => todo!(), self.span = *span;
ListPattern(list) => todo!(), match self.bound(name) {
PairPattern(key, patt) => todo!(), Some(_) => self.err(format!("name `{name}` is already bound")),
DictPattern(dict) => todo!(), None => self.bind(name.to_string()),
}
}
}
}
AsPattern(name, r#type) => {
match self.bound(name) {
Some((name, _span, _)) => {
self.err(format!("name `{name}` is already bound"));
}
None => {
self.bind(name.to_string());
}
}
let as_type = *r#type;
match as_type {
"nil" | "bool" | "number" | "keyword" | "string" | "tuple" | "dict"
| "list" | "fn" | "box" => (),
_ => self.err(format!("unknown type `:{as_type}`")),
}
}
Splattern(splatted) => {
if !self.status.last_term {
self.err("splats in patterns must come last".to_string());
}
match splatted.as_ref() {
(Ast::Placeholder, _) => (),
(Ast::Word(name), span) => match self.bound(name) {
Some(_) => {
self.span = *span;
self.err(format!("name `{name}` is already bound"))
}
None => self.bind(name.to_string()),
},
_ => unreachable!(),
}
}
TuplePattern(terms) | ListPattern(terms) | DictPattern(terms) => {
if terms.is_empty() {
return;
}
for term in terms.iter().take(terms.len() - 1) {
let (patt, span) = term;
self.ast = patt;
self.span = *span;
self.validate();
}
self.status.last_term = true;
let (patt, span) = terms.last().unwrap();
self.ast = patt;
self.span = *span;
self.validate();
self.status.last_term = false;
}
PairPattern(_, patt) => {
let (patt, span) = patt.as_ref();
self.ast = patt;
self.span = *span;
self.validate();
}
// terminals can never be invalid // terminals can never be invalid
Nil | Boolean(_) | Number(_) | Keyword(_) | String(_) => (), Nil | Boolean(_) | Number(_) | Keyword(_) | String(_) => (),
// terminal patterns can never be invalid // terminal patterns can never be invalid