Dynalib Utils
tz_private.h
Go to the documentation of this file.
1 #ifndef TZ_PRIVATE_H
2 #define TZ_PRIVATE_H
3 
4 // The MIT License (MIT)
5 //
6 // Copyright (c) 2015, 2016 Howard Hinnant
7 //
8 // Permission is hereby granted, free of charge, to any person obtaining a copy
9 // of this software and associated documentation files (the "Software"), to deal
10 // in the Software without restriction, including without limitation the rights
11 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 // copies of the Software, and to permit persons to whom the Software is
13 // furnished to do so, subject to the following conditions:
14 //
15 // The above copyright notice and this permission notice shall be included in all
16 // copies or substantial portions of the Software.
17 //
18 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
24 // SOFTWARE.
25 //
26 // Our apologies. When the previous paragraph was written, lowercase had not yet
27 // been invented (that would involve another several millennia of evolution).
28 // We did not mean to shout.
29 
30 #if !defined(_MSC_VER) || (_MSC_VER >= 1900)
31 #include "tz.h"
32 #else
33 #include "date.h"
34 #include <vector>
35 #endif
36 
37 namespace date
38 {
39 
40 namespace detail
41 {
42 
43 #if !USE_OS_TZDB
44 
45 enum class tz {utc, local, standard};
46 
47 //forward declare to avoid warnings in gcc 6.2
48 class MonthDayTime;
49 std::istream& operator>>(std::istream& is, MonthDayTime& x);
50 std::ostream& operator<<(std::ostream& os, const MonthDayTime& x);
51 
52 
54 {
55 private:
56  struct pair
57  {
58 #if defined(_MSC_VER) && (_MSC_VER < 1900)
59  pair() : month_day_(date::jan / 1), weekday_(0U) {}
60 
61  pair(const date::month_day& month_day, const date::weekday& weekday)
62  : month_day_(month_day), weekday_(weekday) {}
63 #endif
64 
65  date::month_day month_day_;
66  date::weekday weekday_;
67  };
68 
69  enum Type {month_day, month_last_dow, lteq, gteq};
70 
71  Type type_{month_day};
72 
73 #if !defined(_MSC_VER) || (_MSC_VER >= 1900)
74  union U
75 #else
76  struct U
77 #endif
78  {
79  date::month_day month_day_;
80  date::month_weekday_last month_weekday_last_;
81  pair month_day_weekday_;
82 
83 #if !defined(_MSC_VER) || (_MSC_VER >= 1900)
84  U() : month_day_{date::jan/1} {}
85 #else
86  U() :
87  month_day_(date::jan/1),
88  month_weekday_last_(date::month(0U), date::weekday_last(date::weekday(0U)))
89  {}
90 
91 #endif // !defined(_MSC_VER) || (_MSC_VER >= 1900)
92 
93  U& operator=(const date::month_day& x);
94  U& operator=(const date::month_weekday_last& x);
95  U& operator=(const pair& x);
96  } u;
97 
98  std::chrono::hours h_{0};
99  std::chrono::minutes m_{0};
100  std::chrono::seconds s_{0};
101  tz zone_{tz::local};
102 
103 public:
104  MonthDayTime() = default;
105  MonthDayTime(local_seconds tp, tz timezone);
106  MonthDayTime(const date::month_day& md, tz timezone);
107 
108  date::day day() const;
109  date::month month() const;
110  tz zone() const {return zone_;}
111 
112  void canonicalize(date::year y);
113 
115  to_sys(date::year y, std::chrono::seconds offset, std::chrono::seconds save) const;
116  sys_days to_sys_days(date::year y) const;
117 
118  sys_seconds to_time_point(date::year y) const;
119  int compare(date::year y, const MonthDayTime& x, date::year yx,
120  std::chrono::seconds offset, std::chrono::minutes prev_save) const;
121 
122  friend std::istream& operator>>(std::istream& is, MonthDayTime& x);
123  friend std::ostream& operator<<(std::ostream& os, const MonthDayTime& x);
124 };
125 
126 // A Rule specifies one or more set of datetimes without using an offset.
127 // Multiple dates are specified with multiple years. The years in effect
128 // go from starting_year_ to ending_year_, inclusive. starting_year_ <=
129 // ending_year_. save_ is in effect for times from the specified time
130 // onward, including the specified time. When the specified time is
131 // local, it uses the save_ from the chronologically previous Rule, or if
132 // there is none, 0.
133 
134 //forward declare to avoid warnings in gcc 6.2
135 class Rule;
136 bool operator==(const Rule& x, const Rule& y);
137 bool operator<(const Rule& x, const Rule& y);
138 bool operator==(const Rule& x, const date::year& y);
139 bool operator<(const Rule& x, const date::year& y);
140 bool operator==(const date::year& x, const Rule& y);
141 bool operator<(const date::year& x, const Rule& y);
142 bool operator==(const Rule& x, const std::string& y);
143 bool operator<(const Rule& x, const std::string& y);
144 bool operator==(const std::string& x, const Rule& y);
145 bool operator<(const std::string& x, const Rule& y);
146 std::ostream& operator<<(std::ostream& os, const Rule& r);
147 
148 class Rule
149 {
150 private:
151  std::string name_;
152  date::year starting_year_{0};
153  date::year ending_year_{0};
154  MonthDayTime starting_at_;
155  std::chrono::minutes save_{0};
156  std::string abbrev_;
157 
158 public:
159  Rule() = default;
160  explicit Rule(const std::string& s);
161  Rule(const Rule& r, date::year starting_year, date::year ending_year);
162 
163  const std::string& name() const {return name_;}
164  const std::string& abbrev() const {return abbrev_;}
165 
166  const MonthDayTime& mdt() const {return starting_at_;}
167  const date::year& starting_year() const {return starting_year_;}
168  const date::year& ending_year() const {return ending_year_;}
169  const std::chrono::minutes& save() const {return save_;}
170 
171  static void split_overlaps(std::vector<Rule>& rules);
172 
173  friend bool operator==(const Rule& x, const Rule& y);
174  friend bool operator<(const Rule& x, const Rule& y);
175  friend bool operator==(const Rule& x, const date::year& y);
176  friend bool operator<(const Rule& x, const date::year& y);
177  friend bool operator==(const date::year& x, const Rule& y);
178  friend bool operator<(const date::year& x, const Rule& y);
179  friend bool operator==(const Rule& x, const std::string& y);
180  friend bool operator<(const Rule& x, const std::string& y);
181  friend bool operator==(const std::string& x, const Rule& y);
182  friend bool operator<(const std::string& x, const Rule& y);
183 
184  friend std::ostream& operator<<(std::ostream& os, const Rule& r);
185 
186 private:
187  date::day day() const;
188  date::month month() const;
189  static void split_overlaps(std::vector<Rule>& rules, std::size_t i, std::size_t& e);
190  static bool overlaps(const Rule& x, const Rule& y);
191  static void split(std::vector<Rule>& rules, std::size_t i, std::size_t k,
192  std::size_t& e);
193 };
194 
195 inline bool operator!=(const Rule& x, const Rule& y) {return !(x == y);}
196 inline bool operator> (const Rule& x, const Rule& y) {return y < x;}
197 inline bool operator<=(const Rule& x, const Rule& y) {return !(y < x);}
198 inline bool operator>=(const Rule& x, const Rule& y) {return !(x < y);}
199 
200 inline bool operator!=(const Rule& x, const date::year& y) {return !(x == y);}
201 inline bool operator> (const Rule& x, const date::year& y) {return y < x;}
202 inline bool operator<=(const Rule& x, const date::year& y) {return !(y < x);}
203 inline bool operator>=(const Rule& x, const date::year& y) {return !(x < y);}
204 
205 inline bool operator!=(const date::year& x, const Rule& y) {return !(x == y);}
206 inline bool operator> (const date::year& x, const Rule& y) {return y < x;}
207 inline bool operator<=(const date::year& x, const Rule& y) {return !(y < x);}
208 inline bool operator>=(const date::year& x, const Rule& y) {return !(x < y);}
209 
210 inline bool operator!=(const Rule& x, const std::string& y) {return !(x == y);}
211 inline bool operator> (const Rule& x, const std::string& y) {return y < x;}
212 inline bool operator<=(const Rule& x, const std::string& y) {return !(y < x);}
213 inline bool operator>=(const Rule& x, const std::string& y) {return !(x < y);}
214 
215 inline bool operator!=(const std::string& x, const Rule& y) {return !(x == y);}
216 inline bool operator> (const std::string& x, const Rule& y) {return y < x;}
217 inline bool operator<=(const std::string& x, const Rule& y) {return !(y < x);}
218 inline bool operator>=(const std::string& x, const Rule& y) {return !(x < y);}
219 
220 struct zonelet
221 {
222  enum tag {has_rule, has_save, is_empty};
223 
224  std::chrono::seconds gmtoff_;
225  tag tag_ = has_rule;
226 
227 #if !defined(_MSC_VER) || (_MSC_VER >= 1900)
228  union U
229 #else
230  struct U
231 #endif
232  {
233  std::string rule_;
234  std::chrono::minutes save_;
235 
236  ~U() {}
237  U() {}
238  U(const U&) {}
239  U& operator=(const U&) = delete;
240  } u;
241 
242  std::string format_;
243  date::year until_year_{0};
248  std::chrono::minutes initial_save_{};
249  std::string initial_abbrev_;
250  std::pair<const Rule*, date::year> first_rule_{nullptr, date::year::min()};
251  std::pair<const Rule*, date::year> last_rule_{nullptr, date::year::max()};
252 
253  ~zonelet();
254  zonelet();
255  zonelet(const zonelet& i);
256  zonelet& operator=(const zonelet&) = delete;
257 };
258 
259 #else // USE_OS_TZDB
260 
261 struct ttinfo
262 {
263  std::int32_t tt_gmtoff;
264  unsigned char tt_isdst;
265  unsigned char tt_abbrind;
266  unsigned char pad[2];
267 };
268 
269 static_assert(sizeof(ttinfo) == 8, "");
270 
271 struct expanded_ttinfo
272 {
273  std::chrono::seconds offset;
274  std::string abbrev;
275  bool is_dst;
276 };
277 
278 struct transition
279 {
280  sys_seconds timepoint;
281  const expanded_ttinfo* info;
282 
283  transition(sys_seconds tp, const expanded_ttinfo* i = nullptr)
284  : timepoint(tp)
285  , info(i)
286  {}
287 
288  friend
289  std::ostream&
290  operator<<(std::ostream& os, const transition& t)
291  {
292  using namespace date;
293  using namespace std::chrono;
294  using date::operator<<;
295  os << t.timepoint << "Z ";
296  if (t.info->offset >= seconds{0})
297  os << '+';
298  os << make_time(t.info->offset);
299  if (t.info->is_dst > 0)
300  os << " daylight ";
301  else
302  os << " standard ";
303  os << t.info->abbrev;
304  return os;
305  }
306 };
307 
308 #endif // USE_OS_TZDB
309 
310 } // namespace detail
311 
312 } // namespace date
313 
314 #if defined(_MSC_VER) && (_MSC_VER < 1900)
315 #include "tz.h"
316 #endif
317 
318 #endif // TZ_PRIVATE_H
MonthDayTime until_date_
Definition: tz_private.h:244
Definition: date.h:475
const MonthDayTime & mdt() const
Definition: tz_private.h:166
std::string format_
Definition: tz_private.h:242
bool operator<=(const Rule &x, const Rule &y)
Definition: tz_private.h:197
std::chrono::minutes save_
Definition: tz_private.h:234
sys_time< std::chrono::seconds > sys_seconds
Definition: date.h:164
U(const U &)
Definition: tz_private.h:238
Definition: tz_private.h:228
tag
Definition: tz_private.h:222
sys_time< days > sys_days
Definition: date.h:163
Definition: date.h:79
sys_seconds until_utc_
Definition: tz_private.h:245
Definition: date.h:365
~U()
Definition: tz_private.h:236
bool operator>(const Rule &x, const Rule &y)
Definition: tz_private.h:196
bool operator>=(const Rule &x, const Rule &y)
Definition: tz_private.h:198
local_seconds until_std_
Definition: tz_private.h:246
Definition: date.h:612
std::string initial_abbrev_
Definition: tz_private.h:249
static CONSTCD11 year min() NOEXCEPT
Definition: date.h:1560
std::string rule_
Definition: tz_private.h:233
tz zone() const
Definition: tz_private.h:110
tz
Definition: tz_private.h:45
CONSTDATA date::month jan
Definition: date.h:1836
CONSTCD11 time_of_day< std::chrono::duration< Rep, Period > > make_time(const std::chrono::duration< Rep, Period > &d)
Definition: date.h:4264
bool operator<(const Rule &x, const Rule &y)
Definition: tz.cpp:1164
bool operator==(const Rule &x, const Rule &y)
Definition: tz.cpp:1155
const std::string & abbrev() const
Definition: tz_private.h:164
const date::year & ending_year() const
Definition: tz_private.h:168
Definition: tz_private.h:220
local_time< std::chrono::seconds > local_seconds
Definition: date.h:171
Definition: date.h:537
local_seconds until_loc_
Definition: tz_private.h:247
std::istream & operator>>(std::istream &is, MonthDayTime &x)
Definition: tz.cpp:946
U()
Definition: tz_private.h:237
static CONSTCD11 year max() NOEXCEPT
Definition: date.h:1568
const date::year & starting_year() const
Definition: tz_private.h:167
const std::string & name() const
Definition: tz_private.h:163
const std::chrono::minutes & save() const
Definition: tz_private.h:169
std::ostream & operator<<(std::ostream &os, const MonthDayTime &x)
Definition: tz.cpp:1040
Definition: date.h:289
Definition: date.h:411
Definition: date.h:327
std::chrono::seconds gmtoff_
Definition: tz_private.h:224
Definition: tz_private.h:148
Definition: tz_private.h:53
bool operator!=(const Rule &x, const Rule &y)
Definition: tz_private.h:195