{-# LANGUAGE ScopedTypeVariables, PatternSynonyms, MultiWayIf #-}
{-# LANGUAGE DeriveFunctor #-}
module GHC.Core.Unify (
tcMatchTy, tcMatchTyKi,
tcMatchTys, tcMatchTyKis,
tcMatchTyX, tcMatchTysX, tcMatchTyKisX,
tcMatchTyX_BM, ruleMatchTyKiX,
tcUnifyTy, tcUnifyTys, tcUnifyFunDeps, tcUnifyDebugger,
tcUnifyTysFG, tcUnifyTyForInjectivity,
BindTvFun, BindFamFun, BindFlag(..),
matchBindTv, alwaysBindTv, alwaysBindFam, dontCareBindFam,
UnifyResult, UnifyResultM(..), MaybeApartReason(..),
typesCantMatch, typesAreApart,
liftCoMatch
) where
import GHC.Prelude
import GHC.Types.Var
import GHC.Types.Var.Env
import GHC.Types.Var.Set
import GHC.Builtin.Names( tYPETyConKey, cONSTRAINTTyConKey )
import GHC.Core.Type hiding ( getTvSubstEnv )
import GHC.Core.Coercion hiding ( getCvSubstEnv )
import GHC.Core.Predicate( scopedSort )
import GHC.Core.TyCon
import GHC.Core.Predicate( CanEqLHS(..), canEqLHS_maybe )
import GHC.Core.TyCon.Env
import GHC.Core.TyCo.Rep
import GHC.Core.TyCo.Compare ( eqType, tcEqType, tcEqTyConAppArgs )
import GHC.Core.TyCo.FVs ( tyCoVarsOfCoList, tyCoFVsOfTypes )
import GHC.Core.TyCo.Subst ( mkTvSubst )
import GHC.Core.Map.Type
import GHC.Core.Multiplicity
import GHC.Utils.FV( FV, fvVarList )
import GHC.Utils.Misc
import GHC.Utils.Outputable
import GHC.Types.Basic( SwapFlag(..) )
import GHC.Types.Unique.FM
import GHC.Exts( oneShot )
import GHC.Utils.Panic
import GHC.Data.Pair
import GHC.Data.TrieMap
import GHC.Data.Maybe( orElse )
import Control.Monad
import qualified Data.Semigroup as S
import GHC.Builtin.Types.Prim (fUNTyCon)
data BindFlag
= BindMe
| DontBindMe
deriving BindFlag -> BindFlag -> Bool
(BindFlag -> BindFlag -> Bool)
-> (BindFlag -> BindFlag -> Bool) -> Eq BindFlag
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: BindFlag -> BindFlag -> Bool
== :: BindFlag -> BindFlag -> Bool
$c/= :: BindFlag -> BindFlag -> Bool
/= :: BindFlag -> BindFlag -> Bool
Eq
type BindTvFun = TyCoVar -> Type -> BindFlag
type BindFamFun = TyCon -> [Type] -> Type -> BindFlag
matchBindTv :: TyCoVarSet -> BindTvFun
matchBindTv :: TyCoVarSet -> BindTvFun
matchBindTv TyCoVarSet
tvs Var
tv Type
_ty
| Var
tv Var -> TyCoVarSet -> Bool
`elemVarSet` TyCoVarSet
tvs = BindFlag
BindMe
| Bool
otherwise = BindFlag
DontBindMe
alwaysBindTv :: BindTvFun
alwaysBindTv :: BindTvFun
alwaysBindTv Var
_tv Type
_ty = BindFlag
BindMe
alwaysBindFam :: BindFamFun
alwaysBindFam :: BindFamFun
alwaysBindFam TyCon
_tc [Type]
_args Type
_rhs = BindFlag
BindMe
dontCareBindFam :: HasCallStack => BindFamFun
dontCareBindFam :: HasCallStack => BindFamFun
dontCareBindFam TyCon
tc [Type]
args Type
rhs
= String -> SDoc -> BindFlag
forall a. HasCallStack => String -> SDoc -> a
pprPanic String
"dontCareBindFam" (SDoc -> BindFlag) -> SDoc -> BindFlag
forall a b. (a -> b) -> a -> b
$
[SDoc] -> SDoc
forall doc. IsDoc doc => [doc] -> doc
vcat [ TyCon -> SDoc
forall a. Outputable a => a -> SDoc
ppr TyCon
tc SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> [Type] -> SDoc
forall a. Outputable a => a -> SDoc
ppr [Type]
args, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"rhs" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
rhs ]
neverBindFam :: BindFamFun
neverBindFam :: BindFamFun
neverBindFam TyCon
_tc [Type]
_args Type
_rhs = BindFlag
DontBindMe
tcMatchTy :: HasDebugCallStack => Type -> Type -> Maybe Subst
tcMatchTy :: HasDebugCallStack => Type -> Type -> Maybe Subst
tcMatchTy Type
ty1 Type
ty2 = [Type] -> [Type] -> Maybe Subst
HasDebugCallStack => [Type] -> [Type] -> Maybe Subst
tcMatchTys [Type
ty1] [Type
ty2]
tcMatchTyX_BM :: HasDebugCallStack
=> BindTvFun -> Subst
-> Type -> Type -> Maybe Subst
tcMatchTyX_BM :: HasDebugCallStack =>
BindTvFun -> Subst -> Type -> Type -> Maybe Subst
tcMatchTyX_BM BindTvFun
bind_tv Subst
subst Type
ty1 Type
ty2
= HasDebugCallStack =>
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
tc_match_tys_x BindTvFun
bind_tv Bool
False Subst
subst [Type
ty1] [Type
ty2]
tcMatchTyKi :: HasDebugCallStack => Type -> Type -> Maybe Subst
tcMatchTyKi :: HasDebugCallStack => Type -> Type -> Maybe Subst
tcMatchTyKi Type
ty1 Type
ty2
= HasDebugCallStack =>
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
tc_match_tys BindTvFun
alwaysBindTv Bool
True [Type
ty1] [Type
ty2]
tcMatchTyX :: HasDebugCallStack
=> Subst
-> Type
-> Type
-> Maybe Subst
tcMatchTyX :: HasDebugCallStack => Subst -> Type -> Type -> Maybe Subst
tcMatchTyX Subst
subst Type
ty1 Type
ty2
= HasDebugCallStack =>
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
tc_match_tys_x BindTvFun
alwaysBindTv Bool
False Subst
subst [Type
ty1] [Type
ty2]
tcMatchTys :: HasDebugCallStack
=> [Type]
-> [Type]
-> Maybe Subst
tcMatchTys :: HasDebugCallStack => [Type] -> [Type] -> Maybe Subst
tcMatchTys [Type]
tys1 [Type]
tys2
= HasDebugCallStack =>
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
tc_match_tys BindTvFun
alwaysBindTv Bool
False [Type]
tys1 [Type]
tys2
tcMatchTyKis :: HasDebugCallStack
=> [Type]
-> [Type]
-> Maybe Subst
tcMatchTyKis :: HasDebugCallStack => [Type] -> [Type] -> Maybe Subst
tcMatchTyKis [Type]
tys1 [Type]
tys2
= HasDebugCallStack =>
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
tc_match_tys BindTvFun
alwaysBindTv Bool
True [Type]
tys1 [Type]
tys2
tcMatchTysX :: HasDebugCallStack
=> Subst
-> [Type]
-> [Type]
-> Maybe Subst
tcMatchTysX :: HasDebugCallStack => Subst -> [Type] -> [Type] -> Maybe Subst
tcMatchTysX Subst
subst [Type]
tys1 [Type]
tys2
= HasDebugCallStack =>
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
tc_match_tys_x BindTvFun
alwaysBindTv Bool
False Subst
subst [Type]
tys1 [Type]
tys2
tcMatchTyKisX :: HasDebugCallStack
=> Subst
-> [Type]
-> [Type]
-> Maybe Subst
tcMatchTyKisX :: HasDebugCallStack => Subst -> [Type] -> [Type] -> Maybe Subst
tcMatchTyKisX Subst
subst [Type]
tys1 [Type]
tys2
= HasDebugCallStack =>
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
tc_match_tys_x BindTvFun
alwaysBindTv Bool
True Subst
subst [Type]
tys1 [Type]
tys2
tc_match_tys :: HasDebugCallStack
=> BindTvFun
-> Bool
-> [Type]
-> [Type]
-> Maybe Subst
tc_match_tys :: HasDebugCallStack =>
BindTvFun -> Bool -> [Type] -> [Type] -> Maybe Subst
tc_match_tys BindTvFun
bind_me Bool
match_kis [Type]
tys1 [Type]
tys2
= HasDebugCallStack =>
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
tc_match_tys_x BindTvFun
bind_me Bool
match_kis (InScopeSet -> Subst
mkEmptySubst InScopeSet
in_scope) [Type]
tys1 [Type]
tys2
where
in_scope :: InScopeSet
in_scope = TyCoVarSet -> InScopeSet
mkInScopeSet ([Type] -> TyCoVarSet
tyCoVarsOfTypes [Type]
tys1 TyCoVarSet -> TyCoVarSet -> TyCoVarSet
`unionVarSet` [Type] -> TyCoVarSet
tyCoVarsOfTypes [Type]
tys2)
tc_match_tys_x :: HasDebugCallStack
=> BindTvFun
-> Bool
-> Subst
-> [Type]
-> [Type]
-> Maybe Subst
tc_match_tys_x :: HasDebugCallStack =>
BindTvFun -> Bool -> Subst -> [Type] -> [Type] -> Maybe Subst
tc_match_tys_x BindTvFun
bind_tv Bool
match_kis (Subst InScopeSet
in_scope IdSubstEnv
id_env TvSubstEnv
tv_env CvSubstEnv
cv_env) [Type]
tys1 [Type]
tys2
= case BindFamFun
-> BindTvFun
-> Bool
-> Bool
-> Bool
-> MultiplicityFlag
-> RnEnv2
-> TvSubstEnv
-> CvSubstEnv
-> [Type]
-> [Type]
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
tc_unify_tys BindFamFun
alwaysBindFam
BindTvFun
bind_tv
Bool
False
Bool
False
Bool
match_kis
MultiplicityFlag
RespectMultiplicities
(InScopeSet -> RnEnv2
mkRnEnv2 InScopeSet
in_scope) TvSubstEnv
tv_env CvSubstEnv
cv_env [Type]
tys1 [Type]
tys2 of
Unifiable (TvSubstEnv
tv_env', CvSubstEnv
cv_env')
-> Subst -> Maybe Subst
forall a. a -> Maybe a
Just (Subst -> Maybe Subst) -> Subst -> Maybe Subst
forall a b. (a -> b) -> a -> b
$ InScopeSet -> IdSubstEnv -> TvSubstEnv -> CvSubstEnv -> Subst
Subst InScopeSet
in_scope IdSubstEnv
id_env TvSubstEnv
tv_env' CvSubstEnv
cv_env'
UnifyResultM (TvSubstEnv, CvSubstEnv)
_ -> Maybe Subst
forall a. Maybe a
Nothing
ruleMatchTyKiX
:: TyCoVarSet
-> RnEnv2
-> TvSubstEnv
-> Type
-> Type
-> Maybe TvSubstEnv
ruleMatchTyKiX :: TyCoVarSet
-> RnEnv2 -> TvSubstEnv -> Type -> Type -> Maybe TvSubstEnv
ruleMatchTyKiX TyCoVarSet
tmpl_tvs RnEnv2
rn_env TvSubstEnv
tenv Type
tmpl Type
target
= case BindFamFun
-> BindTvFun
-> Bool
-> Bool
-> Bool
-> MultiplicityFlag
-> RnEnv2
-> TvSubstEnv
-> CvSubstEnv
-> [Type]
-> [Type]
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
tc_unify_tys BindFamFun
neverBindFam (TyCoVarSet -> BindTvFun
matchBindTv TyCoVarSet
tmpl_tvs)
Bool
False
Bool
False
Bool
True
MultiplicityFlag
IgnoreMultiplicities
RnEnv2
rn_env TvSubstEnv
tenv CvSubstEnv
emptyCvSubstEnv [Type
tmpl] [Type
target] of
Unifiable (TvSubstEnv
tenv', CvSubstEnv
_) -> TvSubstEnv -> Maybe TvSubstEnv
forall a. a -> Maybe a
Just TvSubstEnv
tenv'
UnifyResultM (TvSubstEnv, CvSubstEnv)
_ -> Maybe TvSubstEnv
forall a. Maybe a
Nothing
typesCantMatch :: [(Type,Type)] -> Bool
typesCantMatch :: [(Type, Type)] -> Bool
typesCantMatch [(Type, Type)]
prs = ((Type, Type) -> Bool) -> [(Type, Type)] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any ((Type -> Type -> Bool) -> (Type, Type) -> Bool
forall a b c. (a -> b -> c) -> (a, b) -> c
uncurry Type -> Type -> Bool
typesAreApart) [(Type, Type)]
prs
typesAreApart :: Type -> Type -> Bool
typesAreApart :: Type -> Type -> Bool
typesAreApart Type
t1 Type
t2 = case BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tcUnifyTysFG BindFamFun
alwaysBindFam BindTvFun
alwaysBindTv [Type
t1] [Type
t2] of
UnifyResult
SurelyApart -> Bool
True
UnifyResult
_ -> Bool
False
tcUnifyTy :: Type -> Type
-> Maybe Subst
tcUnifyTy :: Type -> Type -> Maybe Subst
tcUnifyTy Type
t1 Type
t2 = BindTvFun -> [Type] -> [Type] -> Maybe Subst
tcUnifyTys BindTvFun
alwaysBindTv [Type
t1] [Type
t2]
tcUnifyDebugger :: Type -> Type -> Maybe Subst
tcUnifyDebugger :: Type -> Type -> Maybe Subst
tcUnifyDebugger Type
t1 Type
t2
= case Bool -> BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tc_unify_tys_fg
Bool
True
BindFamFun
neverBindFam
BindTvFun
alwaysBindTv
[Type
t1] [Type
t2] of
Unifiable Subst
subst -> Subst -> Maybe Subst
forall a. a -> Maybe a
Just Subst
subst
UnifyResult
_ -> Maybe Subst
forall a. Maybe a
Nothing
tcUnifyFunDeps :: TyCoVarSet
-> [Type] -> [Type]
-> Maybe Subst
tcUnifyFunDeps :: TyCoVarSet -> [Type] -> [Type] -> Maybe Subst
tcUnifyFunDeps TyCoVarSet
qtvs [Type]
tys1 [Type]
tys2
= case Bool -> BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tc_unify_tys_fg
Bool
True
HasCallStack => BindFamFun
BindFamFun
dontCareBindFam
(TyCoVarSet -> BindTvFun
matchBindTv TyCoVarSet
qtvs)
[Type]
tys1 [Type]
tys2 of
Unifiable Subst
subst -> Subst -> Maybe Subst
forall a. a -> Maybe a
Just Subst
subst
UnifyResult
_ -> Maybe Subst
forall a. Maybe a
Nothing
tcUnifyTyForInjectivity
:: AmIUnifying
-> InScopeSet
-> Type -> Type
-> Maybe Subst
tcUnifyTyForInjectivity :: Bool -> InScopeSet -> Type -> Type -> Maybe Subst
tcUnifyTyForInjectivity Bool
unif InScopeSet
in_scope Type
t1 Type
t2
= case BindFamFun
-> BindTvFun
-> Bool
-> Bool
-> Bool
-> MultiplicityFlag
-> RnEnv2
-> TvSubstEnv
-> CvSubstEnv
-> [Type]
-> [Type]
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
tc_unify_tys BindFamFun
alwaysBindFam BindTvFun
alwaysBindTv
Bool
unif
Bool
True
Bool
False
MultiplicityFlag
RespectMultiplicities
RnEnv2
rn_env TvSubstEnv
emptyTvSubstEnv CvSubstEnv
emptyCvSubstEnv
[Type
t1] [Type
t2] of
Unifiable (TvSubstEnv
tv_subst, CvSubstEnv
_cv_subst) -> Subst -> Maybe Subst
forall a. a -> Maybe a
Just (Subst -> Maybe Subst) -> Subst -> Maybe Subst
forall a b. (a -> b) -> a -> b
$ TvSubstEnv -> Subst
maybe_fix TvSubstEnv
tv_subst
MaybeApart MaybeApartReason
_reason (TvSubstEnv
tv_subst, CvSubstEnv
_cv_subst) -> Subst -> Maybe Subst
forall a. a -> Maybe a
Just (Subst -> Maybe Subst) -> Subst -> Maybe Subst
forall a b. (a -> b) -> a -> b
$ TvSubstEnv -> Subst
maybe_fix TvSubstEnv
tv_subst
UnifyResultM (TvSubstEnv, CvSubstEnv)
SurelyApart -> Maybe Subst
forall a. Maybe a
Nothing
where
rn_env :: RnEnv2
rn_env = InScopeSet -> RnEnv2
mkRnEnv2 InScopeSet
in_scope
maybe_fix :: TvSubstEnv -> Subst
maybe_fix | Bool
unif = InScopeSet -> TvSubstEnv -> Subst
niFixSubst InScopeSet
in_scope
| Bool
otherwise = InScopeSet -> TvSubstEnv -> Subst
mkTvSubst InScopeSet
in_scope
tcUnifyTys :: BindTvFun
-> [Type] -> [Type]
-> Maybe Subst
tcUnifyTys :: BindTvFun -> [Type] -> [Type] -> Maybe Subst
tcUnifyTys BindTvFun
bind_fn [Type]
tys1 [Type]
tys2
= case BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tcUnifyTysFG BindFamFun
neverBindFam BindTvFun
bind_fn [Type]
tys1 [Type]
tys2 of
Unifiable Subst
result -> Subst -> Maybe Subst
forall a. a -> Maybe a
Just Subst
result
UnifyResult
_ -> Maybe Subst
forall a. Maybe a
Nothing
tcUnifyTysFG :: BindFamFun -> BindTvFun
-> [Type] -> [Type]
-> UnifyResult
tcUnifyTysFG :: BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tcUnifyTysFG BindFamFun
bind_fam BindTvFun
bind_tv [Type]
tys1 [Type]
tys2
= Bool -> BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tc_unify_tys_fg Bool
False BindFamFun
bind_fam BindTvFun
bind_tv [Type]
tys1 [Type]
tys2
tc_unify_tys_fg :: Bool
-> BindFamFun -> BindTvFun
-> [Type] -> [Type]
-> UnifyResult
tc_unify_tys_fg :: Bool -> BindFamFun -> BindTvFun -> [Type] -> [Type] -> UnifyResult
tc_unify_tys_fg Bool
match_kis BindFamFun
bind_fam BindTvFun
bind_tv [Type]
tys1 [Type]
tys2
= do { (tv_env, _) <- BindFamFun
-> BindTvFun
-> Bool
-> Bool
-> Bool
-> MultiplicityFlag
-> RnEnv2
-> TvSubstEnv
-> CvSubstEnv
-> [Type]
-> [Type]
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
tc_unify_tys BindFamFun
bind_fam BindTvFun
bind_tv
Bool
True
Bool
False
Bool
match_kis
MultiplicityFlag
RespectMultiplicities RnEnv2
rn_env
TvSubstEnv
emptyTvSubstEnv CvSubstEnv
emptyCvSubstEnv
[Type]
tys1 [Type]
tys2
; return $ niFixSubst in_scope tv_env }
where
in_scope :: InScopeSet
in_scope = TyCoVarSet -> InScopeSet
mkInScopeSet (TyCoVarSet -> InScopeSet) -> TyCoVarSet -> InScopeSet
forall a b. (a -> b) -> a -> b
$ [Type] -> TyCoVarSet
tyCoVarsOfTypes [Type]
tys1 TyCoVarSet -> TyCoVarSet -> TyCoVarSet
`unionVarSet` [Type] -> TyCoVarSet
tyCoVarsOfTypes [Type]
tys2
rn_env :: RnEnv2
rn_env = InScopeSet -> RnEnv2
mkRnEnv2 InScopeSet
in_scope
tc_unify_tys :: BindFamFun -> BindTvFun
-> AmIUnifying
-> Bool
-> Bool
-> MultiplicityFlag
-> RnEnv2
-> TvSubstEnv
-> CvSubstEnv
-> [Type] -> [Type]
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
tc_unify_tys :: BindFamFun
-> BindTvFun
-> Bool
-> Bool
-> Bool
-> MultiplicityFlag
-> RnEnv2
-> TvSubstEnv
-> CvSubstEnv
-> [Type]
-> [Type]
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
tc_unify_tys BindFamFun
bind_fam BindTvFun
bind_tv Bool
unif Bool
inj_check Bool
match_kis MultiplicityFlag
match_mults RnEnv2
rn_env TvSubstEnv
tv_env CvSubstEnv
cv_env [Type]
tys1 [Type]
tys2
= TvSubstEnv
-> CvSubstEnv -> UM () -> UnifyResultM (TvSubstEnv, CvSubstEnv)
initUM TvSubstEnv
tv_env CvSubstEnv
cv_env (UM () -> UnifyResultM (TvSubstEnv, CvSubstEnv))
-> UM () -> UnifyResultM (TvSubstEnv, CvSubstEnv)
forall a b. (a -> b) -> a -> b
$
do { Bool -> UM () -> UM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when Bool
match_kis (UM () -> UM ()) -> UM () -> UM ()
forall a b. (a -> b) -> a -> b
$
UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
kis1 [Type]
kis2
; UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
tys1 [Type]
tys2 }
where
env :: UMEnv
env = UMEnv { um_bind_tv_fun :: BindTvFun
um_bind_tv_fun = BindTvFun
bind_tv
, um_bind_fam_fun :: BindFamFun
um_bind_fam_fun = BindFamFun
bind_fam
, um_foralls :: TyCoVarSet
um_foralls = TyCoVarSet
emptyVarSet
, um_unif :: Bool
um_unif = Bool
unif
, um_inj_tf :: Bool
um_inj_tf = Bool
inj_check
, um_arr_mult :: MultiplicityFlag
um_arr_mult = MultiplicityFlag
match_mults
, um_rn_env :: RnEnv2
um_rn_env = RnEnv2
rn_env }
kis1 :: [Type]
kis1 = (Type -> Type) -> [Type] -> [Type]
forall a b. (a -> b) -> [a] -> [b]
map HasDebugCallStack => Type -> Type
Type -> Type
typeKind [Type]
tys1
kis2 :: [Type]
kis2 = (Type -> Type) -> [Type] -> [Type]
forall a b. (a -> b) -> [a] -> [b]
map HasDebugCallStack => Type -> Type
Type -> Type
typeKind [Type]
tys2
type UnifyResult = UnifyResultM Subst
data UnifyResultM a = Unifiable a
| MaybeApart MaybeApartReason
a
| SurelyApart
deriving (forall a b. (a -> b) -> UnifyResultM a -> UnifyResultM b)
-> (forall a b. a -> UnifyResultM b -> UnifyResultM a)
-> Functor UnifyResultM
forall a b. a -> UnifyResultM b -> UnifyResultM a
forall a b. (a -> b) -> UnifyResultM a -> UnifyResultM b
forall (f :: * -> *).
(forall a b. (a -> b) -> f a -> f b)
-> (forall a b. a -> f b -> f a) -> Functor f
$cfmap :: forall a b. (a -> b) -> UnifyResultM a -> UnifyResultM b
fmap :: forall a b. (a -> b) -> UnifyResultM a -> UnifyResultM b
$c<$ :: forall a b. a -> UnifyResultM b -> UnifyResultM a
<$ :: forall a b. a -> UnifyResultM b -> UnifyResultM a
Functor
data MaybeApartReason
= MARTypeFamily
| MARInfinite
| MARTypeVsConstraint
| MARCast
combineMAR :: MaybeApartReason -> MaybeApartReason -> MaybeApartReason
combineMAR :: MaybeApartReason -> MaybeApartReason -> MaybeApartReason
combineMAR MaybeApartReason
MARInfinite MaybeApartReason
_ = MaybeApartReason
MARInfinite
combineMAR MaybeApartReason
MARTypeFamily MaybeApartReason
r = MaybeApartReason
r
combineMAR MaybeApartReason
MARTypeVsConstraint MaybeApartReason
r = MaybeApartReason
r
combineMAR MaybeApartReason
MARCast MaybeApartReason
r = MaybeApartReason
r
instance Outputable MaybeApartReason where
ppr :: MaybeApartReason -> SDoc
ppr MaybeApartReason
MARTypeFamily = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"MARTypeFamily"
ppr MaybeApartReason
MARInfinite = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"MARInfinite"
ppr MaybeApartReason
MARTypeVsConstraint = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"MARTypeVsConstraint"
ppr MaybeApartReason
MARCast = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"MARCast"
instance Semigroup MaybeApartReason where
<> :: MaybeApartReason -> MaybeApartReason -> MaybeApartReason
(<>) = MaybeApartReason -> MaybeApartReason -> MaybeApartReason
combineMAR
instance Applicative UnifyResultM where
pure :: forall a. a -> UnifyResultM a
pure = a -> UnifyResultM a
forall a. a -> UnifyResultM a
Unifiable
<*> :: forall a b.
UnifyResultM (a -> b) -> UnifyResultM a -> UnifyResultM b
(<*>) = UnifyResultM (a -> b) -> UnifyResultM a -> UnifyResultM b
forall (m :: * -> *) a b. Monad m => m (a -> b) -> m a -> m b
ap
instance Monad UnifyResultM where
UnifyResultM a
SurelyApart >>= :: forall a b.
UnifyResultM a -> (a -> UnifyResultM b) -> UnifyResultM b
>>= a -> UnifyResultM b
_ = UnifyResultM b
forall a. UnifyResultM a
SurelyApart
MaybeApart MaybeApartReason
r1 a
x >>= a -> UnifyResultM b
f = case a -> UnifyResultM b
f a
x of
Unifiable b
y -> MaybeApartReason -> b -> UnifyResultM b
forall a. MaybeApartReason -> a -> UnifyResultM a
MaybeApart MaybeApartReason
r1 b
y
MaybeApart MaybeApartReason
r2 b
y -> MaybeApartReason -> b -> UnifyResultM b
forall a. MaybeApartReason -> a -> UnifyResultM a
MaybeApart (MaybeApartReason
r1 MaybeApartReason -> MaybeApartReason -> MaybeApartReason
forall a. Semigroup a => a -> a -> a
S.<> MaybeApartReason
r2) b
y
UnifyResultM b
SurelyApart -> UnifyResultM b
forall a. UnifyResultM a
SurelyApart
Unifiable a
x >>= a -> UnifyResultM b
f = a -> UnifyResultM b
f a
x
instance Outputable a => Outputable (UnifyResultM a) where
ppr :: UnifyResultM a -> SDoc
ppr UnifyResultM a
SurelyApart = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"SurelyApart"
ppr (Unifiable a
x) = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Unifiable" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> a -> SDoc
forall a. Outputable a => a -> SDoc
ppr a
x
ppr (MaybeApart MaybeApartReason
r a
x) = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"MaybeApart" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> MaybeApartReason -> SDoc
forall a. Outputable a => a -> SDoc
ppr MaybeApartReason
r SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> a -> SDoc
forall a. Outputable a => a -> SDoc
ppr a
x
niFixSubst :: InScopeSet -> TvSubstEnv -> Subst
niFixSubst :: InScopeSet -> TvSubstEnv -> Subst
niFixSubst InScopeSet
in_scope TvSubstEnv
tenv
| Bool
not_fixpoint = InScopeSet -> TvSubstEnv -> Subst
niFixSubst InScopeSet
in_scope ((Type -> Type) -> TvSubstEnv -> TvSubstEnv
forall a b. (a -> b) -> VarEnv a -> VarEnv b
mapVarEnv (HasDebugCallStack => Subst -> Type -> Type
Subst -> Type -> Type
substTy Subst
subst) TvSubstEnv
tenv)
| Bool
otherwise = Subst
subst
where
range_fvs :: FV
range_fvs :: FV
range_fvs = [Type] -> FV
tyCoFVsOfTypes (TvSubstEnv -> [Type]
forall {k} (key :: k) elt. UniqFM key elt -> [elt]
nonDetEltsUFM TvSubstEnv
tenv)
range_tvs :: [TyVar]
range_tvs :: [Var]
range_tvs = FV -> [Var]
fvVarList FV
range_fvs
not_fixpoint :: Bool
not_fixpoint = (Var -> Bool) -> [Var] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any Var -> Bool
in_domain [Var]
range_tvs
in_domain :: Var -> Bool
in_domain Var
tv = Var
tv Var -> TvSubstEnv -> Bool
forall a. Var -> VarEnv a -> Bool
`elemVarEnv` TvSubstEnv
tenv
free_tvs :: [Var]
free_tvs = [Var] -> [Var]
scopedSort ((Var -> Bool) -> [Var] -> [Var]
forall a. (a -> Bool) -> [a] -> [a]
filterOut Var -> Bool
in_domain [Var]
range_tvs)
subst :: Subst
subst = (Subst -> Var -> Subst) -> Subst -> [Var] -> Subst
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' Subst -> Var -> Subst
add_free_tv
(InScopeSet -> TvSubstEnv -> Subst
mkTvSubst InScopeSet
in_scope TvSubstEnv
tenv)
[Var]
free_tvs
add_free_tv :: Subst -> TyVar -> Subst
add_free_tv :: Subst -> Var -> Subst
add_free_tv Subst
subst Var
tv
= Subst -> Var -> Type -> Subst
extendTvSubst Subst
subst Var
tv (Var -> Type
mkTyVarTy Var
tv')
where
tv' :: Var
tv' = (Type -> Type) -> Var -> Var
updateTyVarKind (HasDebugCallStack => Subst -> Type -> Type
Subst -> Type -> Type
substTy Subst
subst) Var
tv
type AmIUnifying = Bool
type InType = Type
type OutCoercion = Coercion
unify_ty :: UMEnv
-> InType -> InType
-> OutCoercion
-> UM ()
unify_ty :: UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
_env (TyConApp TyCon
tc1 []) (TyConApp TyCon
tc2 []) OutCoercion
_kco
| TyCon
tc1 TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2
= () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
unify_ty UMEnv
env Type
ty1 Type
ty2 OutCoercion
kco
| Just Type
ty1' <- Type -> Maybe Type
coreView Type
ty1 = UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1' Type
ty2 OutCoercion
kco
| Just Type
ty2' <- Type -> Maybe Type
coreView Type
ty2 = UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1 Type
ty2' OutCoercion
kco
unify_ty UMEnv
env (CastTy Type
ty1 OutCoercion
co1) Type
ty2 OutCoercion
kco
| UMEnv -> TyCoVarSet -> Bool
mentionsForAllBoundTyVarsL UMEnv
env (OutCoercion -> TyCoVarSet
tyCoVarsOfCo OutCoercion
co1)
= MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARCast
| UMEnv -> Bool
um_unif UMEnv
env
= UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1 Type
ty2 (OutCoercion
co1 HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion
kco)
| Bool
otherwise
= do { subst <- UMEnv -> UM Subst
getSubst UMEnv
env
; let co' = HasDebugCallStack => Subst -> OutCoercion -> OutCoercion
Subst -> OutCoercion -> OutCoercion
substCo Subst
subst OutCoercion
co1
; unify_ty env ty1 ty2 (co' `mkTransCo` kco) }
unify_ty UMEnv
env Type
ty1 (CastTy Type
ty2 OutCoercion
co2) OutCoercion
kco
| UMEnv -> TyCoVarSet -> Bool
mentionsForAllBoundTyVarsR UMEnv
env (OutCoercion -> TyCoVarSet
tyCoVarsOfCo OutCoercion
co2)
= MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARCast
| Bool
otherwise
= UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1 Type
ty2 (OutCoercion
kco HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion -> OutCoercion
mkSymCo OutCoercion
co2)
unify_ty UMEnv
env (AppTy Type
ty1a Type
ty1b) Type
ty2 OutCoercion
_kco
| Just (Type
ty2a, Type
ty2b) <- Type -> Maybe (Type, Type)
tcSplitAppTyNoView_maybe Type
ty2
= UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app UMEnv
env Type
ty1a [Type
ty1b] Type
ty2a [Type
ty2b]
unify_ty UMEnv
env Type
ty1 (AppTy Type
ty2a Type
ty2b) OutCoercion
_kco
| Just (Type
ty1a, Type
ty1b) <- Type -> Maybe (Type, Type)
tcSplitAppTyNoView_maybe Type
ty1
= UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app UMEnv
env Type
ty1a [Type
ty1b] Type
ty2a [Type
ty2b]
unify_ty UMEnv
_ (LitTy TyLit
x) (LitTy TyLit
y) OutCoercion
_kco | TyLit
x TyLit -> TyLit -> Bool
forall a. Eq a => a -> a -> Bool
== TyLit
y = () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
unify_ty UMEnv
env (ForAllTy (Bndr Var
tv1 ForAllTyFlag
_) Type
ty1) (ForAllTy (Bndr Var
tv2 ForAllTyFlag
_) Type
ty2) OutCoercion
kco
= do { UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env (Var -> Type
varType Var
tv1) (Var -> Type
varType Var
tv2) (Type -> OutCoercion
mkNomReflCo Type
liftedTypeKind)
; let env' :: UMEnv
env' = UMEnv -> Var -> Var -> UMEnv
umRnBndr2 UMEnv
env Var
tv1 Var
tv2
; UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env' Type
ty1 Type
ty2 OutCoercion
kco }
unify_ty UMEnv
env (CoercionTy OutCoercion
co1) (CoercionTy OutCoercion
co2) OutCoercion
kco
= do { c_subst <- UM CvSubstEnv
getCvSubstEnv
; case co1 of
CoVarCo Var
cv
| Bool -> Bool
not (UMEnv -> Bool
um_unif UMEnv
env)
, Bool -> Bool
not (Var
cv Var -> CvSubstEnv -> Bool
forall a. Var -> VarEnv a -> Bool
`elemVarEnv` CvSubstEnv
c_subst)
, let (OutCoercion
_mult_co, OutCoercion
co_l, OutCoercion
co_r) = HasDebugCallStack =>
OutCoercion -> (OutCoercion, OutCoercion, OutCoercion)
OutCoercion -> (OutCoercion, OutCoercion, OutCoercion)
decomposeFunCo OutCoercion
kco
rhs_co :: OutCoercion
rhs_co = OutCoercion
co_l HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion
co2 HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion -> OutCoercion
mkSymCo OutCoercion
co_r
, BindFlag
BindMe <- UMEnv -> BindTvFun
um_bind_tv_fun UMEnv
env Var
cv (OutCoercion -> Type
CoercionTy OutCoercion
rhs_co)
-> if UMEnv -> TyCoVarSet -> Bool
mentionsForAllBoundTyVarsR UMEnv
env (OutCoercion -> TyCoVarSet
tyCoVarsOfCo OutCoercion
co2)
then UM ()
forall a. UM a
surelyApart
else Var -> OutCoercion -> UM ()
extendCvEnv Var
cv OutCoercion
rhs_co
OutCoercion
_ -> () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return () }
unify_ty UMEnv
env (TyVarTy Var
tv1) Type
ty2 OutCoercion
kco
= UMEnv -> CanEqLHS -> Type -> OutCoercion -> UM ()
uVarOrFam UMEnv
env (Var -> CanEqLHS
TyVarLHS Var
tv1) Type
ty2 OutCoercion
kco
unify_ty UMEnv
env Type
ty1 (TyVarTy Var
tv2) OutCoercion
kco
| UMEnv -> Bool
um_unif UMEnv
env
= UMEnv -> CanEqLHS -> Type -> OutCoercion -> UM ()
uVarOrFam (UMEnv -> UMEnv
umSwapRn UMEnv
env) (Var -> CanEqLHS
TyVarLHS Var
tv2) Type
ty1 (OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco)
unify_ty UMEnv
env Type
ty1 Type
ty2 OutCoercion
kco
| Just (TyCon
tc,[Type]
tys) <- Maybe (TyCon, [Type])
mb_sat_fam_app1
= UMEnv -> CanEqLHS -> Type -> OutCoercion -> UM ()
uVarOrFam UMEnv
env (TyCon -> [Type] -> CanEqLHS
TyFamLHS TyCon
tc [Type]
tys) Type
ty2 OutCoercion
kco
| UMEnv -> Bool
um_unif UMEnv
env
, Just (TyCon
tc,[Type]
tys) <- Maybe (TyCon, [Type])
mb_sat_fam_app2
= UMEnv -> CanEqLHS -> Type -> OutCoercion -> UM ()
uVarOrFam (UMEnv -> UMEnv
umSwapRn UMEnv
env) (TyCon -> [Type] -> CanEqLHS
TyFamLHS TyCon
tc [Type]
tys) Type
ty1 (OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco)
| Just (TyCon
tc1, [Type]
_) <- Maybe (TyCon, [Type])
mb_tc_app1
, TyCon -> Bool
isTypeFamilyTyCon TyCon
tc1
= if | Just (Type
ty1a, Type
ty1b) <- Type -> Maybe (Type, Type)
tcSplitAppTyNoView_maybe Type
ty1
, Just (Type
ty2a, Type
ty2b) <- Type -> Maybe (Type, Type)
tcSplitAppTyNoView_maybe Type
ty2
-> UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app UMEnv
env Type
ty1a [Type
ty1b] Type
ty2a [Type
ty2b]
| Bool
otherwise -> UM ()
forall a. UM a
surelyApart
| Just (TyCon
tc2, [Type]
_) <- Maybe (TyCon, [Type])
mb_tc_app2
, TyCon -> Bool
isTypeFamilyTyCon TyCon
tc2
= if | Just (Type
ty1a, Type
ty1b) <- Type -> Maybe (Type, Type)
tcSplitAppTyNoView_maybe Type
ty1
, Just (Type
ty2a, Type
ty2b) <- Type -> Maybe (Type, Type)
tcSplitAppTyNoView_maybe Type
ty2
-> UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app UMEnv
env Type
ty1a [Type
ty1b] Type
ty2a [Type
ty2b]
| Bool
otherwise -> UM ()
forall a. UM a
surelyApart
| Just (TyCon
tc1, [Type]
tys1) <- Maybe (TyCon, [Type])
mb_tc_app1
, Just (TyCon
tc2, [Type]
tys2) <- Maybe (TyCon, [Type])
mb_tc_app2
, TyCon
tc1 TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2
= do { Bool -> SDoc -> UM ()
forall (m :: * -> *).
(HasCallStack, Applicative m) =>
Bool -> SDoc -> m ()
massertPpr (TyCon -> Role -> Bool
isInjectiveTyCon TyCon
tc1 Role
Nominal) (TyCon -> SDoc
forall a. Outputable a => a -> SDoc
ppr TyCon
tc1)
; UMEnv -> TyCon -> [Type] -> [Type] -> UM ()
unify_tc_app UMEnv
env TyCon
tc1 [Type]
tys1 [Type]
tys2
}
| Just (TyCon
tc1,[Type]
_) <- Maybe (TyCon, [Type])
mb_tc_app1, let u1 :: Unique
u1 = TyCon -> Unique
tyConUnique TyCon
tc1
, Just (TyCon
tc2,[Type]
_) <- Maybe (TyCon, [Type])
mb_tc_app2, let u2 :: Unique
u2 = TyCon -> Unique
tyConUnique TyCon
tc2
, (Unique
u1 Unique -> Unique -> Bool
forall a. Eq a => a -> a -> Bool
== Unique
tYPETyConKey Bool -> Bool -> Bool
&& Unique
u2 Unique -> Unique -> Bool
forall a. Eq a => a -> a -> Bool
== Unique
cONSTRAINTTyConKey) Bool -> Bool -> Bool
||
(Unique
u2 Unique -> Unique -> Bool
forall a. Eq a => a -> a -> Bool
== Unique
tYPETyConKey Bool -> Bool -> Bool
&& Unique
u1 Unique -> Unique -> Bool
forall a. Eq a => a -> a -> Bool
== Unique
cONSTRAINTTyConKey)
= MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARTypeVsConstraint
| FunTy {} <- Type
ty1
, FunTy {} <- Type
ty2
= MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARTypeVsConstraint
where
mb_tc_app1 :: Maybe (TyCon, [Type])
mb_tc_app1 = HasDebugCallStack => Type -> Maybe (TyCon, [Type])
Type -> Maybe (TyCon, [Type])
splitTyConApp_maybe Type
ty1
mb_tc_app2 :: Maybe (TyCon, [Type])
mb_tc_app2 = HasDebugCallStack => Type -> Maybe (TyCon, [Type])
Type -> Maybe (TyCon, [Type])
splitTyConApp_maybe Type
ty2
mb_sat_fam_app1 :: Maybe (TyCon, [Type])
mb_sat_fam_app1 = Type -> Maybe (TyCon, [Type])
isSatFamApp Type
ty1
mb_sat_fam_app2 :: Maybe (TyCon, [Type])
mb_sat_fam_app2 = Type -> Maybe (TyCon, [Type])
isSatFamApp Type
ty2
unify_ty UMEnv
_ Type
_ Type
_ OutCoercion
_ = UM ()
forall a. UM a
surelyApart
unify_tc_app :: UMEnv -> TyCon -> [Type] -> [Type] -> UM ()
unify_tc_app :: UMEnv -> TyCon -> [Type] -> [Type] -> UM ()
unify_tc_app UMEnv
env TyCon
tc [Type]
tys1 [Type]
tys2
| TyCon
tc TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
fUNTyCon
, MultiplicityFlag
IgnoreMultiplicities <- UMEnv -> MultiplicityFlag
um_arr_mult UMEnv
env
, (Type
_mult1 : [Type]
no_mult_tys1) <- [Type]
tys1
, (Type
_mult2 : [Type]
no_mult_tys2) <- [Type]
tys2
=
UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
no_mult_tys1 [Type]
no_mult_tys2
| Bool
otherwise
= UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
tys1 [Type]
tys2
unify_ty_app :: UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app :: UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app UMEnv
env Type
ty1 [Type]
ty1args Type
ty2 [Type]
ty2args
| Just (Type
ty1', Type
ty1a) <- HasDebugCallStack => Type -> Maybe (Type, Type)
Type -> Maybe (Type, Type)
splitAppTyNoView_maybe Type
ty1
, Just (Type
ty2', Type
ty2a) <- HasDebugCallStack => Type -> Maybe (Type, Type)
Type -> Maybe (Type, Type)
splitAppTyNoView_maybe Type
ty2
= UMEnv -> Type -> [Type] -> Type -> [Type] -> UM ()
unify_ty_app UMEnv
env Type
ty1' (Type
ty1a Type -> [Type] -> [Type]
forall a. a -> [a] -> [a]
: [Type]
ty1args) Type
ty2' (Type
ty2a Type -> [Type] -> [Type]
forall a. a -> [a] -> [a]
: [Type]
ty2args)
| Bool
otherwise
= do { let ki1 :: Type
ki1 = HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
ty1
ki2 :: Type
ki2 = HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
ty2
; UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ki1 Type
ki2 (Type -> OutCoercion
mkNomReflCo Type
liftedTypeKind)
; UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1 Type
ty2 (Type -> OutCoercion
mkNomReflCo Type
ki2)
; UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
ty1args [Type]
ty2args }
unify_tys :: UMEnv -> [Type] -> [Type] -> UM ()
unify_tys :: UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
orig_xs [Type]
orig_ys
= [Type] -> [Type] -> UM ()
go [Type]
orig_xs [Type]
orig_ys
where
go :: [Type] -> [Type] -> UM ()
go [] [] = () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
go (Type
x:[Type]
xs) (Type
y:[Type]
ys)
= do { UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
x Type
y (Type -> OutCoercion
mkNomReflCo (Type -> OutCoercion) -> Type -> OutCoercion
forall a b. (a -> b) -> a -> b
$ HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
y)
; [Type] -> [Type] -> UM ()
go [Type]
xs [Type]
ys }
go [Type]
_ [Type]
_ = UM ()
forall a. UM a
surelyApart
isSatFamApp :: Type -> Maybe (TyCon, [Type])
isSatFamApp :: Type -> Maybe (TyCon, [Type])
isSatFamApp (TyConApp TyCon
tc [Type]
tys)
| TyCon -> Bool
isTypeFamilyTyCon TyCon
tc
Bool -> Bool -> Bool
&& Bool -> Bool
not ([Type]
tys [Type] -> Int -> Bool
forall a. [a] -> Int -> Bool
`lengthExceeds` TyCon -> Int
tyConArity TyCon
tc)
= (TyCon, [Type]) -> Maybe (TyCon, [Type])
forall a. a -> Maybe a
Just (TyCon
tc, [Type]
tys)
isSatFamApp Type
_ = Maybe (TyCon, [Type])
forall a. Maybe a
Nothing
uVarOrFam :: UMEnv -> CanEqLHS -> InType -> OutCoercion -> UM ()
uVarOrFam :: UMEnv -> CanEqLHS -> Type -> OutCoercion -> UM ()
uVarOrFam UMEnv
env CanEqLHS
ty1 Type
ty2 OutCoercion
kco
= do { substs <- UM UMState
getSubstEnvs
; go NotSwapped substs ty1 ty2 kco }
where
go :: SwapFlag -> UMState -> CanEqLHS -> Type -> OutCoercion -> UM ()
go SwapFlag
swapped UMState
substs (TyVarLHS Var
tv1) Type
ty2 OutCoercion
kco
= SwapFlag -> UMState -> Var -> Type -> OutCoercion -> UM ()
go_tv SwapFlag
swapped UMState
substs Var
tv1 Type
ty2 OutCoercion
kco
go SwapFlag
swapped UMState
substs (TyFamLHS TyCon
tc [Type]
tys) Type
ty2 OutCoercion
kco
= SwapFlag
-> UMState -> TyCon -> [Type] -> Type -> OutCoercion -> UM ()
go_fam SwapFlag
swapped UMState
substs TyCon
tc [Type]
tys Type
ty2 OutCoercion
kco
go_tv :: SwapFlag -> UMState -> Var -> Type -> OutCoercion -> UM ()
go_tv SwapFlag
swapped UMState
substs Var
tv1 Type
ty2 OutCoercion
kco
| Just Type
ty1' <- TvSubstEnv -> Var -> Maybe Type
forall a. VarEnv a -> Var -> Maybe a
lookupVarEnv (UMState -> TvSubstEnv
um_tv_env UMState
substs) Var
tv1'
=
if | UMEnv -> Bool
um_unif UMEnv
env -> UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1' Type
ty2 OutCoercion
kco
| (Type
ty1' Type -> OutCoercion -> Type
`mkCastTy` OutCoercion
kco) HasDebugCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`tcEqType` Type
ty2 -> () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
| Bool
otherwise -> UM ()
forall a. UM a
surelyApart
| TyVarTy Var
tv2 <- Type
ty2
, let tv2' :: Var
tv2' = UMEnv -> Var -> Var
umRnOccR UMEnv
env Var
tv2
, Var
tv1' Var -> Var -> Bool
forall a. Eq a => a -> a -> Bool
== Var
tv2'
= if | UMEnv -> Bool
um_unif UMEnv
env -> () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
| Bool
tv1_is_bindable -> Var -> Type -> UM ()
extendTvEnv Var
tv1' Type
ty2
| Bool
otherwise -> () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
| Bool
tv1_is_bindable
, Bool -> Bool
not (UMEnv -> TyCoVarSet -> Bool
mentionsForAllBoundTyVarsR UMEnv
env TyCoVarSet
ty2_fvs)
, Bool -> Bool
not Bool
occurs_check
=
Var -> Type -> UM ()
extendTvEnv Var
tv1' Type
rhs
| UMEnv -> Bool
um_unif UMEnv
env
, SwapFlag
NotSwapped <- SwapFlag
swapped
, Just CanEqLHS
lhs2 <- Type -> Maybe CanEqLHS
canEqLHS_maybe Type
ty2
= SwapFlag -> UMState -> CanEqLHS -> Type -> OutCoercion -> UM ()
go SwapFlag
IsSwapped UMState
substs CanEqLHS
lhs2 (Var -> Type
mkTyVarTy Var
tv1) (OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco)
| Bool
occurs_check = MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARInfinite
| Bool
otherwise = UM ()
forall a. UM a
surelyApart
where
tv1' :: Var
tv1' = UMEnv -> Var -> Var
umRnOccL UMEnv
env Var
tv1
ty2_fvs :: TyCoVarSet
ty2_fvs = Type -> TyCoVarSet
tyCoVarsOfType Type
ty2
rhs_fvs :: TyCoVarSet
rhs_fvs = TyCoVarSet
ty2_fvs TyCoVarSet -> TyCoVarSet -> TyCoVarSet
`unionVarSet` OutCoercion -> TyCoVarSet
tyCoVarsOfCo OutCoercion
kco
rhs :: Type
rhs = Type
ty2 Type -> OutCoercion -> Type
`mkCastTy` OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco
tv1_is_bindable :: Bool
tv1_is_bindable | Bool -> Bool
not (Var
tv1' Var -> TyCoVarSet -> Bool
`elemVarSet` UMEnv -> TyCoVarSet
um_foralls UMEnv
env)
, BindFlag
BindMe <- UMEnv -> BindTvFun
um_bind_tv_fun UMEnv
env Var
tv1' Type
rhs
= Bool
True
| Bool
otherwise
= Bool
False
occurs_check :: Bool
occurs_check = UMEnv -> Bool
um_unif UMEnv
env Bool -> Bool -> Bool
&&
TvSubstEnv -> Var -> TyCoVarSet -> Bool
occursCheck (UMState -> TvSubstEnv
um_tv_env UMState
substs) Var
tv1 TyCoVarSet
rhs_fvs
go_fam :: SwapFlag
-> UMState -> TyCon -> [Type] -> Type -> OutCoercion -> UM ()
go_fam SwapFlag
swapped UMState
substs TyCon
tc1 [Type]
tys1 Type
ty2 OutCoercion
kco
| Bool -> Bool
not (TyCoVarSet -> Bool
isEmptyVarSet (UMEnv -> TyCoVarSet
um_foralls UMEnv
env))
= MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARTypeFamily
| Just Type
ty1' <- FamSubstEnv -> TyCon -> [Type] -> Maybe Type
lookupFamEnv (UMState -> FamSubstEnv
um_fam_env UMState
substs) TyCon
tc1 [Type]
tys1
= if | UMEnv -> Bool
um_unif UMEnv
env -> UMEnv -> Type -> Type -> OutCoercion -> UM ()
unify_ty UMEnv
env Type
ty1' Type
ty2 OutCoercion
kco
| (Type
ty1' Type -> OutCoercion -> Type
`mkCastTy` OutCoercion
kco) HasDebugCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`tcEqType` Type
ty2 -> () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
| Bool
otherwise -> MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARTypeFamily
| Just (TyCon
tc2, [Type]
tys2) <- Type -> Maybe (TyCon, [Type])
isSatFamApp Type
ty2
, TyCon
tc1 TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2
= TyCon -> [Type] -> [Type] -> OutCoercion -> UM ()
go_fam_fam TyCon
tc1 [Type]
tys1 [Type]
tys2 OutCoercion
kco
| BindFlag
BindMe <- UMEnv -> BindFamFun
um_bind_fam_fun UMEnv
env TyCon
tc1 [Type]
tys1 Type
rhs
=
do { TyCon -> [Type] -> Type -> UM ()
extendFamEnv TyCon
tc1 [Type]
tys1 Type
rhs
; MaybeApartReason -> UM ()
maybeApart MaybeApartReason
MARTypeFamily }
| UMEnv -> Bool
um_unif UMEnv
env
, SwapFlag
NotSwapped <- SwapFlag
swapped
, Just CanEqLHS
lhs2 <- Type -> Maybe CanEqLHS
canEqLHS_maybe Type
ty2
= SwapFlag -> UMState -> CanEqLHS -> Type -> OutCoercion -> UM ()
go SwapFlag
IsSwapped UMState
substs CanEqLHS
lhs2 (TyCon -> [Type] -> Type
mkTyConApp TyCon
tc1 [Type]
tys1) (OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco)
| Bool
otherwise
= UM ()
forall a. UM a
surelyApart
where
rhs :: Type
rhs = Type
ty2 Type -> OutCoercion -> Type
`mkCastTy` OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco
go_fam_fam :: TyCon -> [Type] -> [Type] -> OutCoercion -> UM ()
go_fam_fam TyCon
tc [Type]
tys1 [Type]
tys2 OutCoercion
kco
| [Type] -> [Type] -> Bool
tcEqTyConAppArgs [Type]
tys1 [Type]
tys2
= () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
| Bool
otherwise
= do { UM ()
bind_fam_if_poss
; UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
inj_tys1 [Type]
inj_tys2
; Bool -> UM () -> UM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (UMEnv -> Bool
um_inj_tf UMEnv
env) (UM () -> UM ()) -> UM () -> UM ()
forall a b. (a -> b) -> a -> b
$
MaybeApartReason -> UM () -> UM ()
don'tBeSoSure MaybeApartReason
MARTypeFamily (UM () -> UM ()) -> UM () -> UM ()
forall a b. (a -> b) -> a -> b
$
UMEnv -> [Type] -> [Type] -> UM ()
unify_tys UMEnv
env [Type]
noninj_tys1 [Type]
noninj_tys2 }
where
inj :: [Bool]
inj = case TyCon -> Injectivity
tyConInjectivityInfo TyCon
tc of
Injectivity
NotInjective -> Bool -> [Bool]
forall a. a -> [a]
repeat Bool
False
Injective [Bool]
bs -> [Bool]
bs
([Type]
inj_tys1, [Type]
noninj_tys1) = [Bool] -> [Type] -> ([Type], [Type])
forall a. [Bool] -> [a] -> ([a], [a])
partitionByList [Bool]
inj [Type]
tys1
([Type]
inj_tys2, [Type]
noninj_tys2) = [Bool] -> [Type] -> ([Type], [Type])
forall a. [Bool] -> [a] -> ([a], [a])
partitionByList [Bool]
inj [Type]
tys2
bind_fam_if_poss :: UM ()
bind_fam_if_poss | BindFlag
BindMe <- UMEnv -> BindFamFun
um_bind_fam_fun UMEnv
env TyCon
tc [Type]
tys1 Type
rhs1
= TyCon -> [Type] -> Type -> UM ()
extendFamEnv TyCon
tc [Type]
tys1 Type
rhs1
| UMEnv -> Bool
um_unif UMEnv
env
, BindFlag
BindMe <- UMEnv -> BindFamFun
um_bind_fam_fun UMEnv
env TyCon
tc [Type]
tys2 Type
rhs2
= TyCon -> [Type] -> Type -> UM ()
extendFamEnv TyCon
tc [Type]
tys2 Type
rhs2
| Bool
otherwise
= () -> UM ()
forall a. a -> UM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
rhs1 :: Type
rhs1 = TyCon -> [Type] -> Type
mkTyConApp TyCon
tc [Type]
tys2 Type -> OutCoercion -> Type
`mkCastTy` OutCoercion -> OutCoercion
mkSymCo OutCoercion
kco
rhs2 :: Type
rhs2 = TyCon -> [Type] -> Type
mkTyConApp TyCon
tc [Type]
tys1 Type -> OutCoercion -> Type
`mkCastTy` OutCoercion
kco
occursCheck :: TvSubstEnv -> TyVar -> TyCoVarSet -> Bool
occursCheck :: TvSubstEnv -> Var -> TyCoVarSet -> Bool
occursCheck TvSubstEnv
env Var
tv1 TyCoVarSet
tvs
= (Var -> Bool) -> TyCoVarSet -> Bool
anyVarSet Var -> Bool
bad TyCoVarSet
tvs
where
bad :: Var -> Bool
bad Var
tv | Just Type
ty <- TvSubstEnv -> Var -> Maybe Type
forall a. VarEnv a -> Var -> Maybe a
lookupVarEnv TvSubstEnv
env Var
tv
= (Var -> Bool) -> TyCoVarSet -> Bool
anyVarSet Var -> Bool
bad (Type -> TyCoVarSet
tyCoVarsOfType Type
ty)
| Bool
otherwise
= Var
tv Var -> Var -> Bool
forall a. Eq a => a -> a -> Bool
== Var
tv1
data UMEnv
= UMEnv { UMEnv -> Bool
um_unif :: AmIUnifying
, UMEnv -> Bool
um_inj_tf :: Bool
, UMEnv -> MultiplicityFlag
um_arr_mult :: MultiplicityFlag
, UMEnv -> RnEnv2
um_rn_env :: RnEnv2
, UMEnv -> TyCoVarSet
um_foralls :: TyVarSet
, UMEnv -> BindTvFun
um_bind_tv_fun :: BindTvFun
, UMEnv -> BindFamFun
um_bind_fam_fun :: BindFamFun
}
type FamSubstEnv = TyConEnv (ListMap TypeMap Type)
lookupFamEnv :: FamSubstEnv -> TyCon -> [Type] -> Maybe Type
lookupFamEnv :: FamSubstEnv -> TyCon -> [Type] -> Maybe Type
lookupFamEnv FamSubstEnv
env TyCon
tc [Type]
tys
= do { tys_map <- FamSubstEnv -> TyCon -> Maybe (ListMap TypeMap Type)
forall a. TyConEnv a -> TyCon -> Maybe a
lookupTyConEnv FamSubstEnv
env TyCon
tc
; lookupTM tys tys_map }
data UMState = UMState
{ UMState -> TvSubstEnv
um_tv_env :: TvSubstEnv
, UMState -> CvSubstEnv
um_cv_env :: CvSubstEnv
, UMState -> FamSubstEnv
um_fam_env :: FamSubstEnv }
newtype UM a
= UM' { forall a. UM a -> UMState -> UnifyResultM (UMState, a)
unUM :: UMState -> UnifyResultM (UMState, a) }
pattern UM :: (UMState -> UnifyResultM (UMState, a)) -> UM a
pattern $mUM :: forall {r} {a}.
UM a
-> ((UMState -> UnifyResultM (UMState, a)) -> r)
-> ((# #) -> r)
-> r
$bUM :: forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM m <- UM' m
where
UM UMState -> UnifyResultM (UMState, a)
m = (UMState -> UnifyResultM (UMState, a)) -> UM a
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM' ((UMState -> UnifyResultM (UMState, a))
-> UMState -> UnifyResultM (UMState, a)
forall a b. (a -> b) -> a -> b
oneShot UMState -> UnifyResultM (UMState, a)
m)
{-# COMPLETE UM #-}
instance Functor UM where
fmap :: forall a b. (a -> b) -> UM a -> UM b
fmap a -> b
f (UM UMState -> UnifyResultM (UMState, a)
m) = (UMState -> UnifyResultM (UMState, b)) -> UM b
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM (\UMState
s -> ((UMState, a) -> (UMState, b))
-> UnifyResultM (UMState, a) -> UnifyResultM (UMState, b)
forall a b. (a -> b) -> UnifyResultM a -> UnifyResultM b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (\(UMState
s', a
v) -> (UMState
s', a -> b
f a
v)) (UMState -> UnifyResultM (UMState, a)
m UMState
s))
instance Applicative UM where
pure :: forall a. a -> UM a
pure a
a = (UMState -> UnifyResultM (UMState, a)) -> UM a
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM (\UMState
s -> (UMState, a) -> UnifyResultM (UMState, a)
forall a. a -> UnifyResultM a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (UMState
s, a
a))
<*> :: forall a b. UM (a -> b) -> UM a -> UM b
(<*>) = UM (a -> b) -> UM a -> UM b
forall (m :: * -> *) a b. Monad m => m (a -> b) -> m a -> m b
ap
instance Monad UM where
{-# INLINE (>>=) #-}
UM a
m >>= :: forall a b. UM a -> (a -> UM b) -> UM b
>>= a -> UM b
k = (UMState -> UnifyResultM (UMState, b)) -> UM b
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM (\UMState
state ->
do { (state', v) <- UM a -> UMState -> UnifyResultM (UMState, a)
forall a. UM a -> UMState -> UnifyResultM (UMState, a)
unUM UM a
m UMState
state
; unUM (k v) state' })
instance MonadFail UM where
fail :: forall a. String -> UM a
fail String
_ = (UMState -> UnifyResultM (UMState, a)) -> UM a
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM (\UMState
_ -> UnifyResultM (UMState, a)
forall a. UnifyResultM a
SurelyApart)
initUM :: TvSubstEnv
-> CvSubstEnv
-> UM ()
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
initUM :: TvSubstEnv
-> CvSubstEnv -> UM () -> UnifyResultM (TvSubstEnv, CvSubstEnv)
initUM TvSubstEnv
subst_env CvSubstEnv
cv_subst_env UM ()
um
= case UM () -> UMState -> UnifyResultM (UMState, ())
forall a. UM a -> UMState -> UnifyResultM (UMState, a)
unUM UM ()
um UMState
state of
Unifiable (UMState
state, ()
_) -> (TvSubstEnv, CvSubstEnv) -> UnifyResultM (TvSubstEnv, CvSubstEnv)
forall a. a -> UnifyResultM a
Unifiable (UMState -> (TvSubstEnv, CvSubstEnv)
get UMState
state)
MaybeApart MaybeApartReason
r (UMState
state, ()
_) -> MaybeApartReason
-> (TvSubstEnv, CvSubstEnv)
-> UnifyResultM (TvSubstEnv, CvSubstEnv)
forall a. MaybeApartReason -> a -> UnifyResultM a
MaybeApart MaybeApartReason
r (UMState -> (TvSubstEnv, CvSubstEnv)
get UMState
state)
UnifyResultM (UMState, ())
SurelyApart -> UnifyResultM (TvSubstEnv, CvSubstEnv)
forall a. UnifyResultM a
SurelyApart
where
state :: UMState
state = UMState { um_tv_env :: TvSubstEnv
um_tv_env = TvSubstEnv
subst_env
, um_cv_env :: CvSubstEnv
um_cv_env = CvSubstEnv
cv_subst_env
, um_fam_env :: FamSubstEnv
um_fam_env = FamSubstEnv
forall a. TyConEnv a
emptyTyConEnv }
get :: UMState -> (TvSubstEnv, CvSubstEnv)
get (UMState { um_tv_env :: UMState -> TvSubstEnv
um_tv_env = TvSubstEnv
tv_env, um_cv_env :: UMState -> CvSubstEnv
um_cv_env = CvSubstEnv
cv_env }) = (TvSubstEnv
tv_env, CvSubstEnv
cv_env)
getTvSubstEnv :: UM TvSubstEnv
getTvSubstEnv :: UM TvSubstEnv
getTvSubstEnv = (UMState -> UnifyResultM (UMState, TvSubstEnv)) -> UM TvSubstEnv
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, TvSubstEnv)) -> UM TvSubstEnv)
-> (UMState -> UnifyResultM (UMState, TvSubstEnv)) -> UM TvSubstEnv
forall a b. (a -> b) -> a -> b
$ \UMState
state -> (UMState, TvSubstEnv) -> UnifyResultM (UMState, TvSubstEnv)
forall a. a -> UnifyResultM a
Unifiable (UMState
state, UMState -> TvSubstEnv
um_tv_env UMState
state)
getCvSubstEnv :: UM CvSubstEnv
getCvSubstEnv :: UM CvSubstEnv
getCvSubstEnv = (UMState -> UnifyResultM (UMState, CvSubstEnv)) -> UM CvSubstEnv
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, CvSubstEnv)) -> UM CvSubstEnv)
-> (UMState -> UnifyResultM (UMState, CvSubstEnv)) -> UM CvSubstEnv
forall a b. (a -> b) -> a -> b
$ \UMState
state -> (UMState, CvSubstEnv) -> UnifyResultM (UMState, CvSubstEnv)
forall a. a -> UnifyResultM a
Unifiable (UMState
state, UMState -> CvSubstEnv
um_cv_env UMState
state)
getSubstEnvs :: UM UMState
getSubstEnvs :: UM UMState
getSubstEnvs = (UMState -> UnifyResultM (UMState, UMState)) -> UM UMState
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, UMState)) -> UM UMState)
-> (UMState -> UnifyResultM (UMState, UMState)) -> UM UMState
forall a b. (a -> b) -> a -> b
$ \UMState
state -> (UMState, UMState) -> UnifyResultM (UMState, UMState)
forall a. a -> UnifyResultM a
Unifiable (UMState
state, UMState
state)
getSubst :: UMEnv -> UM Subst
getSubst :: UMEnv -> UM Subst
getSubst UMEnv
env = do { tv_env <- UM TvSubstEnv
getTvSubstEnv
; cv_env <- getCvSubstEnv
; let in_scope = RnEnv2 -> InScopeSet
rnInScopeSet (UMEnv -> RnEnv2
um_rn_env UMEnv
env)
; return (mkTCvSubst in_scope tv_env cv_env) }
extendTvEnv :: TyVar -> Type -> UM ()
extendTvEnv :: Var -> Type -> UM ()
extendTvEnv Var
tv Type
ty = (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, ())) -> UM ())
-> (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a b. (a -> b) -> a -> b
$ \UMState
state ->
(UMState, ()) -> UnifyResultM (UMState, ())
forall a. a -> UnifyResultM a
Unifiable (UMState
state { um_tv_env = extendVarEnv (um_tv_env state) tv ty }, ())
extendCvEnv :: CoVar -> Coercion -> UM ()
extendCvEnv :: Var -> OutCoercion -> UM ()
extendCvEnv Var
cv OutCoercion
co = (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, ())) -> UM ())
-> (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a b. (a -> b) -> a -> b
$ \UMState
state ->
(UMState, ()) -> UnifyResultM (UMState, ())
forall a. a -> UnifyResultM a
Unifiable (UMState
state { um_cv_env = extendVarEnv (um_cv_env state) cv co }, ())
extendFamEnv :: TyCon -> [Type] -> Type -> UM ()
extendFamEnv :: TyCon -> [Type] -> Type -> UM ()
extendFamEnv TyCon
tc [Type]
tys Type
ty = (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, ())) -> UM ())
-> (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a b. (a -> b) -> a -> b
$ \UMState
state ->
(UMState, ()) -> UnifyResultM (UMState, ())
forall a. a -> UnifyResultM a
Unifiable (UMState
state { um_fam_env = extend (um_fam_env state) tc }, ())
where
extend :: FamSubstEnv -> TyCon -> FamSubstEnv
extend :: FamSubstEnv -> TyCon -> FamSubstEnv
extend = (Maybe (ListMap TypeMap Type) -> Maybe (ListMap TypeMap Type))
-> FamSubstEnv -> TyCon -> FamSubstEnv
forall a. (Maybe a -> Maybe a) -> TyConEnv a -> TyCon -> TyConEnv a
alterTyConEnv Maybe (ListMap TypeMap Type) -> Maybe (ListMap TypeMap Type)
alter_tm
alter_tm :: Maybe (ListMap TypeMap Type) -> Maybe (ListMap TypeMap Type)
alter_tm :: Maybe (ListMap TypeMap Type) -> Maybe (ListMap TypeMap Type)
alter_tm Maybe (ListMap TypeMap Type)
m_elt = ListMap TypeMap Type -> Maybe (ListMap TypeMap Type)
forall a. a -> Maybe a
Just (Key (ListMap TypeMap)
-> XT Type -> ListMap TypeMap Type -> ListMap TypeMap Type
forall b.
Key (ListMap TypeMap)
-> XT b -> ListMap TypeMap b -> ListMap TypeMap b
forall (m :: * -> *) b. TrieMap m => Key m -> XT b -> m b -> m b
alterTM [Type]
Key (ListMap TypeMap)
tys (\Maybe Type
_ -> Type -> Maybe Type
forall a. a -> Maybe a
Just Type
ty) (Maybe (ListMap TypeMap Type)
m_elt Maybe (ListMap TypeMap Type)
-> ListMap TypeMap Type -> ListMap TypeMap Type
forall a. Maybe a -> a -> a
`orElse` ListMap TypeMap Type
forall a. ListMap TypeMap a
forall (m :: * -> *) a. TrieMap m => m a
emptyTM))
umRnBndr2 :: UMEnv -> TyCoVar -> TyCoVar -> UMEnv
umRnBndr2 :: UMEnv -> Var -> Var -> UMEnv
umRnBndr2 UMEnv
env Var
v1 Var
v2
= UMEnv
env { um_rn_env = rn_env', um_foralls = um_foralls env `extendVarSet` v' }
where
(RnEnv2
rn_env', Var
v') = RnEnv2 -> Var -> Var -> (RnEnv2, Var)
rnBndr2_var (UMEnv -> RnEnv2
um_rn_env UMEnv
env) Var
v1 Var
v2
mentionsForAllBoundTyVarsL, mentionsForAllBoundTyVarsR :: UMEnv -> VarSet -> Bool
mentionsForAllBoundTyVarsL :: UMEnv -> TyCoVarSet -> Bool
mentionsForAllBoundTyVarsL = (RnEnv2 -> Var -> Bool) -> UMEnv -> TyCoVarSet -> Bool
mentions_forall_bound_tvs RnEnv2 -> Var -> Bool
inRnEnvL
mentionsForAllBoundTyVarsR :: UMEnv -> TyCoVarSet -> Bool
mentionsForAllBoundTyVarsR = (RnEnv2 -> Var -> Bool) -> UMEnv -> TyCoVarSet -> Bool
mentions_forall_bound_tvs RnEnv2 -> Var -> Bool
inRnEnvR
mentions_forall_bound_tvs :: (RnEnv2 -> TyVar -> Bool) -> UMEnv -> VarSet -> Bool
mentions_forall_bound_tvs :: (RnEnv2 -> Var -> Bool) -> UMEnv -> TyCoVarSet -> Bool
mentions_forall_bound_tvs RnEnv2 -> Var -> Bool
in_rn_env UMEnv
env TyCoVarSet
varset
| TyCoVarSet -> Bool
isEmptyVarSet (UMEnv -> TyCoVarSet
um_foralls UMEnv
env) = Bool
False
| (Var -> Bool) -> TyCoVarSet -> Bool
anyVarSet (RnEnv2 -> Var -> Bool
in_rn_env (UMEnv -> RnEnv2
um_rn_env UMEnv
env)) TyCoVarSet
varset = Bool
True
| Bool
otherwise = Bool
False
don'tBeSoSure :: MaybeApartReason -> UM () -> UM ()
don'tBeSoSure :: MaybeApartReason -> UM () -> UM ()
don'tBeSoSure MaybeApartReason
r UM ()
um = (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM ((UMState -> UnifyResultM (UMState, ())) -> UM ())
-> (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a b. (a -> b) -> a -> b
$ \ UMState
state ->
case UM () -> UMState -> UnifyResultM (UMState, ())
forall a. UM a -> UMState -> UnifyResultM (UMState, a)
unUM UM ()
um UMState
state of
UnifyResultM (UMState, ())
SurelyApart -> MaybeApartReason -> (UMState, ()) -> UnifyResultM (UMState, ())
forall a. MaybeApartReason -> a -> UnifyResultM a
MaybeApart MaybeApartReason
r (UMState
state, ())
UnifyResultM (UMState, ())
other -> UnifyResultM (UMState, ())
other
umRnOccL :: UMEnv -> TyVar -> TyVar
umRnOccL :: UMEnv -> Var -> Var
umRnOccL UMEnv
env Var
v = RnEnv2 -> Var -> Var
rnOccL (UMEnv -> RnEnv2
um_rn_env UMEnv
env) Var
v
umRnOccR :: UMEnv -> TyVar -> TyVar
umRnOccR :: UMEnv -> Var -> Var
umRnOccR UMEnv
env Var
v = RnEnv2 -> Var -> Var
rnOccR (UMEnv -> RnEnv2
um_rn_env UMEnv
env) Var
v
umSwapRn :: UMEnv -> UMEnv
umSwapRn :: UMEnv -> UMEnv
umSwapRn UMEnv
env = UMEnv
env { um_rn_env = rnSwap (um_rn_env env) }
maybeApart :: MaybeApartReason -> UM ()
maybeApart :: MaybeApartReason -> UM ()
maybeApart MaybeApartReason
r = (UMState -> UnifyResultM (UMState, ())) -> UM ()
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM (\UMState
state -> MaybeApartReason -> (UMState, ()) -> UnifyResultM (UMState, ())
forall a. MaybeApartReason -> a -> UnifyResultM a
MaybeApart MaybeApartReason
r (UMState
state, ()))
surelyApart :: UM a
surelyApart :: forall a. UM a
surelyApart = (UMState -> UnifyResultM (UMState, a)) -> UM a
forall a. (UMState -> UnifyResultM (UMState, a)) -> UM a
UM (\UMState
_ -> UnifyResultM (UMState, a)
forall a. UnifyResultM a
SurelyApart)
data MatchEnv = ME { MatchEnv -> TyCoVarSet
me_tmpls :: TyVarSet
, MatchEnv -> RnEnv2
me_env :: RnEnv2 }
liftCoMatch :: TyCoVarSet -> Type -> Coercion -> Maybe LiftingContext
liftCoMatch :: TyCoVarSet -> Type -> OutCoercion -> Maybe LiftingContext
liftCoMatch TyCoVarSet
tmpls Type
ty OutCoercion
co
= do { cenv1 <- MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
forall a. VarEnv a
emptyVarEnv Type
ki OutCoercion
ki_co OutCoercion
ki_ki_co OutCoercion
ki_ki_co
; cenv2 <- ty_co_match menv cenv1 ty co
(mkNomReflCo co_lkind) (mkNomReflCo co_rkind)
; return (LC (mkEmptySubst in_scope) cenv2) }
where
menv :: MatchEnv
menv = ME { me_tmpls :: TyCoVarSet
me_tmpls = TyCoVarSet
tmpls, me_env :: RnEnv2
me_env = InScopeSet -> RnEnv2
mkRnEnv2 InScopeSet
in_scope }
in_scope :: InScopeSet
in_scope = TyCoVarSet -> InScopeSet
mkInScopeSet (TyCoVarSet
tmpls TyCoVarSet -> TyCoVarSet -> TyCoVarSet
`unionVarSet` OutCoercion -> TyCoVarSet
tyCoVarsOfCo OutCoercion
co)
ki :: Type
ki = HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
ty
ki_co :: OutCoercion
ki_co = HasDebugCallStack => OutCoercion -> OutCoercion
OutCoercion -> OutCoercion
promoteCoercion OutCoercion
co
ki_ki_co :: OutCoercion
ki_ki_co = Type -> OutCoercion
mkNomReflCo Type
liftedTypeKind
Pair Type
co_lkind Type
co_rkind = HasDebugCallStack => OutCoercion -> Pair Type
OutCoercion -> Pair Type
coercionKind OutCoercion
ki_co
ty_co_match :: MatchEnv
-> LiftCoEnv
-> Type
-> Coercion
-> Coercion
-> Coercion
-> Maybe LiftCoEnv
ty_co_match :: MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
co OutCoercion
lkco OutCoercion
rkco
| Just Type
ty' <- Type -> Maybe Type
coreView Type
ty = MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty' OutCoercion
co OutCoercion
lkco OutCoercion
rkco
| Type -> TyCoVarSet
tyCoVarsOfType Type
ty TyCoVarSet -> CvSubstEnv -> Bool
forall a. TyCoVarSet -> VarEnv a -> Bool
`isNotInDomainOf` CvSubstEnv
subst
, Just (Type
ty', Role
_) <- OutCoercion -> Maybe (Type, Role)
isReflCo_maybe OutCoercion
co
, Type
ty HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` Type
ty'
= CvSubstEnv -> Maybe CvSubstEnv
forall a. a -> Maybe a
Just CvSubstEnv
subst
where
isNotInDomainOf :: VarSet -> VarEnv a -> Bool
isNotInDomainOf :: forall a. TyCoVarSet -> VarEnv a -> Bool
isNotInDomainOf TyCoVarSet
set VarEnv a
env
= (Var -> Bool) -> TyCoVarSet -> Bool
noneSet (\Var
v -> Var -> VarEnv a -> Bool
forall a. Var -> VarEnv a -> Bool
elemVarEnv Var
v VarEnv a
env) TyCoVarSet
set
noneSet :: (Var -> Bool) -> VarSet -> Bool
noneSet :: (Var -> Bool) -> TyCoVarSet -> Bool
noneSet Var -> Bool
f = (Var -> Bool) -> TyCoVarSet -> Bool
allVarSet (Bool -> Bool
not (Bool -> Bool) -> (Var -> Bool) -> Var -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Var -> Bool
f)
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
co OutCoercion
lkco OutCoercion
rkco
| CastTy Type
ty' OutCoercion
co' <- Type
ty
= let empty_subst :: Subst
empty_subst = InScopeSet -> Subst
mkEmptySubst (RnEnv2 -> InScopeSet
rnInScopeSet (MatchEnv -> RnEnv2
me_env MatchEnv
menv))
substed_co_l :: OutCoercion
substed_co_l = HasDebugCallStack => Subst -> OutCoercion -> OutCoercion
Subst -> OutCoercion -> OutCoercion
substCo (Subst -> CvSubstEnv -> Subst
liftEnvSubstLeft Subst
empty_subst CvSubstEnv
subst) OutCoercion
co'
substed_co_r :: OutCoercion
substed_co_r = HasDebugCallStack => Subst -> OutCoercion -> OutCoercion
Subst -> OutCoercion -> OutCoercion
substCo (Subst -> CvSubstEnv -> Subst
liftEnvSubstRight Subst
empty_subst CvSubstEnv
subst) OutCoercion
co'
in
MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty' OutCoercion
co (OutCoercion
substed_co_l HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion
lkco)
(OutCoercion
substed_co_r HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion
rkco)
| SymCo OutCoercion
co' <- OutCoercion
co
= CvSubstEnv -> CvSubstEnv
swapLiftCoEnv (CvSubstEnv -> CvSubstEnv) -> Maybe CvSubstEnv -> Maybe CvSubstEnv
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv (CvSubstEnv -> CvSubstEnv
swapLiftCoEnv CvSubstEnv
subst) Type
ty OutCoercion
co' OutCoercion
rkco OutCoercion
lkco
ty_co_match MatchEnv
menv CvSubstEnv
subst (TyVarTy Var
tv1) OutCoercion
co OutCoercion
lkco OutCoercion
rkco
| Just OutCoercion
co1' <- CvSubstEnv -> Var -> Maybe OutCoercion
forall a. VarEnv a -> Var -> Maybe a
lookupVarEnv CvSubstEnv
subst Var
tv1'
= if RnEnv2 -> OutCoercion -> OutCoercion -> Bool
eqCoercionX (RnEnv2 -> RnEnv2
nukeRnEnvL RnEnv2
rn_env) OutCoercion
co1' OutCoercion
co
then CvSubstEnv -> Maybe CvSubstEnv
forall a. a -> Maybe a
Just CvSubstEnv
subst
else Maybe CvSubstEnv
forall a. Maybe a
Nothing
| Var
tv1' Var -> TyCoVarSet -> Bool
`elemVarSet` MatchEnv -> TyCoVarSet
me_tmpls MatchEnv
menv
= if (Var -> Bool) -> [Var] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any (RnEnv2 -> Var -> Bool
inRnEnvR RnEnv2
rn_env) (OutCoercion -> [Var]
tyCoVarsOfCoList OutCoercion
co)
then Maybe CvSubstEnv
forall a. Maybe a
Nothing
else CvSubstEnv -> Maybe CvSubstEnv
forall a. a -> Maybe a
Just (CvSubstEnv -> Maybe CvSubstEnv) -> CvSubstEnv -> Maybe CvSubstEnv
forall a b. (a -> b) -> a -> b
$ CvSubstEnv -> Var -> OutCoercion -> CvSubstEnv
forall a. VarEnv a -> Var -> a -> VarEnv a
extendVarEnv CvSubstEnv
subst Var
tv1' (OutCoercion -> CvSubstEnv) -> OutCoercion -> CvSubstEnv
forall a b. (a -> b) -> a -> b
$
OutCoercion -> OutCoercion -> OutCoercion -> OutCoercion
castCoercionKind OutCoercion
co (OutCoercion -> OutCoercion
mkSymCo OutCoercion
lkco) (OutCoercion -> OutCoercion
mkSymCo OutCoercion
rkco)
| Bool
otherwise
= Maybe CvSubstEnv
forall a. Maybe a
Nothing
where
rn_env :: RnEnv2
rn_env = MatchEnv -> RnEnv2
me_env MatchEnv
menv
tv1' :: Var
tv1' = RnEnv2 -> Var -> Var
rnOccL RnEnv2
rn_env Var
tv1
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty (SubCo OutCoercion
co) OutCoercion
lkco OutCoercion
rkco
= MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
co OutCoercion
lkco OutCoercion
rkco
ty_co_match MatchEnv
menv CvSubstEnv
subst (AppTy Type
ty1a Type
ty1b) OutCoercion
co OutCoercion
_lkco OutCoercion
_rkco
| Just (OutCoercion
co2, OutCoercion
arg2) <- OutCoercion -> Maybe (OutCoercion, OutCoercion)
splitAppCo_maybe OutCoercion
co
= MatchEnv
-> CvSubstEnv
-> Type
-> [Type]
-> OutCoercion
-> [OutCoercion]
-> Maybe CvSubstEnv
ty_co_match_app MatchEnv
menv CvSubstEnv
subst Type
ty1a [Type
ty1b] OutCoercion
co2 [OutCoercion
arg2]
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty1 (AppCo OutCoercion
co2 OutCoercion
arg2) OutCoercion
_lkco OutCoercion
_rkco
| Just (Type
ty1a, Type
ty1b) <- HasDebugCallStack => Type -> Maybe (Type, Type)
Type -> Maybe (Type, Type)
splitAppTyNoView_maybe Type
ty1
= MatchEnv
-> CvSubstEnv
-> Type
-> [Type]
-> OutCoercion
-> [OutCoercion]
-> Maybe CvSubstEnv
ty_co_match_app MatchEnv
menv CvSubstEnv
subst Type
ty1a [Type
ty1b] OutCoercion
co2 [OutCoercion
arg2]
ty_co_match MatchEnv
menv CvSubstEnv
subst (TyConApp TyCon
tc1 [Type]
tys) (TyConAppCo Role
_ TyCon
tc2 [OutCoercion]
cos) OutCoercion
_lkco OutCoercion
_rkco
= MatchEnv
-> CvSubstEnv
-> TyCon
-> [Type]
-> TyCon
-> [OutCoercion]
-> Maybe CvSubstEnv
ty_co_match_tc MatchEnv
menv CvSubstEnv
subst TyCon
tc1 [Type]
tys TyCon
tc2 [OutCoercion]
cos
ty_co_match MatchEnv
menv CvSubstEnv
subst (FunTy { ft_mult :: Type -> Type
ft_mult = Type
w, ft_arg :: Type -> Type
ft_arg = Type
ty1, ft_res :: Type -> Type
ft_res = Type
ty2 })
(FunCo { fco_mult :: OutCoercion -> OutCoercion
fco_mult = OutCoercion
co_w, fco_arg :: OutCoercion -> OutCoercion
fco_arg = OutCoercion
co1, fco_res :: OutCoercion -> OutCoercion
fco_res = OutCoercion
co2 }) OutCoercion
_lkco OutCoercion
_rkco
= MatchEnv
-> CvSubstEnv -> [Type] -> [OutCoercion] -> Maybe CvSubstEnv
ty_co_match_args MatchEnv
menv CvSubstEnv
subst [Type
w, Type
rep1, Type
rep2, Type
ty1, Type
ty2]
[OutCoercion
co_w, OutCoercion
co1_rep, OutCoercion
co2_rep, OutCoercion
co1, OutCoercion
co2]
where
rep1 :: Type
rep1 = HasDebugCallStack => Type -> Type
Type -> Type
getRuntimeRep Type
ty1
rep2 :: Type
rep2 = HasDebugCallStack => Type -> Type
Type -> Type
getRuntimeRep Type
ty2
co1_rep :: OutCoercion
co1_rep = HasDebugCallStack => OutCoercion -> OutCoercion
OutCoercion -> OutCoercion
mkRuntimeRepCo OutCoercion
co1
co2_rep :: OutCoercion
co2_rep = HasDebugCallStack => OutCoercion -> OutCoercion
OutCoercion -> OutCoercion
mkRuntimeRepCo OutCoercion
co2
ty_co_match MatchEnv
menv CvSubstEnv
subst (ForAllTy (Bndr Var
tv1 ForAllTyFlag
vis1t) Type
ty1)
(ForAllCo Var
tv2 ForAllTyFlag
vis1c ForAllTyFlag
vis2c OutCoercion
kind_co2 OutCoercion
co2)
OutCoercion
lkco OutCoercion
rkco
| Var -> Bool
isTyVar Var
tv1 Bool -> Bool -> Bool
&& Var -> Bool
isTyVar Var
tv2
, ForAllTyFlag
vis1t ForAllTyFlag -> ForAllTyFlag -> Bool
forall a. Eq a => a -> a -> Bool
== ForAllTyFlag
vis1c Bool -> Bool -> Bool
&& ForAllTyFlag
vis1c ForAllTyFlag -> ForAllTyFlag -> Bool
forall a. Eq a => a -> a -> Bool
== ForAllTyFlag
vis2c
= do { subst1 <- MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst (Var -> Type
tyVarKind Var
tv1) OutCoercion
kind_co2
OutCoercion
ki_ki_co OutCoercion
ki_ki_co
; let rn_env0 = MatchEnv -> RnEnv2
me_env MatchEnv
menv
rn_env1 = RnEnv2 -> Var -> Var -> RnEnv2
rnBndr2 RnEnv2
rn_env0 Var
tv1 Var
tv2
menv' = MatchEnv
menv { me_env = rn_env1 }
; ty_co_match menv' subst1 ty1 co2 lkco rkco }
where
ki_ki_co :: OutCoercion
ki_ki_co = Type -> OutCoercion
mkNomReflCo Type
liftedTypeKind
ty_co_match MatchEnv
_ CvSubstEnv
subst (CoercionTy {}) OutCoercion
_ OutCoercion
_ OutCoercion
_
= CvSubstEnv -> Maybe CvSubstEnv
forall a. a -> Maybe a
Just CvSubstEnv
subst
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty (GRefl Role
r Type
t (MCo OutCoercion
co)) OutCoercion
lkco OutCoercion
rkco
= MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty (Role -> Type -> MCoercion -> OutCoercion
GRefl Role
r Type
t MCoercion
MRefl) OutCoercion
lkco (OutCoercion
rkco HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion -> OutCoercion
mkSymCo OutCoercion
co)
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
co1 OutCoercion
lkco OutCoercion
rkco
| Just (CastTy Type
t OutCoercion
co, Role
r) <- OutCoercion -> Maybe (Type, Role)
isReflCo_maybe OutCoercion
co1
= let kco' :: OutCoercion
kco' = OutCoercion -> OutCoercion
mkSymCo OutCoercion
co
in MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty (Role -> Type -> OutCoercion
mkReflCo Role
r Type
t) (OutCoercion
lkco HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion
kco')
(OutCoercion
rkco HasDebugCallStack => OutCoercion -> OutCoercion -> OutCoercion
OutCoercion -> OutCoercion -> OutCoercion
`mkTransCo` OutCoercion
kco')
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
co OutCoercion
lkco OutCoercion
rkco
| Just OutCoercion
co' <- OutCoercion -> Maybe OutCoercion
pushRefl OutCoercion
co = MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
co' OutCoercion
lkco OutCoercion
rkco
| Bool
otherwise = Maybe CvSubstEnv
forall a. Maybe a
Nothing
ty_co_match_tc :: MatchEnv -> LiftCoEnv
-> TyCon -> [Type]
-> TyCon -> [Coercion]
-> Maybe LiftCoEnv
ty_co_match_tc :: MatchEnv
-> CvSubstEnv
-> TyCon
-> [Type]
-> TyCon
-> [OutCoercion]
-> Maybe CvSubstEnv
ty_co_match_tc MatchEnv
menv CvSubstEnv
subst TyCon
tc1 [Type]
tys1 TyCon
tc2 [OutCoercion]
cos2
= do { Bool -> Maybe ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard (TyCon
tc1 TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2)
; MatchEnv
-> CvSubstEnv -> [Type] -> [OutCoercion] -> Maybe CvSubstEnv
ty_co_match_args MatchEnv
menv CvSubstEnv
subst [Type]
tys1 [OutCoercion]
cos2 }
ty_co_match_app :: MatchEnv -> LiftCoEnv
-> Type -> [Type] -> Coercion -> [Coercion]
-> Maybe LiftCoEnv
ty_co_match_app :: MatchEnv
-> CvSubstEnv
-> Type
-> [Type]
-> OutCoercion
-> [OutCoercion]
-> Maybe CvSubstEnv
ty_co_match_app MatchEnv
menv CvSubstEnv
subst Type
ty1 [Type]
ty1args OutCoercion
co2 [OutCoercion]
co2args
| Just (Type
ty1', Type
ty1a) <- HasDebugCallStack => Type -> Maybe (Type, Type)
Type -> Maybe (Type, Type)
splitAppTyNoView_maybe Type
ty1
, Just (OutCoercion
co2', OutCoercion
co2a) <- OutCoercion -> Maybe (OutCoercion, OutCoercion)
splitAppCo_maybe OutCoercion
co2
= MatchEnv
-> CvSubstEnv
-> Type
-> [Type]
-> OutCoercion
-> [OutCoercion]
-> Maybe CvSubstEnv
ty_co_match_app MatchEnv
menv CvSubstEnv
subst Type
ty1' (Type
ty1a Type -> [Type] -> [Type]
forall a. a -> [a] -> [a]
: [Type]
ty1args) OutCoercion
co2' (OutCoercion
co2a OutCoercion -> [OutCoercion] -> [OutCoercion]
forall a. a -> [a] -> [a]
: [OutCoercion]
co2args)
| Bool
otherwise
= do { subst1 <- MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ki1 OutCoercion
ki2 OutCoercion
ki_ki_co OutCoercion
ki_ki_co
; let Pair lkco rkco = mkNomReflCo <$> coercionKind ki2
; subst2 <- ty_co_match menv subst1 ty1 co2 lkco rkco
; ty_co_match_args menv subst2 ty1args co2args }
where
ki1 :: Type
ki1 = HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
ty1
ki2 :: OutCoercion
ki2 = HasDebugCallStack => OutCoercion -> OutCoercion
OutCoercion -> OutCoercion
promoteCoercion OutCoercion
co2
ki_ki_co :: OutCoercion
ki_ki_co = Type -> OutCoercion
mkNomReflCo Type
liftedTypeKind
ty_co_match_args :: MatchEnv -> LiftCoEnv -> [Type] -> [Coercion]
-> Maybe LiftCoEnv
ty_co_match_args :: MatchEnv
-> CvSubstEnv -> [Type] -> [OutCoercion] -> Maybe CvSubstEnv
ty_co_match_args MatchEnv
menv CvSubstEnv
subst (Type
ty:[Type]
tys) (OutCoercion
arg:[OutCoercion]
args)
= do { let Pair Type
lty Type
rty = HasDebugCallStack => OutCoercion -> Pair Type
OutCoercion -> Pair Type
coercionKind OutCoercion
arg
lkco :: OutCoercion
lkco = Type -> OutCoercion
mkNomReflCo (HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
lty)
rkco :: OutCoercion
rkco = Type -> OutCoercion
mkNomReflCo (HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
rty)
; subst' <- MatchEnv
-> CvSubstEnv
-> Type
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> Maybe CvSubstEnv
ty_co_match MatchEnv
menv CvSubstEnv
subst Type
ty OutCoercion
arg OutCoercion
lkco OutCoercion
rkco
; ty_co_match_args menv subst' tys args }
ty_co_match_args MatchEnv
_ CvSubstEnv
subst [] [] = CvSubstEnv -> Maybe CvSubstEnv
forall a. a -> Maybe a
Just CvSubstEnv
subst
ty_co_match_args MatchEnv
_ CvSubstEnv
_ [Type]
_ [OutCoercion]
_ = Maybe CvSubstEnv
forall a. Maybe a
Nothing
pushRefl :: Coercion -> Maybe Coercion
pushRefl :: OutCoercion -> Maybe OutCoercion
pushRefl OutCoercion
co =
case (OutCoercion -> Maybe (Type, Role)
isReflCo_maybe OutCoercion
co) of
Just (AppTy Type
ty1 Type
ty2, Role
Nominal)
-> OutCoercion -> Maybe OutCoercion
forall a. a -> Maybe a
Just (OutCoercion -> OutCoercion -> OutCoercion
AppCo (Role -> Type -> OutCoercion
mkReflCo Role
Nominal Type
ty1) (Type -> OutCoercion
mkNomReflCo Type
ty2))
Just (FunTy FunTyFlag
af Type
w Type
ty1 Type
ty2, Role
r)
-> OutCoercion -> Maybe OutCoercion
forall a. a -> Maybe a
Just (Role
-> FunTyFlag
-> FunTyFlag
-> OutCoercion
-> OutCoercion
-> OutCoercion
-> OutCoercion
FunCo Role
r FunTyFlag
af FunTyFlag
af (Role -> Type -> OutCoercion
mkReflCo Role
r Type
w) (Role -> Type -> OutCoercion
mkReflCo Role
r Type
ty1) (Role -> Type -> OutCoercion
mkReflCo Role
r Type
ty2))
Just (TyConApp TyCon
tc [Type]
tys, Role
r)
-> OutCoercion -> Maybe OutCoercion
forall a. a -> Maybe a
Just (Role -> TyCon -> [OutCoercion] -> OutCoercion
TyConAppCo Role
r TyCon
tc ((Role -> Type -> OutCoercion) -> [Role] -> [Type] -> [OutCoercion]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith Role -> Type -> OutCoercion
mkReflCo (Role -> TyCon -> [Role]
tyConRoleListX Role
r TyCon
tc) [Type]
tys))
Just (ForAllTy (Bndr Var
tv ForAllTyFlag
vis) Type
ty, Role
r)
-> OutCoercion -> Maybe OutCoercion
forall a. a -> Maybe a
Just (ForAllCo { fco_tcv :: Var
fco_tcv = Var
tv, fco_visL :: ForAllTyFlag
fco_visL = ForAllTyFlag
vis, fco_visR :: ForAllTyFlag
fco_visR = ForAllTyFlag
vis
, fco_kind :: OutCoercion
fco_kind = Type -> OutCoercion
mkNomReflCo (Var -> Type
varType Var
tv)
, fco_body :: OutCoercion
fco_body = Role -> Type -> OutCoercion
mkReflCo Role
r Type
ty })
Maybe (Type, Role)
_ -> Maybe OutCoercion
forall a. Maybe a
Nothing