module Language.Haskell.Liquid.Transforms.RefSplit (
splitXRelatedRefs
) where
import Prelude hiding (error)
import Data.List (partition)
import Text.PrettyPrint.HughesPJ
import Language.Haskell.Liquid.Types
import Language.Haskell.Liquid.Types.PrettyPrint ()
import Language.Fixpoint.Types hiding (Predicate)
import Language.Fixpoint.Misc
splitXRelatedRefs :: Symbol -> SpecType -> (SpecType, SpecType)
splitXRelatedRefs x t = splitRType x t
splitRType :: Symbol
-> RType c tv (UReft Reft)
-> (RType c tv (UReft Reft), RType c tv (UReft Reft))
splitRType f (RVar a r) = (RVar a r1, RVar a r2)
where
(r1, r2) = splitRef f r
splitRType f (RFun x tx t r) = (RFun x tx1 t1 r1, RFun x tx2 t2 r2)
where
(tx1, tx2) = splitRType f tx
(t1, t2) = splitRType f t
(r1, r2) = splitRef f r
splitRType f (RAllT v t) = (RAllT v t1, RAllT v t2)
where
(t1, t2) = splitRType f t
splitRType f (RAllP p t) = (RAllP p t1, RAllP p t2)
where
(t1, t2) = splitRType f t
splitRType f (RAllS s t) = (RAllS s t1, RAllS s t2)
where
(t1, t2) = splitRType f t
splitRType f (RApp c ts rs r) = (RApp c ts1 rs1 r1, RApp c ts2 rs2 r2)
where
(ts1, ts2) = unzip (splitRType f <$> ts)
(rs1, rs2) = unzip (splitUReft f <$> rs)
(r1, r2) = splitRef f r
splitRType f (RAllE x tx t) = (RAllE x tx1 t1, RAllE x tx2 t2)
where
(tx1, tx2) = splitRType f tx
(t1, t2) = splitRType f t
splitRType f (REx x tx t) = (REx x tx1 t1, REx x tx2 t2)
where
(tx1, tx2) = splitRType f tx
(t1, t2) = splitRType f t
splitRType _ (RExprArg e) = (RExprArg e, RExprArg e)
splitRType f (RAppTy tx t r) = (RAppTy tx1 t1 r1, RAppTy tx2 t2 r2)
where
(tx1, tx2) = splitRType f tx
(t1, t2) = splitRType f t
(r1, r2) = splitRef f r
splitRType f (RRTy xs r o t) = (RRTy xs1 r1 o t1, RRTy xs2 r2 o t2)
where
(xs1, xs2) = unzip (go <$> xs)
(r1, r2) = splitRef f r
(t1, t2) = splitRType f t
go (x, t) = let (t1, t2) = splitRType f t in ((x,t1), (x, t2))
splitRType f (RHole r) = (RHole r1, RHole r2)
where
(r1, r2) = splitRef f r
splitUReft :: Symbol -> RTProp c tv (UReft Reft) -> (RTProp c tv (UReft Reft), RTProp c tv (UReft Reft))
splitUReft x (RProp xs (RHole r)) = (RProp xs (RHole r1), RProp xs (RHole r2))
where
(r1, r2) = splitRef x r
splitUReft x (RProp xs t) = (RProp xs t1, RProp xs t2)
where
(t1, t2) = splitRType x t
splitRef :: Symbol -> UReft Reft -> (UReft Reft, UReft Reft)
splitRef f (MkUReft r p s) = (MkUReft r1 p1 s, MkUReft r2 p2 s)
where
(r1, r2) = splitReft f r
(p1, p2) = splitPred f p
splitReft :: Symbol -> Reft -> (Reft, Reft)
splitReft f (Reft (v, xs)) = (Reft (v, pAnd xs1), Reft (v, pAnd xs2))
where
(xs1, xs2) = partition (isFree f) (unPAnd xs)
unPAnd (PAnd ps) = concatMap unPAnd ps
unPAnd p = [p]
splitPred :: Symbol -> Predicate -> (Predicate, Predicate)
splitPred f (Pr ps) = (Pr ps1, Pr ps2)
where
(ps1, ps2) = partition g ps
g p = any (isFree f) (thd3 <$> pargs p)
class IsFree a where
isFree :: Symbol -> a -> Bool
instance (Subable x) => (IsFree x) where
isFree x p = x `elem` syms p
instance Show (UReft Reft) where
show = render . pprint