VirtualBox

source: vbox/trunk/src/VBox/VMM/include/PGMInline.h@ 39669

Last change on this file since 39669 was 39034, checked in by vboxsync, 13 years ago

VMM,INTNET: Addressing unused variable warnings.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 49.9 KB
Line 
1/* $Id: PGMInline.h 39034 2011-10-19 11:43:52Z vboxsync $ */
2/** @file
3 * PGM - Inlined functions.
4 */
5
6/*
7 * Copyright (C) 2006-2010 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#ifndef ___PGMInline_h
19#define ___PGMInline_h
20
21#include <VBox/cdefs.h>
22#include <VBox/types.h>
23#include <VBox/err.h>
24#include <VBox/vmm/stam.h>
25#include <VBox/param.h>
26#include <VBox/vmm/vmm.h>
27#include <VBox/vmm/mm.h>
28#include <VBox/vmm/pdmcritsect.h>
29#include <VBox/vmm/pdmapi.h>
30#include <VBox/dis.h>
31#include <VBox/vmm/dbgf.h>
32#include <VBox/log.h>
33#include <VBox/vmm/gmm.h>
34#include <VBox/vmm/hwaccm.h>
35#include <iprt/asm.h>
36#include <iprt/assert.h>
37#include <iprt/avl.h>
38#include <iprt/critsect.h>
39#include <iprt/sha.h>
40
41
42
43/** @addtogroup grp_pgm_int Internals
44 * @internal
45 * @{
46 */
47
48/**
49 * Gets the PGMRAMRANGE structure for a guest page.
50 *
51 * @returns Pointer to the RAM range on success.
52 * @returns NULL on a VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS condition.
53 *
54 * @param pVM The VM handle.
55 * @param GCPhys The GC physical address.
56 */
57DECLINLINE(PPGMRAMRANGE) pgmPhysGetRange(PVM pVM, RTGCPHYS GCPhys)
58{
59 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
60 if (!pRam || GCPhys - pRam->GCPhys >= pRam->cb)
61 pRam = pgmPhysGetRangeSlow(pVM, GCPhys);
62 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbHits));
63 return pRam;
64}
65
66
67/**
68 * Gets the PGMRAMRANGE structure for a guest page, if unassigned get the ram
69 * range above it.
70 *
71 * @returns Pointer to the RAM range on success.
72 * @returns NULL if the address is located after the last range.
73 *
74 * @param pVM The VM handle.
75 * @param GCPhys The GC physical address.
76 */
77DECLINLINE(PPGMRAMRANGE) pgmPhysGetRangeAtOrAbove(PVM pVM, RTGCPHYS GCPhys)
78{
79 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
80 if ( !pRam
81 || (GCPhys - pRam->GCPhys) >= pRam->cb)
82 return pgmPhysGetRangeAtOrAboveSlow(pVM, GCPhys);
83 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbHits));
84 return pRam;
85}
86
87
88
89/**
90 * Gets the PGMPAGE structure for a guest page.
91 *
92 * @returns Pointer to the page on success.
93 * @returns NULL on a VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS condition.
94 *
95 * @param pVM The VM handle.
96 * @param GCPhys The GC physical address.
97 */
98DECLINLINE(PPGMPAGE) pgmPhysGetPage(PVM pVM, RTGCPHYS GCPhys)
99{
100 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
101 RTGCPHYS off;
102 if ( !pRam
103 || (off = GCPhys - pRam->GCPhys) >= pRam->cb)
104 return pgmPhysGetPageSlow(pVM, GCPhys);
105 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbHits));
106 return &pRam->aPages[off >> PAGE_SHIFT];
107}
108
109
110/**
111 * Gets the PGMPAGE structure for a guest page.
112 *
113 * Old Phys code: Will make sure the page is present.
114 *
115 * @returns VBox status code.
116 * @retval VINF_SUCCESS and a valid *ppPage on success.
117 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if the address isn't valid.
118 *
119 * @param pVM The VM handle.
120 * @param GCPhys The GC physical address.
121 * @param ppPage Where to store the page pointer on success.
122 */
123DECLINLINE(int) pgmPhysGetPageEx(PVM pVM, RTGCPHYS GCPhys, PPPGMPAGE ppPage)
124{
125 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
126 RTGCPHYS off;
127 if ( !pRam
128 || (off = GCPhys - pRam->GCPhys) >= pRam->cb)
129 return pgmPhysGetPageExSlow(pVM, GCPhys, ppPage);
130 *ppPage = &pRam->aPages[(GCPhys - pRam->GCPhys) >> PAGE_SHIFT];
131 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbHits));
132 return VINF_SUCCESS;
133}
134
135
136
137
138/**
139 * Gets the PGMPAGE structure for a guest page.
140 *
141 * Old Phys code: Will make sure the page is present.
142 *
143 * @returns VBox status code.
144 * @retval VINF_SUCCESS and a valid *ppPage on success.
145 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if the address isn't valid.
146 *
147 * @param pVM The VM handle.
148 * @param GCPhys The GC physical address.
149 * @param ppPage Where to store the page pointer on success.
150 * @param ppRamHint Where to read and store the ram list hint.
151 * The caller initializes this to NULL before the call.
152 */
153DECLINLINE(int) pgmPhysGetPageWithHintEx(PVM pVM, RTGCPHYS GCPhys, PPPGMPAGE ppPage, PPGMRAMRANGE *ppRamHint)
154{
155 RTGCPHYS off;
156 PPGMRAMRANGE pRam = *ppRamHint;
157 if ( !pRam
158 || RT_UNLIKELY((off = GCPhys - pRam->GCPhys) >= pRam->cb))
159 {
160 pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
161 if ( !pRam
162 || (off = GCPhys - pRam->GCPhys) >= pRam->cb)
163 return pgmPhysGetPageAndRangeExSlow(pVM, GCPhys, ppPage, ppRamHint);
164
165 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbHits));
166 *ppRamHint = pRam;
167 }
168 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
169 return VINF_SUCCESS;
170}
171
172
173/**
174 * Gets the PGMPAGE structure for a guest page together with the PGMRAMRANGE.
175 *
176 * @returns Pointer to the page on success.
177 * @returns NULL on a VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS condition.
178 *
179 * @param pVM The VM handle.
180 * @param GCPhys The GC physical address.
181 * @param ppPage Where to store the pointer to the PGMPAGE structure.
182 * @param ppRam Where to store the pointer to the PGMRAMRANGE structure.
183 */
184DECLINLINE(int) pgmPhysGetPageAndRangeEx(PVM pVM, RTGCPHYS GCPhys, PPPGMPAGE ppPage, PPGMRAMRANGE *ppRam)
185{
186 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
187 RTGCPHYS off;
188 if ( !pRam
189 || (off = GCPhys - pRam->GCPhys) >= pRam->cb)
190 return pgmPhysGetPageAndRangeExSlow(pVM, GCPhys, ppPage, ppRam);
191
192 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbHits));
193 *ppRam = pRam;
194 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
195 return VINF_SUCCESS;
196}
197
198
199/**
200 * Convert GC Phys to HC Phys.
201 *
202 * @returns VBox status.
203 * @param pVM The VM handle.
204 * @param GCPhys The GC physical address.
205 * @param pHCPhys Where to store the corresponding HC physical address.
206 *
207 * @deprecated Doesn't deal with zero, shared or write monitored pages.
208 * Avoid when writing new code!
209 */
210DECLINLINE(int) pgmRamGCPhys2HCPhys(PVM pVM, RTGCPHYS GCPhys, PRTHCPHYS pHCPhys)
211{
212 PPGMPAGE pPage;
213 int rc = pgmPhysGetPageEx(pVM, GCPhys, &pPage);
214 if (RT_FAILURE(rc))
215 return rc;
216 *pHCPhys = PGM_PAGE_GET_HCPHYS(pPage) | (GCPhys & PAGE_OFFSET_MASK);
217 return VINF_SUCCESS;
218}
219
220#if defined(VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0) || defined(IN_RC)
221
222/**
223 * Inlined version of the ring-0 version of the host page mapping code
224 * that optimizes access to pages already in the set.
225 *
226 * @returns VINF_SUCCESS. Will bail out to ring-3 on failure.
227 * @param pVCpu The current CPU.
228 * @param HCPhys The physical address of the page.
229 * @param ppv Where to store the mapping address.
230 */
231DECLINLINE(int) pgmRZDynMapHCPageInlined(PVMCPU pVCpu, RTHCPHYS HCPhys, void **ppv RTLOG_COMMA_SRC_POS_DECL)
232{
233 PPGMMAPSET pSet = &pVCpu->pgm.s.AutoSet;
234
235 STAM_PROFILE_START(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapHCPageInl, a);
236 Assert(!(HCPhys & PAGE_OFFSET_MASK));
237 Assert(pSet->cEntries <= RT_ELEMENTS(pSet->aEntries));
238
239 unsigned iHash = PGMMAPSET_HASH(HCPhys);
240 unsigned iEntry = pSet->aiHashTable[iHash];
241 if ( iEntry < pSet->cEntries
242 && pSet->aEntries[iEntry].HCPhys == HCPhys
243 && pSet->aEntries[iEntry].cInlinedRefs < UINT16_MAX - 1)
244 {
245 pSet->aEntries[iEntry].cInlinedRefs++;
246 *ppv = pSet->aEntries[iEntry].pvPage;
247 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapHCPageInlHits);
248 }
249 else
250 {
251 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapHCPageInlMisses);
252 pgmRZDynMapHCPageCommon(pSet, HCPhys, ppv RTLOG_COMMA_SRC_POS_ARGS);
253 }
254
255 STAM_PROFILE_STOP(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapHCPageInl, a);
256 return VINF_SUCCESS;
257}
258
259
260/**
261 * Inlined version of the guest page mapping code that optimizes access to pages
262 * already in the set.
263 *
264 * @returns VBox status code, see pgmRZDynMapGCPageCommon for details.
265 * @param pVM The VM handle.
266 * @param pVCpu The current CPU.
267 * @param GCPhys The guest physical address of the page.
268 * @param ppv Where to store the mapping address.
269 */
270DECLINLINE(int) pgmRZDynMapGCPageV2Inlined(PVM pVM, PVMCPU pVCpu, RTGCPHYS GCPhys, void **ppv RTLOG_COMMA_SRC_POS_DECL)
271{
272 STAM_PROFILE_START(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInl, a);
273 AssertMsg(!(GCPhys & PAGE_OFFSET_MASK), ("%RGp\n", GCPhys));
274
275 /*
276 * Get the ram range.
277 */
278 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
279 RTGCPHYS off;
280 if ( !pRam
281 || (off = GCPhys - pRam->GCPhys) >= pRam->cb
282 /** @todo || page state stuff */
283 )
284 {
285 /* This case is not counted into StatRZDynMapGCPageInl. */
286 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlRamMisses);
287 return pgmRZDynMapGCPageCommon(pVM, pVCpu, GCPhys, ppv RTLOG_COMMA_SRC_POS_ARGS);
288 }
289
290 RTHCPHYS HCPhys = PGM_PAGE_GET_HCPHYS(&pRam->aPages[off >> PAGE_SHIFT]);
291 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlRamHits);
292
293 /*
294 * pgmRZDynMapHCPageInlined with out stats.
295 */
296 PPGMMAPSET pSet = &pVCpu->pgm.s.AutoSet;
297 Assert(!(HCPhys & PAGE_OFFSET_MASK));
298 Assert(pSet->cEntries <= RT_ELEMENTS(pSet->aEntries));
299
300 unsigned iHash = PGMMAPSET_HASH(HCPhys);
301 unsigned iEntry = pSet->aiHashTable[iHash];
302 if ( iEntry < pSet->cEntries
303 && pSet->aEntries[iEntry].HCPhys == HCPhys
304 && pSet->aEntries[iEntry].cInlinedRefs < UINT16_MAX - 1)
305 {
306 pSet->aEntries[iEntry].cInlinedRefs++;
307 *ppv = pSet->aEntries[iEntry].pvPage;
308 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlHits);
309 }
310 else
311 {
312 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlMisses);
313 pgmRZDynMapHCPageCommon(pSet, HCPhys, ppv RTLOG_COMMA_SRC_POS_ARGS);
314 }
315
316 STAM_PROFILE_STOP(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInl, a);
317 return VINF_SUCCESS;
318}
319
320
321/**
322 * Inlined version of the ring-0 version of guest page mapping that optimizes
323 * access to pages already in the set.
324 *
325 * @returns VBox status code, see pgmRZDynMapGCPageCommon for details.
326 * @param pVCpu The current CPU.
327 * @param GCPhys The guest physical address of the page.
328 * @param ppv Where to store the mapping address.
329 */
330DECLINLINE(int) pgmRZDynMapGCPageInlined(PVMCPU pVCpu, RTGCPHYS GCPhys, void **ppv RTLOG_COMMA_SRC_POS_DECL)
331{
332 return pgmRZDynMapGCPageV2Inlined(pVCpu->CTX_SUFF(pVM), pVCpu, GCPhys, ppv RTLOG_COMMA_SRC_POS_ARGS);
333}
334
335
336/**
337 * Inlined version of the ring-0 version of the guest byte mapping code
338 * that optimizes access to pages already in the set.
339 *
340 * @returns VBox status code, see pgmRZDynMapGCPageCommon for details.
341 * @param pVCpu The current CPU.
342 * @param HCPhys The physical address of the page.
343 * @param ppv Where to store the mapping address. The offset is
344 * preserved.
345 */
346DECLINLINE(int) pgmRZDynMapGCPageOffInlined(PVMCPU pVCpu, RTGCPHYS GCPhys, void **ppv RTLOG_COMMA_SRC_POS_DECL)
347{
348 STAM_PROFILE_START(&pVCpu->pgm.s.StatRZDynMapGCPageInl, a);
349
350 /*
351 * Get the ram range.
352 */
353 PVM pVM = pVCpu->CTX_SUFF(pVM);
354 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)];
355 RTGCPHYS off;
356 if ( !pRam
357 || (off = GCPhys - pRam->GCPhys) >= pRam->cb
358 /** @todo || page state stuff */
359 )
360 {
361 /* This case is not counted into StatRZDynMapGCPageInl. */
362 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlRamMisses);
363 return pgmRZDynMapGCPageCommon(pVM, pVCpu, GCPhys, ppv RTLOG_COMMA_SRC_POS_ARGS);
364 }
365
366 RTHCPHYS HCPhys = PGM_PAGE_GET_HCPHYS(&pRam->aPages[off >> PAGE_SHIFT]);
367 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlRamHits);
368
369 /*
370 * pgmRZDynMapHCPageInlined with out stats.
371 */
372 PPGMMAPSET pSet = &pVCpu->pgm.s.AutoSet;
373 Assert(!(HCPhys & PAGE_OFFSET_MASK));
374 Assert(pSet->cEntries <= RT_ELEMENTS(pSet->aEntries));
375
376 unsigned iHash = PGMMAPSET_HASH(HCPhys);
377 unsigned iEntry = pSet->aiHashTable[iHash];
378 if ( iEntry < pSet->cEntries
379 && pSet->aEntries[iEntry].HCPhys == HCPhys
380 && pSet->aEntries[iEntry].cInlinedRefs < UINT16_MAX - 1)
381 {
382 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlHits);
383 pSet->aEntries[iEntry].cInlinedRefs++;
384 *ppv = (void *)((uintptr_t)pSet->aEntries[iEntry].pvPage | (PAGE_OFFSET_MASK & (uintptr_t)GCPhys));
385 }
386 else
387 {
388 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInlMisses);
389 pgmRZDynMapHCPageCommon(pSet, HCPhys, ppv RTLOG_COMMA_SRC_POS_ARGS);
390 *ppv = (void *)((uintptr_t)*ppv | (PAGE_OFFSET_MASK & (uintptr_t)GCPhys));
391 }
392
393 STAM_PROFILE_STOP(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZDynMapGCPageInl, a);
394 return VINF_SUCCESS;
395}
396
397#endif /* VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
398#if defined(IN_RC) || defined(VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0)
399
400/**
401 * Maps the page into current context (RC and maybe R0).
402 *
403 * @returns pointer to the mapping.
404 * @param pVM Pointer to the PGM instance data.
405 * @param pPage The page.
406 */
407DECLINLINE(void *) pgmPoolMapPageInlined(PVM pVM, PPGMPOOLPAGE pPage RTLOG_COMMA_SRC_POS_DECL)
408{
409 if (pPage->idx >= PGMPOOL_IDX_FIRST)
410 {
411 Assert(pPage->idx < pVM->pgm.s.CTX_SUFF(pPool)->cCurPages);
412 void *pv;
413 pgmRZDynMapHCPageInlined(VMMGetCpu(pVM), pPage->Core.Key, &pv RTLOG_COMMA_SRC_POS_ARGS);
414 return pv;
415 }
416 AssertFatalMsgFailed(("pgmPoolMapPageInlined invalid page index %x\n", pPage->idx));
417}
418
419/**
420 * Maps the page into current context (RC and maybe R0).
421 *
422 * @returns pointer to the mapping.
423 * @param pVM Pointer to the PGM instance data.
424 * @param pVCpu The current CPU.
425 * @param pPage The page.
426 */
427DECLINLINE(void *) pgmPoolMapPageV2Inlined(PVM pVM, PVMCPU pVCpu, PPGMPOOLPAGE pPage RTLOG_COMMA_SRC_POS_DECL)
428{
429 if (pPage->idx >= PGMPOOL_IDX_FIRST)
430 {
431 Assert(pPage->idx < pVM->pgm.s.CTX_SUFF(pPool)->cCurPages);
432 void *pv;
433 Assert(pVCpu == VMMGetCpu(pVM));
434 pgmRZDynMapHCPageInlined(pVCpu, pPage->Core.Key, &pv RTLOG_COMMA_SRC_POS_ARGS);
435 return pv;
436 }
437 AssertFatalMsgFailed(("pgmPoolMapPageV2Inlined invalid page index %x\n", pPage->idx));
438}
439
440#endif /* VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 || IN_RC */
441#ifndef IN_RC
442
443/**
444 * Queries the Physical TLB entry for a physical guest page,
445 * attempting to load the TLB entry if necessary.
446 *
447 * @returns VBox status code.
448 * @retval VINF_SUCCESS on success
449 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
450 *
451 * @param pVM The VM handle.
452 * @param GCPhys The address of the guest page.
453 * @param ppTlbe Where to store the pointer to the TLB entry.
454 */
455DECLINLINE(int) pgmPhysPageQueryTlbe(PVM pVM, RTGCPHYS GCPhys, PPPGMPAGEMAPTLBE ppTlbe)
456{
457 int rc;
458 PPGMPAGEMAPTLBE pTlbe = &pVM->pgm.s.CTXSUFF(PhysTlb).aEntries[PGM_PAGEMAPTLB_IDX(GCPhys)];
459 if (pTlbe->GCPhys == (GCPhys & X86_PTE_PAE_PG_MASK))
460 {
461 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PageMapTlbHits));
462 rc = VINF_SUCCESS;
463 }
464 else
465 rc = pgmPhysPageLoadIntoTlb(pVM, GCPhys);
466 *ppTlbe = pTlbe;
467 return rc;
468}
469
470
471/**
472 * Queries the Physical TLB entry for a physical guest page,
473 * attempting to load the TLB entry if necessary.
474 *
475 * @returns VBox status code.
476 * @retval VINF_SUCCESS on success
477 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
478 *
479 * @param pVM The VM handle.
480 * @param pPage Pointer to the PGMPAGE structure corresponding to
481 * GCPhys.
482 * @param GCPhys The address of the guest page.
483 * @param ppTlbe Where to store the pointer to the TLB entry.
484 */
485DECLINLINE(int) pgmPhysPageQueryTlbeWithPage(PVM pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, PPPGMPAGEMAPTLBE ppTlbe)
486{
487 int rc;
488 PPGMPAGEMAPTLBE pTlbe = &pVM->pgm.s.CTXSUFF(PhysTlb).aEntries[PGM_PAGEMAPTLB_IDX(GCPhys)];
489 if (pTlbe->GCPhys == (GCPhys & X86_PTE_PAE_PG_MASK))
490 {
491 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PageMapTlbHits));
492 rc = VINF_SUCCESS;
493 AssertPtr(pTlbe->pv);
494# if !defined(VBOX_WITH_2X_4GB_ADDR_SPACE) || defined(IN_RING3)
495 Assert(!pTlbe->pMap || RT_VALID_PTR(pTlbe->pMap->pv));
496# endif
497 }
498 else
499 rc = pgmPhysPageLoadIntoTlbWithPage(pVM, pPage, GCPhys);
500 *ppTlbe = pTlbe;
501 return rc;
502}
503
504#endif /* !IN_RC */
505
506
507/**
508 * Enables write monitoring for an allocated page.
509 *
510 * The caller is responsible for updating the shadow page tables.
511 *
512 * @param pVM The VM handle.
513 * @param pPage The page to write monitor.
514 * @param GCPhysPage The address of the page.
515 */
516DECLINLINE(void) pgmPhysPageWriteMonitor(PVM pVM, PPGMPAGE pPage, RTGCPHYS GCPhysPage)
517{
518 Assert(PGM_PAGE_GET_STATE(pPage) == PGM_PAGE_STATE_ALLOCATED);
519 PGM_LOCK_ASSERT_OWNER(pVM);
520
521 PGM_PAGE_SET_STATE(pVM, pPage, PGM_PAGE_STATE_WRITE_MONITORED);
522 pVM->pgm.s.cMonitoredPages++;
523
524 /* Large pages must disabled. */
525 if (PGM_PAGE_GET_PDE_TYPE(pPage) == PGM_PAGE_PDE_TYPE_PDE)
526 {
527 PPGMPAGE pFirstPage = pgmPhysGetPage(pVM, GCPhysPage & X86_PDE2M_PAE_PG_MASK);
528 AssertFatal(pFirstPage);
529 if (PGM_PAGE_GET_PDE_TYPE(pFirstPage) == PGM_PAGE_PDE_TYPE_PDE)
530 {
531 PGM_PAGE_SET_PDE_TYPE(pVM, pFirstPage, PGM_PAGE_PDE_TYPE_PDE_DISABLED);
532 pVM->pgm.s.cLargePagesDisabled++;
533 }
534 else
535 Assert(PGM_PAGE_GET_PDE_TYPE(pFirstPage) == PGM_PAGE_PDE_TYPE_PDE_DISABLED);
536 }
537}
538
539
540/**
541 * Checks if the no-execute (NX) feature is active (EFER.NXE=1).
542 *
543 * Only used when the guest is in PAE or long mode. This is inlined so that we
544 * can perform consistency checks in debug builds.
545 *
546 * @returns true if it is, false if it isn't.
547 * @param pVCpu The current CPU.
548 */
549DECL_FORCE_INLINE(bool) pgmGstIsNoExecuteActive(PVMCPU pVCpu)
550{
551 Assert(pVCpu->pgm.s.fNoExecuteEnabled == CPUMIsGuestNXEnabled(pVCpu));
552 Assert(CPUMIsGuestInPAEMode(pVCpu) || CPUMIsGuestInLongMode(pVCpu));
553 return pVCpu->pgm.s.fNoExecuteEnabled;
554}
555
556
557/**
558 * Checks if the page size extension (PSE) is currently enabled (CR4.PSE=1).
559 *
560 * Only used when the guest is in paged 32-bit mode. This is inlined so that
561 * we can perform consistency checks in debug builds.
562 *
563 * @returns true if it is, false if it isn't.
564 * @param pVCpu The current CPU.
565 */
566DECL_FORCE_INLINE(bool) pgmGst32BitIsPageSizeExtActive(PVMCPU pVCpu)
567{
568 Assert(pVCpu->pgm.s.fGst32BitPageSizeExtension == CPUMIsGuestPageSizeExtEnabled(pVCpu));
569 Assert(!CPUMIsGuestInPAEMode(pVCpu));
570 Assert(!CPUMIsGuestInLongMode(pVCpu));
571 return pVCpu->pgm.s.fGst32BitPageSizeExtension;
572}
573
574
575/**
576 * Calculated the guest physical address of the large (4 MB) page in 32 bits paging mode.
577 * Takes PSE-36 into account.
578 *
579 * @returns guest physical address
580 * @param pVM The VM handle.
581 * @param Pde Guest Pde
582 */
583DECLINLINE(RTGCPHYS) pgmGstGet4MBPhysPage(PVM pVM, X86PDE Pde)
584{
585 RTGCPHYS GCPhys = Pde.u & X86_PDE4M_PG_MASK;
586 GCPhys |= (RTGCPHYS)Pde.b.u8PageNoHigh << 32;
587
588 return GCPhys & pVM->pgm.s.GCPhys4MBPSEMask;
589}
590
591
592/**
593 * Gets the address the guest page directory (32-bit paging).
594 *
595 * @returns VBox status code.
596 * @param pVCpu The current CPU.
597 * @param ppPd Where to return the mapping. This is always set.
598 */
599DECLINLINE(int) pgmGstGet32bitPDPtrEx(PVMCPU pVCpu, PX86PD *ppPd)
600{
601#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
602 int rc = pgmRZDynMapGCPageInlined(pVCpu, pVCpu->pgm.s.GCPhysCR3, (void **)ppPd RTLOG_COMMA_SRC_POS);
603 if (RT_FAILURE(rc))
604 {
605 *ppPd = NULL;
606 return rc;
607 }
608#else
609 *ppPd = pVCpu->pgm.s.CTX_SUFF(pGst32BitPd);
610 if (RT_UNLIKELY(!*ppPd))
611 return pgmGstLazyMap32BitPD(pVCpu, ppPd);
612#endif
613 return VINF_SUCCESS;
614}
615
616
617/**
618 * Gets the address the guest page directory (32-bit paging).
619 *
620 * @returns Pointer the page directory entry in question.
621 * @param pVCpu The current CPU.
622 */
623DECLINLINE(PX86PD) pgmGstGet32bitPDPtr(PVMCPU pVCpu)
624{
625#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
626 PX86PD pGuestPD = NULL;
627 int rc = pgmRZDynMapGCPageInlined(pVCpu, pVCpu->pgm.s.GCPhysCR3, (void **)&pGuestPD RTLOG_COMMA_SRC_POS);
628 if (RT_FAILURE(rc))
629 {
630 AssertMsg(rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc));
631 return NULL;
632 }
633#else
634 PX86PD pGuestPD = pVCpu->pgm.s.CTX_SUFF(pGst32BitPd);
635 if (RT_UNLIKELY(!pGuestPD))
636 {
637 int rc = pgmGstLazyMap32BitPD(pVCpu, &pGuestPD);
638 if (RT_FAILURE(rc))
639 return NULL;
640 }
641#endif
642 return pGuestPD;
643}
644
645
646/**
647 * Gets the guest page directory pointer table.
648 *
649 * @returns VBox status code.
650 * @param pVCpu The current CPU.
651 * @param ppPdpt Where to return the mapping. This is always set.
652 */
653DECLINLINE(int) pgmGstGetPaePDPTPtrEx(PVMCPU pVCpu, PX86PDPT *ppPdpt)
654{
655#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
656 int rc = pgmRZDynMapGCPageOffInlined(pVCpu, pVCpu->pgm.s.GCPhysCR3, (void **)ppPdpt RTLOG_COMMA_SRC_POS);
657 if (RT_FAILURE(rc))
658 {
659 *ppPdpt = NULL;
660 return rc;
661 }
662#else
663 *ppPdpt = pVCpu->pgm.s.CTX_SUFF(pGstPaePdpt);
664 if (RT_UNLIKELY(!*ppPdpt))
665 return pgmGstLazyMapPaePDPT(pVCpu, ppPdpt);
666#endif
667 return VINF_SUCCESS;
668}
669
670/**
671 * Gets the guest page directory pointer table.
672 *
673 * @returns Pointer to the page directory in question.
674 * @returns NULL if the page directory is not present or on an invalid page.
675 * @param pVCpu The current CPU.
676 */
677DECLINLINE(PX86PDPT) pgmGstGetPaePDPTPtr(PVMCPU pVCpu)
678{
679 PX86PDPT pGuestPdpt;
680 int rc = pgmGstGetPaePDPTPtrEx(pVCpu, &pGuestPdpt);
681 AssertMsg(RT_SUCCESS(rc) || rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc)); NOREF(rc);
682 return pGuestPdpt;
683}
684
685
686/**
687 * Gets the guest page directory pointer table entry for the specified address.
688 *
689 * @returns Pointer to the page directory in question.
690 * @returns NULL if the page directory is not present or on an invalid page.
691 * @param pVCpu The current CPU
692 * @param GCPtr The address.
693 */
694DECLINLINE(PX86PDPE) pgmGstGetPaePDPEPtr(PVMCPU pVCpu, RTGCPTR GCPtr)
695{
696 AssertGCPtr32(GCPtr);
697
698#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
699 PX86PDPT pGuestPDPT = NULL;
700 int rc = pgmRZDynMapGCPageOffInlined(pVCpu, pVCpu->pgm.s.GCPhysCR3, (void **)&pGuestPDPT RTLOG_COMMA_SRC_POS);
701 AssertRCReturn(rc, NULL);
702#else
703 PX86PDPT pGuestPDPT = pVCpu->pgm.s.CTX_SUFF(pGstPaePdpt);
704 if (RT_UNLIKELY(!pGuestPDPT))
705 {
706 int rc = pgmGstLazyMapPaePDPT(pVCpu, &pGuestPDPT);
707 if (RT_FAILURE(rc))
708 return NULL;
709 }
710#endif
711 return &pGuestPDPT->a[(GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_PAE];
712}
713
714
715/**
716 * Gets the page directory entry for the specified address.
717 *
718 * @returns The page directory entry in question.
719 * @returns A non-present entry if the page directory is not present or on an invalid page.
720 * @param pVCpu The handle of the virtual CPU.
721 * @param GCPtr The address.
722 */
723DECLINLINE(X86PDEPAE) pgmGstGetPaePDE(PVMCPU pVCpu, RTGCPTR GCPtr)
724{
725 AssertGCPtr32(GCPtr);
726 PX86PDPT pGuestPDPT = pgmGstGetPaePDPTPtr(pVCpu);
727 if (RT_LIKELY(pGuestPDPT))
728 {
729 const unsigned iPdpt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_PAE;
730 if ( pGuestPDPT->a[iPdpt].n.u1Present
731 && !(pGuestPDPT->a[iPdpt].u & pVCpu->pgm.s.fGstPaeMbzPdpeMask) )
732 {
733 const unsigned iPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
734#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
735 PX86PDPAE pGuestPD = NULL;
736 int rc = pgmRZDynMapGCPageInlined(pVCpu,
737 pGuestPDPT->a[iPdpt].u & X86_PDPE_PG_MASK,
738 (void **)&pGuestPD
739 RTLOG_COMMA_SRC_POS);
740 if (RT_SUCCESS(rc))
741 return pGuestPD->a[iPD];
742 AssertMsg(rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc));
743#else
744 PX86PDPAE pGuestPD = pVCpu->pgm.s.CTX_SUFF(apGstPaePDs)[iPdpt];
745 if ( !pGuestPD
746 || (pGuestPDPT->a[iPdpt].u & X86_PDPE_PG_MASK) != pVCpu->pgm.s.aGCPhysGstPaePDs[iPdpt])
747 pgmGstLazyMapPaePD(pVCpu, iPdpt, &pGuestPD);
748 if (pGuestPD)
749 return pGuestPD->a[iPD];
750#endif
751 }
752 }
753
754 X86PDEPAE ZeroPde = {0};
755 return ZeroPde;
756}
757
758
759/**
760 * Gets the page directory pointer table entry for the specified address
761 * and returns the index into the page directory
762 *
763 * @returns Pointer to the page directory in question.
764 * @returns NULL if the page directory is not present or on an invalid page.
765 * @param pVCpu The current CPU.
766 * @param GCPtr The address.
767 * @param piPD Receives the index into the returned page directory
768 * @param pPdpe Receives the page directory pointer entry. Optional.
769 */
770DECLINLINE(PX86PDPAE) pgmGstGetPaePDPtr(PVMCPU pVCpu, RTGCPTR GCPtr, unsigned *piPD, PX86PDPE pPdpe)
771{
772 AssertGCPtr32(GCPtr);
773
774 /* The PDPE. */
775 PX86PDPT pGuestPDPT = pgmGstGetPaePDPTPtr(pVCpu);
776 if (RT_UNLIKELY(!pGuestPDPT))
777 return NULL;
778 const unsigned iPdpt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_PAE;
779 if (pPdpe)
780 *pPdpe = pGuestPDPT->a[iPdpt];
781 if (!pGuestPDPT->a[iPdpt].n.u1Present)
782 return NULL;
783 if (RT_UNLIKELY(pVCpu->pgm.s.fGstPaeMbzPdpeMask & pGuestPDPT->a[iPdpt].u))
784 return NULL;
785
786 /* The PDE. */
787#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
788 PX86PDPAE pGuestPD = NULL;
789 int rc = pgmRZDynMapGCPageInlined(pVCpu,
790 pGuestPDPT->a[iPdpt].u & X86_PDPE_PG_MASK,
791 (void **)&pGuestPD
792 RTLOG_COMMA_SRC_POS);
793 if (RT_FAILURE(rc))
794 {
795 AssertMsg(rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc));
796 return NULL;
797 }
798#else
799 PX86PDPAE pGuestPD = pVCpu->pgm.s.CTX_SUFF(apGstPaePDs)[iPdpt];
800 if ( !pGuestPD
801 || (pGuestPDPT->a[iPdpt].u & X86_PDPE_PG_MASK) != pVCpu->pgm.s.aGCPhysGstPaePDs[iPdpt])
802 pgmGstLazyMapPaePD(pVCpu, iPdpt, &pGuestPD);
803#endif
804
805 *piPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
806 return pGuestPD;
807}
808
809#ifndef IN_RC
810
811/**
812 * Gets the page map level-4 pointer for the guest.
813 *
814 * @returns VBox status code.
815 * @param pVCpu The current CPU.
816 * @param ppPml4 Where to return the mapping. Always set.
817 */
818DECLINLINE(int) pgmGstGetLongModePML4PtrEx(PVMCPU pVCpu, PX86PML4 *ppPml4)
819{
820#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
821 int rc = pgmRZDynMapGCPageInlined(pVCpu, pVCpu->pgm.s.GCPhysCR3, (void **)ppPml4 RTLOG_COMMA_SRC_POS);
822 if (RT_FAILURE(rc))
823 {
824 *ppPml4 = NULL;
825 return rc;
826 }
827#else
828 *ppPml4 = pVCpu->pgm.s.CTX_SUFF(pGstAmd64Pml4);
829 if (RT_UNLIKELY(!*ppPml4))
830 return pgmGstLazyMapPml4(pVCpu, ppPml4);
831#endif
832 return VINF_SUCCESS;
833}
834
835
836/**
837 * Gets the page map level-4 pointer for the guest.
838 *
839 * @returns Pointer to the PML4 page.
840 * @param pVCpu The current CPU.
841 */
842DECLINLINE(PX86PML4) pgmGstGetLongModePML4Ptr(PVMCPU pVCpu)
843{
844 PX86PML4 pGuestPml4;
845 int rc = pgmGstGetLongModePML4PtrEx(pVCpu, &pGuestPml4);
846 AssertMsg(RT_SUCCESS(rc) || rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc)); NOREF(rc);
847 return pGuestPml4;
848}
849
850
851/**
852 * Gets the pointer to a page map level-4 entry.
853 *
854 * @returns Pointer to the PML4 entry.
855 * @param pVCpu The current CPU.
856 * @param iPml4 The index.
857 * @remarks Only used by AssertCR3.
858 */
859DECLINLINE(PX86PML4E) pgmGstGetLongModePML4EPtr(PVMCPU pVCpu, unsigned int iPml4)
860{
861#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
862 PX86PML4 pGuestPml4;
863 int rc = pgmRZDynMapGCPageInlined(pVCpu, pVCpu->pgm.s.GCPhysCR3, (void **)&pGuestPml4 RTLOG_COMMA_SRC_POS);
864 AssertRCReturn(rc, NULL);
865#else
866 PX86PML4 pGuestPml4 = pVCpu->pgm.s.CTX_SUFF(pGstAmd64Pml4);
867 if (RT_UNLIKELY(!pGuestPml4))
868 {
869 int rc = pgmGstLazyMapPml4(pVCpu, &pGuestPml4);
870 AssertRCReturn(rc, NULL);
871 }
872#endif
873 return &pGuestPml4->a[iPml4];
874}
875
876
877/**
878 * Gets the page directory entry for the specified address.
879 *
880 * @returns The page directory entry in question.
881 * @returns A non-present entry if the page directory is not present or on an invalid page.
882 * @param pVCpu The current CPU.
883 * @param GCPtr The address.
884 */
885DECLINLINE(X86PDEPAE) pgmGstGetLongModePDE(PVMCPU pVCpu, RTGCPTR64 GCPtr)
886{
887 /*
888 * Note! To keep things simple, ASSUME invalid physical addresses will
889 * cause X86_TRAP_PF_RSVD. This isn't a problem until we start
890 * supporting 52-bit wide physical guest addresses.
891 */
892 PCX86PML4 pGuestPml4 = pgmGstGetLongModePML4Ptr(pVCpu);
893 const unsigned iPml4 = (GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
894 if ( RT_LIKELY(pGuestPml4)
895 && pGuestPml4->a[iPml4].n.u1Present
896 && !(pGuestPml4->a[iPml4].u & pVCpu->pgm.s.fGstAmd64MbzPml4eMask) )
897 {
898 PCX86PDPT pPdptTemp;
899 int rc = PGM_GCPHYS_2_PTR_BY_VMCPU(pVCpu, pGuestPml4->a[iPml4].u & X86_PML4E_PG_MASK, &pPdptTemp);
900 if (RT_SUCCESS(rc))
901 {
902 const unsigned iPdpt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_AMD64;
903 if ( pPdptTemp->a[iPdpt].n.u1Present
904 && !(pPdptTemp->a[iPdpt].u & pVCpu->pgm.s.fGstAmd64MbzPdpeMask) )
905 {
906 PCX86PDPAE pPD;
907 rc = PGM_GCPHYS_2_PTR_BY_VMCPU(pVCpu, pPdptTemp->a[iPdpt].u & X86_PDPE_PG_MASK, &pPD);
908 if (RT_SUCCESS(rc))
909 {
910 const unsigned iPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
911 return pPD->a[iPD];
912 }
913 }
914 }
915 AssertMsg(RT_SUCCESS(rc) || rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc));
916 }
917
918 X86PDEPAE ZeroPde = {0};
919 return ZeroPde;
920}
921
922
923/**
924 * Gets the GUEST page directory pointer for the specified address.
925 *
926 * @returns The page directory in question.
927 * @returns NULL if the page directory is not present or on an invalid page.
928 * @param pVCpu The current CPU.
929 * @param GCPtr The address.
930 * @param ppPml4e Page Map Level-4 Entry (out)
931 * @param pPdpe Page directory pointer table entry (out)
932 * @param piPD Receives the index into the returned page directory
933 */
934DECLINLINE(PX86PDPAE) pgmGstGetLongModePDPtr(PVMCPU pVCpu, RTGCPTR64 GCPtr, PX86PML4E *ppPml4e, PX86PDPE pPdpe, unsigned *piPD)
935{
936 /* The PMLE4. */
937 PX86PML4 pGuestPml4 = pgmGstGetLongModePML4Ptr(pVCpu);
938 if (RT_UNLIKELY(!pGuestPml4))
939 return NULL;
940 const unsigned iPml4 = (GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
941 PCX86PML4E pPml4e = *ppPml4e = &pGuestPml4->a[iPml4];
942 if (!pPml4e->n.u1Present)
943 return NULL;
944 if (RT_UNLIKELY(pPml4e->u & pVCpu->pgm.s.fGstAmd64MbzPml4eMask))
945 return NULL;
946
947 /* The PDPE. */
948 PCX86PDPT pPdptTemp;
949 int rc = PGM_GCPHYS_2_PTR_BY_VMCPU(pVCpu, pPml4e->u & X86_PML4E_PG_MASK, &pPdptTemp);
950 if (RT_FAILURE(rc))
951 {
952 AssertMsg(rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc));
953 return NULL;
954 }
955 const unsigned iPdpt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_AMD64;
956 *pPdpe = pPdptTemp->a[iPdpt];
957 if (!pPdpe->n.u1Present)
958 return NULL;
959 if (RT_UNLIKELY(pPdpe->u & pVCpu->pgm.s.fGstAmd64MbzPdpeMask))
960 return NULL;
961
962 /* The PDE. */
963 PX86PDPAE pPD;
964 rc = PGM_GCPHYS_2_PTR_BY_VMCPU(pVCpu, pPdptTemp->a[iPdpt].u & X86_PDPE_PG_MASK, &pPD);
965 if (RT_FAILURE(rc))
966 {
967 AssertMsg(rc == VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS, ("%Rrc\n", rc));
968 return NULL;
969 }
970
971 *piPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
972 return pPD;
973}
974
975#endif /* !IN_RC */
976
977/**
978 * Gets the shadow page directory, 32-bit.
979 *
980 * @returns Pointer to the shadow 32-bit PD.
981 * @param pVCpu The current CPU.
982 */
983DECLINLINE(PX86PD) pgmShwGet32BitPDPtr(PVMCPU pVCpu)
984{
985 return (PX86PD)PGMPOOL_PAGE_2_PTR_V2(pVCpu->CTX_SUFF(pVM), pVCpu, pVCpu->pgm.s.CTX_SUFF(pShwPageCR3));
986}
987
988
989/**
990 * Gets the shadow page directory entry for the specified address, 32-bit.
991 *
992 * @returns Shadow 32-bit PDE.
993 * @param pVCpu The current CPU.
994 * @param GCPtr The address.
995 */
996DECLINLINE(X86PDE) pgmShwGet32BitPDE(PVMCPU pVCpu, RTGCPTR GCPtr)
997{
998 const unsigned iPd = (GCPtr >> X86_PD_SHIFT) & X86_PD_MASK;
999
1000 PX86PD pShwPde = pgmShwGet32BitPDPtr(pVCpu);
1001 if (!pShwPde)
1002 {
1003 X86PDE ZeroPde = {0};
1004 return ZeroPde;
1005 }
1006 return pShwPde->a[iPd];
1007}
1008
1009
1010/**
1011 * Gets the pointer to the shadow page directory entry for the specified
1012 * address, 32-bit.
1013 *
1014 * @returns Pointer to the shadow 32-bit PDE.
1015 * @param pVCpu The current CPU.
1016 * @param GCPtr The address.
1017 */
1018DECLINLINE(PX86PDE) pgmShwGet32BitPDEPtr(PVMCPU pVCpu, RTGCPTR GCPtr)
1019{
1020 const unsigned iPd = (GCPtr >> X86_PD_SHIFT) & X86_PD_MASK;
1021
1022 PX86PD pPde = pgmShwGet32BitPDPtr(pVCpu);
1023 AssertReturn(pPde, NULL);
1024 return &pPde->a[iPd];
1025}
1026
1027
1028/**
1029 * Gets the shadow page pointer table, PAE.
1030 *
1031 * @returns Pointer to the shadow PAE PDPT.
1032 * @param pVCpu The current CPU.
1033 */
1034DECLINLINE(PX86PDPT) pgmShwGetPaePDPTPtr(PVMCPU pVCpu)
1035{
1036 return (PX86PDPT)PGMPOOL_PAGE_2_PTR_V2(pVCpu->CTX_SUFF(pVM), pVCpu, pVCpu->pgm.s.CTX_SUFF(pShwPageCR3));
1037}
1038
1039
1040/**
1041 * Gets the shadow page directory for the specified address, PAE.
1042 *
1043 * @returns Pointer to the shadow PD.
1044 * @param pVCpu The current CPU.
1045 * @param GCPtr The address.
1046 */
1047DECLINLINE(PX86PDPAE) pgmShwGetPaePDPtr(PVMCPU pVCpu, RTGCPTR GCPtr)
1048{
1049 const unsigned iPdpt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_PAE;
1050 PX86PDPT pPdpt = pgmShwGetPaePDPTPtr(pVCpu);
1051
1052 if (!pPdpt->a[iPdpt].n.u1Present)
1053 return NULL;
1054
1055 /* Fetch the pgm pool shadow descriptor. */
1056 PVM pVM = pVCpu->CTX_SUFF(pVM);
1057 PPGMPOOLPAGE pShwPde = pgmPoolGetPage(pVM->pgm.s.CTX_SUFF(pPool), pPdpt->a[iPdpt].u & X86_PDPE_PG_MASK);
1058 AssertReturn(pShwPde, NULL);
1059
1060 return (PX86PDPAE)PGMPOOL_PAGE_2_PTR_V2(pVM, pVCpu, pShwPde);
1061}
1062
1063
1064/**
1065 * Gets the shadow page directory for the specified address, PAE.
1066 *
1067 * @returns Pointer to the shadow PD.
1068 * @param pVCpu The current CPU.
1069 * @param GCPtr The address.
1070 */
1071DECLINLINE(PX86PDPAE) pgmShwGetPaePDPtr(PVMCPU pVCpu, PX86PDPT pPdpt, RTGCPTR GCPtr)
1072{
1073 const unsigned iPdpt = (GCPtr >> X86_PDPT_SHIFT) & X86_PDPT_MASK_PAE;
1074
1075 if (!pPdpt->a[iPdpt].n.u1Present)
1076 return NULL;
1077
1078 /* Fetch the pgm pool shadow descriptor. */
1079 PVM pVM = pVCpu->CTX_SUFF(pVM);
1080 PPGMPOOLPAGE pShwPde = pgmPoolGetPage(pVM->pgm.s.CTX_SUFF(pPool), pPdpt->a[iPdpt].u & X86_PDPE_PG_MASK);
1081 AssertReturn(pShwPde, NULL);
1082
1083 return (PX86PDPAE)PGMPOOL_PAGE_2_PTR_V2(pVM, pVCpu, pShwPde);
1084}
1085
1086
1087/**
1088 * Gets the shadow page directory entry, PAE.
1089 *
1090 * @returns PDE.
1091 * @param pVCpu The current CPU.
1092 * @param GCPtr The address.
1093 */
1094DECLINLINE(X86PDEPAE) pgmShwGetPaePDE(PVMCPU pVCpu, RTGCPTR GCPtr)
1095{
1096 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
1097
1098 PX86PDPAE pShwPde = pgmShwGetPaePDPtr(pVCpu, GCPtr);
1099 if (!pShwPde)
1100 {
1101 X86PDEPAE ZeroPde = {0};
1102 return ZeroPde;
1103 }
1104 return pShwPde->a[iPd];
1105}
1106
1107
1108/**
1109 * Gets the pointer to the shadow page directory entry for an address, PAE.
1110 *
1111 * @returns Pointer to the PDE.
1112 * @param pVCpu The current CPU.
1113 * @param GCPtr The address.
1114 * @remarks Only used by AssertCR3.
1115 */
1116DECLINLINE(PX86PDEPAE) pgmShwGetPaePDEPtr(PVMCPU pVCpu, RTGCPTR GCPtr)
1117{
1118 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
1119
1120 PX86PDPAE pPde = pgmShwGetPaePDPtr(pVCpu, GCPtr);
1121 AssertReturn(pPde, NULL);
1122 return &pPde->a[iPd];
1123}
1124
1125#ifndef IN_RC
1126
1127/**
1128 * Gets the shadow page map level-4 pointer.
1129 *
1130 * @returns Pointer to the shadow PML4.
1131 * @param pVCpu The current CPU.
1132 */
1133DECLINLINE(PX86PML4) pgmShwGetLongModePML4Ptr(PVMCPU pVCpu)
1134{
1135 return (PX86PML4)PGMPOOL_PAGE_2_PTR_V2(pVCpu->CTX_SUFF(pVM), pVCpu, pVCpu->pgm.s.CTX_SUFF(pShwPageCR3));
1136}
1137
1138
1139/**
1140 * Gets the shadow page map level-4 entry for the specified address.
1141 *
1142 * @returns The entry.
1143 * @param pVCpu The current CPU.
1144 * @param GCPtr The address.
1145 */
1146DECLINLINE(X86PML4E) pgmShwGetLongModePML4E(PVMCPU pVCpu, RTGCPTR GCPtr)
1147{
1148 const unsigned iPml4 = ((RTGCUINTPTR64)GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
1149 PX86PML4 pShwPml4 = pgmShwGetLongModePML4Ptr(pVCpu);
1150
1151 if (!pShwPml4)
1152 {
1153 X86PML4E ZeroPml4e = {0};
1154 return ZeroPml4e;
1155 }
1156 return pShwPml4->a[iPml4];
1157}
1158
1159
1160/**
1161 * Gets the pointer to the specified shadow page map level-4 entry.
1162 *
1163 * @returns The entry.
1164 * @param pVCpu The current CPU.
1165 * @param iPml4 The PML4 index.
1166 */
1167DECLINLINE(PX86PML4E) pgmShwGetLongModePML4EPtr(PVMCPU pVCpu, unsigned int iPml4)
1168{
1169 PX86PML4 pShwPml4 = pgmShwGetLongModePML4Ptr(pVCpu);
1170 if (!pShwPml4)
1171 return NULL;
1172 return &pShwPml4->a[iPml4];
1173}
1174
1175#endif /* !IN_RC */
1176
1177
1178/**
1179 * Cached physical handler lookup.
1180 *
1181 * @returns Physical handler covering @a GCPhys.
1182 * @param pVM The VM handle.
1183 * @param GCPhys The lookup address.
1184 */
1185DECLINLINE(PPGMPHYSHANDLER) pgmHandlerPhysicalLookup(PVM pVM, RTGCPHYS GCPhys)
1186{
1187 PPGMPHYSHANDLER pHandler = pVM->pgm.s.CTX_SUFF(pLastPhysHandler);
1188 if ( pHandler
1189 && GCPhys >= pHandler->Core.Key
1190 && GCPhys < pHandler->Core.KeyLast)
1191 {
1192 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysHandlerLookupHits));
1193 return pHandler;
1194 }
1195
1196 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysHandlerLookupMisses));
1197 pHandler = (PPGMPHYSHANDLER)RTAvlroGCPhysRangeGet(&pVM->pgm.s.CTX_SUFF(pTrees)->PhysHandlers, GCPhys);
1198 if (pHandler)
1199 pVM->pgm.s.CTX_SUFF(pLastPhysHandler) = pHandler;
1200 return pHandler;
1201}
1202
1203
1204/**
1205 * Gets the page state for a physical handler.
1206 *
1207 * @returns The physical handler page state.
1208 * @param pCur The physical handler in question.
1209 */
1210DECLINLINE(unsigned) pgmHandlerPhysicalCalcState(PPGMPHYSHANDLER pCur)
1211{
1212 switch (pCur->enmType)
1213 {
1214 case PGMPHYSHANDLERTYPE_PHYSICAL_WRITE:
1215 return PGM_PAGE_HNDL_PHYS_STATE_WRITE;
1216
1217 case PGMPHYSHANDLERTYPE_MMIO:
1218 case PGMPHYSHANDLERTYPE_PHYSICAL_ALL:
1219 return PGM_PAGE_HNDL_PHYS_STATE_ALL;
1220
1221 default:
1222 AssertFatalMsgFailed(("Invalid type %d\n", pCur->enmType));
1223 }
1224}
1225
1226
1227/**
1228 * Gets the page state for a virtual handler.
1229 *
1230 * @returns The virtual handler page state.
1231 * @param pCur The virtual handler in question.
1232 * @remarks This should never be used on a hypervisor access handler.
1233 */
1234DECLINLINE(unsigned) pgmHandlerVirtualCalcState(PPGMVIRTHANDLER pCur)
1235{
1236 switch (pCur->enmType)
1237 {
1238 case PGMVIRTHANDLERTYPE_WRITE:
1239 return PGM_PAGE_HNDL_VIRT_STATE_WRITE;
1240 case PGMVIRTHANDLERTYPE_ALL:
1241 return PGM_PAGE_HNDL_VIRT_STATE_ALL;
1242 default:
1243 AssertFatalMsgFailed(("Invalid type %d\n", pCur->enmType));
1244 }
1245}
1246
1247
1248/**
1249 * Clears one physical page of a virtual handler.
1250 *
1251 * @param pVM The VM handle.
1252 * @param pCur Virtual handler structure.
1253 * @param iPage Physical page index.
1254 *
1255 * @remark Only used when PGM_SYNC_UPDATE_PAGE_BIT_VIRTUAL is being set, so no
1256 * need to care about other handlers in the same page.
1257 */
1258DECLINLINE(void) pgmHandlerVirtualClearPage(PVM pVM, PPGMVIRTHANDLER pCur, unsigned iPage)
1259{
1260 const PPGMPHYS2VIRTHANDLER pPhys2Virt = &pCur->aPhysToVirt[iPage];
1261
1262 /*
1263 * Remove the node from the tree (it's supposed to be in the tree if we get here!).
1264 */
1265#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
1266 AssertReleaseMsg(pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_IN_TREE,
1267 ("pPhys2Virt=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
1268 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias));
1269#endif
1270 if (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_IS_HEAD)
1271 {
1272 /* We're the head of the alias chain. */
1273 PPGMPHYS2VIRTHANDLER pRemove = (PPGMPHYS2VIRTHANDLER)RTAvlroGCPhysRemove(&pVM->pgm.s.CTX_SUFF(pTrees)->PhysToVirtHandlers, pPhys2Virt->Core.Key); NOREF(pRemove);
1274#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
1275 AssertReleaseMsg(pRemove != NULL,
1276 ("pPhys2Virt=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
1277 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias));
1278 AssertReleaseMsg(pRemove == pPhys2Virt,
1279 ("wanted: pPhys2Virt=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n"
1280 " got: pRemove=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
1281 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias,
1282 pRemove, pRemove->Core.Key, pRemove->Core.KeyLast, pRemove->offVirtHandler, pRemove->offNextAlias));
1283#endif
1284 if (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK)
1285 {
1286 /* Insert the next list in the alias chain into the tree. */
1287 PPGMPHYS2VIRTHANDLER pNext = (PPGMPHYS2VIRTHANDLER)((intptr_t)pPhys2Virt + (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK));
1288#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
1289 AssertReleaseMsg(pNext->offNextAlias & PGMPHYS2VIRTHANDLER_IN_TREE,
1290 ("pNext=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
1291 pNext, pNext->Core.Key, pNext->Core.KeyLast, pNext->offVirtHandler, pNext->offNextAlias));
1292#endif
1293 pNext->offNextAlias |= PGMPHYS2VIRTHANDLER_IS_HEAD;
1294 bool fRc = RTAvlroGCPhysInsert(&pVM->pgm.s.CTX_SUFF(pTrees)->PhysToVirtHandlers, &pNext->Core);
1295 AssertRelease(fRc);
1296 }
1297 }
1298 else
1299 {
1300 /* Locate the previous node in the alias chain. */
1301 PPGMPHYS2VIRTHANDLER pPrev = (PPGMPHYS2VIRTHANDLER)RTAvlroGCPhysGet(&pVM->pgm.s.CTX_SUFF(pTrees)->PhysToVirtHandlers, pPhys2Virt->Core.Key);
1302#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
1303 AssertReleaseMsg(pPrev != pPhys2Virt,
1304 ("pPhys2Virt=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32} pPrev=%p\n",
1305 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias, pPrev));
1306#endif
1307 for (;;)
1308 {
1309 PPGMPHYS2VIRTHANDLER pNext = (PPGMPHYS2VIRTHANDLER)((intptr_t)pPrev + (pPrev->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK));
1310 if (pNext == pPhys2Virt)
1311 {
1312 /* unlink. */
1313 LogFlow(("pgmHandlerVirtualClearPage: removed %p:{.offNextAlias=%#RX32} from alias chain. prev %p:{.offNextAlias=%#RX32} [%RGp-%RGp]\n",
1314 pPhys2Virt, pPhys2Virt->offNextAlias, pPrev, pPrev->offNextAlias, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast));
1315 if (!(pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK))
1316 pPrev->offNextAlias &= ~PGMPHYS2VIRTHANDLER_OFF_MASK;
1317 else
1318 {
1319 PPGMPHYS2VIRTHANDLER pNewNext = (PPGMPHYS2VIRTHANDLER)((intptr_t)pPhys2Virt + (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK));
1320 pPrev->offNextAlias = ((intptr_t)pNewNext - (intptr_t)pPrev)
1321 | (pPrev->offNextAlias & ~PGMPHYS2VIRTHANDLER_OFF_MASK);
1322 }
1323 break;
1324 }
1325
1326 /* next */
1327 if (pNext == pPrev)
1328 {
1329#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
1330 AssertReleaseMsg(pNext != pPrev,
1331 ("pPhys2Virt=%p:{.Core.Key=%RGp, .Core.KeyLast=%RGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32} pPrev=%p\n",
1332 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias, pPrev));
1333#endif
1334 break;
1335 }
1336 pPrev = pNext;
1337 }
1338 }
1339 Log2(("PHYS2VIRT: Removing %RGp-%RGp %#RX32 %s\n",
1340 pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offNextAlias, R3STRING(pCur->pszDesc)));
1341 pPhys2Virt->offNextAlias = 0;
1342 pPhys2Virt->Core.KeyLast = NIL_RTGCPHYS; /* require reinsert */
1343
1344 /*
1345 * Clear the ram flags for this page.
1346 */
1347 PPGMPAGE pPage = pgmPhysGetPage(pVM, pPhys2Virt->Core.Key);
1348 AssertReturnVoid(pPage);
1349 PGM_PAGE_SET_HNDL_VIRT_STATE(pPage, PGM_PAGE_HNDL_VIRT_STATE_NONE);
1350}
1351
1352
1353/**
1354 * Internal worker for finding a 'in-use' shadow page give by it's physical address.
1355 *
1356 * @returns Pointer to the shadow page structure.
1357 * @param pPool The pool.
1358 * @param idx The pool page index.
1359 */
1360DECLINLINE(PPGMPOOLPAGE) pgmPoolGetPageByIdx(PPGMPOOL pPool, unsigned idx)
1361{
1362 AssertFatalMsg(idx >= PGMPOOL_IDX_FIRST && idx < pPool->cCurPages, ("idx=%d\n", idx));
1363 return &pPool->aPages[idx];
1364}
1365
1366
1367/**
1368 * Clear references to guest physical memory.
1369 *
1370 * @param pPool The pool.
1371 * @param pPoolPage The pool page.
1372 * @param pPhysPage The physical guest page tracking structure.
1373 * @param iPte Shadow PTE index
1374 */
1375DECLINLINE(void) pgmTrackDerefGCPhys(PPGMPOOL pPool, PPGMPOOLPAGE pPoolPage, PPGMPAGE pPhysPage, uint16_t iPte)
1376{
1377 /*
1378 * Just deal with the simple case here.
1379 */
1380# ifdef VBOX_STRICT
1381 PVM pVM = pPool->CTX_SUFF(pVM); NOREF(pVM);
1382# endif
1383# ifdef LOG_ENABLED
1384 const unsigned uOrg = PGM_PAGE_GET_TRACKING(pPhysPage);
1385# endif
1386 const unsigned cRefs = PGM_PAGE_GET_TD_CREFS(pPhysPage);
1387 if (cRefs == 1)
1388 {
1389 Assert(pPoolPage->idx == PGM_PAGE_GET_TD_IDX(pPhysPage));
1390 Assert(iPte == PGM_PAGE_GET_PTE_INDEX(pPhysPage));
1391 /* Invalidate the tracking data. */
1392 PGM_PAGE_SET_TRACKING(pVM, pPhysPage, 0);
1393 }
1394 else
1395 pgmPoolTrackPhysExtDerefGCPhys(pPool, pPoolPage, pPhysPage, iPte);
1396 Log2(("pgmTrackDerefGCPhys: %x -> %x pPhysPage=%R[pgmpage]\n", uOrg, PGM_PAGE_GET_TRACKING(pPhysPage), pPhysPage ));
1397}
1398
1399
1400/**
1401 * Moves the page to the head of the age list.
1402 *
1403 * This is done when the cached page is used in one way or another.
1404 *
1405 * @param pPool The pool.
1406 * @param pPage The cached page.
1407 */
1408DECLINLINE(void) pgmPoolCacheUsed(PPGMPOOL pPool, PPGMPOOLPAGE pPage)
1409{
1410 PGM_LOCK_ASSERT_OWNER(pPool->CTX_SUFF(pVM));
1411
1412 /*
1413 * Move to the head of the age list.
1414 */
1415 if (pPage->iAgePrev != NIL_PGMPOOL_IDX)
1416 {
1417 /* unlink */
1418 pPool->aPages[pPage->iAgePrev].iAgeNext = pPage->iAgeNext;
1419 if (pPage->iAgeNext != NIL_PGMPOOL_IDX)
1420 pPool->aPages[pPage->iAgeNext].iAgePrev = pPage->iAgePrev;
1421 else
1422 pPool->iAgeTail = pPage->iAgePrev;
1423
1424 /* insert at head */
1425 pPage->iAgePrev = NIL_PGMPOOL_IDX;
1426 pPage->iAgeNext = pPool->iAgeHead;
1427 Assert(pPage->iAgeNext != NIL_PGMPOOL_IDX); /* we would've already been head then */
1428 pPool->iAgeHead = pPage->idx;
1429 pPool->aPages[pPage->iAgeNext].iAgePrev = pPage->idx;
1430 }
1431}
1432
1433/**
1434 * Locks a page to prevent flushing (important for cr3 root pages or shadow pae pd pages).
1435 *
1436 * @param pVM VM Handle.
1437 * @param pPage PGM pool page
1438 */
1439DECLINLINE(void) pgmPoolLockPage(PPGMPOOL pPool, PPGMPOOLPAGE pPage)
1440{
1441 PGM_LOCK_ASSERT_OWNER(pPool->CTX_SUFF(pVM));
1442 ASMAtomicIncU32(&pPage->cLocked);
1443}
1444
1445
1446/**
1447 * Unlocks a page to allow flushing again
1448 *
1449 * @param pVM VM Handle.
1450 * @param pPage PGM pool page
1451 */
1452DECLINLINE(void) pgmPoolUnlockPage(PPGMPOOL pPool, PPGMPOOLPAGE pPage)
1453{
1454 PGM_LOCK_ASSERT_OWNER(pPool->CTX_SUFF(pVM));
1455 Assert(pPage->cLocked);
1456 ASMAtomicDecU32(&pPage->cLocked);
1457}
1458
1459
1460/**
1461 * Checks if the page is locked (e.g. the active CR3 or one of the four PDs of a PAE PDPT)
1462 *
1463 * @returns VBox status code.
1464 * @param pPage PGM pool page
1465 */
1466DECLINLINE(bool) pgmPoolIsPageLocked(PPGMPOOLPAGE pPage)
1467{
1468 if (pPage->cLocked)
1469 {
1470 LogFlow(("pgmPoolIsPageLocked found root page %d\n", pPage->enmKind));
1471 if (pPage->cModifications)
1472 pPage->cModifications = 1; /* reset counter (can't use 0, or else it will be reinserted in the modified list) */
1473 return true;
1474 }
1475 return false;
1476}
1477
1478
1479/**
1480 * Tells if mappings are to be put into the shadow page table or not.
1481 *
1482 * @returns boolean result
1483 * @param pVM VM handle.
1484 */
1485DECL_FORCE_INLINE(bool) pgmMapAreMappingsEnabled(PVM pVM)
1486{
1487#ifdef PGM_WITHOUT_MAPPINGS
1488 /* There are no mappings in VT-x and AMD-V mode. */
1489 Assert(pVM->pgm.s.fMappingsDisabled);
1490 return false;
1491#else
1492 return !pVM->pgm.s.fMappingsDisabled;
1493#endif
1494}
1495
1496
1497/**
1498 * Checks if the mappings are floating and enabled.
1499 *
1500 * @returns true / false.
1501 * @param pVM The VM handle.
1502 */
1503DECL_FORCE_INLINE(bool) pgmMapAreMappingsFloating(PVM pVM)
1504{
1505#ifdef PGM_WITHOUT_MAPPINGS
1506 /* There are no mappings in VT-x and AMD-V mode. */
1507 Assert(pVM->pgm.s.fMappingsDisabled);
1508 return false;
1509#else
1510 return !pVM->pgm.s.fMappingsDisabled
1511 && !pVM->pgm.s.fMappingsFixed;
1512#endif
1513}
1514
1515/** @} */
1516
1517#endif
1518
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