{-# LANGUAGE DeriveFunctor, FlexibleContexts, FlexibleInstances, MultiParamTypeClasses, RankNTypes, TypeApplications, TypeOperators, UndecidableInstances #-} module Main where import Control.Effect import Control.Effect.Carrier import Control.Effect.Pure import Control.Effect.Writer import Control.Effect.State import Control.Monad (ap, replicateM_) import Criterion.Main import Data.Monoid (Sum(..)) main :: IO () main = defaultMain [ bgroup "WriterC" [ bgroup "Cod" [ bench "100" $ whnf (run . runCod pure . execWriter @(Sum Int) . runCod pure . tellLoop) 100 , bench "1000" $ whnf (run . runCod pure . execWriter @(Sum Int) . runCod pure . tellLoop) 1000 , bench "10000" $ whnf (run . runCod pure . execWriter @(Sum Int) . runCod pure . tellLoop) 10000 ] , bgroup "standalone" [ bench "100" $ whnf (run . execWriter @(Sum Int) . tellLoop) 100 , bench "1000" $ whnf (run . execWriter @(Sum Int) . tellLoop) 1000 , bench "10000" $ whnf (run . execWriter @(Sum Int) . tellLoop) 10000 ] ] , bgroup "Strict StateC" [ bgroup "Cod" [ bench "100" $ whnf (run . runCod pure . execState @(Sum Int) 0 . runCod pure . modLoop) 100 , bench "1000" $ whnf (run . runCod pure . execState @(Sum Int) 0 . runCod pure . modLoop) 1000 , bench "10000" $ whnf (run . runCod pure . execWriter @(Sum Int) . runCod pure . tellLoop) 10000 ] , bgroup "standalone" [ bench "100" $ whnf (run . execState @(Sum Int) 0 . modLoop) 100 , bench "1000" $ whnf (run . execState @(Sum Int) 0 . modLoop) 1000 , bench "10000" $ whnf (run . execWriter @(Sum Int) . tellLoop) 10000 ] ] ] tellLoop :: (Applicative m, Carrier sig m, Member (Writer (Sum Int)) sig) => Int -> m () tellLoop i = replicateM_ i (tell (Sum (1 :: Int))) modLoop :: (Applicative m, Carrier sig m, Member (State (Sum Int)) sig) => Int -> m () modLoop i = replicateM_ i (modify (+ (Sum (1 :: Int)))) newtype Cod m a = Cod { unCod :: forall b . (a -> m b) -> m b } deriving (Functor) runCod :: (a -> m b) -> Cod m a -> m b runCod = flip unCod instance Applicative (Cod m) where pure a = Cod (\ k -> k a) (<*>) = ap instance Monad (Cod m) where Cod a >>= f = Cod (\ k -> a (runCod k . f)) instance Carrier sig m => Carrier sig (Cod m) where eff op = Cod (\ k -> eff (hmap (runCod pure) (fmap' (runCod k) op)))