VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/TMAllVirtual.cpp@ 40138

Last change on this file since 40138 was 39402, checked in by vboxsync, 13 years ago

VMM: don't use generic IPE status codes, use specific ones. Part 1.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 38.0 KB
Line 
1/* $Id: TMAllVirtual.cpp 39402 2011-11-23 16:25:04Z vboxsync $ */
2/** @file
3 * TM - Timeout Manager, Virtual Time, All Contexts.
4 */
5
6/*
7 * Copyright (C) 2006-2007 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
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_TM
23#include <VBox/vmm/tm.h>
24#include <VBox/vmm/dbgftrace.h>
25#ifdef IN_RING3
26# include <VBox/vmm/rem.h>
27# include <iprt/thread.h>
28#endif
29#include "TMInternal.h"
30#include <VBox/vmm/vm.h>
31#include <VBox/vmm/vmm.h>
32#include <VBox/err.h>
33#include <VBox/log.h>
34#include <VBox/sup.h>
35
36#include <iprt/time.h>
37#include <iprt/assert.h>
38#include <iprt/asm.h>
39#include <iprt/asm-math.h>
40
41
42
43/**
44 * Helper function that's used by the assembly routines when something goes bust.
45 *
46 * @param pData Pointer to the data structure.
47 * @param u64NanoTS The calculated nano ts.
48 * @param u64DeltaPrev The delta relative to the previously returned timestamp.
49 * @param u64PrevNanoTS The previously returned timestamp (as it was read it).
50 */
51DECLEXPORT(void) tmVirtualNanoTSBad(PRTTIMENANOTSDATA pData, uint64_t u64NanoTS, uint64_t u64DeltaPrev, uint64_t u64PrevNanoTS)
52{
53 //PVM pVM = (PVM)((uint8_t *)pData - RT_OFFSETOF(VM, CTXALLSUFF(s.tm.VirtualGetRawData)));
54 pData->cBadPrev++;
55 if ((int64_t)u64DeltaPrev < 0)
56 LogRel(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64\n",
57 u64DeltaPrev, u64PrevNanoTS, u64NanoTS));
58 else
59 Log(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64 (debugging?)\n",
60 u64DeltaPrev, u64PrevNanoTS, u64NanoTS));
61}
62
63
64/**
65 * Called the first time somebody asks for the time or when the GIP
66 * is mapped/unmapped.
67 *
68 * This should never ever happen.
69 */
70DECLEXPORT(uint64_t) tmVirtualNanoTSRediscover(PRTTIMENANOTSDATA pData)
71{
72 NOREF(pData);
73 //PVM pVM = (PVM)((uint8_t *)pData - RT_OFFSETOF(VM, CTXALLSUFF(s.tm.VirtualGetRawData)));
74 PSUPGLOBALINFOPAGE pGip = g_pSUPGlobalInfoPage;
75 AssertFatalMsgFailed(("pGip=%p u32Magic=%#x\n", pGip, VALID_PTR(pGip) ? pGip->u32Magic : 0));
76#ifndef _MSC_VER
77 return 0; /* gcc false positive warning */
78#endif
79}
80
81
82#if 1
83
84/**
85 * Wrapper around the IPRT GIP time methods.
86 */
87DECLINLINE(uint64_t) tmVirtualGetRawNanoTS(PVM pVM)
88{
89# ifdef IN_RING3
90 uint64_t u64 = CTXALLSUFF(pVM->tm.s.pfnVirtualGetRaw)(&CTXALLSUFF(pVM->tm.s.VirtualGetRawData));
91# else /* !IN_RING3 */
92 uint32_t cPrevSteps = pVM->tm.s.CTX_SUFF(VirtualGetRawData).c1nsSteps;
93 uint64_t u64 = pVM->tm.s.CTX_SUFF(pfnVirtualGetRaw)(&pVM->tm.s.CTX_SUFF(VirtualGetRawData));
94 if (cPrevSteps != pVM->tm.s.CTX_SUFF(VirtualGetRawData).c1nsSteps)
95 VMCPU_FF_SET(VMMGetCpu(pVM), VMCPU_FF_TO_R3);
96# endif /* !IN_RING3 */
97 /*DBGFTRACE_POS_U64(pVM, u64);*/
98 return u64;
99}
100
101#else
102
103/**
104 * This is (mostly) the same as rtTimeNanoTSInternal() except
105 * for the two globals which live in TM.
106 *
107 * @returns Nanosecond timestamp.
108 * @param pVM The VM handle.
109 */
110static uint64_t tmVirtualGetRawNanoTS(PVM pVM)
111{
112 uint64_t u64Delta;
113 uint32_t u32NanoTSFactor0;
114 uint64_t u64TSC;
115 uint64_t u64NanoTS;
116 uint32_t u32UpdateIntervalTSC;
117 uint64_t u64PrevNanoTS;
118
119 /*
120 * Read the GIP data and the previous value.
121 */
122 for (;;)
123 {
124 uint32_t u32TransactionId;
125 PSUPGLOBALINFOPAGE pGip = g_pSUPGlobalInfoPage;
126#ifdef IN_RING3
127 if (RT_UNLIKELY(!pGip || pGip->u32Magic != SUPGLOBALINFOPAGE_MAGIC))
128 return RTTimeSystemNanoTS();
129#endif
130
131 if (pGip->u32Mode != SUPGIPMODE_ASYNC_TSC)
132 {
133 u32TransactionId = pGip->aCPUs[0].u32TransactionId;
134#ifdef RT_OS_L4
135 Assert((u32TransactionId & 1) == 0);
136#endif
137 u32UpdateIntervalTSC = pGip->aCPUs[0].u32UpdateIntervalTSC;
138 u64NanoTS = pGip->aCPUs[0].u64NanoTS;
139 u64TSC = pGip->aCPUs[0].u64TSC;
140 u32NanoTSFactor0 = pGip->u32UpdateIntervalNS;
141 u64Delta = ASMReadTSC();
142 u64PrevNanoTS = ASMAtomicReadU64(&pVM->tm.s.u64VirtualRawPrev);
143 if (RT_UNLIKELY( pGip->aCPUs[0].u32TransactionId != u32TransactionId
144 || (u32TransactionId & 1)))
145 continue;
146 }
147 else
148 {
149 /* SUPGIPMODE_ASYNC_TSC */
150 PSUPGIPCPU pGipCpu;
151
152 uint8_t u8ApicId = ASMGetApicId();
153 if (RT_LIKELY(u8ApicId < RT_ELEMENTS(pGip->aCPUs)))
154 pGipCpu = &pGip->aCPUs[u8ApicId];
155 else
156 {
157 AssertMsgFailed(("%x\n", u8ApicId));
158 pGipCpu = &pGip->aCPUs[0];
159 }
160
161 u32TransactionId = pGipCpu->u32TransactionId;
162#ifdef RT_OS_L4
163 Assert((u32TransactionId & 1) == 0);
164#endif
165 u32UpdateIntervalTSC = pGipCpu->u32UpdateIntervalTSC;
166 u64NanoTS = pGipCpu->u64NanoTS;
167 u64TSC = pGipCpu->u64TSC;
168 u32NanoTSFactor0 = pGip->u32UpdateIntervalNS;
169 u64Delta = ASMReadTSC();
170 u64PrevNanoTS = ASMAtomicReadU64(&pVM->tm.s.u64VirtualRawPrev);
171#ifdef IN_RC
172 Assert(!(ASMGetFlags() & X86_EFL_IF));
173#else
174 if (RT_UNLIKELY(u8ApicId != ASMGetApicId()))
175 continue;
176 if (RT_UNLIKELY( pGipCpu->u32TransactionId != u32TransactionId
177 || (u32TransactionId & 1)))
178 continue;
179#endif
180 }
181 break;
182 }
183
184 /*
185 * Calc NanoTS delta.
186 */
187 u64Delta -= u64TSC;
188 if (u64Delta > u32UpdateIntervalTSC)
189 {
190 /*
191 * We've expired the interval, cap it. If we're here for the 2nd
192 * time without any GIP update in-between, the checks against
193 * pVM->tm.s.u64VirtualRawPrev below will force 1ns stepping.
194 */
195 u64Delta = u32UpdateIntervalTSC;
196 }
197#if !defined(_MSC_VER) || defined(RT_ARCH_AMD64) /* GCC makes very pretty code from these two inline calls, while MSC cannot. */
198 u64Delta = ASMMult2xU32RetU64((uint32_t)u64Delta, u32NanoTSFactor0);
199 u64Delta = ASMDivU64ByU32RetU32(u64Delta, u32UpdateIntervalTSC);
200#else
201 __asm
202 {
203 mov eax, dword ptr [u64Delta]
204 mul dword ptr [u32NanoTSFactor0]
205 div dword ptr [u32UpdateIntervalTSC]
206 mov dword ptr [u64Delta], eax
207 xor edx, edx
208 mov dword ptr [u64Delta + 4], edx
209 }
210#endif
211
212 /*
213 * Calculate the time and compare it with the previously returned value.
214 *
215 * Since this function is called *very* frequently when the VM is running
216 * and then mostly on EMT, we can restrict the valid range of the delta
217 * (-1s to 2*GipUpdates) and simplify/optimize the default path.
218 */
219 u64NanoTS += u64Delta;
220 uint64_t u64DeltaPrev = u64NanoTS - u64PrevNanoTS;
221 if (RT_LIKELY(u64DeltaPrev < 1000000000 /* 1s */))
222 /* frequent - less than 1s since last call. */;
223 else if ( (int64_t)u64DeltaPrev < 0
224 && (int64_t)u64DeltaPrev + u32NanoTSFactor0 * 2 > 0)
225 {
226 /* occasional - u64NanoTS is in the 'past' relative to previous returns. */
227 ASMAtomicIncU32(&pVM->tm.s.CTX_SUFF(VirtualGetRawData).c1nsSteps);
228 u64NanoTS = u64PrevNanoTS + 1;
229#ifndef IN_RING3
230 VM_FF_SET(pVM, VM_FF_TO_R3); /* S10 hack */
231#endif
232 }
233 else if (u64PrevNanoTS)
234 {
235 /* Something has gone bust, if negative offset it's real bad. */
236 ASMAtomicIncU32(&pVM->tm.s.CTX_SUFF(VirtualGetRawData).cBadPrev);
237 if ((int64_t)u64DeltaPrev < 0)
238 LogRel(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64 u64Delta=%#RX64\n",
239 u64DeltaPrev, u64PrevNanoTS, u64NanoTS, u64Delta));
240 else
241 Log(("TM: u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64 u64Delta=%#RX64 (debugging?)\n",
242 u64DeltaPrev, u64PrevNanoTS, u64NanoTS, u64Delta));
243#ifdef DEBUG_bird
244 /** @todo there are some hickups during boot and reset that can cause 2-5 seconds delays. Investigate... */
245 AssertMsg(u64PrevNanoTS > UINT64_C(100000000000) /* 100s */,
246 ("u64DeltaPrev=%RI64 u64PrevNanoTS=0x%016RX64 u64NanoTS=0x%016RX64 u64Delta=%#RX64\n",
247 u64DeltaPrev, u64PrevNanoTS, u64NanoTS, u64Delta));
248#endif
249 }
250 /* else: We're resuming (see TMVirtualResume). */
251 if (RT_LIKELY(ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualRawPrev, u64NanoTS, u64PrevNanoTS)))
252 return u64NanoTS;
253
254 /*
255 * Attempt updating the previous value, provided we're still ahead of it.
256 *
257 * There is no point in recalculating u64NanoTS because we got preempted or if
258 * we raced somebody while the GIP was updated, since these are events
259 * that might occur at any point in the return path as well.
260 */
261 for (int cTries = 50;;)
262 {
263 u64PrevNanoTS = ASMAtomicReadU64(&pVM->tm.s.u64VirtualRawPrev);
264 if (u64PrevNanoTS >= u64NanoTS)
265 break;
266 if (ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualRawPrev, u64NanoTS, u64PrevNanoTS))
267 break;
268 AssertBreak(--cTries <= 0);
269 if (cTries < 25 && !VM_IS_EMT(pVM)) /* give up early */
270 break;
271 }
272
273 return u64NanoTS;
274}
275
276#endif
277
278
279/**
280 * Get the time when we're not running at 100%
281 *
282 * @returns The timestamp.
283 * @param pVM The VM handle.
284 */
285static uint64_t tmVirtualGetRawNonNormal(PVM pVM)
286{
287 /*
288 * Recalculate the RTTimeNanoTS() value for the period where
289 * warp drive has been enabled.
290 */
291 uint64_t u64 = tmVirtualGetRawNanoTS(pVM);
292 u64 -= pVM->tm.s.u64VirtualWarpDriveStart;
293 u64 *= pVM->tm.s.u32VirtualWarpDrivePercentage;
294 u64 /= 100;
295 u64 += pVM->tm.s.u64VirtualWarpDriveStart;
296
297 /*
298 * Now we apply the virtual time offset.
299 * (Which is the negated tmVirtualGetRawNanoTS() value for when the virtual
300 * machine started if it had been running continuously without any suspends.)
301 */
302 u64 -= pVM->tm.s.u64VirtualOffset;
303 return u64;
304}
305
306
307/**
308 * Get the raw virtual time.
309 *
310 * @returns The current time stamp.
311 * @param pVM The VM handle.
312 */
313DECLINLINE(uint64_t) tmVirtualGetRaw(PVM pVM)
314{
315 if (RT_LIKELY(!pVM->tm.s.fVirtualWarpDrive))
316 return tmVirtualGetRawNanoTS(pVM) - pVM->tm.s.u64VirtualOffset;
317 return tmVirtualGetRawNonNormal(pVM);
318}
319
320
321/**
322 * Inlined version of tmVirtualGetEx.
323 */
324DECLINLINE(uint64_t) tmVirtualGet(PVM pVM, bool fCheckTimers)
325{
326 uint64_t u64;
327 if (RT_LIKELY(pVM->tm.s.cVirtualTicking))
328 {
329 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualGet);
330 u64 = tmVirtualGetRaw(pVM);
331
332 /*
333 * Use the chance to check for expired timers.
334 */
335 if (fCheckTimers)
336 {
337 PVMCPU pVCpuDst = &pVM->aCpus[pVM->tm.s.idTimerCpu];
338 if ( !VMCPU_FF_ISSET(pVCpuDst, VMCPU_FF_TIMER)
339 && !pVM->tm.s.fRunningQueues
340 && ( pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL].u64Expire <= u64
341 || ( pVM->tm.s.fVirtualSyncTicking
342 && pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire <= u64 - pVM->tm.s.offVirtualSync
343 )
344 )
345 && !pVM->tm.s.fRunningQueues
346 )
347 {
348 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualGetSetFF);
349 Log5(("TMAllVirtual(%u): FF: %d -> 1\n", __LINE__, VMCPU_FF_ISPENDING(pVCpuDst, VMCPU_FF_TIMER)));
350 VMCPU_FF_SET(pVCpuDst, VMCPU_FF_TIMER);
351#ifdef IN_RING3
352 REMR3NotifyTimerPending(pVM, pVCpuDst);
353 VMR3NotifyCpuFFU(pVCpuDst->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM);
354#endif
355 }
356 }
357 }
358 else
359 u64 = pVM->tm.s.u64Virtual;
360 return u64;
361}
362
363
364/**
365 * Gets the current TMCLOCK_VIRTUAL time
366 *
367 * @returns The timestamp.
368 * @param pVM VM handle.
369 *
370 * @remark While the flow of time will never go backwards, the speed of the
371 * progress varies due to inaccurate RTTimeNanoTS and TSC. The latter can be
372 * influenced by power saving (SpeedStep, PowerNow!), while the former
373 * makes use of TSC and kernel timers.
374 */
375VMM_INT_DECL(uint64_t) TMVirtualGet(PVM pVM)
376{
377 return tmVirtualGet(pVM, true /*fCheckTimers*/);
378}
379
380
381/**
382 * Gets the current TMCLOCK_VIRTUAL time without checking
383 * timers or anything.
384 *
385 * Meaning, this has no side effect on FFs like TMVirtualGet may have.
386 *
387 * @returns The timestamp.
388 * @param pVM VM handle.
389 *
390 * @remarks See TMVirtualGet.
391 */
392VMM_INT_DECL(uint64_t) TMVirtualGetNoCheck(PVM pVM)
393{
394 return tmVirtualGet(pVM, false /*fCheckTimers*/);
395}
396
397
398/**
399 * Converts the dead line interval from TMCLOCK_VIRTUAL to host nano seconds.
400 *
401 * @returns Host nano second count.
402 * @param pVM The VM handle.
403 * @param cVirtTicksToDeadline The TMCLOCK_VIRTUAL interval.
404 */
405DECLINLINE(uint64_t) tmVirtualVirtToNsDeadline(PVM pVM, uint64_t cVirtTicksToDeadline)
406{
407 if (RT_UNLIKELY(pVM->tm.s.fVirtualWarpDrive))
408 return ASMMultU64ByU32DivByU32(cVirtTicksToDeadline, 100, pVM->tm.s.u32VirtualWarpDrivePercentage);
409 return cVirtTicksToDeadline;
410}
411
412
413/**
414 * tmVirtualSyncGetLocked worker for handling catch-up when owning the lock.
415 *
416 * @returns The timestamp.
417 * @param pVM VM handle.
418 * @param u64 raw virtual time.
419 * @param off offVirtualSync.
420 * @param pcNsToDeadline Where to return the number of nano seconds to
421 * the next virtual sync timer deadline. Can be
422 * NULL.
423 */
424DECLINLINE(uint64_t) tmVirtualSyncGetHandleCatchUpLocked(PVM pVM, uint64_t u64, uint64_t off, uint64_t *pcNsToDeadline)
425{
426 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetLocked);
427
428 /*
429 * Don't make updates until we've check the timer queue.
430 */
431 bool fUpdatePrev = true;
432 bool fUpdateOff = true;
433 bool fStop = false;
434 const uint64_t u64Prev = pVM->tm.s.u64VirtualSyncCatchUpPrev;
435 uint64_t u64Delta = u64 - u64Prev;
436 if (RT_LIKELY(!(u64Delta >> 32)))
437 {
438 uint64_t u64Sub = ASMMultU64ByU32DivByU32(u64Delta, pVM->tm.s.u32VirtualSyncCatchUpPercentage, 100);
439 if (off > u64Sub + pVM->tm.s.offVirtualSyncGivenUp)
440 {
441 off -= u64Sub;
442 Log4(("TM: %'RU64/-%'8RU64: sub %RU32 [vsghcul]\n", u64 - off, off - pVM->tm.s.offVirtualSyncGivenUp, u64Sub));
443 }
444 else
445 {
446 /* we've completely caught up. */
447 STAM_PROFILE_ADV_STOP(&pVM->tm.s.StatVirtualSyncCatchup, c);
448 off = pVM->tm.s.offVirtualSyncGivenUp;
449 fStop = true;
450 Log4(("TM: %'RU64/0: caught up [vsghcul]\n", u64));
451 }
452 }
453 else
454 {
455 /* More than 4 seconds since last time (or negative), ignore it. */
456 fUpdateOff = false;
457 fUpdatePrev = !(u64Delta & RT_BIT_64(63));
458 Log(("TMVirtualGetSync: u64Delta=%RX64\n", u64Delta));
459 }
460
461 /*
462 * Complete the calculation of the current TMCLOCK_VIRTUAL_SYNC time. The current
463 * approach is to never pass the head timer. So, when we do stop the clock and
464 * set the timer pending flag.
465 */
466 u64 -= off;
467
468 uint64_t u64Last = ASMAtomicUoReadU64(&pVM->tm.s.u64VirtualSync);
469 if (u64Last > u64)
470 {
471 u64 = u64Last + 1;
472 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetAdjLast);
473 }
474
475 uint64_t u64Expire = ASMAtomicReadU64(&pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire);
476 if (u64 < u64Expire)
477 {
478 ASMAtomicWriteU64(&pVM->tm.s.u64VirtualSync, u64);
479 if (fUpdateOff)
480 ASMAtomicWriteU64(&pVM->tm.s.offVirtualSync, off);
481 if (fStop)
482 ASMAtomicWriteBool(&pVM->tm.s.fVirtualSyncCatchUp, false);
483 if (fUpdatePrev)
484 ASMAtomicWriteU64(&pVM->tm.s.u64VirtualSyncCatchUpPrev, u64);
485 if (pcNsToDeadline)
486 {
487 uint64_t cNsToDeadline = u64Expire - u64;
488 if (pVM->tm.s.fVirtualSyncCatchUp)
489 cNsToDeadline = ASMMultU64ByU32DivByU32(cNsToDeadline, 100,
490 pVM->tm.s.u32VirtualSyncCatchUpPercentage + 100);
491 *pcNsToDeadline = tmVirtualVirtToNsDeadline(pVM, cNsToDeadline);
492 }
493 PDMCritSectLeave(&pVM->tm.s.VirtualSyncLock);
494 }
495 else
496 {
497 u64 = u64Expire;
498 ASMAtomicWriteU64(&pVM->tm.s.u64VirtualSync, u64);
499 ASMAtomicWriteBool(&pVM->tm.s.fVirtualSyncTicking, false);
500
501 VM_FF_SET(pVM, VM_FF_TM_VIRTUAL_SYNC);
502 PVMCPU pVCpuDst = &pVM->aCpus[pVM->tm.s.idTimerCpu];
503 VMCPU_FF_SET(pVCpuDst, VMCPU_FF_TIMER);
504 Log5(("TMAllVirtual(%u): FF: %d -> 1\n", __LINE__, VMCPU_FF_ISPENDING(pVCpuDst, VMCPU_FF_TIMER)));
505 Log4(("TM: %'RU64/-%'8RU64: exp tmr=>ff [vsghcul]\n", u64, pVM->tm.s.offVirtualSync - pVM->tm.s.offVirtualSyncGivenUp));
506 PDMCritSectLeave(&pVM->tm.s.VirtualSyncLock);
507
508 if (pcNsToDeadline)
509 *pcNsToDeadline = 0;
510#ifdef IN_RING3
511 REMR3NotifyTimerPending(pVM, pVCpuDst);
512 VMR3NotifyCpuFFU(pVCpuDst->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM);
513#endif
514 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetSetFF);
515 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetExpired);
516 }
517 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetLocked);
518
519 Log6(("tmVirtualSyncGetHandleCatchUpLocked -> %'RU64\n", u64));
520 DBGFTRACE_U64_TAG(pVM, u64, "tmVirtualSyncGetHandleCatchUpLocked");
521 return u64;
522}
523
524
525/**
526 * tmVirtualSyncGetEx worker for when we get the lock.
527 *
528 * @returns timesamp.
529 * @param pVM The VM handle.
530 * @param u64 The virtual clock timestamp.
531 * @param pcNsToDeadline Where to return the number of nano seconds to
532 * the next virtual sync timer deadline. Can be
533 * NULL.
534 */
535DECLINLINE(uint64_t) tmVirtualSyncGetLocked(PVM pVM, uint64_t u64, uint64_t *pcNsToDeadline)
536{
537 /*
538 * Not ticking?
539 */
540 if (!pVM->tm.s.fVirtualSyncTicking)
541 {
542 u64 = ASMAtomicUoReadU64(&pVM->tm.s.u64VirtualSync);
543 PDMCritSectLeave(&pVM->tm.s.VirtualSyncLock);
544 if (pcNsToDeadline)
545 *pcNsToDeadline = 0;
546 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetLocked);
547 Log6(("tmVirtualSyncGetLocked -> %'RU64 [stopped]\n", u64));
548 DBGFTRACE_U64_TAG(pVM, u64, "tmVirtualSyncGetLocked-stopped");
549 return u64;
550 }
551
552 /*
553 * Handle catch up in a separate function.
554 */
555 uint64_t off = ASMAtomicUoReadU64(&pVM->tm.s.offVirtualSync);
556 if (ASMAtomicUoReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
557 return tmVirtualSyncGetHandleCatchUpLocked(pVM, u64, off, pcNsToDeadline);
558
559 /*
560 * Complete the calculation of the current TMCLOCK_VIRTUAL_SYNC time. The current
561 * approach is to never pass the head timer. So, when we do stop the clock and
562 * set the timer pending flag.
563 */
564 u64 -= off;
565
566 uint64_t u64Last = ASMAtomicUoReadU64(&pVM->tm.s.u64VirtualSync);
567 if (u64Last > u64)
568 {
569 u64 = u64Last + 1;
570 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetAdjLast);
571 }
572
573 uint64_t u64Expire = ASMAtomicReadU64(&pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire);
574 if (u64 < u64Expire)
575 {
576 ASMAtomicWriteU64(&pVM->tm.s.u64VirtualSync, u64);
577 PDMCritSectLeave(&pVM->tm.s.VirtualSyncLock);
578 if (pcNsToDeadline)
579 *pcNsToDeadline = tmVirtualVirtToNsDeadline(pVM, u64Expire - u64);
580 }
581 else
582 {
583 u64 = u64Expire;
584 ASMAtomicWriteU64(&pVM->tm.s.u64VirtualSync, u64);
585 ASMAtomicWriteBool(&pVM->tm.s.fVirtualSyncTicking, false);
586
587 VM_FF_SET(pVM, VM_FF_TM_VIRTUAL_SYNC);
588 PVMCPU pVCpuDst = &pVM->aCpus[pVM->tm.s.idTimerCpu];
589 VMCPU_FF_SET(pVCpuDst, VMCPU_FF_TIMER);
590 Log5(("TMAllVirtual(%u): FF: %d -> 1\n", __LINE__, !!VMCPU_FF_ISPENDING(pVCpuDst, VMCPU_FF_TIMER)));
591 Log4(("TM: %'RU64/-%'8RU64: exp tmr=>ff [vsgl]\n", u64, pVM->tm.s.offVirtualSync - pVM->tm.s.offVirtualSyncGivenUp));
592 PDMCritSectLeave(&pVM->tm.s.VirtualSyncLock);
593
594#ifdef IN_RING3
595 REMR3NotifyTimerPending(pVM, pVCpuDst);
596 VMR3NotifyCpuFFU(pVCpuDst->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM);
597#endif
598 if (pcNsToDeadline)
599 *pcNsToDeadline = 0;
600 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetSetFF);
601 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetExpired);
602 }
603 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetLocked);
604 Log6(("tmVirtualSyncGetLocked -> %'RU64\n", u64));
605 DBGFTRACE_U64_TAG(pVM, u64, "tmVirtualSyncGetLocked");
606 return u64;
607}
608
609
610/**
611 * Gets the current TMCLOCK_VIRTUAL_SYNC time.
612 *
613 * @returns The timestamp.
614 * @param pVM VM handle.
615 * @param fCheckTimers Check timers or not
616 * @param pcNsToDeadline Where to return the number of nano seconds to
617 * the next virtual sync timer deadline. Can be
618 * NULL.
619 * @thread EMT.
620 */
621DECLINLINE(uint64_t) tmVirtualSyncGetEx(PVM pVM, bool fCheckTimers, uint64_t *pcNsToDeadline)
622{
623 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGet);
624
625 uint64_t u64;
626 if (!pVM->tm.s.fVirtualSyncTicking)
627 {
628 if (pcNsToDeadline)
629 *pcNsToDeadline = 0;
630 u64 = pVM->tm.s.u64VirtualSync;
631 DBGFTRACE_U64_TAG(pVM, u64, "tmVirtualSyncGetEx-stopped1");
632 return u64;
633 }
634
635 /*
636 * Query the virtual clock and do the usual expired timer check.
637 */
638 Assert(pVM->tm.s.cVirtualTicking);
639 u64 = tmVirtualGetRaw(pVM);
640 if (fCheckTimers)
641 {
642 PVMCPU pVCpuDst = &pVM->aCpus[pVM->tm.s.idTimerCpu];
643 if ( !VMCPU_FF_ISSET(pVCpuDst, VMCPU_FF_TIMER)
644 && pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL].u64Expire <= u64)
645 {
646 Log5(("TMAllVirtual(%u): FF: 0 -> 1\n", __LINE__));
647 VMCPU_FF_SET(pVCpuDst, VMCPU_FF_TIMER);
648#ifdef IN_RING3
649 REMR3NotifyTimerPending(pVM, pVCpuDst);
650 VMR3NotifyCpuFFU(pVCpuDst->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM /** @todo |VMNOTIFYFF_FLAGS_POKE*/);
651#endif
652 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetSetFF);
653 }
654 }
655
656 /*
657 * If we can get the lock, get it. The result is much more reliable.
658 *
659 * Note! This is where all clock source devices branch off because they
660 * will be owning the lock already. The 'else' is taken by code
661 * which is less picky or hasn't been adjusted yet
662 */
663 if (PDMCritSectTryEnter(&pVM->tm.s.VirtualSyncLock) == VINF_SUCCESS)
664 return tmVirtualSyncGetLocked(pVM, u64, pcNsToDeadline);
665
666 /*
667 * When the clock is ticking, not doing catch ups and not running into an
668 * expired time, we can get away without locking. Try this first.
669 */
670 uint64_t off;
671 if (ASMAtomicUoReadBool(&pVM->tm.s.fVirtualSyncTicking))
672 {
673 if (!ASMAtomicUoReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
674 {
675 off = ASMAtomicReadU64(&pVM->tm.s.offVirtualSync);
676 if (RT_LIKELY( ASMAtomicUoReadBool(&pVM->tm.s.fVirtualSyncTicking)
677 && !ASMAtomicUoReadBool(&pVM->tm.s.fVirtualSyncCatchUp)
678 && off == ASMAtomicReadU64(&pVM->tm.s.offVirtualSync)))
679 {
680 off = u64 - off;
681 uint64_t const u64Expire = ASMAtomicReadU64(&pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire);
682 if (off < u64Expire)
683 {
684 if (pcNsToDeadline)
685 *pcNsToDeadline = tmVirtualVirtToNsDeadline(pVM, u64Expire - off);
686 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetLockless);
687 Log6(("tmVirtualSyncGetEx -> %'RU64 [lockless]\n", off));
688 DBGFTRACE_U64_TAG(pVM, off, "tmVirtualSyncGetEx-lockless");
689 return off;
690 }
691 }
692 }
693 }
694 else
695 {
696 off = ASMAtomicReadU64(&pVM->tm.s.u64VirtualSync);
697 if (RT_LIKELY(!ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncTicking)))
698 {
699 if (pcNsToDeadline)
700 *pcNsToDeadline = 0;
701 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetLockless);
702 Log6(("tmVirtualSyncGetEx -> %'RU64 [lockless/stopped]\n", off));
703 DBGFTRACE_U64_TAG(pVM, off, "tmVirtualSyncGetEx-stopped2");
704 return off;
705 }
706 }
707
708 /*
709 * Read the offset and adjust if we're playing catch-up.
710 *
711 * The catch-up adjusting work by us decrementing the offset by a percentage of
712 * the time elapsed since the previous TMVirtualGetSync call.
713 *
714 * It's possible to get a very long or even negative interval between two read
715 * for the following reasons:
716 * - Someone might have suspended the process execution, frequently the case when
717 * debugging the process.
718 * - We might be on a different CPU which TSC isn't quite in sync with the
719 * other CPUs in the system.
720 * - Another thread is racing us and we might have been preempted while inside
721 * this function.
722 *
723 * Assuming nano second virtual time, we can simply ignore any intervals which has
724 * any of the upper 32 bits set.
725 */
726 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
727 int cOuterTries = 42;
728 for (;; cOuterTries--)
729 {
730 /* Try grab the lock, things get simpler when owning the lock. */
731 int rcLock = PDMCritSectTryEnter(&pVM->tm.s.VirtualSyncLock);
732 if (RT_SUCCESS_NP(rcLock))
733 return tmVirtualSyncGetLocked(pVM, u64, pcNsToDeadline);
734
735 /* Re-check the ticking flag. */
736 if (!ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncTicking))
737 {
738 off = ASMAtomicReadU64(&pVM->tm.s.u64VirtualSync);
739 if ( ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncTicking)
740 && cOuterTries > 0)
741 continue;
742 if (pcNsToDeadline)
743 *pcNsToDeadline = 0;
744 Log6(("tmVirtualSyncGetEx -> %'RU64 [stopped]\n", off));
745 DBGFTRACE_U64_TAG(pVM, off, "tmVirtualSyncGetEx-stopped3");
746 return off;
747 }
748
749 off = ASMAtomicReadU64(&pVM->tm.s.offVirtualSync);
750 if (ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
751 {
752 /* No changes allowed, try get a consistent set of parameters. */
753 uint64_t const u64Prev = ASMAtomicReadU64(&pVM->tm.s.u64VirtualSyncCatchUpPrev);
754 uint64_t const offGivenUp = ASMAtomicReadU64(&pVM->tm.s.offVirtualSyncGivenUp);
755 uint32_t const u32Pct = ASMAtomicReadU32(&pVM->tm.s.u32VirtualSyncCatchUpPercentage);
756 if ( ( u64Prev == ASMAtomicReadU64(&pVM->tm.s.u64VirtualSyncCatchUpPrev)
757 && offGivenUp == ASMAtomicReadU64(&pVM->tm.s.offVirtualSyncGivenUp)
758 && u32Pct == ASMAtomicReadU32(&pVM->tm.s.u32VirtualSyncCatchUpPercentage)
759 && ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
760 || cOuterTries <= 0)
761 {
762 uint64_t u64Delta = u64 - u64Prev;
763 if (RT_LIKELY(!(u64Delta >> 32)))
764 {
765 uint64_t u64Sub = ASMMultU64ByU32DivByU32(u64Delta, u32Pct, 100);
766 if (off > u64Sub + offGivenUp)
767 {
768 off -= u64Sub;
769 Log4(("TM: %'RU64/-%'8RU64: sub %RU32 [NoLock]\n", u64 - off, pVM->tm.s.offVirtualSync - offGivenUp, u64Sub));
770 }
771 else
772 {
773 /* we've completely caught up. */
774 STAM_PROFILE_ADV_STOP(&pVM->tm.s.StatVirtualSyncCatchup, c);
775 off = offGivenUp;
776 Log4(("TM: %'RU64/0: caught up [NoLock]\n", u64));
777 }
778 }
779 else
780 /* More than 4 seconds since last time (or negative), ignore it. */
781 Log(("TMVirtualGetSync: u64Delta=%RX64 (NoLock)\n", u64Delta));
782
783 /* Check that we're still running and in catch up. */
784 if ( ASMAtomicUoReadBool(&pVM->tm.s.fVirtualSyncTicking)
785 && ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
786 break;
787 if (cOuterTries <= 0)
788 break; /* enough */
789 }
790 }
791 else if ( off == ASMAtomicReadU64(&pVM->tm.s.offVirtualSync)
792 && !ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
793 break; /* Got an consistent offset */
794 else if (cOuterTries <= 0)
795 break; /* enough */
796 }
797 if (cOuterTries <= 0)
798 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetELoop);
799
800 /*
801 * Complete the calculation of the current TMCLOCK_VIRTUAL_SYNC time. The current
802 * approach is to never pass the head timer. So, when we do stop the clock and
803 * set the timer pending flag.
804 */
805 u64 -= off;
806/** @todo u64VirtualSyncLast */
807 uint64_t u64Expire = ASMAtomicReadU64(&pVM->tm.s.CTX_SUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire);
808 if (u64 >= u64Expire)
809 {
810 PVMCPU pVCpuDst = &pVM->aCpus[pVM->tm.s.idTimerCpu];
811 if (!VMCPU_FF_ISSET(pVCpuDst, VMCPU_FF_TIMER))
812 {
813 Log5(("TMAllVirtual(%u): FF: %d -> 1 (NoLock)\n", __LINE__, VMCPU_FF_ISPENDING(pVCpuDst, VMCPU_FF_TIMER)));
814 VM_FF_SET(pVM, VM_FF_TM_VIRTUAL_SYNC); /* Hmm? */
815 VMCPU_FF_SET(pVCpuDst, VMCPU_FF_TIMER);
816#ifdef IN_RING3
817 REMR3NotifyTimerPending(pVM, pVCpuDst);
818 VMR3NotifyCpuFFU(pVCpuDst->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM);
819#endif
820 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetSetFF);
821 Log4(("TM: %'RU64/-%'8RU64: exp tmr=>ff [NoLock]\n", u64, pVM->tm.s.offVirtualSync - pVM->tm.s.offVirtualSyncGivenUp));
822 }
823 else
824 Log4(("TM: %'RU64/-%'8RU64: exp tmr [NoLock]\n", u64, pVM->tm.s.offVirtualSync - pVM->tm.s.offVirtualSyncGivenUp));
825 if (pcNsToDeadline)
826 *pcNsToDeadline = 0;
827 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualSyncGetExpired);
828 }
829 else if (pcNsToDeadline)
830 {
831 uint64_t cNsToDeadline = u64Expire - u64;
832 if (ASMAtomicReadBool(&pVM->tm.s.fVirtualSyncCatchUp))
833 cNsToDeadline = ASMMultU64ByU32DivByU32(cNsToDeadline, 100,
834 ASMAtomicReadU32(&pVM->tm.s.u32VirtualSyncCatchUpPercentage) + 100);
835 *pcNsToDeadline = tmVirtualVirtToNsDeadline(pVM, cNsToDeadline);
836 }
837
838 Log6(("tmVirtualSyncGetEx -> %'RU64\n", u64));
839 DBGFTRACE_U64_TAG(pVM, u64, "tmVirtualSyncGetEx-nolock");
840 return u64;
841}
842
843
844/**
845 * Gets the current TMCLOCK_VIRTUAL_SYNC time.
846 *
847 * @returns The timestamp.
848 * @param pVM VM handle.
849 * @thread EMT.
850 * @remarks May set the timer and virtual sync FFs.
851 */
852VMM_INT_DECL(uint64_t) TMVirtualSyncGet(PVM pVM)
853{
854 return tmVirtualSyncGetEx(pVM, true /*fCheckTimers*/, NULL /*pcNsToDeadline*/);
855}
856
857
858/**
859 * Gets the current TMCLOCK_VIRTUAL_SYNC time without checking timers running on
860 * TMCLOCK_VIRTUAL.
861 *
862 * @returns The timestamp.
863 * @param pVM VM handle.
864 * @thread EMT.
865 * @remarks May set the timer and virtual sync FFs.
866 */
867VMM_INT_DECL(uint64_t) TMVirtualSyncGetNoCheck(PVM pVM)
868{
869 return tmVirtualSyncGetEx(pVM, false /*fCheckTimers*/, NULL /*pcNsToDeadline*/);
870}
871
872
873/**
874 * Gets the current TMCLOCK_VIRTUAL_SYNC time.
875 *
876 * @returns The timestamp.
877 * @param pVM VM handle.
878 * @param fCheckTimers Check timers on the virtual clock or not.
879 * @thread EMT.
880 * @remarks May set the timer and virtual sync FFs.
881 */
882VMM_INT_DECL(uint64_t) TMVirtualSyncGetEx(PVM pVM, bool fCheckTimers)
883{
884 return tmVirtualSyncGetEx(pVM, fCheckTimers, NULL /*pcNsToDeadline*/);
885}
886
887
888/**
889 * Gets the current TMCLOCK_VIRTUAL_SYNC time and ticks to the next deadline
890 * without checking timers running on TMCLOCK_VIRTUAL.
891 *
892 * @returns The timestamp.
893 * @param pVM VM handle.
894 * @param pcNsToDeadline Where to return the number of nano seconds to
895 * the next virtual sync timer deadline.
896 * @thread EMT.
897 * @remarks May set the timer and virtual sync FFs.
898 */
899VMM_INT_DECL(uint64_t) TMVirtualSyncGetWithDeadlineNoCheck(PVM pVM, uint64_t *pcNsToDeadline)
900{
901 uint64_t cNsToDeadlineTmp; /* try convince the compiler to skip the if tests. */
902 uint64_t u64Now = tmVirtualSyncGetEx(pVM, false /*fCheckTimers*/, &cNsToDeadlineTmp);
903 *pcNsToDeadline = cNsToDeadlineTmp;
904 return u64Now;
905}
906
907
908/**
909 * Gets the number of nano seconds to the next virtual sync deadline.
910 *
911 * @returns The number of TMCLOCK_VIRTUAL ticks.
912 * @param pVM VM handle.
913 * @thread EMT.
914 * @remarks May set the timer and virtual sync FFs.
915 */
916VMM_INT_DECL(uint64_t) TMVirtualSyncGetNsToDeadline(PVM pVM)
917{
918 uint64_t cNsToDeadline;
919 tmVirtualSyncGetEx(pVM, false /*fCheckTimers*/, &cNsToDeadline);
920 return cNsToDeadline;
921}
922
923
924/**
925 * Gets the current lag of the synchronous virtual clock (relative to the virtual clock).
926 *
927 * @return The current lag.
928 * @param pVM VM handle.
929 */
930VMM_INT_DECL(uint64_t) TMVirtualSyncGetLag(PVM pVM)
931{
932 return pVM->tm.s.offVirtualSync - pVM->tm.s.offVirtualSyncGivenUp;
933}
934
935
936/**
937 * Get the current catch-up percent.
938 *
939 * @return The current catch0up percent. 0 means running at the same speed as the virtual clock.
940 * @param pVM VM handle.
941 */
942VMM_INT_DECL(uint32_t) TMVirtualSyncGetCatchUpPct(PVM pVM)
943{
944 if (pVM->tm.s.fVirtualSyncCatchUp)
945 return pVM->tm.s.u32VirtualSyncCatchUpPercentage;
946 return 0;
947}
948
949
950/**
951 * Gets the current TMCLOCK_VIRTUAL frequency.
952 *
953 * @returns The frequency.
954 * @param pVM VM handle.
955 */
956VMM_INT_DECL(uint64_t) TMVirtualGetFreq(PVM pVM)
957{
958 NOREF(pVM);
959 return TMCLOCK_FREQ_VIRTUAL;
960}
961
962
963/**
964 * Worker for TMR3PauseClocks.
965 *
966 * @returns VINF_SUCCESS or VERR_TM_VIRTUAL_TICKING_IPE (asserted).
967 * @param pVM The VM handle.
968 */
969int tmVirtualPauseLocked(PVM pVM)
970{
971 uint32_t c = ASMAtomicDecU32(&pVM->tm.s.cVirtualTicking);
972 AssertMsgReturn(c < pVM->cCpus, ("%u vs %u\n", c, pVM->cCpus), VERR_TM_VIRTUAL_TICKING_IPE);
973 if (c == 0)
974 {
975 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualPause);
976 pVM->tm.s.u64Virtual = tmVirtualGetRaw(pVM);
977 ASMAtomicWriteBool(&pVM->tm.s.fVirtualSyncTicking, false);
978 }
979 return VINF_SUCCESS;
980}
981
982
983/**
984 * Worker for TMR3ResumeClocks.
985 *
986 * @returns VINF_SUCCESS or VERR_TM_VIRTUAL_TICKING_IPE (asserted).
987 * @param pVM The VM handle.
988 */
989int tmVirtualResumeLocked(PVM pVM)
990{
991 uint32_t c = ASMAtomicIncU32(&pVM->tm.s.cVirtualTicking);
992 AssertMsgReturn(c <= pVM->cCpus, ("%u vs %u\n", c, pVM->cCpus), VERR_TM_VIRTUAL_TICKING_IPE);
993 if (c == 1)
994 {
995 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualResume);
996 pVM->tm.s.u64VirtualRawPrev = 0;
997 pVM->tm.s.u64VirtualWarpDriveStart = tmVirtualGetRawNanoTS(pVM);
998 pVM->tm.s.u64VirtualOffset = pVM->tm.s.u64VirtualWarpDriveStart - pVM->tm.s.u64Virtual;
999 ASMAtomicWriteBool(&pVM->tm.s.fVirtualSyncTicking, true);
1000 }
1001 return VINF_SUCCESS;
1002}
1003
1004
1005/**
1006 * Converts from virtual ticks to nanoseconds.
1007 *
1008 * @returns nanoseconds.
1009 * @param pVM The VM handle.
1010 * @param u64VirtualTicks The virtual ticks to convert.
1011 * @remark There could be rounding errors here. We just do a simple integer divide
1012 * without any adjustments.
1013 */
1014VMM_INT_DECL(uint64_t) TMVirtualToNano(PVM pVM, uint64_t u64VirtualTicks)
1015{
1016 NOREF(pVM);
1017 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
1018 return u64VirtualTicks;
1019}
1020
1021
1022/**
1023 * Converts from virtual ticks to microseconds.
1024 *
1025 * @returns microseconds.
1026 * @param pVM The VM handle.
1027 * @param u64VirtualTicks The virtual ticks to convert.
1028 * @remark There could be rounding errors here. We just do a simple integer divide
1029 * without any adjustments.
1030 */
1031VMM_INT_DECL(uint64_t) TMVirtualToMicro(PVM pVM, uint64_t u64VirtualTicks)
1032{
1033 NOREF(pVM);
1034 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
1035 return u64VirtualTicks / 1000;
1036}
1037
1038
1039/**
1040 * Converts from virtual ticks to milliseconds.
1041 *
1042 * @returns milliseconds.
1043 * @param pVM The VM handle.
1044 * @param u64VirtualTicks The virtual ticks to convert.
1045 * @remark There could be rounding errors here. We just do a simple integer divide
1046 * without any adjustments.
1047 */
1048VMM_INT_DECL(uint64_t) TMVirtualToMilli(PVM pVM, uint64_t u64VirtualTicks)
1049{
1050 NOREF(pVM);
1051 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
1052 return u64VirtualTicks / 1000000;
1053}
1054
1055
1056/**
1057 * Converts from nanoseconds to virtual ticks.
1058 *
1059 * @returns virtual ticks.
1060 * @param pVM The VM handle.
1061 * @param u64NanoTS The nanosecond value ticks to convert.
1062 * @remark There could be rounding and overflow errors here.
1063 */
1064VMM_INT_DECL(uint64_t) TMVirtualFromNano(PVM pVM, uint64_t u64NanoTS)
1065{
1066 NOREF(pVM);
1067 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
1068 return u64NanoTS;
1069}
1070
1071
1072/**
1073 * Converts from microseconds to virtual ticks.
1074 *
1075 * @returns virtual ticks.
1076 * @param pVM The VM handle.
1077 * @param u64MicroTS The microsecond value ticks to convert.
1078 * @remark There could be rounding and overflow errors here.
1079 */
1080VMM_INT_DECL(uint64_t) TMVirtualFromMicro(PVM pVM, uint64_t u64MicroTS)
1081{
1082 NOREF(pVM);
1083 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
1084 return u64MicroTS * 1000;
1085}
1086
1087
1088/**
1089 * Converts from milliseconds to virtual ticks.
1090 *
1091 * @returns virtual ticks.
1092 * @param pVM The VM handle.
1093 * @param u64MilliTS The millisecond value ticks to convert.
1094 * @remark There could be rounding and overflow errors here.
1095 */
1096VMM_INT_DECL(uint64_t) TMVirtualFromMilli(PVM pVM, uint64_t u64MilliTS)
1097{
1098 NOREF(pVM);
1099 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
1100 return u64MilliTS * 1000000;
1101}
1102
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