diff options
Diffstat (limited to 'testsuite/tests/plugins/simple-plugin/Simple/Plugin.hs')
-rw-r--r-- | testsuite/tests/plugins/simple-plugin/Simple/Plugin.hs | 82 |
1 files changed, 82 insertions, 0 deletions
diff --git a/testsuite/tests/plugins/simple-plugin/Simple/Plugin.hs b/testsuite/tests/plugins/simple-plugin/Simple/Plugin.hs new file mode 100644 index 0000000000..d362518e4f --- /dev/null +++ b/testsuite/tests/plugins/simple-plugin/Simple/Plugin.hs @@ -0,0 +1,82 @@ +{-# LANGUAGE TemplateHaskell #-} + +module Simple.Plugin(plugin) where + +import UniqFM +import GhcPlugins +import qualified ErrUtils + +-- For annotation tests +import Simple.DataStructures + +import Control.Monad +import Data.Monoid +import Data.Dynamic +import qualified Language.Haskell.TH as TH + +plugin :: Plugin +plugin = defaultPlugin { + installCoreToDos = install + } + +install :: [CommandLineOption] -> [CoreToDo] -> CoreM [CoreToDo] +install options todos = do + putMsgS $ "Simple Plugin Passes Queried" + putMsgS $ "Got options: " ++ unwords options + + -- Create some actual passes to continue the test. + return $ CoreDoPluginPass "Main pass" mainPass + : todos + +findNameBinds :: String -> [CoreBind] -> First Name +findNameBinds target = mconcat . map (findNameBind target) + +findNameBind :: String -> CoreBind -> First Name +findNameBind target (NonRec b e) = findNameBndr target b +findNameBind target (Rec bes) = mconcat (map (findNameBndr target . fst) bes) + +findNameBndr :: String -> CoreBndr -> First Name +findNameBndr target b + = if getOccString (varName b) == target + then First (Just (varName b)) + else First Nothing + + +mainPass :: ModGuts -> CoreM ModGuts +mainPass guts = do + putMsgS "Simple Plugin Pass Run" + anns <- getAnnotations deserializeWithData guts + bindsOnlyPass (mapM (changeBind anns Nothing)) guts + +changeBind :: UniqFM [ReplaceWith] -> Maybe String -> CoreBind -> CoreM CoreBind +changeBind anns mb_replacement (NonRec b e) = changeBindPr anns mb_replacement b e >>= (return . uncurry NonRec) +changeBind anns mb_replacement (Rec bes) = liftM Rec $ mapM (uncurry (changeBindPr anns mb_replacement)) bes + +changeBindPr :: UniqFM [ReplaceWith] -> Maybe String -> CoreBndr -> CoreExpr -> CoreM (CoreBndr, CoreExpr) +changeBindPr anns mb_replacement b e = do + case lookupWithDefaultUFM anns [] b of + [] -> do + e' <- changeExpr anns mb_replacement e + return (b, e') + [ReplaceWith replace_string] -> do + e' <- changeExpr anns (Just replace_string) e + return (b, e') + _ -> error $ "Too many change_anns on one binder:" ++ showSDoc (ppr b) + +changeExpr :: UniqFM [ReplaceWith] -> Maybe String -> CoreExpr -> CoreM CoreExpr +changeExpr anns mb_replacement e = let go = changeExpr anns mb_replacement in case e of + Lit (MachStr _) -> case mb_replacement of + Nothing -> return e + Just replacement -> do + putMsgS "Performing Replacement" + return $ Lit (MachStr (mkFastString replacement)) + App e1 e2 -> liftM2 App (go e1) (go e2) + Lam b e -> liftM (Lam b) (go e) + Let bind e -> liftM2 Let (changeBind anns mb_replacement bind) (go e) + Case e b ty alts -> liftM4 Case (go e) (return b) (return ty) (mapM (changeAlt anns mb_replacement) alts) + Cast e coerce -> liftM2 Cast (go e) (return coerce) + Note note e -> liftM (Note note) (go e) + _ -> return e + +changeAlt :: UniqFM [ReplaceWith] -> Maybe String -> CoreAlt -> CoreM CoreAlt +changeAlt anns mb_replacement (con, bs, e) = liftM (\e' -> (con, bs, e')) (changeExpr anns mb_replacement e) |