{-# LANGUAGE RankNTypes #-} module Main (main) where import Common (commonMain) import Control.Monad.Identity import qualified Control.Monad.Trans.Reader as R import qualified Control.Monad.Trans.State.Strict as S import Criterion.Main import Data.Monoid import Pipes import Pipes.Lift defaultMax :: Int defaultMax = 10000 main :: IO () main = commonMain defaultMax liftBenchmarks iter :: forall m a . (Monad m , Ord a, Num a) => (a -> m a) -> a -> Effect m a iter a vmax = loop 0 where loop n | n > vmax = return vmax | otherwise = do x <- lift $ a n loop $! x s_bench :: Int -> Effect (S.StateT Int Identity) Int s_bench = iter (\n -> S.get >>= (\a -> S.put $! a + n) >> return (n + 1)) r_bench :: Int -> Effect (R.ReaderT Int Identity) Int r_bench = iter (\n -> R.ask >>= (\a -> return $ n + a)) -- Run before Proxy runB :: (a -> Effect Identity r) -> a -> r runB f a = runIdentity $ runEffect $ f a -- Run after Proxy runA :: (Monad m) => (m r -> Identity a) -> Effect m r -> a runA f a = runIdentity $ f (runEffect a) liftBenchmarks :: Int -> [Benchmark] liftBenchmarks vmax = let applyBench = map ($ vmax) in [ bgroup "ReaderT" $ let defT f = (\d -> f d 1) in applyBench [ bench "runReaderP_B" . whnf (runB (runReaderP 1) . r_bench) , bench "runReaderP_A" . whnf (runA (defT R.runReaderT) . r_bench) ] , bgroup "StateT" $ let defT f = (\s -> f s 0) in applyBench [ bench "runStateP_B" . nf (runB (runStateP 0) . s_bench) , bench "runStateP_A" . nf (runA (defT S.runStateT) . s_bench) , bench "evalStateP_B" . whnf (runB (evalStateP 0) . s_bench) , bench "evalStateP_A" . whnf (runA (defT S.evalStateT) . s_bench) , bench "execStateP_B" . whnf (runB (execStateP 0) . s_bench) , bench "execStateP_A" . whnf (runA (defT S.execStateT) . s_bench) ] ]