/root/bitcoin/src/util/time.cpp
Line | Count | Source |
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 <tinyformat.h> |
10 | | #include <util/check.h> |
11 | | #include <util/strencodings.h> |
12 | | |
13 | | #include <atomic> |
14 | | #include <chrono> |
15 | | #include <optional> |
16 | | #include <string> |
17 | | #include <string_view> |
18 | | #include <thread> |
19 | | |
20 | 0 | void UninterruptibleSleep(const std::chrono::microseconds& n) { std::this_thread::sleep_for(n); } |
21 | | |
22 | | static std::atomic<std::chrono::seconds> g_mock_time{}; //!< For testing |
23 | | std::atomic<bool> g_used_system_time{false}; |
24 | | static std::atomic<MockableSteadyClock::mock_time_point::duration> g_mock_steady_time{}; //!< For testing |
25 | | |
26 | | NodeClock::time_point NodeClock::now() noexcept |
27 | 0 | { |
28 | 0 | const auto mocktime{g_mock_time.load(std::memory_order_relaxed)}; |
29 | 0 | if (!mocktime.count()) { |
30 | 0 | g_used_system_time = true; |
31 | 0 | } |
32 | 0 | const auto ret{ |
33 | 0 | mocktime.count() ? |
34 | 0 | mocktime : |
35 | 0 | std::chrono::system_clock::now().time_since_epoch()}; |
36 | 0 | assert(ret > 0s); |
37 | 0 | return time_point{ret}; |
38 | 0 | }; |
39 | | |
40 | 0 | void SetMockTime(int64_t nMockTimeIn) { SetMockTime(std::chrono::seconds{nMockTimeIn}); } |
41 | | void SetMockTime(std::chrono::seconds mock_time_in) |
42 | 5.34k | { |
43 | 5.34k | Assert(mock_time_in >= 0s); |
44 | 5.34k | g_mock_time.store(mock_time_in, std::memory_order_relaxed); |
45 | 5.34k | } |
46 | | |
47 | | std::chrono::seconds GetMockTime() |
48 | 0 | { |
49 | 0 | return g_mock_time.load(std::memory_order_relaxed); |
50 | 0 | } |
51 | | |
52 | | MockableSteadyClock::time_point MockableSteadyClock::now() noexcept |
53 | 0 | { |
54 | 0 | const auto mocktime{g_mock_steady_time.load(std::memory_order_relaxed)}; |
55 | 0 | if (!mocktime.count()) { |
56 | 0 | g_used_system_time = true; |
57 | 0 | } |
58 | 0 | const auto ret{ |
59 | 0 | mocktime.count() ? |
60 | 0 | mocktime : |
61 | 0 | std::chrono::steady_clock::now().time_since_epoch()}; |
62 | 0 | return time_point{ret}; |
63 | 0 | }; |
64 | | |
65 | | void MockableSteadyClock::SetMockTime(mock_time_point::duration mock_time_in) |
66 | 0 | { |
67 | 0 | Assert(mock_time_in >= 0s); |
68 | 0 | g_mock_steady_time.store(mock_time_in, std::memory_order_relaxed); |
69 | 0 | } |
70 | | |
71 | | void MockableSteadyClock::ClearMockTime() |
72 | 0 | { |
73 | 0 | g_mock_steady_time.store(0ms, std::memory_order_relaxed); |
74 | 0 | } |
75 | | |
76 | 0 | int64_t GetTime() { return GetTime<std::chrono::seconds>().count(); } |
77 | | |
78 | | std::string FormatISO8601DateTime(int64_t nTime) |
79 | 0 | { |
80 | 0 | const std::chrono::sys_seconds secs{std::chrono::seconds{nTime}}; |
81 | 0 | const auto days{std::chrono::floor<std::chrono::days>(secs)}; |
82 | 0 | const std::chrono::year_month_day ymd{days}; |
83 | 0 | const std::chrono::hh_mm_ss hms{secs - days}; |
84 | 0 | return strprintf("%04i-%02u-%02uT%02i:%02i:%02iZ", signed{ymd.year()}, unsigned{ymd.month()}, unsigned{ymd.day()}, hms.hours().count(), hms.minutes().count(), hms.seconds().count()); |
85 | 0 | } |
86 | | |
87 | | std::string FormatISO8601Date(int64_t nTime) |
88 | 0 | { |
89 | 0 | const std::chrono::sys_seconds secs{std::chrono::seconds{nTime}}; |
90 | 0 | const auto days{std::chrono::floor<std::chrono::days>(secs)}; |
91 | 0 | const std::chrono::year_month_day ymd{days}; |
92 | 0 | return strprintf("%04i-%02u-%02u", signed{ymd.year()}, unsigned{ymd.month()}, unsigned{ymd.day()}); |
93 | 0 | } |
94 | | |
95 | | std::optional<int64_t> ParseISO8601DateTime(std::string_view str) |
96 | 0 | { |
97 | 0 | constexpr auto FMT_SIZE{std::string_view{"2000-01-01T01:01:01Z"}.size()}; |
98 | 0 | if (str.size() != FMT_SIZE || str[4] != '-' || str[7] != '-' || str[10] != 'T' || str[13] != ':' || str[16] != ':' || str[19] != 'Z') { |
99 | 0 | return {}; |
100 | 0 | } |
101 | 0 | const auto year{ToIntegral<uint16_t>(str.substr(0, 4))}; |
102 | 0 | const auto month{ToIntegral<uint8_t>(str.substr(5, 2))}; |
103 | 0 | const auto day{ToIntegral<uint8_t>(str.substr(8, 2))}; |
104 | 0 | const auto hour{ToIntegral<uint8_t>(str.substr(11, 2))}; |
105 | 0 | const auto min{ToIntegral<uint8_t>(str.substr(14, 2))}; |
106 | 0 | const auto sec{ToIntegral<uint8_t>(str.substr(17, 2))}; |
107 | 0 | if (!year || !month || !day || !hour || !min || !sec) { |
108 | 0 | return {}; |
109 | 0 | } |
110 | 0 | const std::chrono::year_month_day ymd{std::chrono::year{*year}, std::chrono::month{*month}, std::chrono::day{*day}}; |
111 | 0 | if (!ymd.ok()) { |
112 | 0 | return {}; |
113 | 0 | } |
114 | 0 | const auto time{std::chrono::hours{*hour} + std::chrono::minutes{*min} + std::chrono::seconds{*sec}}; |
115 | 0 | const auto tp{std::chrono::sys_days{ymd} + time}; |
116 | 0 | return int64_t{TicksSinceEpoch<std::chrono::seconds>(tp)}; |
117 | 0 | } |
118 | | |
119 | | struct timeval MillisToTimeval(int64_t nTimeout) |
120 | 0 | { |
121 | 0 | struct timeval timeout; |
122 | 0 | timeout.tv_sec = nTimeout / 1000; |
123 | 0 | timeout.tv_usec = (nTimeout % 1000) * 1000; |
124 | 0 | return timeout; |
125 | 0 | } |
126 | | |
127 | | struct timeval MillisToTimeval(std::chrono::milliseconds ms) |
128 | 0 | { |
129 | 0 | return MillisToTimeval(count_milliseconds(ms)); |
130 | 0 | } |