QueryPerformanceCounter() retrieves the current value of the performance counter, which is a high resolution (<1us) time stamp that can be used for time-interval measurements. So, use it for MONOTONIC clock.
The GetSystemTimePreciseAsFileTime() function retrieves the current system date and time with the highest possible level of precision (<1us). Use it for real time clock. This function returns a counter representing the number of 100-nanosecond intervals since January 1, 1601. To make it compatible with Linux CLOCK_REALTIME, we need to calculate the 100-nanoseconds counter value till 01/01/1970. An upcoming commit implements gettimeofday() using the same clock, so, carve out a function. Signed-off-by: Gurucharan Shetty <gshe...@nicira.com> --- lib/timeval.c | 64 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++ lib/timeval.h | 13 ++++++++++++ 2 files changed, 77 insertions(+) diff --git a/lib/timeval.c b/lib/timeval.c index 3e7719f..07abb4c 100644 --- a/lib/timeval.c +++ b/lib/timeval.c @@ -74,6 +74,11 @@ static long long int deadline = LLONG_MAX; * up. */ DEFINE_STATIC_PER_THREAD_DATA(long long int, last_wakeup, 0); +#ifdef _WIN32 +/* Number of 100 ns intervals from January 1, 1601 till January 1, 1970. */ +static ULARGE_INTEGER unix_epoch; +#endif + static void log_poll_interval(long long int last_wakeup); static struct rusage *get_recent_rusage(void); static void refresh_rusage(void); @@ -96,6 +101,16 @@ do_init_time(void) { struct timespec ts; +#ifdef _WIN32 + /* Calculate number of 100-nanosecond intervals till 01/01/1970. */ + SYSTEMTIME unix_epoch_st = { 1970, 1, 0, 1, 0, 0, 0, 0}; + FILETIME unix_epoch_ft; + + SystemTimeToFileTime(&unix_epoch_st, &unix_epoch_ft); + unix_epoch.LowPart = unix_epoch_ft.dwLowDateTime; + unix_epoch.HighPart = unix_epoch_ft.dwHighDateTime; +#endif + coverage_init(); init_clock(&monotonic_clock, (!clock_gettime(CLOCK_MONOTONIC, &ts) @@ -325,6 +340,55 @@ time_boot_msec(void) return boot_time; } +#ifdef _WIN32 +static ULARGE_INTEGER +xgetfiletime(void) +{ + ULARGE_INTEGER current_time; + FILETIME current_time_ft; + + /* Returns current time in UTC as a 64-bit value representing the number + * of 100-nanosecond intervals since January 1, 1601 . */ + GetSystemTimePreciseAsFileTime(¤t_time_ft); + current_time.LowPart = current_time_ft.dwLowDateTime; + current_time.HighPart = current_time_ft.dwHighDateTime; + + return current_time; +} + +void +clock_gettime(clock_t id, struct timespec *ts) +{ + if (id == CLOCK_MONOTONIC) { + static LARGE_INTEGER freq; + LARGE_INTEGER count; + long long int ns; + + if (!freq.QuadPart) { + /* Number of counts per second. */ + QueryPerformanceFrequency(&freq); + } + /* Total number of counts from a starting point. */ + QueryPerformanceCounter(&count); + + /* Total nano seconds from a starting point. */ + ns = (double) count.QuadPart / freq.QuadPart * 1000000000; + + ts->tv_sec = count.QuadPart / freq.QuadPart; + ts->tv_nsec = ns % 1000000000; + } else if (id == CLOCK_REALTIME) { + ULARGE_INTEGER current_time = xgetfiletime(); + + /* Time from Epoch to now. */ + ts->tv_sec = (current_time.QuadPart - unix_epoch.QuadPart) / 10000000; + ts->tv_nsec = ((current_time.QuadPart - unix_epoch.QuadPart) % + 10000000) * 100; + } else { + return -1; + } +} +#endif /* _WIN32 */ + void xgettimeofday(struct timeval *tv) { diff --git a/lib/timeval.h b/lib/timeval.h index c207f23..1016879 100644 --- a/lib/timeval.h +++ b/lib/timeval.h @@ -40,6 +40,19 @@ BUILD_ASSERT_DECL(TYPE_IS_SIGNED(time_t)); #define TIME_MAX TYPE_MAXIMUM(time_t) #define TIME_MIN TYPE_MINIMUM(time_t) +#ifdef _WIN32 +typedef unsigned int clockid_t; + +#ifndef CLOCK_MONOTONIC +#define CLOCK_MONOTONIC 1 +#endif + +#ifndef CLOCK_REALTIME +#define CLOCK_REALTIME 2 +#endif + +#endif /* _WIN32 */ + struct tm_msec { struct tm tm; int msec; -- 1.7.9.5 _______________________________________________ dev mailing list dev@openvswitch.org http://openvswitch.org/mailman/listinfo/dev