VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/time/timesup.cpp@ 25721

Last change on this file since 25721 was 21337, checked in by vboxsync, 16 years ago

IPRT,HostDrv,AddDrv: Export public IPRT symbols for the linux kernel (pain).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 8.0 KB
Line 
1/* $Id: timesup.cpp 21337 2009-07-07 14:58:27Z vboxsync $ */
2/** @file
3 * IPRT - Time using SUPLib.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#define LOG_GROUP RTLOGGROUP_TIME
36#include <iprt/time.h>
37#include "internal/iprt.h"
38
39#include <iprt/asm.h>
40#include <iprt/assert.h>
41#include <iprt/err.h>
42#include <iprt/log.h>
43#ifndef IN_GUEST
44# include <VBox/sup.h>
45# include <VBox/x86.h>
46#endif
47#include "internal/time.h"
48
49
50/*******************************************************************************
51* Internal Functions *
52*******************************************************************************/
53#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
54static DECLCALLBACK(void) rtTimeNanoTSInternalBitch(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS);
55static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalFallback(PRTTIMENANOTSDATA pData);
56static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalRediscover(PRTTIMENANOTSDATA pData);
57#endif
58
59
60/*******************************************************************************
61* Global Variables *
62*******************************************************************************/
63#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
64/** The previous timestamp value returned by RTTimeNanoTS. */
65static uint64_t g_TimeNanoTSPrev = 0;
66
67/** The RTTimeNanoTS data structure that's passed down to the worker functions. */
68static RTTIMENANOTSDATA g_TimeNanoTSData =
69{
70 /* .pu64Prev = */ &g_TimeNanoTSPrev,
71 /* .pfnBad = */ rtTimeNanoTSInternalBitch,
72 /* .pfnRediscover = */ rtTimeNanoTSInternalRediscover,
73 /* .pvDummy = */ NULL,
74 /* .c1nsSteps = */ 0,
75 /* .cExpired = */ 0,
76 /* .cBadPrev = */ 0,
77 /* .cUpdateRaces = */ 0
78};
79
80/** The index into g_apfnWorkers for the function to use.
81 * This cannot be a pointer because that'll break down in GC due to code relocation. */
82static uint32_t g_iWorker = 0;
83/** Array of rtTimeNanoTSInternal worker functions.
84 * This array is indexed by g_iWorker. */
85static const PFNTIMENANOTSINTERNAL g_apfnWorkers[] =
86{
87#define RTTIMENANO_WORKER_DETECT 0
88 rtTimeNanoTSInternalRediscover,
89#define RTTIMENANO_WORKER_SYNC_CPUID 1
90 RTTimeNanoTSLegacySync,
91#define RTTIMENANO_WORKER_ASYNC_CPUID 2
92 RTTimeNanoTSLegacyAsync,
93#define RTTIMENANO_WORKER_SYNC_LFENCE 3
94 RTTimeNanoTSLFenceSync,
95#define RTTIMENANO_WORKER_ASYNC_LFENCE 4
96 RTTimeNanoTSLFenceAsync,
97#define RTTIMENANO_WORKER_FALLBACK 5
98 rtTimeNanoTSInternalFallback,
99};
100
101
102/**
103 * Helper function that's used by the assembly routines when something goes bust.
104 *
105 * @param pData Pointer to the data structure.
106 * @param u64NanoTS The calculated nano ts.
107 * @param u64DeltaPrev The delta relative to the previously returned timestamp.
108 * @param u64PrevNanoTS The previously returned timestamp (as it was read it).
109 */
110static DECLCALLBACK(void) rtTimeNanoTSInternalBitch(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS)
111{
112 pData->cBadPrev++;
113 if ((int64_t)u64DeltaPrev < 0)
114 LogRel(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64\n",
115 u64DeltaPrev, u64PrevNanoTS, u64NanoTS));
116 else
117 Log(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64 (debugging?)\n",
118 u64DeltaPrev, u64PrevNanoTS, u64NanoTS));
119}
120
121/**
122 * Fallback function.
123 */
124static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalFallback(PRTTIMENANOTSDATA pData)
125{
126 PSUPGLOBALINFOPAGE pGip = g_pSUPGlobalInfoPage;
127 if ( pGip
128 && pGip->u32Magic == SUPGLOBALINFOPAGE_MAGIC
129 && ( pGip->u32Mode == SUPGIPMODE_SYNC_TSC
130 || pGip->u32Mode == SUPGIPMODE_ASYNC_TSC))
131 return rtTimeNanoTSInternalRediscover(pData);
132 NOREF(pData);
133#if defined(IN_RING3) /** @todo Add ring-0 RTTimeSystemNanoTS to all hosts. */
134 return RTTimeSystemNanoTS();
135#else
136 return 0;
137#endif
138}
139
140
141/**
142 * Called the first time somebody asks for the time or when the GIP
143 * is mapped/unmapped.
144 */
145static DECLCALLBACK(uint64_t) rtTimeNanoTSInternalRediscover(PRTTIMENANOTSDATA pData)
146{
147 uint32_t iWorker;
148 PSUPGLOBALINFOPAGE pGip = g_pSUPGlobalInfoPage;
149 if ( pGip
150 && pGip->u32Magic == SUPGLOBALINFOPAGE_MAGIC
151 && ( pGip->u32Mode == SUPGIPMODE_SYNC_TSC
152 || pGip->u32Mode == SUPGIPMODE_ASYNC_TSC))
153 {
154 if (ASMCpuId_EDX(1) & X86_CPUID_FEATURE_EDX_SSE2)
155 iWorker = pGip->u32Mode == SUPGIPMODE_SYNC_TSC
156 ? RTTIMENANO_WORKER_SYNC_LFENCE
157 : RTTIMENANO_WORKER_ASYNC_LFENCE;
158 else
159 iWorker = pGip->u32Mode == SUPGIPMODE_SYNC_TSC
160 ? RTTIMENANO_WORKER_SYNC_CPUID
161 : RTTIMENANO_WORKER_ASYNC_CPUID;
162 }
163 else
164 iWorker = RTTIMENANO_WORKER_FALLBACK;
165
166 ASMAtomicXchgU32((uint32_t volatile *)&g_iWorker, iWorker);
167 return g_apfnWorkers[iWorker](pData);
168}
169
170#endif /* !IN_GUEST && !RT_NO_GIP */
171
172
173/**
174 * Internal worker for getting the current nanosecond timestamp.
175 */
176DECLINLINE(uint64_t) rtTimeNanoTSInternal(void)
177{
178#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
179 return g_apfnWorkers[g_iWorker](&g_TimeNanoTSData);
180#else
181 return RTTimeSystemNanoTS();
182#endif
183}
184
185
186/**
187 * Gets the current nanosecond timestamp.
188 *
189 * @returns nanosecond timestamp.
190 */
191RTDECL(uint64_t) RTTimeNanoTS(void)
192{
193 return rtTimeNanoTSInternal();
194}
195RT_EXPORT_SYMBOL(RTTimeNanoTS);
196
197
198/**
199 * Gets the current millisecond timestamp.
200 *
201 * @returns millisecond timestamp.
202 */
203RTDECL(uint64_t) RTTimeMilliTS(void)
204{
205 return rtTimeNanoTSInternal() / 1000000;
206}
207RT_EXPORT_SYMBOL(RTTimeMilliTS);
208
209
210#if !defined(IN_GUEST) && !defined(RT_NO_GIP)
211/**
212 * Debugging the time api.
213 *
214 * @returns the number of 1ns steps which has been applied by RTTimeNanoTS().
215 */
216RTDECL(uint32_t) RTTimeDbgSteps(void)
217{
218 return g_TimeNanoTSData.c1nsSteps;
219}
220RT_EXPORT_SYMBOL(RTTimeDbgSteps);
221
222
223/**
224 * Debugging the time api.
225 *
226 * @returns the number of times the TSC interval expired RTTimeNanoTS().
227 */
228RTDECL(uint32_t) RTTimeDbgExpired(void)
229{
230 return g_TimeNanoTSData.cExpired;
231}
232RT_EXPORT_SYMBOL(RTTimeDbgExpired);
233
234
235/**
236 * Debugging the time api.
237 *
238 * @returns the number of bad previous values encountered by RTTimeNanoTS().
239 */
240RTDECL(uint32_t) RTTimeDbgBad(void)
241{
242 return g_TimeNanoTSData.cBadPrev;
243}
244RT_EXPORT_SYMBOL(RTTimeDbgBad);
245
246
247/**
248 * Debugging the time api.
249 *
250 * @returns the number of update races in RTTimeNanoTS().
251 */
252RTDECL(uint32_t) RTTimeDbgRaces(void)
253{
254 return g_TimeNanoTSData.cUpdateRaces;
255}
256RT_EXPORT_SYMBOL(RTTimeDbgRaces);
257#endif /* !IN_GUEST && !RT_NO_GIP */
258
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