summaryrefslogtreecommitdiff
path: root/libraries/base/System/CPUTime.hsc
diff options
context:
space:
mode:
Diffstat (limited to 'libraries/base/System/CPUTime.hsc')
-rw-r--r--libraries/base/System/CPUTime.hsc148
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