C++ Cross-Platform High-Resolution Timer

前端 未结 14 2173
不知归路
不知归路 2020-11-22 12:45

I\'m looking to implement a simple timer mechanism in C++. The code should work in Windows and Linux. The resolution should be as precise as possible (at least millisecond a

相关标签:
14条回答
  • 2020-11-22 13:41

    The ACE library has portable high resolution timers also.

    Doxygen for high res timer:
    http://www.dre.vanderbilt.edu/Doxygen/5.7.2/html/ace/a00244.html

    0 讨论(0)
  • 2020-11-22 13:45

    Late to the party here, but I'm working in a legacy codebase that can't be upgraded to c++11 yet. Nobody on our team is very skilled in c++, so adding a library like STL is proving difficult (on top of potential concerns others have raised about deployment issues). I really needed an extremely simple cross platform timer that could live by itself without anything beyond bare-bones standard system libraries. Here's what I found:

    http://www.songho.ca/misc/timer/timer.html

    Reposting the entire source here just so it doesn't get lost if the site ever dies:

        //////////////////////////////////////////////////////////////////////////////
    // Timer.cpp
    // =========
    // High Resolution Timer.
    // This timer is able to measure the elapsed time with 1 micro-second accuracy
    // in both Windows, Linux and Unix system 
    //
    //  AUTHOR: Song Ho Ahn (song.ahn@gmail.com) - http://www.songho.ca/misc/timer/timer.html
    // CREATED: 2003-01-13
    // UPDATED: 2017-03-30
    //
    // Copyright (c) 2003 Song Ho Ahn
    //////////////////////////////////////////////////////////////////////////////
    
    #include "Timer.h"
    #include <stdlib.h>
    
    ///////////////////////////////////////////////////////////////////////////////
    // constructor
    ///////////////////////////////////////////////////////////////////////////////
    Timer::Timer()
    {
    #if defined(WIN32) || defined(_WIN32)
        QueryPerformanceFrequency(&frequency);
        startCount.QuadPart = 0;
        endCount.QuadPart = 0;
    #else
        startCount.tv_sec = startCount.tv_usec = 0;
        endCount.tv_sec = endCount.tv_usec = 0;
    #endif
    
        stopped = 0;
        startTimeInMicroSec = 0;
        endTimeInMicroSec = 0;
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // distructor
    ///////////////////////////////////////////////////////////////////////////////
    Timer::~Timer()
    {
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // start timer.
    // startCount will be set at this point.
    ///////////////////////////////////////////////////////////////////////////////
    void Timer::start()
    {
        stopped = 0; // reset stop flag
    #if defined(WIN32) || defined(_WIN32)
        QueryPerformanceCounter(&startCount);
    #else
        gettimeofday(&startCount, NULL);
    #endif
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // stop the timer.
    // endCount will be set at this point.
    ///////////////////////////////////////////////////////////////////////////////
    void Timer::stop()
    {
        stopped = 1; // set timer stopped flag
    
    #if defined(WIN32) || defined(_WIN32)
        QueryPerformanceCounter(&endCount);
    #else
        gettimeofday(&endCount, NULL);
    #endif
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // compute elapsed time in micro-second resolution.
    // other getElapsedTime will call this first, then convert to correspond resolution.
    ///////////////////////////////////////////////////////////////////////////////
    double Timer::getElapsedTimeInMicroSec()
    {
    #if defined(WIN32) || defined(_WIN32)
        if(!stopped)
            QueryPerformanceCounter(&endCount);
    
        startTimeInMicroSec = startCount.QuadPart * (1000000.0 / frequency.QuadPart);
        endTimeInMicroSec = endCount.QuadPart * (1000000.0 / frequency.QuadPart);
    #else
        if(!stopped)
            gettimeofday(&endCount, NULL);
    
        startTimeInMicroSec = (startCount.tv_sec * 1000000.0) + startCount.tv_usec;
        endTimeInMicroSec = (endCount.tv_sec * 1000000.0) + endCount.tv_usec;
    #endif
    
        return endTimeInMicroSec - startTimeInMicroSec;
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // divide elapsedTimeInMicroSec by 1000
    ///////////////////////////////////////////////////////////////////////////////
    double Timer::getElapsedTimeInMilliSec()
    {
        return this->getElapsedTimeInMicroSec() * 0.001;
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // divide elapsedTimeInMicroSec by 1000000
    ///////////////////////////////////////////////////////////////////////////////
    double Timer::getElapsedTimeInSec()
    {
        return this->getElapsedTimeInMicroSec() * 0.000001;
    }
    
    
    
    ///////////////////////////////////////////////////////////////////////////////
    // same as getElapsedTimeInSec()
    ///////////////////////////////////////////////////////////////////////////////
    double Timer::getElapsedTime()
    {
        return this->getElapsedTimeInSec();
    }
    

    and the header file:

    //////////////////////////////////////////////////////////////////////////////
    // Timer.h
    // =======
    // High Resolution Timer.
    // This timer is able to measure the elapsed time with 1 micro-second accuracy
    // in both Windows, Linux and Unix system 
    //
    //  AUTHOR: Song Ho Ahn (song.ahn@gmail.com) - http://www.songho.ca/misc/timer/timer.html
    // CREATED: 2003-01-13
    // UPDATED: 2017-03-30
    //
    // Copyright (c) 2003 Song Ho Ahn
    //////////////////////////////////////////////////////////////////////////////
    
    #ifndef TIMER_H_DEF
    #define TIMER_H_DEF
    
    #if defined(WIN32) || defined(_WIN32)   // Windows system specific
    #include <windows.h>
    #else          // Unix based system specific
    #include <sys/time.h>
    #endif
    
    
    class Timer
    {
    public:
        Timer();                                    // default constructor
        ~Timer();                                   // default destructor
    
        void   start();                             // start timer
        void   stop();                              // stop the timer
        double getElapsedTime();                    // get elapsed time in second
        double getElapsedTimeInSec();               // get elapsed time in second (same as getElapsedTime)
        double getElapsedTimeInMilliSec();          // get elapsed time in milli-second
        double getElapsedTimeInMicroSec();          // get elapsed time in micro-second
    
    
    protected:
    
    
    private:
        double startTimeInMicroSec;                 // starting time in micro-second
        double endTimeInMicroSec;                   // ending time in micro-second
        int    stopped;                             // stop flag 
    #if defined(WIN32) || defined(_WIN32)
        LARGE_INTEGER frequency;                    // ticks per second
        LARGE_INTEGER startCount;                   //
        LARGE_INTEGER endCount;                     //
    #else
        timeval startCount;                         //
        timeval endCount;                           //
    #endif
    };
    
    #endif // TIMER_H_DEF
    
    0 讨论(0)
  • 2020-11-22 13:47

    SDL2 has an excellent cross-platform high-resolution timer. If however you need sub-millisecond accuracy, I wrote a very small cross-platform timer library here. It is compatible with both C++03 and C++11/higher versions of C++.

    0 讨论(0)
  • 2020-11-22 13:48

    The first answer to C++ library questions is generally BOOST: http://www.boost.org/doc/libs/1_40_0/libs/timer/timer.htm. Does this do what you want? Probably not but it's a start.

    The problem is you want portable and timer functions are not universal in OSes.

    0 讨论(0)
  • 2020-11-22 13:51

    For C++03:

    Boost.Timer might work, but it depends on the C function clock and so may not have good enough resolution for you.

    Boost.Date_Time includes a ptime class that's been recommended on Stack Overflow before. See its docs on microsec_clock::local_time and microsec_clock::universal_time, but note its caveat that "Win32 systems often do not achieve microsecond resolution via this API."

    STLsoft provides, among other things, thin cross-platform (Windows and Linux/Unix) C++ wrappers around OS-specific APIs. Its performance library has several classes that would do what you need. (To make it cross platform, pick a class like performance_counter that exists in both the winstl and unixstl namespaces, then use whichever namespace matches your platform.)

    For C++11 and above:

    The std::chrono library has this functionality built in. See this answer by @HowardHinnant for details.

    0 讨论(0)
  • 2020-11-22 13:52

    Updated answer for an old question:

    In C++11 you can portably get to the highest resolution timer with:

    #include <iostream>
    #include <chrono>
    #include "chrono_io"
    
    int main()
    {
        typedef std::chrono::high_resolution_clock Clock;
        auto t1 = Clock::now();
        auto t2 = Clock::now();
        std::cout << t2-t1 << '\n';
    }
    

    Example output:

    74 nanoseconds
    

    "chrono_io" is an extension to ease I/O issues with these new types and is freely available here.

    There is also an implementation of <chrono> available in boost (might still be on tip-of-trunk, not sure it has been released).

    Update

    This is in response to Ben's comment below that subsequent calls to std::chrono::high_resolution_clock take several milliseconds in VS11. Below is a <chrono>-compatible workaround. However it only works on Intel hardware, you need to dip into inline assembly (syntax to do that varies with compiler), and you have to hardwire the machine's clock speed into the clock:

    #include <chrono>
    
    struct clock
    {
        typedef unsigned long long                 rep;
        typedef std::ratio<1, 2800000000>          period; // My machine is 2.8 GHz
        typedef std::chrono::duration<rep, period> duration;
        typedef std::chrono::time_point<clock>     time_point;
        static const bool is_steady =              true;
    
        static time_point now() noexcept
        {
            unsigned lo, hi;
            asm volatile("rdtsc" : "=a" (lo), "=d" (hi));
            return time_point(duration(static_cast<rep>(hi) << 32 | lo));
        }
    
    private:
    
        static
        unsigned
        get_clock_speed()
        {
            int mib[] = {CTL_HW, HW_CPU_FREQ};
            const std::size_t namelen = sizeof(mib)/sizeof(mib[0]);
            unsigned freq;
            size_t freq_len = sizeof(freq);
            if (sysctl(mib, namelen, &freq, &freq_len, nullptr, 0) != 0)
                return 0;
            return freq;
        }
    
        static
        bool
        check_invariants()
        {
            static_assert(1 == period::num, "period must be 1/freq");
            assert(get_clock_speed() == period::den);
            static_assert(std::is_same<rep, duration::rep>::value,
                          "rep and duration::rep must be the same type");
            static_assert(std::is_same<period, duration::period>::value,
                          "period and duration::period must be the same type");
            static_assert(std::is_same<duration, time_point::duration>::value,
                          "duration and time_point::duration must be the same type");
            return true;
        }
    
        static const bool invariants;
    };
    
    const bool clock::invariants = clock::check_invariants();
    

    So it isn't portable. But if you want to experiment with a high resolution clock on your own intel hardware, it doesn't get finer than this. Though be forewarned, today's clock speeds can dynamically change (they aren't really a compile-time constant). And with a multiprocessor machine you can even get time stamps from different processors. But still, experiments on my hardware work fairly well. If you're stuck with millisecond resolution, this could be a workaround.

    This clock has a duration in terms of your cpu's clock speed (as you reported it). I.e. for me this clock ticks once every 1/2,800,000,000 of a second. If you want to, you can convert this to nanoseconds (for example) with:

    using std::chrono::nanoseconds;
    using std::chrono::duration_cast;
    auto t0 = clock::now();
    auto t1 = clock::now();
    nanoseconds ns = duration_cast<nanoseconds>(t1-t0);
    

    The conversion will truncate fractions of a cpu cycle to form the nanosecond. Other rounding modes are possible, but that's a different topic.

    For me this will return a duration as low as 18 clock ticks, which truncates to 6 nanoseconds.

    I've added some "invariant checking" to the above clock, the most important of which is checking that the clock::period is correct for the machine. Again, this is not portable code, but if you're using this clock, you've already committed to that. The private get_clock_speed() function shown here gets the maximum cpu frequency on OS X, and that should be the same number as the constant denominator of clock::period.

    Adding this will save you a little debugging time when you port this code to your new machine and forget to update the clock::period to the speed of your new machine. All of the checking is done either at compile-time or at program startup time. So it won't impact the performance of clock::now() in the least.

    0 讨论(0)
提交回复
热议问题