1 // mac/chrono.cpp --------------------------------------------------------------//
3 // Copyright Beman Dawes 2008
4 // Copyright 2009-2010 Vicente J. Botet Escriba
6 // Distributed under the Boost Software License, Version 1.0.
7 // See http://www.boost.org/LICENSE_1_0.txt
10 //----------------------------------------------------------------------------//
12 //----------------------------------------------------------------------------//
14 #include <sys/time.h> //for gettimeofday and timeval
15 #include <mach/mach_time.h> // mach_absolute_time, mach_timebase_info_data_t
16 #include <boost/assert.hpp>
25 // gettimeofday is the most precise "system time" available on this platform.
26 // It returns the number of microseconds since New Years 1970 in a struct called timeval
27 // which has a field for seconds and a field for microseconds.
28 // Fill in the timeval and then convert that to the time_point
29 system_clock::time_point
30 system_clock::now() BOOST_NOEXCEPT
34 return time_point(seconds(tv.tv_sec) + microseconds(tv.tv_usec));
37 #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
38 system_clock::time_point
39 system_clock::now(system::error_code & ec)
43 if (!::boost::chrono::is_throws(ec))
47 return time_point(seconds(tv.tv_sec) + microseconds(tv.tv_usec));
50 // Take advantage of the fact that on this platform time_t is nothing but
51 // an integral count of seconds since New Years 1970 (same epoch as timeval).
52 // Just get the duration out of the time_point and truncate it to seconds.
54 system_clock::to_time_t(const time_point& t) BOOST_NOEXCEPT
56 return time_t(duration_cast<seconds>(t.time_since_epoch()).count());
59 // Just turn the time_t into a count of seconds and construct a time_point with it.
60 system_clock::time_point
61 system_clock::from_time_t(time_t t) BOOST_NOEXCEPT
63 return system_clock::time_point(seconds(t));
66 namespace chrono_detail
71 // Note, in this implementation steady_clock and high_resolution_clock
72 // are the same clock. They are both based on mach_absolute_time().
73 // mach_absolute_time() * MachInfo.numer / MachInfo.denom is the number of
74 // nanoseconds since the computer booted up. MachInfo.numer and MachInfo.denom
75 // are run time constants supplied by the OS. This clock has no relationship
76 // to the Gregorian calendar. It's main use is as a high resolution timer.
78 // MachInfo.numer / MachInfo.denom is often 1 on the latest equipment. Specialize
79 // for that case as an optimization.
84 return mach_absolute_time();
87 #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
90 steady_simplified_ec(system::error_code & ec)
92 if (!::boost::chrono::is_throws(ec))
96 return mach_absolute_time();
102 compute_steady_factor(kern_return_t& err)
104 mach_timebase_info_data_t MachInfo;
105 err = mach_timebase_info(&MachInfo);
109 return static_cast<double>(MachInfo.numer) / MachInfo.denom;
117 const double factor = chrono_detail::compute_steady_factor(err);
120 BOOST_ASSERT(0 && "Boost::Chrono - Internal Error");
122 return static_cast<steady_clock::rep>(mach_absolute_time() * factor);
125 #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
128 steady_full_ec(system::error_code & ec)
131 const double factor = chrono_detail::compute_steady_factor(err);
134 if (::boost::chrono::is_throws(ec))
136 boost::throw_exception(
137 system::system_error(
139 ::boost::system::system_category(),
140 "chrono::steady_clock" ));
144 ec.assign( errno, ::boost::system::system_category() );
145 return steady_clock::rep();
148 if (!::boost::chrono::is_throws(ec))
152 return static_cast<steady_clock::rep>(mach_absolute_time() * factor);
156 typedef steady_clock::rep (*FP)();
157 #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
158 typedef steady_clock::rep (*FP_ec)(system::error_code &);
163 init_steady_clock(kern_return_t & err)
165 mach_timebase_info_data_t MachInfo;
166 err = mach_timebase_info(&MachInfo);
172 if (MachInfo.numer == MachInfo.denom)
174 return &chrono_detail::steady_simplified;
176 return &chrono_detail::steady_full;
179 #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
182 init_steady_clock_ec(kern_return_t & err)
184 mach_timebase_info_data_t MachInfo;
185 err = mach_timebase_info(&MachInfo);
191 if (MachInfo.numer == MachInfo.denom)
193 return &chrono_detail::steady_simplified_ec;
195 return &chrono_detail::steady_full_ec;
200 steady_clock::time_point
201 steady_clock::now() BOOST_NOEXCEPT
204 chrono_detail::FP fp = chrono_detail::init_steady_clock(err);
207 BOOST_ASSERT(0 && "Boost::Chrono - Internal Error");
209 return time_point(duration(fp()));
212 #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
213 steady_clock::time_point
214 steady_clock::now(system::error_code & ec)
217 chrono_detail::FP_ec fp = chrono_detail::init_steady_clock_ec(err);
220 if (::boost::chrono::is_throws(ec))
222 boost::throw_exception(
223 system::system_error(
225 ::boost::system::system_category(),
226 "chrono::steady_clock" ));
230 ec.assign( err, ::boost::system::system_category() );
234 if (!::boost::chrono::is_throws(ec))
238 return time_point(duration(fp(ec)));
241 } // namespace chrono