{-# LANGUAGE Trustworthy #-}
module Criterion.Monad
(
Criterion
, withConfig
, getGen
) where
import Control.Monad.Reader (asks, runReaderT)
import Control.Monad.Trans (liftIO)
import Criterion.Monad.Internal (Criterion(..), Crit(..))
import Criterion.Types hiding (measure)
import Data.IORef (IORef, newIORef, readIORef, writeIORef)
import System.IO.CodePage (withCP65001)
import System.Random.MWC (GenIO, createSystemRandom)
withConfig :: Config -> Criterion a -> IO a
withConfig :: forall a. Config -> Criterion a -> IO a
withConfig Config
cfg (Criterion ReaderT Crit IO a
act) = IO a -> IO a
forall a. IO a -> IO a
withCP65001 (IO a -> IO a) -> IO a -> IO a
forall a b. (a -> b) -> a -> b
$ do
g <- Maybe (Gen RealWorld) -> IO (IORef (Maybe (Gen RealWorld)))
forall a. a -> IO (IORef a)
newIORef Maybe (Gen RealWorld)
forall a. Maybe a
Nothing
runReaderT act (Crit cfg g)
getGen :: Criterion GenIO
getGen :: Criterion GenIO
getGen = (Crit -> IORef (Maybe (Gen RealWorld)))
-> IO (Gen RealWorld) -> Criterion (Gen RealWorld)
forall a. (Crit -> IORef (Maybe a)) -> IO a -> Criterion a
memoise Crit -> IORef (Maybe (Gen RealWorld))
Crit -> IORef (Maybe GenIO)
gen IO (Gen RealWorld)
IO GenIO
createSystemRandom
memoise :: (Crit -> IORef (Maybe a)) -> IO a -> Criterion a
memoise :: forall a. (Crit -> IORef (Maybe a)) -> IO a -> Criterion a
memoise Crit -> IORef (Maybe a)
ref IO a
generate = do
r <- ReaderT Crit IO (IORef (Maybe a)) -> Criterion (IORef (Maybe a))
forall a. ReaderT Crit IO a -> Criterion a
Criterion (ReaderT Crit IO (IORef (Maybe a)) -> Criterion (IORef (Maybe a)))
-> ReaderT Crit IO (IORef (Maybe a)) -> Criterion (IORef (Maybe a))
forall a b. (a -> b) -> a -> b
$ (Crit -> IORef (Maybe a)) -> ReaderT Crit IO (IORef (Maybe a))
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Crit -> IORef (Maybe a)
ref
liftIO $ do
mv <- readIORef r
case mv of
Just a
rv -> a -> IO a
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return a
rv
Maybe a
Nothing -> do
rv <- IO a
generate
writeIORef r (Just rv)
return rv