module Data.Array.Accelerate.Debug.Timed (
timed,
elapsed,
) where
import Data.Array.Accelerate.Debug.Trace
import Data.Array.Accelerate.Debug.Flags
import Control.Monad.Trans ( MonadIO )
import Text.Printf
#if ACCELERATE_DEBUG
import Control.Applicative
import Control.Monad.Trans ( liftIO )
import Data.Int
import Data.List
import Data.Time.Clock
import System.CPUTime
import Prelude
import GHC.Stats
#endif
timed :: MonadIO m => Mode -> (Double -> Double -> String) -> m a -> m a
#ifdef ACCELERATE_DEBUG
timed mode fmt action = do
enabled <- liftIO $ queryFlag mode
if enabled
then do
with_gc <- liftIO $ (&&) <$> getGCStatsEnabled <*> queryFlag verbose
if with_gc
then timed_gc fmt action
else timed_simpl fmt action
else
action
#else
timed _ _ action = action
#endif
#ifdef ACCELERATE_DEBUG
timed_simpl :: MonadIO m => (Double -> Double -> String) -> m a -> m a
timed_simpl fmt action = do
wall0 <- liftIO getCurrentTime
cpu0 <- liftIO getCPUTime
res <- action
wall1 <- liftIO getCurrentTime
cpu1 <- liftIO getCPUTime
let wallTime = realToFrac (diffUTCTime wall1 wall0)
cpuTime = fromIntegral (cpu1 cpu0) * 1E-12
liftIO $ putTraceMsg (fmt wallTime cpuTime)
return res
timed_gc :: MonadIO m => (Double -> Double -> String) -> m a -> m a
timed_gc fmt action = do
gc0 <- liftIO getGCStats
res <- action
gc1 <- liftIO getGCStats
let toDouble :: Int64 -> Double
toDouble = fromIntegral
allocated = toDouble (bytesAllocated gc1 bytesAllocated gc0)
copied = toDouble (bytesCopied gc1 bytesCopied gc0)
totalWall = wallSeconds gc1 wallSeconds gc0
totalCPU = cpuSeconds gc1 cpuSeconds gc0
mutatorWall = mutatorWallSeconds gc1 mutatorWallSeconds gc0
mutatorCPU = mutatorCpuSeconds gc1 mutatorCpuSeconds gc0
gcWall = gcWallSeconds gc1 gcWallSeconds gc0
gcCPU = gcCpuSeconds gc1 gcCpuSeconds gc0
liftIO . putTraceMsg $ intercalate "\n"
[ fmt totalWall totalCPU
, printf " %s allocated on the heap" (showFFloatSIBase (Just 1) 1024 allocated "B")
, printf " %s copied during GC (%d collections)" (showFFloatSIBase (Just 1) 1024 copied "B") (numGcs gc1 numGcs gc0)
, printf " MUT: %s" (elapsed mutatorWall mutatorCPU)
, printf " GC: %s" (elapsed gcWall gcCPU)
]
return res
#endif
elapsed :: Double -> Double -> String
elapsed wallTime cpuTime =
printf "%s (wall), %s (cpu)"
(showFFloatSIBase (Just 3) 1000 wallTime "s")
(showFFloatSIBase (Just 3) 1000 cpuTime "s")