use crate::etypes::{
BoundedTyvar, Ctx, Defined, FreeTyvar, Handleable, ImportExport, TypeBound, Tyvar,
};
use crate::substitute::{self, Substitution, Unvoidable};
pub enum ResolvedTyvar<'a> {
Definite(Defined<'a>),
#[allow(unused)]
Bound(u32),
E(u32, u32, TypeBound<'a>),
U(u32, u32, TypeBound<'a>),
}
impl<'p, 'a> Ctx<'p, 'a> {
fn lookup_uvar<'c>(&'c self, o: u32, i: u32) -> &'c (BoundedTyvar<'a>, bool) {
&self.parents().nth(o as usize).unwrap().uvars[i as usize]
}
fn lookup_evar<'c>(&'c self, o: u32, i: u32) -> &'c (BoundedTyvar<'a>, Option<Defined<'a>>) {
&self.parents().nth(o as usize).unwrap().evars[i as usize]
}
pub fn var_bound<'c>(&'c self, tv: &Tyvar) -> &'c TypeBound<'a> {
match tv {
Tyvar::Bound(_) => panic!("Requested bound for Bound tyvar"),
Tyvar::Free(FreeTyvar::U(o, i)) => &self.lookup_uvar(*o, *i).0.bound,
Tyvar::Free(FreeTyvar::E(o, i)) => &self.lookup_evar(*o, *i).0.bound,
}
}
pub fn resolve_tyvar<'c>(&'c self, v: &Tyvar) -> ResolvedTyvar<'a> {
let check_deftype = |dt: &Defined<'a>| match dt {
Defined::Handleable(Handleable::Var(v_)) => self.resolve_tyvar(v_),
_ => ResolvedTyvar::Definite(dt.clone()),
};
match *v {
Tyvar::Bound(i) => ResolvedTyvar::Bound(i),
Tyvar::Free(FreeTyvar::E(o, i)) => {
let (tv, def) = self.lookup_evar(o, i);
match (&tv.bound, def) {
(TypeBound::Eq(dt), _) => check_deftype(dt),
(_, Some(dt)) => check_deftype(dt),
(tb, _) => ResolvedTyvar::E(o, i, tb.clone()),
}
}
Tyvar::Free(FreeTyvar::U(o, i)) => {
let (tv, _) = self.lookup_uvar(o, i);
match &tv.bound {
TypeBound::Eq(dt) => check_deftype(dt),
tb => ResolvedTyvar::U(o, i, tb.clone()),
}
}
}
}
pub fn bound_to_evars(
&mut self,
origin: Option<&'a str>,
vs: &[BoundedTyvar<'a>],
) -> substitute::Opening {
let mut sub = substitute::Opening::new(false, self.evars.len() as u32);
for var in vs {
let var = var.push_origin(origin.map(ImportExport::Export));
let bound = sub.bounded_tyvar(&var).not_void();
self.evars.push((bound, None));
sub.next();
}
sub
}
pub fn bound_to_uvars(
&mut self,
origin: Option<&'a str>,
vs: &[BoundedTyvar<'a>],
imported: bool,
) -> substitute::Opening {
let mut sub = substitute::Opening::new(true, self.uvars.len() as u32);
for var in vs {
let var = var.push_origin(origin.map(ImportExport::Import));
let bound = sub.bounded_tyvar(&var).not_void();
self.uvars.push((bound, imported));
sub.next();
}
sub
}
}