VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/HWACCMAll.cpp@ 31768

Last change on this file since 31768 was 30327, checked in by vboxsync, 15 years ago

Paranoia

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.4 KB
Line 
1/* $Id: HWACCMAll.cpp 30327 2010-06-21 12:43:53Z vboxsync $ */
2/** @file
3 * HWACCM - 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_HWACCM
23#include <VBox/hwaccm.h>
24#include <VBox/pgm.h>
25#include "HWACCMInternal.h"
26#include <VBox/vm.h>
27#include <VBox/x86.h>
28#include <VBox/hwacc_vmx.h>
29#include <VBox/hwacc_svm.h>
30#include <VBox/err.h>
31#include <VBox/log.h>
32#include <iprt/param.h>
33#include <iprt/assert.h>
34#include <iprt/asm.h>
35#include <iprt/string.h>
36
37/**
38 * Queues a page for invalidation
39 *
40 * @returns VBox status code.
41 * @param pVCpu The VMCPU to operate on.
42 * @param GCVirt Page to invalidate
43 */
44void hwaccmQueueInvlPage(PVMCPU pVCpu, RTGCPTR GCVirt)
45{
46 /* Nothing to do if a TLB flush is already pending */
47 if (VMCPU_FF_ISSET(pVCpu, VMCPU_FF_TLB_FLUSH))
48 return;
49#if 1
50 VMCPU_FF_SET(pVCpu, VMCPU_FF_TLB_FLUSH);
51#else
52 Be very careful when activating this code!
53 if (iPage == RT_ELEMENTS(pVCpu->hwaccm.s.TlbShootdown.aPages))
54 VMCPU_FF_SET(pVCpu, VMCPU_FF_TLB_FLUSH);
55 else
56 VMCPU_FF_SET(pVCpu, VMCPU_FF_TLB_SHOOTDOWN);
57#endif
58}
59
60/**
61 * Invalidates a guest page
62 *
63 * @returns VBox status code.
64 * @param pVCpu The VMCPU to operate on.
65 * @param GCVirt Page to invalidate
66 */
67VMMDECL(int) HWACCMInvalidatePage(PVMCPU pVCpu, RTGCPTR GCVirt)
68{
69 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatFlushPageManual);
70#ifdef IN_RING0
71 PVM pVM = pVCpu->CTX_SUFF(pVM);
72 if (pVM->hwaccm.s.vmx.fSupported)
73 return VMXR0InvalidatePage(pVM, pVCpu, GCVirt);
74
75 Assert(pVM->hwaccm.s.svm.fSupported);
76 return SVMR0InvalidatePage(pVM, pVCpu, GCVirt);
77#endif
78
79 hwaccmQueueInvlPage(pVCpu, GCVirt);
80 return VINF_SUCCESS;
81}
82
83/**
84 * Flushes the guest TLB
85 *
86 * @returns VBox status code.
87 * @param pVCpu The VMCPU to operate on.
88 */
89VMMDECL(int) HWACCMFlushTLB(PVMCPU pVCpu)
90{
91 LogFlow(("HWACCMFlushTLB\n"));
92
93 VMCPU_FF_SET(pVCpu, VMCPU_FF_TLB_FLUSH);
94 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatFlushTLBManual);
95 return VINF_SUCCESS;
96}
97
98#ifdef IN_RING0
99/**
100 * Dummy RTMpOnSpecific handler since RTMpPokeCpu couldn't be used.
101 *
102 */
103static DECLCALLBACK(void) hwaccmFlushHandler(RTCPUID idCpu, void *pvUser1, void *pvUser2)
104{
105 return;
106}
107
108/**
109 * Wrapper for RTMpPokeCpu to deal with VERR_NOT_SUPPORTED
110 *
111 */
112void hwaccmMpPokeCpu(PVMCPU pVCpu, RTCPUID idHostCpu)
113{
114 uint32_t cWorldSwitchExit = pVCpu->hwaccm.s.cWorldSwitchExit;
115
116 Assert(idHostCpu == pVCpu->idHostCpu);
117
118 STAM_PROFILE_ADV_START(&pVCpu->hwaccm.s.StatPoke, x);
119 int rc = RTMpPokeCpu(idHostCpu);
120 STAM_PROFILE_ADV_STOP(&pVCpu->hwaccm.s.StatPoke, x);
121 /* Not implemented on some platforms (Darwin, Linux kernel < 2.6.19); fall back to a less efficient implementation (broadcast). */
122 if (rc == VERR_NOT_SUPPORTED)
123 {
124 STAM_PROFILE_ADV_START(&pVCpu->hwaccm.s.StatSpinPoke, z);
125 /* synchronous. */
126 RTMpOnSpecific(idHostCpu, hwaccmFlushHandler, 0, 0);
127 STAM_PROFILE_ADV_STOP(&pVCpu->hwaccm.s.StatSpinPoke, z);
128 }
129 else
130 {
131 if (rc == VINF_SUCCESS)
132 STAM_PROFILE_ADV_START(&pVCpu->hwaccm.s.StatSpinPoke, z);
133 else
134 STAM_PROFILE_ADV_START(&pVCpu->hwaccm.s.StatSpinPokeFailed, z);
135
136 /* Spin until the VCPU has switched back. */
137 while ( pVCpu->hwaccm.s.fCheckedTLBFlush
138 && cWorldSwitchExit == pVCpu->hwaccm.s.cWorldSwitchExit)
139 {
140 ASMNopPause();
141 }
142 if (rc == VINF_SUCCESS)
143 STAM_PROFILE_ADV_STOP(&pVCpu->hwaccm.s.StatSpinPoke, z);
144 else
145 STAM_PROFILE_ADV_STOP(&pVCpu->hwaccm.s.StatSpinPokeFailed, z);
146 }
147}
148#endif
149
150#ifndef IN_RC
151/**
152 * Invalidates a guest page on all VCPUs.
153 *
154 * @returns VBox status code.
155 * @param pVM The VM to operate on.
156 * @param GCVirt Page to invalidate
157 */
158VMMDECL(int) HWACCMInvalidatePageOnAllVCpus(PVM pVM, RTGCPTR GCPtr)
159{
160 VMCPUID idCurCpu = VMMGetCpuId(pVM);
161
162 STAM_COUNTER_INC(&pVM->aCpus[idCurCpu].hwaccm.s.StatFlushPage);
163
164 for (VMCPUID idCpu = 0; idCpu < pVM->cCpus; idCpu++)
165 {
166 PVMCPU pVCpu = &pVM->aCpus[idCpu];
167
168 /* Nothing to do if a TLB flush is already pending; the VCPU should have already been poked if it were active */
169 if (VMCPU_FF_ISSET(pVCpu, VMCPU_FF_TLB_FLUSH))
170 continue;
171
172 if (pVCpu->idCpu == idCurCpu)
173 {
174 HWACCMInvalidatePage(pVCpu, GCPtr);
175 }
176 else
177 {
178 hwaccmQueueInvlPage(pVCpu, GCPtr);
179 if (pVCpu->hwaccm.s.fCheckedTLBFlush)
180 {
181 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatTlbShootdown);
182#ifdef IN_RING0
183 RTCPUID idHostCpu = pVCpu->hwaccm.s.idEnteredCpu;
184 if (idHostCpu != NIL_RTCPUID)
185 hwaccmMpPokeCpu(pVCpu, idHostCpu);
186#else
187 VMR3NotifyCpuFFU(pVCpu->pUVCpu, VMNOTIFYFF_FLAGS_POKE);
188#endif
189 }
190 else
191 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatFlushPageManual);
192 }
193 }
194
195 return VINF_SUCCESS;
196}
197
198
199/**
200 * Flush the TLBs of all VCPUs
201 *
202 * @returns VBox status code.
203 * @param pVM The VM to operate on.
204 */
205VMMDECL(int) HWACCMFlushTLBOnAllVCpus(PVM pVM)
206{
207 if (pVM->cCpus == 1)
208 return HWACCMFlushTLB(&pVM->aCpus[0]);
209
210 VMCPUID idThisCpu = VMMGetCpuId(pVM);
211
212 STAM_COUNTER_INC(&pVM->aCpus[idThisCpu].hwaccm.s.StatFlushTLB);
213
214 for (VMCPUID idCpu = 0; idCpu < pVM->cCpus; idCpu++)
215 {
216 PVMCPU pVCpu = &pVM->aCpus[idCpu];
217
218 /* Nothing to do if a TLB flush is already pending; the VCPU should have already been poked if it were active */
219 if (VMCPU_FF_ISSET(pVCpu, VMCPU_FF_TLB_FLUSH))
220 continue;
221
222 VMCPU_FF_SET(pVCpu, VMCPU_FF_TLB_FLUSH);
223 if (idThisCpu == idCpu)
224 continue;
225
226 if (pVCpu->hwaccm.s.fCheckedTLBFlush)
227 {
228 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatTlbShootdownFlush);
229#ifdef IN_RING0
230 RTCPUID idHostCpu = pVCpu->hwaccm.s.idEnteredCpu;
231 if (idHostCpu != NIL_RTCPUID)
232 hwaccmMpPokeCpu(pVCpu, idHostCpu);
233#else
234 VMR3NotifyCpuFFU(pVCpu->pUVCpu, VMNOTIFYFF_FLAGS_POKE);
235#endif
236 }
237 else
238 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatFlushTLBManual);
239 }
240 return VINF_SUCCESS;
241}
242#endif
243
244/**
245 * Checks if nested paging is enabled
246 *
247 * @returns boolean
248 * @param pVM The VM to operate on.
249 */
250VMMDECL(bool) HWACCMIsNestedPagingActive(PVM pVM)
251{
252 return HWACCMIsEnabled(pVM) && pVM->hwaccm.s.fNestedPaging;
253}
254
255/**
256 * Return the shadow paging mode for nested paging/ept
257 *
258 * @returns shadow paging mode
259 * @param pVM The VM to operate on.
260 */
261VMMDECL(PGMMODE) HWACCMGetShwPagingMode(PVM pVM)
262{
263 Assert(HWACCMIsNestedPagingActive(pVM));
264 if (pVM->hwaccm.s.svm.fSupported)
265 return PGMMODE_NESTED;
266
267 Assert(pVM->hwaccm.s.vmx.fSupported);
268 return PGMMODE_EPT;
269}
270
271/**
272 * Invalidates a guest page by physical address
273 *
274 * NOTE: Assumes the current instruction references this physical page though a virtual address!!
275 *
276 * @returns VBox status code.
277 * @param pVM The VM to operate on.
278 * @param GCPhys Page to invalidate
279 */
280VMMDECL(int) HWACCMInvalidatePhysPage(PVM pVM, RTGCPHYS GCPhys)
281{
282 if (!HWACCMIsNestedPagingActive(pVM))
283 return VINF_SUCCESS;
284
285#ifdef IN_RING0
286 if (pVM->hwaccm.s.vmx.fSupported)
287 {
288 VMCPUID idThisCpu = VMMGetCpuId(pVM);
289
290 for (VMCPUID idCpu = 0; idCpu < pVM->cCpus; idCpu++)
291 {
292 PVMCPU pVCpu = &pVM->aCpus[idCpu];
293
294 if (idThisCpu == idCpu)
295 {
296 VMXR0InvalidatePhysPage(pVM, pVCpu, GCPhys);
297 continue;
298 }
299
300 VMCPU_FF_SET(pVCpu, VMCPU_FF_TLB_FLUSH);
301 if (pVCpu->hwaccm.s.fCheckedTLBFlush)
302 {
303 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatTlbShootdownFlush);
304# ifdef IN_RING0
305 RTCPUID idHostCpu = pVCpu->hwaccm.s.idEnteredCpu;
306 if (idHostCpu != NIL_RTCPUID)
307 hwaccmMpPokeCpu(pVCpu, idHostCpu);
308# else
309 VMR3NotifyCpuFFU(pVCpu->pUVCpu, VMNOTIFYFF_FLAGS_POKE);
310# endif
311 }
312 else
313 STAM_COUNTER_INC(&pVCpu->hwaccm.s.StatFlushTLBManual);
314 }
315 return VINF_SUCCESS;
316 }
317
318 Assert(pVM->hwaccm.s.svm.fSupported);
319 /* AMD-V doesn't support invalidation with guest physical addresses; see comment in SVMR0InvalidatePhysPage. */
320 HWACCMFlushTLBOnAllVCpus(pVM);
321#else
322 HWACCMFlushTLBOnAllVCpus(pVM);
323#endif
324 return VINF_SUCCESS;
325}
326
327/**
328 * Checks if an interrupt event is currently pending.
329 *
330 * @returns Interrupt event pending state.
331 * @param pVM The VM to operate on.
332 */
333VMMDECL(bool) HWACCMHasPendingIrq(PVM pVM)
334{
335 PVMCPU pVCpu = VMMGetCpu(pVM);
336 return !!pVCpu->hwaccm.s.Event.fPending;
337}
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