{-# LINE 1 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
{-# LANGUAGE CPP, ForeignFunctionInterface #-}
{-# LINE 2 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
module Trace.Hpc.Reflect
  ( clearTix
  , examineTix
  , updateTix
  ) where

import Trace.Hpc.Tix


{-# LINE 25 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}

import Foreign.C.String
import Foreign.Marshal.Array
import Foreign.Ptr
import Foreign.Storable ( Storable(..) )
import Data.Word 
import Trace.Hpc.Util
import System.IO.Unsafe


{-# LINE 35 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}

foreign import ccall unsafe hs_hpc_rootModule :: IO (Ptr ())

modInfo :: [ModuleInfo]
modInfo = unsafePerformIO $ do
      ptr <- hs_hpc_rootModule 
      moduleInfoList ptr

data ModuleInfo = ModuleInfo String Word32 Hash (Ptr Word64) 

moduleInfoList :: Ptr () -> IO [ModuleInfo]
moduleInfoList ptr
  | ptr == nullPtr = return []
  | otherwise = do
        cModName  <- ((\hsc_ptr -> peekByteOff hsc_ptr 0)) ptr
{-# LINE 50 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
        modName   <- peekCString cModName
        tickCount <- ((\hsc_ptr -> peekByteOff hsc_ptr 4)) ptr
{-# LINE 52 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
        hashNo    <- ((\hsc_ptr -> peekByteOff hsc_ptr 12)) ptr
{-# LINE 53 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
        tixArr    <- ((\hsc_ptr -> peekByteOff hsc_ptr 16)) ptr
{-# LINE 54 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
	next      <- ((\hsc_ptr -> peekByteOff hsc_ptr 20)) ptr
{-# LINE 55 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}
        rest      <- moduleInfoList next
        return $ ModuleInfo modName tickCount (toHash (hashNo :: Int)) tixArr : rest

clearTix :: IO ()
clearTix = do
      sequence_ [ pokeArray ptr $ take (fromIntegral count) $ repeat 0
      	      	| ModuleInfo _mod count _hash ptr <- modInfo
		]
      return ()


examineTix :: IO Tix
examineTix = do
      mods <- sequence [ do tixs <- peekArray (fromIntegral count) ptr
      	      	       	    return $ TixModule mod' hash (fromIntegral count)
			    	   $ map fromIntegral tixs
      	      	       | (ModuleInfo mod' count hash ptr) <- modInfo
		       ]
      return $ Tix mods

-- requirement that the tix be of the same shape as the 
-- internal tix.
updateTix :: Tix -> IO ()
updateTix (Tix modTixes) 
  | length modTixes /= length modInfo = error "updateTix failed"
  | otherwise = do
      sequence_ [ pokeArray ptr $ map fromIntegral tixs
      	      	| (ModuleInfo mod1 count1 hash1 ptr,
		   TixModule mod2 hash2 count2 tixs) <- zip modInfo modTixes
		, if mod1 /= mod2 
		|| (fromIntegral count1) /= count2 
		|| hash1 /= hash2
		|| length tixs /= count2
		  then error "updateTix failed"
		  else True
		]
      return ()


{-# LINE 94 "libraries/hpc/./Trace/Hpc/Reflect.hsc" #-}