Qt Reference Documentation

ClockCycleCounter.cpp Example File

demos/spectrum/3rdparty/fftreal/stopwatch/ClockCycleCounter.cpp
 /*****************************************************************************

         ClockCycleCounter.cpp
         Copyright (c) 2003 Laurent de Soras

 --- Legal stuff ---

 This library is free software; you can redistribute it and/or
 modify it under the terms of the GNU Lesser General Public
 License as published by the Free Software Foundation; either
 version 2.1 of the License, or (at your option) any later version.

 This library is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 Lesser General Public License for more details.

 You should have received a copy of the GNU Lesser General Public
 License along with this library; if not, write to the Free Software
 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA

 *Tab=3***********************************************************************/

 #if defined (_MSC_VER)
         #pragma warning (1 : 4130) // "'operator' : logical operation on address of string constant"
         #pragma warning (1 : 4223) // "nonstandard extension used : non-lvalue array converted to pointer"
         #pragma warning (1 : 4705) // "statement has no effect"
         #pragma warning (1 : 4706) // "assignment within conditional expression"
         #pragma warning (4 : 4786) // "identifier was truncated to '255' characters in the debug information"
         #pragma warning (4 : 4800) // "forcing value to bool 'true' or 'false' (performance warning)"
         #pragma warning (4 : 4355) // "'this' : used in base member initializer list"
 #endif

 /*\\\ INCLUDE FILES \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\*/

 #include        "ClockCycleCounter.h"

 #include        <cassert>

 namespace stopwatch
 {

 /*\\\ PUBLIC \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\*/

 /*
 ==============================================================================
 Name: ctor
 Description:
         The first object constructed initialise global data. This first
         construction may be a bit slow.
 Throws: Nothing
 ==============================================================================
 */

 ClockCycleCounter::ClockCycleCounter ()
 :       _start_time (0)
 ,       _state (0)
 ,       _best_score (-1)
 {
         if (! _init_flag)
         {
                 // Should be executed in this order
                 compute_clk_mul ();
                 compute_measure_time_total ();
                 compute_measure_time_lap ();

                 // Restores object state
                 _start_time = 0;
                 _state      = 0;
                 _best_score = -1;

                 _init_flag = true;
         }
 }

 /*
 ==============================================================================
 Name: get_time_total
 Description:
         Gives the time elapsed between the latest stop_lap() and start() calls.
 Returns:
         The duration, in clock cycles.
 Throws: Nothing
 ==============================================================================
 */

 Int64   ClockCycleCounter::get_time_total () const
 {
         const Int64             duration = _state - _start_time;
         assert (duration >= 0);

         const Int64             t = max (
                 duration - _measure_time_total,
                 static_cast <Int64> (0)
         );

         return (t * _clk_mul);
 }

 /*
 ==============================================================================
 Name: get_time_best_lap
 Description:
         Gives the smallest time between two consecutive stop_lap() or between
         the stop_lap() and start(). The value is reset by a call to start().
         Call this function only after a stop_lap().
         The time is amputed from the duration of the stop_lap() call itself.
 Returns:
         The smallest duration, in clock cycles.
 Throws: Nothing
 ==============================================================================
 */

 Int64   ClockCycleCounter::get_time_best_lap () const
 {
         assert (_best_score >= 0);

         const Int64             t1 = max (
                 _best_score - _measure_time_lap,
                 static_cast <Int64> (0)
         );
         const Int64             t = min (t1, get_time_total ());

         return (t * _clk_mul);
 }

 /*\\\ PROTECTED \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\*/

 /*\\\ PRIVATE \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\*/

 #if defined (__MACOS__)

 static inline double    stopwatch_ClockCycleCounter_get_time_s ()
 {
         const Nanoseconds       ns = AbsoluteToNanoseconds (UpTime ());

         return (ns.hi * 4294967296e-9 + ns.lo * 1e-9);
 }

 #endif  // __MACOS__

 /*
 ==============================================================================
 Name: compute_clk_mul
 Description:
         This function, only for PowerPC/MacOS computers, computes the multiplier
         required to deduce clock cycles from the internal counter.
 Throws: Nothing
 ==============================================================================
 */

 void    ClockCycleCounter::compute_clk_mul ()
 {
         assert (! _init_flag);

 #if defined (__MACOS__)

         long                            clk_speed_mhz = CurrentProcessorSpeed ();
         const Int64             clk_speed =
                 static_cast <Int64> (clk_speed_mhz) * (1000L*1000L);

         const double    start_time_s = stopwatch_ClockCycleCounter_get_time_s ();
         start ();

         const double    duration = 0.01;        // Seconds
         while (stopwatch_ClockCycleCounter_get_time_s () - start_time_s < duration)
         {
                 continue;
         }

         const double    stop_time_s = stopwatch_ClockCycleCounter_get_time_s ();
         stop ();

         const double    diff_time_s = stop_time_s - start_time_s;
         const double    nbr_cycles = diff_time_s * static_cast <double> (clk_speed);

         const Int64             diff_time_c = _state - _start_time;
         const double    clk_mul = nbr_cycles / static_cast <double> (diff_time_c);

         _clk_mul = round_int (clk_mul);

 #endif  // __MACOS__
 }

 void    ClockCycleCounter::compute_measure_time_total ()
 {
         start ();
         spend_time ();

         Int64                           best_result = 0x7FFFFFFFL;      // Should be enough
         long                            nbr_tests = 100;
         for (long cnt = 0; cnt < nbr_tests; ++cnt)
         {
                 start ();
                 stop_lap ();
                 const Int64             duration = _state - _start_time;
                 best_result = min (best_result, duration);
         }

         _measure_time_total = best_result;
 }

 /*
 ==============================================================================
 Name: compute_measure_time_lap
 Description:
         Computes the duration of one stop_lap() call and store it. It will be used
         later to get the real duration of the measured operation (by subtracting
         the measurement duration).
 Throws: Nothing
 ==============================================================================
 */

 void    ClockCycleCounter::compute_measure_time_lap ()
 {
         start ();
         spend_time ();

         long                            nbr_tests = 10;
         for (long cnt = 0; cnt < nbr_tests; ++cnt)
         {
                 stop_lap ();
                 stop_lap ();
                 stop_lap ();
                 stop_lap ();
         }

         _measure_time_lap = _best_score;
 }

 void    ClockCycleCounter::spend_time ()
 {
         const Int64             nbr_clocks = 500;       // Number of clock cycles to spend

         const Int64             start = read_clock_counter ();
         Int64                           current;

         do
         {
                 current = read_clock_counter ();
         }
         while ((current - start) * _clk_mul < nbr_clocks);
 }

 Int64   ClockCycleCounter::_measure_time_total = 0;
 Int64   ClockCycleCounter::_measure_time_lap = 0;
 int     ClockCycleCounter::_clk_mul = 1;
 bool    ClockCycleCounter::_init_flag = false;

 }       // namespace stopwatch

 /*\\\ EOF \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\*/