----------------------------------------------------------------------------- -- | -- Module : Control.Concurrent.Speculation.List -- Copyright : (C) 2010-2011 Edward Kmett, -- License : BSD-style (see the file LICENSE) -- -- Maintainer : Edward Kmett -- Stability : provisional -- Portability : portable -- ---------------------------------------------------------------------------- module Control.Concurrent.Speculation.List ( -- * Speculative scans scan, scanBy , scanMap, scanMapBy , scanr, scanrBy , scanl, scanlBy , scanr1, scanr1By , scanl1, scanl1By {- -- ** Speculative monadic scans , scanrM, scanrByM , scanlM, scanlByM -- * Speculative transactional monadic scans , scanrSTM, scanrBySTM , scanlSTM, scanlBySTM -} ) where import Prelude hiding (foldl, foldl1, foldr, foldr1 , any, all, and, or, mapM_, sequence_ , elem, notElem, sum, product , minimum, maximum, concat, concatMap , scanr, scanl, scanr1, scanl1 ) import Data.Monoid import qualified Data.List as List import Control.Concurrent.Speculation import Control.Concurrent.Speculation.Internal -- | Given a valid estimator @g@, @'scan' g xs@ converts @xs@ into a list of the prefix sums. -- -- @g n@ should supply an estimate of the value of the monoidal summation over the first @n@ elements of the container. -- -- If @g n@ is accurate a reasonable percentage of the time and faster to compute than the prefix sum, then this can -- provide increased opportunities for parallelism. scan :: (Monoid m, Eq m) => (Int -> m) -> [m] -> [m] scan = scanBy (==) {-# INLINE scan #-} -- | 'scan' using 'specBy' scanBy :: Monoid m => (m -> m -> Bool) -> (Int -> m) -> [m] -> [m] scanBy cmp g = scanrBy cmp g mappend mempty {-# INLINE scanBy #-} -- | Given a valid estimator @g@, @'scanMap' g f xs@ converts @xs@ into a list of the prefix sums. -- -- @g n@ should supply an estimate of the value of the monoidal summation over the first @n@ elements of the container. -- -- If @g n@ is accurate a reasonable percentage of the time and faster to compute than the scan, then this can -- provide increased opportunities for parallelism. -- -- > scan = scanMap id -- > scanMap = scanMapBy (==) scanMap :: (Monoid m, Eq m) => (Int -> m) -> (a -> m) -> [a] -> [m] scanMap = scanMapBy (==) {-# INLINE scanMap #-} scanMapBy :: Monoid m => (m -> m -> Bool) -> (Int -> m) -> (a -> m) -> [a] -> [m] scanMapBy cmp g f = scanrBy cmp g (mappend . f) mempty {-# INLINE scanMapBy #-} -- | Given a valid estimator @g@, @'scanr' g f z xs@ yields the same answer as @'scanr'' f z xs@. -- -- @g n@ should supply an estimate of the value returned from scanning over the last @n@ elements of the container. -- -- If @g n@ is accurate a reasonable percentage of the time and faster to compute than the scan, then this can -- provide increased opportunities for parallelism. scanr :: Eq b => (Int -> b) -> (a -> b -> b) -> b -> [a] -> [b] scanr = scanrBy (==) {-# INLINE scanr #-} scanrBy :: (b -> b -> Bool) -> (Int -> b) -> (a -> b -> b) -> b -> [a] -> [b] scanrBy cmp g f z = map extractAcc . List.scanr mf (Acc 0 z) where mf a (Acc n b) = let n' = n + 1 in Acc n' (specBy' cmp (g n') (f a) b) {-# INLINE scanrBy #-} scanl :: Eq b => (Int -> b) -> (b -> a -> b) -> b -> [a] -> [b] scanl = scanlBy (==) {-# INLINE scanl #-} scanlBy :: (b -> b -> Bool) -> (Int -> b) -> (b -> a -> b) -> b -> [a] -> [b] scanlBy cmp g f z = map extractAcc . List.scanl mf (Acc 0 z) where mf (Acc n a) b = let n' = n + 1 in Acc n' (specBy' cmp (g n') (`f` b) a) {-# INLINE scanlBy #-} scanr1 :: Eq a => (Int -> a) -> (a -> a -> a) -> [a] -> [a] scanr1 = scanr1By (==) {-# INLINE scanr1 #-} scanr1By :: (a -> a -> Bool) -> (Int -> a) -> (a -> a -> a) -> [a] -> [a] scanr1By cmp g f xs = map (fromMaybeAcc undefined) $ List.scanr mf NothingAcc xs where mf a (JustAcc n b) = let n' = n + 1 in JustAcc n' (specBy' cmp (g n') (f a) b) mf a NothingAcc = JustAcc 1 a {-# INLINE scanr1By #-} scanl1 :: Eq a => (Int -> a) -> (a -> a -> a) -> [a] -> [a] scanl1 = scanl1By (==) {-# INLINE scanl1 #-} scanl1By :: (a -> a -> Bool) -> (Int -> a) -> (a -> a -> a) -> [a] -> [a] scanl1By cmp g f xs = map (fromMaybeAcc undefined) $ List.scanl mf NothingAcc xs where mf (JustAcc n a) b = let n' = n + 1 in JustAcc n' (specBy' cmp (g n') (`f` b) a) mf NothingAcc b = JustAcc 1 b {-# INLINE scanl1By #-}