From 0b2d2ccee2023198d60b48154b9b211e47b782ec Mon Sep 17 00:00:00 2001 From: NanoTech Date: Thu, 8 Dec 2016 09:20:39 +0000 Subject: Replace Cow<'i, str> with &'i str in Expr Cow::Owned is never used in Expr --- src/context.rs | 5 ++--- src/core.rs | 62 +++++++++++++++++++++++++---------------------------- src/grammar.lalrpop | 7 +++--- src/lexer.rs | 10 ++++----- src/typecheck.rs | 24 ++++++++++----------- 5 files changed, 50 insertions(+), 58 deletions(-) (limited to 'src') diff --git a/src/context.rs b/src/context.rs index 4d6abf2..b72182d 100644 --- a/src/context.rs +++ b/src/context.rs @@ -1,4 +1,3 @@ -use std::borrow::Cow; use std::collections::HashMap; /// A `(Context a)` associates `Text` labels with values of type `a` @@ -14,7 +13,7 @@ use std::collections::HashMap; /// `n`th occurrence of a given key. /// #[derive(Debug, Clone)] -pub struct Context<'i, T>(HashMap, Vec>); +pub struct Context<'i, T>(HashMap<&'i str, Vec>); impl<'i, T> Context<'i, T> { /// An empty context with no key-value pairs @@ -41,7 +40,7 @@ impl<'i, T> Context<'i, T> { impl<'i, T: Clone> Context<'i, T> { /// Add a key-value pair to the `Context` - pub fn insert(&self, k: Cow<'i, str>, v: T) -> Self { + pub fn insert(&self, k: &'i str, v: T) -> Self { let mut ctx = (*self).clone(); { let m = ctx.0.entry(k).or_insert(vec![]); diff --git a/src/core.rs b/src/core.rs index 5bf90db..0bc42d4 100644 --- a/src/core.rs +++ b/src/core.rs @@ -1,5 +1,4 @@ #![allow(non_snake_case)] -use std::borrow::Cow; use std::collections::HashMap; use std::path::PathBuf; @@ -95,8 +94,8 @@ pub enum Path { /// Zero indices are omitted when pretty-printing `Var`s and non-zero indices /// appear as a numeric suffix. /// -#[derive(Debug, Clone, PartialEq, Eq)] // (Eq, Show) -pub struct V<'i>(pub Cow<'i, str>, pub usize); +#[derive(Debug, Copy, Clone, PartialEq, Eq)] // (Eq, Show) +pub struct V<'i>(pub &'i str, pub usize); /* instance IsString Var where @@ -115,15 +114,15 @@ pub enum Expr<'i, S, A> { /// `Var (V x n) ~ x@n` Var(V<'i>), /// `Lam x A b ~ λ(x : A) -> b` - Lam(Cow<'i, str>, Box>, Box>), + Lam(&'i str, Box>, Box>), /// `Pi "_" A B ~ A -> B` /// `Pi x A B ~ ∀(x : A) -> B` - Pi(Cow<'i, str>, Box>, Box>), + Pi(&'i str, Box>, Box>), /// `App f A ~ f A` App(Box>, Box>), /// `Let x Nothing r e ~ let x = r in e` /// `Let x (Just t) r e ~ let x : t = r in e` - Let(Cow<'i, str>, Option>>, Box>, Box>), + Let(&'i str, Option>>, Box>, Box>), /// `Annot x t ~ x : t` Annot(Box>, Box>), /// `Bool ~ Bool` @@ -198,19 +197,19 @@ pub enum Expr<'i, S, A> { /// `OptionalFold ~ Optional/fold` OptionalFold, /// `Record [(k1, t1), (k2, t2)] ~ { k1 : t1, k2 : t1 }` - Record(HashMap, Expr<'i, S, A>>), + Record(HashMap<&'i str, Expr<'i, S, A>>), /// `RecordLit [(k1, v1), (k2, v2)] ~ { k1 = v1, k2 = v2 }` - RecordLit(HashMap, Expr<'i, S, A>>), + RecordLit(HashMap<&'i str, Expr<'i, S, A>>), /// `Union [(k1, t1), (k2, t2)] ~ < k1 : t1, k2 : t2 >` - Union(HashMap, Expr<'i, S, A>>), + Union(HashMap<&'i str, Expr<'i, S, A>>), /// `UnionLit (k1, v1) [(k2, t2), (k3, t3)] ~ < k1 = t1, k2 : t2, k3 : t3 >` - UnionLit(Cow<'i, str>, Box>, HashMap, Expr<'i, S, A>>), + UnionLit(&'i str, Box>, HashMap<&'i str, Expr<'i, S, A>>), /// `Combine x y ~ x ∧ y` Combine(Box>, Box>), /// `Merge x y t ~ merge x y : t` Merge(Box>, Box>, Box>), /// `Field e x ~ e.x` - Field(Box>, Cow<'i, str>), + Field(Box>, &'i str), /// `Note S x ~ e` Note(S, Box>), /// `Embed path ~ path` @@ -254,18 +253,18 @@ impl<'i, S, A> Expr<'i, S, A> { impl<'i> From<&'i str> for V<'i> { fn from(s: &'i str) -> Self { - V(Cow::Borrowed(s), 0) + V(s, 0) } } impl<'i, S, A> From<&'i str> for Expr<'i, S, A> { fn from(s: &'i str) -> Self { - Expr::Var(V(Cow::Borrowed(s), 0)) + Expr::Var(s.into()) } } pub fn pi<'i, S, A, Name, Et, Ev>(var: Name, ty: Et, value: Ev) -> Expr<'i, S, A> - where Name: Into>, + where Name: Into<&'i str>, Et: Into>, Ev: Into> { @@ -378,30 +377,28 @@ pub fn shift<'i, S, T, A>(d: isize, v: V, e: Expr<'i, S, A>) -> Expr<'i, T, A> A: ::std::fmt::Debug, { use Expr::*; + let V(x, n) = v; match e { Const(a) => Const(a), Var(V(x2, n2)) => { - let V(x, n) = v; let n3 = if x == x2 && n <= n2 { add_ui(n2, d) } else { n2 }; Var(V(x2, n3)) } Lam(x2, tA, b) => { - let V(x, n) = v; let n2 = if x == x2 { n + 1 } else { n }; - let tA2 = shift(d, V(x.clone(), n ), *tA); - let b2 = shift(d, V(x, n2), *b); + let tA2 = shift(d, V(x, n ), *tA); + let b2 = shift(d, V(x, n2), *b); Lam(x2, bx(tA2), bx(b2)) } Pi(x2, tA, tB) => { - let V(x, n) = v; let n2 = if x == x2 { n + 1 } else { n }; - let tA2 = shift(d, V(x.clone(), n ), *tA); + let tA2 = shift(d, V(x, n ), *tA); let tB2 = shift(d, V(x, n2), *tB); pi(x2, tA2, tB2) } App(f, a) => { - let f2 = shift(d, v.clone(), *f); - let a2 = shift(d, v, *a); + let f2 = shift(d, v, *f); + let a2 = shift(d, v, *a); App(bx(f2), bx(a2)) } /* @@ -419,7 +416,7 @@ shift d v (Annot a b) = Annot a' b' b' = shift d v b */ BoolLit(a) => BoolLit(a), - BoolAnd(a, b) => BoolAnd(bx(shift(d, v.clone(), *a)), bx(shift(d, v, *b))), + BoolAnd(a, b) => BoolAnd(bx(shift(d, v, *a)), bx(shift(d, v, *b))), /* shift d v (BoolOr a b) = BoolOr a' b' where @@ -533,30 +530,29 @@ pub fn subst<'i, S, T, A>(v: V<'i>, a: Expr<'i, S, A>, b: Expr<'i, T, A>) -> Exp A: Clone + ::std::fmt::Debug, { use Expr::*; + let V(x, n) = v; match (a, b) { (_, Const(a)) => Const(a), (e, Lam(y, tA, b)) => { - let V(x, n) = v; let n2 = if x == y { n + 1 } else { n }; - let tA2 = subst(V(x.clone(), n), e.clone(), *tA); - let b2 = subst(V(x, n2), shift(1, V(y.clone(), 0), e), *b); + let tA2 = subst(V(x, n), e.clone(), *tA); + let b2 = subst(V(x, n2), shift(1, V(y, 0), e), *b); Lam(y, bx(tA2), bx(b2)) } (e, Pi(y, tA, tB)) => { - let V(x, n) = v; let n2 = if x == y { n + 1 } else { n }; - let tA2 = subst(V(x.clone(), n), e.clone(), *tA); - let tB2 = subst(V(x, n2), shift(1, V(y.clone(), 0), e), *tB); + let tA2 = subst(V(x, n), e.clone(), *tA); + let tB2 = subst(V(x, n2), shift(1, V(y, 0), e), *tB); pi(y, tA2, tB2) } (e, App(f, a)) => { - let f2 = subst(v.clone(), e.clone(), *f); + let f2 = subst(v, e.clone(), *f); let a2 = subst(v, e, *a); app(f2, a2) } (e, Var(v2)) => if v == v2 { e } else { Var(v2) }, (e, ListLit(a, b)) => { - let b2 = b.into_iter().map(|be| subst(v.clone(), e.clone(), be)).collect(); + let b2 = b.into_iter().map(|be| subst(v, e.clone(), be)).collect(); let a2 = subst(v, e, *a); ListLit(bx(a2), b2) } @@ -599,8 +595,8 @@ pub fn normalize(e: Expr) -> Expr App(f, a) => match normalize::(*f) { Lam(x, _A, b) => { // Beta reduce let vx0 = V(x, 0); - let a2 = shift::( 1, vx0.clone(), *a); - let b2 = subst::(vx0.clone(), a2, *b); + let a2 = shift::( 1, vx0, *a); + let b2 = subst::(vx0, a2, *b); let b3 = shift::(-1, vx0, b2); normalize(b3) } diff --git a/src/grammar.lalrpop b/src/grammar.lalrpop index 91f8b8d..63c17ad 100644 --- a/src/grammar.lalrpop +++ b/src/grammar.lalrpop @@ -4,7 +4,6 @@ use core::Expr::*; use grammar_util::*; use lexer::*; -use std::borrow::Cow; use std::collections::HashMap; use std::iter; use std::iter::FromIterator; @@ -25,7 +24,7 @@ extern { Nat => Tok::Natural(), Text => Tok::Text(), Bool => Tok::Bool(), - Label => Tok::Identifier(>), + Label => Tok::Identifier(<&'input str>), Const => Tok::Const(), Let => Tok::Keyword(Keyword::Let), In => Tok::Keyword(Keyword::In), @@ -65,7 +64,7 @@ ExprB: BoxExpr<'input> = { Lambda "("