243 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			243 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| 
								 | 
							
								//  mac/chrono.cpp  --------------------------------------------------------------//
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//  Copyright Beman Dawes 2008
							 | 
						||
| 
								 | 
							
								//  Copyright 2009-2010 Vicente J. Botet Escriba
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//  Distributed under the Boost Software License, Version 1.0.
							 | 
						||
| 
								 | 
							
								//  See http://www.boost.org/LICENSE_1_0.txt
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//----------------------------------------------------------------------------//
							 | 
						||
| 
								 | 
							
								//                                 Mac                                        //
							 | 
						||
| 
								 | 
							
								//----------------------------------------------------------------------------//
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <sys/time.h> //for gettimeofday and timeval
							 | 
						||
| 
								 | 
							
								#include <mach/mach_time.h>  // mach_absolute_time, mach_timebase_info_data_t
							 | 
						||
| 
								 | 
							
								#include <boost/assert.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace boost
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								namespace chrono
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// system_clock
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// gettimeofday is the most precise "system time" available on this platform.
							 | 
						||
| 
								 | 
							
								// It returns the number of microseconds since New Years 1970 in a struct called timeval
							 | 
						||
| 
								 | 
							
								// which has a field for seconds and a field for microseconds.
							 | 
						||
| 
								 | 
							
								//    Fill in the timeval and then convert that to the time_point
							 | 
						||
| 
								 | 
							
								system_clock::time_point
							 | 
						||
| 
								 | 
							
								system_clock::now() BOOST_NOEXCEPT
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    timeval tv;
							 | 
						||
| 
								 | 
							
								    gettimeofday(&tv, 0);
							 | 
						||
| 
								 | 
							
								    return time_point(seconds(tv.tv_sec) + microseconds(tv.tv_usec));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
							 | 
						||
| 
								 | 
							
								system_clock::time_point
							 | 
						||
| 
								 | 
							
								system_clock::now(system::error_code & ec)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    timeval tv;
							 | 
						||
| 
								 | 
							
								    gettimeofday(&tv, 0);
							 | 
						||
| 
								 | 
							
								    if (!BOOST_CHRONO_IS_THROWS(ec))
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        ec.clear();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return time_point(seconds(tv.tv_sec) + microseconds(tv.tv_usec));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								// Take advantage of the fact that on this platform time_t is nothing but
							 | 
						||
| 
								 | 
							
								//    an integral count of seconds since New Years 1970 (same epoch as timeval).
							 | 
						||
| 
								 | 
							
								//    Just get the duration out of the time_point and truncate it to seconds.
							 | 
						||
| 
								 | 
							
								time_t
							 | 
						||
| 
								 | 
							
								system_clock::to_time_t(const time_point& t) BOOST_NOEXCEPT
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return time_t(duration_cast<seconds>(t.time_since_epoch()).count());
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Just turn the time_t into a count of seconds and construct a time_point with it.
							 | 
						||
| 
								 | 
							
								system_clock::time_point
							 | 
						||
| 
								 | 
							
								system_clock::from_time_t(time_t t) BOOST_NOEXCEPT
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return system_clock::time_point(seconds(t));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace chrono_detail
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// steady_clock
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Note, in this implementation steady_clock and high_resolution_clock
							 | 
						||
| 
								 | 
							
								//   are the same clock.  They are both based on mach_absolute_time().
							 | 
						||
| 
								 | 
							
								//   mach_absolute_time() * MachInfo.numer / MachInfo.denom is the number of
							 | 
						||
| 
								 | 
							
								//   nanoseconds since the computer booted up.  MachInfo.numer and MachInfo.denom
							 | 
						||
| 
								 | 
							
								//   are run time constants supplied by the OS.  This clock has no relationship
							 | 
						||
| 
								 | 
							
								//   to the Gregorian calendar.  It's main use is as a high resolution timer.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// MachInfo.numer / MachInfo.denom is often 1 on the latest equipment.  Specialize
							 | 
						||
| 
								 | 
							
								//   for that case as an optimization.
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								steady_clock::rep
							 | 
						||
| 
								 | 
							
								steady_simplified()
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return mach_absolute_time();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								steady_clock::rep
							 | 
						||
| 
								 | 
							
								steady_simplified_ec(system::error_code & ec)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    if (!BOOST_CHRONO_IS_THROWS(ec))
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        ec.clear();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return mach_absolute_time();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								double
							 | 
						||
| 
								 | 
							
								compute_steady_factor(kern_return_t& err)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    mach_timebase_info_data_t MachInfo;
							 | 
						||
| 
								 | 
							
								    err = mach_timebase_info(&MachInfo);
							 | 
						||
