mirror of
https://github.com/composewell/streamly.git
synced 2024-10-27 04:04:16 +03:00
128 lines
4.3 KiB
Haskell
128 lines
4.3 KiB
Haskell
{-# OPTIONS_GHC -Wno-deprecations #-}
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- Investigate specific benchmarks more closely in isolation, possibly looking
|
|
-- at GHC generated code for optimizing specific problematic cases.
|
|
-------------------------------------------------------------------------------
|
|
|
|
{-# LANGUAGE FlexibleContexts #-}
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
|
|
|
import Data.Char (ord)
|
|
import Streamly.Internal.Data.Stream (Stream)
|
|
import System.IO (hSeek, SeekMode(..), openFile, IOMode(..))
|
|
|
|
import qualified Streamly.Data.Array as A
|
|
import qualified Streamly.Internal.FileSystem.Handle as IFH
|
|
import qualified Streamly.Data.Fold as FL
|
|
import qualified Streamly.Internal.Data.Stream as S
|
|
import qualified Streamly.Internal.Data.Stream.IsStream as Internal
|
|
|
|
import Test.Tasty.Bench
|
|
import System.Random
|
|
|
|
maxValue :: Int
|
|
maxValue = 100000
|
|
|
|
drain :: Monad m => S.Stream m a -> m ()
|
|
drain = S.fold FL.drain
|
|
|
|
{-# INLINE sourceUnfoldrM #-}
|
|
sourceUnfoldrM :: Monad m => Stream m Int
|
|
sourceUnfoldrM = S.unfoldrM step 0
|
|
where
|
|
step cnt =
|
|
if cnt > maxValue
|
|
then return Nothing
|
|
else return (Just (cnt, cnt + 1))
|
|
|
|
{-# INLINE sourceUnfoldrMN #-}
|
|
sourceUnfoldrMN :: Monad m => Int -> Stream m Int
|
|
sourceUnfoldrMN n = S.unfoldrM step n
|
|
where
|
|
step cnt =
|
|
if cnt > n
|
|
then return Nothing
|
|
else return (Just (cnt, cnt + 1))
|
|
|
|
{-# INLINE sourceUnfoldr #-}
|
|
sourceUnfoldr :: Monad m => Int -> Stream m Int
|
|
sourceUnfoldr n = S.unfoldr step n
|
|
where
|
|
step cnt =
|
|
if cnt > n + maxValue
|
|
then Nothing
|
|
else Just (cnt, cnt + 1)
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- take-drop composition
|
|
-------------------------------------------------------------------------------
|
|
|
|
takeAllDropOne :: Monad m => Stream m Int -> Stream m Int
|
|
takeAllDropOne = S.drop 1 . S.take maxValue
|
|
|
|
-- Requires -fspec-constr-recursive=5 for better fused code
|
|
-- The number depends on how many times we compose it
|
|
|
|
{-# INLINE takeDrop #-}
|
|
takeDrop :: Monad m => Stream m Int -> m ()
|
|
takeDrop = drain .
|
|
takeAllDropOne . takeAllDropOne . takeAllDropOne . takeAllDropOne
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- dropWhileFalse composition
|
|
-------------------------------------------------------------------------------
|
|
|
|
dropWhileFalse :: Monad m => Stream m Int -> Stream m Int
|
|
dropWhileFalse = S.dropWhile (> maxValue)
|
|
|
|
-- Requires -fspec-constr-recursive=5 for better fused code
|
|
-- The number depends on how many times we compose it
|
|
|
|
{-# INLINE dropWhileFalseX4 #-}
|
|
dropWhileFalseX4 :: Monad m => Stream m Int -> m ()
|
|
dropWhileFalseX4 = drain
|
|
. dropWhileFalse . dropWhileFalse . dropWhileFalse . dropWhileFalse
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- iteration
|
|
-------------------------------------------------------------------------------
|
|
|
|
{-# INLINE iterateSource #-}
|
|
iterateSource
|
|
:: Monad m
|
|
=> (Stream m Int -> Stream m Int) -> Int -> Int -> Stream m Int
|
|
iterateSource g i n = f i (sourceUnfoldrMN n)
|
|
where
|
|
f (0 :: Int) m = g m
|
|
f x m = g (f (x - 1) m)
|
|
|
|
-- Keep only the benchmark that is to be investiagted and comment out the rest.
|
|
-- We keep all of them enabled by default for testing the build.
|
|
main :: IO ()
|
|
main = do
|
|
defaultMain [bench "unfoldr" $ nfIO $
|
|
randomRIO (1,1) >>= \n -> drain (sourceUnfoldr n)]
|
|
defaultMain [bench "take-drop" $ nfIO $ takeDrop sourceUnfoldrM]
|
|
defaultMain [bench "dropWhileFalseX4" $
|
|
nfIO $ dropWhileFalseX4 sourceUnfoldrM]
|
|
defaultMain [bench "iterate-mapM" $
|
|
nfIO $ drain $ iterateSource (S.mapM return) 100000 10]
|
|
|
|
inText <- openFile "benchmark/text-processing/gutenberg-500.txt" ReadMode
|
|
defaultMain [mkBenchText "splitOn abc...xyz" inText $ do
|
|
S.fold FL.length
|
|
(Internal.splitOnSeq
|
|
(A.fromList
|
|
$ map (fromIntegral . ord) "abcdefghijklmnopqrstuvwxyz"
|
|
)
|
|
FL.drain
|
|
$ IFH.read inText
|
|
)
|
|
>>= print
|
|
]
|
|
where
|
|
|
|
mkBenchText name h action =
|
|
env (hSeek h AbsoluteSeek 0) (\_ -> bench name $ nfIO action)
|