Path: blob/main/contrib/llvm-project/libcxx/include/__chrono/convert_to_tm.h
35262 views
// -*- C++ -*-1//===----------------------------------------------------------------------===//2//3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.4// See https://llvm.org/LICENSE.txt for license information.5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception6//7//===----------------------------------------------------------------------===//89#ifndef _LIBCPP___CHRONO_CONVERT_TO_TM_H10#define _LIBCPP___CHRONO_CONVERT_TO_TM_H1112#include <__chrono/calendar.h>13#include <__chrono/concepts.h>14#include <__chrono/day.h>15#include <__chrono/duration.h>16#include <__chrono/file_clock.h>17#include <__chrono/hh_mm_ss.h>18#include <__chrono/local_info.h>19#include <__chrono/month.h>20#include <__chrono/month_weekday.h>21#include <__chrono/monthday.h>22#include <__chrono/statically_widen.h>23#include <__chrono/sys_info.h>24#include <__chrono/system_clock.h>25#include <__chrono/time_point.h>26#include <__chrono/weekday.h>27#include <__chrono/year.h>28#include <__chrono/year_month.h>29#include <__chrono/year_month_day.h>30#include <__chrono/year_month_weekday.h>31#include <__chrono/zoned_time.h>32#include <__concepts/same_as.h>33#include <__config>34#include <__format/format_error.h>35#include <__memory/addressof.h>36#include <__type_traits/is_convertible.h>37#include <__type_traits/is_specialization.h>38#include <cstdint>39#include <ctime>40#include <limits>4142#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)43# pragma GCC system_header44#endif4546_LIBCPP_PUSH_MACROS47#include <__undef_macros>4849_LIBCPP_BEGIN_NAMESPACE_STD5051#if _LIBCPP_STD_VER >= 205253// Conerts a chrono date and weekday to a given _Tm type.54//55// This is an implementation detail for the function56// template <class _Tm, class _ChronoT>57// _Tm __convert_to_tm(const _ChronoT& __value)58//59// This manually converts the two values to the proper type. It is possible to60// convert from sys_days to time_t and then to _Tm. But this leads to the Y2K61// bug when time_t is a 32-bit signed integer. Chrono considers years beyond62// the year 2038 valid, so instead do the transformation manually.63template <class _Tm, class _Date>64requires(same_as<_Date, chrono::year_month_day> || same_as<_Date, chrono::year_month_day_last>)65_LIBCPP_HIDE_FROM_ABI _Tm __convert_to_tm(const _Date& __date, chrono::weekday __weekday) {66_Tm __result = {};67# ifdef __GLIBC__68__result.tm_zone = "UTC";69# endif70__result.tm_year = static_cast<int>(__date.year()) - 1900;71__result.tm_mon = static_cast<unsigned>(__date.month()) - 1;72__result.tm_mday = static_cast<unsigned>(__date.day());73__result.tm_wday = static_cast<unsigned>(__weekday.c_encoding());74__result.tm_yday =75(static_cast<chrono::sys_days>(__date) -76static_cast<chrono::sys_days>(chrono::year_month_day{__date.year(), chrono::January, chrono::day{1}}))77.count();7879return __result;80}8182template <class _Tm, class _Duration>83_LIBCPP_HIDE_FROM_ABI _Tm __convert_to_tm(const chrono::sys_time<_Duration> __tp) {84chrono::sys_days __days = chrono::floor<chrono::days>(__tp);85chrono::year_month_day __ymd{__days};8687_Tm __result = std::__convert_to_tm<_Tm>(chrono::year_month_day{__ymd}, chrono::weekday{__days});8889uint64_t __sec =90chrono::duration_cast<chrono::seconds>(__tp - chrono::time_point_cast<chrono::seconds>(__days)).count();91__sec %= 24 * 3600;92__result.tm_hour = __sec / 3600;93__sec %= 3600;94__result.tm_min = __sec / 60;95__result.tm_sec = __sec % 60;9697return __result;98}99100// Convert a chrono (calendar) time point, or dururation to the given _Tm type,101// which must have the same properties as std::tm.102template <class _Tm, class _ChronoT>103_LIBCPP_HIDE_FROM_ABI _Tm __convert_to_tm(const _ChronoT& __value) {104_Tm __result = {};105# ifdef __GLIBC__106__result.tm_zone = "UTC";107# endif108109if constexpr (__is_time_point<_ChronoT>) {110if constexpr (same_as<typename _ChronoT::clock, chrono::system_clock>)111return std::__convert_to_tm<_Tm>(__value);112else if constexpr (same_as<typename _ChronoT::clock, chrono::file_clock>)113return std::__convert_to_tm<_Tm>(_ChronoT::clock::to_sys(__value));114else if constexpr (same_as<typename _ChronoT::clock, chrono::local_t>)115return std::__convert_to_tm<_Tm>(chrono::sys_time<typename _ChronoT::duration>{__value.time_since_epoch()});116else117static_assert(sizeof(_ChronoT) == 0, "TODO: Add the missing clock specialization");118} else if constexpr (chrono::__is_duration<_ChronoT>::value) {119// [time.format]/6120// ... However, if a flag refers to a "time of day" (e.g. %H, %I, %p,121// etc.), then a specialization of duration is interpreted as the time of122// day elapsed since midnight.123124// Not all values can be converted to hours, it may run into ratio125// conversion errors. In that case the conversion to seconds works.126if constexpr (is_convertible_v<_ChronoT, chrono::hours>) {127auto __hour = chrono::floor<chrono::hours>(__value);128auto __sec = chrono::duration_cast<chrono::seconds>(__value - __hour);129__result.tm_hour = __hour.count() % 24;130__result.tm_min = __sec.count() / 60;131__result.tm_sec = __sec.count() % 60;132} else {133uint64_t __sec = chrono::duration_cast<chrono::seconds>(__value).count();134__sec %= 24 * 3600;135__result.tm_hour = __sec / 3600;136__sec %= 3600;137__result.tm_min = __sec / 60;138__result.tm_sec = __sec % 60;139}140} else if constexpr (same_as<_ChronoT, chrono::day>)141__result.tm_mday = static_cast<unsigned>(__value);142else if constexpr (same_as<_ChronoT, chrono::month>)143__result.tm_mon = static_cast<unsigned>(__value) - 1;144else if constexpr (same_as<_ChronoT, chrono::year>)145__result.tm_year = static_cast<int>(__value) - 1900;146else if constexpr (same_as<_ChronoT, chrono::weekday>)147__result.tm_wday = __value.c_encoding();148else if constexpr (same_as<_ChronoT, chrono::weekday_indexed> || same_as<_ChronoT, chrono::weekday_last>)149__result.tm_wday = __value.weekday().c_encoding();150else if constexpr (same_as<_ChronoT, chrono::month_day>) {151__result.tm_mday = static_cast<unsigned>(__value.day());152__result.tm_mon = static_cast<unsigned>(__value.month()) - 1;153} else if constexpr (same_as<_ChronoT, chrono::month_day_last>) {154__result.tm_mon = static_cast<unsigned>(__value.month()) - 1;155} else if constexpr (same_as<_ChronoT, chrono::month_weekday> || same_as<_ChronoT, chrono::month_weekday_last>) {156__result.tm_wday = __value.weekday_indexed().weekday().c_encoding();157__result.tm_mon = static_cast<unsigned>(__value.month()) - 1;158} else if constexpr (same_as<_ChronoT, chrono::year_month>) {159__result.tm_year = static_cast<int>(__value.year()) - 1900;160__result.tm_mon = static_cast<unsigned>(__value.month()) - 1;161} else if constexpr (same_as<_ChronoT, chrono::year_month_day> || same_as<_ChronoT, chrono::year_month_day_last>) {162return std::__convert_to_tm<_Tm>(163chrono::year_month_day{__value}, chrono::weekday{static_cast<chrono::sys_days>(__value)});164} else if constexpr (same_as<_ChronoT, chrono::year_month_weekday> ||165same_as<_ChronoT, chrono::year_month_weekday_last>) {166return std::__convert_to_tm<_Tm>(chrono::year_month_day{static_cast<chrono::sys_days>(__value)}, __value.weekday());167} else if constexpr (__is_hh_mm_ss<_ChronoT>) {168__result.tm_sec = __value.seconds().count();169__result.tm_min = __value.minutes().count();170// In libc++ hours is stored as a long. The type in std::tm is an int. So171// the overflow can only occur when hour uses more bits than an int172// provides.173if constexpr (sizeof(std::chrono::hours::rep) > sizeof(__result.tm_hour))174if (__value.hours().count() > std::numeric_limits<decltype(__result.tm_hour)>::max())175std::__throw_format_error("Formatting hh_mm_ss, encountered an hour overflow");176__result.tm_hour = __value.hours().count();177# if !defined(_LIBCPP_HAS_NO_EXPERIMENTAL_TZDB)178} else if constexpr (same_as<_ChronoT, chrono::sys_info>) {179// Has no time information.180} else if constexpr (same_as<_ChronoT, chrono::local_info>) {181// Has no time information.182# if !defined(_LIBCPP_HAS_NO_TIME_ZONE_DATABASE) && !defined(_LIBCPP_HAS_NO_FILESYSTEM) && \183!defined(_LIBCPP_HAS_NO_LOCALIZATION)184} else if constexpr (__is_specialization_v<_ChronoT, chrono::zoned_time>) {185return std::__convert_to_tm<_Tm>(186chrono::sys_time<typename _ChronoT::duration>{__value.get_local_time().time_since_epoch()});187# endif188# endif // !defined(_LIBCPP_HAS_NO_EXPERIMENTAL_TZDB)189} else190static_assert(sizeof(_ChronoT) == 0, "Add the missing type specialization");191192return __result;193}194195#endif // if _LIBCPP_STD_VER >= 20196197_LIBCPP_END_NAMESPACE_STD198199_LIBCPP_POP_MACROS200201#endif // _LIBCPP___CHRONO_CONVERT_TO_TM_H202203204