1 | // -*- C++ -*- |
2 | //===----------------------------------------------------------------------===// |
3 | // |
4 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
5 | // See https://llvm.org/LICENSE.txt for license information. |
6 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
7 | // |
8 | //===----------------------------------------------------------------------===// |
9 | |
10 | // For information see https://libcxx.llvm.org/DesignDocs/TimeZone.html |
11 | |
12 | #ifndef _LIBCPP___CHRONO_TIME_ZONE_H |
13 | #define _LIBCPP___CHRONO_TIME_ZONE_H |
14 | |
15 | #include <version> |
16 | // Enable the contents of the header only when libc++ was built with experimental features enabled. |
17 | #if _LIBCPP_HAS_EXPERIMENTAL_TZDB |
18 | |
19 | # include <__chrono/calendar.h> |
20 | # include <__chrono/duration.h> |
21 | # include <__chrono/exception.h> |
22 | # include <__chrono/local_info.h> |
23 | # include <__chrono/sys_info.h> |
24 | # include <__chrono/system_clock.h> |
25 | # include <__compare/strong_order.h> |
26 | # include <__config> |
27 | # include <__memory/unique_ptr.h> |
28 | # include <__type_traits/common_type.h> |
29 | # include <string_view> |
30 | |
31 | # if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) |
32 | # pragma GCC system_header |
33 | # endif |
34 | |
35 | _LIBCPP_PUSH_MACROS |
36 | # include <__undef_macros> |
37 | |
38 | _LIBCPP_BEGIN_NAMESPACE_STD |
39 | |
40 | # if _LIBCPP_STD_VER >= 20 && _LIBCPP_HAS_TIME_ZONE_DATABASE && _LIBCPP_HAS_FILESYSTEM && _LIBCPP_HAS_LOCALIZATION |
41 | |
42 | namespace chrono { |
43 | |
44 | enum class choose { earliest, latest }; |
45 | |
46 | class _LIBCPP_AVAILABILITY_TZDB time_zone { |
47 | _LIBCPP_HIDE_FROM_ABI time_zone() = default; |
48 | |
49 | public: |
50 | class __impl; // public so it can be used by make_unique. |
51 | |
52 | // The "constructor". |
53 | // |
54 | // The default constructor is private to avoid the constructor from being |
55 | // part of the ABI. Instead use an __ugly_named function as an ABI interface, |
56 | // since that gives us the ability to change it in the future. |
57 | [[nodiscard]] _LIBCPP_EXPORTED_FROM_ABI static time_zone __create(unique_ptr<__impl>&& __p); |
58 | |
59 | _LIBCPP_EXPORTED_FROM_ABI ~time_zone(); |
60 | |
61 | _LIBCPP_HIDE_FROM_ABI time_zone(time_zone&&) = default; |
62 | _LIBCPP_HIDE_FROM_ABI time_zone& operator=(time_zone&&) = default; |
63 | |
64 | [[nodiscard]] _LIBCPP_HIDE_FROM_ABI string_view name() const noexcept { return __name(); } |
65 | |
66 | template <class _Duration> |
67 | [[nodiscard]] _LIBCPP_HIDE_FROM_ABI sys_info get_info(const sys_time<_Duration>& __time) const { |
68 | return __get_info(chrono::time_point_cast<seconds>(__time)); |
69 | } |
70 | |
71 | template <class _Duration> |
72 | [[nodiscard]] _LIBCPP_HIDE_FROM_ABI local_info get_info(const local_time<_Duration>& __time) const { |
73 | return __get_info(chrono::time_point_cast<seconds>(__time)); |
74 | } |
75 | |
76 | // We don't apply nodiscard here since this function throws on many inputs, |
77 | // so it could be used as a validation. |
78 | template <class _Duration> |
79 | _LIBCPP_HIDE_FROM_ABI sys_time<common_type_t<_Duration, seconds>> to_sys(const local_time<_Duration>& __time) const { |
80 | local_info __info = get_info(__time); |
81 | switch (__info.result) { |
82 | case local_info::unique: |
83 | return sys_time<common_type_t<_Duration, seconds>>{__time.time_since_epoch() - __info.first.offset}; |
84 | |
85 | case local_info::nonexistent: |
86 | chrono::__throw_nonexistent_local_time(__time, __info); |
87 | |
88 | case local_info::ambiguous: |
89 | chrono::__throw_ambiguous_local_time(__time, __info); |
90 | } |
91 | |
92 | // TODO TZDB The Standard does not specify anything in these cases. |
93 | _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( |
94 | __info.result != -1, "cannot convert the local time; it would be before the minimum system clock value" ); |
95 | _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( |
96 | __info.result != -2, "cannot convert the local time; it would be after the maximum system clock value" ); |
97 | |
98 | return {}; |
99 | } |
100 | |
101 | template <class _Duration> |
102 | [[nodiscard]] _LIBCPP_HIDE_FROM_ABI sys_time<common_type_t<_Duration, seconds>> |
103 | to_sys(const local_time<_Duration>& __time, choose __z) const { |
104 | local_info __info = get_info(__time); |
105 | switch (__info.result) { |
106 | case local_info::unique: // first and second are the same |
107 | return sys_time<common_type_t<_Duration, seconds>>{__time.time_since_epoch() - __info.first.offset}; |
108 | |
109 | case local_info::nonexistent: |
110 | // first and second are the same |
111 | // All non-existing values are converted to the same time. |
112 | return sys_time<common_type_t<_Duration, seconds>>{__info.first.end}; |
113 | |
114 | case local_info::ambiguous: |
115 | switch (__z) { |
116 | case choose::earliest: |
117 | return sys_time<common_type_t<_Duration, seconds>>{__time.time_since_epoch() - __info.first.offset}; |
118 | |
119 | case choose::latest: |
120 | return sys_time<common_type_t<_Duration, seconds>>{__time.time_since_epoch() - __info.second.offset}; |
121 | |
122 | // Note a value out of bounds is not specified. |
123 | } |
124 | } |
125 | |
126 | // TODO TZDB The standard does not specify anything in these cases. |
127 | _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( |
128 | __info.result != -1, "cannot convert the local time; it would be before the minimum system clock value" ); |
129 | _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( |
130 | __info.result != -2, "cannot convert the local time; it would be after the maximum system clock value" ); |
131 | |
132 | return {}; |
133 | } |
134 | |
135 | template <class _Duration> |
136 | [[nodiscard]] _LIBCPP_HIDE_FROM_ABI local_time<common_type_t<_Duration, seconds>> |
137 | to_local(const sys_time<_Duration>& __time) const { |
138 | using _Dp = common_type_t<_Duration, seconds>; |
139 | |
140 | sys_info __info = get_info(__time); |
141 | |
142 | _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( |
143 | __info.offset >= chrono::seconds{0} || __time.time_since_epoch() >= _Dp::min() - __info.offset, |
144 | "cannot convert the system time; it would be before the minimum local clock value" ); |
145 | |
146 | _LIBCPP_ASSERT_ARGUMENT_WITHIN_DOMAIN( |
147 | __info.offset <= chrono::seconds{0} || __time.time_since_epoch() <= _Dp::max() - __info.offset, |
148 | "cannot convert the system time; it would be after the maximum local clock value" ); |
149 | |
150 | return local_time<_Dp>{__time.time_since_epoch() + __info.offset}; |
151 | } |
152 | |
153 | [[nodiscard]] _LIBCPP_HIDE_FROM_ABI const __impl& __implementation() const noexcept { return *__impl_; } |
154 | |
155 | private: |
156 | [[nodiscard]] _LIBCPP_EXPORTED_FROM_ABI string_view __name() const noexcept; |
157 | |
158 | [[nodiscard]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_EXPORTED_FROM_ABI sys_info __get_info(sys_seconds __time) const; |
159 | [[nodiscard]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_EXPORTED_FROM_ABI local_info __get_info(local_seconds __time) const; |
160 | |
161 | unique_ptr<__impl> __impl_; |
162 | }; |
163 | |
164 | [[nodiscard]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_HIDE_FROM_ABI inline bool |
165 | operator==(const time_zone& __x, const time_zone& __y) noexcept { |
166 | return __x.name() == __y.name(); |
167 | } |
168 | |
169 | [[nodiscard]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_HIDE_FROM_ABI inline strong_ordering |
170 | operator<=>(const time_zone& __x, const time_zone& __y) noexcept { |
171 | return __x.name() <=> __y.name(); |
172 | } |
173 | |
174 | } // namespace chrono |
175 | |
176 | # endif // _LIBCPP_STD_VER >= 20 && _LIBCPP_HAS_TIME_ZONE_DATABASE && _LIBCPP_HAS_FILESYSTEM && |
177 | // _LIBCPP_HAS_LOCALIZATION |
178 | |
179 | _LIBCPP_END_NAMESPACE_STD |
180 | |
181 | _LIBCPP_POP_MACROS |
182 | |
183 | #endif // _LIBCPP_HAS_EXPERIMENTAL_TZDB |
184 | |
185 | #endif // _LIBCPP___CHRONO_TIME_ZONE_H |
186 | |