Toolbox snapshot
The Reactive C++ Toolbox
Loading...
Searching...
No Matches
Time.hpp
Go to the documentation of this file.
1// The Reactive C++ Toolbox.
2// Copyright (C) 2013-2019 Swirly Cloud Limited
3// Copyright (C) 2022 Reactive Markets Limited
4//
5// Licensed under the Apache License, Version 2.0 (the "License");
6// you may not use this file except in compliance with the License.
7// You may obtain a copy of the License at
8//
9// http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing, software
12// distributed under the License is distributed on an "AS IS" BASIS,
13// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14// See the License for the specific language governing permissions and
15// limitations under the License.
16
17#ifndef TOOLBOX_SYS_TIME_HPP
18#define TOOLBOX_SYS_TIME_HPP
19
24
25#include <chrono>
26#include <format>
27#include <optional>
28
29namespace toolbox {
30inline namespace sys {
31using namespace std::literals::chrono_literals;
32using namespace std::literals::string_view_literals;
33
34using Seconds = std::chrono::seconds;
35using Decis = std::chrono::duration<int64_t, std::deci>;
36using Millis = std::chrono::milliseconds;
37using Micros = std::chrono::microseconds;
38using Nanos = std::chrono::nanoseconds;
39
42
44
45struct MonoClock {
47 using period = Duration::period;
48 using rep = Duration::rep;
49 using time_point = std::chrono::time_point<MonoClock, Duration>;
50
51 static constexpr int Id{CLOCK_MONOTONIC};
52 static constexpr bool is_steady{true};
53
54 static constexpr time_point max() noexcept
55 {
56 using namespace std::chrono;
57 using FromPoint = std::chrono::time_point<MonoClock, seconds>;
58 constexpr seconds secs{std::numeric_limits<int>::max()};
60 }
61
63
64 static constexpr std::time_t to_time_t(const time_point& tp) noexcept
65 {
66 using namespace std::chrono;
67 return duration_cast<seconds>(tp.time_since_epoch()).count();
68 }
69
70 static constexpr time_point from_time_t(std::time_t t) noexcept
71 {
72 using namespace std::chrono;
73 using FromPoint = std::chrono::time_point<MonoClock, seconds>;
75 }
76};
77
78struct WallClock {
80 using period = Duration::period;
81 using rep = Duration::rep;
82 using time_point = std::chrono::time_point<WallClock, Duration>;
83
84 static constexpr int Id{CLOCK_REALTIME};
85 static constexpr bool is_steady{false};
86
87 static constexpr time_point max() noexcept
88 {
89 using namespace std::chrono;
90 using FromPoint = std::chrono::time_point<WallClock, seconds>;
91 constexpr seconds secs{std::numeric_limits<int>::max()};
93 }
94
96
97 static constexpr std::time_t to_time_t(const time_point& tp) noexcept
98 {
99 using namespace std::chrono;
100 return duration_cast<seconds>(tp.time_since_epoch()).count();
101 }
102
103 static constexpr time_point from_time_t(std::time_t t) noexcept
104 {
105 using namespace std::chrono;
106 using FromPoint = std::chrono::time_point<WallClock, seconds>;
108 }
109};
110
113
125 public:
126 static CyclTime current() noexcept { return {}; }
127 static CyclTime now() noexcept
128 {
129 time_ = Time::now();
130 return {}; // Empty tag.
131 }
133 static CyclTime now(WallTime wall_time) noexcept
134 {
135 time_ = Time::now(wall_time);
136 return {}; // Empty tag.
137 }
138 MonoTime mono_time() const noexcept { return time_.mono_time; }
139 WallTime wall_time() const noexcept { return time_.wall_time; }
140 void set_wall_time(WallTime wall_time) noexcept { time_.wall_time = wall_time; }
141
142 private:
143 CyclTime() = default;
144 struct Time {
145 static Time now() noexcept { return {MonoClock::now(), WallClock::now()}; };
146 static Time now(WallTime wall_time) noexcept { return {MonoClock::now(), wall_time}; };
147 MonoTime mono_time{};
148 WallTime wall_time{};
149 };
150 static thread_local Time time_;
151};
152
153template <typename RepT, typename PeriodT>
154constexpr bool is_zero(std::chrono::duration<RepT, PeriodT> d) noexcept
155{
156 return d == decltype(d){};
157}
158
159template <typename ClockT>
160constexpr bool is_zero(std::chrono::time_point<ClockT, Duration> t) noexcept
161{
162 return t == decltype(t){};
163}
164
165template <typename ClockT, typename DurationT>
166constexpr DurationT time_since_epoch(std::chrono::time_point<ClockT, Duration> t) noexcept
167{
168 using namespace std::chrono;
169 const Duration d{t.time_since_epoch()};
171}
172
173template <typename ClockT>
174constexpr std::int64_t ms_since_epoch(std::chrono::time_point<ClockT, Duration> t) noexcept
175{
176 using namespace std::chrono;
178}
179
180template <typename ClockT>
181constexpr std::int64_t us_since_epoch(std::chrono::time_point<ClockT, Duration> t) noexcept
182{
183 using namespace std::chrono;
185}
186
187template <typename ClockT>
188constexpr std::int64_t ns_since_epoch(std::chrono::time_point<ClockT, Duration> t) noexcept
189{
190 using namespace std::chrono;
191 const nanoseconds ns{t.time_since_epoch()};
192 return ns.count();
193}
194
195template <typename ClockT, typename RepT, typename PeriodT>
196constexpr auto to_time(std::chrono::duration<RepT, PeriodT> d) noexcept
197{
198 using namespace std::chrono;
199 return std::chrono::time_point<ClockT, Duration>{duration_cast<Duration>(d)};
200}
201
202template <typename ClockT>
203constexpr auto to_time(timeval tv) noexcept
204{
205 using namespace std::chrono;
206 return to_time<ClockT>(seconds{tv.tv_sec} + microseconds{tv.tv_usec});
207}
208
209template <typename ClockT>
210constexpr auto to_time(timespec ts) noexcept
211{
212 using namespace std::chrono;
213 return to_time<ClockT>(seconds{ts.tv_sec} + nanoseconds{ts.tv_nsec});
214}
215
216template <typename RepT, typename PeriodT>
217constexpr timeval to_timeval(std::chrono::duration<RepT, PeriodT> d) noexcept
218{
219 using namespace std::chrono;
220 const auto us = duration_cast<microseconds>(d).count();
221 return {static_cast<time_t>(us / 1'000'000L), static_cast<suseconds_t>(us % 1'000'000L)};
222}
223
224template <typename ClockT>
225constexpr timeval to_timeval(std::chrono::time_point<ClockT, Duration> t) noexcept
226{
227 using namespace std::chrono;
229}
230
231template <typename RepT, typename PeriodT>
232constexpr timespec to_timespec(std::chrono::duration<RepT, PeriodT> d) noexcept
233{
234 using namespace std::chrono;
235 const auto ns = duration_cast<nanoseconds>(d).count();
236 return {static_cast<time_t>(ns / 1'000'000'000L), static_cast<long>(ns % 1'000'000'000L)};
237}
238
239template <typename ClockT>
240constexpr timespec to_timespec(std::chrono::time_point<ClockT, Duration> t) noexcept
241{
242 using namespace std::chrono;
244}
245
246template <typename StreamT>
247 requires Streamable<StreamT>
249{
250 os << ns_since_epoch(t);
251 return os;
252}
253
254template <typename StreamT>
255 requires Streamable<StreamT>
257{
258 os << ns_since_epoch(t);
259 return os;
260}
261
262template <typename DurationT>
263struct PutTime {
265 const char* fmt;
266};
267
268template <typename DurationT = Seconds>
269auto put_time(WallTime t, const char* fmt) noexcept
270{
271 return PutTime<DurationT>{t, fmt};
272}
273
274template <typename DurationT, typename StreamT>
275 requires Streamable<StreamT>
277{
278 const auto t = WallClock::to_time_t(pt.time);
279 struct tm gmt;
280 os << to_string(std::put_time(gmtime_r(&t, &gmt), pt.fmt));
281
282 if constexpr (std::is_same_v<DurationT, Nanos>) {
283 const auto ns = ns_since_epoch<WallClock>(pt.time);
284 os << '.' << std::format("{:0>9}", ns % 1'000'000'000L);
285 } else if constexpr (std::is_same_v<DurationT, Micros>) {
286 const auto us = us_since_epoch<WallClock>(pt.time);
287 os << '.' << std::format("{:0>6}", us % 1'000'000L);
288 } else if constexpr (std::is_same_v<DurationT, Millis>) {
289 const auto ms = ms_since_epoch<WallClock>(pt.time);
290 os << '.' << std::format("{:0>3}", ms % 1'000L);
291 } else if constexpr (std::is_same_v<DurationT, Seconds>) {
292 } else {
293 static_assert(AlwaysFalse<DurationT>::value);
294 }
295 return os;
296}
297
298constexpr bool is_zero(timeval tv) noexcept
299{
300 return tv.tv_sec == 0 && tv.tv_usec == 0;
301}
302
303inline void clear(timeval& tv) noexcept
304{
305 tv.tv_sec = tv.tv_usec = 0;
306}
307
309{
310 timeval tv;
311 tv.tv_sec = lhs.tv_sec + rhs.tv_sec;
312 tv.tv_usec = lhs.tv_usec + rhs.tv_usec;
313 if (tv.tv_usec >= 1'000'000) {
314 ++tv.tv_sec;
315 tv.tv_usec -= 1'000'000;
316 }
317 return tv;
318}
319
321{
322 timeval tv;
323 tv.tv_sec = lhs.tv_sec - rhs.tv_sec;
324 tv.tv_usec = lhs.tv_usec - rhs.tv_usec;
325 if (tv.tv_usec < 0) {
326 --tv.tv_sec;
327 tv.tv_usec += 1'000'000;
328 }
329 return tv;
330}
331
332constexpr bool is_zero(timespec ts) noexcept
333{
334 return ts.tv_sec == 0 && ts.tv_nsec == 0;
335}
336
337inline void clear(timespec& ts) noexcept
338{
339 ts.tv_sec = ts.tv_nsec = 0;
340}
341
343{
344 timespec ts;
345 ts.tv_sec = lhs.tv_sec + rhs.tv_sec;
346 ts.tv_nsec = lhs.tv_nsec + rhs.tv_nsec;
347 if (ts.tv_nsec >= 1'000'000'000) {
348 ++ts.tv_sec;
349 ts.tv_nsec -= 1'000'000'000;
350 }
351 return ts;
352}
353
355{
356 timespec ts;
357 ts.tv_sec = lhs.tv_sec - rhs.tv_sec;
358 ts.tv_nsec = lhs.tv_nsec - rhs.tv_nsec;
359 if (ts.tv_nsec < 0) {
360 --ts.tv_sec;
361 ts.tv_nsec += 1'000'000'000;
362 }
363 return ts;
364}
365
369constexpr auto parse_nanos(std::string_view sv) noexcept
370{
371 // clang-format off
372 constexpr int c[] = {
373 0,
374 100'000'000,
375 10'000'000,
376 1'000'000,
377 100'000,
378 10'000,
379 1'000,
380 100,
381 10,
382 1
383 };
384 // clang-format on
385
386 // Truncate to ensure that we process no more than 9 decimal places.
387 sv = sv.substr(0, 9);
388 auto it = sv.begin(), end = sv.end();
389
390 int ns{0};
391 if (isdigit(*it)) {
392 ns = *it++ - '0';
393 while (it != end && isdigit(*it)) {
394 ns *= 10;
395 ns += *it++ - '0';
396 }
397 }
398 return Nanos{ns * c[it - sv.begin()]};
399}
400static_assert(parse_nanos("000000001") == 1ns);
401
408constexpr std::optional<Nanos> parse_time_only(std::string_view sv) noexcept
409{
410 using namespace std::chrono;
411
412 // clang-format off
413 if (sv.size() < 8
414 || !isdigit(sv[0])
415 || !isdigit(sv[1])
416 || sv[2] != ':'
417 || !isdigit(sv[3])
418 || !isdigit(sv[4])
419 || sv[5] != ':'
420 || !isdigit(sv[6])
421 || !isdigit(sv[7])) {
422 // Invalid format.
423 return {};
424 }
425 // clang-format on
426 const hours h{(sv[0] - '0') * 10 + sv[1] - '0'};
427 const minutes m{(sv[3] - '0') * 10 + sv[4] - '0'};
428 const seconds s{(sv[6] - '0') * 10 + sv[7] - '0'};
429 Nanos ns{h + m + s};
430 if (sv.size() > 8) {
431 if (sv[8] != '.') {
432 // Invalid delimiter.
433 return {};
434 }
435 ns += parse_nanos(sv.substr(9));
436 }
437 return ns;
438}
439static_assert(*parse_time_only("12:00:00"sv) == 12h);
440
441} // namespace sys
442inline namespace util {
443
444template <typename RepT, typename PeriodT>
445struct TypeTraits<std::chrono::duration<RepT, PeriodT>> {
446 static auto from_string(std::string_view sv) noexcept
447 {
448 using namespace std::chrono;
450 using Rep = typename Duration::rep;
452 }
453 static auto from_string(const std::string& s) noexcept
454 {
455 return from_string(std::string_view{s});
456 }
457};
458
459template <>
461 static auto from_string(std::string_view sv) noexcept
462 {
464 }
465 static auto from_string(const std::string& s) noexcept
466 {
467 return from_string(std::string_view{s});
468 }
469};
470
471} // namespace util
472} // namespace toolbox
473
474#endif // TOOLBOX_SYS_TIME_HPP
#define TOOLBOX_API
Definition Config.h:39
MonoTime mono_time() const noexcept
Definition Time.hpp:138
static CyclTime now() noexcept
Definition Time.hpp:127
static CyclTime current() noexcept
Definition Time.hpp:126
WallTime wall_time() const noexcept
Definition Time.hpp:139
static CyclTime now(WallTime wall_time) noexcept
This overload allows users to override wall-clock time.
Definition Time.hpp:133
void set_wall_time(WallTime wall_time) noexcept
Definition Time.hpp:140
STL namespace.
ostream & operator<<(ostream &os, const pair< T, U > &p)
Definition Parser.ut.cpp:29
constexpr bool is_zero(std::chrono::duration< RepT, PeriodT > d) noexcept
Definition Time.hpp:154
constexpr std::int64_t us_since_epoch(std::chrono::time_point< ClockT, Duration > t) noexcept
Definition Time.hpp:181
constexpr std::optional< Nanos > parse_time_only(std::string_view sv) noexcept
Definition Time.hpp:408
constexpr std::int64_t ms_since_epoch(std::chrono::time_point< ClockT, Duration > t) noexcept
Definition Time.hpp:174
constexpr DurationT time_since_epoch(std::chrono::time_point< ClockT, Duration > t) noexcept
Definition Time.hpp:166
constexpr std::int64_t ns_since_epoch(std::chrono::time_point< ClockT, Duration > t) noexcept
Definition Time.hpp:188
auto put_time(WallTime t, const char *fmt) noexcept
Definition Time.hpp:269
timeval operator-(timeval lhs, timeval rhs) noexcept
Definition Time.hpp:320
std::chrono::microseconds Micros
Definition Time.hpp:37
std::chrono::milliseconds Millis
Definition Time.hpp:36
WallClock::time_point WallTime
Definition Time.hpp:112
constexpr auto parse_nanos(std::string_view sv) noexcept
Definition Time.hpp:369
TOOLBOX_WEAK Nanos get_time(clockid_t clock_id) noexcept
Definition Time.cpp:25
MonoClock::time_point MonoTime
Definition Time.hpp:111
std::chrono::seconds Seconds
Definition Time.hpp:34
std::chrono::nanoseconds Nanos
Definition Time.hpp:38
constexpr auto to_time(std::chrono::duration< RepT, PeriodT > d) noexcept
Definition Time.hpp:196
constexpr timeval to_timeval(std::chrono::duration< RepT, PeriodT > d) noexcept
Definition Time.hpp:217
constexpr timespec to_timespec(std::chrono::duration< RepT, PeriodT > d) noexcept
Definition Time.hpp:232
Nanos NanoTime
Definition Time.hpp:40
std::chrono::duration< int64_t, std::deci > Decis
Definition Time.hpp:35
timeval operator+(timeval lhs, timeval rhs) noexcept
Definition Time.hpp:308
Nanos Duration
Definition Time.hpp:41
void clear(timeval &tv) noexcept
Definition Time.hpp:303
std::string to_string(ValueT &&val)
Definition String.hpp:54
constexpr bool isdigit(int c) noexcept
Definition Utility.hpp:40
std::string_view sv
Definition Tokeniser.hpp:26
constexpr auto bind() noexcept
Definition Slot.hpp:97
static constexpr int Id
Definition Time.hpp:51
std::chrono::time_point< MonoClock, Duration > time_point
Definition Time.hpp:49
Duration::rep rep
Definition Time.hpp:48
static constexpr std::time_t to_time_t(const time_point &tp) noexcept
Definition Time.hpp:64
static time_point now() noexcept
Definition Time.hpp:62
static constexpr time_point max() noexcept
Definition Time.hpp:54
static constexpr time_point from_time_t(std::time_t t) noexcept
Definition Time.hpp:70
Duration::period period
Definition Time.hpp:47
static constexpr bool is_steady
Definition Time.hpp:52
const char * fmt
Definition Time.hpp:265
static constexpr time_point from_time_t(std::time_t t) noexcept
Definition Time.hpp:103
std::chrono::time_point< WallClock, Duration > time_point
Definition Time.hpp:82
Duration::rep rep
Definition Time.hpp:81
static constexpr bool is_steady
Definition Time.hpp:85
static constexpr std::time_t to_time_t(const time_point &tp) noexcept
Definition Time.hpp:97
static constexpr time_point max() noexcept
Definition Time.hpp:87
static constexpr int Id
Definition Time.hpp:84
static time_point now() noexcept
Definition Time.hpp:95
Duration::period period
Definition Time.hpp:80
static auto from_string(std::string_view sv) noexcept
Definition Time.hpp:461
static auto from_string(const std::string &s) noexcept
Definition Time.hpp:465
static auto from_string(std::string_view sv) noexcept
Definition Time.hpp:446
static auto from_string(const std::string &s) noexcept
Definition Time.hpp:453
static constexpr auto from_string(StringT &&s) noexcept(noexcept(ValueT{s}))