trunk/branch integration: VxWorks & Wince

This commit is contained in:
Marian Krivos 2011-08-23 07:13:59 +00:00
parent 35f44e5a2d
commit fd733bcca4
2 changed files with 138 additions and 17 deletions

View File

@ -1,7 +1,7 @@
// //
// Timestamp.cpp // Timestamp.cpp
// //
// $Id: //poco/svn/Foundation/src/Timestamp.cpp#2 $ // $Id: //poco/1.4/Foundation/src/Timestamp.cpp#2 $
// //
// Library: Foundation // Library: Foundation
// Package: DateTime // Package: DateTime
@ -40,11 +40,121 @@
#if defined(POCO_OS_FAMILY_UNIX) #if defined(POCO_OS_FAMILY_UNIX)
#include <time.h> #include <time.h>
#include <unistd.h> #include <unistd.h>
#if defined(POCO_VXWORKS)
#include <timers.h>
#else
#include <sys/time.h> #include <sys/time.h>
#include <sys/times.h> #include <sys/times.h>
#endif
#elif defined(POCO_OS_FAMILY_WINDOWS) #elif defined(POCO_OS_FAMILY_WINDOWS)
#include "Poco/UnWindows.h" #include "Poco/UnWindows.h"
#if defined(_WIN32_WCE)
#include <cmath>
#endif #endif
#endif
#if defined(_WIN32_WCE)
//
// See <http://community.opennetcf.com/articles/cf/archive/2007/11/20/getting-a-millisecond-resolution-datetime-under-windows-ce.aspx>
// 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 { namespace Poco {
@ -111,21 +221,33 @@ void Timestamp::update()
{ {
#if defined(POCO_OS_FAMILY_WINDOWS) #if defined(POCO_OS_FAMILY_WINDOWS)
FILETIME ft; FILETIME ft;
GetSystemTimeAsFileTime(&ft); #if defined(_WIN32_WCE)
ULARGE_INTEGER epoch; // UNIX epoch (1970-01-01 00:00:00) expressed in Windows NT FILETIME GetSystemTimeAsFileTimeWithMillisecondResolution(&ft);
epoch.LowPart = 0xD53E8000; #else
epoch.HighPart = 0x019DB1DE; 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; ULARGE_INTEGER ts;
ts.LowPart = ft.dwLowDateTime; ts.LowPart = ft.dwLowDateTime;
ts.HighPart = ft.dwHighDateTime; ts.HighPart = ft.dwHighDateTime;
ts.QuadPart -= epoch.QuadPart; ts.QuadPart -= epoch.QuadPart;
_ts = ts.QuadPart/10; _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 #else
struct timeval tv; struct timeval tv;
if (gettimeofday(&tv, NULL)) if (gettimeofday(&tv, NULL))
throw SystemException("cannot get time of day"); throw SystemException("cannot get time of day");
_ts = TimeVal(tv.tv_sec)*resolution() + tv.tv_usec; _ts = TimeVal(tv.tv_sec)*resolution() + tv.tv_usec;
@ -169,5 +291,4 @@ void Timestamp::toFileTimeNP(UInt32& fileTimeLow, UInt32& fileTimeHigh) const
#endif #endif
} // namespace Poco } // namespace Poco

View File

@ -49,13 +49,13 @@ public:
{ {
tzset(); tzset();
} }
int timeZone() int timeZone()
{ {
#if defined(__APPLE__) || defined(__FreeBSD__) // no timezone global var #if defined(__APPLE__) || defined(__FreeBSD__) || defined(POCO_ANDROID) // no timezone global var
std::time_t now = std::time(NULL); std::time_t now = std::time(NULL);
struct std::tm t; struct std::tm t;
gmtime_r(&now, &t); gmtime_r(&now, &t);
std::time_t utc = std::mktime(&t); std::time_t utc = std::mktime(&t);
return now - utc; return now - utc;
#elif defined(__CYGWIN__) #elif defined(__CYGWIN__)