// // Timestamp.cpp // // $Id: //poco/1.4/Foundation/src/Timestamp.cpp#2 $ // // Library: Foundation // Package: DateTime // Module: Timestamp // // Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH. // and Contributors. // // SPDX-License-Identifier: BSL-1.0 // #include "Poco/Timestamp.h" #include "Poco/Timespan.h" #include "Poco/Exception.h" #include #undef min #undef max #include #if defined(POCO_OS_FAMILY_UNIX) #include #include #if defined(POCO_VXWORKS) #include #else #include #include #endif #elif defined(POCO_OS_FAMILY_WINDOWS) #include "Poco/UnWindows.h" #if defined(_WIN32_WCE) #include #endif #endif #if defined(_WIN32_WCE) // // See // for an explanation of the following code. // // In short: Windows CE system time in most cases only has a resolution of one second. // But we want millisecond resolution. // namespace { class TickOffset { public: TickOffset() { SYSTEMTIME st1, st2; std::memset(&st1, 0, sizeof(SYSTEMTIME)); std::memset(&st2, 0, sizeof(SYSTEMTIME)); GetSystemTime(&st1); while (true) { GetSystemTime(&st2); // wait for a rollover if (st1.wSecond != st2.wSecond) { _offset = GetTickCount() % 1000; break; } } } void calibrate(int seconds) { SYSTEMTIME st1, st2; systemTime(&st1); WORD s = st1.wSecond; int sum = 0; int remaining = seconds; while (remaining > 0) { systemTime(&st2); WORD s2 = st2.wSecond; if (s != s2) { remaining--; // store the offset from zero sum += (st2.wMilliseconds > 500) ? (st2.wMilliseconds - 1000) : st2.wMilliseconds; s = st2.wSecond; } } // adjust the offset by the average deviation from zero (round to the integer farthest from zero) if (sum < 0) _offset += (int) std::floor(sum / (float)seconds); else _offset += (int) std::ceil(sum / (float)seconds); } void systemTime(SYSTEMTIME* pST) { std::memset(pST, 0, sizeof(SYSTEMTIME)); WORD tick = GetTickCount() % 1000; GetSystemTime(pST); WORD ms = (tick >= _offset) ? (tick - _offset) : (1000 - (_offset - tick)); pST->wMilliseconds = ms; } void systemTimeAsFileTime(FILETIME* pFT) { SYSTEMTIME st; systemTime(&st); SystemTimeToFileTime(&st, pFT); } private: WORD _offset; }; static TickOffset offset; void GetSystemTimeAsFileTimeWithMillisecondResolution(FILETIME* pFT) { offset.systemTimeAsFileTime(pFT); } } // namespace #endif // defined(_WIN32_WCE) namespace Poco { const Timestamp::TimeVal Timestamp::TIMEVAL_MIN = std::numeric_limits::min(); const Timestamp::TimeVal Timestamp::TIMEVAL_MAX = std::numeric_limits::max(); Timestamp::Timestamp() { update(); } Timestamp::Timestamp(TimeVal tv) { _ts = tv; } Timestamp::Timestamp(const Timestamp& other) { _ts = other._ts; } Timestamp::~Timestamp() { } Timestamp& Timestamp::operator = (const Timestamp& other) { _ts = other._ts; return *this; } Timestamp& Timestamp::operator = (TimeVal tv) { _ts = tv; return *this; } void Timestamp::swap(Timestamp& timestamp) { std::swap(_ts, timestamp._ts); } Timestamp Timestamp::fromEpochTime(std::time_t t) { return Timestamp(TimeVal(t)*resolution()); } Timestamp Timestamp::fromUtcTime(UtcTimeVal val) { val -= (TimeDiff(0x01b21dd2) << 32) + 0x13814000; val /= 10; return Timestamp(val); } void Timestamp::update() { #if defined(POCO_OS_FAMILY_WINDOWS) FILETIME ft; #if defined(_WIN32_WCE) GetSystemTimeAsFileTimeWithMillisecondResolution(&ft); #else GetSystemTimeAsFileTime(&ft); #endif ULARGE_INTEGER epoch; // UNIX epoch (1970-01-01 00:00:00) expressed in Windows NT FILETIME epoch.LowPart = 0xD53E8000; epoch.HighPart = 0x019DB1DE; ULARGE_INTEGER ts; ts.LowPart = ft.dwLowDateTime; ts.HighPart = ft.dwHighDateTime; ts.QuadPart -= epoch.QuadPart; _ts = ts.QuadPart/10; #elif defined(POCO_VXWORKS) struct timespec ts; if (clock_gettime(CLOCK_REALTIME, &ts)) throw SystemException("cannot get time of day"); _ts = TimeVal(ts.tv_sec)*resolution() + ts.tv_nsec/1000; #else struct timeval tv; if (gettimeofday(&tv, NULL)) throw SystemException("cannot get time of day"); _ts = TimeVal(tv.tv_sec)*resolution() + tv.tv_usec; #endif } Timestamp Timestamp::operator + (const Timespan& span) const { return *this + span.totalMicroseconds(); } Timestamp Timestamp::operator - (const Timespan& span) const { return *this - span.totalMicroseconds(); } Timestamp& Timestamp::operator += (const Timespan& span) { return *this += span.totalMicroseconds(); } Timestamp& Timestamp::operator -= (const Timespan& span) { return *this -= span.totalMicroseconds(); } #if defined(_WIN32) Timestamp Timestamp::fromFileTimeNP(UInt32 fileTimeLow, UInt32 fileTimeHigh) { ULARGE_INTEGER epoch; // UNIX epoch (1970-01-01 00:00:00) expressed in Windows NT FILETIME epoch.LowPart = 0xD53E8000; epoch.HighPart = 0x019DB1DE; ULARGE_INTEGER ts; ts.LowPart = fileTimeLow; ts.HighPart = fileTimeHigh; ts.QuadPart -= epoch.QuadPart; return Timestamp(ts.QuadPart/10); } void Timestamp::toFileTimeNP(UInt32& fileTimeLow, UInt32& fileTimeHigh) const { ULARGE_INTEGER epoch; // UNIX epoch (1970-01-01 00:00:00) expressed in Windows NT FILETIME epoch.LowPart = 0xD53E8000; epoch.HighPart = 0x019DB1DE; ULARGE_INTEGER ts; ts.QuadPart = _ts*10; ts.QuadPart += epoch.QuadPart; fileTimeLow = ts.LowPart; fileTimeHigh = ts.HighPart; } #endif } // namespace Poco