diff options
| author | simonpj@microsoft.com <unknown> | 2010-10-07 10:37:00 +0000 | 
|---|---|---|
| committer | simonpj@microsoft.com <unknown> | 2010-10-07 10:37:00 +0000 | 
| commit | afd6da0d938747f45b16c8d6eb5d786e59a21218 (patch) | |
| tree | 25d64cc7f7f67fcfea2f68a0903c1c9c4a39fb0c /compiler | |
| parent | 5c248c7dfa9b789df31ca5186c4f181f1e8eb42b (diff) | |
| download | haskell-afd6da0d938747f45b16c8d6eb5d786e59a21218.tar.gz | |
Improve the rule-matcher
Previously it was rejecting the match
  Template: forall s t. map s t
  Actual:   map Int t
which should obviously be fine.  It turns out that this kind of match
comes up when specialising.  By freshening that t we could avoid the
difficulty, but morally the (forall t) binds t and the rule should
be alpha-equivalent regardless of the forall'd variables.
This patch makes it so, and incidentally makes matching a little
more efficient.  See Note [Eta expansion] in VarEnv.
Diffstat (limited to 'compiler')
| -rw-r--r-- | compiler/basicTypes/VarEnv.lhs | 56 | ||||
| -rw-r--r-- | compiler/specialise/Rules.lhs | 4 | 
2 files changed, 41 insertions, 19 deletions
| diff --git a/compiler/basicTypes/VarEnv.lhs b/compiler/basicTypes/VarEnv.lhs index bf3f96dd26..5a087079a3 100644 --- a/compiler/basicTypes/VarEnv.lhs +++ b/compiler/basicTypes/VarEnv.lhs @@ -36,6 +36,7 @@ module VarEnv (  	-- ** Operations on RnEnv2s  	mkRnEnv2, rnBndr2, rnBndrs2, rnOccL, rnOccR, inRnEnvL, inRnEnvR,  	rnBndrL, rnBndrR, nukeRnEnvL, nukeRnEnvR, extendRnInScopeList, +        rnEtaL, rnEtaR,  	rnInScope, rnInScopeSet, lookupRnInScope,  	-- * TidyEnv and its operation @@ -237,36 +238,43 @@ rnBndr2 (RV2 { envL = envL, envR = envR, in_scope = in_scope }) bL bR  rnBndrL :: RnEnv2 -> Var -> (RnEnv2, Var)  -- ^ Similar to 'rnBndr2' but used when there's a binder on the left --- side only. Useful when eta-expanding +-- side only.  rnBndrL (RV2 { envL = envL, envR = envR, in_scope = in_scope }) bL    = (RV2 { envL     = extendVarEnv envL bL new_b -	 , envR     = extendVarEnv envR new_b new_b 	-- Note [rnBndrLR] +         , envR     = envR  	 , in_scope = extendInScopeSet in_scope new_b }, new_b)    where      new_b = uniqAway in_scope bL  rnBndrR :: RnEnv2 -> Var -> (RnEnv2, Var)  -- ^ Similar to 'rnBndr2' but used when there's a binder on the right --- side only. Useful when eta-expanding +-- side only.  rnBndrR (RV2 { envL = envL, envR = envR, in_scope = in_scope }) bR -  = (RV2 { envL     = extendVarEnv envL new_b new_b	-- Note [rnBndrLR] -	 , envR     = extendVarEnv envR bR new_b +  = (RV2 { envR     = extendVarEnv envR bR new_b +         , envL     = envL  	 , in_scope = extendInScopeSet in_scope new_b }, new_b)    where      new_b = uniqAway in_scope bR --- Note [rnBndrLR]  --- ~~~~~~~~~~~~~~~ --- Notice that in rnBndrL, rnBndrR, we extend envR, envL respectively --- with a binding [new_b -> new_b], where new_b is the new binder. --- This is important when doing eta expansion; e.g. matching (\x.M) ~ N --- In effect we switch to (\x'.M) ~ (\x'.N x'), where x' is new_b --- So we must add x' to the env of both L and R.  (x' is fresh, so it --- can't capture anything in N.)   --- --- If we don't do this, we can get silly matches like ---	forall a.  \y.a  ~   v --- succeeding with [x -> v y], which is bogus of course  +rnEtaL :: RnEnv2 -> Var -> (RnEnv2, Var) +-- ^ Similar to 'rnBndrL' but used for eta expansion +-- See Note [Eta expansion] +rnEtaL (RV2 { envL = envL, envR = envR, in_scope = in_scope }) bL +  = (RV2 { envL     = extendVarEnv envL bL new_b +	 , envR     = extendVarEnv envR new_b new_b 	-- Note [Eta expansion] +	 , in_scope = extendInScopeSet in_scope new_b }, new_b) +  where +    new_b = uniqAway in_scope bL + +rnEtaR :: RnEnv2 -> Var -> (RnEnv2, Var) +-- ^ Similar to 'rnBndr2' but used for eta expansion +-- See Note [Eta expansion] +rnEtaR (RV2 { envL = envL, envR = envR, in_scope = in_scope }) bR +  = (RV2 { envL     = extendVarEnv envL new_b new_b	-- Note [Eta expansion] +	 , envR     = extendVarEnv envR bR new_b +	 , in_scope = extendInScopeSet in_scope new_b }, new_b) +  where +    new_b = uniqAway in_scope bR  rnOccL, rnOccR :: RnEnv2 -> Var -> Var  -- ^ Look up the renaming of an occurrence in the left or right term @@ -287,6 +295,20 @@ nukeRnEnvL env = env { envL = emptyVarEnv }  nukeRnEnvR env = env { envR = emptyVarEnv }  \end{code} +Note [Eta expansion] +~~~~~~~~~~~~~~~~~~~~ +When matching +     (\x.M) ~ N +we rename x to x' with, where x' is not in scope in  +either term.  Then we want to behave as if we'd seen +     (\x'.M) ~ (\x'.N x') +Since x' isn't in scope in N, the form (\x'. N x') doesn't +capture any variables in N.  But we must nevertheless extend +the envR with a binding [x' -> x'], to support the occurs check. +For example, if we don't do this, we can get silly matches like +	forall a.  (\y.a)  ~   v +succeeding with [a -> v y], which is bogus of course. +  %************************************************************************  %*									* diff --git a/compiler/specialise/Rules.lhs b/compiler/specialise/Rules.lhs index 87999a4dfd..b4b996248b 100644 --- a/compiler/specialise/Rules.lhs +++ b/compiler/specialise/Rules.lhs @@ -652,7 +652,7 @@ match idu menv subst (Lam x1 e1) (Lam x2 e2)  match idu menv subst (Lam x1 e1) e2    = match idu menv' subst e1 (App e2 (varToCoreExpr new_x))    where -    (rn_env', new_x) = rnBndrL (me_env menv) x1 +    (rn_env', new_x) = rnEtaL (me_env menv) x1      menv' = menv { me_env = rn_env' }  -- Eta expansion the other way @@ -660,7 +660,7 @@ match idu menv subst (Lam x1 e1) e2  match idu menv subst e1 (Lam x2 e2)    = match idu menv' subst (App e1 (varToCoreExpr new_x)) e2    where -    (rn_env', new_x) = rnBndrR (me_env menv) x2 +    (rn_env', new_x) = rnEtaR (me_env menv) x2      menv' = menv { me_env = rn_env' }  match idu menv subst (Case e1 x1 ty1 alts1) (Case e2 x2 ty2 alts2) | 
