VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/TMAllCpu.cpp@ 28875

Last change on this file since 28875 was 28800, checked in by vboxsync, 15 years ago

Automated rebranding to Oracle copyright/license strings via filemuncher

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 9.9 KB
Line 
1/* $Id: TMAllCpu.cpp 28800 2010-04-27 08:22:32Z vboxsync $ */
2/** @file
3 * TM - Timeout Manager, CPU 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/tm.h>
24#include "../TMInternal.h"
25#include <VBox/vm.h>
26#include <VBox/sup.h>
27
28#include <VBox/param.h>
29#include <VBox/err.h>
30#include <iprt/assert.h>
31#include <iprt/asm.h>
32#include <VBox/log.h>
33
34
35/**
36 * Gets the raw cpu tick from current virtual time.
37 */
38DECLINLINE(uint64_t) tmCpuTickGetRawVirtual(PVM pVM, bool fCheckTimers)
39{
40 uint64_t u64 = TMVirtualSyncGetEx(pVM, fCheckTimers);
41 if (u64 != TMCLOCK_FREQ_VIRTUAL)
42 u64 = ASMMultU64ByU32DivByU32(u64, pVM->tm.s.cTSCTicksPerSecond, TMCLOCK_FREQ_VIRTUAL);
43 return u64;
44}
45
46
47/**
48 * Resumes the CPU timestamp counter ticking.
49 *
50 * @returns VBox status code.
51 * @param pVM The VM to operate on.
52 * @param pVCpu The VMCPU to operate on.
53 * @internal
54 */
55int tmCpuTickResume(PVM pVM, PVMCPU pVCpu)
56{
57 if (!pVCpu->tm.s.fTSCTicking)
58 {
59 pVCpu->tm.s.fTSCTicking = true;
60 if (pVM->tm.s.fTSCVirtualized)
61 {
62 /** @todo Test that pausing and resuming doesn't cause lag! (I.e. that we're
63 * unpaused before the virtual time and stopped after it. */
64 if (pVM->tm.s.fTSCUseRealTSC)
65 pVCpu->tm.s.offTSCRawSrc = ASMReadTSC() - pVCpu->tm.s.u64TSC;
66 else
67 pVCpu->tm.s.offTSCRawSrc = tmCpuTickGetRawVirtual(pVM, false /* don't check for pending timers */)
68 - pVCpu->tm.s.u64TSC;
69 }
70 return VINF_SUCCESS;
71 }
72 AssertFailed();
73 return VERR_INTERNAL_ERROR;
74}
75
76
77/**
78 * Pauses the CPU timestamp counter ticking.
79 *
80 * @returns VBox status code.
81 * @param pVM The VM to operate on.
82 * @param pVCpu The VMCPU to operate on.
83 * @internal
84 */
85int tmCpuTickPause(PVM pVM, PVMCPU pVCpu)
86{
87 if (pVCpu->tm.s.fTSCTicking)
88 {
89 pVCpu->tm.s.u64TSC = TMCpuTickGetNoCheck(pVCpu);
90 pVCpu->tm.s.fTSCTicking = false;
91 return VINF_SUCCESS;
92 }
93 AssertFailed();
94 return VERR_INTERNAL_ERROR;
95}
96
97
98/**
99 * Checks if AMD-V / VT-x can use an offsetted hardware TSC or not.
100 *
101 * @returns true/false accordingly.
102 * @param pVCpu The VMCPU to operate on.
103 * @param poffRealTSC The offset against the TSC of the current CPU.
104 * Can be NULL.
105 * @thread EMT.
106 */
107VMM_INT_DECL(bool) TMCpuTickCanUseRealTSC(PVMCPU pVCpu, uint64_t *poffRealTSC)
108{
109 PVM pVM = pVCpu->CTX_SUFF(pVM);
110
111 /*
112 * We require:
113 * 1. A fixed TSC, this is checked at init time.
114 * 2. That the TSC is ticking (we shouldn't be here if it isn't)
115 * 3. Either that we're using the real TSC as time source or
116 * a) we don't have any lag to catch up, and
117 * b) the virtual sync clock hasn't been halted by an expired timer, and
118 * c) we're not using warp drive (accelerated virtual guest time).
119 */
120 if ( pVM->tm.s.fMaybeUseOffsettedHostTSC
121 && RT_LIKELY(pVCpu->tm.s.fTSCTicking)
122 && ( pVM->tm.s.fTSCUseRealTSC
123 || ( !pVM->tm.s.fVirtualSyncCatchUp
124 && RT_LIKELY(pVM->tm.s.fVirtualSyncTicking)
125 && !pVM->tm.s.fVirtualWarpDrive))
126 )
127 {
128 if (!pVM->tm.s.fTSCUseRealTSC)
129 {
130 /* The source is the timer synchronous virtual clock. */
131 Assert(pVM->tm.s.fTSCVirtualized);
132
133 if (poffRealTSC)
134 {
135 uint64_t u64Now = tmCpuTickGetRawVirtual(pVM, false /* don't check for pending timers */)
136 - pVCpu->tm.s.offTSCRawSrc;
137 /** @todo When we start collecting statistics on how much time we spend executing
138 * guest code before exiting, we should check this against the next virtual sync
139 * timer timeout. If it's lower than the avg. length, we should trap rdtsc to increase
140 * the chance that we'll get interrupted right after the timer expired. */
141 *poffRealTSC = u64Now - ASMReadTSC();
142 }
143 }
144 else if (poffRealTSC)
145 {
146 /* The source is the real TSC. */
147 if (pVM->tm.s.fTSCVirtualized)
148 *poffRealTSC = pVCpu->tm.s.offTSCRawSrc;
149 else
150 *poffRealTSC = 0;
151 }
152 /** @todo count this? */
153 return true;
154 }
155
156#ifdef VBOX_WITH_STATISTICS
157 /* Sample the reason for refusing. */
158 if (!pVM->tm.s.fMaybeUseOffsettedHostTSC)
159 STAM_COUNTER_INC(&pVM->tm.s.StatTSCNotFixed);
160 else if (!pVCpu->tm.s.fTSCTicking)
161 STAM_COUNTER_INC(&pVM->tm.s.StatTSCNotTicking);
162 else if (!pVM->tm.s.fTSCUseRealTSC)
163 {
164 if (pVM->tm.s.fVirtualSyncCatchUp)
165 {
166 if (pVM->tm.s.u32VirtualSyncCatchUpPercentage <= 10)
167 STAM_COUNTER_INC(&pVM->tm.s.StatTSCCatchupLE010);
168 else if (pVM->tm.s.u32VirtualSyncCatchUpPercentage <= 25)
169 STAM_COUNTER_INC(&pVM->tm.s.StatTSCCatchupLE025);
170 else if (pVM->tm.s.u32VirtualSyncCatchUpPercentage <= 100)
171 STAM_COUNTER_INC(&pVM->tm.s.StatTSCCatchupLE100);
172 else
173 STAM_COUNTER_INC(&pVM->tm.s.StatTSCCatchupOther);
174 }
175 else if (!pVM->tm.s.fVirtualSyncTicking)
176 STAM_COUNTER_INC(&pVM->tm.s.StatTSCSyncNotTicking);
177 else if (pVM->tm.s.fVirtualWarpDrive)
178 STAM_COUNTER_INC(&pVM->tm.s.StatTSCWarp);
179 }
180#endif
181 return false;
182}
183
184
185/**
186 * Read the current CPU timstamp counter.
187 *
188 * @returns Gets the CPU tsc.
189 * @param pVCpu The VMCPU to operate on.
190 */
191DECLINLINE(uint64_t) tmCpuTickGetInternal(PVMCPU pVCpu, bool fCheckTimers)
192{
193 uint64_t u64;
194
195 if (RT_LIKELY(pVCpu->tm.s.fTSCTicking))
196 {
197 PVM pVM = pVCpu->CTX_SUFF(pVM);
198 if (pVM->tm.s.fTSCVirtualized)
199 {
200 if (pVM->tm.s.fTSCUseRealTSC)
201 u64 = ASMReadTSC();
202 else
203 u64 = tmCpuTickGetRawVirtual(pVM, fCheckTimers);
204 u64 -= pVCpu->tm.s.offTSCRawSrc;
205 }
206 else
207 u64 = ASMReadTSC();
208
209 /* Never return a value lower than what the guest has already seen. */
210 if (u64 < pVCpu->tm.s.u64TSCLastSeen)
211 {
212 STAM_COUNTER_INC(&pVM->tm.s.StatTSCUnderflow);
213 pVCpu->tm.s.u64TSCLastSeen += 64; /* @todo choose a good increment here */
214 u64 = pVCpu->tm.s.u64TSCLastSeen;
215 }
216 }
217 else
218 u64 = pVCpu->tm.s.u64TSC;
219 return u64;
220}
221
222
223/**
224 * Read the current CPU timstamp counter.
225 *
226 * @returns Gets the CPU tsc.
227 * @param pVCpu The VMCPU to operate on.
228 */
229VMMDECL(uint64_t) TMCpuTickGet(PVMCPU pVCpu)
230{
231 return tmCpuTickGetInternal(pVCpu, true /* fCheckTimers */);
232}
233
234
235/**
236 * Read the current CPU timstamp counter, don't check for expired timers.
237 *
238 * @returns Gets the CPU tsc.
239 * @param pVCpu The VMCPU to operate on.
240 */
241VMM_INT_DECL(uint64_t) TMCpuTickGetNoCheck(PVMCPU pVCpu)
242{
243 return tmCpuTickGetInternal(pVCpu, false /* fCheckTimers */);
244}
245
246
247/**
248 * Sets the current CPU timestamp counter.
249 *
250 * @returns VBox status code.
251 * @param pVM The VM handle.
252 * @param pVCpu The virtual CPU to operate on.
253 * @param u64Tick The new timestamp value.
254 *
255 * @thread EMT which TSC is to be set.
256 */
257VMM_INT_DECL(int) TMCpuTickSet(PVM pVM, PVMCPU pVCpu, uint64_t u64Tick)
258{
259 VMCPU_ASSERT_EMT(pVCpu);
260 STAM_COUNTER_INC(&pVM->tm.s.StatTSCSet);
261
262 /*
263 * This is easier to do when the TSC is paused since resume will
264 * do all the calcuations for us. Actually, we don't need to
265 * call tmCpuTickPause here since we overwrite u64TSC anyway.
266 */
267 bool fTSCTicking = pVCpu->tm.s.fTSCTicking;
268 pVCpu->tm.s.fTSCTicking = false;
269 pVCpu->tm.s.u64TSC = u64Tick;
270 pVCpu->tm.s.u64TSCLastSeen = u64Tick;
271 if (fTSCTicking)
272 tmCpuTickResume(pVM, pVCpu);
273 /** @todo Try help synchronizing it better among the virtual CPUs? */
274
275 return VINF_SUCCESS;
276}
277
278/**
279 * Sets the last seen CPU timestamp counter.
280 *
281 * @returns VBox status code.
282 * @param pVCpu The virtual CPU to operate on.
283 * @param u64LastSeenTick The last seen timestamp value.
284 *
285 * @thread EMT which TSC is to be set.
286 */
287VMM_INT_DECL(int) TMCpuTickSetLastSeen(PVMCPU pVCpu, uint64_t u64LastSeenTick)
288{
289 VMCPU_ASSERT_EMT(pVCpu);
290
291 LogFlow(("TMCpuTickSetLastSeen %RX64\n", u64LastSeenTick));
292 if (pVCpu->tm.s.u64TSCLastSeen < u64LastSeenTick)
293 pVCpu->tm.s.u64TSCLastSeen = u64LastSeenTick;
294 return VINF_SUCCESS;
295}
296
297/**
298 * Gets the last seen CPU timestamp counter.
299 *
300 * @returns last seen TSC
301 * @param pVCpu The virtual CPU to operate on.
302 *
303 * @thread EMT which TSC is to be set.
304 */
305VMM_INT_DECL(uint64_t) TMCpuTickGetLastSeen(PVMCPU pVCpu)
306{
307 VMCPU_ASSERT_EMT(pVCpu);
308
309 return pVCpu->tm.s.u64TSCLastSeen;
310}
311
312
313/**
314 * Get the timestamp frequency.
315 *
316 * @returns Number of ticks per second.
317 * @param pVM The VM.
318 */
319VMMDECL(uint64_t) TMCpuTicksPerSecond(PVM pVM)
320{
321 if (pVM->tm.s.fTSCUseRealTSC)
322 {
323 uint64_t cTSCTicksPerSecond = SUPGetCpuHzFromGIP(g_pSUPGlobalInfoPage);
324 if (RT_LIKELY(cTSCTicksPerSecond != ~(uint64_t)0))
325 return cTSCTicksPerSecond;
326 }
327 return pVM->tm.s.cTSCTicksPerSecond;
328}
329
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