19d6b68b6SMark de Wever //===----------------------------------------------------------------------===// 29d6b68b6SMark de Wever // 39d6b68b6SMark de Wever // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 49d6b68b6SMark de Wever // See https://llvm.org/LICENSE.txt for license information. 59d6b68b6SMark de Wever // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 69d6b68b6SMark de Wever // 79d6b68b6SMark de Wever //===----------------------------------------------------------------------===// 89d6b68b6SMark de Wever 99d6b68b6SMark de Wever // For information see https://libcxx.llvm.org/DesignDocs/TimeZone.html 109d6b68b6SMark de Wever 119d6b68b6SMark de Wever // TODO TZDB look at optimizations 129d6b68b6SMark de Wever // 139d6b68b6SMark de Wever // The current algorithm is correct but not efficient. For example, in a named 149d6b68b6SMark de Wever // rule based continuation finding the next rule does quite a bit of work, 159d6b68b6SMark de Wever // returns the next rule and "forgets" its state. This could be better. 169d6b68b6SMark de Wever // 179d6b68b6SMark de Wever // It would be possible to cache lookups. If a time for a zone is calculated its 189d6b68b6SMark de Wever // sys_info could be kept and the next lookup could test whether the time is in 199d6b68b6SMark de Wever // a "known" sys_info. The wording in the Standard hints at this slowness by 209d6b68b6SMark de Wever // "suggesting" this could be implemented on the user's side. 219d6b68b6SMark de Wever 229d6b68b6SMark de Wever // TODO TZDB look at removing quirks 239d6b68b6SMark de Wever // 249d6b68b6SMark de Wever // The code has some special rules to adjust the timing at the continuation 259d6b68b6SMark de Wever // switches. This works correctly, but some of the places feel odd. It would be 269d6b68b6SMark de Wever // good to investigate this further and see whether all quirks are needed or 279d6b68b6SMark de Wever // that there are better fixes. 289d6b68b6SMark de Wever // 299d6b68b6SMark de Wever // These quirks often use a 12h interval; this is the scan interval of zdump, 309d6b68b6SMark de Wever // which implies there are no sys_info objects with a duration of less than 12h. 319d6b68b6SMark de Wever 329d6b68b6SMark de Wever #include <algorithm> 339d6b68b6SMark de Wever #include <cctype> 349d6b68b6SMark de Wever #include <chrono> 359d6b68b6SMark de Wever #include <expected> 369d6b68b6SMark de Wever #include <map> 379d6b68b6SMark de Wever #include <numeric> 389d6b68b6SMark de Wever #include <ranges> 399d6b68b6SMark de Wever 409d6b68b6SMark de Wever #include "include/tzdb/time_zone_private.h" 419d6b68b6SMark de Wever #include "include/tzdb/tzdb_list_private.h" 429d6b68b6SMark de Wever 439d6b68b6SMark de Wever // TODO TZDB remove debug printing 449d6b68b6SMark de Wever #ifdef PRINT 459d6b68b6SMark de Wever # include <print> 469d6b68b6SMark de Wever #endif 479d6b68b6SMark de Wever 489d6b68b6SMark de Wever _LIBCPP_BEGIN_NAMESPACE_STD 499d6b68b6SMark de Wever 509d6b68b6SMark de Wever #ifdef PRINT 519d6b68b6SMark de Wever template <> 529d6b68b6SMark de Wever struct formatter<chrono::sys_info, char> { 539d6b68b6SMark de Wever template <class ParseContext> 549d6b68b6SMark de Wever constexpr typename ParseContext::iterator parse(ParseContext& ctx) { 559d6b68b6SMark de Wever return ctx.begin(); 569d6b68b6SMark de Wever } 579d6b68b6SMark de Wever 589d6b68b6SMark de Wever template <class FormatContext> 599d6b68b6SMark de Wever typename FormatContext::iterator format(const chrono::sys_info& info, FormatContext& ctx) const { 609d6b68b6SMark de Wever return std::format_to( 619d6b68b6SMark de Wever ctx.out(), "[{}, {}) {:%Q%q} {:%Q%q} {}", info.begin, info.end, info.offset, info.save, info.abbrev); 629d6b68b6SMark de Wever } 639d6b68b6SMark de Wever }; 649d6b68b6SMark de Wever #endif 659d6b68b6SMark de Wever 669d6b68b6SMark de Wever namespace chrono { 679d6b68b6SMark de Wever 689d6b68b6SMark de Wever //===----------------------------------------------------------------------===// 699d6b68b6SMark de Wever // Details 709d6b68b6SMark de Wever //===----------------------------------------------------------------------===// 719d6b68b6SMark de Wever 729d6b68b6SMark de Wever struct __sys_info { 739d6b68b6SMark de Wever sys_info __info; 749d6b68b6SMark de Wever bool __can_merge; // Can the returned sys_info object be merged with 759d6b68b6SMark de Wever }; 769d6b68b6SMark de Wever 779d6b68b6SMark de Wever // Return type for helper function to get a sys_info. 789d6b68b6SMark de Wever // - The expected result returns the "best" sys_info object. This object can be 799d6b68b6SMark de Wever // before the requested time. Sometimes sys_info objects from different 809d6b68b6SMark de Wever // continuations share their offset, save, and abbrev and these objects are 819d6b68b6SMark de Wever // merged to one sys_info object. The __can_merge flag determines whether the 829d6b68b6SMark de Wever // current result can be merged with the next result. 839d6b68b6SMark de Wever // - The unexpected result means no sys_info object was found and the time is 849d6b68b6SMark de Wever // the time to be used for the next search iteration. 859d6b68b6SMark de Wever using __sys_info_result = expected<__sys_info, sys_seconds>; 869d6b68b6SMark de Wever 879d6b68b6SMark de Wever template <ranges::forward_range _Range, 889d6b68b6SMark de Wever class _Type, 899d6b68b6SMark de Wever class _Proj = identity, 909d6b68b6SMark de Wever indirect_strict_weak_order<const _Type*, projected<ranges::iterator_t<_Range>, _Proj>> _Comp = ranges::less> 919d6b68b6SMark de Wever [[nodiscard]] static ranges::borrowed_iterator_t<_Range> 929d6b68b6SMark de Wever __binary_find(_Range&& __r, const _Type& __value, _Comp __comp = {}, _Proj __proj = {}) { 939d6b68b6SMark de Wever auto __end = ranges::end(__r); 949d6b68b6SMark de Wever auto __ret = ranges::lower_bound(ranges::begin(__r), __end, __value, __comp, __proj); 959d6b68b6SMark de Wever if (__ret == __end) 969d6b68b6SMark de Wever return __end; 979d6b68b6SMark de Wever 989d6b68b6SMark de Wever // When the value does not match the predicate it's equal and a valid result 999d6b68b6SMark de Wever // was found. 1009d6b68b6SMark de Wever return !std::invoke(__comp, __value, std::invoke(__proj, *__ret)) ? __ret : __end; 1019d6b68b6SMark de Wever } 1029d6b68b6SMark de Wever 1039d6b68b6SMark de Wever // Format based on https://data.iana.org/time-zones/tz-how-to.html 1049d6b68b6SMark de Wever // 1059d6b68b6SMark de Wever // 1 a time zone abbreviation that is a string of three or more characters that 1069d6b68b6SMark de Wever // are either ASCII alphanumerics, "+", or "-" 1079d6b68b6SMark de Wever // 2 the string "%z", in which case the "%z" will be replaced by a numeric time 1089d6b68b6SMark de Wever // zone abbreviation 1099d6b68b6SMark de Wever // 3 a pair of time zone abbreviations separated by a slash ('/'), in which 1109d6b68b6SMark de Wever // case the first string is the abbreviation for the standard time name and 1119d6b68b6SMark de Wever // the second string is the abbreviation for the daylight saving time name 1129d6b68b6SMark de Wever // 4 a string containing "%s", in which case the "%s" will be replaced by the 1139d6b68b6SMark de Wever // text in the appropriate Rule's LETTER column, and the resulting string 1149d6b68b6SMark de Wever // should be a time zone abbreviation 1159d6b68b6SMark de Wever // 1169d6b68b6SMark de Wever // Rule 1 is not strictly validated since America/Barbados uses a two letter 1179d6b68b6SMark de Wever // abbreviation AT. 1189d6b68b6SMark de Wever [[nodiscard]] static string 1199d6b68b6SMark de Wever __format(const __tz::__continuation& __continuation, const string& __letters, seconds __save) { 1209d6b68b6SMark de Wever bool __shift = false; 1219d6b68b6SMark de Wever string __result; 1229d6b68b6SMark de Wever for (char __c : __continuation.__format) { 1239d6b68b6SMark de Wever if (__shift) { 1249d6b68b6SMark de Wever switch (__c) { 1259d6b68b6SMark de Wever case 's': 1269d6b68b6SMark de Wever std::ranges::copy(__letters, std::back_inserter(__result)); 1279d6b68b6SMark de Wever break; 1289d6b68b6SMark de Wever 1299d6b68b6SMark de Wever case 'z': { 1309d6b68b6SMark de Wever if (__continuation.__format.size() != 2) 1319d6b68b6SMark de Wever std::__throw_runtime_error( 1329d6b68b6SMark de Wever std::format("corrupt tzdb FORMAT field: %z should be the entire contents, instead contains '{}'", 1339d6b68b6SMark de Wever __continuation.__format) 1349d6b68b6SMark de Wever .c_str()); 1359d6b68b6SMark de Wever chrono::hh_mm_ss __offset{__continuation.__stdoff + __save}; 1369d6b68b6SMark de Wever if (__offset.is_negative()) { 1379d6b68b6SMark de Wever __result += '-'; 1389d6b68b6SMark de Wever __offset = chrono::hh_mm_ss{-(__continuation.__stdoff + __save)}; 1399d6b68b6SMark de Wever } else 1409d6b68b6SMark de Wever __result += '+'; 1419d6b68b6SMark de Wever 1429d6b68b6SMark de Wever if (__offset.minutes() != 0min) 1439d6b68b6SMark de Wever std::format_to(std::back_inserter(__result), "{:%H%M}", __offset); 1449d6b68b6SMark de Wever else 1459d6b68b6SMark de Wever std::format_to(std::back_inserter(__result), "{:%H}", __offset); 1469d6b68b6SMark de Wever } break; 1479d6b68b6SMark de Wever 1489d6b68b6SMark de Wever default: 1499d6b68b6SMark de Wever std::__throw_runtime_error( 1509d6b68b6SMark de Wever std::format("corrupt tzdb FORMAT field: invalid sequence '%{}' found, expected %s or %z", __c).c_str()); 1519d6b68b6SMark de Wever } 1529d6b68b6SMark de Wever __shift = false; 1539d6b68b6SMark de Wever 1549d6b68b6SMark de Wever } else if (__c == '/') { 1559d6b68b6SMark de Wever if (__save != 0s) 1569d6b68b6SMark de Wever __result.clear(); 1579d6b68b6SMark de Wever else 1589d6b68b6SMark de Wever break; 1599d6b68b6SMark de Wever 1609d6b68b6SMark de Wever } else if (__c == '%') { 1619d6b68b6SMark de Wever __shift = true; 1629d6b68b6SMark de Wever } else if (__c == '+' || __c == '-' || std::isalnum(__c)) { 1639d6b68b6SMark de Wever __result.push_back(__c); 1649d6b68b6SMark de Wever } else { 1659d6b68b6SMark de Wever std::__throw_runtime_error( 1669d6b68b6SMark de Wever std::format( 1679d6b68b6SMark de Wever "corrupt tzdb FORMAT field: invalid character '{}' found, expected +, -, or an alphanumeric value", __c) 1689d6b68b6SMark de Wever .c_str()); 1699d6b68b6SMark de Wever } 1709d6b68b6SMark de Wever } 1719d6b68b6SMark de Wever 1729d6b68b6SMark de Wever if (__shift) 1739d6b68b6SMark de Wever std::__throw_runtime_error("corrupt tzdb FORMAT field: input ended with the start of the escape sequence '%'"); 1749d6b68b6SMark de Wever 1759d6b68b6SMark de Wever if (__result.empty()) 1769d6b68b6SMark de Wever std::__throw_runtime_error("corrupt tzdb FORMAT field: result is empty"); 1779d6b68b6SMark de Wever 1789d6b68b6SMark de Wever return __result; 1799d6b68b6SMark de Wever } 1809d6b68b6SMark de Wever 1819d6b68b6SMark de Wever [[nodiscard]] static sys_seconds __to_sys_seconds(year_month_day __ymd, seconds __seconds) { 1829d6b68b6SMark de Wever seconds __result = static_cast<sys_days>(__ymd).time_since_epoch() + __seconds; 1839d6b68b6SMark de Wever return sys_seconds{__result}; 1849d6b68b6SMark de Wever } 1859d6b68b6SMark de Wever 1869d6b68b6SMark de Wever [[nodiscard]] static seconds __at_to_sys_seconds(const __tz::__continuation& __continuation) { 1879d6b68b6SMark de Wever switch (__continuation.__at.__clock) { 1889d6b68b6SMark de Wever case __tz::__clock::__local: 1899d6b68b6SMark de Wever return __continuation.__at.__time - __continuation.__stdoff - 1909d6b68b6SMark de Wever std::visit( 1919d6b68b6SMark de Wever [](const auto& __value) { 1929d6b68b6SMark de Wever using _Tp = decay_t<decltype(__value)>; 1939d6b68b6SMark de Wever if constexpr (same_as<_Tp, monostate>) 1949d6b68b6SMark de Wever return chrono::seconds{0}; 1959d6b68b6SMark de Wever else if constexpr (same_as<_Tp, __tz::__save>) 1969d6b68b6SMark de Wever return chrono::duration_cast<seconds>(__value.__time); 1979d6b68b6SMark de Wever else if constexpr (same_as<_Tp, std::string>) 1989d6b68b6SMark de Wever // For a named rule based continuation the SAVE depends on the RULE 1999d6b68b6SMark de Wever // active at the end. This should be determined separately. 2009d6b68b6SMark de Wever return chrono::seconds{0}; 2019d6b68b6SMark de Wever else 202*7f845cbaSNikolas Klauser static_assert(false); 2039d6b68b6SMark de Wever 2049d6b68b6SMark de Wever std::__libcpp_unreachable(); 2059d6b68b6SMark de Wever }, 2069d6b68b6SMark de Wever __continuation.__rules); 2079d6b68b6SMark de Wever 2089d6b68b6SMark de Wever case __tz::__clock::__universal: 2099d6b68b6SMark de Wever return __continuation.__at.__time; 2109d6b68b6SMark de Wever 2119d6b68b6SMark de Wever case __tz::__clock::__standard: 2129d6b68b6SMark de Wever return __continuation.__at.__time - __continuation.__stdoff; 2139d6b68b6SMark de Wever } 2149d6b68b6SMark de Wever std::__libcpp_unreachable(); 2159d6b68b6SMark de Wever } 2169d6b68b6SMark de Wever 2179d6b68b6SMark de Wever [[nodiscard]] static year_month_day __to_year_month_day(year __year, month __month, __tz::__on __on) { 2189d6b68b6SMark de Wever return std::visit( 2199d6b68b6SMark de Wever [&](const auto& __value) { 2209d6b68b6SMark de Wever using _Tp = decay_t<decltype(__value)>; 2219d6b68b6SMark de Wever if constexpr (same_as<_Tp, chrono::day>) 2229d6b68b6SMark de Wever return year_month_day{__year, __month, __value}; 2239d6b68b6SMark de Wever else if constexpr (same_as<_Tp, weekday_last>) 2249d6b68b6SMark de Wever return year_month_day{static_cast<sys_days>(year_month_weekday_last{__year, __month, __value})}; 2259d6b68b6SMark de Wever else if constexpr (same_as<_Tp, __tz::__constrained_weekday>) 2269d6b68b6SMark de Wever return __value(__year, __month); 2279d6b68b6SMark de Wever else 228*7f845cbaSNikolas Klauser static_assert(false); 2299d6b68b6SMark de Wever 2309d6b68b6SMark de Wever std::__libcpp_unreachable(); 2319d6b68b6SMark de Wever }, 2329d6b68b6SMark de Wever __on); 2339d6b68b6SMark de Wever } 2349d6b68b6SMark de Wever 2359d6b68b6SMark de Wever [[nodiscard]] static sys_seconds __until_to_sys_seconds(const __tz::__continuation& __continuation) { 2369d6b68b6SMark de Wever // Does UNTIL contain the magic value for the last continuation? 2379d6b68b6SMark de Wever if (__continuation.__year == chrono::year::min()) 2389d6b68b6SMark de Wever return sys_seconds::max(); 2399d6b68b6SMark de Wever 2409d6b68b6SMark de Wever year_month_day __ymd = chrono::__to_year_month_day(__continuation.__year, __continuation.__in, __continuation.__on); 2419d6b68b6SMark de Wever return chrono::__to_sys_seconds(__ymd, chrono::__at_to_sys_seconds(__continuation)); 2429d6b68b6SMark de Wever } 2439d6b68b6SMark de Wever 2449d6b68b6SMark de Wever // Holds the UNTIL time for a continuation with a named rule. 2459d6b68b6SMark de Wever // 2469d6b68b6SMark de Wever // Unlike continuations with an fixed SAVE named rules have a variable SAVE. 2479d6b68b6SMark de Wever // This means when the UNTIL uses the local wall time the actual UNTIL value can 2489d6b68b6SMark de Wever // only be determined when the SAVE is known. This class holds that abstraction. 2499d6b68b6SMark de Wever class __named_rule_until { 2509d6b68b6SMark de Wever public: 2519d6b68b6SMark de Wever explicit __named_rule_until(const __tz::__continuation& __continuation) 2529d6b68b6SMark de Wever : __until_{chrono::__until_to_sys_seconds(__continuation)}, 2539d6b68b6SMark de Wever __needs_adjustment_{ 2549d6b68b6SMark de Wever // The last continuation of a ZONE has no UNTIL which basically is 2559d6b68b6SMark de Wever // until the end of _local_ time. This value is expressed by 2569d6b68b6SMark de Wever // sys_seconds::max(). Subtracting the SAVE leaves large value. 2579d6b68b6SMark de Wever // However SAVE can be negative, which would add a value to maximum 2589d6b68b6SMark de Wever // leading to undefined behaviour. In practice this often results in 2599d6b68b6SMark de Wever // an overflow to a very small value. 2609d6b68b6SMark de Wever __until_ != sys_seconds::max() && __continuation.__at.__clock == __tz::__clock::__local} {} 2619d6b68b6SMark de Wever 2629d6b68b6SMark de Wever // Gives the unadjusted until value, this is useful when the SAVE is not known 2639d6b68b6SMark de Wever // at all. 2649d6b68b6SMark de Wever sys_seconds __until() const noexcept { return __until_; } 2659d6b68b6SMark de Wever 2669d6b68b6SMark de Wever bool __needs_adjustment() const noexcept { return __needs_adjustment_; } 2679d6b68b6SMark de Wever 2689d6b68b6SMark de Wever // Returns the UNTIL adjusted for SAVE. 2699d6b68b6SMark de Wever sys_seconds operator()(seconds __save) const noexcept { return __until_ - __needs_adjustment_ * __save; } 2709d6b68b6SMark de Wever 2719d6b68b6SMark de Wever private: 2729d6b68b6SMark de Wever sys_seconds __until_; 2739d6b68b6SMark de Wever bool __needs_adjustment_; 2749d6b68b6SMark de Wever }; 2759d6b68b6SMark de Wever 2769d6b68b6SMark de Wever [[nodiscard]] static seconds __at_to_seconds(seconds __stdoff, const __tz::__rule& __rule) { 2779d6b68b6SMark de Wever switch (__rule.__at.__clock) { 2789d6b68b6SMark de Wever case __tz::__clock::__local: 2799d6b68b6SMark de Wever // Local time and standard time behave the same. This is not 2809d6b68b6SMark de Wever // correct. Local time needs to adjust for the current saved time. 2819d6b68b6SMark de Wever // To know the saved time the rules need to be known and sorted. 2829d6b68b6SMark de Wever // This needs a time so to avoid the chicken and egg adjust the 2839d6b68b6SMark de Wever // saving of the local time later. 2849d6b68b6SMark de Wever return __rule.__at.__time - __stdoff; 2859d6b68b6SMark de Wever 2869d6b68b6SMark de Wever case __tz::__clock::__universal: 2879d6b68b6SMark de Wever return __rule.__at.__time; 2889d6b68b6SMark de Wever 2899d6b68b6SMark de Wever case __tz::__clock::__standard: 2909d6b68b6SMark de Wever return __rule.__at.__time - __stdoff; 2919d6b68b6SMark de Wever } 2929d6b68b6SMark de Wever std::__libcpp_unreachable(); 2939d6b68b6SMark de Wever } 2949d6b68b6SMark de Wever 2959d6b68b6SMark de Wever [[nodiscard]] static sys_seconds __from_to_sys_seconds(seconds __stdoff, const __tz::__rule& __rule, year __year) { 2969d6b68b6SMark de Wever year_month_day __ymd = chrono::__to_year_month_day(__year, __rule.__in, __rule.__on); 2979d6b68b6SMark de Wever 2989d6b68b6SMark de Wever seconds __at = chrono::__at_to_seconds(__stdoff, __rule); 2999d6b68b6SMark de Wever return chrono::__to_sys_seconds(__ymd, __at); 3009d6b68b6SMark de Wever } 3019d6b68b6SMark de Wever 3029d6b68b6SMark de Wever [[nodiscard]] static sys_seconds __from_to_sys_seconds(seconds __stdoff, const __tz::__rule& __rule) { 3039d6b68b6SMark de Wever return chrono::__from_to_sys_seconds(__stdoff, __rule, __rule.__from); 3049d6b68b6SMark de Wever } 3059d6b68b6SMark de Wever 3069d6b68b6SMark de Wever [[nodiscard]] static const vector<__tz::__rule>& 3079d6b68b6SMark de Wever __get_rules(const __tz::__rules_storage_type& __rules_db, const string& __rule_name) { 3089d6b68b6SMark de Wever auto __result = chrono::__binary_find(__rules_db, __rule_name, {}, [](const auto& __p) { return __p.first; }); 3099d6b68b6SMark de Wever if (__result == std::end(__rules_db)) 3109d6b68b6SMark de Wever std::__throw_runtime_error(("corrupt tzdb: rule '" + __rule_name + " 'does not exist").c_str()); 3119d6b68b6SMark de Wever 3129d6b68b6SMark de Wever return __result->second; 3139d6b68b6SMark de Wever } 3149d6b68b6SMark de Wever 3159d6b68b6SMark de Wever // Returns the letters field for a time before the first rule. 3169d6b68b6SMark de Wever // 3179d6b68b6SMark de Wever // Per https://data.iana.org/time-zones/tz-how-to.html 3189d6b68b6SMark de Wever // One wrinkle, not fully explained in zic.8.txt, is what happens when switching 3199d6b68b6SMark de Wever // to a named rule. To what values should the SAVE and LETTER data be 3209d6b68b6SMark de Wever // initialized? 3219d6b68b6SMark de Wever // 3229d6b68b6SMark de Wever // 1 If at least one transition has happened, use the SAVE and LETTER data from 3239d6b68b6SMark de Wever // the most recent. 3249d6b68b6SMark de Wever // 2 If switching to a named rule before any transition has happened, assume 3259d6b68b6SMark de Wever // standard time (SAVE zero), and use the LETTER data from the earliest 3269d6b68b6SMark de Wever // transition with a SAVE of zero. 3279d6b68b6SMark de Wever // 3289d6b68b6SMark de Wever // This function implements case 2. 3299d6b68b6SMark de Wever [[nodiscard]] static string __letters_before_first_rule(const vector<__tz::__rule>& __rules) { 3309d6b68b6SMark de Wever auto __letters = 3319d6b68b6SMark de Wever __rules // 3329d6b68b6SMark de Wever | views::filter([](const __tz::__rule& __rule) { return __rule.__save.__time == 0s; }) // 3339d6b68b6SMark de Wever | views::transform([](const __tz::__rule& __rule) { return __rule.__letters; }) // 3349d6b68b6SMark de Wever | views::take(1); 3359d6b68b6SMark de Wever 3369d6b68b6SMark de Wever if (__letters.empty()) 3379d6b68b6SMark de Wever std::__throw_runtime_error("corrupt tzdb: rule has zero entries"); 3389d6b68b6SMark de Wever 3399d6b68b6SMark de Wever return __letters.front(); 3409d6b68b6SMark de Wever } 3419d6b68b6SMark de Wever 3429d6b68b6SMark de Wever // Determines the information based on the continuation and the rules. 3439d6b68b6SMark de Wever // 3449d6b68b6SMark de Wever // There are several special cases to take into account 3459d6b68b6SMark de Wever // 3469d6b68b6SMark de Wever // === Entries before the first rule becomes active === 3479d6b68b6SMark de Wever // Asia/Hong_Kong 3489d6b68b6SMark de Wever // 9 - JST 1945 N 18 2 // (1) 3499d6b68b6SMark de Wever // 8 HK HK%sT // (2) 3509d6b68b6SMark de Wever // R HK 1946 o - Ap 21 0 1 S // (3) 3519d6b68b6SMark de Wever // There (1) is active until Novemer 18th 1945 at 02:00, after this time 3529d6b68b6SMark de Wever // (2) becomes active. The first rule entry for HK (3) becomes active 3539d6b68b6SMark de Wever // from April 21st 1945 at 01:00. In the period between (2) is active. 3549d6b68b6SMark de Wever // This entry has an offset. 3559d6b68b6SMark de Wever // This entry has no save, letters, or dst flag. So in the period 3569d6b68b6SMark de Wever // after (1) and until (3) no rule entry is associated with the time. 3579d6b68b6SMark de Wever 3589d6b68b6SMark de Wever [[nodiscard]] static sys_info __get_sys_info_before_first_rule( 3599d6b68b6SMark de Wever sys_seconds __begin, 3609d6b68b6SMark de Wever sys_seconds __end, 3619d6b68b6SMark de Wever const __tz::__continuation& __continuation, 3629d6b68b6SMark de Wever const vector<__tz::__rule>& __rules) { 3639d6b68b6SMark de Wever return sys_info{ 3649d6b68b6SMark de Wever __begin, 3659d6b68b6SMark de Wever __end, 3669d6b68b6SMark de Wever __continuation.__stdoff, 3679d6b68b6SMark de Wever chrono::minutes(0), 3689d6b68b6SMark de Wever chrono::__format(__continuation, __letters_before_first_rule(__rules), 0s)}; 3699d6b68b6SMark de Wever } 3709d6b68b6SMark de Wever 3719d6b68b6SMark de Wever // Returns the sys_info object for a time before the first rule. 3729d6b68b6SMark de Wever // When this first rule has a SAVE of 0s the sys_info for the time before the 3739d6b68b6SMark de Wever // first rule and for the first rule are identical and will be merged. 3749d6b68b6SMark de Wever [[nodiscard]] static sys_info __get_sys_info_before_first_rule( 3759d6b68b6SMark de Wever sys_seconds __begin, 3769d6b68b6SMark de Wever sys_seconds __rule_end, // The end used when SAVE != 0s 3779d6b68b6SMark de Wever sys_seconds __next_end, // The end used when SAVE == 0s the times are merged 3789d6b68b6SMark de Wever const __tz::__continuation& __continuation, 3799d6b68b6SMark de Wever const vector<__tz::__rule>& __rules, 3809d6b68b6SMark de Wever vector<__tz::__rule>::const_iterator __rule) { 3819d6b68b6SMark de Wever if (__rule->__save.__time != 0s) 3829d6b68b6SMark de Wever return __get_sys_info_before_first_rule(__begin, __rule_end, __continuation, __rules); 3839d6b68b6SMark de Wever 3849d6b68b6SMark de Wever return sys_info{ 3859d6b68b6SMark de Wever __begin, __next_end, __continuation.__stdoff, 0min, chrono::__format(__continuation, __rule->__letters, 0s)}; 3869d6b68b6SMark de Wever } 3879d6b68b6SMark de Wever 3889d6b68b6SMark de Wever [[nodiscard]] static seconds __at_to_seconds(seconds __stdoff, seconds __save, const __tz::__rule& __rule) { 3899d6b68b6SMark de Wever switch (__rule.__at.__clock) { 3909d6b68b6SMark de Wever case __tz::__clock::__local: 3919d6b68b6SMark de Wever return __rule.__at.__time - __stdoff - __save; 3929d6b68b6SMark de Wever 3939d6b68b6SMark de Wever case __tz::__clock::__universal: 3949d6b68b6SMark de Wever return __rule.__at.__time; 3959d6b68b6SMark de Wever 3969d6b68b6SMark de Wever case __tz::__clock::__standard: 3979d6b68b6SMark de Wever return __rule.__at.__time - __stdoff; 3989d6b68b6SMark de Wever } 3999d6b68b6SMark de Wever std::__libcpp_unreachable(); 4009d6b68b6SMark de Wever } 4019d6b68b6SMark de Wever 4029d6b68b6SMark de Wever [[nodiscard]] static sys_seconds 4039d6b68b6SMark de Wever __rule_to_sys_seconds(seconds __stdoff, seconds __save, const __tz::__rule& __rule, year __year) { 4049d6b68b6SMark de Wever year_month_day __ymd = chrono::__to_year_month_day(__year, __rule.__in, __rule.__on); 4059d6b68b6SMark de Wever 4069d6b68b6SMark de Wever seconds __at = chrono::__at_to_seconds(__stdoff, __save, __rule); 4079d6b68b6SMark de Wever return chrono::__to_sys_seconds(__ymd, __at); 4089d6b68b6SMark de Wever } 4099d6b68b6SMark de Wever 4109d6b68b6SMark de Wever // Returns the first rule after __time. 4119d6b68b6SMark de Wever // Note that a rule can be "active" in multiple years, this may result in an 4129d6b68b6SMark de Wever // infinite loop where the same rule is returned every time, use __current to 4139d6b68b6SMark de Wever // guard against that. 4149d6b68b6SMark de Wever // 4159d6b68b6SMark de Wever // When no next rule exists the returned time will be sys_seconds::max(). This 4169d6b68b6SMark de Wever // can happen in practice. For example, 4179d6b68b6SMark de Wever // 4189d6b68b6SMark de Wever // R So 1945 o - May 24 2 2 M 4199d6b68b6SMark de Wever // R So 1945 o - S 24 3 1 S 4209d6b68b6SMark de Wever // R So 1945 o - N 18 2s 0 - 4219d6b68b6SMark de Wever // 4229d6b68b6SMark de Wever // Has 3 rules that are all only active in 1945. 4239d6b68b6SMark de Wever [[nodiscard]] static pair<sys_seconds, vector<__tz::__rule>::const_iterator> 4249d6b68b6SMark de Wever __next_rule(sys_seconds __time, 4259d6b68b6SMark de Wever seconds __stdoff, 4269d6b68b6SMark de Wever seconds __save, 4279d6b68b6SMark de Wever const vector<__tz::__rule>& __rules, 4289d6b68b6SMark de Wever vector<__tz::__rule>::const_iterator __current) { 4299d6b68b6SMark de Wever year __year = year_month_day{chrono::floor<days>(__time)}.year(); 4309d6b68b6SMark de Wever 4319d6b68b6SMark de Wever // Note it would probably be better to store the pairs in a vector and then 4329d6b68b6SMark de Wever // use min() to get the smallest element 4339d6b68b6SMark de Wever map<sys_seconds, vector<__tz::__rule>::const_iterator> __candidates; 4349d6b68b6SMark de Wever // Note this evaluates all rules which is a waste of effort; when the entries 4359d6b68b6SMark de Wever // are beyond the current year's "next year" (where "next year" is not always 4369d6b68b6SMark de Wever // year + 1) the algorithm should end. 4379d6b68b6SMark de Wever for (auto __it = __rules.begin(); __it != __rules.end(); ++__it) { 4389d6b68b6SMark de Wever for (year __y = __it->__from; __y <= __it->__to; ++__y) { 4399d6b68b6SMark de Wever // Adding the current entry for the current year may lead to infinite 4409d6b68b6SMark de Wever // loops due to the SAVE adjustment. Skip these entries. 4419d6b68b6SMark de Wever if (__y == __year && __it == __current) 4429d6b68b6SMark de Wever continue; 4439d6b68b6SMark de Wever 4449d6b68b6SMark de Wever sys_seconds __t = chrono::__rule_to_sys_seconds(__stdoff, __save, *__it, __y); 4459d6b68b6SMark de Wever if (__t <= __time) 4469d6b68b6SMark de Wever continue; 4479d6b68b6SMark de Wever 4489d6b68b6SMark de Wever _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN(!__candidates.contains(__t), "duplicated rule"); 4499d6b68b6SMark de Wever __candidates[__t] = __it; 4509d6b68b6SMark de Wever break; 4519d6b68b6SMark de Wever } 4529d6b68b6SMark de Wever } 4539d6b68b6SMark de Wever 4549d6b68b6SMark de Wever if (!__candidates.empty()) [[likely]] { 4559d6b68b6SMark de Wever auto __it = __candidates.begin(); 4569d6b68b6SMark de Wever 4579d6b68b6SMark de Wever // When no rule is selected the time before the first rule and the first rule 4589d6b68b6SMark de Wever // should not be merged. 4599d6b68b6SMark de Wever if (__time == sys_seconds::min()) 4609d6b68b6SMark de Wever return *__it; 4619d6b68b6SMark de Wever 4629d6b68b6SMark de Wever // There can be two constitutive rules that are the same. For example, 4639d6b68b6SMark de Wever // Hong Kong 4649d6b68b6SMark de Wever // 4659d6b68b6SMark de Wever // R HK 1973 o - D 30 3:30 1 S (R1) 4669d6b68b6SMark de Wever // R HK 1965 1976 - Ap Su>=16 3:30 1 S (R2) 4679d6b68b6SMark de Wever // 4689d6b68b6SMark de Wever // 1973-12-29 19:30:00 R1 becomes active. 4699d6b68b6SMark de Wever // 1974-04-20 18:30:00 R2 becomes active. 4709d6b68b6SMark de Wever // Both rules have a SAVE of 1 hour and LETTERS are S for both of them. 4719d6b68b6SMark de Wever while (__it != __candidates.end()) { 4729d6b68b6SMark de Wever if (__current->__save.__time != __it->second->__save.__time || __current->__letters != __it->second->__letters) 4739d6b68b6SMark de Wever return *__it; 4749d6b68b6SMark de Wever 4759d6b68b6SMark de Wever ++__it; 4769d6b68b6SMark de Wever } 4779d6b68b6SMark de Wever } 4789d6b68b6SMark de Wever 4799d6b68b6SMark de Wever return {sys_seconds::max(), __rules.end()}; 4809d6b68b6SMark de Wever } 4819d6b68b6SMark de Wever 4829d6b68b6SMark de Wever // Returns the first rule of a set of rules. 4839d6b68b6SMark de Wever // This is not always the first of the listed rules. For example 4849d6b68b6SMark de Wever // R Sa 2008 2009 - Mar Su>=8 0 0 - 4859d6b68b6SMark de Wever // R Sa 2007 2008 - O Su>=8 0 1 - 4869d6b68b6SMark de Wever // The transition in October 2007 happens before the transition in March 2008. 4879d6b68b6SMark de Wever [[nodiscard]] static vector<__tz::__rule>::const_iterator 4889d6b68b6SMark de Wever __first_rule(seconds __stdoff, const vector<__tz::__rule>& __rules) { 4899d6b68b6SMark de Wever return chrono::__next_rule(sys_seconds::min(), __stdoff, 0s, __rules, __rules.end()).second; 4909d6b68b6SMark de Wever } 4919d6b68b6SMark de Wever 4929d6b68b6SMark de Wever [[nodiscard]] static __sys_info_result __get_sys_info_rule( 4939d6b68b6SMark de Wever sys_seconds __time, 4949d6b68b6SMark de Wever sys_seconds __continuation_begin, 4959d6b68b6SMark de Wever const __tz::__continuation& __continuation, 4969d6b68b6SMark de Wever const vector<__tz::__rule>& __rules) { 4979d6b68b6SMark de Wever auto __rule = chrono::__first_rule(__continuation.__stdoff, __rules); 4989d6b68b6SMark de Wever _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN(__rule != __rules.end(), "the set of rules has no first rule"); 4999d6b68b6SMark de Wever 5009d6b68b6SMark de Wever // Avoid selecting a time before the start of the continuation 5019d6b68b6SMark de Wever __time = std::max(__time, __continuation_begin); 5029d6b68b6SMark de Wever 5039d6b68b6SMark de Wever sys_seconds __rule_begin = chrono::__from_to_sys_seconds(__continuation.__stdoff, *__rule); 5049d6b68b6SMark de Wever 5059d6b68b6SMark de Wever // The time sought is very likely inside the current rule. 5069d6b68b6SMark de Wever // When the continuation's UNTIL uses the local clock there are edge cases 5079d6b68b6SMark de Wever // where this is not true. 5089d6b68b6SMark de Wever // 5099d6b68b6SMark de Wever // Start to walk the rules to find the proper one. 5109d6b68b6SMark de Wever // 5119d6b68b6SMark de Wever // For now we just walk all the rules TODO TZDB investigate whether a smarter 5129d6b68b6SMark de Wever // algorithm would work. 5139d6b68b6SMark de Wever auto __next = chrono::__next_rule(__rule_begin, __continuation.__stdoff, __rule->__save.__time, __rules, __rule); 5149d6b68b6SMark de Wever 5159d6b68b6SMark de Wever // Ignore small steps, this happens with America/Punta_Arenas for the 5169d6b68b6SMark de Wever // transition 5179d6b68b6SMark de Wever // -4:42:46 - SMT 1927 S 5189d6b68b6SMark de Wever // -5 x -05/-04 1932 S 5199d6b68b6SMark de Wever // ... 5209d6b68b6SMark de Wever // 5219d6b68b6SMark de Wever // R x 1927 1931 - S 1 0 1 - 5229d6b68b6SMark de Wever // R x 1928 1932 - Ap 1 0 0 - 5239d6b68b6SMark de Wever // 5249d6b68b6SMark de Wever // America/Punta_Arenas Thu Sep 1 04:42:45 1927 UT = Thu Sep 1 00:42:45 1927 -04 isdst=1 gmtoff=-14400 5259d6b68b6SMark de Wever // America/Punta_Arenas Sun Apr 1 03:59:59 1928 UT = Sat Mar 31 23:59:59 1928 -04 isdst=1 gmtoff=-14400 5269d6b68b6SMark de Wever // America/Punta_Arenas Sun Apr 1 04:00:00 1928 UT = Sat Mar 31 23:00:00 1928 -05 isdst=0 gmtoff=-18000 5279d6b68b6SMark de Wever // 5289d6b68b6SMark de Wever // Without this there will be a transition 5299d6b68b6SMark de Wever // [1927-09-01 04:42:45, 1927-09-01 05:00:00) -05:00:00 0min -05 5309d6b68b6SMark de Wever 5319d6b68b6SMark de Wever if (sys_seconds __begin = __rule->__save.__time != 0s ? __rule_begin : __next.first; __time < __begin) { 5329d6b68b6SMark de Wever if (__continuation_begin == sys_seconds::min() || __begin - __continuation_begin > 12h) 5339d6b68b6SMark de Wever return __sys_info{__get_sys_info_before_first_rule( 5349d6b68b6SMark de Wever __continuation_begin, __rule_begin, __next.first, __continuation, __rules, __rule), 5359d6b68b6SMark de Wever false}; 5369d6b68b6SMark de Wever 5379d6b68b6SMark de Wever // Europe/Berlin 5389d6b68b6SMark de Wever // 1 c CE%sT 1945 May 24 2 (C1) 5399d6b68b6SMark de Wever // 1 So CE%sT 1946 (C2) 5409d6b68b6SMark de Wever // 5419d6b68b6SMark de Wever // R c 1944 1945 - Ap M>=1 2s 1 S (R1) 5429d6b68b6SMark de Wever // 5439d6b68b6SMark de Wever // R So 1945 o - May 24 2 2 M (R2) 5449d6b68b6SMark de Wever // 5459d6b68b6SMark de Wever // When C2 becomes active the time would be before the first rule R2, 5469d6b68b6SMark de Wever // giving a 1 hour sys_info. 5479d6b68b6SMark de Wever seconds __save = __rule->__save.__time; 5489d6b68b6SMark de Wever __named_rule_until __continuation_end{__continuation}; 5499d6b68b6SMark de Wever sys_seconds __sys_info_end = std::min(__continuation_end(__save), __next.first); 5509d6b68b6SMark de Wever 5519d6b68b6SMark de Wever return __sys_info{ 5529d6b68b6SMark de Wever sys_info{__continuation_begin, 5539d6b68b6SMark de Wever __sys_info_end, 5549d6b68b6SMark de Wever __continuation.__stdoff + __save, 5559d6b68b6SMark de Wever chrono::duration_cast<minutes>(__save), 5569d6b68b6SMark de Wever chrono::__format(__continuation, __rule->__letters, __save)}, 5579d6b68b6SMark de Wever __sys_info_end == __continuation_end(__save)}; 5589d6b68b6SMark de Wever } 5599d6b68b6SMark de Wever 5609d6b68b6SMark de Wever // See above for America/Asuncion 5619d6b68b6SMark de Wever if (__rule->__save.__time == 0s && __time < __next.first) { 5629d6b68b6SMark de Wever return __sys_info{ 5639d6b68b6SMark de Wever sys_info{__continuation_begin, 5649d6b68b6SMark de Wever __next.first, 5659d6b68b6SMark de Wever __continuation.__stdoff, 5669d6b68b6SMark de Wever 0min, 5679d6b68b6SMark de Wever chrono::__format(__continuation, __rule->__letters, 0s)}, 5689d6b68b6SMark de Wever false}; 5699d6b68b6SMark de Wever } 5709d6b68b6SMark de Wever 5719d6b68b6SMark de Wever if (__rule->__save.__time != 0s) { 5729d6b68b6SMark de Wever // another fix for America/Punta_Arenas when not at the start of the 5739d6b68b6SMark de Wever // sys_info object. 5749d6b68b6SMark de Wever seconds __save = __rule->__save.__time; 5759d6b68b6SMark de Wever if (__continuation_begin >= __rule_begin - __save && __time < __next.first) { 5769d6b68b6SMark de Wever return __sys_info{ 5779d6b68b6SMark de Wever sys_info{__continuation_begin, 5789d6b68b6SMark de Wever __next.first, 5799d6b68b6SMark de Wever __continuation.__stdoff + __save, 5809d6b68b6SMark de Wever chrono::duration_cast<minutes>(__save), 5819d6b68b6SMark de Wever chrono::__format(__continuation, __rule->__letters, __save)}, 5829d6b68b6SMark de Wever false}; 5839d6b68b6SMark de Wever } 5849d6b68b6SMark de Wever } 5859d6b68b6SMark de Wever 5869d6b68b6SMark de Wever __named_rule_until __continuation_end{__continuation}; 5879d6b68b6SMark de Wever while (__next.second != __rules.end()) { 5889d6b68b6SMark de Wever #ifdef PRINT 5899d6b68b6SMark de Wever std::print( 5909d6b68b6SMark de Wever stderr, 5919d6b68b6SMark de Wever "Rule for {}: [{}, {}) off={} save={} duration={}\n", 5929d6b68b6SMark de Wever __time, 5939d6b68b6SMark de Wever __rule_begin, 5949d6b68b6SMark de Wever __next.first, 5959d6b68b6SMark de Wever __continuation.__stdoff, 5969d6b68b6SMark de Wever __rule->__save.__time, 5979d6b68b6SMark de Wever __next.first - __rule_begin); 5989d6b68b6SMark de Wever #endif 5999d6b68b6SMark de Wever 6009d6b68b6SMark de Wever sys_seconds __end = __continuation_end(__rule->__save.__time); 6019d6b68b6SMark de Wever 6029d6b68b6SMark de Wever sys_seconds __sys_info_begin = std::max(__continuation_begin, __rule_begin); 6039d6b68b6SMark de Wever sys_seconds __sys_info_end = std::min(__end, __next.first); 6049d6b68b6SMark de Wever seconds __diff = chrono::abs(__sys_info_end - __sys_info_begin); 6059d6b68b6SMark de Wever 6069d6b68b6SMark de Wever if (__diff < 12h) { 6079d6b68b6SMark de Wever // Z America/Argentina/Buenos_Aires -3:53:48 - LMT 1894 O 31 6089d6b68b6SMark de Wever // -4:16:48 - CMT 1920 May 6099d6b68b6SMark de Wever // -4 - -04 1930 D 6109d6b68b6SMark de Wever // -4 A -04/-03 1969 O 5 6119d6b68b6SMark de Wever // -3 A -03/-02 1999 O 3 6129d6b68b6SMark de Wever // -4 A -04/-03 2000 Mar 3 6139d6b68b6SMark de Wever // ... 6149d6b68b6SMark de Wever // 6159d6b68b6SMark de Wever // ... 6169d6b68b6SMark de Wever // R A 1989 1992 - O Su>=15 0 1 - 6179d6b68b6SMark de Wever // R A 1999 o - O Su>=1 0 1 - 6189d6b68b6SMark de Wever // R A 2000 o - Mar 3 0 0 - 6199d6b68b6SMark de Wever // R A 2007 o - D 30 0 1 - 6209d6b68b6SMark de Wever // ... 6219d6b68b6SMark de Wever 6229d6b68b6SMark de Wever // The 1999 switch uses the same rule, but with a different stdoff. 6239d6b68b6SMark de Wever // R A 1999 o - O Su>=1 0 1 - 6249d6b68b6SMark de Wever // stdoff -3 -> 1999-10-03 03:00:00 6259d6b68b6SMark de Wever // stdoff -4 -> 1999-10-03 04:00:00 6269d6b68b6SMark de Wever // This generates an invalid entry and this is evaluated as a transition. 6279d6b68b6SMark de Wever // Looking at the zdump like output in libc++ this generates jumps in 6289d6b68b6SMark de Wever // the UTC time. 6299d6b68b6SMark de Wever 6309d6b68b6SMark de Wever __rule = __next.second; 6319d6b68b6SMark de Wever __next = __next_rule(__next.first, __continuation.__stdoff, __rule->__save.__time, __rules, __rule); 6329d6b68b6SMark de Wever __end = __continuation_end(__rule->__save.__time); 6339d6b68b6SMark de Wever __sys_info_end = std::min(__end, __next.first); 6349d6b68b6SMark de Wever } 6359d6b68b6SMark de Wever 6369d6b68b6SMark de Wever if ((__time >= __rule_begin && __time < __next.first) || __next.first >= __end) { 6379d6b68b6SMark de Wever __sys_info_begin = std::max(__continuation_begin, __rule_begin); 6389d6b68b6SMark de Wever __sys_info_end = std::min(__end, __next.first); 6399d6b68b6SMark de Wever 6409d6b68b6SMark de Wever return __sys_info{ 6419d6b68b6SMark de Wever sys_info{__sys_info_begin, 6429d6b68b6SMark de Wever __sys_info_end, 6439d6b68b6SMark de Wever __continuation.__stdoff + __rule->__save.__time, 6449d6b68b6SMark de Wever chrono::duration_cast<minutes>(__rule->__save.__time), 6459d6b68b6SMark de Wever chrono::__format(__continuation, __rule->__letters, __rule->__save.__time)}, 6469d6b68b6SMark de Wever __sys_info_end == __end}; 6479d6b68b6SMark de Wever } 6489d6b68b6SMark de Wever 6499d6b68b6SMark de Wever __rule_begin = __next.first; 6509d6b68b6SMark de Wever __rule = __next.second; 6519d6b68b6SMark de Wever __next = __next_rule(__rule_begin, __continuation.__stdoff, __rule->__save.__time, __rules, __rule); 6529d6b68b6SMark de Wever } 6539d6b68b6SMark de Wever 6549d6b68b6SMark de Wever return __sys_info{ 6559d6b68b6SMark de Wever sys_info{std::max(__continuation_begin, __rule_begin), 6569d6b68b6SMark de Wever __continuation_end(__rule->__save.__time), 6579d6b68b6SMark de Wever __continuation.__stdoff + __rule->__save.__time, 6589d6b68b6SMark de Wever chrono::duration_cast<minutes>(__rule->__save.__time), 6599d6b68b6SMark de Wever chrono::__format(__continuation, __rule->__letters, __rule->__save.__time)}, 6609d6b68b6SMark de Wever true}; 6619d6b68b6SMark de Wever } 6629d6b68b6SMark de Wever 6639d6b68b6SMark de Wever [[nodiscard]] static __sys_info_result __get_sys_info_basic( 6649d6b68b6SMark de Wever sys_seconds __time, sys_seconds __continuation_begin, const __tz::__continuation& __continuation, seconds __save) { 6659d6b68b6SMark de Wever sys_seconds __continuation_end = chrono::__until_to_sys_seconds(__continuation); 6669d6b68b6SMark de Wever return __sys_info{ 6679d6b68b6SMark de Wever sys_info{__continuation_begin, 6689d6b68b6SMark de Wever __continuation_end, 6699d6b68b6SMark de Wever __continuation.__stdoff + __save, 6709d6b68b6SMark de Wever chrono::duration_cast<minutes>(__save), 6719d6b68b6SMark de Wever __continuation.__format}, 6729d6b68b6SMark de Wever true}; 6739d6b68b6SMark de Wever } 6749d6b68b6SMark de Wever 6759d6b68b6SMark de Wever [[nodiscard]] static __sys_info_result 6769d6b68b6SMark de Wever __get_sys_info(sys_seconds __time, 6779d6b68b6SMark de Wever sys_seconds __continuation_begin, 6789d6b68b6SMark de Wever const __tz::__continuation& __continuation, 6799d6b68b6SMark de Wever const __tz::__rules_storage_type& __rules_db) { 6809d6b68b6SMark de Wever return std::visit( 6819d6b68b6SMark de Wever [&](const auto& __value) { 6829d6b68b6SMark de Wever using _Tp = decay_t<decltype(__value)>; 6839d6b68b6SMark de Wever if constexpr (same_as<_Tp, std::string>) 6849d6b68b6SMark de Wever return chrono::__get_sys_info_rule( 6859d6b68b6SMark de Wever __time, __continuation_begin, __continuation, __get_rules(__rules_db, __value)); 6869d6b68b6SMark de Wever else if constexpr (same_as<_Tp, monostate>) 6879d6b68b6SMark de Wever return chrono::__get_sys_info_basic(__time, __continuation_begin, __continuation, chrono::seconds(0)); 6889d6b68b6SMark de Wever else if constexpr (same_as<_Tp, __tz::__save>) 6899d6b68b6SMark de Wever return chrono::__get_sys_info_basic(__time, __continuation_begin, __continuation, __value.__time); 6909d6b68b6SMark de Wever else 691*7f845cbaSNikolas Klauser static_assert(false); 6929d6b68b6SMark de Wever 6939d6b68b6SMark de Wever std::__libcpp_unreachable(); 6949d6b68b6SMark de Wever }, 6959d6b68b6SMark de Wever __continuation.__rules); 6969d6b68b6SMark de Wever } 6979d6b68b6SMark de Wever 6989d6b68b6SMark de Wever // The transition from one continuation to the next continuation may result in 6999d6b68b6SMark de Wever // two constitutive continuations with the same "offset" information. 7009d6b68b6SMark de Wever // [time.zone.info.sys]/3 7019d6b68b6SMark de Wever // The begin and end data members indicate that, for the associated time_zone 7029d6b68b6SMark de Wever // and time_point, the offset and abbrev are in effect in the range 7039d6b68b6SMark de Wever // [begin, end). This information can be used to efficiently iterate the 7049d6b68b6SMark de Wever // transitions of a time_zone. 7059d6b68b6SMark de Wever // 7069d6b68b6SMark de Wever // Note that this does considers a change in the SAVE field not to be a 7079d6b68b6SMark de Wever // different sys_info, zdump does consider this different. 7089d6b68b6SMark de Wever // LWG XXXX The sys_info range should be affected by save 7099d6b68b6SMark de Wever // matches the behaviour of the Standard and zdump. 7109d6b68b6SMark de Wever // 7119d6b68b6SMark de Wever // Iff the "offsets" are the same '__current.__end' is replaced with 7129d6b68b6SMark de Wever // '__next.__end', which effectively merges the two objects in one object. The 7139d6b68b6SMark de Wever // function returns true if a merge occurred. 7149d6b68b6SMark de Wever [[nodiscard]] bool __merge_continuation(sys_info& __current, const sys_info& __next) { 7159d6b68b6SMark de Wever if (__current.end != __next.begin) 7169d6b68b6SMark de Wever return false; 7179d6b68b6SMark de Wever 7189d6b68b6SMark de Wever if (__current.offset != __next.offset || __current.abbrev != __next.abbrev || __current.save != __next.save) 7199d6b68b6SMark de Wever return false; 7209d6b68b6SMark de Wever 7219d6b68b6SMark de Wever __current.end = __next.end; 7229d6b68b6SMark de Wever return true; 7239d6b68b6SMark de Wever } 7249d6b68b6SMark de Wever 7259d6b68b6SMark de Wever //===----------------------------------------------------------------------===// 7269d6b68b6SMark de Wever // Public API 7279d6b68b6SMark de Wever //===----------------------------------------------------------------------===// 7289d6b68b6SMark de Wever 7299d6b68b6SMark de Wever [[nodiscard]] _LIBCPP_EXPORTED_FROM_ABI time_zone time_zone::__create(unique_ptr<time_zone::__impl>&& __p) { 7309d6b68b6SMark de Wever _LIBCPP_ASSERT_NON_NULL(__p != nullptr, "initialized time_zone without a valid pimpl object"); 7319d6b68b6SMark de Wever time_zone result; 7329d6b68b6SMark de Wever result.__impl_ = std::move(__p); 7339d6b68b6SMark de Wever return result; 7349d6b68b6SMark de Wever } 7359d6b68b6SMark de Wever 7369d6b68b6SMark de Wever _LIBCPP_EXPORTED_FROM_ABI time_zone::~time_zone() = default; 7379d6b68b6SMark de Wever 7389d6b68b6SMark de Wever [[nodiscard]] _LIBCPP_EXPORTED_FROM_ABI string_view time_zone::__name() const noexcept { return __impl_->__name(); } 7399d6b68b6SMark de Wever 7409d6b68b6SMark de Wever [[nodiscard]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_EXPORTED_FROM_ABI sys_info 7419d6b68b6SMark de Wever time_zone::__get_info(sys_seconds __time) const { 7429d6b68b6SMark de Wever optional<sys_info> __result; 7439d6b68b6SMark de Wever bool __valid_result = false; // true iff __result.has_value() is true and 7449d6b68b6SMark de Wever // __result.begin <= __time < __result.end is true. 7459d6b68b6SMark de Wever bool __can_merge = false; 7469d6b68b6SMark de Wever sys_seconds __continuation_begin = sys_seconds::min(); 7479d6b68b6SMark de Wever // Iterates over the Zone entry and its continuations. Internally the Zone 7489d6b68b6SMark de Wever // entry is split in a Zone information and the first continuation. The last 7499d6b68b6SMark de Wever // continuation has no UNTIL field. This means the loop should always find a 7509d6b68b6SMark de Wever // continuation. 7519d6b68b6SMark de Wever // 7529d6b68b6SMark de Wever // For more information on background of zone information please consult the 7539d6b68b6SMark de Wever // following information 7549d6b68b6SMark de Wever // [zic manual](https://www.man7.org/linux/man-pages/man8/zic.8.html) 7559d6b68b6SMark de Wever // [tz source info](https://data.iana.org/time-zones/tz-how-to.html) 7569d6b68b6SMark de Wever // On POSIX systems the zdump tool can be useful: 7579d6b68b6SMark de Wever // zdump -v Asia/Hong_Kong 7589d6b68b6SMark de Wever // Gives all transitions in the Hong Kong time zone. 7599d6b68b6SMark de Wever // 7609d6b68b6SMark de Wever // During iteration the result for the current continuation is returned. If 7619d6b68b6SMark de Wever // no continuation is applicable it will return the end time as "error". When 7629d6b68b6SMark de Wever // two continuations are contiguous and contain the "same" information these 7639d6b68b6SMark de Wever // ranges are merged as one range. 7649d6b68b6SMark de Wever // The merging requires keeping any result that occurs before __time, 7659d6b68b6SMark de Wever // likewise when a valid result is found the algorithm needs to test the next 7669d6b68b6SMark de Wever // continuation to see whether it can be merged. For example, Africa/Ceuta 7679d6b68b6SMark de Wever // Continuations 7689d6b68b6SMark de Wever // 0 s WE%sT 1929 (C1) 7699d6b68b6SMark de Wever // 0 - WET 1967 (C2) 7709d6b68b6SMark de Wever // 0 Sp WE%sT 1984 Mar 16 (C3) 7719d6b68b6SMark de Wever // 7729d6b68b6SMark de Wever // Rules 7739d6b68b6SMark de Wever // R s 1926 1929 - O Sa>=1 24s 0 - (R1) 7749d6b68b6SMark de Wever // 7759d6b68b6SMark de Wever // R Sp 1967 o - Jun 3 12 1 S (R2) 7769d6b68b6SMark de Wever // 7779d6b68b6SMark de Wever // The rule R1 is the last rule used in C1. The rule R2 is the first rule in 7789d6b68b6SMark de Wever // C3. Since R2 is the first rule this means when a continuation uses this 7799d6b68b6SMark de Wever // rule its value prior to R2 will be SAVE 0 LETTERS of the first entry with a 7809d6b68b6SMark de Wever // SAVE of 0, in this case WET. 7819d6b68b6SMark de Wever // This gives the following changes in the information. 7829d6b68b6SMark de Wever // 1928-10-07 00:00:00 C1 R1 becomes active: offset 0 save 0 abbrev WET 7839d6b68b6SMark de Wever // 1929-01-01 00:00:00 C2 becomes active: offset 0 save 0 abbrev WET 7849d6b68b6SMark de Wever // 1967-01-01 00:00:00 C3 becomes active: offset 0 save 0 abbrev WET 7859d6b68b6SMark de Wever // 1967-06-03 12:00:00 C3 R2 becomes active: offset 0 save 1 abbrev WEST 7869d6b68b6SMark de Wever // 7879d6b68b6SMark de Wever // The first 3 entries are contiguous and contain the same information, this 7889d6b68b6SMark de Wever // means the period [1928-10-07 00:00:00, 1967-06-03 12:00:00) should be 7899d6b68b6SMark de Wever // returned in one sys_info object. 7909d6b68b6SMark de Wever 7919d6b68b6SMark de Wever const auto& __continuations = __impl_->__continuations(); 7929d6b68b6SMark de Wever const __tz::__rules_storage_type& __rules_db = __impl_->__rules_db(); 7939d6b68b6SMark de Wever for (auto __it = __continuations.begin(); __it != __continuations.end(); ++__it) { 7949d6b68b6SMark de Wever const auto& __continuation = *__it; 7959d6b68b6SMark de Wever __sys_info_result __sys_info = chrono::__get_sys_info(__time, __continuation_begin, __continuation, __rules_db); 7969d6b68b6SMark de Wever 7979d6b68b6SMark de Wever if (__sys_info) { 7989d6b68b6SMark de Wever _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( 7999d6b68b6SMark de Wever __sys_info->__info.begin < __sys_info->__info.end, "invalid sys_info range"); 8009d6b68b6SMark de Wever 8019d6b68b6SMark de Wever // Filters out dummy entries 8029d6b68b6SMark de Wever // Z America/Argentina/Buenos_Aires -3:53:48 - LMT 1894 O 31 8039d6b68b6SMark de Wever // ... 8049d6b68b6SMark de Wever // -4 A -04/-03 2000 Mar 3 (C1) 8059d6b68b6SMark de Wever // -3 A -03/-02 (C2) 8069d6b68b6SMark de Wever // 8079d6b68b6SMark de Wever // ... 8089d6b68b6SMark de Wever // R A 2000 o - Mar 3 0 0 - 8099d6b68b6SMark de Wever // R A 2007 o - D 30 0 1 - 8109d6b68b6SMark de Wever // ... 8119d6b68b6SMark de Wever // 8129d6b68b6SMark de Wever // This results in an entry 8139d6b68b6SMark de Wever // [2000-03-03 03:00:00, 2000-03-03 04:00:00) -10800s 60min -03 8149d6b68b6SMark de Wever // for [C1 & R1, C1, R2) which due to the end of the continuation is an 8159d6b68b6SMark de Wever // one hour "sys_info". Instead the entry should be ignored and replaced 8169d6b68b6SMark de Wever // by [C2 & R1, C2 & R2) which is the proper range 8179d6b68b6SMark de Wever // "[2000-03-03 03:00:00, 2007-12-30 03:00:00) -02:00:00 60min -02 8189d6b68b6SMark de Wever 8199d6b68b6SMark de Wever if (std::holds_alternative<string>(__continuation.__rules) && __sys_info->__can_merge && 8209d6b68b6SMark de Wever __sys_info->__info.begin + 12h > __sys_info->__info.end) { 8219d6b68b6SMark de Wever __continuation_begin = __sys_info->__info.begin; 8229d6b68b6SMark de Wever continue; 8239d6b68b6SMark de Wever } 8249d6b68b6SMark de Wever 8259d6b68b6SMark de Wever if (!__result) { 8269d6b68b6SMark de Wever // First entry found, always keep it. 8279d6b68b6SMark de Wever __result = __sys_info->__info; 8289d6b68b6SMark de Wever 8299d6b68b6SMark de Wever __valid_result = __time >= __result->begin && __time < __result->end; 8309d6b68b6SMark de Wever __can_merge = __sys_info->__can_merge; 8319d6b68b6SMark de Wever } else if (__can_merge && chrono::__merge_continuation(*__result, __sys_info->__info)) { 8329d6b68b6SMark de Wever // The results are merged, update the result state. This may 8339d6b68b6SMark de Wever // "overwrite" a valid sys_info object with another valid sys_info 8349d6b68b6SMark de Wever // object. 8359d6b68b6SMark de Wever __valid_result = __time >= __result->begin && __time < __result->end; 8369d6b68b6SMark de Wever __can_merge = __sys_info->__can_merge; 8379d6b68b6SMark de Wever } else { 8389d6b68b6SMark de Wever // Here things get interesting: 8399d6b68b6SMark de Wever // For example, America/Argentina/San_Luis 8409d6b68b6SMark de Wever // 8419d6b68b6SMark de Wever // -3 A -03/-02 2008 Ja 21 (C1) 8429d6b68b6SMark de Wever // -4 Sa -04/-03 2009 O 11 (C2) 8439d6b68b6SMark de Wever // 8449d6b68b6SMark de Wever // R A 2007 o - D 30 0 1 - (R1) 8459d6b68b6SMark de Wever // 8469d6b68b6SMark de Wever // R Sa 2007 2008 - O Su>=8 0 1 - (R2) 8479d6b68b6SMark de Wever // 8489d6b68b6SMark de Wever // Based on C1 & R1 the end time of C1 is 2008-01-21 03:00:00 8499d6b68b6SMark de Wever // Based on C2 & R2 the end time of C1 is 2008-01-21 02:00:00 8509d6b68b6SMark de Wever // In this case the earlier time is the real time of the transition. 8519d6b68b6SMark de Wever // However the algorithm used gives 2008-01-21 03:00:00. 8529d6b68b6SMark de Wever // 8539d6b68b6SMark de Wever // So we need to calculate the previous UNTIL in the current context and 8549d6b68b6SMark de Wever // see whether it's earlier. 8559d6b68b6SMark de Wever 8569d6b68b6SMark de Wever // The results could not be merged. 8579d6b68b6SMark de Wever // - When we have a valid result that result is the final result. 8589d6b68b6SMark de Wever // - Otherwise the result we had is before __time and the result we got 8599d6b68b6SMark de Wever // is at a later time (possibly valid). This result is always better 8609d6b68b6SMark de Wever // than the previous result. 8619d6b68b6SMark de Wever if (__valid_result) { 8629d6b68b6SMark de Wever return *__result; 8639d6b68b6SMark de Wever } else { 8649d6b68b6SMark de Wever _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( 8659d6b68b6SMark de Wever __it != __continuations.begin(), "the first rule should always seed the result"); 8669d6b68b6SMark de Wever const auto& __last = *(__it - 1); 8679d6b68b6SMark de Wever if (std::holds_alternative<string>(__last.__rules)) { 8689d6b68b6SMark de Wever // Europe/Berlin 8699d6b68b6SMark de Wever // 1 c CE%sT 1945 May 24 2 (C1) 8709d6b68b6SMark de Wever // 1 So CE%sT 1946 (C2) 8719d6b68b6SMark de Wever // 8729d6b68b6SMark de Wever // R c 1944 1945 - Ap M>=1 2s 1 S (R1) 8739d6b68b6SMark de Wever // 8749d6b68b6SMark de Wever // R So 1945 o - May 24 2 2 M (R2) 8759d6b68b6SMark de Wever // 8769d6b68b6SMark de Wever // When C2 becomes active the time would be before the first rule R2, 8779d6b68b6SMark de Wever // giving a 1 hour sys_info. This is not valid and the results need 8789d6b68b6SMark de Wever // merging. 8799d6b68b6SMark de Wever 8809d6b68b6SMark de Wever if (__result->end != __sys_info->__info.begin) { 8819d6b68b6SMark de Wever // When the UTC gap between the rules is due to the change of 8829d6b68b6SMark de Wever // offsets adjust the new time to remove the gap. 8839d6b68b6SMark de Wever sys_seconds __end = __result->end - __result->offset; 8849d6b68b6SMark de Wever sys_seconds __begin = __sys_info->__info.begin - __sys_info->__info.offset; 8859d6b68b6SMark de Wever if (__end == __begin) { 8869d6b68b6SMark de Wever __sys_info->__info.begin = __result->end; 8879d6b68b6SMark de Wever } 8889d6b68b6SMark de Wever } 8899d6b68b6SMark de Wever } 8909d6b68b6SMark de Wever 8919d6b68b6SMark de Wever __result = __sys_info->__info; 8929d6b68b6SMark de Wever __valid_result = __time >= __result->begin && __time < __result->end; 8939d6b68b6SMark de Wever __can_merge = __sys_info->__can_merge; 8949d6b68b6SMark de Wever } 8959d6b68b6SMark de Wever } 8969d6b68b6SMark de Wever __continuation_begin = __result->end; 8979d6b68b6SMark de Wever } else { 8989d6b68b6SMark de Wever __continuation_begin = __sys_info.error(); 8999d6b68b6SMark de Wever } 9009d6b68b6SMark de Wever } 9019d6b68b6SMark de Wever if (__valid_result) 9029d6b68b6SMark de Wever return *__result; 9039d6b68b6SMark de Wever 9049d6b68b6SMark de Wever std::__throw_runtime_error("tzdb: corrupt db"); 9059d6b68b6SMark de Wever } 9069d6b68b6SMark de Wever 9079d6b68b6SMark de Wever // Is the "__local_time" present in "__first" and "__second". If so the 9089d6b68b6SMark de Wever // local_info has an ambiguous result. 9099d6b68b6SMark de Wever [[nodiscard]] static bool 9109d6b68b6SMark de Wever __is_ambiguous(local_seconds __local_time, const sys_info& __first, const sys_info& __second) { 9119d6b68b6SMark de Wever std::chrono::local_seconds __end_first{__first.end.time_since_epoch() + __first.offset}; 9129d6b68b6SMark de Wever std::chrono::local_seconds __begin_second{__second.begin.time_since_epoch() + __second.offset}; 9139d6b68b6SMark de Wever 9149d6b68b6SMark de Wever return __local_time < __end_first && __local_time >= __begin_second; 9159d6b68b6SMark de Wever } 9169d6b68b6SMark de Wever 9179d6b68b6SMark de Wever // Determines the result of the "__local_time". This expects the object 9189d6b68b6SMark de Wever // "__first" to be earlier in time than "__second". 9199d6b68b6SMark de Wever [[nodiscard]] static local_info 9209d6b68b6SMark de Wever __get_info(local_seconds __local_time, const sys_info& __first, const sys_info& __second) { 9219d6b68b6SMark de Wever std::chrono::local_seconds __end_first{__first.end.time_since_epoch() + __first.offset}; 9229d6b68b6SMark de Wever std::chrono::local_seconds __begin_second{__second.begin.time_since_epoch() + __second.offset}; 9239d6b68b6SMark de Wever 9249d6b68b6SMark de Wever if (__local_time < __end_first) { 9259d6b68b6SMark de Wever if (__local_time >= __begin_second) 9269d6b68b6SMark de Wever // |--------| 9279d6b68b6SMark de Wever // |------| 9289d6b68b6SMark de Wever // ^ 9299d6b68b6SMark de Wever return {local_info::ambiguous, __first, __second}; 9309d6b68b6SMark de Wever 9319d6b68b6SMark de Wever // |--------| 9329d6b68b6SMark de Wever // |------| 9339d6b68b6SMark de Wever // ^ 9349d6b68b6SMark de Wever return {local_info::unique, __first, sys_info{}}; 9359d6b68b6SMark de Wever } 9369d6b68b6SMark de Wever 9379d6b68b6SMark de Wever if (__local_time < __begin_second) 9389d6b68b6SMark de Wever // |--------| 9399d6b68b6SMark de Wever // |------| 9409d6b68b6SMark de Wever // ^ 9419d6b68b6SMark de Wever return {local_info::nonexistent, __first, __second}; 9429d6b68b6SMark de Wever 9439d6b68b6SMark de Wever // |--------| 9449d6b68b6SMark de Wever // |------| 9459d6b68b6SMark de Wever // ^ 9469d6b68b6SMark de Wever return {local_info::unique, __second, sys_info{}}; 9479d6b68b6SMark de Wever } 9489d6b68b6SMark de Wever 9499d6b68b6SMark de Wever [[nodiscard]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_EXPORTED_FROM_ABI local_info 9509d6b68b6SMark de Wever time_zone::__get_info(local_seconds __local_time) const { 9519d6b68b6SMark de Wever seconds __local_seconds = __local_time.time_since_epoch(); 9529d6b68b6SMark de Wever 9539d6b68b6SMark de Wever /* An example of a typical year with a DST switch displayed in local time. 9549d6b68b6SMark de Wever * 9559d6b68b6SMark de Wever * At the first of April the time goes forward one hour. This means the 9569d6b68b6SMark de Wever * time marked with ~~ is not a valid local time. This is represented by the 9579d6b68b6SMark de Wever * nonexistent value in local_info.result. 9589d6b68b6SMark de Wever * 9599d6b68b6SMark de Wever * At the first of November the time goes backward one hour. This means the 9609d6b68b6SMark de Wever * time marked with ^^ happens twice. This is represented by the ambiguous 9619d6b68b6SMark de Wever * value in local_info.result. 9629d6b68b6SMark de Wever * 9639d6b68b6SMark de Wever * 2020.11.01 2021.04.01 2021.11.01 9649d6b68b6SMark de Wever * offset +05 offset +05 offset +05 9659d6b68b6SMark de Wever * save 0s save 1h save 0s 9669d6b68b6SMark de Wever * |------------//----------| 9679d6b68b6SMark de Wever * |---------//--------------| 9689d6b68b6SMark de Wever * |------------- 9699d6b68b6SMark de Wever * ~~ ^^ 9709d6b68b6SMark de Wever * 9719d6b68b6SMark de Wever * These shifts can happen due to changes in the current time zone for a 9729d6b68b6SMark de Wever * location. For example, Indian/Kerguelen switched only once. In 1950 from an 9739d6b68b6SMark de Wever * offset of 0 hours to an offset of +05 hours. 9749d6b68b6SMark de Wever * 9759d6b68b6SMark de Wever * During all these shifts the UTC time will not have gaps. 9769d6b68b6SMark de Wever */ 9779d6b68b6SMark de Wever 9789d6b68b6SMark de Wever // The code needs to determine the system time for the local time. There is no 9799d6b68b6SMark de Wever // information available. Assume the offset between system time and local time 9809d6b68b6SMark de Wever // is 0s. This gives an initial estimate. 9819d6b68b6SMark de Wever sys_seconds __guess{__local_seconds}; 9829d6b68b6SMark de Wever sys_info __info = __get_info(__guess); 9839d6b68b6SMark de Wever 9849d6b68b6SMark de Wever // At this point the offset can be used to determine an estimate for the local 9859d6b68b6SMark de Wever // time. Before doing that, determine the offset and validate whether the 9869d6b68b6SMark de Wever // local time is the range [chrono::local_seconds::min(), 9879d6b68b6SMark de Wever // chrono::local_seconds::max()). 9889d6b68b6SMark de Wever if (__local_seconds < 0s && __info.offset > 0s) 9899d6b68b6SMark de Wever if (__local_seconds - chrono::local_seconds::min().time_since_epoch() < __info.offset) 9909d6b68b6SMark de Wever return {-1, __info, {}}; 9919d6b68b6SMark de Wever 9929d6b68b6SMark de Wever if (__local_seconds > 0s && __info.offset < 0s) 9939d6b68b6SMark de Wever if (chrono::local_seconds::max().time_since_epoch() - __local_seconds < -__info.offset) 9949d6b68b6SMark de Wever return {-2, __info, {}}; 9959d6b68b6SMark de Wever 9969d6b68b6SMark de Wever // Based on the information found in the sys_info, the local time can be 9979d6b68b6SMark de Wever // converted to a system time. This resulting time can be in the following 9989d6b68b6SMark de Wever // locations of the sys_info: 9999d6b68b6SMark de Wever // 10009d6b68b6SMark de Wever // |---------//--------------| 10019d6b68b6SMark de Wever // 1 2.1 2.2 2.3 3 10029d6b68b6SMark de Wever // 10039d6b68b6SMark de Wever // 1. The estimate is before the returned sys_info object. 10049d6b68b6SMark de Wever // The result is either non-existent or unique in the previous sys_info. 10059d6b68b6SMark de Wever // 2. The estimate is in the sys_info object 10069d6b68b6SMark de Wever // - If the sys_info begin is not sys_seconds::min(), then it might be at 10079d6b68b6SMark de Wever // 2.1 and could be ambiguous with the previous or unique. 10089d6b68b6SMark de Wever // - If sys_info end is not sys_seconds::max(), then it might be at 2.3 10099d6b68b6SMark de Wever // and could be ambiguous with the next or unique. 10109d6b68b6SMark de Wever // - Else it is at 2.2 and always unique. This case happens when a 10119d6b68b6SMark de Wever // time zone has no transitions. For example, UTC or GMT+1. 10129d6b68b6SMark de Wever // 3. The estimate is after the returned sys_info object. 10139d6b68b6SMark de Wever // The result is either non-existent or unique in the next sys_info. 10149d6b68b6SMark de Wever // 10159d6b68b6SMark de Wever // There is no specification where the "middle" starts. Similar issues can 10169d6b68b6SMark de Wever // happen when sys_info objects are "short", then "unique in the next" could 10179d6b68b6SMark de Wever // become "ambiguous in the next and the one following". Theoretically there 10189d6b68b6SMark de Wever // is the option of the following time-line 10199d6b68b6SMark de Wever // 10209d6b68b6SMark de Wever // |------------| 10219d6b68b6SMark de Wever // |----| 10229d6b68b6SMark de Wever // |-----------------| 10239d6b68b6SMark de Wever // 10249d6b68b6SMark de Wever // However the local_info object only has 2 sys_info objects, so this option 10259d6b68b6SMark de Wever // is not tested. 10269d6b68b6SMark de Wever 10279d6b68b6SMark de Wever sys_seconds __sys_time{__local_seconds - __info.offset}; 10289d6b68b6SMark de Wever if (__sys_time < __info.begin) 10299d6b68b6SMark de Wever // Case 1 before __info 10309d6b68b6SMark de Wever return chrono::__get_info(__local_time, __get_info(__info.begin - 1s), __info); 10319d6b68b6SMark de Wever 10329d6b68b6SMark de Wever if (__sys_time >= __info.end) 10339d6b68b6SMark de Wever // Case 3 after __info 10349d6b68b6SMark de Wever return chrono::__get_info(__local_time, __info, __get_info(__info.end)); 10359d6b68b6SMark de Wever 10369d6b68b6SMark de Wever // Case 2 in __info 10379d6b68b6SMark de Wever if (__info.begin != sys_seconds::min()) { 10389d6b68b6SMark de Wever // Case 2.1 Not at the beginning, when not ambiguous the result should test 10399d6b68b6SMark de Wever // case 2.3. 10409d6b68b6SMark de Wever sys_info __prev = __get_info(__info.begin - 1s); 10419d6b68b6SMark de Wever if (__is_ambiguous(__local_time, __prev, __info)) 10429d6b68b6SMark de Wever return {local_info::ambiguous, __prev, __info}; 10439d6b68b6SMark de Wever } 10449d6b68b6SMark de Wever 10459d6b68b6SMark de Wever if (__info.end == sys_seconds::max()) 10469d6b68b6SMark de Wever // At the end so it's case 2.2 10479d6b68b6SMark de Wever return {local_info::unique, __info, sys_info{}}; 10489d6b68b6SMark de Wever 10499d6b68b6SMark de Wever // This tests case 2.2 or case 2.3. 10509d6b68b6SMark de Wever return chrono::__get_info(__local_time, __info, __get_info(__info.end)); 10519d6b68b6SMark de Wever } 10529d6b68b6SMark de Wever 10539d6b68b6SMark de Wever } // namespace chrono 10549d6b68b6SMark de Wever 10559d6b68b6SMark de Wever _LIBCPP_END_NAMESPACE_STD 1056