diff --git a/kernel/src/declaration.rs b/kernel/src/declaration.rs index 81feeddaf152dd6655ec51854008abd22676b620..ae64e75261e57ae72f8842e5a0e445b1ac63fdb6 100644 --- a/kernel/src/declaration.rs +++ b/kernel/src/declaration.rs @@ -1,20 +1,50 @@ use crate::term::Term; +use crate::universe::UniverseLevel; - -/// Declarations constructed through commands. A declaration describes a constant in the environment, whether it's a definition with +/// Declarations constructed through commands. A declaration describes a constant in the environment, whether it's a definition with /// a corresponding term, or an axiom with only a type. -/// univ_vars corresponds to the number of universe variables bound to the declaration. +/// univ_vars corresponds to the number of universe variables bound to the declaration. /// No universe variable can be "free" in a term, meaning that for all Var(i) in ty or term, i<univ_vars. /// Additionally, ty and term *should* in theory always have the same number of universe variables, and as such, only a single method is needed. -/// However, additional checks to ensure this invariant will have to be put in place. For now, when constructing declarations, only the number of -/// universes in term are counted. Since ty has to be infered, or at least checked against the infered type before being added, it has to have <= universes than term. -#[derive(Clone)] +/// However, additional checks to ensure this invariant will have to be put in place. For now, when constructing declarations, only the number of +/// universes in ty are counted. +#[derive(Clone, Debug)] pub struct Declaration { - ty : Term, - term : Option<Term>, - univ_vars : usize + ty: Term, + term: Option<Term>, + univ_vars: usize, } impl Declaration { - -} \ No newline at end of file + pub fn make(term: Option<Term>, ty: Term) -> Declaration { + Declaration { + ty: ty.clone(), + term, + univ_vars: ty.univ_vars(), + } + } + + pub fn get_type(&self, vec: &[UniverseLevel]) -> Term { + if self.univ_vars != vec.len() { + panic!( + "wrong type of universe arguments, expected {}, found {}", + self.univ_vars, + vec.len() + ) + } else { + self.ty.substitute_univs(vec) + } + } + + pub fn get_term(&self, vec: &[UniverseLevel]) -> Option<Term> { + if self.univ_vars != vec.len() { + panic!( + "wrong type of universe arguments, expected {}, found {}", + self.univ_vars, + vec.len() + ) + } else { + self.clone().term.map(|x| x.substitute_univs(vec)) + } + } +} diff --git a/kernel/src/environment.rs b/kernel/src/environment.rs index 01721aca033df6aa75fbd1ad9049ad599f6b0024..93118e9456712f0da17faa09336a31ae265d39f1 100644 --- a/kernel/src/environment.rs +++ b/kernel/src/environment.rs @@ -1,11 +1,13 @@ use crate::error::KernelError; use crate::term::Term; use crate::declaration::Declaration; +use crate::term::Term; +use crate::universe::UniverseLevel; use derive_more::From; use std::collections::{hash_map, HashMap}; /// Global Environment, contains the term and type of every definitions, denoted by their strings. -#[derive(Clone, Default, From)] +#[derive(Clone, Default, Debug, From)] pub struct Environment(HashMap<String, Declaration>); impl Environment { @@ -25,12 +27,12 @@ impl Environment { } /// Returns the term linked to a definition in a given environment. - pub fn get_term(&self, s: &String) -> Option<Term> { - self.0.get(s).map(|(t, _)| t.clone()) + pub fn get_term(&self, s: &String, vec: &[UniverseLevel]) -> Option<Term> { + self.0.get(s).and_then(|decl| decl.get_term(vec)) } /// Returns the type linked to a definition in a given environment. - pub fn get_type(&self, s: &String) -> Option<Term> { - self.0.get(s).map(|(_, t)| t.clone()) + pub fn get_type(&self, s: &String, vec: &[UniverseLevel]) -> Option<Term> { + self.0.get(s).map(|decl| decl.get_type(vec)) } } diff --git a/kernel/src/lib.rs b/kernel/src/lib.rs index d3642508b65403f12893aa24a14d9982d048e283..59a7d9f01df6c2cc17db258fb881677d78fae0cd 100644 --- a/kernel/src/lib.rs +++ b/kernel/src/lib.rs @@ -4,12 +4,12 @@ #![feature(box_syntax)] mod command; +mod declaration; mod environment; mod error; mod term; mod type_checker; mod universe; -mod declaration; pub use derive_more; pub use num_bigint; diff --git a/kernel/src/term.rs b/kernel/src/term.rs index 4abd86860d0c36ec2afb7ae3463b1e894e519b45..bba9d85284c1fe51187a3ae102c44c71339bae40 100644 --- a/kernel/src/term.rs +++ b/kernel/src/term.rs @@ -12,8 +12,11 @@ pub enum Term { #[display(fmt = "{}", _0)] Var(DeBruijnIndex), + /// Constants can be universe polymorphic. As such, it is important to know what universes + /// they're assigned to. The vector serves to represent the universes by which the variables + /// present in the constant need to be substituted for. #[display(fmt = "{}", _0)] - Const(String), + Const(String, Vec<UniverseLevel>), #[display(fmt = "Prop")] Prop, @@ -40,9 +43,9 @@ impl Term { App(box Abs(_, box t1), box t2) => t1.substitute(t2, 1), App(box t1, box t2) => App(box t1.beta_reduction(env), box t2.clone()), Abs(x, box t) => Abs(x.clone(), box t.beta_reduction(env)), - Const(s) => match env.get_term(s) { + Const(s, vec) => match env.get_term(s, vec) { Some(t) => t, - None => unreachable!(), + None => self.clone(), }, _ => self.clone(), } @@ -91,13 +94,35 @@ impl Term { whnf @ Abs(_, _) => App(box whnf, t2.clone()).beta_reduction(env).whnf(env), _ => self.clone(), }, - Const(s) => match env.get_term(s) { + Const(s, vec) => match env.get_term(s, vec) { Some(t) => t, - None => unreachable!(), + None => self.clone(), }, _ => self.clone(), } } + + pub fn univ_vars(&self) -> usize { + match self { + Var(_) => 0, + Const(..) => 0, + Prop => 0, + Type(u) => u.clone().univ_vars(), + App(t1, t2) | Abs(t1, t2) | Prod(t1, t2) => t1.univ_vars().max(t2.univ_vars()), + } + } + + pub fn substitute_univs(&self, vec: &[UniverseLevel]) -> Term { + match self { + Var(_) => self.clone(), + Const(..) => self.clone(), + Prop => self.clone(), + Type(u) => Type(u.clone().substitute(vec)), + App(t1, t2) => App(box t1.substitute_univs(vec), box t2.substitute_univs(vec)), + Abs(t1, t2) => Abs(box t1.substitute_univs(vec), box t2.substitute_univs(vec)), + Prod(t1, t2) => Prod(box t1.substitute_univs(vec), box t2.substitute_univs(vec)), + } + } } #[cfg(test)] diff --git a/kernel/src/type_checker.rs b/kernel/src/type_checker.rs index 611840a8f18eca941841902e1f270b49f85d8987..7c1130257f4b03bd5eb4618792ebe85ac8d3a67e 100644 --- a/kernel/src/type_checker.rs +++ b/kernel/src/type_checker.rs @@ -114,7 +114,7 @@ impl Term { Type(i) => Ok(Type(i.clone() + 1)), Var(i) => Ok(ctx.types[ctx.lvl - *i].clone()), - Const(s) => match env.get_type(s) { + Const(s, vec) => match env.get_type(s, vec) { Some(ty) => Ok(ty), None => Err(KernelError::ConstNotFound(s.clone())), }, @@ -342,7 +342,7 @@ mod tests { assert!(t.infer(&Environment::new()).is_err()); let wf_prod = Prod(box Prop, box Prop); - println!("{:?}",wf_prod.infer(&Environment::new())); + println!("{:?}", wf_prod.infer(&Environment::new())); assert!(wf_prod.check(&Type(0.into()), &Environment::new()).is_ok()); let wf_prod = Prod(box Prop, box Var(1.into())); @@ -379,8 +379,10 @@ mod tests { let mut env = Environment::new(); env.insert("foo".into(), id_prop.clone(), Prop).unwrap(); - assert!(t.check(&Const("foo".into()), &env).is_ok()); - assert!(id_prop.is_def_eq(&Const("foo".into()), &env).is_ok()); + assert!(t.check(&Const("foo".into(), Vec::new()), &env).is_ok()); + assert!(id_prop + .is_def_eq(&Const("foo".into(), Vec::new()), &env) + .is_ok()); } #[test] @@ -401,8 +403,10 @@ mod tests { let mut env = Environment::new(); env.insert("foo".into(), id_prop, Prop).unwrap(); - assert!(Const("foo".into()).infer(&env).is_ok()); - assert!(Const("foo".into()).infer(&Environment::new()).is_err()); + assert!(Const("foo".into(), Vec::new()).infer(&env).is_ok()); + assert!(Const("foo".into(), Vec::new()) + .infer(&Environment::new()) + .is_err()); } #[test] diff --git a/kernel/src/universe.rs b/kernel/src/universe.rs index 4ffa5a1b23a2550f68756c8e19bdc989d17630ce..2def2419097456b6168ca1b5e61c39a645f3b381 100644 --- a/kernel/src/universe.rs +++ b/kernel/src/universe.rs @@ -41,16 +41,23 @@ impl From<usize> for UniverseLevel { use UniverseLevel::*; impl UniverseLevel { - /*fn substitute(self, u: &UniverseLevel, lvl : usize) -> Self { + pub fn univ_vars(self) -> usize { match self { - UnivZero => UnivZero, - UnivSucc(v) => UnivSucc(box v.substitute(u),lvl), - UnivMax(u1, u2) => UnivMax(box u1.substitute(u,lvl), box u2.substitute(u,lvl)), - UnivVar(var) if var == lvl => u.clone(), - UnivVar(var) if var>lvl => UnivVar(var - 1), - _ => self + Zero => 0, + Succ(n) => n.univ_vars(), + Max(n, m) => n.univ_vars().max(m.univ_vars()), + Var(n) => n + 1, } - }*/ + } + + pub fn substitute(self, univs: &[UniverseLevel]) -> Self { + match self { + Zero => Zero, + Succ(v) => Succ(box v.substitute(univs)), + Max(u1, u2) => Max(box u1.substitute(univs), box u2.substitute(univs)), + Var(var) => univs[var].clone(), + } + } fn geq(&self, u2: &UniverseLevel, n: i64) -> bool { match (self, u2) { @@ -60,7 +67,7 @@ impl UniverseLevel { (_, Succ(box l)) if self.geq(l, n + 1) => true, (_, Max(box l1, box l2)) if self.geq(l1, n) || self.geq(l2, n) => true, (Max(box l1, box l2), _) if l1.geq(u2, n) && l2.geq(u2, n) => true, - _ => false + _ => false, } } @@ -75,7 +82,7 @@ mod tests { #[test] fn univ_eq() { - assert!(format!("{}",Max(box Zero, box Zero)) == "max (Zero) (Zero)"); + assert!(format!("{}", Max(box Zero, box Zero)) == "max (Zero) (Zero)"); assert!(&Zero.is_eq(&Default::default())); assert!(!&Zero.is_eq(&Succ(box Zero))); assert!(!&Succ(box Zero).is_eq(&Zero)); @@ -83,6 +90,7 @@ mod tests { assert!(&Max(box Zero, box Var(0)).is_eq(&Var(0))); assert!(&Max(box Var(1), box Var(0)).is_eq(&Max(box Var(0), box Var(1)))); assert!(!&Max(box Var(1), box Var(1)).is_eq(&Max(box Var(0), box Var(1)))); - assert!(&Succ(box Max(box Var(1), box Var(0))).is_eq(&Max(box Succ(box Var(0)), box Succ(box Var(1))))); + assert!(&Succ(box Max(box Var(1), box Var(0))) + .is_eq(&Max(box Succ(box Var(0)), box Succ(box Var(1))))); } }