{-# LANGUAGE CPP #-}
module GHC.Core.Coercion.Opt
( optCoercion
, OptCoercionOpts (..)
)
where
import GHC.Prelude
import GHC.Tc.Utils.TcType ( exactTyCoVarsOfType )
import GHC.Core.TyCo.Rep
import GHC.Core.TyCo.Subst
import GHC.Core.TyCo.Compare( eqType, eqForAllVis )
import GHC.Core.Coercion
import GHC.Core.Type as Type hiding( substTyVarBndr, substTy )
import GHC.Core.TyCon
import GHC.Core.Coercion.Axiom
import GHC.Core.Unify
import GHC.Types.Basic( SwapFlag(..), flipSwap, isSwapped, pickSwap, notSwapped )
import GHC.Types.Var
import GHC.Types.Var.Set
import GHC.Types.Var.Env
import GHC.Data.Pair
import GHC.Utils.Outputable
import GHC.Utils.Constants (debugIsOn)
import GHC.Utils.Misc
import GHC.Utils.Panic
import Control.Monad ( zipWithM )
newtype OptCoercionOpts = OptCoercionOpts
{ OptCoercionOpts -> Bool
optCoercionEnabled :: Bool
}
optCoercion :: OptCoercionOpts -> Subst -> Coercion -> NormalCo
optCoercion :: OptCoercionOpts -> Subst -> Coercion -> Coercion
optCoercion OptCoercionOpts
opts Subst
env Coercion
co
| OptCoercionOpts -> Bool
optCoercionEnabled OptCoercionOpts
opts
= Subst -> Coercion -> Coercion
optCoercion' Subst
env Coercion
co
| Bool
otherwise
= HasDebugCallStack => Subst -> Coercion -> Coercion
Subst -> Coercion -> Coercion
substCo Subst
env Coercion
co
optCoercion' :: Subst -> Coercion -> NormalCo
optCoercion' :: Subst -> Coercion -> Coercion
optCoercion' Subst
env Coercion
co
| Bool
debugIsOn
= let out_co :: Coercion
out_co = LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
lc SwapFlag
NotSwapped Coercion
co
(Pair Type
in_ty1 Type
in_ty2, Role
in_role) = Coercion -> (Pair Type, Role)
coercionKindRole Coercion
co
(Pair Type
out_ty1 Type
out_ty2, Role
out_role) = Coercion -> (Pair Type, Role)
coercionKindRole Coercion
out_co
details :: SDoc
details = [SDoc] -> SDoc
forall doc. IsDoc doc => [doc] -> doc
vcat [ String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_co:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
co
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_ty1:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
in_ty1
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_ty2:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
in_ty2
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"out_co:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
out_co
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"out_ty1:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
out_ty1
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"out_ty2:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
out_ty2
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
in_role
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"out_role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
out_role
]
in
Bool -> String -> SDoc -> Coercion -> Coercion
forall a. HasCallStack => Bool -> String -> SDoc -> a -> a
warnPprTrace (Bool -> Bool
not (Coercion -> Bool
isReflCo Coercion
out_co) Bool -> Bool -> Bool
&& Coercion -> Bool
isReflexiveCo Coercion
out_co)
String
"optCoercion: reflexive but not refl" SDoc
details (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
Coercion
out_co
| Bool
otherwise
= LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
lc SwapFlag
NotSwapped Coercion
co
where
lc :: LiftingContext
lc = Subst -> LiftingContext
mkSubstLiftingContext Subst
env
type NormalCo = Coercion
type NormalNonIdCo = NormalCo
type ReprFlag = Bool
opt_co1 :: LiftingContext
-> SwapFlag
-> Coercion -> NormalCo
opt_co1 :: LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
env SwapFlag
sym Coercion
co = LiftingContext -> SwapFlag -> Role -> Coercion -> Coercion
opt_co2 LiftingContext
env SwapFlag
sym (Coercion -> Role
coercionRole Coercion
co) Coercion
co
opt_co2 :: LiftingContext
-> SwapFlag
-> Role
-> Coercion -> NormalCo
opt_co2 :: LiftingContext -> SwapFlag -> Role -> Coercion -> Coercion
opt_co2 LiftingContext
env SwapFlag
sym Role
Phantom Coercion
co = LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_phantom LiftingContext
env SwapFlag
sym Coercion
co
opt_co2 LiftingContext
env SwapFlag
sym Role
r Coercion
co = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
r Coercion
co
opt_co3 :: LiftingContext -> SwapFlag -> Maybe Role -> Role -> Coercion -> NormalCo
opt_co3 :: LiftingContext
-> SwapFlag -> Maybe Role -> Role -> Coercion -> Coercion
opt_co3 LiftingContext
env SwapFlag
sym (Just Role
Phantom) Role
_ Coercion
co = LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_phantom LiftingContext
env SwapFlag
sym Coercion
co
opt_co3 LiftingContext
env SwapFlag
sym (Just Role
Representational) Role
r Coercion
co = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
True Role
r Coercion
co
opt_co3 LiftingContext
env SwapFlag
sym Maybe Role
_ Role
r Coercion
co = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
r Coercion
co
opt_co4, opt_co4' :: LiftingContext -> SwapFlag -> ReprFlag
-> Role -> Coercion -> NormalCo
opt_co4 :: LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4'
opt_co4' :: LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (Refl Type
ty)
= Bool -> SDoc -> Coercion -> Coercion
forall a. HasCallStack => Bool -> SDoc -> a -> a
assertPpr (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
Nominal)
(String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Expected role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
r SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$
String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Found role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
Nominal SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$
String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Type:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
ty) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
sym (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ HasDebugCallStack => Role -> LiftingContext -> Type -> Coercion
Role -> LiftingContext -> Type -> Coercion
liftCoSubst (Bool -> Role -> Role
chooseRole Bool
rep Role
r) LiftingContext
env Type
ty
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (GRefl Role
_r Type
ty MCoercion
MRefl)
= Bool -> SDoc -> Coercion -> Coercion
forall a. HasCallStack => Bool -> SDoc -> a -> a
assertPpr (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
_r)
(String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Expected role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
r SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$
String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Found role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
_r SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$
String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Type:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
ty) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
sym (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ HasDebugCallStack => Role -> LiftingContext -> Type -> Coercion
Role -> LiftingContext -> Type -> Coercion
liftCoSubst (Bool -> Role -> Role
chooseRole Bool
rep Role
r) LiftingContext
env Type
ty
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (GRefl Role
_r Type
ty (MCo Coercion
kco))
= Bool -> SDoc -> Coercion -> Coercion
forall a. HasCallStack => Bool -> SDoc -> a -> a
assertPpr (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
_r)
(String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Expected role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
r SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$
String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Found role:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Role -> SDoc
forall a. Outputable a => a -> SDoc
ppr Role
_r SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$
String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"Type:" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
ty) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
if Coercion -> Bool
isGReflCo Coercion
kco Bool -> Bool -> Bool
|| Coercion -> Bool
isGReflCo Coercion
kco'
then SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
sym Coercion
ty_co
else SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
sym (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ HasDebugCallStack =>
Role -> Type -> Coercion -> Coercion -> Coercion
Role -> Type -> Coercion -> Coercion -> Coercion
mk_coherence_right_co Role
r' (HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
ty_co) Coercion
kco' Coercion
ty_co
where
r' :: Role
r' = Bool -> Role -> Role
chooseRole Bool
rep Role
r
ty_co :: Coercion
ty_co = HasDebugCallStack => Role -> LiftingContext -> Type -> Coercion
Role -> LiftingContext -> Type -> Coercion
liftCoSubst Role
r' LiftingContext
env Type
ty
kco' :: Coercion
kco' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
NotSwapped Bool
False Role
Nominal Coercion
kco
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (SymCo Coercion
co) = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env (SwapFlag -> SwapFlag
flipSwap SwapFlag
sym) Bool
rep Role
r Coercion
co
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r g :: Coercion
g@(TyConAppCo Role
_r TyCon
tc [Coercion]
cos)
= Bool -> Coercion -> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
_r) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
case (Bool
rep, Role
r) of
(Bool
True, Role
Nominal) ->
HasDebugCallStack => Role -> TyCon -> [Coercion] -> Coercion
Role -> TyCon -> [Coercion] -> Coercion
mkTyConAppCo Role
Representational TyCon
tc
((Maybe Role -> Role -> Coercion -> Coercion)
-> [Maybe Role] -> [Role] -> [Coercion] -> [Coercion]
forall a b c d. (a -> b -> c -> d) -> [a] -> [b] -> [c] -> [d]
zipWith3 (LiftingContext
-> SwapFlag -> Maybe Role -> Role -> Coercion -> Coercion
opt_co3 LiftingContext
env SwapFlag
sym)
((Role -> Maybe Role) -> [Role] -> [Maybe Role]
forall a b. (a -> b) -> [a] -> [b]
map Role -> Maybe Role
forall a. a -> Maybe a
Just (TyCon -> [Role]
tyConRoleListRepresentational TyCon
tc))
(Role -> [Role]
forall a. a -> [a]
repeat Role
Nominal)
[Coercion]
cos)
(Bool
False, Role
Nominal) ->
HasDebugCallStack => Role -> TyCon -> [Coercion] -> Coercion
Role -> TyCon -> [Coercion] -> Coercion
mkTyConAppCo Role
Nominal TyCon
tc ((Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map (LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
Nominal) [Coercion]
cos)
(Bool
_, Role
Representational) ->
HasDebugCallStack => Role -> TyCon -> [Coercion] -> Coercion
Role -> TyCon -> [Coercion] -> Coercion
mkTyConAppCo Role
r TyCon
tc ((Role -> Coercion -> Coercion)
-> [Role] -> [Coercion] -> [Coercion]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith (LiftingContext -> SwapFlag -> Role -> Coercion -> Coercion
opt_co2 LiftingContext
env SwapFlag
sym)
(TyCon -> [Role]
tyConRoleListRepresentational TyCon
tc)
[Coercion]
cos)
(Bool
_, Role
Phantom) -> String -> SDoc -> Coercion
forall a. HasCallStack => String -> SDoc -> a
pprPanic String
"opt_co4 sees a phantom!" (Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
g)
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (AppCo Coercion
co1 Coercion
co2)
= Coercion -> Coercion -> Coercion
mkAppCo (LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
r Coercion
co1)
(LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
Nominal Coercion
co2)
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (ForAllCo { fco_tcv :: Coercion -> CoVar
fco_tcv = CoVar
tv, fco_visL :: Coercion -> ForAllTyFlag
fco_visL = ForAllTyFlag
visL, fco_visR :: Coercion -> ForAllTyFlag
fco_visR = ForAllTyFlag
visR
, fco_kind :: Coercion -> Coercion
fco_kind = Coercion
k_co, fco_body :: Coercion -> Coercion
fco_body = Coercion
co })
= case LiftingContext
-> SwapFlag
-> CoVar
-> Coercion
-> (LiftingContext, CoVar, Coercion)
optForAllCoBndr LiftingContext
env SwapFlag
sym CoVar
tv Coercion
k_co of
(LiftingContext
env', CoVar
tv', Coercion
k_co') -> HasDebugCallStack =>
CoVar
-> ForAllTyFlag -> ForAllTyFlag -> Coercion -> Coercion -> Coercion
CoVar
-> ForAllTyFlag -> ForAllTyFlag -> Coercion -> Coercion -> Coercion
mkForAllCo CoVar
tv' ForAllTyFlag
visL' ForAllTyFlag
visR' Coercion
k_co' (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env' SwapFlag
sym Bool
rep Role
r Coercion
co
where
!(ForAllTyFlag
visL', ForAllTyFlag
visR') = SwapFlag
-> (ForAllTyFlag, ForAllTyFlag) -> (ForAllTyFlag, ForAllTyFlag)
forall a. SwapFlag -> (a, a) -> (a, a)
swapSym SwapFlag
sym (ForAllTyFlag
visL, ForAllTyFlag
visR)
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (FunCo Role
_r FunTyFlag
afl FunTyFlag
afr Coercion
cow Coercion
co1 Coercion
co2)
= Bool -> Coercion -> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
_r) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
Role
-> FunTyFlag
-> FunTyFlag
-> Coercion
-> Coercion
-> Coercion
-> Coercion
mkFunCo2 Role
r' FunTyFlag
afl' FunTyFlag
afr' Coercion
cow' Coercion
co1' Coercion
co2'
where
co1' :: Coercion
co1' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
r Coercion
co1
co2' :: Coercion
co2' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
r Coercion
co2
cow' :: Coercion
cow' = LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
env SwapFlag
sym Coercion
cow
!r' :: Role
r' | Bool
rep = Role
Representational
| Bool
otherwise = Role
r
!(FunTyFlag
afl', FunTyFlag
afr') = SwapFlag -> (FunTyFlag, FunTyFlag) -> (FunTyFlag, FunTyFlag)
forall a. SwapFlag -> (a, a) -> (a, a)
swapSym SwapFlag
sym (FunTyFlag
afl, FunTyFlag
afr)
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (CoVarCo CoVar
cv)
| Just Coercion
co <- LiftingContext -> CoVar -> Maybe Coercion
lcLookupCoVar LiftingContext
env CoVar
cv
=
LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 (LiftingContext -> LiftingContext
zapLiftingContext LiftingContext
env) SwapFlag
sym Bool
rep Role
r Coercion
co
| Type
ty1 HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` Type
ty2
= Role -> Type -> Coercion
mkReflCo (Bool -> Role -> Role
chooseRole Bool
rep Role
r) Type
ty1
| Bool
otherwise
= Bool -> Coercion -> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (CoVar -> Bool
isCoVar CoVar
cv1) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
Bool -> Role -> Coercion -> Coercion
wrapRole Bool
rep Role
r (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
sym (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
CoVar -> Coercion
CoVarCo CoVar
cv1
where
Pair Type
ty1 Type
ty2 = HasDebugCallStack => CoVar -> Pair Type
CoVar -> Pair Type
coVarTypes CoVar
cv1
cv1 :: CoVar
cv1 = case InScopeSet -> CoVar -> Maybe CoVar
lookupInScope (LiftingContext -> InScopeSet
lcInScopeSet LiftingContext
env) CoVar
cv of
Just CoVar
cv1 -> CoVar
cv1
Maybe CoVar
Nothing -> Bool -> String -> SDoc -> CoVar -> CoVar
forall a. HasCallStack => Bool -> String -> SDoc -> a -> a
warnPprTrace Bool
True
String
"opt_co: not in scope"
(CoVar -> SDoc
forall a. Outputable a => a -> SDoc
ppr CoVar
cv SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$ LiftingContext -> SDoc
forall a. Outputable a => a -> SDoc
ppr LiftingContext
env)
CoVar
cv
opt_co4' LiftingContext
_ SwapFlag
_ Bool
_ Role
_ (HoleCo CoercionHole
h)
= String -> SDoc -> Coercion
forall a. HasCallStack => String -> SDoc -> a
pprPanic String
"opt_univ fell into a hole" (CoercionHole -> SDoc
forall a. Outputable a => a -> SDoc
ppr CoercionHole
h)
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (AxiomCo CoAxiomRule
con [Coercion]
cos)
= Bool
-> (Bool -> Role -> Coercion -> Coercion)
-> Bool
-> Role
-> Coercion
-> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== CoAxiomRule -> Role
coAxiomRuleRole CoAxiomRule
con )
Bool -> Role -> Coercion -> Coercion
wrapRole Bool
rep (CoAxiomRule -> Role
coAxiomRuleRole CoAxiomRule
con) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
sym (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
CoAxiomRule -> [Coercion] -> Coercion
AxiomCo CoAxiomRule
con ((Role -> Coercion -> Coercion)
-> [Role] -> [Coercion] -> [Coercion]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith (LiftingContext -> SwapFlag -> Role -> Coercion -> Coercion
opt_co2 LiftingContext
env SwapFlag
NotSwapped)
(CoAxiomRule -> [Role]
coAxiomRuleArgRoles CoAxiomRule
con)
[Coercion]
cos)
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (UnivCo { uco_prov :: Coercion -> UnivCoProvenance
uco_prov = UnivCoProvenance
prov, uco_lty :: Coercion -> Type
uco_lty = Type
t1
, uco_rty :: Coercion -> Type
uco_rty = Type
t2, uco_deps :: Coercion -> [Coercion]
uco_deps = [Coercion]
deps })
= LiftingContext
-> SwapFlag
-> UnivCoProvenance
-> [Coercion]
-> Role
-> Type
-> Type
-> Coercion
opt_univ LiftingContext
env SwapFlag
sym UnivCoProvenance
prov [Coercion]
deps (Bool -> Role -> Role
chooseRole Bool
rep Role
r) Type
t1 Type
t2
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (TransCo Coercion
co1 Coercion
co2)
| SwapFlag -> Bool
isSwapped SwapFlag
sym = HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
in_scope Coercion
co2' Coercion
co1'
| Bool
otherwise = HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
in_scope Coercion
co1' Coercion
co2'
where
co1' :: Coercion
co1' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
r Coercion
co1
co2' :: Coercion
co2' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
r Coercion
co2
in_scope :: InScopeSet
in_scope = LiftingContext -> InScopeSet
lcInScopeSet LiftingContext
env
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (SelCo CoSel
cs Coercion
co)
= Bool -> Role -> Coercion -> Coercion
wrapRole Bool
rep Role
r (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo CoSel
cs (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
env SwapFlag
sym Coercion
co
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (LRCo LeftOrRight
lr Coercion
co)
| Just (Coercion, Coercion)
pr_co <- Coercion -> Maybe (Coercion, Coercion)
splitAppCo_maybe Coercion
co
= Bool
-> (LiftingContext
-> SwapFlag -> Bool -> Role -> Coercion -> Coercion)
-> LiftingContext
-> SwapFlag
-> Bool
-> Role
-> Coercion
-> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
Nominal )
LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
Nominal (LeftOrRight -> (Coercion, Coercion) -> Coercion
forall {a}. LeftOrRight -> (a, a) -> a
pick_lr LeftOrRight
lr (Coercion, Coercion)
pr_co)
| Just (Coercion, Coercion)
pr_co <- Coercion -> Maybe (Coercion, Coercion)
splitAppCo_maybe Coercion
co'
= Bool -> Coercion -> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
Nominal) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
if Bool
rep
then LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 (LiftingContext -> LiftingContext
zapLiftingContext LiftingContext
env) SwapFlag
NotSwapped Bool
True Role
Nominal (LeftOrRight -> (Coercion, Coercion) -> Coercion
forall {a}. LeftOrRight -> (a, a) -> a
pick_lr LeftOrRight
lr (Coercion, Coercion)
pr_co)
else LeftOrRight -> (Coercion, Coercion) -> Coercion
forall {a}. LeftOrRight -> (a, a) -> a
pick_lr LeftOrRight
lr (Coercion, Coercion)
pr_co
| Bool
otherwise
= Bool -> Role -> Coercion -> Coercion
wrapRole Bool
rep Role
Nominal (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ LeftOrRight -> Coercion -> Coercion
LRCo LeftOrRight
lr Coercion
co'
where
co' :: Coercion
co' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
Nominal Coercion
co
pick_lr :: LeftOrRight -> (a, a) -> a
pick_lr LeftOrRight
CLeft (a
l, a
_) = a
l
pick_lr LeftOrRight
CRight (a
_, a
r) = a
r
opt_co4' LiftingContext
env SwapFlag
sym Bool
rep Role
r (InstCo Coercion
fun_co Coercion
arg_co)
| Just (CoVar
tv, ForAllTyFlag
_visL, ForAllTyFlag
_visR, Coercion
k_co, Coercion
body_co) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_ty_maybe Coercion
fun_co
, let arg_co' :: Coercion
arg_co' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
NotSwapped Bool
False Role
Nominal Coercion
arg_co
k_co' :: Coercion
k_co' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
NotSwapped Bool
False Role
Nominal Coercion
k_co
s2' :: Type
s2' = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
arg_co'
tv_co :: Coercion
tv_co = HasDebugCallStack =>
Role -> Type -> Coercion -> Coercion -> Coercion
Role -> Type -> Coercion -> Coercion -> Coercion
mk_coherence_right_co Role
Nominal Type
s2' (Coercion -> Coercion
mkSymCo Coercion
k_co') Coercion
arg_co'
= LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 (LiftingContext -> CoVar -> Coercion -> LiftingContext
extendLiftingContext LiftingContext
env CoVar
tv Coercion
tv_co) SwapFlag
sym Bool
rep Role
r Coercion
body_co
| Just (CoVar
cv, ForAllTyFlag
_visL, ForAllTyFlag
_visR, Coercion
_kind_co, Coercion
body_co) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_co_maybe Coercion
fun_co
, CoercionTy Coercion
h1 <- HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
arg_co
, let h1' :: Coercion
h1' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
NotSwapped Bool
False Role
Nominal Coercion
h1
= LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 (LiftingContext -> CoVar -> Coercion -> LiftingContext
extendLiftingContextCvSubst LiftingContext
env CoVar
cv Coercion
h1') SwapFlag
sym Bool
rep Role
r Coercion
body_co
| Just (CoVar
tv', ForAllTyFlag
_visL, ForAllTyFlag
_visR, Coercion
k_co', Coercion
body_co') <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_ty_maybe Coercion
fun_co'
, let s2' :: Type
s2' = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
arg_co'
tv_co :: Coercion
tv_co = HasDebugCallStack =>
Role -> Type -> Coercion -> Coercion -> Coercion
Role -> Type -> Coercion -> Coercion -> Coercion
mk_coherence_right_co Role
Nominal Type
s2' (Coercion -> Coercion
mkSymCo Coercion
k_co') Coercion
arg_co'
env' :: LiftingContext
env' = LiftingContext -> CoVar -> Coercion -> LiftingContext
extendLiftingContext (LiftingContext -> LiftingContext
zapLiftingContext LiftingContext
env) CoVar
tv' Coercion
tv_co
= LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env' SwapFlag
NotSwapped Bool
False Role
r' Coercion
body_co'
| Just (CoVar
cv', ForAllTyFlag
_visL, ForAllTyFlag
_visR, Coercion
_kind_co', Coercion
body_co') <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_co_maybe Coercion
fun_co'
, CoercionTy Coercion
h1' <- HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
arg_co'
, let env' :: LiftingContext
env' = LiftingContext -> CoVar -> Coercion -> LiftingContext
extendLiftingContextCvSubst (LiftingContext -> LiftingContext
zapLiftingContext LiftingContext
env) CoVar
cv' Coercion
h1'
= LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env' SwapFlag
NotSwapped Bool
False Role
r' Coercion
body_co'
| Bool
otherwise = Coercion -> Coercion -> Coercion
InstCo Coercion
fun_co' Coercion
arg_co'
where
fun_co' :: Coercion
fun_co' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
rep Role
r Coercion
fun_co
arg_co' :: Coercion
arg_co' = LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
Nominal Coercion
arg_co
r' :: Role
r' = Bool -> Role -> Role
chooseRole Bool
rep Role
r
opt_co4' LiftingContext
env SwapFlag
sym Bool
_rep Role
r (KindCo Coercion
co)
= Bool -> Coercion -> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
Nominal) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
let kco' :: Coercion
kco' = HasDebugCallStack => Coercion -> Coercion
Coercion -> Coercion
promoteCoercion Coercion
co in
case Coercion
kco' of
KindCo Coercion
co' -> HasDebugCallStack => Coercion -> Coercion
Coercion -> Coercion
promoteCoercion (LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
env SwapFlag
sym Coercion
co')
Coercion
_ -> LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
Nominal Coercion
kco'
opt_co4' LiftingContext
env SwapFlag
sym Bool
_ Role
r (SubCo Coercion
co)
= Bool -> Coercion -> Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
Representational) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
True Role
Nominal Coercion
co
opt_phantom :: LiftingContext -> SwapFlag -> Coercion -> NormalCo
opt_phantom :: LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_phantom LiftingContext
env SwapFlag
sym (UnivCo { uco_prov :: Coercion -> UnivCoProvenance
uco_prov = UnivCoProvenance
prov, uco_lty :: Coercion -> Type
uco_lty = Type
t1
, uco_rty :: Coercion -> Type
uco_rty = Type
t2, uco_deps :: Coercion -> [Coercion]
uco_deps = [Coercion]
deps })
= LiftingContext
-> SwapFlag
-> UnivCoProvenance
-> [Coercion]
-> Role
-> Type
-> Type
-> Coercion
opt_univ LiftingContext
env SwapFlag
sym UnivCoProvenance
prov [Coercion]
deps Role
Phantom Type
t1 Type
t2
opt_phantom LiftingContext
env SwapFlag
sym Coercion
co
= LiftingContext
-> SwapFlag
-> UnivCoProvenance
-> [Coercion]
-> Role
-> Type
-> Type
-> Coercion
opt_univ LiftingContext
env SwapFlag
sym UnivCoProvenance
PhantomProv [Coercion -> Coercion
mkKindCo Coercion
co] Role
Phantom Type
ty1 Type
ty2
where
Pair Type
ty1 Type
ty2 = HasDebugCallStack => Coercion -> Pair Type
Coercion -> Pair Type
coercionKind Coercion
co
opt_univ :: LiftingContext -> SwapFlag -> UnivCoProvenance
-> [Coercion]
-> Role -> Type -> Type -> Coercion
opt_univ :: LiftingContext
-> SwapFlag
-> UnivCoProvenance
-> [Coercion]
-> Role
-> Type
-> Type
-> Coercion
opt_univ LiftingContext
env SwapFlag
sym UnivCoProvenance
prov [Coercion]
deps Role
role Type
ty1 Type
ty2
= let ty1' :: Type
ty1' = Subst -> Type -> Type
substTyUnchecked (LiftingContext -> Subst
lcSubstLeft LiftingContext
env) Type
ty1
ty2' :: Type
ty2' = Subst -> Type -> Type
substTyUnchecked (LiftingContext -> Subst
lcSubstRight LiftingContext
env) Type
ty2
deps' :: [Coercion]
deps' = (Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map (LiftingContext -> SwapFlag -> Coercion -> Coercion
opt_co1 LiftingContext
env SwapFlag
sym) [Coercion]
deps
(Type
ty1'', Type
ty2'') = SwapFlag -> (Type, Type) -> (Type, Type)
forall a. SwapFlag -> (a, a) -> (a, a)
swapSym SwapFlag
sym (Type
ty1', Type
ty2')
in
UnivCoProvenance -> [Coercion] -> Role -> Type -> Type -> Coercion
mkUnivCo UnivCoProvenance
prov [Coercion]
deps' Role
role Type
ty1'' Type
ty2''
opt_transList :: HasDebugCallStack => InScopeSet -> [NormalCo] -> [NormalCo] -> [NormalCo]
opt_transList :: HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is = String
-> (Coercion -> Coercion -> Coercion)
-> [Coercion]
-> [Coercion]
-> [Coercion]
forall a b c.
HasDebugCallStack =>
String -> (a -> b -> c) -> [a] -> [b] -> [c]
zipWithEqual String
"opt_transList" (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is)
opt_trans :: HasDebugCallStack => InScopeSet -> NormalCo -> NormalCo -> NormalCo
opt_trans :: HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1 Coercion
co2
=
HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans' InScopeSet
is Coercion
co1 Coercion
co2
opt_trans' :: HasDebugCallStack => InScopeSet -> NormalCo -> NormalCo -> NormalCo
opt_trans' :: HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans' InScopeSet
is Coercion
co1 Coercion
co2
| Coercion -> Bool
isReflCo Coercion
co1 = Coercion
co2
| Bool
otherwise = HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans1 InScopeSet
is Coercion
co1 Coercion
co2
opt_trans1 :: HasDebugCallStack => InScopeSet -> NormalNonIdCo -> NormalCo -> NormalCo
opt_trans1 :: HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans1 InScopeSet
is Coercion
co1 Coercion
co2
| Coercion -> Bool
isReflCo Coercion
co2 = Coercion
co1
| Bool
otherwise = HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans2 InScopeSet
is Coercion
co1 Coercion
co2
opt_trans2 :: HasDebugCallStack => InScopeSet -> NormalNonIdCo -> NormalNonIdCo -> NormalCo
opt_trans2 :: HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans2 InScopeSet
is (TransCo Coercion
co1a Coercion
co1b) Coercion
co2
= HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1a (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1b Coercion
co2)
opt_trans2 InScopeSet
is Coercion
co1 Coercion
co2
| Just Coercion
co <- HasDebugCallStack =>
InScopeSet -> Coercion -> Coercion -> Maybe Coercion
InScopeSet -> Coercion -> Coercion -> Maybe Coercion
opt_trans_rule InScopeSet
is Coercion
co1 Coercion
co2
= Coercion
co
opt_trans2 InScopeSet
is Coercion
co1 (TransCo Coercion
co2a Coercion
co2b)
| Just Coercion
co1_2a <- HasDebugCallStack =>
InScopeSet -> Coercion -> Coercion -> Maybe Coercion
InScopeSet -> Coercion -> Coercion -> Maybe Coercion
opt_trans_rule InScopeSet
is Coercion
co1 Coercion
co2a
= if Coercion -> Bool
isReflCo Coercion
co1_2a
then Coercion
co2b
else HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans1 InScopeSet
is Coercion
co1_2a Coercion
co2b
opt_trans2 InScopeSet
_ Coercion
co1 Coercion
co2
= HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
mk_trans_co Coercion
co1 Coercion
co2
opt_trans_rule :: HasDebugCallStack => InScopeSet -> NormalNonIdCo -> NormalNonIdCo -> Maybe NormalCo
opt_trans_rule :: HasDebugCallStack =>
InScopeSet -> Coercion -> Coercion -> Maybe Coercion
opt_trans_rule InScopeSet
_ Coercion
in_co1 Coercion
in_co2
| Bool -> SDoc -> Bool -> Bool
forall a. HasCallStack => Bool -> SDoc -> a -> a
assertPpr (HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
in_co1 HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
in_co2)
([SDoc] -> SDoc
forall doc. IsDoc doc => [doc] -> doc
vcat [ String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_co1" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
in_co1
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_co2" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
in_co2
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_co1 kind" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Pair Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (HasDebugCallStack => Coercion -> Pair Type
Coercion -> Pair Type
coercionKind Coercion
in_co1)
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"in_co2 kind" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Pair Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (HasDebugCallStack => Coercion -> Pair Type
Coercion -> Pair Type
coercionKind Coercion
in_co2)
, SDoc
HasCallStack => SDoc
callStackDoc ]) (Bool -> Bool) -> Bool -> Bool
forall a b. (a -> b) -> a -> b
$
Bool
False
= String -> Maybe Coercion
forall a. HasCallStack => String -> a
panic String
"opt_trans_rule"
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(GRefl Role
r1 Type
t1 (MCo Coercion
co1)) in_co2 :: Coercion
in_co2@(GRefl Role
r2 Type
_t2 (MCo Coercion
co2))
= Bool -> Maybe Coercion -> Maybe Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r1 Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
r2) (Maybe Coercion -> Maybe Coercion)
-> Maybe Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"GRefl" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
Role -> Type -> Coercion -> Coercion
mk_grefl_right_co Role
r1 Type
t1 (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1 Coercion
co2)
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(SelCo CoSel
d1 Coercion
co1) in_co2 :: Coercion
in_co2@(SelCo CoSel
d2 Coercion
co2)
| CoSel
d1 CoSel -> CoSel -> Bool
forall a. Eq a => a -> a -> Bool
== CoSel
d2
, Coercion -> Role
coercionRole Coercion
co1 Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Coercion -> Role
coercionRole Coercion
co2
, Coercion
co1 Coercion -> Coercion -> Bool
`compatible_co` Coercion
co2
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"PushNth" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo CoSel
d1 (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1 Coercion
co2)
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(LRCo LeftOrRight
d1 Coercion
co1) in_co2 :: Coercion
in_co2@(LRCo LeftOrRight
d2 Coercion
co2)
| LeftOrRight
d1 LeftOrRight -> LeftOrRight -> Bool
forall a. Eq a => a -> a -> Bool
== LeftOrRight
d2
, Coercion
co1 Coercion -> Coercion -> Bool
`compatible_co` Coercion
co2
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"PushLR" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
LeftOrRight -> Coercion -> Coercion
mkLRCo LeftOrRight
d1 (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1 Coercion
co2)
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(InstCo Coercion
co1 Coercion
ty1) in_co2 :: Coercion
in_co2@(InstCo Coercion
co2 Coercion
ty2)
| Coercion
ty1 Coercion -> Coercion -> Bool
`eqCoercion` Coercion
ty2
, Coercion
co1 Coercion -> Coercion -> Bool
`compatible_co` Coercion
co2
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"TrPushInst" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
Coercion -> Coercion -> Coercion
mkInstCo (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1 Coercion
co2) Coercion
ty1
opt_trans_rule InScopeSet
_
in_co1 :: Coercion
in_co1@(UnivCo { uco_prov :: Coercion -> UnivCoProvenance
uco_prov = UnivCoProvenance
p1, uco_role :: Coercion -> Role
uco_role = Role
r1, uco_lty :: Coercion -> Type
uco_lty = Type
tyl1, uco_deps :: Coercion -> [Coercion]
uco_deps = [Coercion]
deps1 })
in_co2 :: Coercion
in_co2@(UnivCo { uco_prov :: Coercion -> UnivCoProvenance
uco_prov = UnivCoProvenance
p2, uco_role :: Coercion -> Role
uco_role = Role
r2, uco_rty :: Coercion -> Type
uco_rty = Type
tyr2, uco_deps :: Coercion -> [Coercion]
uco_deps = [Coercion]
deps2 })
| UnivCoProvenance
p1 UnivCoProvenance -> UnivCoProvenance -> Bool
forall a. Eq a => a -> a -> Bool
== UnivCoProvenance
p2
= Bool -> Maybe Coercion -> Maybe Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r1 Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
r2) (Maybe Coercion -> Maybe Coercion)
-> Maybe Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"UnivCo" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
UnivCoProvenance -> [Coercion] -> Role -> Type -> Type -> Coercion
mkUnivCo UnivCoProvenance
p1 ([Coercion]
deps1 [Coercion] -> [Coercion] -> [Coercion]
forall a. [a] -> [a] -> [a]
++ [Coercion]
deps2) Role
r1 Type
tyl1 Type
tyr2
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(TyConAppCo Role
r1 TyCon
tc1 [Coercion]
cos1) in_co2 :: Coercion
in_co2@(TyConAppCo Role
r2 TyCon
tc2 [Coercion]
cos2)
| TyCon
tc1 TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2
= Bool -> Maybe Coercion -> Maybe Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r1 Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
r2) (Maybe Coercion -> Maybe Coercion)
-> Maybe Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"PushTyConApp" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack => Role -> TyCon -> [Coercion] -> Coercion
Role -> TyCon -> [Coercion] -> Coercion
mkTyConAppCo Role
r1 TyCon
tc1 (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos1 [Coercion]
cos2)
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(FunCo Role
r1 FunTyFlag
afl1 FunTyFlag
afr1 Coercion
w1 Coercion
co1a Coercion
co1b)
in_co2 :: Coercion
in_co2@(FunCo Role
r2 FunTyFlag
afl2 FunTyFlag
afr2 Coercion
w2 Coercion
co2a Coercion
co2b)
= Bool -> Maybe Coercion -> Maybe Coercion
forall a. HasCallStack => Bool -> a -> a
assert (Role
r1 Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
== Role
r2) (Maybe Coercion -> Maybe Coercion)
-> Maybe Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
Bool -> Maybe Coercion -> Maybe Coercion
forall a. HasCallStack => Bool -> a -> a
assert (FunTyFlag
afr1 FunTyFlag -> FunTyFlag -> Bool
forall a. Eq a => a -> a -> Bool
== FunTyFlag
afl2) (Maybe Coercion -> Maybe Coercion)
-> Maybe Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"PushFun" Coercion
in_co1 Coercion
in_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
Role
-> FunTyFlag
-> FunTyFlag
-> Coercion
-> Coercion
-> Coercion
-> Coercion
mkFunCo2 Role
r1 FunTyFlag
afl1 FunTyFlag
afr2 (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
w1 Coercion
w2)
(HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1a Coercion
co2a)
(HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1b Coercion
co2b)
opt_trans_rule InScopeSet
is in_co1 :: Coercion
in_co1@(AppCo Coercion
co1a Coercion
co1b) in_co2 :: Coercion
in_co2@(AppCo Coercion
co2a Coercion
co2b)
= InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app InScopeSet
is Coercion
in_co1 Coercion
in_co2 Coercion
co1a [Coercion
co1b] Coercion
co2a [Coercion
co2b]
opt_trans_rule InScopeSet
is co1 :: Coercion
co1@(TyConAppCo Role
r TyCon
tc [Coercion]
cos1) Coercion
co2
| Just [Coercion]
cos2 <- TyCon -> Coercion -> Maybe [Coercion]
etaTyConAppCo_maybe TyCon
tc Coercion
co2
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"EtaCompL" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack => Role -> TyCon -> [Coercion] -> Coercion
Role -> TyCon -> [Coercion] -> Coercion
mkTyConAppCo Role
r TyCon
tc (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos1 [Coercion]
cos2)
opt_trans_rule InScopeSet
is Coercion
co1 co2 :: Coercion
co2@(TyConAppCo Role
r TyCon
tc [Coercion]
cos2)
| Just [Coercion]
cos1 <- TyCon -> Coercion -> Maybe [Coercion]
etaTyConAppCo_maybe TyCon
tc Coercion
co1
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"EtaCompR" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack => Role -> TyCon -> [Coercion] -> Coercion
Role -> TyCon -> [Coercion] -> Coercion
mkTyConAppCo Role
r TyCon
tc (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos1 [Coercion]
cos2)
opt_trans_rule InScopeSet
is co1 :: Coercion
co1@(AppCo Coercion
co1a Coercion
co1b) Coercion
co2
| Just (Coercion
co2a,Coercion
co2b) <- Coercion -> Maybe (Coercion, Coercion)
etaAppCo_maybe Coercion
co2
= InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app InScopeSet
is Coercion
co1 Coercion
co2 Coercion
co1a [Coercion
co1b] Coercion
co2a [Coercion
co2b]
opt_trans_rule InScopeSet
is Coercion
co1 co2 :: Coercion
co2@(AppCo Coercion
co2a Coercion
co2b)
| Just (Coercion
co1a,Coercion
co1b) <- Coercion -> Maybe (Coercion, Coercion)
etaAppCo_maybe Coercion
co1
= InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app InScopeSet
is Coercion
co1 Coercion
co2 Coercion
co1a [Coercion
co1b] Coercion
co2a [Coercion
co2b]
opt_trans_rule InScopeSet
is Coercion
co1 Coercion
co2
| Just (CoVar
tv1, ForAllTyFlag
visL1, ForAllTyFlag
_visR1, Coercion
eta1, Coercion
r1) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_ty_maybe Coercion
co1
, Just (CoVar
tv2, ForAllTyFlag
_visL2, ForAllTyFlag
visR2, Coercion
eta2, Coercion
r2) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_ty_maybe Coercion
co2
= CoVar
-> Coercion
-> Coercion
-> CoVar
-> Coercion
-> Coercion
-> ForAllTyFlag
-> ForAllTyFlag
-> Maybe Coercion
push_trans CoVar
tv1 Coercion
eta1 Coercion
r1 CoVar
tv2 Coercion
eta2 Coercion
r2 ForAllTyFlag
visL1 ForAllTyFlag
visR2
| Just (CoVar
tv2, ForAllTyFlag
_visL2, ForAllTyFlag
visR2, Coercion
eta2, Coercion
r2) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_ty_maybe Coercion
co2
, Just (CoVar
tv1, ForAllTyFlag
visL1, ForAllTyFlag
_visR1, Coercion
eta1, Coercion
r1) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_ty_maybe Coercion
co1
= CoVar
-> Coercion
-> Coercion
-> CoVar
-> Coercion
-> Coercion
-> ForAllTyFlag
-> ForAllTyFlag
-> Maybe Coercion
push_trans CoVar
tv1 Coercion
eta1 Coercion
r1 CoVar
tv2 Coercion
eta2 Coercion
r2 ForAllTyFlag
visL1 ForAllTyFlag
visR2
where
push_trans :: CoVar
-> Coercion
-> Coercion
-> CoVar
-> Coercion
-> Coercion
-> ForAllTyFlag
-> ForAllTyFlag
-> Maybe Coercion
push_trans CoVar
tv1 Coercion
eta1 Coercion
r1 CoVar
tv2 Coercion
eta2 Coercion
r2 ForAllTyFlag
visL ForAllTyFlag
visR
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"EtaAllTy_ty" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack =>
CoVar
-> ForAllTyFlag -> ForAllTyFlag -> Coercion -> Coercion -> Coercion
CoVar
-> ForAllTyFlag -> ForAllTyFlag -> Coercion -> Coercion -> Coercion
mkForAllCo CoVar
tv1 ForAllTyFlag
visL ForAllTyFlag
visR (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
eta1 Coercion
eta2) (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is' Coercion
r1 Coercion
r2')
where
is' :: InScopeSet
is' = InScopeSet
is InScopeSet -> CoVar -> InScopeSet
`extendInScopeSet` CoVar
tv1
r2' :: Coercion
r2' = [CoVar] -> [Type] -> Coercion -> Coercion
substCoWithUnchecked [CoVar
tv2] [Type -> Coercion -> Type
mkCastTy (CoVar -> Type
TyVarTy CoVar
tv1) Coercion
eta1] Coercion
r2
opt_trans_rule InScopeSet
is Coercion
co1 Coercion
co2
| Just (CoVar
cv1, ForAllTyFlag
visL1, ForAllTyFlag
_visR1, Coercion
eta1, Coercion
r1) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_co_maybe Coercion
co1
, Just (CoVar
cv2, ForAllTyFlag
_visL2, ForAllTyFlag
visR2, Coercion
eta2, Coercion
r2) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_co_maybe Coercion
co2
= CoVar
-> Coercion
-> Coercion
-> CoVar
-> Coercion
-> Coercion
-> ForAllTyFlag
-> ForAllTyFlag
-> Maybe Coercion
push_trans CoVar
cv1 Coercion
eta1 Coercion
r1 CoVar
cv2 Coercion
eta2 Coercion
r2 ForAllTyFlag
visL1 ForAllTyFlag
visR2
| Just (CoVar
cv2, ForAllTyFlag
_visL2, ForAllTyFlag
visR2, Coercion
eta2, Coercion
r2) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_co_maybe Coercion
co2
, Just (CoVar
cv1, ForAllTyFlag
visL1, ForAllTyFlag
_visR1, Coercion
eta1, Coercion
r1) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_co_maybe Coercion
co1
= CoVar
-> Coercion
-> Coercion
-> CoVar
-> Coercion
-> Coercion
-> ForAllTyFlag
-> ForAllTyFlag
-> Maybe Coercion
push_trans CoVar
cv1 Coercion
eta1 Coercion
r1 CoVar
cv2 Coercion
eta2 Coercion
r2 ForAllTyFlag
visL1 ForAllTyFlag
visR2
where
push_trans :: CoVar
-> Coercion
-> Coercion
-> CoVar
-> Coercion
-> Coercion
-> ForAllTyFlag
-> ForAllTyFlag
-> Maybe Coercion
push_trans CoVar
cv1 Coercion
eta1 Coercion
r1 CoVar
cv2 Coercion
eta2 Coercion
r2 ForAllTyFlag
visL ForAllTyFlag
visR
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"EtaAllTy_co" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack =>
CoVar
-> ForAllTyFlag -> ForAllTyFlag -> Coercion -> Coercion -> Coercion
CoVar
-> ForAllTyFlag -> ForAllTyFlag -> Coercion -> Coercion -> Coercion
mkForAllCo CoVar
cv1 ForAllTyFlag
visL ForAllTyFlag
visR (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
eta1 Coercion
eta2) (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is' Coercion
r1 Coercion
r2')
where
is' :: InScopeSet
is' = InScopeSet
is InScopeSet -> CoVar -> InScopeSet
`extendInScopeSet` CoVar
cv1
role :: Role
role = CoVar -> Role
coVarRole CoVar
cv1
eta1' :: Coercion
eta1' = Role -> Role -> Coercion -> Coercion
downgradeRole Role
role Role
Nominal Coercion
eta1
n1 :: Coercion
n1 = HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo (Int -> Role -> CoSel
SelTyCon Int
2 Role
role) Coercion
eta1'
n2 :: Coercion
n2 = HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo (Int -> Role -> CoSel
SelTyCon Int
3 Role
role) Coercion
eta1'
r2' :: Coercion
r2' = HasDebugCallStack => Subst -> Coercion -> Coercion
Subst -> Coercion -> Coercion
substCo ([CoVar] -> [Coercion] -> Subst
HasDebugCallStack => [CoVar] -> [Coercion] -> Subst
zipCvSubst [CoVar
cv2] [(Coercion -> Coercion
mkSymCo Coercion
n1) HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
`mk_trans_co`
(CoVar -> Coercion
mkCoVarCo CoVar
cv1) HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
`mk_trans_co` Coercion
n2])
Coercion
r2
opt_trans_rule InScopeSet
is Coercion
co1 Coercion
co2
| Just (SwapFlag
sym1, CoAxiomRule
axr1, [Coercion]
cos1) <- Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe Coercion
co1
, Just (SwapFlag
sym2, CoAxiomRule
axr2, [Coercion]
cos2) <- Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe Coercion
co2
, CoAxiomRule
axr1 CoAxiomRule -> CoAxiomRule -> Bool
forall a. Eq a => a -> a -> Bool
== CoAxiomRule
axr2
, SwapFlag
sym1 SwapFlag -> SwapFlag -> Bool
forall a. Eq a => a -> a -> Bool
== SwapFlag -> SwapFlag
flipSwap SwapFlag
sym2
, Just (TyCon
tc, Role
role, CoAxBranch
branch) <- CoAxiomRule -> Maybe (TyCon, Role, CoAxBranch)
coAxiomRuleBranch_maybe CoAxiomRule
axr1
, let qtvs :: [CoVar]
qtvs = CoAxBranch -> [CoVar]
coAxBranchTyVars CoAxBranch
branch [CoVar] -> [CoVar] -> [CoVar]
forall a. [a] -> [a] -> [a]
++ CoAxBranch -> [CoVar]
coAxBranchCoVars CoAxBranch
branch
lhs :: Type
lhs = TyCon -> [Type] -> Type
mkTyConApp TyCon
tc (CoAxBranch -> [Type]
coAxBranchLHS CoAxBranch
branch)
rhs :: Type
rhs = CoAxBranch -> Type
coAxBranchRHS CoAxBranch
branch
pivot_tvs :: TyCoVarSet
pivot_tvs = Type -> TyCoVarSet
exactTyCoVarsOfType (SwapFlag -> Type -> Type -> Type
forall a. SwapFlag -> a -> a -> a
pickSwap SwapFlag
sym2 Type
lhs Type
rhs)
, (CoVar -> Bool) -> [CoVar] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all (CoVar -> TyCoVarSet -> Bool
`elemVarSet` TyCoVarSet
pivot_tvs) [CoVar]
qtvs
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"TrPushAxSym" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
if SwapFlag -> Bool
isSwapped SwapFlag
sym2
then Role -> [CoVar] -> [Coercion] -> Type -> Coercion
liftCoSubstWith Role
role [CoVar]
qtvs (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos1 ((Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map Coercion -> Coercion
mkSymCo [Coercion]
cos2)) Type
lhs
else Role -> [CoVar] -> [Coercion] -> Type -> Coercion
liftCoSubstWith Role
role [CoVar]
qtvs (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is ((Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map Coercion -> Coercion
mkSymCo [Coercion]
cos1) [Coercion]
cos2) Type
rhs
| Just (SwapFlag
sym, CoAxiomRule
axr, [Coercion]
cos1) <- Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe Coercion
co1
, SwapFlag -> Bool
isSwapped SwapFlag
sym
, Just [Coercion]
cos2 <- SwapFlag -> CoAxiomRule -> Coercion -> Maybe [Coercion]
matchNewtypeBranch SwapFlag
sym CoAxiomRule
axr Coercion
co2
, let newAxInst :: Coercion
newAxInst = CoAxiomRule -> [Coercion] -> Coercion
AxiomCo CoAxiomRule
axr (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is ((Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map Coercion -> Coercion
mkSymCo [Coercion]
cos2) [Coercion]
cos1)
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"TrPushSymAxR" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$ Coercion -> Coercion
SymCo Coercion
newAxInst
| Just (SwapFlag
sym, CoAxiomRule
axr, [Coercion]
cos1) <- Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe Coercion
co1
, SwapFlag -> Bool
notSwapped SwapFlag
sym
, Just [Coercion]
cos2 <- SwapFlag -> CoAxiomRule -> Coercion -> Maybe [Coercion]
matchNewtypeBranch SwapFlag
sym CoAxiomRule
axr Coercion
co2
, let newAxInst :: Coercion
newAxInst = CoAxiomRule -> [Coercion] -> Coercion
AxiomCo CoAxiomRule
axr (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos1 [Coercion]
cos2)
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"TrPushAxR" Coercion
co1 Coercion
co2 Coercion
newAxInst
| Just (SwapFlag
sym, CoAxiomRule
axr, [Coercion]
cos2) <- Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe Coercion
co2
, SwapFlag -> Bool
isSwapped SwapFlag
sym
, Just [Coercion]
cos1 <- SwapFlag -> CoAxiomRule -> Coercion -> Maybe [Coercion]
matchNewtypeBranch (SwapFlag -> SwapFlag
flipSwap SwapFlag
sym) CoAxiomRule
axr Coercion
co1
, let newAxInst :: Coercion
newAxInst = CoAxiomRule -> [Coercion] -> Coercion
AxiomCo CoAxiomRule
axr (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos2 ((Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map Coercion -> Coercion
mkSymCo [Coercion]
cos1))
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"TrPushSymAxL" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$ Coercion -> Coercion
SymCo Coercion
newAxInst
| Just (SwapFlag
sym, CoAxiomRule
axr, [Coercion]
cos2) <- Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe Coercion
co2
, SwapFlag -> Bool
notSwapped SwapFlag
sym
, Just [Coercion]
cos1 <- SwapFlag -> CoAxiomRule -> Coercion -> Maybe [Coercion]
matchNewtypeBranch (SwapFlag -> SwapFlag
flipSwap SwapFlag
sym) CoAxiomRule
axr Coercion
co1
, let newAxInst :: Coercion
newAxInst = CoAxiomRule -> [Coercion] -> Coercion
AxiomCo CoAxiomRule
axr (HasDebugCallStack =>
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
InScopeSet -> [Coercion] -> [Coercion] -> [Coercion]
opt_transList InScopeSet
is [Coercion]
cos1 [Coercion]
cos2)
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"TrPushAxL" Coercion
co1 Coercion
co2 Coercion
newAxInst
opt_trans_rule InScopeSet
_ Coercion
co1 Coercion
co2
| let ty1 :: Type
ty1 = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
co1
r :: Role
r = Coercion -> Role
coercionRole Coercion
co1
ty2 :: Type
ty2 = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
co2
, Type
ty1 HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` Type
ty2
= String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
"RedTypeDirRefl" Coercion
co1 Coercion
co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
Role -> Type -> Coercion
mkReflCo Role
r Type
ty2
opt_trans_rule InScopeSet
_ Coercion
_ Coercion
_ = Maybe Coercion
forall a. Maybe a
Nothing
opt_trans_rule_app :: InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app :: InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app InScopeSet
is Coercion
orig_co1 Coercion
orig_co2 Coercion
co1a [Coercion]
co1bs Coercion
co2a [Coercion]
co2bs
| AppCo Coercion
co1aa Coercion
co1ab <- Coercion
co1a
, Just (Coercion
co2aa, Coercion
co2ab) <- Coercion -> Maybe (Coercion, Coercion)
etaAppCo_maybe Coercion
co2a
= InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app InScopeSet
is Coercion
orig_co1 Coercion
orig_co2 Coercion
co1aa (Coercion
co1abCoercion -> [Coercion] -> [Coercion]
forall a. a -> [a] -> [a]
:[Coercion]
co1bs) Coercion
co2aa (Coercion
co2abCoercion -> [Coercion] -> [Coercion]
forall a. a -> [a] -> [a]
:[Coercion]
co2bs)
| AppCo Coercion
co2aa Coercion
co2ab <- Coercion
co2a
, Just (Coercion
co1aa, Coercion
co1ab) <- Coercion -> Maybe (Coercion, Coercion)
etaAppCo_maybe Coercion
co1a
= InScopeSet
-> Coercion
-> Coercion
-> Coercion
-> [Coercion]
-> Coercion
-> [Coercion]
-> Maybe Coercion
opt_trans_rule_app InScopeSet
is Coercion
orig_co1 Coercion
orig_co2 Coercion
co1aa (Coercion
co1abCoercion -> [Coercion] -> [Coercion]
forall a. a -> [a] -> [a]
:[Coercion]
co1bs) Coercion
co2aa (Coercion
co2abCoercion -> [Coercion] -> [Coercion]
forall a. a -> [a] -> [a]
:[Coercion]
co2bs)
| Bool
otherwise
= Bool -> Maybe Coercion -> Maybe Coercion
forall a. HasCallStack => Bool -> a -> a
assert ([Coercion]
co1bs [Coercion] -> [Coercion] -> Bool
forall a b. [a] -> [b] -> Bool
`equalLength` [Coercion]
co2bs) (Maybe Coercion -> Maybe Coercion)
-> Maybe Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule (String
"EtaApps:" String -> String -> String
forall a. [a] -> [a] -> [a]
++ Int -> String
forall a. Show a => a -> String
show ([Coercion] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [Coercion]
co1bs)) Coercion
orig_co1 Coercion
orig_co2 (Coercion -> Maybe Coercion) -> Coercion -> Maybe Coercion
forall a b. (a -> b) -> a -> b
$
let rt1a :: Type
rt1a = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
co1a
lt2a :: Type
lt2a = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
co2a
rt2a :: Role
rt2a = Coercion -> Role
coercionRole Coercion
co2a
rt1bs :: [Type]
rt1bs = (Coercion -> Type) -> [Coercion] -> [Type]
forall a b. (a -> b) -> [a] -> [b]
map HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind [Coercion]
co1bs
lt2bs :: [Type]
lt2bs = (Coercion -> Type) -> [Coercion] -> [Type]
forall a b. (a -> b) -> [a] -> [b]
map HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind [Coercion]
co2bs
rt2bs :: [Role]
rt2bs = (Coercion -> Role) -> [Coercion] -> [Role]
forall a b. (a -> b) -> [a] -> [b]
map Coercion -> Role
coercionRole [Coercion]
co2bs
kcoa :: Coercion
kcoa = Coercion -> Coercion
mkKindCo (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$ HasDebugCallStack => Type -> Type -> Coercion
Type -> Type -> Coercion
buildCoercion Type
lt2a Type
rt1a
kcobs :: [Coercion]
kcobs = (Coercion -> Coercion) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> [a] -> [b]
map Coercion -> Coercion
mkKindCo ([Coercion] -> [Coercion]) -> [Coercion] -> [Coercion]
forall a b. (a -> b) -> a -> b
$ (Type -> Type -> Coercion) -> [Type] -> [Type] -> [Coercion]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith HasDebugCallStack => Type -> Type -> Coercion
Type -> Type -> Coercion
buildCoercion [Type]
lt2bs [Type]
rt1bs
co2a' :: Coercion
co2a' = Role -> Type -> Coercion -> Coercion -> Coercion
mkCoherenceLeftCo Role
rt2a Type
lt2a Coercion
kcoa Coercion
co2a
co2bs' :: [Coercion]
co2bs' = (Role -> Type -> Coercion -> Coercion)
-> [Role] -> [Type] -> [Coercion] -> [Coercion]
forall a b c d. (a -> b -> c -> d) -> [a] -> [b] -> [c] -> [d]
zipWith3 Role -> Type -> Coercion -> Coercion
mkGReflLeftCo [Role]
rt2bs [Type]
lt2bs [Coercion]
kcobs
co2bs'' :: [Coercion]
co2bs'' = (Coercion -> Coercion -> Coercion)
-> [Coercion] -> [Coercion] -> [Coercion]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
mk_trans_co [Coercion]
co2bs' [Coercion]
co2bs
in
Coercion -> [Coercion] -> Coercion
mkAppCos (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is Coercion
co1a Coercion
co2a')
((Coercion -> Coercion -> Coercion)
-> [Coercion] -> [Coercion] -> [Coercion]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith (HasDebugCallStack => InScopeSet -> Coercion -> Coercion -> Coercion
InScopeSet -> Coercion -> Coercion -> Coercion
opt_trans InScopeSet
is) [Coercion]
co1bs [Coercion]
co2bs'')
fireTransRule :: String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule :: String -> Coercion -> Coercion -> Coercion -> Maybe Coercion
fireTransRule String
_rule Coercion
_co1 Coercion
_co2 Coercion
res
= Coercion -> Maybe Coercion
forall a. a -> Maybe a
Just Coercion
res
swapSym :: SwapFlag -> (a,a) -> (a,a)
swapSym :: forall a. SwapFlag -> (a, a) -> (a, a)
swapSym SwapFlag
IsSwapped (a
x,a
y) = (a
y,a
x)
swapSym SwapFlag
NotSwapped (a
x,a
y) = (a
x,a
y)
wrapSym :: SwapFlag -> Coercion -> Coercion
wrapSym :: SwapFlag -> Coercion -> Coercion
wrapSym SwapFlag
IsSwapped Coercion
co = Coercion -> Coercion
mkSymCo Coercion
co
wrapSym SwapFlag
NotSwapped Coercion
co = Coercion
co
wrapRole :: ReprFlag
-> Role
-> Coercion -> Coercion
wrapRole :: Bool -> Role -> Coercion -> Coercion
wrapRole Bool
False Role
_ = Coercion -> Coercion
forall a. a -> a
id
wrapRole Bool
True Role
current = Role -> Role -> Coercion -> Coercion
downgradeRole Role
Representational Role
current
chooseRole :: ReprFlag
-> Role
-> Role
chooseRole :: Bool -> Role -> Role
chooseRole Bool
True Role
_ = Role
Representational
chooseRole Bool
_ Role
r = Role
r
isAxiomCo_maybe :: Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe :: Coercion -> Maybe (SwapFlag, CoAxiomRule, [Coercion])
isAxiomCo_maybe (SymCo (AxiomCo CoAxiomRule
ax [Coercion]
cos)) = (SwapFlag, CoAxiomRule, [Coercion])
-> Maybe (SwapFlag, CoAxiomRule, [Coercion])
forall a. a -> Maybe a
Just (SwapFlag
IsSwapped, CoAxiomRule
ax, [Coercion]
cos)
isAxiomCo_maybe (AxiomCo CoAxiomRule
ax [Coercion]
cos) = (SwapFlag, CoAxiomRule, [Coercion])
-> Maybe (SwapFlag, CoAxiomRule, [Coercion])
forall a. a -> Maybe a
Just (SwapFlag
NotSwapped, CoAxiomRule
ax, [Coercion]
cos)
isAxiomCo_maybe Coercion
_ = Maybe (SwapFlag, CoAxiomRule, [Coercion])
forall a. Maybe a
Nothing
matchNewtypeBranch :: SwapFlag
-> CoAxiomRule
-> Coercion -> Maybe [Coercion]
matchNewtypeBranch :: SwapFlag -> CoAxiomRule -> Coercion -> Maybe [Coercion]
matchNewtypeBranch SwapFlag
sym CoAxiomRule
axr Coercion
co
| Just (TyCon
tc,CoAxBranch
branch) <- CoAxiomRule -> Maybe (TyCon, CoAxBranch)
isNewtypeAxiomRule_maybe CoAxiomRule
axr
, CoAxBranch { cab_tvs :: CoAxBranch -> [CoVar]
cab_tvs = [CoVar]
qtvs
, cab_cvs :: CoAxBranch -> [CoVar]
cab_cvs = []
, cab_roles :: CoAxBranch -> [Role]
cab_roles = [Role]
roles
, cab_lhs :: CoAxBranch -> [Type]
cab_lhs = [Type]
lhs
, cab_rhs :: CoAxBranch -> Type
cab_rhs = Type
rhs } <- CoAxBranch
branch
, Just LiftingContext
subst <- TyCoVarSet -> Type -> Coercion -> Maybe LiftingContext
liftCoMatch ([CoVar] -> TyCoVarSet
mkVarSet [CoVar]
qtvs)
(SwapFlag -> Type -> Type -> Type
forall a. SwapFlag -> a -> a -> a
pickSwap SwapFlag
sym Type
rhs (TyCon -> [Type] -> Type
mkTyConApp TyCon
tc [Type]
lhs))
Coercion
co
, (CoVar -> Bool) -> [CoVar] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all (CoVar -> LiftingContext -> Bool
`isMappedByLC` LiftingContext
subst) [CoVar]
qtvs
= (Role -> CoVar -> Maybe Coercion)
-> [Role] -> [CoVar] -> Maybe [Coercion]
forall (m :: * -> *) a b c.
Applicative m =>
(a -> b -> m c) -> [a] -> [b] -> m [c]
zipWithM (LiftingContext -> Role -> CoVar -> Maybe Coercion
liftCoSubstTyVar LiftingContext
subst) [Role]
roles [CoVar]
qtvs
| Bool
otherwise
= Maybe [Coercion]
forall a. Maybe a
Nothing
compatible_co :: Coercion -> Coercion -> Bool
compatible_co :: Coercion -> Coercion -> Bool
compatible_co Coercion
co1 Coercion
co2
= Type
x1 HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` Type
x2
where
x1 :: Type
x1 = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
co1
x2 :: Type
x2 = HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
co2
etaForAllCo_ty_maybe :: Coercion -> Maybe (TyVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_ty_maybe :: Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_ty_maybe Coercion
co
| Just (CoVar
tv, ForAllTyFlag
visL, ForAllTyFlag
visR, Coercion
kind_co, Coercion
r) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_ty_maybe Coercion
co
= (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
forall a. a -> Maybe a
Just (CoVar
tv, ForAllTyFlag
visL, ForAllTyFlag
visR, Coercion
kind_co, Coercion
r)
| (Pair Type
ty1 Type
ty2, Role
role) <- Coercion -> (Pair Type, Role)
coercionKindRole Coercion
co
, Just (Bndr CoVar
tv1 ForAllTyFlag
vis1, Type
_) <- Type -> Maybe (ForAllTyBinder, Type)
splitForAllForAllTyBinder_maybe Type
ty1
, CoVar -> Bool
isTyVar CoVar
tv1
, Just (Bndr CoVar
tv2 ForAllTyFlag
vis2, Type
_) <- Type -> Maybe (ForAllTyBinder, Type)
splitForAllForAllTyBinder_maybe Type
ty2
, CoVar -> Bool
isTyVar CoVar
tv2
, (Role
role Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
/= Role
Nominal) Bool -> Bool -> Bool
|| (ForAllTyFlag
vis1 ForAllTyFlag -> ForAllTyFlag -> Bool
`eqForAllVis` ForAllTyFlag
vis2)
, let kind_co :: Coercion
kind_co = HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo CoSel
SelForAll Coercion
co
= (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
forall a. a -> Maybe a
Just ( CoVar
tv1, ForAllTyFlag
vis1, ForAllTyFlag
vis2, Coercion
kind_co
, Coercion -> Coercion -> Coercion
mkInstCo Coercion
co (Role -> Type -> Coercion -> Coercion
mk_grefl_right_co Role
Nominal (CoVar -> Type
TyVarTy CoVar
tv1) Coercion
kind_co))
| Bool
otherwise
= Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
forall a. Maybe a
Nothing
etaForAllCo_co_maybe :: Coercion -> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_co_maybe :: Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
etaForAllCo_co_maybe Coercion
co
| Just (CoVar
cv, ForAllTyFlag
visL, ForAllTyFlag
visR, Coercion
kind_co, Coercion
r) <- Coercion
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
splitForAllCo_co_maybe Coercion
co
= (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
forall a. a -> Maybe a
Just (CoVar
cv, ForAllTyFlag
visL, ForAllTyFlag
visR, Coercion
kind_co, Coercion
r)
| (Pair Type
ty1 Type
ty2, Role
role) <- Coercion -> (Pair Type, Role)
coercionKindRole Coercion
co
, Just (Bndr CoVar
cv1 ForAllTyFlag
vis1, Type
_) <- Type -> Maybe (ForAllTyBinder, Type)
splitForAllForAllTyBinder_maybe Type
ty1
, CoVar -> Bool
isCoVar CoVar
cv1
, Just (Bndr CoVar
cv2 ForAllTyFlag
vis2, Type
_) <- Type -> Maybe (ForAllTyBinder, Type)
splitForAllForAllTyBinder_maybe Type
ty2
, CoVar -> Bool
isCoVar CoVar
cv2
, (Role
role Role -> Role -> Bool
forall a. Eq a => a -> a -> Bool
/= Role
Nominal)
= let kind_co :: Coercion
kind_co = HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo CoSel
SelForAll Coercion
co
r :: Role
r = CoVar -> Role
coVarRole CoVar
cv1
l_co :: Coercion
l_co = CoVar -> Coercion
mkCoVarCo CoVar
cv1
kind_co' :: Coercion
kind_co' = Role -> Role -> Coercion -> Coercion
downgradeRole Role
r Role
Nominal Coercion
kind_co
r_co :: Coercion
r_co = Coercion -> Coercion
mkSymCo (HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo (Int -> Role -> CoSel
SelTyCon Int
2 Role
r) Coercion
kind_co')
HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
`mk_trans_co` Coercion
l_co
HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
`mk_trans_co` HasDebugCallStack => CoSel -> Coercion -> Coercion
CoSel -> Coercion -> Coercion
mkSelCo (Int -> Role -> CoSel
SelTyCon Int
3 Role
r) Coercion
kind_co'
in (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
-> Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
forall a. a -> Maybe a
Just ( CoVar
cv1, ForAllTyFlag
vis1, ForAllTyFlag
vis2, Coercion
kind_co
, Coercion -> Coercion -> Coercion
mkInstCo Coercion
co (Role -> Coercion -> Coercion -> Coercion -> Coercion
mkProofIrrelCo Role
Nominal Coercion
kind_co Coercion
l_co Coercion
r_co))
| Bool
otherwise
= Maybe (CoVar, ForAllTyFlag, ForAllTyFlag, Coercion, Coercion)
forall a. Maybe a
Nothing
etaAppCo_maybe :: Coercion -> Maybe (Coercion,Coercion)
etaAppCo_maybe :: Coercion -> Maybe (Coercion, Coercion)
etaAppCo_maybe Coercion
co
| Just (Coercion
co1,Coercion
co2) <- Coercion -> Maybe (Coercion, Coercion)
splitAppCo_maybe Coercion
co
= (Coercion, Coercion) -> Maybe (Coercion, Coercion)
forall a. a -> Maybe a
Just (Coercion
co1,Coercion
co2)
| (Pair Type
ty1 Type
ty2, Role
Nominal) <- Coercion -> (Pair Type, Role)
coercionKindRole Coercion
co
, Just (Type
_,Type
t1) <- Type -> Maybe (Type, Type)
splitAppTy_maybe Type
ty1
, Just (Type
_,Type
t2) <- Type -> Maybe (Type, Type)
splitAppTy_maybe Type
ty2
, let isco1 :: Bool
isco1 = Type -> Bool
isCoercionTy Type
t1
, let isco2 :: Bool
isco2 = Type -> Bool
isCoercionTy Type
t2
, Bool
isco1 Bool -> Bool -> Bool
forall a. Eq a => a -> a -> Bool
== Bool
isco2
= (Coercion, Coercion) -> Maybe (Coercion, Coercion)
forall a. a -> Maybe a
Just (LeftOrRight -> Coercion -> Coercion
LRCo LeftOrRight
CLeft Coercion
co, LeftOrRight -> Coercion -> Coercion
LRCo LeftOrRight
CRight Coercion
co)
| Bool
otherwise
= Maybe (Coercion, Coercion)
forall a. Maybe a
Nothing
etaTyConAppCo_maybe :: TyCon -> Coercion -> Maybe [Coercion]
etaTyConAppCo_maybe :: TyCon -> Coercion -> Maybe [Coercion]
etaTyConAppCo_maybe TyCon
tc (TyConAppCo Role
_ TyCon
tc2 [Coercion]
cos2)
= Bool -> Maybe [Coercion] -> Maybe [Coercion]
forall a. HasCallStack => Bool -> a -> a
assert (TyCon
tc TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2) (Maybe [Coercion] -> Maybe [Coercion])
-> Maybe [Coercion] -> Maybe [Coercion]
forall a b. (a -> b) -> a -> b
$ [Coercion] -> Maybe [Coercion]
forall a. a -> Maybe a
Just [Coercion]
cos2
etaTyConAppCo_maybe TyCon
tc Coercion
co
| Bool -> Bool
not (TyCon -> Bool
tyConMustBeSaturated TyCon
tc)
, (Pair Type
ty1 Type
ty2, Role
r) <- Coercion -> (Pair Type, Role)
coercionKindRole Coercion
co
, Just (TyCon
tc1, [Type]
tys1) <- HasDebugCallStack => Type -> Maybe (TyCon, [Type])
Type -> Maybe (TyCon, [Type])
splitTyConApp_maybe Type
ty1
, Just (TyCon
tc2, [Type]
tys2) <- HasDebugCallStack => Type -> Maybe (TyCon, [Type])
Type -> Maybe (TyCon, [Type])
splitTyConApp_maybe Type
ty2
, TyCon
tc1 TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc2
, TyCon -> Role -> Bool
isInjectiveTyCon TyCon
tc Role
r
, let n :: Int
n = [Type] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [Type]
tys1
, [Type]
tys2 [Type] -> Int -> Bool
forall a. [a] -> Int -> Bool
`lengthIs` Int
n
= Bool -> Maybe [Coercion] -> Maybe [Coercion]
forall a. HasCallStack => Bool -> a -> a
assert (TyCon
tc TyCon -> TyCon -> Bool
forall a. Eq a => a -> a -> Bool
== TyCon
tc1) (Maybe [Coercion] -> Maybe [Coercion])
-> Maybe [Coercion] -> Maybe [Coercion]
forall a b. (a -> b) -> a -> b
$
[Coercion] -> Maybe [Coercion]
forall a. a -> Maybe a
Just (Int -> Coercion -> Infinite Role -> [Coercion]
decomposeCo Int
n Coercion
co (Role -> TyCon -> Infinite Role
tyConRolesX Role
r TyCon
tc1))
| Bool
otherwise
= Maybe [Coercion]
forall a. Maybe a
Nothing
optForAllCoBndr :: LiftingContext -> SwapFlag
-> TyCoVar -> Coercion -> (LiftingContext, TyCoVar, Coercion)
optForAllCoBndr :: LiftingContext
-> SwapFlag
-> CoVar
-> Coercion
-> (LiftingContext, CoVar, Coercion)
optForAllCoBndr LiftingContext
env SwapFlag
sym
= SwapFlag
-> (Coercion -> Coercion)
-> LiftingContext
-> CoVar
-> Coercion
-> (LiftingContext, CoVar, Coercion)
substForAllCoBndrUsingLC SwapFlag
sym (LiftingContext -> SwapFlag -> Bool -> Role -> Coercion -> Coercion
opt_co4 LiftingContext
env SwapFlag
sym Bool
False Role
Nominal) LiftingContext
env
mk_trans_co :: HasDebugCallStack => Coercion -> Coercion -> Coercion
mk_trans_co :: HasDebugCallStack => Coercion -> Coercion -> Coercion
mk_trans_co Coercion
co1 Coercion
co2
= Bool -> SDoc -> Coercion -> Coercion
forall a. HasCallStack => Bool -> SDoc -> a -> a
assertPpr (HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionRKind Coercion
co1 HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
co2)
([SDoc] -> SDoc
forall doc. IsDoc doc => [doc] -> doc
vcat [ String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"co1" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
co1
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"co2" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
co2
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"co1 kind" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Pair Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (HasDebugCallStack => Coercion -> Pair Type
Coercion -> Pair Type
coercionKind Coercion
co1)
, String -> SDoc
forall doc. IsLine doc => String -> doc
text String
"co2 kind" SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Pair Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (HasDebugCallStack => Coercion -> Pair Type
Coercion -> Pair Type
coercionKind Coercion
co2)
, SDoc
HasCallStack => SDoc
callStackDoc ]) (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack => Coercion -> Coercion -> Coercion
Coercion -> Coercion -> Coercion
mkTransCo Coercion
co1 Coercion
co2
mk_coherence_right_co :: HasDebugCallStack => Role -> Type -> CoercionN -> Coercion -> Coercion
mk_coherence_right_co :: HasDebugCallStack =>
Role -> Type -> Coercion -> Coercion -> Coercion
mk_coherence_right_co Role
r Type
ty Coercion
co Coercion
co2
= Type -> Coercion -> Coercion -> Coercion
forall r. HasDebugCallStack => Type -> Coercion -> r -> r
assertGRefl Type
ty Coercion
co (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
HasDebugCallStack =>
Role -> Type -> Coercion -> Coercion -> Coercion
Role -> Type -> Coercion -> Coercion -> Coercion
mkCoherenceRightCo Role
r Type
ty Coercion
co Coercion
co2
assertGRefl :: HasDebugCallStack => Type -> Coercion -> r -> r
assertGRefl :: forall r. HasDebugCallStack => Type -> Coercion -> r -> r
assertGRefl Type
ty Coercion
co r
res
= Bool -> SDoc -> r -> r
forall a. HasCallStack => Bool -> SDoc -> a -> a
assertPpr (HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
ty HasCallStack => Type -> Type -> Bool
Type -> Type -> Bool
`eqType` HasDebugCallStack => Coercion -> Type
Coercion -> Type
coercionLKind Coercion
co)
([SDoc] -> SDoc
forall doc. IsDoc doc => [doc] -> doc
vcat [ String -> Type -> SDoc
pp_ty String
"ty" Type
ty
, String -> Coercion -> SDoc
pp_co String
"co" Coercion
co
, SDoc
HasCallStack => SDoc
callStackDoc ]) (r -> r) -> r -> r
forall a b. (a -> b) -> a -> b
$
r
res
mk_grefl_right_co :: Role -> Type -> CoercionN -> Coercion
mk_grefl_right_co :: Role -> Type -> Coercion -> Coercion
mk_grefl_right_co Role
r Type
ty Coercion
co
= Type -> Coercion -> Coercion -> Coercion
forall r. HasDebugCallStack => Type -> Coercion -> r -> r
assertGRefl Type
ty Coercion
co (Coercion -> Coercion) -> Coercion -> Coercion
forall a b. (a -> b) -> a -> b
$
Role -> Type -> Coercion -> Coercion
mkGReflRightCo Role
r Type
ty Coercion
co
pp_co :: String -> Coercion -> SDoc
pp_co :: String -> Coercion -> SDoc
pp_co String
s Coercion
co = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
s SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> SDoc -> Int -> SDoc -> SDoc
hang (Coercion -> SDoc
forall a. Outputable a => a -> SDoc
ppr Coercion
co) Int
2 (SDoc
dcolon SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Pair Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (HasDebugCallStack => Coercion -> Pair Type
Coercion -> Pair Type
coercionKind Coercion
co))
pp_ty :: String -> Type -> SDoc
pp_ty :: String -> Type -> SDoc
pp_ty String
s Type
ty = String -> SDoc
forall doc. IsLine doc => String -> doc
text String
s SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> SDoc -> Int -> SDoc -> SDoc
hang (Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr Type
ty) Int
2 (SDoc
dcolon SDoc -> SDoc -> SDoc
forall doc. IsLine doc => doc -> doc -> doc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (HasDebugCallStack => Type -> Type
Type -> Type
typeKind Type
ty))