/* -*- C++ -*- */ /* Heap Layers: An Extensible Memory Allocation Infrastructure Copyright (C) 2000-2020 by Emery Berger http://www.emeryberger.com emery@cs.umass.edu Heap Layers is distributed under the terms of the Apache 2.0 license. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 */ #include #include #ifndef HL_TIMEROLD_H #define HL_TIMEROLD_H /** * @class Timer * @brief A portable class for high-resolution timing. * * This class simplifies timing measurements across a number of platforms. * * @code * Timer t; * t.start(); * // do some work * t.stop(); * cout << "That took " << (double) t << " seconds." << endl; * @endcode * */ #if defined(__APPLE__) || defined(__FreeBSD__) #include #endif #if defined(__linux__) && defined(__GNUG__) && defined(__i386__) #include #include #include #include #include #include static void getTime (unsigned long& tlo, unsigned long& thi) { asm volatile ("rdtsc" : "=a"(tlo), "=d" (thi)); } static double getFrequency (void) { static double freq = 0.0; static bool initialized = false; unsigned long LTime0, LTime1, HTime0, HTime1; if (!initialized) { // Compute MHz directly. // Wait for approximately one second. getTime (LTime0, HTime0); // printf ("waiting...\n"); sleep (1); // printf ("done.\n"); getTime (LTime1, HTime1); freq = (double)(LTime1 - LTime0) + (double)(UINT_MAX)*(double)(HTime1 - HTime0); if (LTime1 < LTime0) { freq -= (double)UINT_MAX; } initialized = true; } else { // printf ("wha?\n"); } return freq; } namespace HL { class Timer { public: Timer (void) : timeElapsed (0.0) { _frequency = getFrequency(); // printf ("wooo!\n"); // printf ("freq = %lf\n", frequency); } void start (void) { getTime (currentLo, currentHi); } void stop (void) { unsigned long lo, hi; getTime (lo, hi); double now = (double) hi * 4294967296.0 + lo; double prev = (double) currentHi * 4294967296.0 + currentLo; timeElapsed = (now - prev) / _frequency; } operator double (void) { return timeElapsed; } private: double timeElapsed; unsigned long currentLo, currentHi; double _frequency; }; } #else #ifdef __SVR4 // Solaris #include #include #include #include #include #endif // __SVR4 #include #if defined(unix) || defined(__linux) #include #include #endif #ifdef __sgi #include #include #include #endif #if defined(_WIN32) #include #endif #if defined(__BEOS__) #include #endif namespace HL { class Timer { public: /// Initializes the timer. Timer (void) #if !defined(_WIN32) : _starttime (0), _elapsedtime (0) #endif { } /// Start the timer. void start (void) { _starttime = _time(); } /// Stop the timer. void stop (void) { _elapsedtime += _time() - _starttime; } /// Reset the timer. void reset (void) { _starttime = _elapsedtime; } #if 0 // Set the timer. void set (double secs) { _starttime = 0; _elapsedtime = _sectotime (secs);} #endif /// Return the number of seconds elapsed. operator double (void) { return _timetosec (_elapsedtime); } static double currentTime (void) { TimeType t; t = _time(); return _timetosec (t); } private: // The _timer variable will be different depending on the OS. // We try to use the best timer available. #ifdef __sgi #define TIMER_FOUND long _starttime, _elapsedtime; long _time (void) { struct tms t; long ticks = times (&t); return ticks; } static double _timetosec (long t) { return ((double) (t) / CLK_TCK); } static long _sectotime (double sec) { return (long) sec * CLK_TCK; } #endif #ifdef __SVR4 // Solaris #define TIMER_FOUND typedef hrtime_t TimeType; TimeType _starttime, _elapsedtime; static TimeType _time (void) { return gethrtime(); } static TimeType _sectotime (double sec) { return (hrtime_t) (sec * 1.0e9); } static double _timetosec (TimeType& t) { return ((double) (t) / 1.0e9); } #endif // __SVR4 #if defined(MAC) || defined(macintosh) #define TIMER_FOUND double _starttime, _elapsedtime; double _time (void) { return get_Mac_microseconds(); } double _timetosec (hrtime_t& t) { return t; } #endif // MAC #ifdef _WIN32 #define TIMER_FOUND #ifndef __GNUC__ class TimeType { public: TimeType (void) { largeInt.QuadPart = 0; } operator double& (void) { return (double&) largeInt.QuadPart; } operator LARGE_INTEGER& (void) { return largeInt; } double timeToSec (void) { return (double) largeInt.QuadPart / getFreq(); } private: double getFreq (void) { QueryPerformanceFrequency (&freq); return (double) freq.QuadPart; } LARGE_INTEGER largeInt; LARGE_INTEGER freq; }; TimeType _starttime, _elapsedtime; static TimeType _time (void) { TimeType t; int r = QueryPerformanceCounter (&((LARGE_INTEGER&) t)); assert (r); return t; } static double _timetosec (TimeType& t) { return t.timeToSec(); } #else typedef DWORD TimeType; DWORD _starttime, _elapsedtime; static DWORD _time (void) { return GetTickCount(); } static double _timetosec (DWORD& t) { return (double) t / 100000.0; } static unsigned long _sectotime (double sec) { return (unsigned long)(sec); } #endif #endif // _WIN32 #ifdef __BEOS__ #define TIMER_FOUND bigtime_t _starttime, _elapsedtime; bigtime_t _time(void) { return system_time(); } double _timetosec (bigtime_t& t) { return (double) t / 1000000.0; } bigtime_t _sectotime (double sec) { return (bigtime_t)(sec * 1000000.0); } #endif // __BEOS__ #ifndef TIMER_FOUND typedef long TimeType; TimeType _starttime, _elapsedtime; static TimeType _time (void) { struct timeval t; gettimeofday (&t, NULL); return t.tv_sec * 1000000 + t.tv_usec; } static double _timetosec (TimeType t) { return ((double) (t) / 1000000.0); } static TimeType _sectotime (double sec) { return (TimeType) (sec * 1000000.0); } #endif // TIMER_FOUND #undef TIMER_FOUND }; #ifdef __SVR4 // Solaris class VirtualTimer : public Timer { public: hrtime_t _time (void) { return gethrvtime(); } }; #endif } #endif #endif