The pipeline is finally reasonably clean

This commit is contained in:
2023-09-12 01:26:46 +01:00
parent 6693d93944
commit 8c866967a9
86 changed files with 1959 additions and 1393 deletions

View File

@@ -1,18 +1,16 @@
use hashbrown::HashMap;
use super::alias_map::AliasMap;
use super::decls::{InjectedAsFn, UpdatedFn};
use crate::ast::{Expr, Rule};
use crate::interner::Tok;
use crate::representations::project::{ProjectExt, ProjectModule};
use crate::representations::tree::{ModEntry, ModMember};
use crate::representations::project::{ItemKind, ProjectMod};
use crate::representations::tree::ModMember;
use crate::representations::VName;
use crate::utils::Substack;
fn resolve_rec(
namespace: &[Tok<String>],
alias_map: &AliasMap,
) -> Option<Vec<Tok<String>>> {
) -> Option<VName> {
if let Some(alias) = alias_map.resolve(namespace) {
Some(alias.clone())
} else if let Some((foot, body)) = namespace.split_last() {
@@ -28,7 +26,7 @@ fn resolve(
namespace: &[Tok<String>],
alias_map: &AliasMap,
injected_as: &impl InjectedAsFn,
) -> Option<Vec<Tok<String>>> {
) -> Option<VName> {
injected_as(namespace).or_else(|| {
let next_v = resolve_rec(namespace, alias_map)?;
Some(injected_as(&next_v).unwrap_or(next_v))
@@ -45,69 +43,44 @@ fn process_expr(
.unwrap_or_else(|| expr.clone())
}
// TODO: replace is_injected with injected_as
/// Replace all aliases with the name they're originally defined as
fn apply_aliases_rec(
path: Substack<Tok<String>>,
module: &ProjectModule<VName>,
module: &mut ProjectMod<VName>,
alias_map: &AliasMap,
injected_as: &impl InjectedAsFn,
updated: &impl UpdatedFn,
) -> ProjectModule<VName> {
let items = (module.items.iter())
.map(|(name, ent)| {
let ModEntry { exported, member } = ent;
let member = match member {
ModMember::Item(expr) =>
ModMember::Item(process_expr(expr, alias_map, injected_as)),
ModMember::Sub(module) => {
let subpath = path.push(name.clone());
let new_mod = if !updated(&subpath.iter().rev_vec_clone()) {
module.clone()
} else {
apply_aliases_rec(subpath, module, alias_map, injected_as, updated)
};
ModMember::Sub(new_mod)
},
};
(name.clone(), ModEntry { exported: *exported, member })
})
.collect::<HashMap<_, _>>();
let rules = (module.extra.rules.iter())
.map(|rule| {
let Rule { pattern, prio, template } = rule;
Rule {
prio: *prio,
pattern: (pattern.iter())
.map(|expr| process_expr(expr, alias_map, injected_as))
.collect::<Vec<_>>(),
template: (template.iter())
.map(|expr| process_expr(expr, alias_map, injected_as))
.collect::<Vec<_>>(),
}
})
.collect::<Vec<_>>();
ProjectModule {
items,
imports: module.imports.clone(),
extra: ProjectExt {
rules,
exports: (module.extra.exports.iter())
.map(|(k, v)| {
(k.clone(), resolve(v, alias_map, injected_as).unwrap_or(v.clone()))
})
.collect(),
file: module.extra.file.clone(),
imports_from: module.extra.imports_from.clone(),
},
) {
for (name, entry) in module.entries.iter_mut() {
match &mut entry.member {
ModMember::Sub(sub) => {
let subpath = path.push(name.clone());
apply_aliases_rec(subpath, sub, alias_map, injected_as, updated)
},
ModMember::Item(it) => match &mut it.kind {
ItemKind::None => (),
ItemKind::Const(expr) =>
*expr = process_expr(expr, alias_map, injected_as),
ItemKind::Alias(name) =>
if let Some(alt) = alias_map.resolve(&name) {
*name = alt.clone()
},
},
_ => (),
}
}
for Rule { pattern, prio, template } in module.extra.rules.iter_mut() {
for expr in pattern.iter_mut().chain(template.iter_mut()) {
*expr = process_expr(expr, alias_map, injected_as)
}
}
}
pub fn apply_aliases(
module: &ProjectModule<VName>,
module: &mut ProjectMod<VName>,
alias_map: &AliasMap,
injected_as: &impl InjectedAsFn,
updated: &impl UpdatedFn,
) -> ProjectModule<VName> {
) {
apply_aliases_rec(Substack::Bottom, module, alias_map, injected_as, updated)
}