| 
									
										
										
										
											2013-08-07 18:56:35 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  Created by Phil on 05/08/2013. | 
					
						
							|  |  |  |  *  Copyright 2013 Two Blue Cubes Ltd. All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Distributed under the Boost Software License, Version 1.0. (See accompanying | 
					
						
							|  |  |  |  *  file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "catch_timer.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-25 12:24:34 +01:00
										 |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2013-08-07 18:56:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Catch { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 12:22:16 +01:00
										 |  |  |     auto getCurrentNanosecondsSinceEpoch() -> uint64_t { | 
					
						
							|  |  |  |         return std::chrono::duration_cast<std::chrono::nanoseconds>( std::chrono::high_resolution_clock::now().time_since_epoch() ).count(); | 
					
						
							| 
									
										
										
										
											2013-08-07 18:56:35 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 21:31:28 +01:00
										 |  |  |     auto estimateClockResolution() -> uint64_t { | 
					
						
							| 
									
										
										
										
											2017-08-04 19:23:30 +01:00
										 |  |  |         uint64_t sum = 0; | 
					
						
							|  |  |  |         static const uint64_t iterations = 1000000; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 17:13:17 +01:00
										 |  |  |         for( std::size_t i = 0; i < iterations; ++i ) { | 
					
						
							| 
									
										
										
										
											2017-08-04 19:23:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             uint64_t ticks; | 
					
						
							|  |  |  |             uint64_t baseTicks = getCurrentNanosecondsSinceEpoch(); | 
					
						
							|  |  |  |             do { | 
					
						
							|  |  |  |                 ticks = getCurrentNanosecondsSinceEpoch(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             while( ticks == baseTicks ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto delta = ticks - baseTicks; | 
					
						
							|  |  |  |             sum += delta; | 
					
						
							| 
									
										
										
										
											2017-08-04 13:53:47 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 19:23:30 +01:00
										 |  |  |         // We're just taking the mean, here. To do better we could take the std. dev and exclude outliers
 | 
					
						
							|  |  |  |         // - and potentially do more iterations if there's a high variance.
 | 
					
						
							| 
									
										
										
										
											2017-08-04 21:31:28 +01:00
										 |  |  |         return sum/iterations; | 
					
						
							| 
									
										
										
										
											2017-08-04 19:23:30 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-04 21:31:28 +01:00
										 |  |  |     auto getEstimatedClockResolution() -> uint64_t { | 
					
						
							| 
									
										
										
										
											2017-08-04 19:23:30 +01:00
										 |  |  |         static auto s_resolution = estimateClockResolution(); | 
					
						
							|  |  |  |         return s_resolution; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Timer::start() { | 
					
						
							|  |  |  |        m_nanoseconds = getCurrentNanosecondsSinceEpoch(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     auto Timer::getElapsedNanoseconds() const -> unsigned int { | 
					
						
							|  |  |  |         return static_cast<unsigned int>(getCurrentNanosecondsSinceEpoch() - m_nanoseconds); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     auto Timer::getElapsedMicroseconds() const -> unsigned int { | 
					
						
							|  |  |  |         return static_cast<unsigned int>(getElapsedNanoseconds()/1000); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     auto Timer::getElapsedMilliseconds() const -> unsigned int { | 
					
						
							|  |  |  |         return static_cast<unsigned int>(getElapsedMicroseconds()/1000); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     auto Timer::getElapsedSeconds() const -> double { | 
					
						
							|  |  |  |         return getElapsedMicroseconds()/1000000.0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 13:53:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-07 18:56:35 +01:00
										 |  |  | } // namespace Catch
 |