193 lines
4.6 KiB
C++
193 lines
4.6 KiB
C++
#include "base/timer.hpp"
|
|
|
|
#include "base/assert.hpp"
|
|
#include "base/gmtime.hpp"
|
|
#include "base/timegm.hpp"
|
|
|
|
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <chrono>
|
|
#include <cstdio>
|
|
#include <iomanip>
|
|
#include <sstream>
|
|
|
|
#include <sys/time.h>
|
|
|
|
namespace base
|
|
{
|
|
using std::get_time;
|
|
|
|
Timer::Timer(bool start/* = true*/)
|
|
{
|
|
if (start)
|
|
Reset();
|
|
}
|
|
|
|
// static
|
|
double Timer::LocalTime()
|
|
{
|
|
timeval tv;
|
|
::gettimeofday(&tv, 0);
|
|
return tv.tv_sec + tv.tv_usec / 1000000.0;
|
|
}
|
|
|
|
std::string FormatCurrentTime()
|
|
{
|
|
time_t t = time(NULL);
|
|
std::string s(ctime(&t));
|
|
|
|
replace(s.begin(), s.end(), ' ', '_');
|
|
|
|
ASSERT_GREATER(s.size(), 1, ());
|
|
s.resize(s.size() - 1);
|
|
return s;
|
|
}
|
|
|
|
uint32_t GenerateYYMMDD(int year, int month, int day)
|
|
{
|
|
uint32_t result = (year - 100) * 100;
|
|
result = (result + month + 1) * 100;
|
|
result = result + day;
|
|
return result;
|
|
}
|
|
|
|
uint32_t GenerateYYMMDD(uint64_t secondsSinceEpoch)
|
|
{
|
|
auto const tm = GmTime(SecondsSinceEpochToTimeT(secondsSinceEpoch));
|
|
return GenerateYYMMDD(tm.tm_year, tm.tm_mon, tm.tm_mday);
|
|
}
|
|
|
|
uint64_t YYMMDDToSecondsSinceEpoch(uint32_t yymmdd)
|
|
{
|
|
auto constexpr partsCount = 3;
|
|
// From left to right YY MM DD.
|
|
std::array<int, partsCount> parts{}; // Initialize with zeros.
|
|
for (auto i = partsCount - 1; i >= 0; --i)
|
|
{
|
|
parts[i] = yymmdd % 100;
|
|
yymmdd /= 100;
|
|
}
|
|
ASSERT_EQUAL(yymmdd, 0, ("Version is too big."));
|
|
|
|
ASSERT_GREATER_OR_EQUAL(parts[1], 1, ("Month should be in range [1, 12]"));
|
|
ASSERT_LESS_OR_EQUAL(parts[1], 12, ("Month should be in range [1, 12]"));
|
|
ASSERT_GREATER_OR_EQUAL(parts[2], 1, ("Day should be in range [1, 31]"));
|
|
ASSERT_LESS_OR_EQUAL(parts[2], 31, ("Day should be in range [1, 31]"));
|
|
|
|
std::tm tm{};
|
|
tm.tm_year = parts[0] + 100;
|
|
tm.tm_mon = parts[1] - 1;
|
|
tm.tm_mday = parts[2];
|
|
|
|
return TimeTToSecondsSinceEpoch(TimeGM(tm));
|
|
}
|
|
|
|
uint64_t SecondsSinceEpoch()
|
|
{
|
|
return TimeTToSecondsSinceEpoch(::time(nullptr));
|
|
}
|
|
|
|
std::string TimestampToString(time_t time)
|
|
{
|
|
if (time == INVALID_TIME_STAMP)
|
|
return std::string("INVALID_TIME_STAMP");
|
|
|
|
std::tm * tm = std::gmtime(&time);
|
|
std::ostringstream ss;
|
|
ss << std::put_time(tm, "%FT%TZ");
|
|
return ss.str();
|
|
}
|
|
|
|
std::string SecondsSinceEpochToString(uint64_t secondsSinceEpoch)
|
|
{
|
|
return TimestampToString(SecondsSinceEpochToTimeT(secondsSinceEpoch));
|
|
}
|
|
|
|
namespace
|
|
{
|
|
bool IsValid(tm const & t)
|
|
{
|
|
/// @todo Funny thing, but "00" month is accepted as valid in get_time function.
|
|
/// Seems like a bug in the std library.
|
|
return (t.tm_mday >= 1 && t.tm_mday <= 31 &&
|
|
t.tm_mon >= 0 && t.tm_mon <= 11);
|
|
}
|
|
}
|
|
|
|
time_t StringToTimestamp(std::string const & s)
|
|
{
|
|
// Return current time in the case of failure
|
|
time_t res = INVALID_TIME_STAMP;
|
|
|
|
if (s.size() == 20)
|
|
{
|
|
// Parse UTC format: 1970-01-01T00:00:00Z
|
|
tm t{};
|
|
std::istringstream ss(s);
|
|
ss >> base::get_time(&t, "%Y-%m-%dT%H:%M:%SZ");
|
|
|
|
if (!ss.fail() && IsValid(t))
|
|
res = base::TimeGM(t);
|
|
}
|
|
else if (s.size() == 25)
|
|
{
|
|
// Parse custom time zone offset format: 2012-12-03T00:38:34+03:30
|
|
tm t1{}, t2{};
|
|
char sign = 0;
|
|
std::istringstream ss(s);
|
|
ss >> base::get_time(&t1, "%Y-%m-%dT%H:%M:%S") >> sign >> base::get_time(&t2, "%H:%M");
|
|
|
|
if (!ss.fail() && IsValid(t1))
|
|
{
|
|
time_t const tt = base::TimeGM(t1);
|
|
|
|
// Fix timezone offset
|
|
if (sign == '-')
|
|
res = tt + t2.tm_hour * 3600 + t2.tm_min * 60;
|
|
else if (sign == '+')
|
|
res = tt - t2.tm_hour * 3600 - t2.tm_min * 60;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
HighResTimer::HighResTimer(bool start/* = true*/)
|
|
{
|
|
if (start)
|
|
Reset();
|
|
}
|
|
|
|
void HighResTimer::Reset()
|
|
{
|
|
m_start = std::chrono::high_resolution_clock::now();
|
|
}
|
|
|
|
uint64_t HighResTimer::ElapsedNano() const
|
|
{
|
|
return std::chrono::duration_cast<std::chrono::nanoseconds>(std::chrono::high_resolution_clock::now() - m_start).count();
|
|
}
|
|
|
|
uint64_t HighResTimer::ElapsedMillis() const
|
|
{
|
|
return std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::high_resolution_clock::now() - m_start).count();
|
|
}
|
|
|
|
double HighResTimer::ElapsedSeconds() const
|
|
{
|
|
return std::chrono::duration_cast<std::chrono::duration<double>>(std::chrono::high_resolution_clock::now() - m_start).count();
|
|
}
|
|
|
|
time_t SecondsSinceEpochToTimeT(uint64_t secondsSinceEpoch)
|
|
{
|
|
std::chrono::time_point<std::chrono::system_clock> const tpoint{std::chrono::seconds(secondsSinceEpoch)};
|
|
return std::chrono::system_clock::to_time_t(tpoint);
|
|
}
|
|
|
|
uint64_t TimeTToSecondsSinceEpoch(time_t time)
|
|
{
|
|
auto const tpoint = std::chrono::system_clock::from_time_t(time);
|
|
return std::chrono::duration_cast<std::chrono::seconds>(tpoint.time_since_epoch()).count();
|
|
}
|
|
} // namespace base
|