module Language.HERMIT.Optimize
(
optimize
, query
, run
, interactive
, display
, setPretty
, setPrettyOptions
, at
, phase
, after
, before
, allPhases
) where
import GhcPlugins hiding (singleton, liftIO, display)
import qualified GhcPlugins as GHC
import Control.Monad.Operational
import Control.Monad.State hiding (guard)
import Data.Default
import Language.HERMIT.Core
import Language.HERMIT.External hiding (Query, Shell)
import Language.HERMIT.Kernel.Scoped
import Language.HERMIT.Kure
import Language.HERMIT.Monad
import Language.HERMIT.Plugin
import Language.HERMIT.PrettyPrinter.Common
import qualified Language.HERMIT.PrettyPrinter.Clean as Clean
import qualified Language.HERMIT.Shell.Command as Shell
import System.Console.Haskeline (defaultBehavior)
import System.IO (stdout)
data OInst :: * -> * where
RR :: RewriteH Core -> OInst ()
Query :: TranslateH Core a -> OInst a
Shell :: [External] -> [CommandLineOption] -> OInst ()
Guard :: (PhaseInfo -> Bool) -> OM () -> OInst ()
Focus :: TranslateH Core Path -> OM () -> OInst ()
type OM a = ProgramT OInst (StateT InterpState IO) a
optimize :: ([CommandLineOption] -> OM ()) -> Plugin
optimize f = hermitPlugin $ \ phaseInfo -> runOM phaseInfo . f
data InterpState =
InterpState { isAST :: SAST
, isPretty :: PrettyOptions -> PrettyH Core
, isPrettyOptions :: PrettyOptions
, shellHack :: Maybe ([External], [CommandLineOption])
}
type InterpM a = StateT InterpState IO a
runOM :: PhaseInfo -> OM () -> ModGuts -> CoreM ModGuts
runOM phaseInfo opt = scopedKernel $ \ kernel initSAST ->
let env = mkHermitMEnv $ GHC.liftIO . debug
debug (DebugTick msg) = putStrLn msg
debug (DebugCore msg _c _e) = putStrLn $ "Core: " ++ msg
errorAbortIO err = putStrLn err >> abortS kernel
errorAbort = liftIO . errorAbortIO
initState = InterpState initSAST Clean.corePrettyH def Nothing
eval :: Path -> ProgramT OInst (StateT InterpState IO) () -> InterpM ()
eval path comp = do
sast <- gets isAST
v <- viewT comp
case v of
Return _ -> return ()
RR rr :>>= k -> liftIO (applyS kernel sast (pathR path (extractR rr)) env)
>>= runKureM (\sast' -> modify (\s -> s { isAST = sast' }))
errorAbort >> eval path (k ())
Query tr :>>= k -> liftIO (queryS kernel sast (pathT path (extractT tr)) env)
>>= runKureM (eval path . k) errorAbort
Shell es os :>>= _k -> modify (\s -> s { shellHack = Just (es,os) })
Guard p m :>>= k -> when (p phaseInfo) (eval path m) >> eval path (k ())
Focus tp m :>>= k -> liftIO (queryS kernel sast (extractT tp) env)
>>= runKureM (flip eval m) errorAbort >> eval path (k ())
in do st <- execStateT (eval [] opt) initState
let sast = isAST st
maybe (liftIO (resumeS kernel sast) >>= runKureM return errorAbortIO)
(\(es,os) -> liftIO $ Shell.interactive os defaultBehavior es kernel sast)
(shellHack st)
interactive :: [External] -> [CommandLineOption] -> OM ()
interactive es os = singleton $ Shell es os
run :: RewriteH Core -> OM ()
run = singleton . RR
query :: TranslateH Core a -> OM a
query = singleton . Query
guard :: (PhaseInfo -> Bool) -> OM () -> OM ()
guard p = singleton . Guard p
at :: TranslateH Core Path -> OM () -> OM ()
at tp = singleton . Focus tp
phase :: Int -> OM () -> OM ()
phase n = guard ((n ==) . phaseNum)
after :: CorePass -> OM () -> OM ()
after cp = guard (\phaseInfo -> case phasesDone phaseInfo of
[] -> False
xs -> last xs == cp)
before :: CorePass -> OM () -> OM ()
before cp = guard (\phaseInfo -> case phasesLeft phaseInfo of
(x:_) | cp == x -> True
_ -> False)
allPhases :: OM () -> OM ()
allPhases = guard (const True)
display :: OM ()
display = do
po <- gets isPrettyOptions
gets isPretty >>= query . ($ po) >>= liftIO . Shell.unicodeConsole stdout po
setPretty :: (PrettyOptions -> PrettyH Core) -> OM ()
setPretty pp = modify $ \s -> s { isPretty = pp }
setPrettyOptions :: PrettyOptions -> OM ()
setPrettyOptions po = modify $ \s -> s { isPrettyOptions = po }