VirtualBox

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

Last change on this file since 2230 was 2082, checked in by vboxsync, 18 years ago

spaces

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 15.2 KB
Line 
1/* $Id: TMAllVirtual.cpp 2082 2007-04-13 16:16:13Z vboxsync $ */
2/** @file
3 * TM - Timeout Manager, Virtual Time, All Contexts.
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_TM
27#include <VBox/tm.h>
28#ifdef IN_RING3
29# include <VBox/rem.h>
30#endif
31#include "TMInternal.h"
32#include <VBox/vm.h>
33#include <VBox/err.h>
34#include <VBox/log.h>
35#include <VBox/sup.h>
36
37#include <iprt/time.h>
38#include <iprt/assert.h>
39#include <iprt/asm.h>
40
41
42/*******************************************************************************
43* Internal Functions *
44*******************************************************************************/
45static DECLCALLBACK(int) tmVirtualSetWarpDrive(PVM pVM, uint32_t u32Percent);
46
47
48
49/**
50 * Get the time when we're not running at 100%
51 *
52 * @returns The timestamp.
53 * @param pVM The VM handle.
54 */
55uint64_t tmVirtualGetRawNonNormal(PVM pVM)
56{
57 /*
58 * Recalculate the RTTimeNanoTS() value for the period where
59 * warp drive has been enabled.
60 */
61 uint64_t u64 = RTTimeNanoTS();
62 u64 -= pVM->tm.s.u64VirtualWarpDriveStart;
63 u64 *= pVM->tm.s.u32VirtualWarpDrivePercentage;
64 u64 /= 100;
65 u64 += pVM->tm.s.u64VirtualWarpDriveStart;
66
67 /*
68 * Now we apply the virtual time offset.
69 * (Which is the negate RTTimeNanoTS() value for when the virtual machine
70 * started if it had been running continuously without any suspends.)
71 */
72 u64 -= pVM->tm.s.u64VirtualOffset;
73 return u64;
74}
75
76
77/**
78 * Get the raw virtual time.
79 *
80 * @returns The current time stamp.
81 * @param pVM The VM handle.
82 */
83DECLINLINE(uint64_t) tmVirtualGetRaw(PVM pVM)
84{
85 if (RT_LIKELY(!pVM->tm.s.fVirtualWarpDrive))
86 return RTTimeNanoTS() - pVM->tm.s.u64VirtualOffset;
87 return tmVirtualGetRawNonNormal(pVM);
88}
89
90
91/**
92 * Gets the current TMCLOCK_VIRTUAL time
93 *
94 * @returns The timestamp.
95 * @param pVM VM handle.
96 *
97 * @remark While the flow of time will never go backwards, the speed of the
98 * progress varies due to inaccurate RTTimeNanoTS and TSC. The latter can be
99 * influenced by power saving (SpeedStep, PowerNow!), while the former
100 * makes use of TSC and kernel timers.
101 */
102TMDECL(uint64_t) TMVirtualGet(PVM pVM)
103{
104 return TMVirtualGetEx(pVM, true /* check timers */);
105}
106
107
108/**
109 * Gets the current TMCLOCK_VIRTUAL time
110 *
111 * @returns The timestamp.
112 * @param pVM VM handle.
113 * @param fCheckTimers Check timers or not
114 *
115 * @remark While the flow of time will never go backwards, the speed of the
116 * progress varies due to inaccurate RTTimeNanoTS and TSC. The latter can be
117 * influenced by power saving (SpeedStep, PowerNow!), while the former
118 * makes use of TSC and kernel timers.
119 */
120TMDECL(uint64_t) TMVirtualGetEx(PVM pVM, bool fCheckTimers)
121{
122 uint64_t u64;
123 if (pVM->tm.s.fVirtualTicking)
124 {
125 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualGet);
126 u64 = tmVirtualGetRaw(pVM);
127
128 /*
129 * Use the chance to check for expired timers.
130 */
131 if ( fCheckTimers
132 && !VM_FF_ISSET(pVM, VM_FF_TIMER)
133 && ( pVM->tm.s.CTXALLSUFF(paTimerQueues)[TMCLOCK_VIRTUAL].u64Expire <= u64
134 || ( pVM->tm.s.fVirtualSyncTicking
135 && pVM->tm.s.CTXALLSUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire <= u64 - pVM->tm.s.u64VirtualSyncOffset
136 )
137 )
138 )
139 {
140 VM_FF_SET(pVM, VM_FF_TIMER);
141#ifdef IN_RING3
142 REMR3NotifyTimerPending(pVM);
143 VMR3NotifyFF(pVM, true);
144#endif
145 }
146 }
147 else
148 u64 = pVM->tm.s.u64Virtual;
149 return u64;
150}
151
152
153/**
154 * Gets the current TMCLOCK_VIRTUAL_SYNC time.
155 *
156 * @returns The timestamp.
157 * @param pVM VM handle.
158 */
159TMDECL(uint64_t) TMVirtualGetSync(PVM pVM)
160{
161 uint64_t u64;
162 if (pVM->tm.s.fVirtualSyncTicking)
163 {
164 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualGetSync);
165
166 /*
167 * Do TMVirtualGet() to get the current TMCLOCK_VIRTUAL time.
168 */
169 Assert(pVM->tm.s.fVirtualTicking);
170 u64 = tmVirtualGetRaw(pVM);
171 if ( !VM_FF_ISSET(pVM, VM_FF_TIMER)
172 && pVM->tm.s.CTXALLSUFF(paTimerQueues)[TMCLOCK_VIRTUAL].u64Expire <= u64)
173 {
174 VM_FF_SET(pVM, VM_FF_TIMER);
175#ifdef IN_RING3
176 REMR3NotifyTimerPending(pVM);
177 VMR3NotifyFF(pVM, true);
178#endif
179 }
180
181 /*
182 * Read the offset and adjust if we're playing catch-up.
183 *
184 * The catch-up adjusting work by us decrementing the offset by a percentage of
185 * the time elapsed since the previous TMVritualGetSync call. We take some simple
186 * precautions against racing other threads here, but assume that this isn't going
187 * to be much of a problem since calls to this function is unlikely from threads
188 * other than the EMT.
189 *
190 * It's possible to get a very long or even negative interval between two read
191 * for the following reasons:
192 * - Someone might have suspended the process execution, frequently the case when
193 * debugging the process.
194 * - We might be on a different CPU which TSC isn't quite in sync with the
195 * other CPUs in the system.
196 * - RTTimeNanoTS() is returning sligtly different values in GC, R0 and R3 because
197 * of the static variable it uses with the previous read time.
198 * - Another thread is racing us and we might have been preemnted while inside
199 * this function.
200 *
201 * Assuming nano second virtual time, we can simply ignore any intervals which has
202 * any of the upper 32 bits set. This will have the nice sideeffect of allowing us
203 * to use (faster) 32-bit math.
204 */
205 AssertCompile(TMCLOCK_FREQ_VIRTUAL <= 2000000000); /* (assumes low 32-bit >= 2 seconds) */
206 uint64_t u64Offset = pVM->tm.s.u64VirtualSyncOffset;
207 if (pVM->tm.s.fVirtualSyncCatchUp)
208 {
209 const uint64_t u64Prev = pVM->tm.s.u64VirtualSyncCatchUpPrev;
210 uint64_t u64Delta = u64 - u64Prev;
211 if (!(u64Delta >> 32))
212 {
213 uint32_t u32Sub = ASMDivU64ByU32RetU32(ASMMult2xU32RetU64((uint32_t)u64Delta, pVM->tm.s.u32VirtualSyncCatchupPercentage),
214 100);
215 if (u32Sub < (uint32_t)u64Delta)
216 {
217 const uint64_t u64NewOffset = u64Offset - u32Sub;
218 if (ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualSyncCatchUpPrev, u64, u64Prev))
219 ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualSyncOffset, u64NewOffset, u64Offset);
220 u64Offset = u64NewOffset;
221 }
222 else
223 {
224 /* we've completely caught up. */
225 if ( ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualSyncCatchUpPrev, u64, u64Prev)
226 && ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualSyncOffset, 0, u64Offset))
227 ASMAtomicXchgSize(&pVM->tm.s.fVirtualSyncCatchUp, false);
228 }
229 }
230 else
231 {
232 /* Update the previous TMVirtualGetSync time it's not a negative delta. */
233 if (!(u64Delta >> 63))
234 ASMAtomicCmpXchgU64(&pVM->tm.s.u64VirtualSyncCatchUpPrev, u64, u64Prev);
235 Log(("TMVirtualGetSync: u64Delta=%VRU64\n", u64Delta));
236 }
237 }
238
239 /*
240 * Complete the calculation of the current TMCLOCK_VIRTUAL_SYNC time.
241 * The current approach will not let us pass any expired timer.
242 */
243 u64 -= u64Offset;
244 if (pVM->tm.s.CTXALLSUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire <= u64)
245 {
246 if (!VM_FF_ISSET(pVM, VM_FF_TIMER))
247 {
248 VM_FF_SET(pVM, VM_FF_TIMER);
249#ifdef IN_RING3
250 REMR3NotifyTimerPending(pVM);
251 VMR3NotifyFF(pVM, true);
252#endif
253 }
254 const uint64_t u64Expire = pVM->tm.s.CTXALLSUFF(paTimerQueues)[TMCLOCK_VIRTUAL_SYNC].u64Expire;
255 if (u64Expire < u64)
256 u64 = u64Expire;
257 }
258 }
259 else
260 u64 = pVM->tm.s.u64VirtualSync;
261 return u64;
262}
263
264
265/**
266 * Gets the current TMCLOCK_VIRTUAL frequency.
267 *
268 * @returns The freqency.
269 * @param pVM VM handle.
270 */
271TMDECL(uint64_t) TMVirtualGetFreq(PVM pVM)
272{
273 return TMCLOCK_FREQ_VIRTUAL;
274}
275
276
277//#define TM_CONTINUOUS_TIME
278
279/**
280 * Resumes the virtual clock.
281 *
282 * @returns VINF_SUCCESS on success.
283 * @returns VINF_INTERNAL_ERROR and VBOX_STRICT assertion if called out of order.
284 * @param pVM VM handle.
285 */
286TMDECL(int) TMVirtualResume(PVM pVM)
287{
288 if (!pVM->tm.s.fVirtualTicking)
289 {
290 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualResume);
291 pVM->tm.s.u64VirtualWarpDriveStart = RTTimeNanoTS();
292 pVM->tm.s.u64VirtualOffset = pVM->tm.s.u64VirtualWarpDriveStart - pVM->tm.s.u64Virtual;
293 pVM->tm.s.fVirtualTicking = true;
294 pVM->tm.s.fVirtualSyncTicking = true;
295 return VINF_SUCCESS;
296 }
297
298#ifndef TM_CONTINUOUS_TIME
299 AssertFailed();
300 return VERR_INTERNAL_ERROR;
301#else
302 return VINF_SUCCESS;
303#endif
304}
305
306
307/**
308 * Pauses the virtual clock.
309 *
310 * @returns VINF_SUCCESS on success.
311 * @returns VINF_INTERNAL_ERROR and VBOX_STRICT assertion if called out of order.
312 * @param pVM VM handle.
313 */
314TMDECL(int) TMVirtualPause(PVM pVM)
315{
316 if (pVM->tm.s.fVirtualTicking)
317 {
318#ifndef TM_CONTINUOUS_TIME
319 STAM_COUNTER_INC(&pVM->tm.s.StatVirtualPause);
320 pVM->tm.s.u64Virtual = tmVirtualGetRaw(pVM);
321 pVM->tm.s.fVirtualSyncTicking = false;
322 pVM->tm.s.fVirtualTicking = false;
323#endif
324 return VINF_SUCCESS;
325 }
326
327 AssertFailed();
328 return VERR_INTERNAL_ERROR;
329}
330
331
332/**
333 * Gets the current warp drive percent.
334 *
335 * @returns The warp drive percent.
336 * @param pVM The VM handle.
337 */
338TMDECL(uint32_t) TMVirtualGetWarpDrive(PVM pVM)
339{
340 return pVM->tm.s.u32VirtualWarpDrivePercentage;
341}
342
343
344/**
345 * Sets the warp drive percent of the virtual time.
346 *
347 * @returns VBox status code.
348 * @param pVM The VM handle.
349 * @param u32Percent The new percentage. 100 means normal operation.
350 */
351TMDECL(int) TMVirtualSetWarpDrive(PVM pVM, uint32_t u32Percent)
352{
353#ifdef IN_RING3
354 PVMREQ pReq;
355 int rc = VMR3ReqCall(pVM, &pReq, RT_INDEFINITE_WAIT, (PFNRT)tmVirtualSetWarpDrive, 2, pVM, u32Percent);
356 if (VBOX_SUCCESS(rc))
357 rc = pReq->iStatus;
358 VMR3ReqFree(pReq);
359 return rc;
360#else
361
362 return tmVirtualSetWarpDrive(pVM, u32Percent);
363#endif
364}
365
366
367/**
368 * EMT worker for tmVirtualSetWarpDrive.
369 *
370 * @returns VBox status code.
371 * @param pVM The VM handle.
372 * @param u32Percent See TMVirtualSetWarpDrive().
373 * @internal
374 */
375static DECLCALLBACK(int) tmVirtualSetWarpDrive(PVM pVM, uint32_t u32Percent)
376{
377 /*
378 * Validate it.
379 */
380 AssertMsgReturn(u32Percent >= 2 && u32Percent <= 20000,
381 ("%RX32 is not between 2 and 20000 (inclusive).\n", u32Percent),
382 VERR_INVALID_PARAMETER);
383
384 /*
385 * If the time is running we'll have to pause it before we can change
386 * the warp drive settings.
387 */
388 bool fPaused = pVM->tm.s.fVirtualTicking;
389 if (fPaused)
390 {
391 int rc = TMVirtualPause(pVM);
392 AssertRCReturn(rc, rc);
393 rc = TMCpuTickPause(pVM);
394 AssertRCReturn(rc, rc);
395 }
396
397 pVM->tm.s.u32VirtualWarpDrivePercentage = u32Percent;
398 pVM->tm.s.fVirtualWarpDrive = u32Percent != 100;
399 LogRel(("TM: u32VirtualWarpDrivePercentage=%RI32 fVirtualWarpDrive=%RTbool\n",
400 pVM->tm.s.u32VirtualWarpDrivePercentage, pVM->tm.s.fVirtualWarpDrive));
401
402 if (fPaused)
403 {
404 int rc = TMVirtualResume(pVM);
405 AssertRCReturn(rc, rc);
406 rc = TMCpuTickResume(pVM);
407 AssertRCReturn(rc, rc);
408 }
409
410 return VINF_SUCCESS;
411}
412
413
414/**
415 * Converts from virtual ticks to nanoseconds.
416 *
417 * @returns nanoseconds.
418 * @param pVM The VM handle.
419 * @param u64VirtualTicks The virtual ticks to convert.
420 * @remark There could be rounding errors here. We just do a simple integere divide
421 * without any adjustments.
422 */
423TMDECL(uint64_t) TMVirtualToNano(PVM pVM, uint64_t u64VirtualTicks)
424{
425 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
426 return u64VirtualTicks;
427}
428
429
430/**
431 * Converts from virtual ticks to microseconds.
432 *
433 * @returns microseconds.
434 * @param pVM The VM handle.
435 * @param u64VirtualTicks The virtual ticks to convert.
436 * @remark There could be rounding errors here. We just do a simple integere divide
437 * without any adjustments.
438 */
439TMDECL(uint64_t) TMVirtualToMicro(PVM pVM, uint64_t u64VirtualTicks)
440{
441 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
442 return u64VirtualTicks / 1000;
443}
444
445
446/**
447 * Converts from virtual ticks to milliseconds.
448 *
449 * @returns milliseconds.
450 * @param pVM The VM handle.
451 * @param u64VirtualTicks The virtual ticks to convert.
452 * @remark There could be rounding errors here. We just do a simple integere divide
453 * without any adjustments.
454 */
455TMDECL(uint64_t) TMVirtualToMilli(PVM pVM, uint64_t u64VirtualTicks)
456{
457 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
458 return u64VirtualTicks / 1000000;
459}
460
461
462/**
463 * Converts from nanoseconds to virtual ticks.
464 *
465 * @returns virtual ticks.
466 * @param pVM The VM handle.
467 * @param u64NanoTS The nanosecond value ticks to convert.
468 * @remark There could be rounding and overflow errors here.
469 */
470TMDECL(uint64_t) TMVirtualFromNano(PVM pVM, uint64_t u64NanoTS)
471{
472 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
473 return u64NanoTS;
474}
475
476
477/**
478 * Converts from microseconds to virtual ticks.
479 *
480 * @returns virtual ticks.
481 * @param pVM The VM handle.
482 * @param u64MicroTS The microsecond value ticks to convert.
483 * @remark There could be rounding and overflow errors here.
484 */
485TMDECL(uint64_t) TMVirtualFromMicro(PVM pVM, uint64_t u64MicroTS)
486{
487 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
488 return u64MicroTS * 1000;
489}
490
491
492/**
493 * Converts from milliseconds to virtual ticks.
494 *
495 * @returns virtual ticks.
496 * @param pVM The VM handle.
497 * @param u64MilliTS The millisecond value ticks to convert.
498 * @remark There could be rounding and overflow errors here.
499 */
500TMDECL(uint64_t) TMVirtualFromMilli(PVM pVM, uint64_t u64MilliTS)
501{
502 AssertCompile(TMCLOCK_FREQ_VIRTUAL == 1000000000);
503 return u64MilliTS * 1000000;
504}
505
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