module GHC.RTS.Events.Merge (mergeEventLogs) where
import GHC.RTS.Events
import Data.Monoid
import Data.List (foldl')
import qualified Data.Map as M
import Data.Word (Word32, Word16)
mergeEventLogs :: EventLog -> EventLog -> EventLog
mergeEventLogs (EventLog h1 (Data xs)) (EventLog h2 (Data ys)) =
let headerMap = M.fromList . map (\ et@EventType {num} -> (num, et))
m1 = headerMap $ eventTypes h1
m2 = headerMap $ eventTypes h2
combine et1 et2 | et1 == et2 = et1
combine _ _ = error "can't merge eventlogs with inconsistent headers"
m = M.unionWith combine m1 m2
h = Header $ M.elems m
in h == h `seq`
EventLog h . Data . mergeOn time xs $ shift (maxVars xs) ys
mergeOn :: Ord b => (a -> b) -> [a] -> [a] -> [a]
mergeOn f [] ys = ys
mergeOn f xs [] = xs
mergeOn f (x:xs) (y:ys) | f x <= f y = x : mergeOn f xs (y:ys)
| otherwise = y : mergeOn f (x:xs) ys
data MaxVars = MaxVars { mcapset :: !Word32
, mcap :: !Int
, mthread :: !ThreadId }
instance Monoid MaxVars where
mempty = MaxVars 0 0 0
mappend (MaxVars a b c) (MaxVars x y z) =
MaxVars (max a x) (b + y) (max c z)
mconcat = foldl' mappend mempty
isCap :: Int -> Bool
isCap x = fromIntegral x /= ((1) :: Word16)
maxVars :: [Event] -> MaxVars
maxVars = mconcat . map (maxSpec . spec)
where
maxSpec (Startup n) = mempty { mcap = n }
maxSpec (CreateThread t) = mempty { mthread = t }
maxSpec (CreateSparkThread t) = mempty { mthread = t }
maxSpec (CapsetCreate cs _) = mempty {mcapset = cs + 1 }
maxSpec (EventBlock _ _ es) = maxVars es
maxSpec _ = mempty
sh :: Num a => a -> a -> a
sh x y = x + y
shift :: MaxVars -> [Event] -> [Event]
shift mv@(MaxVars mcs mc mt) = map (\(Event t s) -> Event t $ shift' s)
where
shift' (EventBlock et c bs) = EventBlock et (msh c) $ shift mv bs
where msh x = if isCap x then sh mc x else x
shift' (CreateThread t) = CreateThread $ sh mt t
shift' (RunThread t) = RunThread $ sh mt t
shift' (StopThread t s) = StopThread (sh mt t) s
shift' (ThreadRunnable t) = ThreadRunnable $ sh mt t
shift' (MigrateThread t c) = MigrateThread (sh mt t) (sh mc c)
shift' (WakeupThread t c) = WakeupThread (sh mt t) (sh mc c)
shift' (ThreadLabel t l) = ThreadLabel (sh mt t) l
shift' (CreateSparkThread t) = CreateSparkThread (sh mt t)
shift' (SparkSteal c) = SparkSteal (sh mc c)
shift' (TaskCreate tk c tid) = TaskCreate tk (sh mc c) tid
shift' (TaskMigrate tk c1 c2) = TaskMigrate tk (sh mc c1) (sh mc c2)
shift' (CapCreate c) = CapCreate (sh mc c)
shift' (CapDelete c) = CapDelete (sh mc c)
shift' (CapDisable c) = CapDisable (sh mc c)
shift' (CapEnable c) = CapEnable (sh mc c)
shift' (CapsetCreate cs cst) = CapsetCreate (sh mcs cs) cst
shift' (CapsetDelete cs) = CapsetDelete (sh mcs cs)
shift' (CapsetAssignCap cs c) = CapsetAssignCap (sh mcs cs) (sh mc c)
shift' (CapsetRemoveCap cs c) = CapsetRemoveCap (sh mcs cs) (sh mc c)
shift' (RtsIdentifier cs rts) = RtsIdentifier (sh mcs cs) rts
shift' (ProgramArgs cs as) = ProgramArgs (sh mcs cs) as
shift' (ProgramEnv cs es) = ProgramEnv (sh mcs cs) es
shift' (OsProcessPid cs pid) = OsProcessPid (sh mcs cs) pid
shift' (OsProcessParentPid cs ppid) = OsProcessParentPid (sh mcs cs) ppid
shift' (WallClockTime cs sec nsec) = WallClockTime (sh mcs cs) sec nsec
shift' x = x