27 using period = std::ratio<86400>;
28 using duration = std::chrono::duration<rep, period>;
29 using time_point = std::chrono::time_point<aosn_date>;
30 static constexpr bool is_steady =
false;
35 static time_point
now() noexcept
39 GetSystemTimeAsFileTime(&t);
51 static std::time_t
to_time_t(_In_
const time_point tp)
noexcept
53 return static_cast<std::time_t
>(tp.time_since_epoch().count()) * 86400 - 210866803200;
61 return time_point(duration(
static_cast<rep
>((t + 210866803200) / 86400)));
68 static time_point
from_system(_In_
const SYSTEMTIME& t)
noexcept
70 return from_dmy(
static_cast<uint8_t
>(t.wDay),
static_cast<uint8_t
>(t.wMonth),
static_cast<int32_t
>(t.wYear));
76 static time_point
from_system(_In_
const FILETIME& t)
noexcept
78 uint64_t x = ((
static_cast<uint64_t
>(t.dwHighDateTime)) << 32) | t.dwLowDateTime;
79 return time_point(duration(
static_cast<rep
>(x / 864000000000 + 2305814)));
88 if (!VariantTimeToSystemTime(t, &st))
89 throw std::invalid_argument(
"failed to convert date from VARIANT_DATE");
96 static time_point
from_system(_In_
const struct timespec& t)
noexcept
106 static void to_system(_In_ time_point tp, _Out_ SYSTEMTIME& t)
110 to_dmy(tp, &day, &month, &year);
113 if (year > WORD_MAX) _Unlikely_
114 throw std::range_error(
"year too big");
115 t.wYear =
static_cast<WORD
>(year);
116 t.wDayOfWeek =
static_cast<int>(
day_of_week(tp) + 1) % 7;
126 static void to_system(_In_ time_point tp, _Out_ FILETIME& t)
noexcept
128 uint64_t x = (tp.time_since_epoch().count() - 2305814) * 864000000000;
129 t.dwHighDateTime =
static_cast<DWORD
>(x >> 32);
130 t.dwLowDateTime =
static_cast<DWORD
>(x & 0xffffffff);
136 static void to_system(_In_ time_point tp, _Out_ DATE& t)
140 if (!SystemTimeToVariantTime(&st, &t))
141 throw std::invalid_argument(
"failed to convert date to VARIANT_DATE");
148 static time_point
from_dmy(_In_ uint8_t day, _In_ uint8_t month, _In_ int32_t year)
noexcept
159 int32_t ctmp = (ytmp / 100);
160 int32_t dtmp = ytmp - (100 * ctmp);
161 int32_t result1 = 146097L * ctmp / 4;
162 int32_t result2 = (1461 * dtmp) / 4;
163 int32_t result3 = (153 * mtmp + 2) / 5;
164 return time_point(duration(
static_cast<int32_t
>(result1) + day + result2 + 1721119L + result3));
170 static void to_dmy(_In_
const time_point tp, _Out_opt_ uint8_t* day, _Out_opt_ uint8_t* month, _Out_opt_ int32_t* year)
noexcept
172 int32_t mtmp = tp.time_since_epoch().count() - 1721119L;
173 int32_t yr = (4 * mtmp - 1) / 146097L;
174 mtmp = 4 * mtmp - 1 - 146097L * yr;
175 int32_t da = mtmp / 4;
176 mtmp = (4 * da + 3) / 1461;
177 da = 4 * da + 3 - 1461 * mtmp;
179 int32_t mo = (5 * da - 3) / 153;
180 da = 5 * da - 3 - 153 * mo;
182 yr = 100 * yr + mtmp;
189 if (day) *day =
static_cast<uint8_t
>(da);
190 if (month) *month =
static_cast<uint8_t
>(mo);
191 if (year) *year = yr;
199 return static_cast<uint8_t
>(tp.time_since_epoch().count() % 7);
205 static uint8_t
day_of_week(_In_ uint8_t day, _In_ uint8_t month, _In_ int32_t year)
207 return static_cast<uint8_t
>(
from_dmy(day, month, year).time_since_epoch().count() % 7);
217 using period = std::milli;
218 using duration = std::chrono::duration<rep, period>;
219 using time_point = std::chrono::time_point<aosn_timestamp>;
220 static constexpr bool is_steady =
false;
222 static constexpr rep f_second = 1000;
223 static constexpr rep f_minute = 60;
224 static constexpr rep f_hour = 60;
225 static constexpr rep f_day = 24;
226 static constexpr rep f_week = 7;
228 static constexpr rep one_second = f_second;
229 static constexpr rep one_minute = f_minute * one_second;
230 static constexpr rep one_hour = f_hour * one_minute;
231 static constexpr rep one_day = f_day * one_hour;
232 static constexpr rep one_week = f_week * one_day;
237 static time_point
now() noexcept
241 GetSystemTimeAsFileTime(&t);
253 static std::time_t
to_time_t(_In_
const time_point tp)
noexcept
255 return tp.time_since_epoch().count() / one_second - 210866803200;
263 return time_point(duration((
static_cast<rep
>(t) + 210866803200) * one_second));
270 static time_point
from_system(_In_
const SYSTEMTIME& t)
noexcept
273 static_cast<uint8_t
>(t.wDay),
static_cast<uint8_t
>(t.wMonth),
static_cast<int32_t
>(t.wYear),
274 static_cast<uint8_t
>(t.wHour),
static_cast<uint8_t
>(t.wMinute),
static_cast<uint8_t
>(t.wSecond),
static_cast<uint16_t
>(t.wMilliseconds));
280 static time_point
from_system(_In_
const FILETIME& t)
noexcept
282 uint64_t x = ((
static_cast<uint64_t
>(t.dwHighDateTime)) << 32) | t.dwLowDateTime;
283 return time_point(duration(
static_cast<rep
>(x / 10000 + 199222329600000)));
292 if (!VariantTimeToSystemTime(t, &st))
293 throw std::invalid_argument(
"failed to convert date from VARIANT_DATE");
300 static time_point
from_system(_In_
const struct timespec& t)
noexcept
302 return time_point(duration(
static_cast<rep
>(
from_time_t(t.tv_sec).time_since_epoch().count() + t.tv_nsec / 1000)));
306 static void to_system(_In_ time_point tp, _Out_
struct tm& date)
noexcept
308 uint8_t day, month, hour, minute, second;
309 uint16_t millisecond;
311 to_dmy(tp, &day, &month, &year, &hour, &minute, &second, &millisecond);
312 date.tm_sec = second;
313 date.tm_min = minute;
316 date.tm_mon = month - 1;
317 date.tm_year = year - 1900;
327 static void to_system(_In_ time_point tp, _Out_ SYSTEMTIME& t)
329 uint8_t day, month, hour, minute, second;
330 uint16_t millisecond;
334 &hour, &minute, &second, &millisecond);
337 if (year > WORD_MAX) _Unlikely_
338 throw std::range_error(
"year too big");
339 t.wYear =
static_cast<WORD
>(year);
344 t.wMilliseconds = millisecond;
350 static void to_system(_In_ time_point tp, _Out_ FILETIME& t)
noexcept
352 uint64_t x = (tp.time_since_epoch().count() - 199222329600000) * 10000;
353 t.dwHighDateTime =
static_cast<DWORD
>(x >> 32);
354 t.dwLowDateTime =
static_cast<DWORD
>(x & 0xffffffff);
360 static void to_system(_In_ time_point tp, _Out_ DATE& t)
364 if (!SystemTimeToVariantTime(&st, &t))
365 throw std::invalid_argument(
"failed to convert date to VARIANT_DATE");
372 static aosn_date::time_point
to_date(_In_ time_point tp)
noexcept
374 return aosn_date::time_point(aosn_date::duration(
static_cast<aosn_date::rep
>(tp.time_since_epoch().count() / one_day)));
380 static time_point
from_date(_In_ aosn_date::time_point date)
noexcept
382 return time_point(duration(
static_cast<rep
>(date.time_since_epoch().count()) * one_day));
389 _In_ uint8_t day, _In_ uint8_t month, _In_ int32_t year,
390 _In_ uint8_t hour, _In_ uint8_t minute, _In_ uint8_t second, _In_ uint16_t millisecond)
noexcept
392 return time_point(duration(
393 (
static_cast<rep
>(
aosn_date::from_dmy(day, month, year).time_since_epoch().count()) * one_day) +
394 (
static_cast<rep
>(hour) * one_hour +
static_cast<rep
>(minute) * one_minute +
static_cast<rep
>(second) * one_second + millisecond)));
400 static void to_dmy(_In_
const time_point tp,
401 _Out_opt_ uint8_t* day, _Out_opt_ uint8_t* month, _Out_opt_ int32_t* year,
402 _Out_opt_ uint8_t* hour, _Out_opt_ uint8_t* minute, _Out_opt_ uint8_t* second, _Out_opt_ uint16_t* millisecond)
noexcept
405 int32_t u =
static_cast<int32_t
>(tp.time_since_epoch().count() % one_day);
406 if (millisecond) *millisecond =
static_cast<uint16_t
>(u % f_second);
408 if (second) *second =
static_cast<uint8_t
>(u % f_minute);
410 if (minute) *minute =
static_cast<uint8_t
>(u % f_hour);
412 if (hour) *hour =
static_cast<uint8_t
>(u);
415 template<
class TR = std::
char_traits<
char>,
class AX = std::allocator<
char>>
416 static std::basic_string<char, TR, AX> to_str(_In_
const time_point tp, _In_z_
const char* format, _In_opt_ locale_t
locale)
420 std::basic_string<char, TR, AX> str;
421 char stack_buffer[1024 /
sizeof(char)];
424 n = _strftime_l(stack_buffer, _countof(stack_buffer), format, &date,
locale);
426 n = strftime_l(stack_buffer, _countof(stack_buffer), format, &date,
locale);
429 str.assign(stack_buffer, stack_buffer + n);
432 size_t num_chars = stdex::mul(_countof(stack_buffer), 2);
434 std::unique_ptr<char> buf(
new char[num_chars]);
436 n = _strftime_l(buf.get(), num_chars, format, &date,
locale);
438 n = strftime_l(buf.get(), num_chars, format, &date,
locale);
441 str.assign(buf.get(), buf.get() + n);
444 num_chars = stdex::mul(num_chars, 2);
448 template<
class TR = std::
char_traits<
wchar_t>,
class AX = std::allocator<
wchar_t>>
449 static std::basic_string<wchar_t, TR, AX> to_str(_In_
const time_point tp, _In_z_
const wchar_t* format, _In_opt_ locale_t locale)
453 std::basic_string<wchar_t, TR, AX> str;
454 wchar_t stack_buffer[1024 /
sizeof(wchar_t)];
457 n = _wcsftime_l(stack_buffer, _countof(stack_buffer), format, &date, locale);
459 n = wcsftime_l(stack_buffer, _countof(stack_buffer), format, &date, locale);
462 str.assign(stack_buffer, stack_buffer + n);
465 size_t num_chars = stdex::mul(_countof(stack_buffer), 2);
467 std::unique_ptr<wchar_t> buf(
new wchar_t[num_chars]);
469 n = _wcsftime_l(buf.get(), num_chars, format, &date, locale);
471 n = wcsftime_l(buf.get(), num_chars, format, &date, locale);
474 str.assign(buf.get(), buf.get() + n);
477 num_chars = stdex::mul(num_chars, 2);
481 template<
class TR = std::
char_traits<
char>,
class AX = std::allocator<
char>>
482 static std::basic_string<char, TR, AX> to_rfc822(_In_
const time_point tp)
484 return to_str(tp,
"%a, %d %b %Y %H:%M:%S GMT", stdex::locale_C);
static void to_dmy(const time_point tp, uint8_t *day, uint8_t *month, int32_t *year, uint8_t *hour, uint8_t *minute, uint8_t *second, uint16_t *millisecond) noexcept
Returns calendar day, month, year and time from time point.
Definition chrono.hpp:400