Chris@16
|
1 #ifndef DATE_TIME_HIGHRES_TIME_CLOCK_HPP___
|
Chris@16
|
2 #define DATE_TIME_HIGHRES_TIME_CLOCK_HPP___
|
Chris@16
|
3
|
Chris@16
|
4 /* Copyright (c) 2002,2003,2005 CrystalClear Software, Inc.
|
Chris@16
|
5 * Use, modification and distribution is subject to the
|
Chris@16
|
6 * Boost Software License, Version 1.0. (See accompanying
|
Chris@16
|
7 * file LICENSE_1_0.txt or http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
8 * Author: Jeff Garland, Bart Garst
|
Chris@101
|
9 * $Date$
|
Chris@16
|
10 */
|
Chris@16
|
11
|
Chris@16
|
12
|
Chris@16
|
13 /*! @file microsec_time_clock.hpp
|
Chris@16
|
14 This file contains a high resolution time clock implementation.
|
Chris@16
|
15 */
|
Chris@16
|
16
|
Chris@16
|
17 #include <boost/cstdint.hpp>
|
Chris@16
|
18 #include <boost/shared_ptr.hpp>
|
Chris@16
|
19 #include <boost/detail/workaround.hpp>
|
Chris@16
|
20 #include <boost/date_time/compiler_config.hpp>
|
Chris@16
|
21 #include <boost/date_time/c_time.hpp>
|
Chris@16
|
22 #include <boost/date_time/time_clock.hpp>
|
Chris@16
|
23 #include <boost/date_time/filetime_functions.hpp>
|
Chris@16
|
24
|
Chris@16
|
25 #ifdef BOOST_DATE_TIME_HAS_HIGH_PRECISION_CLOCK
|
Chris@16
|
26
|
Chris@16
|
27 namespace boost {
|
Chris@16
|
28 namespace date_time {
|
Chris@16
|
29
|
Chris@16
|
30 //! A clock providing microsecond level resolution
|
Chris@16
|
31 /*! A high precision clock that measures the local time
|
Chris@16
|
32 * at a resolution up to microseconds and adjusts to the
|
Chris@16
|
33 * resolution of the time system. For example, for the
|
Chris@16
|
34 * a library configuration with nano second resolution,
|
Chris@16
|
35 * the last 3 places of the fractional seconds will always
|
Chris@16
|
36 * be 000 since there are 1000 nano-seconds in a micro second.
|
Chris@16
|
37 */
|
Chris@16
|
38 template<class time_type>
|
Chris@16
|
39 class microsec_clock
|
Chris@16
|
40 {
|
Chris@16
|
41 private:
|
Chris@16
|
42 //! Type for the function used to convert time_t to tm
|
Chris@16
|
43 typedef std::tm* (*time_converter)(const std::time_t*, std::tm*);
|
Chris@16
|
44
|
Chris@16
|
45 public:
|
Chris@16
|
46 typedef typename time_type::date_type date_type;
|
Chris@16
|
47 typedef typename time_type::time_duration_type time_duration_type;
|
Chris@16
|
48 typedef typename time_duration_type::rep_type resolution_traits_type;
|
Chris@16
|
49
|
Chris@16
|
50 //! return a local time object for the given zone, based on computer clock
|
Chris@16
|
51 //JKG -- looks like we could rewrite this against universal_time
|
Chris@16
|
52 template<class time_zone_type>
|
Chris@16
|
53 static time_type local_time(shared_ptr<time_zone_type> tz_ptr)
|
Chris@16
|
54 {
|
Chris@16
|
55 typedef typename time_type::utc_time_type utc_time_type;
|
Chris@16
|
56 typedef second_clock<utc_time_type> second_clock;
|
Chris@16
|
57 // we'll need to know the utc_offset this machine has
|
Chris@16
|
58 // in order to get a utc_time_type set to utc
|
Chris@16
|
59 utc_time_type utc_time = second_clock::universal_time();
|
Chris@16
|
60 time_duration_type utc_offset = second_clock::local_time() - utc_time;
|
Chris@16
|
61 // use micro clock to get a local time with sub seconds
|
Chris@16
|
62 // and adjust it to get a true utc time reading with sub seconds
|
Chris@16
|
63 utc_time = microsec_clock<utc_time_type>::local_time() - utc_offset;
|
Chris@16
|
64 return time_type(utc_time, tz_ptr);
|
Chris@16
|
65 }
|
Chris@16
|
66
|
Chris@16
|
67 //! Returns the local time based on computer clock settings
|
Chris@16
|
68 static time_type local_time()
|
Chris@16
|
69 {
|
Chris@16
|
70 return create_time(&c_time::localtime);
|
Chris@16
|
71 }
|
Chris@16
|
72
|
Chris@16
|
73 //! Returns the UTC time based on computer settings
|
Chris@16
|
74 static time_type universal_time()
|
Chris@16
|
75 {
|
Chris@16
|
76 return create_time(&c_time::gmtime);
|
Chris@16
|
77 }
|
Chris@16
|
78
|
Chris@16
|
79 private:
|
Chris@16
|
80 static time_type create_time(time_converter converter)
|
Chris@16
|
81 {
|
Chris@16
|
82 #ifdef BOOST_HAS_GETTIMEOFDAY
|
Chris@16
|
83 timeval tv;
|
Chris@16
|
84 gettimeofday(&tv, 0); //gettimeofday does not support TZ adjust on Linux.
|
Chris@16
|
85 std::time_t t = tv.tv_sec;
|
Chris@16
|
86 boost::uint32_t sub_sec = tv.tv_usec;
|
Chris@16
|
87 #elif defined(BOOST_HAS_FTIME)
|
Chris@16
|
88 winapi::file_time ft;
|
Chris@16
|
89 winapi::get_system_time_as_file_time(ft);
|
Chris@16
|
90 uint64_t micros = winapi::file_time_to_microseconds(ft); // it will not wrap, since ft is the current time
|
Chris@16
|
91 // and cannot be before 1970-Jan-01
|
Chris@16
|
92 std::time_t t = static_cast<std::time_t>(micros / 1000000UL); // seconds since epoch
|
Chris@101
|
93 // microseconds -- static casts suppress warnings
|
Chris@16
|
94 boost::uint32_t sub_sec = static_cast<boost::uint32_t>(micros % 1000000UL);
|
Chris@16
|
95 #else
|
Chris@16
|
96 #error Internal Boost.DateTime error: BOOST_DATE_TIME_HAS_HIGH_PRECISION_CLOCK is defined, however neither gettimeofday nor FILETIME support is detected.
|
Chris@16
|
97 #endif
|
Chris@16
|
98
|
Chris@16
|
99 std::tm curr;
|
Chris@16
|
100 std::tm* curr_ptr = converter(&t, &curr);
|
Chris@16
|
101 date_type d(static_cast< typename date_type::year_type::value_type >(curr_ptr->tm_year + 1900),
|
Chris@16
|
102 static_cast< typename date_type::month_type::value_type >(curr_ptr->tm_mon + 1),
|
Chris@16
|
103 static_cast< typename date_type::day_type::value_type >(curr_ptr->tm_mday));
|
Chris@16
|
104
|
Chris@16
|
105 //The following line will adjust the fractional second tick in terms
|
Chris@16
|
106 //of the current time system. For example, if the time system
|
Chris@16
|
107 //doesn't support fractional seconds then res_adjust returns 0
|
Chris@16
|
108 //and all the fractional seconds return 0.
|
Chris@16
|
109 int adjust = static_cast< int >(resolution_traits_type::res_adjust() / 1000000);
|
Chris@16
|
110
|
Chris@16
|
111 time_duration_type td(static_cast< typename time_duration_type::hour_type >(curr_ptr->tm_hour),
|
Chris@16
|
112 static_cast< typename time_duration_type::min_type >(curr_ptr->tm_min),
|
Chris@16
|
113 static_cast< typename time_duration_type::sec_type >(curr_ptr->tm_sec),
|
Chris@16
|
114 sub_sec * adjust);
|
Chris@16
|
115
|
Chris@16
|
116 return time_type(d,td);
|
Chris@16
|
117 }
|
Chris@16
|
118 };
|
Chris@16
|
119
|
Chris@16
|
120
|
Chris@16
|
121 } } //namespace date_time
|
Chris@16
|
122
|
Chris@16
|
123 #endif //BOOST_DATE_TIME_HAS_HIGH_PRECISION_CLOCK
|
Chris@16
|
124
|
Chris@16
|
125
|
Chris@16
|
126 #endif
|
Chris@16
|
127
|