chrono.hpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. // win/chrono.cpp --------------------------------------------------------------//
  2. // Copyright Beman Dawes 2008
  3. // Copyright 2009-2010 Vicente J. Botet Escriba
  4. // Distributed under the Boost Software License, Version 1.0.
  5. // See http://www.boost.org/LICENSE_1_0.txt
  6. //----------------------------------------------------------------------------//
  7. // Windows //
  8. //----------------------------------------------------------------------------//
  9. #ifndef BOOST_CHRONO_DETAIL_INLINED_WIN_CHRONO_HPP
  10. #define BOOST_CHRONO_DETAIL_INLINED_WIN_CHRONO_HPP
  11. #include <boost/winapi/time.hpp>
  12. #include <boost/winapi/timers.hpp>
  13. #include <boost/winapi/get_last_error.hpp>
  14. #include <boost/assert.hpp>
  15. namespace boost
  16. {
  17. namespace chrono
  18. {
  19. namespace chrono_detail
  20. {
  21. BOOST_CHRONO_INLINE double get_nanosecs_per_tic() BOOST_NOEXCEPT
  22. {
  23. boost::winapi::LARGE_INTEGER_ freq;
  24. if ( !boost::winapi::QueryPerformanceFrequency( &freq ) )
  25. return 0.0L;
  26. return double(1000000000.0L / freq.QuadPart);
  27. }
  28. }
  29. steady_clock::time_point steady_clock::now() BOOST_NOEXCEPT
  30. {
  31. double nanosecs_per_tic = chrono_detail::get_nanosecs_per_tic();
  32. boost::winapi::LARGE_INTEGER_ pcount;
  33. if ( nanosecs_per_tic <= 0.0L )
  34. {
  35. BOOST_ASSERT(0 && "Boost::Chrono - get_nanosecs_per_tic Internal Error");
  36. return steady_clock::time_point();
  37. }
  38. unsigned times=0;
  39. while ( ! boost::winapi::QueryPerformanceCounter( &pcount ) )
  40. {
  41. if ( ++times > 3 )
  42. {
  43. BOOST_ASSERT(0 && "Boost::Chrono - QueryPerformanceCounter Internal Error");
  44. return steady_clock::time_point();
  45. }
  46. }
  47. return steady_clock::time_point(steady_clock::duration(
  48. static_cast<steady_clock::rep>((nanosecs_per_tic) * pcount.QuadPart)));
  49. }
  50. #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
  51. steady_clock::time_point steady_clock::now( system::error_code & ec )
  52. {
  53. double nanosecs_per_tic = chrono_detail::get_nanosecs_per_tic();
  54. boost::winapi::LARGE_INTEGER_ pcount;
  55. if ( (nanosecs_per_tic <= 0.0L)
  56. || (!boost::winapi::QueryPerformanceCounter( &pcount )) )
  57. {
  58. boost::winapi::DWORD_ cause =
  59. ((nanosecs_per_tic <= 0.0L)
  60. ? ERROR_NOT_SUPPORTED
  61. : boost::winapi::GetLastError());
  62. if (::boost::chrono::is_throws(ec)) {
  63. boost::throw_exception(
  64. system::system_error(
  65. cause,
  66. ::boost::system::system_category(),
  67. "chrono::steady_clock" ));
  68. }
  69. else
  70. {
  71. ec.assign( cause, ::boost::system::system_category() );
  72. return steady_clock::time_point(duration(0));
  73. }
  74. }
  75. if (!::boost::chrono::is_throws(ec))
  76. {
  77. ec.clear();
  78. }
  79. return time_point(duration(
  80. static_cast<steady_clock::rep>(nanosecs_per_tic * pcount.QuadPart)));
  81. }
  82. #endif
  83. BOOST_CHRONO_INLINE
  84. system_clock::time_point system_clock::now() BOOST_NOEXCEPT
  85. {
  86. boost::winapi::FILETIME_ ft;
  87. boost::winapi::GetSystemTimeAsFileTime( &ft ); // never fails
  88. return system_clock::time_point(
  89. system_clock::duration(
  90. ((static_cast<__int64>( ft.dwHighDateTime ) << 32) | ft.dwLowDateTime)
  91. - 116444736000000000LL
  92. //- (134775LL*864000000000LL)
  93. )
  94. );
  95. }
  96. #if !defined BOOST_CHRONO_DONT_PROVIDE_HYBRID_ERROR_HANDLING
  97. BOOST_CHRONO_INLINE
  98. system_clock::time_point system_clock::now( system::error_code & ec )
  99. {
  100. boost::winapi::FILETIME_ ft;
  101. boost::winapi::GetSystemTimeAsFileTime( &ft ); // never fails
  102. if (!::boost::chrono::is_throws(ec))
  103. {
  104. ec.clear();
  105. }
  106. return system_clock::time_point(
  107. system_clock::duration(
  108. ((static_cast<__int64>( ft.dwHighDateTime ) << 32) | ft.dwLowDateTime)
  109. - 116444736000000000LL
  110. //- (134775LL*864000000000LL)
  111. ));
  112. }
  113. #endif
  114. BOOST_CHRONO_INLINE
  115. std::time_t system_clock::to_time_t(const system_clock::time_point& t) BOOST_NOEXCEPT
  116. {
  117. __int64 temp = t.time_since_epoch().count();
  118. temp /= 10000000;
  119. return static_cast<std::time_t>( temp );
  120. }
  121. BOOST_CHRONO_INLINE
  122. system_clock::time_point system_clock::from_time_t(std::time_t t) BOOST_NOEXCEPT
  123. {
  124. __int64 temp = t;
  125. temp *= 10000000;
  126. return time_point(duration(temp));
  127. }
  128. } // namespace chrono
  129. } // namespace boost
  130. #endif