2014-07-08 03:11:14 -04:00
# pragma once
# include "EmberDefines.h"
/// <summary>
/// Timing and CriticalSection classes.
/// </summary>
namespace EmberNs
{
/// <summary>
/// Since the algorithm is so computationally intensive, timing and benchmarking are an integral portion
/// of both the development process and the execution results. This class provides an easy way to time
/// things by simply calling its Tic() and Toc() member functions. It also assists with formatting the
/// elapsed time as a string.
/// </summary>
class EMBER_API Timing
{
public :
/// <summary>
/// Constructor that takes an optional precision argument which specifies how many digits after the decimal place should be printed for seconds.
/// As a convenience, the Tic() function is called automatically.
/// </summary>
/// <param name="precision">The precision of the seconds field of the elapsed time. Default: 2.</param>
Timing ( int precision = 2 )
{
m_Precision = precision ;
Init ( ) ;
Tic ( ) ;
}
/// <summary>
/// Set the begin time.
/// </summary>
2014-09-01 00:25:15 -04:00
/// <returns>The begin time cast to a double</returns>
2014-07-08 03:11:14 -04:00
double Tic ( )
{
2014-09-01 00:25:15 -04:00
m_BeginTime = Clock : : now ( ) ;
2014-07-08 03:11:14 -04:00
return BeginTime ( ) ;
}
2014-09-01 00:25:15 -04:00
2014-07-08 03:11:14 -04:00
/// <summary>
/// Set the end time and optionally output a string showing the elapsed time.
/// </summary>
2014-09-10 01:41:26 -04:00
/// <param name="str">The string to output. Default: nullptr.</param>
2014-07-08 03:11:14 -04:00
/// <param name="fullString">If true, output the string verbatim, else output the text " processing time: " in between str and the formatted time.</param>
/// <returns>The elapsed time in milliseconds as a double</returns>
2014-09-10 01:41:26 -04:00
double Toc ( const char * str = nullptr , bool fullString = false )
2014-07-08 03:11:14 -04:00
{
2014-09-01 00:25:15 -04:00
m_EndTime = Clock : : now ( ) ;
2014-07-08 03:11:14 -04:00
double ms = ElapsedTime ( ) ;
2014-09-10 01:41:26 -04:00
if ( str ! = nullptr )
2014-07-08 03:11:14 -04:00
{
cout < < string ( str ) < < ( fullString ? " " : " processing time: " ) < < Format ( ms ) < < endl ;
}
return ms ;
}
/// <summary>
2014-09-01 00:25:15 -04:00
/// Return the begin time as a double.
2014-07-08 03:11:14 -04:00
/// </summary>
/// <returns></returns>
2014-12-07 02:51:44 -05:00
double BeginTime ( ) { return static_cast < double > ( m_BeginTime . time_since_epoch ( ) . count ( ) ) ; }
2014-07-08 03:11:14 -04:00
/// <summary>
2014-09-01 00:25:15 -04:00
/// Return the end time as a double.
2014-07-08 03:11:14 -04:00
/// </summary>
/// <returns></returns>
2014-12-07 02:51:44 -05:00
double EndTime ( ) { return static_cast < double > ( m_EndTime . time_since_epoch ( ) . count ( ) ) ; }
2014-07-08 03:11:14 -04:00
/// <summary>
/// Return the elapsed time in milliseconds.
/// </summary>
/// <returns>The elapsed time in milliseconds as a double</returns>
2014-09-01 00:25:15 -04:00
double ElapsedTime ( )
{
duration < double > elapsed = duration_cast < milliseconds , Clock : : rep , Clock : : period > ( m_EndTime - m_BeginTime ) ;
return elapsed . count ( ) * 1000.0 ;
}
2014-07-08 03:11:14 -04:00
/// <summary>
/// Formats a specified milliseconds value as a string.
/// This uses some intelligence to determine what to return depending on how much time has elapsed.
/// Days, hours and minutes are only included if 1 or more of them has elapsed. Seconds are always
/// included as a decimal value with the precision the user specified in the constructor.
/// </summary>
/// <param name="ms">The ms</param>
/// <returns>The formatted string</returns>
string Format ( double ms )
{
stringstream ss ;
2014-09-01 00:25:15 -04:00
2014-07-08 03:11:14 -04:00
double x = ms / 1000 ;
double secs = fmod ( x , 60 ) ;
x / = 60 ;
double mins = fmod ( x , 60 ) ;
x / = 60 ;
double hours = fmod ( x , 24 ) ;
x / = 24 ;
double days = x ;
if ( days > = 1 )
2014-12-07 02:51:44 -05:00
ss < < static_cast < int > ( days ) < < " d " ;
2014-07-08 03:11:14 -04:00
if ( hours > = 1 )
2014-12-07 02:51:44 -05:00
ss < < static_cast < int > ( hours ) < < " h " ;
2014-07-08 03:11:14 -04:00
if ( mins > = 1 )
2014-12-07 02:51:44 -05:00
ss < < static_cast < int > ( mins ) < < " m " ;
2014-07-08 03:11:14 -04:00
ss < < std : : fixed < < std : : setprecision ( m_Precision ) < < secs < < " s " ;
return ss . str ( ) ;
}
/// <summary>
/// Return the number of cores in the system.
/// </summary>
/// <returns>The number of cores in the system</returns>
2014-12-11 00:50:15 -05:00
static uint ProcessorCount ( )
2014-07-08 03:11:14 -04:00
{
Init ( ) ;
return m_ProcessorCount ;
}
private :
/// <summary>
/// Query and store the performance info of the system.
/// Since it will never change it only needs to be queried once.
/// This is achieved by keeping static state and performance variables.
/// </summary>
static void Init ( )
{
if ( ! m_TimingInit )
{
2014-09-01 00:25:15 -04:00
m_ProcessorCount = thread : : hardware_concurrency ( ) ;
2014-07-08 03:11:14 -04:00
m_TimingInit = true ;
}
}
int m_Precision ; //How many digits after the decimal place to print for seconds.
2014-09-01 00:25:15 -04:00
time_point < Clock > m_BeginTime ; //The start of the timing, set with Tic().
time_point < Clock > m_EndTime ; //The end of the timing, set with Toc().
2014-07-08 03:11:14 -04:00
static bool m_TimingInit ; //Whether the performance info has bee queried.
2014-12-11 00:50:15 -05:00
static uint m_ProcessorCount ; //The number of cores on the system, set in Init().
2014-07-08 03:11:14 -04:00
} ;
/// <summary>
/// Cross platform critical section class which can be used for thread locking.
/// </summary>
class EMBER_API CriticalSection
{
public :
2014-09-01 00:25:15 -04:00
# ifdef _WIN32
2014-07-08 03:11:14 -04:00
/// <summary>
/// Constructor which initialized the underlying CRITICAL_SECTION object.
/// </summary>
CriticalSection ( ) { InitializeCriticalSection ( & m_CriticalSection ) ; }
/// <summary>
/// Constructor which initialized the underlying CRITICAL_SECTION object
/// with the specified spin count value.
/// </summary>
/// <param name="spinCount">The spin count.</param>
CriticalSection ( DWORD spinCount ) { InitializeCriticalSectionAndSpinCount ( & m_CriticalSection , spinCount ) ; }
2014-09-01 00:25:15 -04:00
2014-07-08 03:11:14 -04:00
/// <summary>
/// Deletes the underlying CRITICAL_SECTION object.
/// </summary>
~ CriticalSection ( ) { DeleteCriticalSection ( & m_CriticalSection ) ; }
/// <summary>
/// Lock the critical section.
/// </summary>
void Enter ( ) { EnterCriticalSection ( & m_CriticalSection ) ; }
/// <summary>
/// Unlock the critical section.
/// </summary>
void Leave ( ) { LeaveCriticalSection ( & m_CriticalSection ) ; }
private :
CRITICAL_SECTION m_CriticalSection ; //The Windows specific critical section object.
# else
/// <summary>
/// Constructor which initialized the underlying pthread_mutex_t object.
/// </summary>
CriticalSection ( )
{
pthread_mutexattr_t attr ;
pthread_mutexattr_settype ( & attr , PTHREAD_MUTEX_NORMAL ) ;
pthread_mutex_init ( & m_CriticalSection , & attr ) ;
pthread_mutexattr_destroy ( & attr ) ;
}
/// <summary>
/// Deletes the underlying pthread_mutex_t object.
/// </summary>
~ CriticalSection ( ) { pthread_mutex_destroy ( & m_CriticalSection ) ; }
/// <summary>
/// Lock the critical section.
/// </summary>
void Enter ( ) { pthread_mutex_lock ( & m_CriticalSection ) ; }
/// <summary>
/// Unlock the critical section.
/// </summary>
void Leave ( ) { pthread_mutex_unlock ( & m_CriticalSection ) ; }
private :
pthread_mutex_t m_CriticalSection ; //The *nix/pthread specific critical section object.
# endif
} ;
2014-09-01 00:25:15 -04:00
}