use crate::lang::env::{Env, EnvEntry, UniqueVariableId};
use crate::lang::UnionIndex;
use crate::lang::{PartialExpr, TcEnv};
use std::collections::HashMap;
impl TcEnv {
pub fn simplify(&mut self, i: UnionIndex) -> UnionIndex {
self.simplify_inner(i, &Env::new(), &mut HashMap::new())
}
fn simplify_inner(
&mut self,
i: UnionIndex,
s: &Env,
var_map: &mut HashMap<UniqueVariableId, usize>,
) -> UnionIndex {
let e_new = match self.values[i.0] {
PartialExpr::Type => PartialExpr::Type,
PartialExpr::Let(v, b) => {
let v = self.simplify_inner(v, s, var_map);
let id = self.new_tc_id();
var_map.insert(id, var_map.len());
let b = self.simplify_inner(b, &s.cons(EnvEntry::RType(id)), var_map);
var_map.remove(&id);
PartialExpr::Let(v, b)
}
PartialExpr::DeBruijnIndex(v) => match s.get(v) {
Some(EnvEntry::CType(_, _)) | Some(EnvEntry::CSubst(_, _)) => unreachable!(),
Some(EnvEntry::RType(id)) => PartialExpr::DeBruijnIndex(var_map.len() - var_map[id] - 1),
Some(EnvEntry::RSubst(subst, subst_env)) => {
return self.simplify_inner(*subst, subst_env, var_map)
}
None => PartialExpr::DeBruijnIndex(v),
},
PartialExpr::FnType(a, b) => {
let a = self.simplify_inner(a, s, var_map);
let id = self.new_tc_id();
var_map.insert(id, var_map.len());
let b = self.simplify_inner(b, &s.cons(EnvEntry::RType(id)), var_map);
var_map.remove(&id);
PartialExpr::FnType(a, b)
}
PartialExpr::FnConstruct(a, b) => {
let a = self.simplify_inner(a, s, var_map);
let id = self.new_tc_id();
var_map.insert(id, var_map.len());
let b = self.simplify_inner(b, &s.cons(EnvEntry::RType(id)), var_map);
var_map.remove(&id);
PartialExpr::FnConstruct(a, b)
}
PartialExpr::FnDestruct(a, b) => {
let a = self.simplify_inner(a, s, var_map);
let b = self.simplify_inner(b, s, var_map);
PartialExpr::FnDestruct(a, b)
}
PartialExpr::Free => PartialExpr::Free,
PartialExpr::Shift(b, i) => {
return self.simplify_inner(b, &s.shift(i.min(s.len())), var_map)
}
};
self.store(e_new, self.value_origins[i.0])
}
}