diff options
Diffstat (limited to 'libraries/base/System/CPUTime.hsc')
-rw-r--r-- | libraries/base/System/CPUTime.hsc | 148 |
1 files changed, 27 insertions, 121 deletions
diff --git a/libraries/base/System/CPUTime.hsc b/libraries/base/System/CPUTime.hsc index 8d2671c49f..a6d934f491 100644 --- a/libraries/base/System/CPUTime.hsc +++ b/libraries/base/System/CPUTime.hsc @@ -1,5 +1,5 @@ {-# LANGUAGE Trustworthy #-} -{-# LANGUAGE CPP, NondecreasingIndentation, CApiFFI #-} +{-# LANGUAGE CPP, CApiFFI #-} ----------------------------------------------------------------------------- -- | @@ -18,144 +18,50 @@ #include "HsFFI.h" #include "HsBaseConfig.h" -module System.CPUTime - ( - getCPUTime, -- :: IO Integer - cpuTimePrecision -- :: Integer - ) where - -import Data.Ratio - -import Foreign -import Foreign.C - --- For struct rusage -#if !defined(mingw32_HOST_OS) && !defined(irix_HOST_OS) -# if HAVE_SYS_RESOURCE_H -# include <sys/resource.h> -# endif +-- For various _POSIX_* #defines +#if defined(HAVE_UNISTD_H) +#include <unistd.h> #endif --- For FILETIME etc. on Windows -#if HAVE_WINDOWS_H -#include <windows.h> -#endif - --- for struct tms -#if HAVE_SYS_TIMES_H -#include <sys/times.h> -#endif +module System.CPUTime + ( getCPUTime + , cpuTimePrecision + ) where -##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 +import System.IO.Unsafe (unsafePerformIO) -#if !defined(mingw32_HOST_OS) -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 -#endif +-- Here is where we decide which backend to use +#if defined(mingw32_HOST_OS) +import qualified System.CPUTime.Windows as I --- ----------------------------------------------------------------------------- --- |Computation 'getCPUTime' returns the number of picoseconds CPU time --- used by the current program. The precision of this result is --- implementation-dependent. +#elif _POSIX_TIMERS > 0 && defined(_POSIX_CPUTIME) +import qualified System.CPUTime.Posix.ClockGetTime as I -getCPUTime :: IO Integer -getCPUTime = do +#elif defined(HAVE_GETRUSAGE) && ! irix_HOST_OS && ! solaris2_HOST_OS +import qualified System.CPUTime.Posix.RUsage as I -#if !defined(mingw32_HOST_OS) --- getrusage() is right royal pain to deal with when targetting multiple +-- @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. --- -#if defined(HAVE_GETRUSAGE) && ! irix_HOST_OS && ! solaris2_HOST_OS - allocaBytes (#const sizeof(struct rusage)) $ \ p_rusage -> do - throwErrnoIfMinus1_ "getrusage" $ getrusage (#const RUSAGE_SELF) p_rusage - - let ru_utime = (#ptr struct rusage, ru_utime) p_rusage - let ru_stime = (#ptr struct rusage, ru_stime) p_rusage - u_sec <- (#peek struct timeval,tv_sec) ru_utime :: IO CTime - u_usec <- (#peek struct timeval,tv_usec) ru_utime :: IO CSUSeconds - s_sec <- (#peek struct timeval,tv_sec) ru_stime :: IO CTime - s_usec <- (#peek struct timeval,tv_usec) ru_stime :: IO CSUSeconds - 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 #elif defined(HAVE_TIMES) - allocaBytes (#const sizeof(struct tms)) $ \ p_tms -> do - _ <- times p_tms - u_ticks <- (#peek struct tms,tms_utime) p_tms :: IO CClock - s_ticks <- (#peek struct tms,tms_stime) p_tms :: IO CClock - return (( (realToInteger u_ticks + realToInteger s_ticks) * 1000000000000) - `div` fromIntegral clockTicks) +import qualified System.CPUTime.Posix.Times as I -type CTms = () -foreign import ccall unsafe times :: Ptr CTms -> IO CClock #else - ioException (IOError Nothing UnsupportedOperation - "getCPUTime" - "can't get CPU time" - Nothing) +import qualified System.CPUTime.Unsupported as I #endif -#else /* win32 */ - -- NOTE: GetProcessTimes() is only supported on NT-based OSes. - -- The counts reported by GetProcessTimes() are in 100-ns (10^-7) units. - allocaBytes (#const sizeof(FILETIME)) $ \ p_creationTime -> do - allocaBytes (#const sizeof(FILETIME)) $ \ p_exitTime -> do - allocaBytes (#const sizeof(FILETIME)) $ \ p_kernelTime -> do - allocaBytes (#const sizeof(FILETIME)) $ \ p_userTime -> do - pid <- getCurrentProcess - ok <- getProcessTimes pid p_creationTime p_exitTime p_kernelTime p_userTime - if toBool ok then do - ut <- ft2psecs p_userTime - kt <- ft2psecs p_kernelTime - return (ut + kt) - else return 0 - where - ft2psecs :: Ptr FILETIME -> IO Integer - ft2psecs ft = do - high <- (#peek FILETIME,dwHighDateTime) ft :: IO Word32 - low <- (#peek FILETIME,dwLowDateTime) ft :: IO Word32 - -- Convert 100-ns units to picosecs (10^-12) - -- => multiply by 10^5. - return (((fromIntegral high) * (2^(32::Int)) + (fromIntegral low)) * 100000) - - -- ToDo: pin down elapsed times to just the OS thread(s) that - -- are evaluating/managing Haskell code. - -type FILETIME = () -type HANDLE = () --- need proper Haskell names (initial lower-case character) -foreign import WINDOWS_CCONV unsafe "GetCurrentProcess" getCurrentProcess :: IO (Ptr HANDLE) -foreign import WINDOWS_CCONV unsafe "GetProcessTimes" getProcessTimes :: Ptr HANDLE -> Ptr FILETIME -> Ptr FILETIME -> Ptr FILETIME -> Ptr FILETIME -> IO CInt - -#endif /* not _WIN32 */ - - --- |The 'cpuTimePrecision' constant is the smallest measurable difference +-- | 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. - cpuTimePrecision :: Integer -cpuTimePrecision = round ((1000000000000::Integer) % fromIntegral (clockTicks)) - -foreign import ccall unsafe clk_tck :: CLong +cpuTimePrecision = unsafePerformIO I.getCpuTimePrecision +{-# NOINLINE cpuTimePrecision #-} -clockTicks :: Int -clockTicks = fromIntegral clk_tck +-- | 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 = I.getCPUTime |