{-# LANGUAGE CPP #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE OverloadedStrings #-}
{-# OPTIONS_GHC -fno-warn-unused-binds #-}
{-# OPTIONS_GHC -fno-warn-unused-imports #-}
module Data.Array.Accelerate.Debug.Internal.Stats (
simplCount, resetSimplCount, dumpSimplStats,
inline, ruleFired, knownBranch, caseElim, caseDefault, betaReduce, substitution, simplifierDone, fusionDone,
) where
import Data.Array.Accelerate.Debug.Internal.Flags
import Data.Array.Accelerate.Debug.Internal.Trace
import Data.Function ( on )
import Data.IORef
import Data.List ( groupBy, sortBy )
import Data.Map ( Map )
import Data.Ord ( comparing )
import Data.Text ( Text )
import Data.Text.Lazy.Builder
import Formatting
import Prettyprinter hiding ( annotate, Doc )
import Prettyprinter.Internal ( SimpleDocStream(..), textSpaces )
import Prettyprinter.Render.Util.Panic ( panicUncaughtFail )
import System.IO.Unsafe
import qualified Data.Map as Map
import qualified Prettyprinter as Pretty
ruleFired, inline, knownBranch, caseElim, caseDefault, betaReduce, substitution :: Text -> a -> a
inline :: forall a. Text -> a -> a
inline = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
Inline
ruleFired :: forall a. Text -> a -> a
ruleFired = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
RuleFired
knownBranch :: forall a. Text -> a -> a
knownBranch = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
KnownBranch
caseElim :: forall a. Text -> a -> a
caseElim = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
CaseElim
caseDefault :: forall a. Text -> a -> a
caseDefault = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
CaseDefault
betaReduce :: forall a. Text -> a -> a
betaReduce = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
BetaReduce
substitution :: forall a. Text -> a -> a
substitution = (Id -> Tick) -> Text -> a -> a
forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
Substitution
simplifierDone, fusionDone :: a -> a
simplifierDone :: forall a. a -> a
simplifierDone = Tick -> a -> a
forall a. Tick -> a -> a
tick Tick
SimplifierDone
fusionDone :: forall a. a -> a
fusionDone = Tick -> a -> a
forall a. Tick -> a -> a
tick Tick
FusionDone
tick :: Tick -> a -> a
#ifdef ACCELERATE_DEBUG
{-# NOINLINE tick #-}
tick t expr = unsafeDupablePerformIO $ do
modifyIORef' statistics (simplTick t)
return expr
#else
{-# INLINE tick #-}
tick :: forall a. Tick -> a -> a
tick Tick
_ a
expr = a
expr
#endif
annotate :: (Id -> Tick) -> Text -> a -> a
annotate :: forall a. (Id -> Tick) -> Text -> a -> a
annotate Id -> Tick
name Text
ctx = Tick -> a -> a
forall a. Tick -> a -> a
tick (Id -> Tick
name (Text -> Id
Id Text
ctx))
data SimplStats
= Simple {-# UNPACK #-} !Int
| Detail {
SimplStats -> Int
ticks :: {-# UNPACK #-} !Int,
SimplStats -> TickCount
details :: !TickCount
}
instance Show SimplStats where
show :: SimplStats -> String
show = Doc -> String
forall a. Show a => a -> String
show (Doc -> String) -> (SimplStats -> Doc) -> SimplStats -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SimplStats -> Doc
pprSimplCount
{-# NOINLINE statistics #-}
statistics :: IORef SimplStats
statistics :: IORef SimplStats
statistics = IO (IORef SimplStats) -> IORef SimplStats
forall a. IO a -> a
unsafePerformIO (IO (IORef SimplStats) -> IORef SimplStats)
-> IO (IORef SimplStats) -> IORef SimplStats
forall a b. (a -> b) -> a -> b
$ SimplStats -> IO (IORef SimplStats)
forall a. a -> IO (IORef a)
newIORef (SimplStats -> IO (IORef SimplStats))
-> IO SimplStats -> IO (IORef SimplStats)
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< IO SimplStats
initSimplCount
initSimplCount :: IO SimplStats
#ifdef ACCELERATE_DEBUG
initSimplCount = do
d <- getFlag dump_simpl_stats
return $! if d then Detail { ticks = 0, details = Map.empty }
else Simple 0
#else
initSimplCount :: IO SimplStats
initSimplCount = SimplStats -> IO SimplStats
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return (SimplStats -> IO SimplStats) -> SimplStats -> IO SimplStats
forall a b. (a -> b) -> a -> b
$! Int -> SimplStats
Simple Int
0
#endif
resetSimplCount :: IO ()
#ifdef ACCELERATE_DEBUG
resetSimplCount = writeIORef statistics =<< initSimplCount
#else
resetSimplCount :: IO ()
resetSimplCount = () -> IO ()
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
#endif
{-# INLINEABLE dumpSimplStats #-}
dumpSimplStats :: IO ()
#ifdef ACCELERATE_DEBUG
dumpSimplStats = do
when dump_simpl_stats $ do
stats <- simplCount
putTraceMsg builder (render (layoutPretty defaultLayoutOptions stats))
resetSimplCount
where
render = \case
SFail -> panicUncaughtFail
SEmpty -> mempty
SChar c rest -> singleton c <> render rest
SText _l t rest -> fromText t <> render rest
SLine i rest -> singleton '\n' <> (fromText (textSpaces i) <> render rest)
SAnnPush _ann rest -> render rest
SAnnPop rest -> render rest
#else
dumpSimplStats :: IO ()
dumpSimplStats = () -> IO ()
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
#endif
simplTick :: Tick -> SimplStats -> SimplStats
simplTick :: Tick -> SimplStats -> SimplStats
simplTick Tick
_ (Simple Int
n) = Int -> SimplStats
Simple (Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
simplTick Tick
t (Detail Int
n TickCount
dts) = Int -> TickCount -> SimplStats
Detail (Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (TickCount
dts TickCount -> Tick -> TickCount
`addTick` Tick
t)
pprSimplCount :: SimplStats -> Doc
pprSimplCount :: SimplStats -> Doc
pprSimplCount (Simple Int
n) = Doc
"Total ticks:" Doc -> Doc -> Doc
forall ann. Doc ann -> Doc ann -> Doc ann
<+> Int -> Doc
forall ann. Int -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Int
n
pprSimplCount (Detail Int
n TickCount
dts)
= [Doc] -> Doc
forall ann. [Doc ann] -> Doc ann
vcat [ Doc
"Total ticks:" Doc -> Doc -> Doc
forall ann. Doc ann -> Doc ann -> Doc ann
<+> Int -> Doc
forall ann. Int -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Int
n
, Doc
forall a. Monoid a => a
mempty
, TickCount -> Doc
pprTickCount TickCount
dts
]
simplCount :: IO Doc
simplCount :: IO Doc
simplCount = SimplStats -> Doc
pprSimplCount (SimplStats -> Doc) -> IO SimplStats -> IO Doc
forall a b. (a -> b) -> IO a -> IO b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
`fmap` IORef SimplStats -> IO SimplStats
forall a. IORef a -> IO a
readIORef IORef SimplStats
statistics
type Doc = Pretty.Doc ()
type TickCount = Map Tick Int
data Id = Id Text
deriving (Id -> Id -> Bool
(Id -> Id -> Bool) -> (Id -> Id -> Bool) -> Eq Id
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Id -> Id -> Bool
== :: Id -> Id -> Bool
$c/= :: Id -> Id -> Bool
/= :: Id -> Id -> Bool
Eq, Eq Id
Eq Id
-> (Id -> Id -> Ordering)
-> (Id -> Id -> Bool)
-> (Id -> Id -> Bool)
-> (Id -> Id -> Bool)
-> (Id -> Id -> Bool)
-> (Id -> Id -> Id)
-> (Id -> Id -> Id)
-> Ord Id
Id -> Id -> Bool
Id -> Id -> Ordering
Id -> Id -> Id
forall a.
Eq a
-> (a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
$ccompare :: Id -> Id -> Ordering
compare :: Id -> Id -> Ordering
$c< :: Id -> Id -> Bool
< :: Id -> Id -> Bool
$c<= :: Id -> Id -> Bool
<= :: Id -> Id -> Bool
$c> :: Id -> Id -> Bool
> :: Id -> Id -> Bool
$c>= :: Id -> Id -> Bool
>= :: Id -> Id -> Bool
$cmax :: Id -> Id -> Id
max :: Id -> Id -> Id
$cmin :: Id -> Id -> Id
min :: Id -> Id -> Id
Ord)
data Tick
= Inline Id
| RuleFired Id
| KnownBranch Id
| CaseElim Id
| CaseDefault Id
| BetaReduce Id
| Substitution Id
| SimplifierDone
| FusionDone
deriving (Tick -> Tick -> Bool
(Tick -> Tick -> Bool) -> (Tick -> Tick -> Bool) -> Eq Tick
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Tick -> Tick -> Bool
== :: Tick -> Tick -> Bool
$c/= :: Tick -> Tick -> Bool
/= :: Tick -> Tick -> Bool
Eq, Eq Tick
Eq Tick
-> (Tick -> Tick -> Ordering)
-> (Tick -> Tick -> Bool)
-> (Tick -> Tick -> Bool)
-> (Tick -> Tick -> Bool)
-> (Tick -> Tick -> Bool)
-> (Tick -> Tick -> Tick)
-> (Tick -> Tick -> Tick)
-> Ord Tick
Tick -> Tick -> Bool
Tick -> Tick -> Ordering
Tick -> Tick -> Tick
forall a.
Eq a
-> (a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
$ccompare :: Tick -> Tick -> Ordering
compare :: Tick -> Tick -> Ordering
$c< :: Tick -> Tick -> Bool
< :: Tick -> Tick -> Bool
$c<= :: Tick -> Tick -> Bool
<= :: Tick -> Tick -> Bool
$c> :: Tick -> Tick -> Bool
> :: Tick -> Tick -> Bool
$c>= :: Tick -> Tick -> Bool
>= :: Tick -> Tick -> Bool
$cmax :: Tick -> Tick -> Tick
max :: Tick -> Tick -> Tick
$cmin :: Tick -> Tick -> Tick
min :: Tick -> Tick -> Tick
Ord)
addTick :: TickCount -> Tick -> TickCount
addTick :: TickCount -> Tick -> TickCount
addTick TickCount
tc Tick
t =
(Maybe Int -> Maybe Int) -> Tick -> TickCount -> TickCount
forall k a.
Ord k =>
(Maybe a -> Maybe a) -> k -> Map k a -> Map k a
Map.alter Maybe Int -> Maybe Int
forall {a}. Num a => Maybe a -> Maybe a
f Tick
t TickCount
tc
where
f :: Maybe a -> Maybe a
f Maybe a
Nothing = a -> Maybe a
forall a. a -> Maybe a
Just a
1
f (Just a
x) = let x' :: a
x' = a
xa -> a -> a
forall a. Num a => a -> a -> a
+a
1 in a
x' a -> Maybe a -> Maybe a
forall a b. a -> b -> b
`seq` a -> Maybe a
forall a. a -> Maybe a
Just a
x'
pprTickCount :: TickCount -> Doc
pprTickCount :: TickCount -> Doc
pprTickCount TickCount
counts =
[Doc] -> Doc
forall ann. [Doc ann] -> Doc ann
vcat (([(Tick, Int)] -> Doc) -> [[(Tick, Int)]] -> [Doc]
forall a b. (a -> b) -> [a] -> [b]
map [(Tick, Int)] -> Doc
pprTickGroup [[(Tick, Int)]]
groups)
where
groups :: [[(Tick, Int)]]
groups = ((Tick, Int) -> (Tick, Int) -> Bool)
-> [(Tick, Int)] -> [[(Tick, Int)]]
forall a. (a -> a -> Bool) -> [a] -> [[a]]
groupBy (Tick, Int) -> (Tick, Int) -> Bool
forall {b}. (Tick, b) -> (Tick, b) -> Bool
sameTag (TickCount -> [(Tick, Int)]
forall k a. Map k a -> [(k, a)]
Map.toList TickCount
counts)
sameTag :: (Tick, b) -> (Tick, b) -> Bool
sameTag = Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
(==) (Int -> Int -> Bool)
-> ((Tick, b) -> Int) -> (Tick, b) -> (Tick, b) -> Bool
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` Tick -> Int
tickToTag (Tick -> Int) -> ((Tick, b) -> Tick) -> (Tick, b) -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Tick, b) -> Tick
forall a b. (a, b) -> a
fst
pprTickGroup :: [(Tick,Int)] -> Doc
pprTickGroup :: [(Tick, Int)] -> Doc
pprTickGroup [] = String -> Doc
forall a. HasCallStack => String -> a
error String
"pprTickGroup"
pprTickGroup [(Tick, Int)]
grp =
Int -> Doc -> Doc
forall ann. Int -> Doc ann -> Doc ann
hang Int
2 ([Doc] -> Doc
forall ann. [Doc ann] -> Doc ann
vcat ([Doc] -> Doc) -> [Doc] -> Doc
forall a b. (a -> b) -> a -> b
$ (Int -> Doc
forall ann. Int -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Int
groupTotal Doc -> Doc -> Doc
forall ann. Doc ann -> Doc ann -> Doc ann
<+> Doc
groupName)
Doc -> [Doc] -> [Doc]
forall a. a -> [a] -> [a]
: [ Int -> Doc
forall ann. Int -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Int
n Doc -> Doc -> Doc
forall ann. Doc ann -> Doc ann -> Doc ann
<+> Tick -> Doc
pprTickCtx Tick
t | (Tick
t,Int
n) <- ((Tick, Int) -> (Tick, Int) -> Ordering)
-> [(Tick, Int)] -> [(Tick, Int)]
forall a. (a -> a -> Ordering) -> [a] -> [a]
sortBy (((Tick, Int) -> (Tick, Int) -> Ordering)
-> (Tick, Int) -> (Tick, Int) -> Ordering
forall a b c. (a -> b -> c) -> b -> a -> c
flip (((Tick, Int) -> Int) -> (Tick, Int) -> (Tick, Int) -> Ordering
forall a b. Ord a => (b -> a) -> b -> b -> Ordering
comparing (Tick, Int) -> Int
forall a b. (a, b) -> b
snd)) [(Tick, Int)]
grp ])
where
groupName :: Doc
groupName = Tick -> Doc
tickToStr ((Tick, Int) -> Tick
forall a b. (a, b) -> a
fst ([(Tick, Int)] -> (Tick, Int)
forall a. HasCallStack => [a] -> a
head [(Tick, Int)]
grp))
groupTotal :: Int
groupTotal = [Int] -> Int
forall a. Num a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Num a) => t a -> a
sum [Int
n | (Tick
_,Int
n) <- [(Tick, Int)]
grp]
tickToTag :: Tick -> Int
tickToTag :: Tick -> Int
tickToTag Inline{} = Int
0
tickToTag RuleFired{} = Int
1
tickToTag KnownBranch{} = Int
2
tickToTag CaseElim{} = Int
3
tickToTag CaseDefault{} = Int
4
tickToTag BetaReduce{} = Int
5
tickToTag Substitution{} = Int
6
tickToTag Tick
SimplifierDone = Int
99
tickToTag Tick
FusionDone = Int
100
tickToStr :: Tick -> Doc
tickToStr :: Tick -> Doc
tickToStr Inline{} = Doc
"Inline"
tickToStr RuleFired{} = Doc
"RuleFired"
tickToStr KnownBranch{} = Doc
"KnownBranch"
tickToStr CaseElim{} = Doc
"CaseElim"
tickToStr CaseDefault{} = Doc
"CaseDefault"
tickToStr BetaReduce{} = Doc
"BetaReduce"
tickToStr Substitution{} = Doc
"Substitution"
tickToStr Tick
SimplifierDone = Doc
"SimplifierDone"
tickToStr Tick
FusionDone = Doc
"FusionDone"
pprTickCtx :: Tick -> Doc
pprTickCtx :: Tick -> Doc
pprTickCtx (Inline Id
v) = Id -> Doc
pprId Id
v
pprTickCtx (RuleFired Id
v) = Id -> Doc
pprId Id
v
pprTickCtx (KnownBranch Id
v) = Id -> Doc
pprId Id
v
pprTickCtx (CaseElim Id
v) = Id -> Doc
pprId Id
v
pprTickCtx (CaseDefault Id
v) = Id -> Doc
pprId Id
v
pprTickCtx (BetaReduce Id
v) = Id -> Doc
pprId Id
v
pprTickCtx (Substitution Id
v) = Id -> Doc
pprId Id
v
pprTickCtx Tick
SimplifierDone = Doc
forall a. Monoid a => a
mempty
pprTickCtx Tick
FusionDone = Doc
forall a. Monoid a => a
mempty
pprId :: Id -> Doc
pprId :: Id -> Doc
pprId (Id Text
s) = Text -> Doc
forall ann. Text -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Text
s