From 03de1d323107916b57def2a39238da14ba23291b Mon Sep 17 00:00:00 2001 From: Nadrieril Date: Wed, 8 May 2019 00:45:59 +0200 Subject: Fix Context shifting --- dhall/src/core/context.rs | 39 ++++++++++++++++++++++++--------------- 1 file changed, 24 insertions(+), 15 deletions(-) (limited to 'dhall/src/core') diff --git a/dhall/src/core/context.rs b/dhall/src/core/context.rs index f07c638..0483c4d 100644 --- a/dhall/src/core/context.rs +++ b/dhall/src/core/context.rs @@ -62,27 +62,24 @@ impl Context { } None } - fn shift(&self, delta: isize, var: &AlphaVar) -> Self - where - T: Shift, - { - Context(Rc::new( - self.0 - .iter() - .map(|(l, i)| (l.clone(), i.shift(delta, var))) - .collect(), - )) - } - fn subst_shift(&self, var: &AlphaVar, val: &Typed) -> Self + /// Given a var that makes sense in the current context, map the given function in such a way + /// that the passed variable always makes sense in the context of the passed item. + /// Once we pass the variable definition, the variable doesn't make sense anymore so we just + /// copy the remaining items. + fn do_with_var( + &self, + var: &AlphaVar, + mut f: impl FnMut(&AlphaVar, &CtxItem) -> CtxItem, + ) -> Self where - T: Clone + Subst, + T: Clone, { let mut vec = Vec::new(); vec.reserve(self.0.len()); let mut var = var.clone(); let mut iter = self.0.iter().rev(); for (l, i) in iter.by_ref() { - vec.push((l.clone(), i.subst_shift(&var, val))); + vec.push((l.clone(), f(&var, i))); if let CtxItem::Kept(_, _) = i { if var == l.into() { break; @@ -97,6 +94,18 @@ impl Context { vec.reverse(); Context(Rc::new(vec)) } + fn shift(&self, delta: isize, var: &AlphaVar) -> Self + where + T: Clone + Shift, + { + self.do_with_var(var, |var, i| i.shift(delta, &var)) + } + fn subst_shift(&self, var: &AlphaVar, val: &Typed) -> Self + where + T: Clone + Subst, + { + self.do_with_var(var, |var, i| i.subst_shift(&var, val)) + } } impl NormalizationContext { @@ -161,7 +170,7 @@ impl Shift for CtxItem { } } -impl Shift for Context { +impl Shift for Context { fn shift(&self, delta: isize, var: &AlphaVar) -> Self { self.shift(delta, var) } -- cgit v1.2.3