Bitcoin ABC 0.33.1
P2P Digital Currency
time.cpp
Go to the documentation of this file.
1// Copyright (c) 2009-2010 Satoshi Nakamoto
2// Copyright (c) 2009-present The Bitcoin Core developers
3// Distributed under the MIT software license, see the accompanying
4// file COPYING or http://www.opensource.org/licenses/mit-license.php.
5
6#include <util/time.h>
7
8#include <compat/compat.h>
9#include <util/check.h>
10#include <util/strencodings.h>
11
12#include <tinyformat.h>
13
14#include <atomic>
15#include <chrono>
16#include <optional>
17#include <string>
18#include <string_view>
19#include <thread>
20
21void UninterruptibleSleep(const std::chrono::microseconds &n) {
22 std::this_thread::sleep_for(n);
23}
24
26static std::atomic<int64_t> nMockTime(0);
27static std::atomic<std::chrono::milliseconds> g_mock_steady_time{};
28
30 const std::chrono::seconds mocktime{
31 nMockTime.load(std::memory_order_relaxed)};
32
33 const auto ret{mocktime.count()
34 ? mocktime
35 : std::chrono::system_clock::now().time_since_epoch()};
36 assert(ret > 0s);
37 return time_point{ret};
38};
39
40template <typename T> static T GetSystemTime() {
41 const auto now = std::chrono::duration_cast<T>(
42 std::chrono::system_clock::now().time_since_epoch());
43 assert(now.count() > 0);
44 return now;
45}
46
48 const auto mocktime{g_mock_steady_time.load(std::memory_order_relaxed)};
49 const auto ret{mocktime.count()
50 ? mocktime
51 : std::chrono::steady_clock::now().time_since_epoch()};
52 return time_point{ret};
53};
54
55void MockableSteadyClock::SetMockTime(std::chrono::milliseconds mock_time_in) {
56 Assert(mock_time_in >= 0s);
57 g_mock_steady_time.store(mock_time_in, std::memory_order_relaxed);
58}
59
61 g_mock_steady_time.store(0ms, std::memory_order_relaxed);
62}
63
64void SetMockTime(int64_t nMockTimeIn) {
65 Assert(nMockTimeIn >= 0);
66 nMockTime.store(nMockTimeIn, std::memory_order_relaxed);
67}
68
69void SetMockTime(std::chrono::seconds mock_time_in) {
70 nMockTime.store(mock_time_in.count(), std::memory_order_relaxed);
71}
72std::chrono::seconds GetMockTime() {
73 return std::chrono::seconds(nMockTime.load(std::memory_order_relaxed));
74}
75
76int64_t GetTimeMillis() {
77 return int64_t{GetSystemTime<std::chrono::milliseconds>().count()};
78}
79
80int64_t GetTime() {
81 return GetTime<std::chrono::seconds>().count();
82}
83
84// According to https://en.cppreference.com/w/cpp/chrono/year.html, the
85// valid year range is [-32767, 32767].
86// Rejecting values outside this year range guards against
87// a segfault (obeserved with GCC 13.3.0 + Debug build) for very negative
88// values (e.g. std::numeric_limits<int64_t>::min()), and is consistent
89// with the behavior of std::chrono::year_month_day::ok()
90static bool IsInvalidTimestamp(int64_t nTime) {
91 constexpr int64_t december_31st_32767{971890963199};
92 constexpr int64_t january_1st_minus32767{-1096193779200};
93 return nTime > december_31st_32767 || nTime < january_1st_minus32767;
94}
95
96std::string FormatISO8601DateTime(int64_t nTime) {
97 if (IsInvalidTimestamp(nTime)) {
98 return {};
99 }
100 const std::chrono::sys_seconds secs{std::chrono::seconds{nTime}};
101 const auto days{std::chrono::floor<std::chrono::days>(secs)};
102 const std::chrono::year_month_day ymd{days};
103 const std::chrono::hh_mm_ss hms{secs - days};
104 return strprintf("%04i-%02u-%02uT%02i:%02i:%02iZ", signed{ymd.year()},
105 unsigned{ymd.month()}, unsigned{ymd.day()},
106 hms.hours().count(), hms.minutes().count(),
107 hms.seconds().count());
108}
109
110std::string FormatISO8601Date(int64_t nTime) {
111 if (IsInvalidTimestamp(nTime)) {
112 return {};
113 }
114 const std::chrono::sys_seconds secs{std::chrono::seconds{nTime}};
115 const auto days{std::chrono::floor<std::chrono::days>(secs)};
116 const std::chrono::year_month_day ymd{days};
117 return strprintf("%04i-%02u-%02u", signed{ymd.year()},
118 unsigned{ymd.month()}, unsigned{ymd.day()});
119}
120
121std::optional<int64_t> ParseISO8601DateTime(std::string_view str) {
122 constexpr auto FMT_SIZE{std::string_view{"2000-01-01T01:01:01Z"}.size()};
123 if (str.size() != FMT_SIZE || str[4] != '-' || str[7] != '-' ||
124 str[10] != 'T' || str[13] != ':' || str[16] != ':' || str[19] != 'Z') {
125 return {};
126 }
127 const auto year{ToIntegral<uint16_t>(str.substr(0, 4))};
128 const auto month{ToIntegral<uint8_t>(str.substr(5, 2))};
129 const auto day{ToIntegral<uint8_t>(str.substr(8, 2))};
130 const auto hour{ToIntegral<uint8_t>(str.substr(11, 2))};
131 const auto min{ToIntegral<uint8_t>(str.substr(14, 2))};
132 const auto sec{ToIntegral<uint8_t>(str.substr(17, 2))};
133 if (!year || !month || !day || !hour || !min || !sec) {
134 return {};
135 }
136 const std::chrono::year_month_day ymd{std::chrono::year{*year},
137 std::chrono::month{*month},
138 std::chrono::day{*day}};
139 if (!ymd.ok()) {
140 return {};
141 }
142 const auto time{std::chrono::hours{*hour} + std::chrono::minutes{*min} +
143 std::chrono::seconds{*sec}};
144 const auto tp{std::chrono::sys_days{ymd} + time};
145 return int64_t{TicksSinceEpoch<std::chrono::seconds>(tp)};
146}
147
148struct timeval MillisToTimeval(int64_t nTimeout) {
149 struct timeval timeout;
150 timeout.tv_sec = nTimeout / 1000;
151 timeout.tv_usec = (nTimeout % 1000) * 1000;
152 return timeout;
153}
154
155struct timeval MillisToTimeval(std::chrono::milliseconds ms) {
157}
#define Assert(val)
Identity function.
Definition: check.h:84
Implement std::hash so RCUPtr can be used as a key for maps or sets.
Definition: rcu.h:259
static time_point now() noexcept
Return current system time or mocked time, if set.
Definition: time.cpp:47
static void SetMockTime(std::chrono::milliseconds mock_time_in)
Set mock time for testing.
Definition: time.cpp:55
static void ClearMockTime()
Clear mock time, go back to system steady clock.
Definition: time.cpp:60
std::chrono::time_point< MockableSteadyClock > time_point
Definition: time.h:44
static time_point now() noexcept
Return current system time or mocked time, if set.
Definition: time.cpp:29
std::chrono::time_point< NodeClock > time_point
Definition: time.h:21
static int count
struct timeval MillisToTimeval(int64_t nTimeout)
Convert milliseconds to a struct timeval for e.g.
Definition: time.cpp:148
int64_t GetTimeMillis()
Returns the system time (not mockable)
Definition: time.cpp:76
static std::atomic< std::chrono::milliseconds > g_mock_steady_time
Definition: time.cpp:27
void UninterruptibleSleep(const std::chrono::microseconds &n)
Definition: time.cpp:21
static std::atomic< int64_t > nMockTime(0)
For testing.
std::chrono::seconds GetMockTime()
For testing.
Definition: time.cpp:72
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
Definition: time.cpp:80
std::string FormatISO8601Date(int64_t nTime)
Definition: time.cpp:110
static bool IsInvalidTimestamp(int64_t nTime)
Definition: time.cpp:90
void SetMockTime(int64_t nMockTimeIn)
DEPRECATED Use SetMockTime with chrono type.
Definition: time.cpp:64
std::optional< int64_t > ParseISO8601DateTime(std::string_view str)
Definition: time.cpp:121
static T GetSystemTime()
Definition: time.cpp:40
std::string FormatISO8601DateTime(int64_t nTime)
ISO 8601 formatting is preferred.
Definition: time.cpp:96
constexpr int64_t count_milliseconds(std::chrono::milliseconds t)
Definition: time.h:88
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1202
assert(!tx.IsCoinBase())