-- | Erase contained let-bindings that have no uses. -- -- Contained bindings are ones that do not perform effects that are -- visible to anything in the calling context. This includes allocation -- and read effects, but not writes or any globally visible effects. -- module DDC.Core.Transform.Prune ( PruneInfo (..) , pruneModule , pruneX) where import DDC.Core.Analysis.Usage import DDC.Core.Simplifier.Base import DDC.Core.Transform.Reannotate import DDC.Core.Transform.TransformUpX import DDC.Core.Fragment import DDC.Core.Check import DDC.Core.Module import DDC.Core.Exp import DDC.Type.Env import DDC.Base.Pretty import Data.Typeable import Control.Monad.Writer (Writer, runWriter, tell) import Data.Monoid (Monoid, mempty, mappend) import qualified Data.Map as Map import qualified Data.Set as Set import qualified DDC.Type.Env as Env import qualified DDC.Core.Collect as C import qualified DDC.Core.Transform.SubstituteXX as S import qualified DDC.Core.Transform.Trim as Trim import qualified DDC.Type.Compounds as T import qualified DDC.Type.Sum as TS import qualified DDC.Type.Transform.Crush as T ------------------------------------------------------------------------------- -- | A summary of what the prune transform did. data PruneInfo = PruneInfo { -- | How many let-bindings we erased. infoBindingsErased :: Int } deriving Typeable instance Pretty PruneInfo where ppr (PruneInfo remo) = text "Prune:" <$> indent 4 (vcat [ text "Removed: " <> int remo]) instance Monoid PruneInfo where mempty = PruneInfo 0 mappend (PruneInfo r1) (PruneInfo r2) = PruneInfo (r1 + r2) ------------------------------------------------------------------------------- -- | Erase pure let-bindings in a module that have no uses. pruneModule :: (Show a, Show n, Ord n, Pretty n) => Profile n -- ^ Profile of the language we're in -> Module a n -> Module a n pruneModule profile mm -- If the language fragment has untracked effects then we can't do -- the prune transform because we risk dropping statements with global -- effects. | not $ featuresTrackedEffects $ profileFeatures profile = mm | otherwise = mm { moduleBody = result $ pruneX profile (moduleKindEnv mm) (moduleTypeEnv mm) $ moduleBody mm } -- | Erase pure let-bindings in an expression that have no uses. pruneX :: (Show a, Show n, Ord n, Pretty n) => Profile n -- ^ Profile of the language we're in -> KindEnv n -- ^ Kind environment -> TypeEnv n -- ^ Type environment -> Exp a n -> TransformResult (Exp a n) pruneX profile kenv tenv xx = {-# SCC pruneX #-} let (xx', info) = transformTypeUsage profile kenv tenv (transformUpMX pruneTrans kenv tenv) xx progress (PruneInfo r) = r > 0 in TransformResult { result = xx' , resultAgain = progress info , resultProgress = progress info , resultInfo = TransformInfo info } -- The prune transform proper needs to have every expression annotated -- with its type an effect, as well the variable usage map. -- -- We generate these annotations here then pass the result off to -- deadCodeTrans to actually erase dead bindings. -- transformTypeUsage profile kenv tenv trans xx = case fst $ checkExp (configOfProfile profile) kenv tenv xx Recon of Right (xx1, _, _,_) -> let xx2 = usageX xx1 (x', info) = runWriter (trans xx2) x'' = reannotate (\(_, AnTEC { annotTail = a }) -> a) x' in (x'', info) Left _ -> error $ renderIndent $ vcat [ text "ddc-coe-simpl.Prune: core type error" ] ------------------------------------------------------------------------------- -- | Annotations used by the dead-code trasnform. type Annot a n = (UsedMap n, AnTEC a n) -- | Apply the dead-code transform to an annotated expression. pruneTrans :: (Show a, Show n, Ord n, Pretty n) => KindEnv n -> TypeEnv n -> Exp (Annot a n) n -> Writer PruneInfo (Exp (Annot a n) n) pruneTrans _ _ xx = case xx of XLet a@(usedMap, antec) (LLet b x1) x2 | isUnusedBind b usedMap , isContainedEffect $ annotEffect antec -> do -- We still need to substitute value into casts let x2' = transformUpX' Trim.trimX $ S.substituteXX b x1 x2 -- Record that we've erased a binding. tell mempty {infoBindingsErased = 1} -- return $ XCast a (weakEff antec) $ XCast a (weakClo a x1) $ x2' _ -> return xx where weakEff antec = CastWeakenEffect $ T.crushEffect $ annotEffect antec weakClo a x1 = CastWeakenClosure $ Trim.trimClosures a ( (map ((XType a) . TVar) $ Set.toList $ C.freeT Env.empty x1) ++ (map (XVar a) $ Set.toList $ C.freeX Env.empty x1)) -- | Check whether this binder has no uses, -- not including weakclo casts, beause we'll substitute the bound -- expression directly into those. isUnusedBind :: Ord n => Bind n -> UsedMap n -> Bool isUnusedBind bb (UsedMap um) = case bb of BName n _ -> case Map.lookup n um of Just useds -> filterUsedInCasts useds == [] Nothing -> True BNone _ -> True _ -> False filterUsedInCasts :: [Used] -> [Used] filterUsedInCasts = filter notCast where notCast UsedInCast = False notCast _ = True -- | A contained effect is one that is not visible to anything else -- in the context. This is allocation and read effects, which are -- not visible from outside the computation performing the effect. isContainedEffect :: Ord n => Effect n -> Bool isContainedEffect eff = all contained $ map T.takeTApps $ sumList $ T.crushEffect eff where contained (c : _args) = case c of TCon (TyConSpec TcConAlloc) -> True TCon (TyConSpec TcConDeepAlloc) -> True TCon (TyConSpec TcConRead) -> True TCon (TyConSpec TcConHeadRead) -> True TCon (TyConSpec TcConDeepRead) -> True _ -> False contained [] = False sumList (TSum ts) = TS.toList ts sumList tt = [tt]