| 
								 | 
							
								    if ( err != 0  ) {
							 | 
						||
| 
								 | 
							
								        return 0;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return static_cast<double>(MachInfo.numer) / MachInfo.denom;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								steady_clock::rep
							 | 
						||
| 
								 | 
							
								steady_full()
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    kern_return_t err;
							 | 
						||
| 
								 | 
							
								    const double factor = chrono_detail::compute_steady_factor(err);
							 | 
						||
| 
								 | 
							
								    if (err != 0)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								      BOOST_ASSERT(0 && "Boost::Chrono - Internal Error");
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return static_cast<steady_clock::rep>(mach_absolute_time() * factor);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								steady_clock::rep
							 | 
						||
| 
								 | 
							
								steady_full_ec(system::error_code & ec)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    kern_return_t err;
							 | 
						||
| 
								 | 
							
								    const double factor = chrono_detail::compute_steady_factor(err);
							 | 
						||
| 
								 | 
							
								    if (err != 0)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (BOOST_CHRONO_IS_THROWS(ec))
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            boost::throw_exception(
							 | 
						||
| 
								 | 
							
								                    system::system_error(
							 | 
						||
| 
								 | 
							
								                            err,
							 | 
						||
| 
								 | 
							
								                            BOOST_CHRONO_SYSTEM_CATEGORY,
							 | 
						||
| 
								 | 
							
								                            "chrono::steady_clock" ));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        else
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            ec.assign( errno, BOOST_CHRONO_SYSTEM_CATEGORY );
							 | 
						||
| 
								 | 
							
								            return steady_clock::rep();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (!BOOST_CHRONO_IS_THROWS(ec))
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        ec.clear();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return static_cast<steady_clock::rep>(mach_absolute_time() * factor);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								typedef steady_clock::rep (*FP)();
							 | 
						||
| 
								 | 
							
								#if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
							 | 
						||
| 
								 | 
							
								typedef steady_clock::rep (*FP_ec)(system::error_code &);
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								FP
							 | 
						||
| 
								 | 
							
								init_steady_clock(kern_return_t & err)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    mach_timebase_info_data_t MachInfo;
							 | 
						||
| 
								 | 
							
								    err = mach_timebase_info(&MachInfo);
							 | 
						||
| 
								 | 
							
								    if ( err != 0  )
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return 0;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (MachInfo.numer == MachInfo.denom)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return &chrono_detail::steady_simplified;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return &chrono_detail::steady_full;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
							 | 
						||
| 
								 | 
							
								BOOST_CHRONO_STATIC
							 | 
						||
| 
								 | 
							
								FP_ec
							 | 
						||
| 
								 | 
							
								init_steady_clock_ec(kern_return_t & err)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    mach_timebase_info_data_t MachInfo;
							 | 
						||
| 
								 | 
							
								    err = mach_timebase_info(&MachInfo);
							 | 
						||
| 
								 | 
							
								    if ( err != 0  )
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return 0;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (MachInfo.numer == MachInfo.denom)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return &chrono_detail::steady_simplified_ec;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return &chrono_detail::steady_full_ec;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								steady_clock::time_point
							 | 
						||
| 
								 | 
							
								steady_clock::now() BOOST_NOEXCEPT
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    kern_return_t err;
							 | 
						||
| 
								 | 
							
								    chrono_detail::FP fp = chrono_detail::init_steady_clock(err);
							 | 
						||
| 
								 | 
							
								    if ( err != 0  )
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								      BOOST_ASSERT(0 && "Boost::Chrono - Internal Error");
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return time_point(duration(fp()));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
							 | 
						||
| 
								 | 
							
								steady_clock::time_point
							 | 
						||
| 
								 | 
							
								steady_clock::now(system::error_code & ec)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    kern_return_t err;
							 | 
						||
| 
								 | 
							
								    chrono_detail::FP_ec fp = chrono_detail::init_steady_clock_ec(err);
							 | 
						||
| 
								 | 
							
								    if ( err != 0  )
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (BOOST_CHRONO_IS_THROWS(ec))
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            boost::throw_exception(
							 | 
						||
| 
								 | 
							
								                    system::system_error(
							 | 
						||
| 
								 | 
							
								                            err,
							 | 
						||
| 
								 | 
							
								                            BOOST_CHRONO_SYSTEM_CATEGORY,
							 | 
						||
| 
								 | 
							
								                            "chrono::steady_clock" ));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        else
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            ec.assign( err, BOOST_CHRONO_SYSTEM_CATEGORY );
							 | 
						||
| 
								 | 
							
								            return time_point();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (!BOOST_CHRONO_IS_THROWS(ec))
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        ec.clear();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return time_point(duration(fp(ec)));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								}  // namespace chrono
							 | 
						||
| 
								 | 
							
								}  // namespace boost
							 |