mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-05-05 09:21:01 +02:00
fae86c38bca5c960462e53975314a0749db5d17d util: Remove unused MilliSleep (MarcoFalke) fa9af06d91e9357e86863781746f0e78a509967e scripted-diff: Replace MilliSleep with UninterruptibleSleep (MarcoFalke) fa4620be782c2bf6b5ffddf4f671194fdd1536f3 util: Add UnintrruptibleSleep (MarcoFalke) Pull request description: We don't use the interruptible feature of boost's sleep anywhere, so replace it with the sleep in `std::thread` ACKs for top commit: ajtowns: ACK fae86c38bca5c960462e53975314a0749db5d17d quick code review practicalswift: ACK fae86c38bca5c960462e53975314a0749db5d17d -- patch looks correct sipa: Concept and code review ACK fae86c38bca5c960462e53975314a0749db5d17d fanquake: ACK fae86c38bca5c960462e53975314a0749db5d17d - note that an instance of `DHAVE_WORKING_BOOST_SLEEP_FOR` was missed in the [linter](https://github.com/bitcoin/bitcoin/blob/master/test/lint/extended-lint-cppcheck.sh#L69), but that can be cleaned up later. Tree-SHA512: 7c0f8eb197664b9f7d9fe6c472c77d384f11c797c913afc31de4b532e3b4fd9ea6dd174f92062ff9d1ec39b25e0900ca7c597435add87f0f2477d9557204848c
117 lines
3.1 KiB
C++
117 lines
3.1 KiB
C++
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
|
// Copyright (c) 2009-2019 The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
#include <config/bitcoin-config.h>
|
|
#endif
|
|
|
|
#include <util/time.h>
|
|
|
|
#include <atomic>
|
|
#include <boost/date_time/posix_time/posix_time.hpp>
|
|
#include <ctime>
|
|
#include <thread>
|
|
|
|
#include <tinyformat.h>
|
|
|
|
void UninterruptibleSleep(const std::chrono::microseconds& n) { std::this_thread::sleep_for(n); }
|
|
|
|
static std::atomic<int64_t> nMockTime(0); //!< For unit testing
|
|
|
|
int64_t GetTime()
|
|
{
|
|
int64_t mocktime = nMockTime.load(std::memory_order_relaxed);
|
|
if (mocktime) return mocktime;
|
|
|
|
time_t now = time(nullptr);
|
|
assert(now > 0);
|
|
return now;
|
|
}
|
|
|
|
template <typename T>
|
|
T GetTime()
|
|
{
|
|
const std::chrono::seconds mocktime{nMockTime.load(std::memory_order_relaxed)};
|
|
|
|
return std::chrono::duration_cast<T>(
|
|
mocktime.count() ?
|
|
mocktime :
|
|
std::chrono::microseconds{GetTimeMicros()});
|
|
}
|
|
template std::chrono::seconds GetTime();
|
|
template std::chrono::milliseconds GetTime();
|
|
template std::chrono::microseconds GetTime();
|
|
|
|
void SetMockTime(int64_t nMockTimeIn)
|
|
{
|
|
nMockTime.store(nMockTimeIn, std::memory_order_relaxed);
|
|
}
|
|
|
|
int64_t GetMockTime()
|
|
{
|
|
return nMockTime.load(std::memory_order_relaxed);
|
|
}
|
|
|
|
int64_t GetTimeMillis()
|
|
{
|
|
int64_t now = (boost::posix_time::microsec_clock::universal_time() -
|
|
boost::posix_time::ptime(boost::gregorian::date(1970,1,1))).total_milliseconds();
|
|
assert(now > 0);
|
|
return now;
|
|
}
|
|
|
|
int64_t GetTimeMicros()
|
|
{
|
|
int64_t now = (boost::posix_time::microsec_clock::universal_time() -
|
|
boost::posix_time::ptime(boost::gregorian::date(1970,1,1))).total_microseconds();
|
|
assert(now > 0);
|
|
return now;
|
|
}
|
|
|
|
int64_t GetSystemTimeInSeconds()
|
|
{
|
|
return GetTimeMicros()/1000000;
|
|
}
|
|
|
|
std::string FormatISO8601DateTime(int64_t nTime) {
|
|
struct tm ts;
|
|
time_t time_val = nTime;
|
|
#ifdef _MSC_VER
|
|
if (gmtime_s(&ts, &time_val) != 0) {
|
|
#else
|
|
if (gmtime_r(&time_val, &ts) == nullptr) {
|
|
#endif
|
|
return {};
|
|
}
|
|
return strprintf("%04i-%02i-%02iT%02i:%02i:%02iZ", ts.tm_year + 1900, ts.tm_mon + 1, ts.tm_mday, ts.tm_hour, ts.tm_min, ts.tm_sec);
|
|
}
|
|
|
|
std::string FormatISO8601Date(int64_t nTime) {
|
|
struct tm ts;
|
|
time_t time_val = nTime;
|
|
#ifdef _MSC_VER
|
|
if (gmtime_s(&ts, &time_val) != 0) {
|
|
#else
|
|
if (gmtime_r(&time_val, &ts) == nullptr) {
|
|
#endif
|
|
return {};
|
|
}
|
|
return strprintf("%04i-%02i-%02i", ts.tm_year + 1900, ts.tm_mon + 1, ts.tm_mday);
|
|
}
|
|
|
|
int64_t ParseISO8601DateTime(const std::string& str)
|
|
{
|
|
static const boost::posix_time::ptime epoch = boost::posix_time::from_time_t(0);
|
|
static const std::locale loc(std::locale::classic(),
|
|
new boost::posix_time::time_input_facet("%Y-%m-%dT%H:%M:%SZ"));
|
|
std::istringstream iss(str);
|
|
iss.imbue(loc);
|
|
boost::posix_time::ptime ptime(boost::date_time::not_a_date_time);
|
|
iss >> ptime;
|
|
if (ptime.is_not_a_date_time() || epoch > ptime)
|
|
return 0;
|
|
return (ptime - epoch).total_seconds();
|
|
}
|