{-# LANGUAGE CPP #-}
module Foreign.Storable.Generic.Plugin (plugin) where
#if MIN_VERSION_GLASGOW_HASKELL(9,6,1,0)
import GHC.Core.Opt.Simplify.Env (SimplMode (sm_phase))
import GHC.Core.Opt.Simplify (so_mode)
#endif
#if MIN_VERSION_GLASGOW_HASKELL(9,0,1,0)
import GHC.Plugins
#else
import GhcPlugins
#endif
import Data.Maybe
import Foreign.Storable.Generic.Plugin.Internal
import Data.IORef
import Data.List
import Control.Monad (when)
import Foreign.Storable.Generic.Plugin.Internal.Error
plugin :: Plugin
plugin :: Plugin
plugin = Plugin
defaultPlugin {
installCoreToDos = install
#if MIN_VERSION_GLASGOW_HASKELL(8,6,1,0)
, pluginRecompile = \[CommandLineOption]
_ -> PluginRecompile -> IO PluginRecompile
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure PluginRecompile
NoForceRecompile
#endif
}
defFlags :: Flags
defFlags = Verbosity -> CrashOnWarning -> Flags
Flags Verbosity
Some CrashOnWarning
False
orderingPass :: Flags -> IORef [[Type]] -> CoreToDo
orderingPass :: Flags -> IORef [[Type]] -> CoreToDo
orderingPass Flags
flags IORef [[Type]]
io_ref = CommandLineOption -> CorePluginPass -> CoreToDo
CoreDoPluginPass CommandLineOption
"GStorable - type ordering"
(Flags -> IORef [[Type]] -> CorePluginPass
groupTypes Flags
flags IORef [[Type]]
io_ref)
substitutionPass :: Flags -> IORef [[Type]] -> CoreToDo
substitutionPass :: Flags -> IORef [[Type]] -> CoreToDo
substitutionPass Flags
flags IORef [[Type]]
io_ref = CommandLineOption -> CorePluginPass -> CoreToDo
CoreDoPluginPass CommandLineOption
"GStorable - substitution"
(Flags -> IORef [[Type]] -> CorePluginPass
gstorableSubstitution Flags
flags IORef [[Type]]
io_ref)
isPhase0 :: CoreToDo
-> Bool
#if MIN_VERSION_GLASGOW_HASKELL(9,6,0,0)
isPhase0 :: CoreToDo -> CrashOnWarning
isPhase0 (CoreDoSimplify SimplifyOpts
simpl_mode) = case SimplMode -> CompilerPhase
sm_phase (SimplMode -> CompilerPhase) -> SimplMode -> CompilerPhase
forall a b. (a -> b) -> a -> b
$ (SimplifyOpts -> SimplMode
so_mode SimplifyOpts
simpl_mode) of
#else
isPhase0 (CoreDoSimplify iters simpl_mode) = case sm_phase $ simpl_mode of
#endif
Phase PhaseNum
0 -> CrashOnWarning
True
CompilerPhase
_ -> CrashOnWarning
False
isPhase0 CoreToDo
_ = CrashOnWarning
False
afterPhase0 :: [CoreToDo] -> Maybe Int
afterPhase0 :: [CoreToDo] -> Maybe PhaseNum
afterPhase0 [CoreToDo]
todos = (CoreToDo -> CrashOnWarning) -> [CoreToDo] -> Maybe PhaseNum
forall a. (a -> CrashOnWarning) -> [a] -> Maybe PhaseNum
findIndex CoreToDo -> CrashOnWarning
isPhase0 [CoreToDo]
todos
isSpecialize :: CoreToDo -> Bool
isSpecialize :: CoreToDo -> CrashOnWarning
isSpecialize CoreToDo
CoreDoSpecialising = CrashOnWarning
True
isSpecialize CoreToDo
_ = CrashOnWarning
False
afterSpecialize :: [CoreToDo] -> Maybe Int
afterSpecialize :: [CoreToDo] -> Maybe PhaseNum
afterSpecialize [CoreToDo]
todos = (CoreToDo -> CrashOnWarning) -> [CoreToDo] -> Maybe PhaseNum
forall a. (a -> CrashOnWarning) -> [a] -> Maybe PhaseNum
findIndex CoreToDo -> CrashOnWarning
isSpecialize [CoreToDo]
todos
setOpts :: Flags -> String -> Flags
setOpts :: Flags -> CommandLineOption -> Flags
setOpts (Flags Verbosity
_ CrashOnWarning
crash) CommandLineOption
"-v0" = Verbosity -> CrashOnWarning -> Flags
Flags Verbosity
None CrashOnWarning
crash
setOpts (Flags Verbosity
_ CrashOnWarning
crash) CommandLineOption
"-v1" = Verbosity -> CrashOnWarning -> Flags
Flags Verbosity
Some CrashOnWarning
crash
setOpts (Flags Verbosity
_ CrashOnWarning
crash) CommandLineOption
"-v2" = Verbosity -> CrashOnWarning -> Flags
Flags Verbosity
All CrashOnWarning
crash
setOpts (Flags Verbosity
verb CrashOnWarning
_ ) CommandLineOption
"-crash" = Verbosity -> CrashOnWarning -> Flags
Flags Verbosity
verb CrashOnWarning
True
setOpts Flags
flags CommandLineOption
opt = Flags
flags
parseOpts :: [CommandLineOption] -> Flags
parseOpts :: [CommandLineOption] -> Flags
parseOpts [CommandLineOption]
opts = (Flags -> CommandLineOption -> Flags)
-> Flags -> [CommandLineOption] -> Flags
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' Flags -> CommandLineOption -> Flags
setOpts Flags
defFlags [CommandLineOption]
opts
putPasses :: Flags -> [CoreToDo] -> Int -> Int -> CoreM [CoreToDo]
putPasses :: Flags -> [CoreToDo] -> PhaseNum -> PhaseNum -> CoreM [CoreToDo]
putPasses Flags
flags [CoreToDo]
todos PhaseNum
ph0 PhaseNum
sp = do
IORef [[Type]]
the_ioref <- IO (IORef [[Type]]) -> CoreM (IORef [[Type]])
forall a. IO a -> CoreM a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO (IORef [[Type]]) -> CoreM (IORef [[Type]]))
-> IO (IORef [[Type]]) -> CoreM (IORef [[Type]])
forall a b. (a -> b) -> a -> b
$ [[Type]] -> IO (IORef [[Type]])
forall a. a -> IO (IORef a)
newIORef []
let ([CoreToDo]
before_spec,[CoreToDo]
after_spec) = PhaseNum -> [CoreToDo] -> ([CoreToDo], [CoreToDo])
forall a. PhaseNum -> [a] -> ([a], [a])
splitAt PhaseNum
sp [CoreToDo]
todos
([CoreToDo]
before_ph0 ,[CoreToDo]
after_ph0) = PhaseNum -> [CoreToDo] -> ([CoreToDo], [CoreToDo])
forall a. PhaseNum -> [a] -> ([a], [a])
splitAt (PhaseNum
ph0PhaseNum -> PhaseNum -> PhaseNum
forall a. Num a => a -> a -> a
-PhaseNum
sp) [CoreToDo]
after_spec
ordering :: CoreToDo
ordering = Flags -> IORef [[Type]] -> CoreToDo
orderingPass Flags
flags IORef [[Type]]
the_ioref
substitute :: CoreToDo
substitute = Flags -> IORef [[Type]] -> CoreToDo
substitutionPass Flags
flags IORef [[Type]]
the_ioref
new_todos :: [CoreToDo]
new_todos = [[CoreToDo]] -> [CoreToDo]
forall (t :: * -> *) a. Foldable t => t [a] -> [a]
concat [[CoreToDo]
before_spec, [CoreToDo
ordering], [CoreToDo]
before_ph0, [CoreToDo
substitute] , [CoreToDo]
after_ph0]
[CoreToDo] -> CoreM [CoreToDo]
forall a. a -> CoreM a
forall (m :: * -> *) a. Monad m => a -> m a
return [CoreToDo]
new_todos
install_err :: Flags -> CoreM ()
install_err :: Flags -> CoreM ()
install_err Flags
flags = do
let (Flags Verbosity
verb CrashOnWarning
to_crash) = Flags
flags
printer :: CoreM ()
printer = case Verbosity
verb of
Verbosity
None -> () -> CoreM ()
forall a. a -> CoreM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
Verbosity
other -> SDoc -> CoreM ()
putMsg (SDoc -> CoreM ()) -> SDoc -> CoreM ()
forall a b. (a -> b) -> a -> b
$ CommandLineOption -> SDoc
forall doc. IsLine doc => CommandLineOption -> doc
text CommandLineOption
"The GStorable plugin requires simplifier phases with inlining and rules on, as well as a specialiser phase."
SDoc -> SDoc -> SDoc
forall doc. IsDoc doc => doc -> doc -> doc
$$ CommandLineOption -> SDoc
forall doc. IsLine doc => CommandLineOption -> doc
text CommandLineOption
"Try to compile the code with -O1 or -O2 optimisation flags."
CoreM ()
printer
CrashOnWarning -> CoreM () -> CoreM ()
forall (f :: * -> *).
Applicative f =>
CrashOnWarning -> f () -> f ()
when CrashOnWarning
to_crash (CoreM () -> CoreM ()) -> CoreM () -> CoreM ()
forall a b. (a -> b) -> a -> b
$ (() -> CoreM ()
forall a. a -> CoreM a
forall (m :: * -> *) a. Monad m => a -> m a
return (() -> CoreM ()) -> () -> CoreM ()
forall a b. (a -> b) -> a -> b
$ CommandLineOption -> ()
forall a. HasCallStack => CommandLineOption -> a
error CommandLineOption
"Crashing...")
install :: [CommandLineOption] -> [CoreToDo] -> CoreM [CoreToDo]
install :: [CommandLineOption] -> [CoreToDo] -> CoreM [CoreToDo]
install [CommandLineOption]
opts [CoreToDo]
todos = do
DynFlags
dyn_flags <- CoreM DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
#if MIN_VERSION_GLASGOW_HASKELL(9,4,0,0)
let opt_level :: PhaseNum
opt_level = DynFlags -> PhaseNum
llvmOptLevel DynFlags
dyn_flags
#else
let opt_level = optLevel dyn_flags
#endif
flags :: Flags
flags = [CommandLineOption] -> Flags
parseOpts [CommandLineOption]
opts
m_phase0 :: Maybe PhaseNum
m_phase0 = [CoreToDo] -> Maybe PhaseNum
afterPhase0 [CoreToDo]
todos
m_spec :: Maybe PhaseNum
m_spec = [CoreToDo] -> Maybe PhaseNum
afterSpecialize [CoreToDo]
todos
case (Maybe PhaseNum
m_phase0, Maybe PhaseNum
m_spec, PhaseNum
opt_level) of
(Maybe PhaseNum
_ ,Maybe PhaseNum
_ ,PhaseNum
0) -> Flags -> CoreM ()
install_err Flags
flags CoreM () -> CoreM [CoreToDo] -> CoreM [CoreToDo]
forall a b. CoreM a -> CoreM b -> CoreM b
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> [CoreToDo] -> CoreM [CoreToDo]
forall a. a -> CoreM a
forall (m :: * -> *) a. Monad m => a -> m a
return [CoreToDo]
todos
(Just PhaseNum
ph0, Just PhaseNum
sp,PhaseNum
_) -> Flags -> [CoreToDo] -> PhaseNum -> PhaseNum -> CoreM [CoreToDo]
putPasses Flags
flags [CoreToDo]
todos (PhaseNum
ph0PhaseNum -> PhaseNum -> PhaseNum
forall a. Num a => a -> a -> a
+PhaseNum
1) (PhaseNum
spPhaseNum -> PhaseNum -> PhaseNum
forall a. Num a => a -> a -> a
+PhaseNum
1)
(Maybe PhaseNum
_ ,Maybe PhaseNum
_ ,PhaseNum
_) -> Flags -> CoreM ()
install_err Flags
flags CoreM () -> CoreM [CoreToDo] -> CoreM [CoreToDo]
forall a b. CoreM a -> CoreM b -> CoreM b
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> [CoreToDo] -> CoreM [CoreToDo]
forall a. a -> CoreM a
forall (m :: * -> *) a. Monad m => a -> m a
return [CoreToDo]
todos