filetime_functions.hpp 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. #ifndef DATE_TIME_FILETIME_FUNCTIONS_HPP__
  2. #define DATE_TIME_FILETIME_FUNCTIONS_HPP__
  3. /* Copyright (c) 2004 CrystalClear Software, Inc.
  4. * Use, modification and distribution is subject to the
  5. * Boost Software License, Version 1.0. (See accompanying
  6. * file LICENSE_1_0.txt or http://www.boost.org/LICENSE_1_0.txt)
  7. * Author: Jeff Garland, Bart Garst
  8. * $Date$
  9. */
  10. /*! @file filetime_functions.hpp
  11. * Function(s) for converting between a FILETIME structure and a
  12. * time object. This file is only available on systems that have
  13. * BOOST_HAS_FTIME defined.
  14. */
  15. #include <boost/date_time/compiler_config.hpp>
  16. #if defined(BOOST_HAS_FTIME) // skip this file if no FILETIME
  17. #if defined(BOOST_USE_WINDOWS_H)
  18. # include <windows.h>
  19. #endif
  20. #include <boost/cstdint.hpp>
  21. #include <boost/date_time/time.hpp>
  22. #include <boost/date_time/date_defs.hpp>
  23. namespace boost {
  24. namespace date_time {
  25. namespace winapi {
  26. #if !defined(BOOST_USE_WINDOWS_H)
  27. extern "C" {
  28. struct FILETIME
  29. {
  30. boost::uint32_t dwLowDateTime;
  31. boost::uint32_t dwHighDateTime;
  32. };
  33. struct SYSTEMTIME
  34. {
  35. boost::uint16_t wYear;
  36. boost::uint16_t wMonth;
  37. boost::uint16_t wDayOfWeek;
  38. boost::uint16_t wDay;
  39. boost::uint16_t wHour;
  40. boost::uint16_t wMinute;
  41. boost::uint16_t wSecond;
  42. boost::uint16_t wMilliseconds;
  43. };
  44. __declspec(dllimport) void __stdcall GetSystemTimeAsFileTime(FILETIME* lpFileTime);
  45. __declspec(dllimport) int __stdcall FileTimeToLocalFileTime(const FILETIME* lpFileTime, FILETIME* lpLocalFileTime);
  46. __declspec(dllimport) void __stdcall GetSystemTime(SYSTEMTIME* lpSystemTime);
  47. __declspec(dllimport) int __stdcall SystemTimeToFileTime(const SYSTEMTIME* lpSystemTime, FILETIME* lpFileTime);
  48. } // extern "C"
  49. #endif // defined(BOOST_USE_WINDOWS_H)
  50. typedef FILETIME file_time;
  51. typedef SYSTEMTIME system_time;
  52. inline void get_system_time_as_file_time(file_time& ft)
  53. {
  54. #if BOOST_WORKAROUND(__MWERKS__, BOOST_TESTED_AT(0x3205))
  55. // Some runtime library implementations expect local times as the norm for ctime.
  56. file_time ft_utc;
  57. GetSystemTimeAsFileTime(&ft_utc);
  58. FileTimeToLocalFileTime(&ft_utc, &ft);
  59. #elif defined(BOOST_HAS_GETSYSTEMTIMEASFILETIME)
  60. GetSystemTimeAsFileTime(&ft);
  61. #else
  62. system_time st;
  63. GetSystemTime(&st);
  64. SystemTimeToFileTime(&st, &ft);
  65. #endif
  66. }
  67. /*!
  68. * The function converts file_time into number of microseconds elapsed since 1970-Jan-01
  69. *
  70. * \note Only dates after 1970-Jan-01 are supported. Dates before will be wrapped.
  71. *
  72. * \note The function is templated on the FILETIME type, so that
  73. * it can be used with both native FILETIME and the ad-hoc
  74. * boost::date_time::winapi::file_time type.
  75. */
  76. template< typename FileTimeT >
  77. inline boost::uint64_t file_time_to_microseconds(FileTimeT const& ft)
  78. {
  79. /* shift is difference between 1970-Jan-01 & 1601-Jan-01
  80. * in 100-nanosecond intervals */
  81. const uint64_t shift = 116444736000000000ULL; // (27111902 << 32) + 3577643008
  82. union {
  83. FileTimeT as_file_time;
  84. uint64_t as_integer; // 100-nanos since 1601-Jan-01
  85. } caster;
  86. caster.as_file_time = ft;
  87. caster.as_integer -= shift; // filetime is now 100-nanos since 1970-Jan-01
  88. return (caster.as_integer / 10); // truncate to microseconds
  89. }
  90. } // namespace winapi
  91. //! Create a time object from an initialized FILETIME struct.
  92. /*!
  93. * Create a time object from an initialized FILETIME struct.
  94. * A FILETIME struct holds 100-nanosecond units (0.0000001). When
  95. * built with microsecond resolution the file_time's sub second value
  96. * will be truncated. Nanosecond resolution has no truncation.
  97. *
  98. * \note The function is templated on the FILETIME type, so that
  99. * it can be used with both native FILETIME and the ad-hoc
  100. * boost::date_time::winapi::file_time type.
  101. */
  102. template< typename TimeT, typename FileTimeT >
  103. inline
  104. TimeT time_from_ftime(const FileTimeT& ft)
  105. {
  106. typedef typename TimeT::date_type date_type;
  107. typedef typename TimeT::date_duration_type date_duration_type;
  108. typedef typename TimeT::time_duration_type time_duration_type;
  109. // https://svn.boost.org/trac/boost/ticket/2523
  110. // Since this function can be called with arbitrary times, including ones that
  111. // are before 1970-Jan-01, we'll have to cast the time a bit differently,
  112. // than it is done in the file_time_to_microseconds function. This allows to
  113. // avoid integer wrapping for dates before 1970-Jan-01.
  114. union {
  115. FileTimeT as_file_time;
  116. uint64_t as_integer; // 100-nanos since 1601-Jan-01
  117. } caster;
  118. caster.as_file_time = ft;
  119. uint64_t sec = caster.as_integer / 10000000UL;
  120. uint32_t sub_sec = (caster.as_integer % 10000000UL) // 100-nanoseconds since the last second
  121. #if !defined(BOOST_DATE_TIME_POSIX_TIME_STD_CONFIG)
  122. / 10; // microseconds since the last second
  123. #else
  124. * 100; // nanoseconds since the last second
  125. #endif
  126. // split sec into usable chunks: days, hours, minutes, & seconds
  127. const uint32_t sec_per_day = 86400; // seconds per day
  128. uint32_t days = static_cast< uint32_t >(sec / sec_per_day);
  129. uint32_t tmp = static_cast< uint32_t >(sec % sec_per_day);
  130. uint32_t hours = tmp / 3600; // sec_per_hour
  131. tmp %= 3600;
  132. uint32_t minutes = tmp / 60; // sec_per_min
  133. tmp %= 60;
  134. uint32_t seconds = tmp; // seconds
  135. date_duration_type dd(days);
  136. date_type d = date_type(1601, Jan, 01) + dd;
  137. return TimeT(d, time_duration_type(hours, minutes, seconds, sub_sec));
  138. }
  139. }} // boost::date_time
  140. #endif // BOOST_HAS_FTIME
  141. #endif // DATE_TIME_FILETIME_FUNCTIONS_HPP__