VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/nt/time-r0drv-nt.cpp@ 53517

Last change on this file since 53517 was 53456, checked in by vboxsync, 10 years ago

RTTime/r0drv/nt: Use higher precision APIs to get times when available.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 4.2 KB
Line 
1/* $Id: time-r0drv-nt.cpp 53456 2014-12-05 12:52:46Z vboxsync $ */
2/** @file
3 * IPRT - Time, Ring-0 Driver, Nt.
4 */
5
6/*
7 * Copyright (C) 2007-2014 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#define LOG_GROUP RTLOGGROUP_TIME
32#include "the-nt-kernel.h"
33#include "internal-r0drv-nt.h"
34#include <iprt/time.h>
35
36
37DECLINLINE(uint64_t) rtTimeGetSystemNanoTS(void)
38{
39 /*
40 * Note! The time source we use here must be exactly the same as in
41 * the ring-3 code!
42 *
43 * Using interrupt time is the simplest and requires the least calculation.
44 * It is also accounting for suspended time. Unfortuantely, there is no
45 * ring-3 for reading it... but that won't stop us.
46 *
47 * Using the tick count is problematic in ring-3 on older windows version
48 * as we can only get the 32-bit tick value, i.e. we'll roll over sooner or
49 * later.
50 */
51#if 1
52 /* Interrupt time. */
53 LARGE_INTEGER InterruptTime;
54 if (g_pfnrtKeQueryInterruptTimePrecise)
55 {
56 ULONG64 QpcTsIgnored;
57 InterruptTime.QuadPart = g_pfnrtKeQueryInterruptTimePrecise(&QpcTsIgnored);
58 }
59# ifdef RT_ARCH_AMD64
60 else
61 InterruptTime.QuadPart = KeQueryInterruptTime(); /* macro */
62# else
63 else if (g_pfnrtKeQueryInterruptTime)
64 InterruptTime.QuadPart = g_pfnrtKeQueryInterruptTime();
65 else
66 {
67 /* NT4 (no API) and pre-init fallback. */
68 do
69 {
70 InterruptTime.HighPart = ((KUSER_SHARED_DATA volatile *)SharedUserData)->InterruptTime.High1Time;
71 InterruptTime.LowPart = ((KUSER_SHARED_DATA volatile *)SharedUserData)->InterruptTime.LowPart;
72 } while (((KUSER_SHARED_DATA volatile *)SharedUserData)->InterruptTime.High2Time != InterruptTime.HighPart);
73 }
74# endif
75 return (uint64_t)InterruptTime.QuadPart * 100;
76#else
77 /* Tick Count (NT4 SP1 has these APIs, haven't got SP0 to check). */
78 LARGE_INTEGER Tick;
79 KeQueryTickCount(&Tick);
80 return (uint64_t)Tick.QuadPart * KeQueryTimeIncrement() * 100;
81#endif
82}
83
84
85RTDECL(uint64_t) RTTimeNanoTS(void)
86{
87 return rtTimeGetSystemNanoTS();
88}
89
90
91RTDECL(uint64_t) RTTimeMilliTS(void)
92{
93 return rtTimeGetSystemNanoTS() / RT_NS_1MS;
94}
95
96
97RTDECL(uint64_t) RTTimeSystemNanoTS(void)
98{
99 return rtTimeGetSystemNanoTS();
100}
101
102
103RTDECL(uint64_t) RTTimeSystemMilliTS(void)
104{
105 return rtTimeGetSystemNanoTS() / RT_NS_1MS;
106}
107
108
109RTDECL(PRTTIMESPEC) RTTimeNow(PRTTIMESPEC pTime)
110{
111 LARGE_INTEGER SystemTime;
112 if (g_pfnrtKeQuerySystemTimePrecise)
113 g_pfnrtKeQuerySystemTimePrecise(&SystemTime);
114#ifdef RT_ARCH_AMD64
115 else
116 KeQuerySystemTime(&SystemTime); /* macro */
117#else
118 else if (g_pfnrtKeQuerySystemTime)
119 g_pfnrtKeQuerySystemTime(&SystemTime);
120 else
121 {
122 do
123 {
124 SystemTime.HighPart = ((KUSER_SHARED_DATA volatile *)SharedUserData)->SystemTime.High1Time;
125 SystemTime.LowPart = ((KUSER_SHARED_DATA volatile *)SharedUserData)->SystemTime.LowPart;
126 } while (((KUSER_SHARED_DATA volatile *)SharedUserData)->SystemTime.High2Time != SystemTime.HighPart);
127 }
128#endif
129 return RTTimeSpecSetNtTime(pTime, SystemTime.QuadPart);
130}
131
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette