summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRichard Eisenberg <eir@cis.upenn.edu>2016-03-25 17:25:25 -0400
committerRichard Eisenberg <eir@cis.upenn.edu>2016-03-26 16:28:27 -0400
commit9f73e46c0f34b8b5e8318e6b488b7dade7db68e3 (patch)
treebcc17d8c3204c28fcae9bf0aaa009e2630d5dd49
parent12a76bebe0864cdf1c9088ed16175d7b34369e24 (diff)
downloadhaskell-9f73e46c0f34b8b5e8318e6b488b7dade7db68e3.tar.gz
Clarify Note [Kind coercions in Unify]
-rw-r--r--compiler/types/Unify.hs25
1 files changed, 24 insertions, 1 deletions
diff --git a/compiler/types/Unify.hs b/compiler/types/Unify.hs
index ed4b2243ff..0b5df14962 100644
--- a/compiler/types/Unify.hs
+++ b/compiler/types/Unify.hs
@@ -79,10 +79,31 @@ may look nothing alike.
So, we pass a kind coercion to the match/unify worker. This coercion witnesses
the equality between the substed kind of the left-hand type and the substed
-kind of the right-hand type. To get this coercion, we first have to match/unify
+kind of the right-hand type. Note that we do not unify kinds at the leaves
+(as we did previously). We thus have
+
+INVARIANT: In the call
+ unify_ty ty1 ty2 kco
+it must be that subst(kco) :: subst(kind(ty1)) ~N subst(kind(ty2)), where
+`subst` is the ambient substitution in the UM monad.
+
+To get this coercion, we first have to match/unify
the kinds before looking at the types. Happily, we need look only one level
up, as all kinds are guaranteed to have kind *.
+When we're working with type applications (either TyConApp or AppTy) we
+need to worry about establishing INVARIANT, as the kinds of the function
+& arguments aren't (necessarily) included in the kind of the result.
+When unifying two TyConApps, this is easy, because the two TyCons are
+the same. Their kinds are thus the same. As long as we unify left-to-right,
+we'll be sure to unify types' kinds before the types themselves. (For example,
+think about Proxy :: forall k. k -> *. Unifying the first args matches up
+the kinds of the second args.)
+
+For AppTy, we must unify the kinds of the functions, but once these are
+unified, we can continue unifying arguments without worrying further about
+kinds.
+
We thought, at one point, that this was all unnecessary: why should casts
be in types in the first place? But they do. In
dependent/should_compile/KindEqualities2, we see, for example
@@ -760,6 +781,7 @@ unify_ty_app ty1 ty1args ty2 ty2args
| otherwise
= do { let ki1 = typeKind ty1
ki2 = typeKind ty2
+ -- See Note [Kind coercions in Unify]
; unify_ty ki1 ki2 (mkNomReflCo liftedTypeKind)
; unify_ty ty1 ty2 (mkNomReflCo ki1)
; unify_tys ty1args ty2args }
@@ -770,6 +792,7 @@ unify_tys orig_xs orig_ys
where
go [] [] = return ()
go (x:xs) (y:ys)
+ -- See Note [Kind coercions in Unify]
= do { unify_ty x y (mkNomReflCo $ typeKind x)
; go xs ys }
go _ _ = maybeApart -- See Note [Lists of different lengths are MaybeApart]