{-# LINE 1 "libraries/base/System/CPUTime.hsc" #-}
{-# LANGUAGE Trustworthy #-}
{-# LINE 2 "libraries/base/System/CPUTime.hsc" #-}
{-# LANGUAGE CPP, NondecreasingIndentation, ForeignFunctionInterface, CApiFFI #-}

-----------------------------------------------------------------------------
-- |
-- Module      :  System.CPUTime
-- Copyright   :  (c) The University of Glasgow 2001
-- License     :  BSD-style (see the file libraries/base/LICENSE)
-- 
-- Maintainer  :  libraries@haskell.org
-- Stability   :  provisional
-- Portability :  portable
--
-- The standard CPUTime library.
--
-----------------------------------------------------------------------------


{-# LINE 19 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 20 "libraries/base/System/CPUTime.hsc" #-}

module System.CPUTime 
        (
         getCPUTime,       -- :: IO Integer
         cpuTimePrecision  -- :: Integer
        ) where

import Prelude

import Data.Ratio


{-# LINE 34 "libraries/base/System/CPUTime.hsc" #-}


{-# LINE 38 "libraries/base/System/CPUTime.hsc" #-}


{-# LINE 40 "libraries/base/System/CPUTime.hsc" #-}
import Foreign.Safe
import Foreign.C

{-# LINE 43 "libraries/base/System/CPUTime.hsc" #-}
import System.IO.Unsafe (unsafePerformIO)

{-# LINE 45 "libraries/base/System/CPUTime.hsc" #-}

-- For _SC_CLK_TCK

{-# LINE 48 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 49 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 50 "libraries/base/System/CPUTime.hsc" #-}

-- For struct rusage

{-# LINE 53 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 54 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 55 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 56 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 57 "libraries/base/System/CPUTime.hsc" #-}

-- For FILETIME etc. on Windows

{-# LINE 62 "libraries/base/System/CPUTime.hsc" #-}

-- for CLK_TCK

{-# LINE 65 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 66 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 67 "libraries/base/System/CPUTime.hsc" #-}

-- for struct tms

{-# LINE 70 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 71 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 72 "libraries/base/System/CPUTime.hsc" #-}


{-# LINE 74 "libraries/base/System/CPUTime.hsc" #-}

#ifdef mingw32_HOST_OS
# if defined(i386_HOST_ARCH)
#  define WINDOWS_CCONV stdcall
# elif defined(x86_64_HOST_ARCH)
#  define WINDOWS_CCONV ccall
# else
#  error Unknown mingw32 arch
# endif
#else
#endif


{-# LINE 87 "libraries/base/System/CPUTime.hsc" #-}
realToInteger :: Real a => a -> Integer
realToInteger ct = round (realToFrac ct :: Double)
  -- CTime, CClock, CUShort etc are in Real but not Fractional, 
  -- so we must convert to Double before we can round it

{-# LINE 92 "libraries/base/System/CPUTime.hsc" #-}


{-# LINE 94 "libraries/base/System/CPUTime.hsc" #-}
-- -----------------------------------------------------------------------------
-- |Computation 'getCPUTime' returns the number of picoseconds CPU time
-- used by the current program.  The precision of this result is
-- implementation-dependent.

getCPUTime :: IO Integer
getCPUTime = do


{-# LINE 103 "libraries/base/System/CPUTime.hsc" #-}
-- getrusage() is right royal pain to deal with when targetting multiple
-- versions of Solaris, since some versions supply it in libc (2.3 and 2.5),
-- while 2.4 has got it in libucb (I wouldn't be too surprised if it was back
-- again in libucb in 2.6..)
--
-- Avoid the problem by resorting to times() instead.
--

{-# LINE 111 "libraries/base/System/CPUTime.hsc" #-}
    allocaBytes (144) $ \ p_rusage -> do
{-# LINE 112 "libraries/base/System/CPUTime.hsc" #-}
    throwErrnoIfMinus1_ "getrusage" $ getrusage (0) p_rusage
{-# LINE 113 "libraries/base/System/CPUTime.hsc" #-}

    let ru_utime = ((\hsc_ptr -> hsc_ptr `plusPtr` 0)) p_rusage
{-# LINE 115 "libraries/base/System/CPUTime.hsc" #-}
    let ru_stime = ((\hsc_ptr -> hsc_ptr `plusPtr` 16)) p_rusage
{-# LINE 116 "libraries/base/System/CPUTime.hsc" #-}
    u_sec  <- ((\hsc_ptr -> peekByteOff hsc_ptr 0))  ru_utime :: IO CTime
{-# LINE 117 "libraries/base/System/CPUTime.hsc" #-}
    u_usec <- ((\hsc_ptr -> peekByteOff hsc_ptr 8)) ru_utime :: IO CSUSeconds
{-# LINE 118 "libraries/base/System/CPUTime.hsc" #-}
    s_sec  <- ((\hsc_ptr -> peekByteOff hsc_ptr 0))  ru_stime :: IO CTime
{-# LINE 119 "libraries/base/System/CPUTime.hsc" #-}
    s_usec <- ((\hsc_ptr -> peekByteOff hsc_ptr 8)) ru_stime :: IO CSUSeconds
{-# LINE 120 "libraries/base/System/CPUTime.hsc" #-}
    return ((realToInteger u_sec * 1000000 + realToInteger u_usec + 
             realToInteger s_sec * 1000000 + realToInteger s_usec) 
                * 1000000)

type CRUsage = ()
foreign import capi unsafe "HsBase.h getrusage" getrusage :: CInt -> Ptr CRUsage -> IO CInt

{-# LINE 142 "libraries/base/System/CPUTime.hsc" #-}


{-# LINE 176 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 177 "libraries/base/System/CPUTime.hsc" #-}

-- |The 'cpuTimePrecision' constant is the smallest measurable difference
-- in CPU time that the implementation can record, and is given as an
-- integral number of picoseconds.


{-# LINE 183 "libraries/base/System/CPUTime.hsc" #-}
cpuTimePrecision :: Integer
cpuTimePrecision = round ((1000000000000::Integer) % fromIntegral (clockTicks))

{-# LINE 186 "libraries/base/System/CPUTime.hsc" #-}


{-# LINE 188 "libraries/base/System/CPUTime.hsc" #-}
clockTicks :: Int
clockTicks =

{-# LINE 193 "libraries/base/System/CPUTime.hsc" #-}
    unsafePerformIO (sysconf (2) >>= return . fromIntegral)
{-# LINE 194 "libraries/base/System/CPUTime.hsc" #-}
foreign import ccall unsafe sysconf :: CInt -> IO CLong

{-# LINE 196 "libraries/base/System/CPUTime.hsc" #-}

{-# LINE 197 "libraries/base/System/CPUTime.hsc" #-}