VirtualBox

source: vbox/trunk/src/VBox/VMM/PGMInternal.h@ 5011

Last change on this file since 5011 was 4978, checked in by vboxsync, 17 years ago

Backed out most of 24659.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 129.5 KB
Line 
1/* $Id: PGMInternal.h 4978 2007-09-22 00:02:40Z vboxsync $ */
2/** @file
3 * PGM - Internal header file.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek 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
18#ifndef ___PGMInternal_h
19#define ___PGMInternal_h
20
21#include <VBox/cdefs.h>
22#include <VBox/types.h>
23#include <VBox/err.h>
24#include <VBox/stam.h>
25#include <VBox/param.h>
26#include <VBox/vmm.h>
27#include <VBox/mm.h>
28#include <VBox/pdmcritsect.h>
29#include <VBox/pdmapi.h>
30#include <VBox/dis.h>
31#include <VBox/dbgf.h>
32#include <VBox/log.h>
33#include <iprt/avl.h>
34#include <iprt/assert.h>
35#include <iprt/critsect.h>
36
37#if !defined(IN_PGM_R3) && !defined(IN_PGM_R0) && !defined(IN_PGM_GC)
38# error "Not in PGM! This is an internal header!"
39#endif
40
41
42/** @defgroup grp_pgm_int Internals
43 * @ingroup grp_pgm
44 * @internal
45 * @{
46 */
47
48
49/** @name PGM Compile Time Config
50 * @{
51 */
52
53/**
54 * Solve page is out of sync issues inside Guest Context (in PGMGC.cpp).
55 * Comment it if it will break something.
56 */
57#define PGM_OUT_OF_SYNC_IN_GC
58
59/**
60 * Virtualize the dirty bit
61 * This also makes a half-hearted attempt at the accessed bit. For full
62 * accessed bit virtualization define PGM_SYNC_ACCESSED_BIT.
63 */
64#define PGM_SYNC_DIRTY_BIT
65
66/**
67 * Fully virtualize the accessed bit.
68 * @remark This requires SYNC_DIRTY_ACCESSED_BITS to be defined!
69 */
70#define PGM_SYNC_ACCESSED_BIT
71
72/**
73 * Check and skip global PDEs for non-global flushes
74 */
75#define PGM_SKIP_GLOBAL_PAGEDIRS_ON_NONGLOBAL_FLUSH
76
77/**
78 * Sync N pages instead of a whole page table
79 */
80#define PGM_SYNC_N_PAGES
81
82/**
83 * Number of pages to sync during a page fault
84 *
85 * When PGMPOOL_WITH_GCPHYS_TRACKING is enabled using high values here
86 * causes a lot of unnecessary extents and also is slower than taking more \#PFs.
87 */
88#define PGM_SYNC_NR_PAGES 8
89
90/**
91 * Number of PGMPhysRead/Write cache entries (must be <= sizeof(uint64_t))
92 */
93#define PGM_MAX_PHYSCACHE_ENTRIES 64
94#define PGM_MAX_PHYSCACHE_ENTRIES_MASK (PGM_MAX_PHYSCACHE_ENTRIES-1)
95
96/**
97 * Enable caching of PGMR3PhysRead/WriteByte/Word/Dword
98 */
99#define PGM_PHYSMEMACCESS_CACHING
100
101/*
102 * Assert Sanity.
103 */
104#if defined(PGM_SYNC_ACCESSED_BIT) && !defined(PGM_SYNC_DIRTY_BIT)
105# error "PGM_SYNC_ACCESSED_BIT requires PGM_SYNC_DIRTY_BIT!"
106#endif
107
108/** @def PGMPOOL_WITH_CACHE
109 * Enable agressive caching using the page pool.
110 *
111 * This requires PGMPOOL_WITH_USER_TRACKING and PGMPOOL_WITH_MONITORING.
112 */
113#define PGMPOOL_WITH_CACHE
114
115/** @def PGMPOOL_WITH_MIXED_PT_CR3
116 * When defined, we'll deal with 'uncachable' pages.
117 */
118#ifdef PGMPOOL_WITH_CACHE
119# define PGMPOOL_WITH_MIXED_PT_CR3
120#endif
121
122/** @def PGMPOOL_WITH_MONITORING
123 * Monitor the guest pages which are shadowed.
124 * When this is enabled, PGMPOOL_WITH_CACHE or PGMPOOL_WITH_GCPHYS_TRACKING must
125 * be enabled as well.
126 * @remark doesn't really work without caching now. (Mixed PT/CR3 change.)
127 */
128#ifdef PGMPOOL_WITH_CACHE
129# define PGMPOOL_WITH_MONITORING
130#endif
131
132/** @def PGMPOOL_WITH_GCPHYS_TRACKING
133 * Tracking the of shadow pages mapping guest physical pages.
134 *
135 * This is very expensive, the current cache prototype is trying to figure out
136 * whether it will be acceptable with an agressive caching policy.
137 */
138#if defined(PGMPOOL_WITH_CACHE) || defined(PGMPOOL_WITH_MONITORING)
139# define PGMPOOL_WITH_GCPHYS_TRACKING
140#endif
141
142/** @def PGMPOOL_WITH_USER_TRACKNG
143 * Tracking users of shadow pages. This is required for the linking of shadow page
144 * tables and physical guest addresses.
145 */
146#if defined(PGMPOOL_WITH_GCPHYS_TRACKING) || defined(PGMPOOL_WITH_CACHE) || defined(PGMPOOL_WITH_MONITORING)
147# define PGMPOOL_WITH_USER_TRACKING
148#endif
149
150/** @def PGMPOOL_CFG_MAX_GROW
151 * The maximum number of pages to add to the pool in one go.
152 */
153#define PGMPOOL_CFG_MAX_GROW (_256K >> PAGE_SHIFT)
154
155/** @def VBOX_STRICT_PGM_HANDLER_VIRTUAL
156 * Enables some extra assertions for virtual handlers (mainly phys2virt related).
157 */
158#ifdef VBOX_STRICT
159# define VBOX_STRICT_PGM_HANDLER_VIRTUAL
160#endif
161/** @} */
162
163
164/** @name PDPTR and PML4 flags.
165 * These are placed in the three bits available for system programs in
166 * the PDPTR and PML4 entries.
167 * @{ */
168/** The entry is a permanent one and it's must always be present.
169 * Never free such an entry. */
170#define PGM_PLXFLAGS_PERMANENT BIT64(10)
171/** @} */
172
173/** @name Page directory flags.
174 * These are placed in the three bits available for system programs in
175 * the page directory entries.
176 * @{ */
177/** Mapping (hypervisor allocated pagetable). */
178#define PGM_PDFLAGS_MAPPING BIT64(10)
179/** Made read-only to facilitate dirty bit tracking. */
180#define PGM_PDFLAGS_TRACK_DIRTY BIT64(11)
181/** @} */
182
183/** @name Page flags.
184 * These are placed in the three bits available for system programs in
185 * the page entries.
186 * @{ */
187/** Made read-only to facilitate dirty bit tracking. */
188#define PGM_PTFLAGS_TRACK_DIRTY BIT64(9)
189
190#ifndef PGM_PTFLAGS_CSAM_VALIDATED
191/** Scanned and approved by CSAM (tm).
192 * NOTE: Must be identical to the one defined in CSAMInternal.h!!
193 * @todo Move PGM_PTFLAGS_* and PGM_PDFLAGS_* to VBox/pgm.h. */
194#define PGM_PTFLAGS_CSAM_VALIDATED BIT64(11)
195#endif
196/** @} */
197
198/** @name Defines used to indicate the shadow and guest paging in the templates.
199 * @{ */
200#define PGM_TYPE_REAL 1
201#define PGM_TYPE_PROT 2
202#define PGM_TYPE_32BIT 3
203#define PGM_TYPE_PAE 4
204#define PGM_TYPE_AMD64 5
205/** @} */
206
207/** Macro for checking if the guest is using paging.
208 * @param uType PGM_TYPE_*
209 * @remark ASSUMES certain order of the PGM_TYPE_* values.
210 */
211#define PGM_WITH_PAGING(uType) ((uType) >= PGM_TYPE_32BIT)
212
213
214/** @def PGM_HCPHYS_2_PTR
215 * Maps a HC physical page pool address to a virtual address.
216 *
217 * @returns VBox status code.
218 * @param pVM The VM handle.
219 * @param HCPhys The HC physical address to map to a virtual one.
220 * @param ppv Where to store the virtual address. No need to cast this.
221 *
222 * @remark In GC this uses PGMGCDynMapHCPage(), so it will consume of the
223 * small page window employeed by that function. Be careful.
224 * @remark There is no need to assert on the result.
225 */
226#ifdef IN_GC
227# define PGM_HCPHYS_2_PTR(pVM, HCPhys, ppv) PGMGCDynMapHCPage(pVM, HCPhys, (void **)(ppv))
228#else
229# define PGM_HCPHYS_2_PTR(pVM, HCPhys, ppv) MMPagePhys2PageEx(pVM, HCPhys, (void **)(ppv))
230#endif
231
232/** @def PGM_GCPHYS_2_PTR
233 * Maps a GC physical page address to a virtual address.
234 *
235 * @returns VBox status code.
236 * @param pVM The VM handle.
237 * @param GCPhys The GC physical address to map to a virtual one.
238 * @param ppv Where to store the virtual address. No need to cast this.
239 *
240 * @remark In GC this uses PGMGCDynMapGCPage(), so it will consume of the
241 * small page window employeed by that function. Be careful.
242 * @remark There is no need to assert on the result.
243 */
244#ifdef IN_GC
245# define PGM_GCPHYS_2_PTR(pVM, GCPhys, ppv) PGMGCDynMapGCPage(pVM, GCPhys, (void **)(ppv))
246#else
247# define PGM_GCPHYS_2_PTR(pVM, GCPhys, ppv) PGMPhysGCPhys2HCPtr(pVM, GCPhys, 1 /* one page only */, (void **)(ppv)) /** @todo this isn't asserting, use PGMRamGCPhys2HCPtr! */
248#endif
249
250/** @def PGM_GCPHYS_2_PTR_EX
251 * Maps a unaligned GC physical page address to a virtual address.
252 *
253 * @returns VBox status code.
254 * @param pVM The VM handle.
255 * @param GCPhys The GC physical address to map to a virtual one.
256 * @param ppv Where to store the virtual address. No need to cast this.
257 *
258 * @remark In GC this uses PGMGCDynMapGCPage(), so it will consume of the
259 * small page window employeed by that function. Be careful.
260 * @remark There is no need to assert on the result.
261 */
262#ifdef IN_GC
263# define PGM_GCPHYS_2_PTR_EX(pVM, GCPhys, ppv) PGMGCDynMapGCPageEx(pVM, GCPhys, (void **)(ppv))
264#else
265# define PGM_GCPHYS_2_PTR_EX(pVM, GCPhys, ppv) PGMPhysGCPhys2HCPtr(pVM, GCPhys, 1 /* one page only */, (void **)(ppv)) /** @todo this isn't asserting, use PGMRamGCPhys2HCPtr! */
266#endif
267
268/** @def PGM_INVL_PG
269 * Invalidates a page when in GC does nothing in HC.
270 *
271 * @param GCVirt The virtual address of the page to invalidate.
272 */
273#ifdef IN_GC
274# define PGM_INVL_PG(GCVirt) ASMInvalidatePage((void *)(GCVirt))
275#else
276# define PGM_INVL_PG(GCVirt) ((void)0)
277#endif
278
279/** @def PGM_INVL_BIG_PG
280 * Invalidates a 4MB page directory entry when in GC does nothing in HC.
281 *
282 * @param GCVirt The virtual address within the page directory to invalidate.
283 */
284#ifdef IN_GC
285# define PGM_INVL_BIG_PG(GCVirt) ASMReloadCR3()
286#else
287# define PGM_INVL_BIG_PG(GCVirt) ((void)0)
288#endif
289
290/** @def PGM_INVL_GUEST_TLBS()
291 * Invalidates all guest TLBs.
292 */
293#ifdef IN_GC
294# define PGM_INVL_GUEST_TLBS() ASMReloadCR3()
295#else
296# define PGM_INVL_GUEST_TLBS() ((void)0)
297#endif
298
299
300/**
301 * Structure for tracking GC Mappings.
302 *
303 * This structure is used by linked list in both GC and HC.
304 */
305typedef struct PGMMAPPING
306{
307 /** Pointer to next entry. */
308 R3PTRTYPE(struct PGMMAPPING *) pNextR3;
309 /** Pointer to next entry. */
310 R0PTRTYPE(struct PGMMAPPING *) pNextR0;
311 /** Pointer to next entry. */
312 GCPTRTYPE(struct PGMMAPPING *) pNextGC;
313 /** Start Virtual address. */
314 RTGCUINTPTR GCPtr;
315 /** Last Virtual address (inclusive). */
316 RTGCUINTPTR GCPtrLast;
317 /** Range size (bytes). */
318 RTGCUINTPTR cb;
319 /** Pointer to relocation callback function. */
320 R3PTRTYPE(PFNPGMRELOCATE) pfnRelocate;
321 /** User argument to the callback. */
322 R3PTRTYPE(void *) pvUser;
323 /** Mapping description / name. For easing debugging. */
324 R3PTRTYPE(const char *) pszDesc;
325 /** Number of page tables. */
326 RTUINT cPTs;
327#if HC_ARCH_BITS != GC_ARCH_BITS
328 RTUINT uPadding0; /**< Alignment padding. */
329#endif
330 /** Array of page table mapping data. Each entry
331 * describes one page table. The array can be longer
332 * than the declared length.
333 */
334 struct
335 {
336 /** The HC physical address of the page table. */
337 RTHCPHYS HCPhysPT;
338 /** The HC physical address of the first PAE page table. */
339 RTHCPHYS HCPhysPaePT0;
340 /** The HC physical address of the second PAE page table. */
341 RTHCPHYS HCPhysPaePT1;
342 /** The HC virtual address of the 32-bit page table. */
343 R3PTRTYPE(PVBOXPT) pPTR3;
344 /** The HC virtual address of the two PAE page table. (i.e 1024 entries instead of 512) */
345 R3PTRTYPE(PX86PTPAE) paPaePTsR3;
346 /** The GC virtual address of the 32-bit page table. */
347 GCPTRTYPE(PVBOXPT) pPTGC;
348 /** The GC virtual address of the two PAE page table. */
349 GCPTRTYPE(PX86PTPAE) paPaePTsGC;
350 /** The GC virtual address of the 32-bit page table. */
351 R0PTRTYPE(PVBOXPT) pPTR0;
352 /** The GC virtual address of the two PAE page table. */
353 R0PTRTYPE(PX86PTPAE) paPaePTsR0;
354 } aPTs[1];
355} PGMMAPPING;
356/** Pointer to structure for tracking GC Mappings. */
357typedef struct PGMMAPPING *PPGMMAPPING;
358
359
360/**
361 * Physical page access handler structure.
362 *
363 * This is used to keep track of physical address ranges
364 * which are being monitored in some kind of way.
365 */
366typedef struct PGMPHYSHANDLER
367{
368 AVLROGCPHYSNODECORE Core;
369 /** Alignment padding. */
370 uint32_t u32Padding;
371 /** Access type. */
372 PGMPHYSHANDLERTYPE enmType;
373 /** Number of pages to update. */
374 uint32_t cPages;
375 /** Pointer to R3 callback function. */
376 R3PTRTYPE(PFNPGMR3PHYSHANDLER) pfnHandlerR3;
377 /** User argument for R3 handlers. */
378 R3PTRTYPE(void *) pvUserR3;
379 /** Pointer to R0 callback function. */
380 R0PTRTYPE(PFNPGMR0PHYSHANDLER) pfnHandlerR0;
381 /** User argument for R0 handlers. */
382 R0PTRTYPE(void *) pvUserR0;
383 /** Pointer to GC callback function. */
384 GCPTRTYPE(PFNPGMGCPHYSHANDLER) pfnHandlerGC;
385 /** User argument for GC handlers. */
386 GCPTRTYPE(void *) pvUserGC;
387 /** Description / Name. For easing debugging. */
388 R3PTRTYPE(const char *) pszDesc;
389#ifdef VBOX_WITH_STATISTICS
390 /** Profiling of this handler. */
391 STAMPROFILE Stat;
392#endif
393} PGMPHYSHANDLER;
394/** Pointer to a physical page access handler structure. */
395typedef PGMPHYSHANDLER *PPGMPHYSHANDLER;
396
397
398/**
399 * Cache node for the physical addresses covered by a virtual handler.
400 */
401typedef struct PGMPHYS2VIRTHANDLER
402{
403 /** Core node for the tree based on physical ranges. */
404 AVLROGCPHYSNODECORE Core;
405 /** Offset from this struct to the PGMVIRTHANDLER structure. */
406 RTGCINTPTR offVirtHandler;
407 /** Offset of the next alias relativer to this one.
408 * Bit 0 is used for indicating whether we're in the tree.
409 * Bit 1 is used for indicating that we're the head node.
410 */
411 int32_t offNextAlias;
412} PGMPHYS2VIRTHANDLER;
413/** Pointer to a phys to virtual handler structure. */
414typedef PGMPHYS2VIRTHANDLER *PPGMPHYS2VIRTHANDLER;
415
416/** The bit in PGMPHYS2VIRTHANDLER::offNextAlias used to indicate that the
417 * node is in the tree. */
418#define PGMPHYS2VIRTHANDLER_IN_TREE BIT(0)
419/** The bit in PGMPHYS2VIRTHANDLER::offNextAlias used to indicate that the
420 * node is in the head of an alias chain.
421 * The PGMPHYS2VIRTHANDLER_IN_TREE is always set if this bit is set. */
422#define PGMPHYS2VIRTHANDLER_IS_HEAD BIT(1)
423/** The mask to apply to PGMPHYS2VIRTHANDLER::offNextAlias to get the offset. */
424#define PGMPHYS2VIRTHANDLER_OFF_MASK (~(int32_t)3)
425
426
427/**
428 * Virtual page access handler structure.
429 *
430 * This is used to keep track of virtual address ranges
431 * which are being monitored in some kind of way.
432 */
433typedef struct PGMVIRTHANDLER
434{
435 /** Core node for the tree based on virtual ranges. */
436 AVLROGCPTRNODECORE Core;
437 /** Number of cache pages. */
438 uint32_t u32Padding;
439 /** Access type. */
440 PGMVIRTHANDLERTYPE enmType;
441 /** Number of cache pages. */
442 uint32_t cPages;
443
444/** @todo The next two members are redundant. It adds some readability though. */
445 /** Start of the range. */
446 RTGCPTR GCPtr;
447 /** End of the range (exclusive). */
448 RTGCPTR GCPtrLast;
449 /** Size of the range (in bytes). */
450 RTGCUINTPTR cb;
451 /** Pointer to the GC callback function. */
452 GCPTRTYPE(PFNPGMGCVIRTHANDLER) pfnHandlerGC;
453 /** Pointer to the HC callback function for invalidation. */
454 R3PTRTYPE(PFNPGMHCVIRTINVALIDATE) pfnInvalidateHC;
455 /** Pointer to the HC callback function. */
456 R3PTRTYPE(PFNPGMHCVIRTHANDLER) pfnHandlerHC;
457 /** Description / Name. For easing debugging. */
458 R3PTRTYPE(const char *) pszDesc;
459#ifdef VBOX_WITH_STATISTICS
460 /** Profiling of this handler. */
461 STAMPROFILE Stat;
462#endif
463 /** Array of cached physical addresses for the monitored ranged. */
464 PGMPHYS2VIRTHANDLER aPhysToVirt[HC_ARCH_BITS == 32 ? 1 : 2];
465} PGMVIRTHANDLER;
466/** Pointer to a virtual page access handler structure. */
467typedef PGMVIRTHANDLER *PPGMVIRTHANDLER;
468
469
470/**
471 * A Physical Guest Page tracking structure.
472 *
473 * The format of this structure is complicated because we have to fit a lot
474 * of information into as few bits as possible. The format is also subject
475 * to change (there is one comming up soon). Which means that for we'll be
476 * using PGM_PAGE_GET_* and PGM_PAGE_SET_* macros for all accessess to the
477 * structure.
478 */
479typedef struct PGMPAGE
480{
481 /** The physical address and a whole lot of other stuff. All bits are used! */
482 RTHCPHYS HCPhys;
483 /** The page state. */
484 uint32_t u2State : 2;
485 /** Flag indicating that a write monitored page was written to when set. */
486 uint32_t fWrittenTo : 1;
487 /** For later. */
488 uint32_t fSomethingElse : 1;
489 /** The Page ID. */
490 uint32_t idPage : 28;
491 uint32_t u32B;
492} PGMPAGE;
493AssertCompileSize(PGMPAGE, 16);
494/** Pointer to a physical guest page. */
495typedef PGMPAGE *PPGMPAGE;
496/** Pointer to a const physical guest page. */
497typedef const PGMPAGE *PCPGMPAGE;
498/** Pointer to a physical guest page pointer. */
499typedef PPGMPAGE *PPPGMPAGE;
500
501/** @name The Page state, PGMPAGE::u2State.
502 * @{ */
503/** The zero page.
504 * This is a per-VM page that's never ever mapped writable. */
505#define PGM_PAGE_STATE_ZERO 0
506/** A allocated page.
507 * This is a per-VM page allocated from the page pool.
508 */
509#define PGM_PAGE_STATE_ALLOCATED 1
510/** A allocated page that's being monitored for writes.
511 * The shadow page table mappings are read-only. When a write occurs, the
512 * fWrittenTo member is set, the page remapped as read-write and the state
513 * moved back to allocated. */
514#define PGM_PAGE_STATE_WRITE_MONITORED 2
515/** The page is shared, aka. copy-on-write.
516 * This is a page that's shared with other VMs. */
517#define PGM_PAGE_STATE_SHARED 3
518/** @} */
519
520
521/**
522 * Gets the page state.
523 * @returns page state (PGM_PAGE_STATE_*).
524 * @param pPage Pointer to the physical guest page tracking structure.
525 */
526#define PGM_PAGE_GET_STATE(pPage) ( (pPage)->u2State )
527
528/**
529 * Sets the page state.
530 * @param pPage Pointer to the physical guest page tracking structure.
531 * @param _uState The new page state.
532 */
533#define PGM_PAGE_SET_STATE(pPage, _uState) \
534 do { (pPage)->u2State = (_uState); } while (0)
535
536
537/**
538 * Gets the host physical address of the guest page.
539 * @returns host physical address (RTHCPHYS).
540 * @param pPage Pointer to the physical guest page tracking structure.
541 */
542#define PGM_PAGE_GET_HCPHYS(pPage) ( (pPage)->HCPhys & UINT64_C(0x0000fffffffff000) )
543
544/**
545 * Sets the host physical address of the guest page.
546 * @param pPage Pointer to the physical guest page tracking structure.
547 * @param _HCPhys The new host physical address.
548 */
549#define PGM_PAGE_SET_HCPHYS(pPage, _HCPhys) \
550 do { (pPage)->HCPhys = (((pPage)->HCPhys) & UINT64_C(0xffff000000000fff)) \
551 | ((_HCPhys) & UINT64_C(0x0000fffffffff000)); } while (0)
552
553/** The chunk shift. (2^20 = 1 MB) */
554#define GMM_CHUNK_SHIFT 20
555/** The allocation chunk size. */
556#define GMM_CHUNK_SIZE (1U << GMM_CHUNK_SHIFT)
557/** The shift factor for converting a page id into a chunk id. */
558#define GMM_CHUNKID_SHIFT (GMM_CHUNK_SHIFT - PAGE_SHIFT)
559/** The NIL Chunk ID value. */
560#define NIL_GMM_CHUNKID 0
561/** The NIL Page ID value. */
562#define NIL_GMM_PAGEID 0
563
564/**
565 * Get the Page ID.
566 * @returns The Page ID; NIL_GMM_PAGEID if it's a ZERO page.
567 * @param pPage Pointer to the physical guest page tracking structure.
568 */
569#define PGM_PAGE_GET_PAGEID(pPage) ( (pPage)->idPage )
570/* later:
571#define PGM_PAGE_GET_PAGEID(pPage) ( ((uint32_t)(pPage)->HCPhys >> (48 - 12))
572 | ((uint32_t)(pPage)->HCPhys & 0xfff) )
573*/
574/**
575 * Sets the Page ID.
576 * @param pPage Pointer to the physical guest page tracking structure.
577 */
578#define PGM_PAGE_SET_PAGEID(pPage, _idPage) do { (pPage)->idPage = (_idPage); } while (0)
579/* later:
580#define PGM_PAGE_SET_PAGEID(pPage, _idPage) do { (pPage)->HCPhys = (((pPage)->HCPhys) & UINT64_C(0x0000fffffffff000)) \
581 | ((_idPage) & 0xfff) \
582 | (((_idPage) & 0x0ffff000) << (48-12)); } while (0)
583*/
584
585/**
586 * Get the Chunk ID.
587 * @returns The Chunk ID; NIL_GMM_CHUNKID if it's a ZERO page.
588 * @param pPage Pointer to the physical guest page tracking structure.
589 */
590#define PGM_PAGE_GET_CHUNKID(pPage) ( (pPage)->idPage >> GMM_CHUNKID_SHIFT )
591/* later:
592#if GMM_CHUNKID_SHIFT == 12
593# define PGM_PAGE_GET_CHUNKID(pPage) ( (uint32_t)((pPage)->HCPhys >> 48) )
594#elif GMM_CHUNKID_SHIFT > 12
595# define PGM_PAGE_GET_CHUNKID(pPage) ( (uint32_t)((pPage)->HCPhys >> (48 + (GMM_CHUNKID_SHIFT - 12)) )
596#elif GMM_CHUNKID_SHIFT < 12
597# define PGM_PAGE_GET_CHUNKID(pPage) ( ( (uint32_t)((pPage)->HCPhys >> 48) << (12 - GMM_CHUNKID_SHIFT) ) \
598 | ( (uint32_t)((pPage)->HCPhys & 0xfff) >> GMM_CHUNKID_SHIFT ) )
599#else
600# error "GMM_CHUNKID_SHIFT isn't defined or something."
601#endif
602*/
603
604/**
605 * Get the index of the page within the allocaiton chunk.
606 * @returns The page index.
607 * @param pPage Pointer to the physical guest page tracking structure.
608 */
609#define PGM_PAGE_GET_PAGE_IN_CHUNK(pPage) ( (pPage)->idPage & (RT_BIT_32(GMM_CHUNKID_SHIFT) - 1) )
610/* later:
611#if GMM_CHUNKID_SHIFT <= 12
612# define PGM_PAGE_GET_PAGE_IN_CHUNK(pPage) ( (uint32_t)((pPage)->HCPhys & (RT_BIT_32(GMM_CHUNKID_SHIFT) - 1)) )
613#else
614# define PGM_PAGE_GET_PAGE_IN_CHUNK(pPage) ( (uint32_t)((pPage)->HCPhys & 0xfff) \
615 | ( (uint32_t)((pPage)->HCPhys >> 48) & (RT_BIT_32(GMM_CHUNKID_SHIFT - 12) - 1) ) )
616#endif
617*/
618
619/**
620 * Checks if the page is 'reserved'.
621 * @returns true/false.
622 * @param pPage Pointer to the physical guest page tracking structure.
623 */
624#define PGM_PAGE_IS_RESERVED(pPage) ( !!((pPage)->HCPhys & MM_RAM_FLAGS_RESERVED) )
625
626/**
627 * Checks if the page is marked for MMIO.
628 * @returns true/false.
629 * @param pPage Pointer to the physical guest page tracking structure.
630 */
631#define PGM_PAGE_IS_MMIO(pPage) ( !!((pPage)->HCPhys & MM_RAM_FLAGS_MMIO) )
632
633/**
634 * Checks if the page is backed by the ZERO page.
635 * @returns true/false.
636 * @param pPage Pointer to the physical guest page tracking structure.
637 */
638#define PGM_PAGE_IS_ZERO(pPage) ( (pPage)->u2State == PGM_PAGE_STATE_ZERO )
639
640/**
641 * Checks if the page is backed by a SHARED page.
642 * @returns true/false.
643 * @param pPage Pointer to the physical guest page tracking structure.
644 */
645#define PGM_PAGE_IS_SHARED(pPage) ( (pPage)->u2State == PGM_PAGE_STATE_SHARED )
646
647
648
649/**
650 * Ram range for GC Phys to HC Phys conversion.
651 *
652 * Can be used for HC Virt to GC Phys and HC Virt to HC Phys
653 * conversions too, but we'll let MM handle that for now.
654 *
655 * This structure is used by linked lists in both GC and HC.
656 */
657typedef struct PGMRAMRANGE
658{
659 /** Pointer to the next RAM range - for HC. */
660 R3R0PTRTYPE(struct PGMRAMRANGE *) pNextHC;
661 /** Pointer to the next RAM range - for GC. */
662 GCPTRTYPE(struct PGMRAMRANGE *) pNextGC;
663 /** Start of the range. Page aligned. */
664 RTGCPHYS GCPhys;
665 /** Last address in the range (inclusive). Page aligned (-1). */
666 RTGCPHYS GCPhysLast;
667 /** Size of the range. (Page aligned of course). */
668 RTGCPHYS cb;
669 /** MM_RAM_* flags */
670 uint32_t fFlags;
671
672 /** HC virtual lookup ranges for chunks. Currently only used with MM_RAM_FLAGS_DYNAMIC_ALLOC ranges. */
673 GCPTRTYPE(PRTHCPTR) pavHCChunkGC;
674 /** HC virtual lookup ranges for chunks. Currently only used with MM_RAM_FLAGS_DYNAMIC_ALLOC ranges. */
675 R3R0PTRTYPE(PRTHCPTR) pavHCChunkHC;
676
677 /** Start of the HC mapping of the range.
678 * For pure MMIO and dynamically allocated ranges this is NULL, while for all ranges this is a valid pointer. */
679 R3PTRTYPE(void *) pvHC;
680
681 /** Array of physical guest page tracking structures. */
682 PGMPAGE aPages[1];
683} PGMRAMRANGE;
684/** Pointer to Ram range for GC Phys to HC Phys conversion. */
685typedef PGMRAMRANGE *PPGMRAMRANGE;
686
687/** Return hc ptr corresponding to the ram range and physical offset */
688#define PGMRAMRANGE_GETHCPTR(pRam, off) \
689 (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC) ? (RTHCPTR)((RTHCUINTPTR)CTXSUFF(pRam->pavHCChunk)[(off >> PGM_DYNAMIC_CHUNK_SHIFT)] + (off & PGM_DYNAMIC_CHUNK_OFFSET_MASK)) \
690 : (RTHCPTR)((RTHCUINTPTR)pRam->pvHC + off);
691
692/** @todo r=bird: fix typename. */
693/**
694 * PGMPhysRead/Write cache entry
695 */
696typedef struct PGMPHYSCACHE_ENTRY
697{
698 /** HC pointer to physical page */
699 R3PTRTYPE(uint8_t *) pbHC;
700 /** GC Physical address for cache entry */
701 RTGCPHYS GCPhys;
702#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32
703 RTGCPHYS u32Padding0; /**< alignment padding. */
704#endif
705} PGMPHYSCACHE_ENTRY;
706
707/**
708 * PGMPhysRead/Write cache to reduce REM memory access overhead
709 */
710typedef struct PGMPHYSCACHE
711{
712 /** Bitmap of valid cache entries */
713 uint64_t aEntries;
714 /** Cache entries */
715 PGMPHYSCACHE_ENTRY Entry[PGM_MAX_PHYSCACHE_ENTRIES];
716} PGMPHYSCACHE;
717
718
719/** Pointer to an allocation chunk ring-3 mapping. */
720typedef struct PGMCHUNKR3MAP *PPGMCHUNKR3MAP;
721/** Pointer to an allocation chunk ring-3 mapping pointer. */
722typedef PPGMCHUNKR3MAP *PPPGMCHUNKR3MAP;
723
724/**
725 * Ring-3 tracking structore for an allocation chunk ring-3 mapping.
726 *
727 * The primary tree (Core) uses the chunk id as key.
728 * The secondary tree (AgeCore) is used for ageing and uses ageing sequence number as key.
729 */
730typedef struct PGMCHUNKR3MAP
731{
732 /** The key is the chunk id. */
733 AVLU32NODECORE Core;
734 /** The key is the ageing sequence number. */
735 AVLLU32NODECORE AgeCore;
736 /** The current age thingy. */
737 uint32_t iAge;
738 /** The current reference count. */
739 uint32_t volatile cRefs;
740 /** The current permanent reference count. */
741 uint32_t volatile cPermRefs;
742 /** The mapping address. */
743 void *pv;
744} PGMCHUNKR3MAP;
745
746/**
747 * Allocation chunk ring-3 mapping TLB entry.
748 */
749typedef struct PGMCHUNKR3MAPTLBE
750{
751 /** The chunk id. */
752 uint32_t volatile idChunk;
753#if HC_ARCH_BITS == 64
754 uint32_t u32Padding; /**< alignment padding. */
755#endif
756 /** The chunk map. */
757 R3R0PTRTYPE(PPGMCHUNKR3MAP) volatile pChunk;
758} PGMCHUNKR3MAPTLBE;
759/** Pointer to the an allocation chunk ring-3 mapping TLB entry. */
760typedef PGMCHUNKR3MAPTLBE *PPGMCHUNKR3MAPTLBE;
761
762/** The number of TLB entries in PGMCHUNKR3MAPTLB.
763 * @remark Must be a power of two value. */
764#define PGM_CHUNKR3MAPTLB_ENTRIES 32
765
766/**
767 * Allocation chunk ring-3 mapping TLB.
768 *
769 * @remarks We use a TLB to speed up lookups by avoiding walking the AVL.
770 * At first glance this might look kinda odd since AVL trees are
771 * supposed to give the most optimial lookup times of all trees
772 * due to their balancing. However, take a tree with 1023 nodes
773 * in it, that's 10 levels, meaning that most searches has to go
774 * down 9 levels before they find what they want. This isn't fast
775 * compared to a TLB hit. There is the factor of cache misses,
776 * and of course the problem with trees and branch prediction.
777 * This is why we use TLBs in front of most of the trees.
778 *
779 * @todo Generalize this TLB + AVL stuff, shouldn't be all that
780 * difficult when we switch to inlined AVL trees (from kStuff).
781 */
782typedef struct PGMCHUNKR3MAPTLB
783{
784 /** The TLB entries. */
785 PGMCHUNKR3MAPTLBE aEntries[PGM_CHUNKR3MAPTLB_ENTRIES];
786} PGMCHUNKR3MAPTLB;
787
788/**
789 * Calculates the index of a guest page in the Ring-3 Chunk TLB.
790 * @returns Chunk TLB index.
791 * @param idChunk The Chunk ID.
792 */
793#define PGM_CHUNKR3MAPTLB_IDX(idChunk) ( (idChunk) & (PGM_CHUNKR3MAPTLB_ENTRIES - 1) )
794
795
796/**
797 * Ring-3 guest page mapping TLB entry.
798 * @remarks used in ring-0 as well at the moment.
799 */
800typedef struct PGMPAGER3MAPTLBE
801{
802 /** Address of the page. */
803 RTGCPHYS volatile GCPhys;
804#if HC_ARCH_BITS == 64
805 uint32_t u32Padding; /**< alignment padding. */
806#endif
807 /** The guest page. */
808 R3R0PTRTYPE(PPGMPAGE) volatile pPage;
809 /** Pointer to the page mapping tracking structure, PGMCHUNKR3MAP. */
810 R3R0PTRTYPE(PPGMCHUNKR3MAP) volatile pMap;
811 /** The address */
812 R3R0PTRTYPE(void *) volatile pv;
813} PGMPAGER3MAPTLBE;
814/** Pointer to an entry in the HC physical TLB. */
815typedef PGMPAGER3MAPTLBE *PPGMPAGER3MAPTLBE;
816
817
818/** The number of entries in the ring-3 guest page mapping TLB.
819 * @remarks The value must be a power of two. */
820#define PGM_PAGER3MAPTLB_ENTRIES 64
821
822/**
823 * Ring-3 guest page mapping TLB.
824 * @remarks used in ring-0 as well at the moment.
825 */
826typedef struct PGMPAGER3MAPTLB
827{
828 /** The TLB entries. */
829 PGMPAGER3MAPTLBE aEntries[PGM_PAGER3MAPTLB_ENTRIES];
830} PGMPAGER3MAPTLB;
831/** Pointer to the ring-3 guest page mapping TLB. */
832typedef PGMPAGER3MAPTLB *PPGMPAGER3MAPTLB;
833
834/**
835 * Calculates the index of the TLB entry for the specified guest page.
836 * @returns Physical TLB index.
837 * @param GCPhys The guest physical address.
838 */
839#define PGM_PAGER3MAPTLB_IDX(GCPhys) ( ((GCPhys) >> PAGE_SHIFT) & (PGM_PAGER3MAPTLB_ENTRIES - 1) )
840
841
842/** @name Context neutrual page mapper TLB.
843 *
844 * Hoping to avoid some code and bug duplication parts of the GCxxx->CCPtr
845 * code is writting in a kind of context neutrual way. Time will show whether
846 * this actually makes sense or not...
847 *
848 * @{ */
849/** @typedef PPGMPAGEMAPTLB
850 * The page mapper TLB pointer type for the current context. */
851/** @typedef PPGMPAGEMAPTLB
852 * The page mapper TLB entry pointer type for the current context. */
853/** @typedef PPGMPAGEMAPTLB
854 * The page mapper TLB entry pointer pointer type for the current context. */
855/** @def PGMPAGEMAPTLB_ENTRIES
856 * The number of TLB entries in the page mapper TLB for the current context. */
857/** @def PGM_PAGEMAPTLB_IDX
858 * Calculate the TLB index for a guest physical address.
859 * @returns The TLB index.
860 * @param GCPhys The guest physical address. */
861/** @typedef PPGMPAGEMAP
862 * Pointer to a page mapper unit for current context. */
863/** @typedef PPPGMPAGEMAP
864 * Pointer to a page mapper unit pointer for current context. */
865#ifdef IN_GC
866// typedef PPGMPAGEGCMAPTLB PPGMPAGEMAPTLB;
867// typedef PPGMPAGEGCMAPTLBE PPGMPAGEMAPTLBE;
868// typedef PPGMPAGEGCMAPTLBE *PPPGMPAGEMAPTLBE;
869# define PGM_PAGEMAPTLB_ENTRIES PGM_PAGEGCMAPTLB_ENTRIES
870# define PGM_PAGEMAPTLB_IDX(GCPhys) PGM_PAGEGCMAPTLB_IDX(GCPhys)
871 typedef void * PPGMPAGEMAP;
872 typedef void ** PPPGMPAGEMAP;
873//#elif IN_RING0
874// typedef PPGMPAGER0MAPTLB PPGMPAGEMAPTLB;
875// typedef PPGMPAGER0MAPTLBE PPGMPAGEMAPTLBE;
876// typedef PPGMPAGER0MAPTLBE *PPPGMPAGEMAPTLBE;
877//# define PGM_PAGEMAPTLB_ENTRIES PGM_PAGER0MAPTLB_ENTRIES
878//# define PGM_PAGEMAPTLB_IDX(GCPhys) PGM_PAGER0MAPTLB_IDX(GCPhys)
879// typedef PPGMCHUNKR0MAP PPGMPAGEMAP;
880// typedef PPPGMCHUNKR0MAP PPPGMPAGEMAP;
881#else
882 typedef PPGMPAGER3MAPTLB PPGMPAGEMAPTLB;
883 typedef PPGMPAGER3MAPTLBE PPGMPAGEMAPTLBE;
884 typedef PPGMPAGER3MAPTLBE *PPPGMPAGEMAPTLBE;
885# define PGM_PAGEMAPTLB_ENTRIES PGM_PAGER3MAPTLB_ENTRIES
886# define PGM_PAGEMAPTLB_IDX(GCPhys) PGM_PAGER3MAPTLB_IDX(GCPhys)
887 typedef PPGMCHUNKR3MAP PPGMPAGEMAP;
888 typedef PPPGMCHUNKR3MAP PPPGMPAGEMAP;
889#endif
890/** @} */
891
892
893/** @name PGM Pool Indexes.
894 * Aka. the unique shadow page identifier.
895 * @{ */
896/** NIL page pool IDX. */
897#define NIL_PGMPOOL_IDX 0
898/** The first normal index. */
899#define PGMPOOL_IDX_FIRST_SPECIAL 1
900/** Page directory (32-bit root). */
901#define PGMPOOL_IDX_PD 1
902/** The extended PAE page directory (2048 entries, works as root currently). */
903#define PGMPOOL_IDX_PAE_PD 2
904/** Page Directory Pointer Table (PAE root, not currently used). */
905#define PGMPOOL_IDX_PDPTR 3
906/** Page Map Level-4 (64-bit root). */
907#define PGMPOOL_IDX_PML4 4
908/** The first normal index. */
909#define PGMPOOL_IDX_FIRST 5
910/** The last valid index. (inclusive, 14 bits) */
911#define PGMPOOL_IDX_LAST 0x3fff
912/** @} */
913
914/** The NIL index for the parent chain. */
915#define NIL_PGMPOOL_USER_INDEX ((uint16_t)0xffff)
916
917/**
918 * Node in the chain linking a shadowed page to it's parent (user).
919 */
920#pragma pack(1)
921typedef struct PGMPOOLUSER
922{
923 /** The index to the next item in the chain. NIL_PGMPOOL_USER_INDEX is no next. */
924 uint16_t iNext;
925 /** The user page index. */
926 uint16_t iUser;
927 /** Index into the user table. */
928 uint16_t iUserTable;
929} PGMPOOLUSER, *PPGMPOOLUSER;
930typedef const PGMPOOLUSER *PCPGMPOOLUSER;
931#pragma pack()
932
933
934/** The NIL index for the phys ext chain. */
935#define NIL_PGMPOOL_PHYSEXT_INDEX ((uint16_t)0xffff)
936
937/**
938 * Node in the chain of physical cross reference extents.
939 */
940#pragma pack(1)
941typedef struct PGMPOOLPHYSEXT
942{
943 /** The index to the next item in the chain. NIL_PGMPOOL_PHYSEXT_INDEX is no next. */
944 uint16_t iNext;
945 /** The user page index. */
946 uint16_t aidx[3];
947} PGMPOOLPHYSEXT, *PPGMPOOLPHYSEXT;
948typedef const PGMPOOLPHYSEXT *PCPGMPOOLPHYSEXT;
949#pragma pack()
950
951
952/**
953 * The kind of page that's being shadowed.
954 */
955typedef enum PGMPOOLKIND
956{
957 /** The virtual invalid 0 entry. */
958 PGMPOOLKIND_INVALID = 0,
959 /** The entry is free (=unused). */
960 PGMPOOLKIND_FREE,
961
962 /** Shw: 32-bit page table; Gst: no paging */
963 PGMPOOLKIND_32BIT_PT_FOR_PHYS,
964 /** Shw: 32-bit page table; Gst: 32-bit page table. */
965 PGMPOOLKIND_32BIT_PT_FOR_32BIT_PT,
966 /** Shw: 32-bit page table; Gst: 4MB page. */
967 PGMPOOLKIND_32BIT_PT_FOR_32BIT_4MB,
968 /** Shw: PAE page table; Gst: no paging */
969 PGMPOOLKIND_PAE_PT_FOR_PHYS,
970 /** Shw: PAE page table; Gst: 32-bit page table. */
971 PGMPOOLKIND_PAE_PT_FOR_32BIT_PT,
972 /** Shw: PAE page table; Gst: Half of a 4MB page. */
973 PGMPOOLKIND_PAE_PT_FOR_32BIT_4MB,
974 /** Shw: PAE page table; Gst: PAE page table. */
975 PGMPOOLKIND_PAE_PT_FOR_PAE_PT,
976 /** Shw: PAE page table; Gst: 2MB page. */
977 PGMPOOLKIND_PAE_PT_FOR_PAE_2MB,
978
979 /** Shw: PAE page directory; Gst: 32-bit page directory. */
980 PGMPOOLKIND_PAE_PD_FOR_32BIT_PD,
981 /** Shw: PAE page directory; Gst: PAE page directory. */
982 PGMPOOLKIND_PAE_PD_FOR_PAE_PD,
983
984 /** Shw: 64-bit page directory pointer table; Gst: 64-bit page directory pointer table. */
985 PGMPOOLKIND_64BIT_PDPTR_FOR_64BIT_PDPTR,
986
987 /** Shw: Root 32-bit page directory. */
988 PGMPOOLKIND_ROOT_32BIT_PD,
989 /** Shw: Root PAE page directory */
990 PGMPOOLKIND_ROOT_PAE_PD,
991 /** Shw: Root PAE page directory pointer table (legacy, 4 entries). */
992 PGMPOOLKIND_ROOT_PDPTR,
993 /** Shw: Root page map level-4 table. */
994 PGMPOOLKIND_ROOT_PML4,
995
996 /** The last valid entry. */
997 PGMPOOLKIND_LAST = PGMPOOLKIND_ROOT_PML4
998} PGMPOOLKIND;
999
1000
1001/**
1002 * The tracking data for a page in the pool.
1003 */
1004typedef struct PGMPOOLPAGE
1005{
1006 /** AVL node code with the (HC) physical address of this page. */
1007 AVLOHCPHYSNODECORE Core;
1008 /** Pointer to the HC mapping of the page. */
1009 R3R0PTRTYPE(void *) pvPageHC;
1010 /** The guest physical address. */
1011 RTGCPHYS GCPhys;
1012 /** The kind of page we're shadowing. (This is really a PGMPOOLKIND enum.) */
1013 uint8_t enmKind;
1014 uint8_t bPadding;
1015 /** The index of this page. */
1016 uint16_t idx;
1017 /** The next entry in the list this page currently resides in.
1018 * It's either in the free list or in the GCPhys hash. */
1019 uint16_t iNext;
1020#ifdef PGMPOOL_WITH_USER_TRACKING
1021 /** Head of the user chain. NIL_PGMPOOL_USER_INDEX if not currently in use. */
1022 uint16_t iUserHead;
1023 /** The number of present entries. */
1024 uint16_t cPresent;
1025 /** The first entry in the table which is present. */
1026 uint16_t iFirstPresent;
1027#endif
1028#ifdef PGMPOOL_WITH_MONITORING
1029 /** The number of modifications to the monitored page. */
1030 uint16_t cModifications;
1031 /** The next modified page. NIL_PGMPOOL_IDX if tail. */
1032 uint16_t iModifiedNext;
1033 /** The previous modified page. NIL_PGMPOOL_IDX if head. */
1034 uint16_t iModifiedPrev;
1035 /** The next page sharing access handler. NIL_PGMPOOL_IDX if tail. */
1036 uint16_t iMonitoredNext;
1037 /** The previous page sharing access handler. NIL_PGMPOOL_IDX if head. */
1038 uint16_t iMonitoredPrev;
1039#endif
1040#ifdef PGMPOOL_WITH_CACHE
1041 /** The next page in the age list. */
1042 uint16_t iAgeNext;
1043 /** The previous page in the age list. */
1044 uint16_t iAgePrev;
1045#endif /* PGMPOOL_WITH_CACHE */
1046 /** Used to indicate that the page is zeroed. */
1047 bool fZeroed;
1048 /** Used to indicate that a PT has non-global entries. */
1049 bool fSeenNonGlobal;
1050 /** Used to indicate that we're monitoring writes to the guest page. */
1051 bool fMonitored;
1052 /** Used to indicate that the page is in the cache (e.g. in the GCPhys hash).
1053 * (All pages are in the age list.) */
1054 bool fCached;
1055 /** This is used by the R3 access handlers when invoked by an async thread.
1056 * It's a hack required because of REMR3NotifyHandlerPhysicalDeregister. */
1057 bool volatile fReusedFlushPending;
1058 /** Used to indicate that the guest is mapping the page is also used as a CR3.
1059 * In these cases the access handler acts differently and will check
1060 * for mapping conflicts like the normal CR3 handler.
1061 * @todo When we change the CR3 shadowing to use pool pages, this flag can be
1062 * replaced by a list of pages which share access handler.
1063 */
1064 bool fCR3Mix;
1065#if HC_ARCH_BITS == 64 || GC_ARCH_BITS == 64
1066 bool Alignment[4]; /**< Align the structure size on a 64-bit boundrary. */
1067#endif
1068} PGMPOOLPAGE, *PPGMPOOLPAGE, **PPPGMPOOLPAGE;
1069
1070
1071#ifdef PGMPOOL_WITH_CACHE
1072/** The hash table size. */
1073# define PGMPOOL_HASH_SIZE 0x40
1074/** The hash function. */
1075# define PGMPOOL_HASH(GCPhys) ( ((GCPhys) >> PAGE_SHIFT) & (PGMPOOL_HASH_SIZE - 1) )
1076#endif
1077
1078
1079/**
1080 * The shadow page pool instance data.
1081 *
1082 * It's all one big allocation made at init time, except for the
1083 * pages that is. The user nodes follows immediatly after the
1084 * page structures.
1085 */
1086typedef struct PGMPOOL
1087{
1088 /** The VM handle - HC Ptr. */
1089 R3R0PTRTYPE(PVM) pVMHC;
1090 /** The VM handle - GC Ptr. */
1091 GCPTRTYPE(PVM) pVMGC;
1092 /** The max pool size. This includes the special IDs. */
1093 uint16_t cMaxPages;
1094 /** The current pool size. */
1095 uint16_t cCurPages;
1096 /** The head of the free page list. */
1097 uint16_t iFreeHead;
1098 /* Padding. */
1099 uint16_t u16Padding;
1100#ifdef PGMPOOL_WITH_USER_TRACKING
1101 /** Head of the chain of free user nodes. */
1102 uint16_t iUserFreeHead;
1103 /** The number of user nodes we've allocated. */
1104 uint16_t cMaxUsers;
1105 /** The number of present page table entries in the entire pool. */
1106 uint32_t cPresent;
1107 /** Pointer to the array of user nodes - GC pointer. */
1108 GCPTRTYPE(PPGMPOOLUSER) paUsersGC;
1109 /** Pointer to the array of user nodes - HC pointer. */
1110 R3R0PTRTYPE(PPGMPOOLUSER) paUsersHC;
1111#endif /* PGMPOOL_WITH_USER_TRACKING */
1112#ifdef PGMPOOL_WITH_GCPHYS_TRACKING
1113 /** Head of the chain of free phys ext nodes. */
1114 uint16_t iPhysExtFreeHead;
1115 /** The number of user nodes we've allocated. */
1116 uint16_t cMaxPhysExts;
1117 /** Pointer to the array of physical xref extent - GC pointer. */
1118 GCPTRTYPE(PPGMPOOLPHYSEXT) paPhysExtsGC;
1119 /** Pointer to the array of physical xref extent nodes - HC pointer. */
1120 R3R0PTRTYPE(PPGMPOOLPHYSEXT) paPhysExtsHC;
1121#endif /* PGMPOOL_WITH_GCPHYS_TRACKING */
1122#ifdef PGMPOOL_WITH_CACHE
1123 /** Hash table for GCPhys addresses. */
1124 uint16_t aiHash[PGMPOOL_HASH_SIZE];
1125 /** The head of the age list. */
1126 uint16_t iAgeHead;
1127 /** The tail of the age list. */
1128 uint16_t iAgeTail;
1129 /** Set if the cache is enabled. */
1130 bool fCacheEnabled;
1131#endif /* PGMPOOL_WITH_CACHE */
1132#ifdef PGMPOOL_WITH_MONITORING
1133 /** Head of the list of modified pages. */
1134 uint16_t iModifiedHead;
1135 /** The current number of modified pages. */
1136 uint16_t cModifiedPages;
1137 /** Access handler, GC. */
1138 GCPTRTYPE(PFNPGMGCPHYSHANDLER) pfnAccessHandlerGC;
1139 /** Access handler, R0. */
1140 R0PTRTYPE(PFNPGMR0PHYSHANDLER) pfnAccessHandlerR0;
1141 /** Access handler, R3. */
1142 R3PTRTYPE(PFNPGMR3PHYSHANDLER) pfnAccessHandlerR3;
1143 /** The access handler description (HC ptr). */
1144 R3PTRTYPE(const char *) pszAccessHandler;
1145#endif /* PGMPOOL_WITH_MONITORING */
1146 /** The number of pages currently in use. */
1147 uint16_t cUsedPages;
1148#ifdef VBOX_WITH_STATISTICS
1149 /** The high wather mark for cUsedPages. */
1150 uint16_t cUsedPagesHigh;
1151 uint32_t Alignment1; /**< Align the next member on a 64-bit boundrary. */
1152 /** Profiling pgmPoolAlloc(). */
1153 STAMPROFILEADV StatAlloc;
1154 /** Profiling pgmPoolClearAll(). */
1155 STAMPROFILE StatClearAll;
1156 /** Profiling pgmPoolFlushAllInt(). */
1157 STAMPROFILE StatFlushAllInt;
1158 /** Profiling pgmPoolFlushPage(). */
1159 STAMPROFILE StatFlushPage;
1160 /** Profiling pgmPoolFree(). */
1161 STAMPROFILE StatFree;
1162 /** Profiling time spent zeroing pages. */
1163 STAMPROFILE StatZeroPage;
1164# ifdef PGMPOOL_WITH_USER_TRACKING
1165 /** Profiling of pgmPoolTrackDeref. */
1166 STAMPROFILE StatTrackDeref;
1167 /** Profiling pgmTrackFlushGCPhysPT. */
1168 STAMPROFILE StatTrackFlushGCPhysPT;
1169 /** Profiling pgmTrackFlushGCPhysPTs. */
1170 STAMPROFILE StatTrackFlushGCPhysPTs;
1171 /** Profiling pgmTrackFlushGCPhysPTsSlow. */
1172 STAMPROFILE StatTrackFlushGCPhysPTsSlow;
1173 /** Number of times we've been out of user records. */
1174 STAMCOUNTER StatTrackFreeUpOneUser;
1175# endif
1176# ifdef PGMPOOL_WITH_GCPHYS_TRACKING
1177 /** Profiling deref activity related tracking GC physical pages. */
1178 STAMPROFILE StatTrackDerefGCPhys;
1179 /** Number of linear searches for a HCPhys in the ram ranges. */
1180 STAMCOUNTER StatTrackLinearRamSearches;
1181 /** The number of failing pgmPoolTrackPhysExtAlloc calls. */
1182 STAMCOUNTER StamTrackPhysExtAllocFailures;
1183# endif
1184# ifdef PGMPOOL_WITH_MONITORING
1185 /** Profiling the GC PT access handler. */
1186 STAMPROFILE StatMonitorGC;
1187 /** Times we've failed interpreting the instruction. */
1188 STAMCOUNTER StatMonitorGCEmulateInstr;
1189 /** Profiling the pgmPoolFlushPage calls made from the GC PT access handler. */
1190 STAMPROFILE StatMonitorGCFlushPage;
1191 /** Times we've detected fork(). */
1192 STAMCOUNTER StatMonitorGCFork;
1193 /** Profiling the GC access we've handled (except REP STOSD). */
1194 STAMPROFILE StatMonitorGCHandled;
1195 /** Times we've failed interpreting a patch code instruction. */
1196 STAMCOUNTER StatMonitorGCIntrFailPatch1;
1197 /** Times we've failed interpreting a patch code instruction during flushing. */
1198 STAMCOUNTER StatMonitorGCIntrFailPatch2;
1199 /** The number of times we've seen rep prefixes we can't handle. */
1200 STAMCOUNTER StatMonitorGCRepPrefix;
1201 /** Profiling the REP STOSD cases we've handled. */
1202 STAMPROFILE StatMonitorGCRepStosd;
1203
1204 /** Profiling the HC PT access handler. */
1205 STAMPROFILE StatMonitorHC;
1206 /** Times we've failed interpreting the instruction. */
1207 STAMCOUNTER StatMonitorHCEmulateInstr;
1208 /** Profiling the pgmPoolFlushPage calls made from the HC PT access handler. */
1209 STAMPROFILE StatMonitorHCFlushPage;
1210 /** Times we've detected fork(). */
1211 STAMCOUNTER StatMonitorHCFork;
1212 /** Profiling the HC access we've handled (except REP STOSD). */
1213 STAMPROFILE StatMonitorHCHandled;
1214 /** The number of times we've seen rep prefixes we can't handle. */
1215 STAMCOUNTER StatMonitorHCRepPrefix;
1216 /** Profiling the REP STOSD cases we've handled. */
1217 STAMPROFILE StatMonitorHCRepStosd;
1218 /** The number of times we're called in an async thread an need to flush. */
1219 STAMCOUNTER StatMonitorHCAsync;
1220 /** The high wather mark for cModifiedPages. */
1221 uint16_t cModifiedPagesHigh;
1222 uint16_t Alignment2[3]; /**< Align the next member on a 64-bit boundrary. */
1223# endif
1224# ifdef PGMPOOL_WITH_CACHE
1225 /** The number of cache hits. */
1226 STAMCOUNTER StatCacheHits;
1227 /** The number of cache misses. */
1228 STAMCOUNTER StatCacheMisses;
1229 /** The number of times we've got a conflict of 'kind' in the cache. */
1230 STAMCOUNTER StatCacheKindMismatches;
1231 /** Number of times we've been out of pages. */
1232 STAMCOUNTER StatCacheFreeUpOne;
1233 /** The number of cacheable allocations. */
1234 STAMCOUNTER StatCacheCacheable;
1235 /** The number of uncacheable allocations. */
1236 STAMCOUNTER StatCacheUncacheable;
1237# endif
1238#elif HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32
1239 uint32_t Alignment1; /**< Align the next member on a 64-bit boundrary. */
1240#endif
1241 /** The AVL tree for looking up a page by its HC physical address. */
1242 AVLOHCPHYSTREE HCPhysTree;
1243 uint32_t Alignment3; /**< Align the next member on a 64-bit boundrary. */
1244 /** Array of pages. (cMaxPages in length)
1245 * The Id is the index into thist array.
1246 */
1247 PGMPOOLPAGE aPages[PGMPOOL_IDX_FIRST];
1248} PGMPOOL, *PPGMPOOL, **PPPGMPOOL;
1249
1250
1251/** @def PGMPOOL_PAGE_2_PTR
1252 * Maps a pool page pool into the current context.
1253 *
1254 * @returns VBox status code.
1255 * @param pVM The VM handle.
1256 * @param pPage The pool page.
1257 *
1258 * @remark In HC this uses PGMGCDynMapHCPage(), so it will consume of the
1259 * small page window employeed by that function. Be careful.
1260 * @remark There is no need to assert on the result.
1261 */
1262#ifdef IN_GC
1263# define PGMPOOL_PAGE_2_PTR(pVM, pPage) pgmGCPoolMapPage((pVM), (pPage))
1264#else
1265# define PGMPOOL_PAGE_2_PTR(pVM, pPage) ((pPage)->pvPageHC)
1266#endif
1267
1268
1269/**
1270 * Trees are using self relative offsets as pointers.
1271 * So, all its data, including the root pointer, must be in the heap for HC and GC
1272 * to have the same layout.
1273 */
1274typedef struct PGMTREES
1275{
1276 /** Physical access handlers (AVL range+offsetptr tree). */
1277 AVLROGCPHYSTREE PhysHandlers;
1278 /** Virtual access handlers (AVL range + GC ptr tree). */
1279 AVLROGCPTRTREE VirtHandlers;
1280 /** Virtual access handlers (Phys range AVL range + offsetptr tree). */
1281 AVLROGCPHYSTREE PhysToVirtHandlers;
1282 uint32_t auPadding[1];
1283} PGMTREES;
1284/** Pointer to PGM trees. */
1285typedef PGMTREES *PPGMTREES;
1286
1287
1288/** @name Paging mode macros
1289 * @{ */
1290#ifdef IN_GC
1291# define PGM_CTX(a,b) a##GC##b
1292# define PGM_CTX_STR(a,b) a "GC" b
1293# define PGM_CTX_DECL(type) PGMGCDECL(type)
1294#else
1295# ifdef IN_RING3
1296# define PGM_CTX(a,b) a##R3##b
1297# define PGM_CTX_STR(a,b) a "R3" b
1298# define PGM_CTX_DECL(type) DECLCALLBACK(type)
1299# else
1300# define PGM_CTX(a,b) a##R0##b
1301# define PGM_CTX_STR(a,b) a "R0" b
1302# define PGM_CTX_DECL(type) PGMDECL(type)
1303# endif
1304#endif
1305
1306#define PGM_GST_NAME_REAL(name) PGM_CTX(pgm,GstReal##name)
1307#define PGM_GST_NAME_GC_REAL_STR(name) "pgmGCGstReal" #name
1308#define PGM_GST_NAME_R0_REAL_STR(name) "pgmR0GstReal" #name
1309#define PGM_GST_NAME_PROT(name) PGM_CTX(pgm,GstProt##name)
1310#define PGM_GST_NAME_GC_PROT_STR(name) "pgmGCGstProt" #name
1311#define PGM_GST_NAME_R0_PROT_STR(name) "pgmR0GstProt" #name
1312#define PGM_GST_NAME_32BIT(name) PGM_CTX(pgm,Gst32Bit##name)
1313#define PGM_GST_NAME_GC_32BIT_STR(name) "pgmGCGst32Bit" #name
1314#define PGM_GST_NAME_R0_32BIT_STR(name) "pgmR0Gst32Bit" #name
1315#define PGM_GST_NAME_PAE(name) PGM_CTX(pgm,GstPAE##name)
1316#define PGM_GST_NAME_GC_PAE_STR(name) "pgmGCGstPAE" #name
1317#define PGM_GST_NAME_R0_PAE_STR(name) "pgmR0GstPAE" #name
1318#define PGM_GST_NAME_AMD64(name) PGM_CTX(pgm,GstAMD64##name)
1319#define PGM_GST_NAME_GC_AMD64_STR(name) "pgmGCGstAMD64" #name
1320#define PGM_GST_NAME_R0_AMD64_STR(name) "pgmR0GstAMD64" #name
1321#define PGM_GST_PFN(name, pVM) ((pVM)->pgm.s.PGM_CTX(pfn,Gst##name))
1322#define PGM_GST_DECL(type, name) PGM_CTX_DECL(type) PGM_GST_NAME(name)
1323
1324#define PGM_SHW_NAME_32BIT(name) PGM_CTX(pgm,Shw32Bit##name)
1325#define PGM_SHW_NAME_GC_32BIT_STR(name) "pgmGCShw32Bit" #name
1326#define PGM_SHW_NAME_R0_32BIT_STR(name) "pgmR0Shw32Bit" #name
1327#define PGM_SHW_NAME_PAE(name) PGM_CTX(pgm,ShwPAE##name)
1328#define PGM_SHW_NAME_GC_PAE_STR(name) "pgmGCShwPAE" #name
1329#define PGM_SHW_NAME_R0_PAE_STR(name) "pgmR0ShwPAE" #name
1330#define PGM_SHW_NAME_AMD64(name) PGM_CTX(pgm,ShwAMD64##name)
1331#define PGM_SHW_NAME_GC_AMD64_STR(name) "pgmGCShwAMD64" #name
1332#define PGM_SHW_NAME_R0_AMD64_STR(name) "pgmR0ShwAMD64" #name
1333#define PGM_SHW_DECL(type, name) PGM_CTX_DECL(type) PGM_SHW_NAME(name)
1334#define PGM_SHW_PFN(name, pVM) ((pVM)->pgm.s.PGM_CTX(pfn,Shw##name))
1335
1336/* Shw_Gst */
1337#define PGM_BTH_NAME_32BIT_REAL(name) PGM_CTX(pgm,Bth32BitReal##name)
1338#define PGM_BTH_NAME_32BIT_PROT(name) PGM_CTX(pgm,Bth32BitProt##name)
1339#define PGM_BTH_NAME_32BIT_32BIT(name) PGM_CTX(pgm,Bth32Bit32Bit##name)
1340#define PGM_BTH_NAME_PAE_REAL(name) PGM_CTX(pgm,BthPAEReal##name)
1341#define PGM_BTH_NAME_PAE_PROT(name) PGM_CTX(pgm,BthPAEProt##name)
1342#define PGM_BTH_NAME_PAE_32BIT(name) PGM_CTX(pgm,BthPAE32Bit##name)
1343#define PGM_BTH_NAME_PAE_PAE(name) PGM_CTX(pgm,BthPAEPAE##name)
1344#define PGM_BTH_NAME_AMD64_REAL(name) PGM_CTX(pgm,BthAMD64Real##name)
1345#define PGM_BTH_NAME_AMD64_PROT(name) PGM_CTX(pgm,BthAMD64Prot##name)
1346#define PGM_BTH_NAME_AMD64_AMD64(name) PGM_CTX(pgm,BthAMD64AMD64##name)
1347#define PGM_BTH_NAME_GC_32BIT_REAL_STR(name) "pgmGCBth32BitReal" #name
1348#define PGM_BTH_NAME_GC_32BIT_PROT_STR(name) "pgmGCBth32BitProt" #name
1349#define PGM_BTH_NAME_GC_32BIT_32BIT_STR(name) "pgmGCBth32Bit32Bit" #name
1350#define PGM_BTH_NAME_GC_PAE_REAL_STR(name) "pgmGCBthPAEReal" #name
1351#define PGM_BTH_NAME_GC_PAE_PROT_STR(name) "pgmGCBthPAEProt" #name
1352#define PGM_BTH_NAME_GC_PAE_32BIT_STR(name) "pgmGCBthPAE32Bit" #name
1353#define PGM_BTH_NAME_GC_PAE_PAE_STR(name) "pgmGCBthPAEPAE" #name
1354#define PGM_BTH_NAME_GC_AMD64_REAL_STR(name) "pgmGCBthAMD64Real" #name
1355#define PGM_BTH_NAME_GC_AMD64_PROT_STR(name) "pgmGCBthAMD64Prot" #name
1356#define PGM_BTH_NAME_GC_AMD64_AMD64_STR(name) "pgmGCBthAMD64AMD64" #name
1357#define PGM_BTH_NAME_R0_32BIT_REAL_STR(name) "pgmR0Bth32BitReal" #name
1358#define PGM_BTH_NAME_R0_32BIT_PROT_STR(name) "pgmR0Bth32BitProt" #name
1359#define PGM_BTH_NAME_R0_32BIT_32BIT_STR(name) "pgmR0Bth32Bit32Bit" #name
1360#define PGM_BTH_NAME_R0_PAE_REAL_STR(name) "pgmR0BthPAEReal" #name
1361#define PGM_BTH_NAME_R0_PAE_PROT_STR(name) "pgmR0BthPAEProt" #name
1362#define PGM_BTH_NAME_R0_PAE_32BIT_STR(name) "pgmR0BthPAE32Bit" #name
1363#define PGM_BTH_NAME_R0_PAE_PAE_STR(name) "pgmR0BthPAEPAE" #name
1364#define PGM_BTH_NAME_R0_AMD64_REAL_STR(name) "pgmR0BthAMD64Real" #name
1365#define PGM_BTH_NAME_R0_AMD64_PROT_STR(name) "pgmR0BthAMD64Prot" #name
1366#define PGM_BTH_NAME_R0_AMD64_AMD64_STR(name) "pgmR0BthAMD64AMD64" #name
1367#define PGM_BTH_DECL(type, name) PGM_CTX_DECL(type) PGM_BTH_NAME(name)
1368#define PGM_BTH_PFN(name, pVM) ((pVM)->pgm.s.PGM_CTX(pfn,Bth##name))
1369/** @} */
1370
1371/**
1372 * Data for each paging mode.
1373 */
1374typedef struct PGMMODEDATA
1375{
1376 /** The guest mode type. */
1377 uint32_t uGstType;
1378 /** The shadow mode type. */
1379 uint32_t uShwType;
1380
1381 /** @name Function pointers for Shadow paging.
1382 * @{
1383 */
1384 DECLR3CALLBACKMEMBER(int, pfnR3ShwRelocate,(PVM pVM, RTGCUINTPTR offDelta));
1385 DECLR3CALLBACKMEMBER(int, pfnR3ShwExit,(PVM pVM));
1386 DECLR3CALLBACKMEMBER(int, pfnR3ShwGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys));
1387 DECLR3CALLBACKMEMBER(int, pfnR3ShwModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1388 DECLR3CALLBACKMEMBER(int, pfnR3ShwGetPDEByIndex,(PVM pVM, uint32_t iPD, PX86PDEPAE pPde));
1389 DECLR3CALLBACKMEMBER(int, pfnR3ShwSetPDEByIndex,(PVM pVM, uint32_t iPD, X86PDEPAE Pde));
1390 DECLR3CALLBACKMEMBER(int, pfnR3ShwModifyPDEByIndex,(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask));
1391
1392 DECLGCCALLBACKMEMBER(int, pfnGCShwGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys));
1393 DECLGCCALLBACKMEMBER(int, pfnGCShwModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1394 DECLGCCALLBACKMEMBER(int, pfnGCShwGetPDEByIndex,(PVM pVM, uint32_t iPD, PX86PDEPAE pPde));
1395 DECLGCCALLBACKMEMBER(int, pfnGCShwSetPDEByIndex,(PVM pVM, uint32_t iPD, X86PDEPAE Pde));
1396 DECLGCCALLBACKMEMBER(int, pfnGCShwModifyPDEByIndex,(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask));
1397
1398 DECLR0CALLBACKMEMBER(int, pfnR0ShwGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys));
1399 DECLR0CALLBACKMEMBER(int, pfnR0ShwModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1400 DECLR0CALLBACKMEMBER(int, pfnR0ShwGetPDEByIndex,(PVM pVM, uint32_t iPD, PX86PDEPAE pPde));
1401 DECLR0CALLBACKMEMBER(int, pfnR0ShwSetPDEByIndex,(PVM pVM, uint32_t iPD, X86PDEPAE Pde));
1402 DECLR0CALLBACKMEMBER(int, pfnR0ShwModifyPDEByIndex,(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask));
1403 /** @} */
1404
1405 /** @name Function pointers for Guest paging.
1406 * @{
1407 */
1408 DECLR3CALLBACKMEMBER(int, pfnR3GstRelocate,(PVM pVM, RTGCUINTPTR offDelta));
1409 DECLR3CALLBACKMEMBER(int, pfnR3GstExit,(PVM pVM));
1410 DECLR3CALLBACKMEMBER(int, pfnR3GstGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTGCPHYS pGCPhys));
1411 DECLR3CALLBACKMEMBER(int, pfnR3GstModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1412 DECLR3CALLBACKMEMBER(int, pfnR3GstGetPDE,(PVM pVM, RTGCUINTPTR GCPtr, PX86PDEPAE pPde));
1413 DECLR3CALLBACKMEMBER(int, pfnR3GstMonitorCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1414 DECLR3CALLBACKMEMBER(int, pfnR3GstUnmonitorCR3,(PVM pVM));
1415 DECLR3CALLBACKMEMBER(int, pfnR3GstMapCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1416 DECLR3CALLBACKMEMBER(int, pfnR3GstUnmapCR3,(PVM pVM));
1417 R3PTRTYPE(PFNPGMR3PHYSHANDLER) pfnHCGstWriteHandlerCR3;
1418 R3PTRTYPE(const char *) pszHCGstWriteHandlerCR3;
1419
1420 DECLGCCALLBACKMEMBER(int, pfnGCGstGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTGCPHYS pGCPhys));
1421 DECLGCCALLBACKMEMBER(int, pfnGCGstModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1422 DECLGCCALLBACKMEMBER(int, pfnGCGstGetPDE,(PVM pVM, RTGCUINTPTR GCPtr, PX86PDEPAE pPde));
1423 DECLGCCALLBACKMEMBER(int, pfnGCGstMonitorCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1424 DECLGCCALLBACKMEMBER(int, pfnGCGstUnmonitorCR3,(PVM pVM));
1425 DECLGCCALLBACKMEMBER(int, pfnGCGstMapCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1426 DECLGCCALLBACKMEMBER(int, pfnGCGstUnmapCR3,(PVM pVM));
1427 GCPTRTYPE(PFNPGMGCPHYSHANDLER) pfnGCGstWriteHandlerCR3;
1428
1429 DECLR0CALLBACKMEMBER(int, pfnR0GstGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTGCPHYS pGCPhys));
1430 DECLR0CALLBACKMEMBER(int, pfnR0GstModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1431 DECLR0CALLBACKMEMBER(int, pfnR0GstGetPDE,(PVM pVM, RTGCUINTPTR GCPtr, PX86PDEPAE pPde));
1432 DECLR0CALLBACKMEMBER(int, pfnR0GstMonitorCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1433 DECLR0CALLBACKMEMBER(int, pfnR0GstUnmonitorCR3,(PVM pVM));
1434 DECLR0CALLBACKMEMBER(int, pfnR0GstMapCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1435 DECLR0CALLBACKMEMBER(int, pfnR0GstUnmapCR3,(PVM pVM));
1436 R0PTRTYPE(PFNPGMR0PHYSHANDLER) pfnR0GstWriteHandlerCR3;
1437 /** @} */
1438
1439 /** @name Function pointers for Both Shadow and Guest paging.
1440 * @{
1441 */
1442 DECLR3CALLBACKMEMBER(int, pfnR3BthRelocate,(PVM pVM, RTGCUINTPTR offDelta));
1443 DECLR3CALLBACKMEMBER(int, pfnR3BthTrap0eHandler,(PVM pVM, RTGCUINT uErr, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault));
1444 DECLR3CALLBACKMEMBER(int, pfnR3BthInvalidatePage,(PVM pVM, RTGCPTR GCPtrPage));
1445 DECLR3CALLBACKMEMBER(int, pfnR3BthSyncCR3,(PVM pVM, uint32_t cr0, uint32_t cr3, uint32_t cr4, bool fGlobal));
1446 DECLR3CALLBACKMEMBER(int, pfnR3BthSyncPage,(PVM pVM, VBOXPDE PdeSrc, RTGCUINTPTR GCPtrPage, unsigned cPages, unsigned uError));
1447 DECLR3CALLBACKMEMBER(int, pfnR3BthPrefetchPage,(PVM pVM, RTGCUINTPTR GCPtrPage));
1448 DECLR3CALLBACKMEMBER(int, pfnR3BthVerifyAccessSyncPage,(PVM pVM, RTGCUINTPTR GCPtrPage, unsigned fFlags, unsigned uError));
1449#ifdef VBOX_STRICT
1450 DECLR3CALLBACKMEMBER(unsigned, pfnR3BthAssertCR3,(PVM pVM, uint32_t cr3, uint32_t cr4, RTGCUINTPTR GCPtr, RTGCUINTPTR cb));
1451#endif
1452
1453 DECLGCCALLBACKMEMBER(int, pfnGCBthTrap0eHandler,(PVM pVM, RTGCUINT uErr, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault));
1454 DECLGCCALLBACKMEMBER(int, pfnGCBthInvalidatePage,(PVM pVM, RTGCPTR GCPtrPage));
1455 DECLGCCALLBACKMEMBER(int, pfnGCBthSyncCR3,(PVM pVM, uint32_t cr0, uint32_t cr3, uint32_t cr4, bool fGlobal));
1456 DECLGCCALLBACKMEMBER(int, pfnGCBthSyncPage,(PVM pVM, VBOXPDE PdeSrc, RTGCUINTPTR GCPtrPage, unsigned cPages, unsigned uError));
1457 DECLGCCALLBACKMEMBER(int, pfnGCBthPrefetchPage,(PVM pVM, RTGCUINTPTR GCPtrPage));
1458 DECLGCCALLBACKMEMBER(int, pfnGCBthVerifyAccessSyncPage,(PVM pVM, RTGCUINTPTR GCPtrPage, unsigned fFlags, unsigned uError));
1459#ifdef VBOX_STRICT
1460 DECLGCCALLBACKMEMBER(unsigned, pfnGCBthAssertCR3,(PVM pVM, uint32_t cr3, uint32_t cr4, RTGCUINTPTR GCPtr, RTGCUINTPTR cb));
1461#endif
1462
1463 DECLR0CALLBACKMEMBER(int, pfnR0BthTrap0eHandler,(PVM pVM, RTGCUINT uErr, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault));
1464 DECLR0CALLBACKMEMBER(int, pfnR0BthInvalidatePage,(PVM pVM, RTGCPTR GCPtrPage));
1465 DECLR0CALLBACKMEMBER(int, pfnR0BthSyncCR3,(PVM pVM, uint32_t cr0, uint32_t cr3, uint32_t cr4, bool fGlobal));
1466 DECLR0CALLBACKMEMBER(int, pfnR0BthSyncPage,(PVM pVM, VBOXPDE PdeSrc, RTGCUINTPTR GCPtrPage, unsigned cPages, unsigned uError));
1467 DECLR0CALLBACKMEMBER(int, pfnR0BthPrefetchPage,(PVM pVM, RTGCUINTPTR GCPtrPage));
1468 DECLR0CALLBACKMEMBER(int, pfnR0BthVerifyAccessSyncPage,(PVM pVM, RTGCUINTPTR GCPtrPage, unsigned fFlags, unsigned uError));
1469#ifdef VBOX_STRICT
1470 DECLR0CALLBACKMEMBER(unsigned, pfnR0BthAssertCR3,(PVM pVM, uint32_t cr3, uint32_t cr4, RTGCUINTPTR GCPtr, RTGCUINTPTR cb));
1471#endif
1472 /** @} */
1473} PGMMODEDATA, *PPGMMODEDATA;
1474
1475
1476
1477/**
1478 * Converts a PGM pointer into a VM pointer.
1479 * @returns Pointer to the VM structure the PGM is part of.
1480 * @param pPGM Pointer to PGM instance data.
1481 */
1482#define PGM2VM(pPGM) ( (PVM)((char*)pPGM - pPGM->offVM) )
1483
1484/**
1485 * PGM Data (part of VM)
1486 */
1487typedef struct PGM
1488{
1489 /** Offset to the VM structure. */
1490 RTINT offVM;
1491
1492 /*
1493 * This will be redefined at least two more times before we're done, I'm sure.
1494 * The current code is only to get on with the coding.
1495 * - 2004-06-10: initial version, bird.
1496 * - 2004-07-02: 1st time, bird.
1497 * - 2004-10-18: 2nd time, bird.
1498 * - 2005-07-xx: 3rd time, bird.
1499 */
1500
1501 /** Pointer to the page table entries for the dynamic page mapping area - GCPtr. */
1502 GCPTRTYPE(PX86PTE) paDynPageMap32BitPTEsGC;
1503 /** Pointer to the page table entries for the dynamic page mapping area - GCPtr. */
1504 GCPTRTYPE(PX86PTEPAE) paDynPageMapPaePTEsGC;
1505
1506 /** The host paging mode. (This is what SUPLib reports.) */
1507 SUPPAGINGMODE enmHostMode;
1508 /** The shadow paging mode. */
1509 PGMMODE enmShadowMode;
1510 /** The guest paging mode. */
1511 PGMMODE enmGuestMode;
1512
1513 /** The current physical address representing in the guest CR3 register. */
1514 RTGCPHYS GCPhysCR3;
1515 /** Pointer to the 5 page CR3 content mapping.
1516 * The first page is always the CR3 (in some form) while the 4 other pages
1517 * are used of the PDs in PAE mode. */
1518 RTGCPTR GCPtrCR3Mapping;
1519 /** The physical address of the currently monitored guest CR3 page.
1520 * When this value is NIL_RTGCPHYS no page is being monitored. */
1521 RTGCPHYS GCPhysGstCR3Monitored;
1522#if HC_ARCH_BITS == 64 || GC_ARCH_BITS == 64
1523 RTGCPHYS GCPhysPadding0; /**< alignment padding. */
1524#endif
1525
1526 /** @name 32-bit Guest Paging.
1527 * @{ */
1528 /** The guest's page directory, HC pointer. */
1529 R3R0PTRTYPE(PVBOXPD) pGuestPDHC;
1530 /** The guest's page directory, static GC mapping. */
1531 GCPTRTYPE(PVBOXPD) pGuestPDGC;
1532 /** @} */
1533
1534 /** @name PAE Guest Paging.
1535 * @{ */
1536 /** The guest's page directory pointer table, static GC mapping. */
1537 GCPTRTYPE(PX86PDPTR) pGstPaePDPTRGC;
1538 /** The guest's page directory pointer table, HC pointer. */
1539 R3R0PTRTYPE(PX86PDPTR) pGstPaePDPTRHC;
1540 /** The guest's page directories, HC pointers.
1541 * These are individual pointers and doesn't have to be adjecent.
1542 * These doesn't have to be update to date - use pgmGstGetPaePD() to access them. */
1543 R3R0PTRTYPE(PX86PDPAE) apGstPaePDsHC[4];
1544 /** The guest's page directories, static GC mapping.
1545 * Unlike the HC array the first entry can be accessed as a 2048 entry PD.
1546 * These doesn't have to be update to date - use pgmGstGetPaePD() to access them. */
1547 GCPTRTYPE(PX86PDPAE) apGstPaePDsGC[4];
1548 /** The physical addresses of the guest page directories (PAE) pointed to by apGstPagePDsHC/GC. */
1549 RTGCPHYS aGCPhysGstPaePDs[4];
1550 /** The physical addresses of the monitored guest page directories (PAE). */
1551 RTGCPHYS aGCPhysGstPaePDsMonitored[4];
1552 /** @} */
1553
1554
1555 /** @name 32-bit Shadow Paging
1556 * @{ */
1557 /** The 32-Bit PD - HC Ptr. */
1558 R3R0PTRTYPE(PX86PD) pHC32BitPD;
1559 /** The 32-Bit PD - GC Ptr. */
1560 GCPTRTYPE(PX86PD) pGC32BitPD;
1561#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32
1562 uint32_t u32Padding1; /**< alignment padding. */
1563#endif
1564 /** The Physical Address (HC) of the 32-Bit PD. */
1565 RTHCPHYS HCPhys32BitPD;
1566 /** @} */
1567
1568 /** @name PAE Shadow Paging
1569 * @{ */
1570 /** The four PDs for the low 4GB - HC Ptr.
1571 * Even though these are 4 pointers, what they point at is a single table.
1572 * Thus, it's possible to walk the 2048 entries starting where apHCPaePDs[0] points. */
1573 R3R0PTRTYPE(PX86PDPAE) apHCPaePDs[4];
1574 /** The four PDs for the low 4GB - GC Ptr.
1575 * Same kind of mapping as apHCPaePDs. */
1576 GCPTRTYPE(PX86PDPAE) apGCPaePDs[4];
1577 /** The Physical Address (HC) of the four PDs for the low 4GB.
1578 * These are *NOT* 4 contiguous pages. */
1579 RTHCPHYS aHCPhysPaePDs[4];
1580 /** The PAE PDPTR - HC Ptr. */
1581 R3R0PTRTYPE(PX86PDPTR) pHCPaePDPTR;
1582 /** The Physical Address (HC) of the PAE PDPTR. */
1583 RTHCPHYS HCPhysPaePDPTR;
1584 /** The PAE PDPTR - GC Ptr. */
1585 GCPTRTYPE(PX86PDPTR) pGCPaePDPTR;
1586 /** @} */
1587
1588 /** @name AMD64 Shadow Paging
1589 * Extends PAE Paging.
1590 * @{ */
1591 /** The Page Map Level 4 table - HC Ptr. */
1592 GCPTRTYPE(PX86PML4) pGCPaePML4;
1593 /** The Page Map Level 4 table - GC Ptr. */
1594 R3R0PTRTYPE(PX86PML4) pHCPaePML4;
1595 /** The Physical Address (HC) of the Page Map Level 4 table. */
1596 RTHCPHYS HCPhysPaePML4;
1597 /** @}*/
1598
1599 /** @name Function pointers for Shadow paging.
1600 * @{
1601 */
1602 DECLR3CALLBACKMEMBER(int, pfnR3ShwRelocate,(PVM pVM, RTGCUINTPTR offDelta));
1603 DECLR3CALLBACKMEMBER(int, pfnR3ShwExit,(PVM pVM));
1604 DECLR3CALLBACKMEMBER(int, pfnR3ShwGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys));
1605 DECLR3CALLBACKMEMBER(int, pfnR3ShwModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1606 DECLR3CALLBACKMEMBER(int, pfnR3ShwGetPDEByIndex,(PVM pVM, uint32_t iPD, PX86PDEPAE pPde));
1607 DECLR3CALLBACKMEMBER(int, pfnR3ShwSetPDEByIndex,(PVM pVM, uint32_t iPD, X86PDEPAE Pde));
1608 DECLR3CALLBACKMEMBER(int, pfnR3ShwModifyPDEByIndex,(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask));
1609
1610 DECLGCCALLBACKMEMBER(int, pfnGCShwGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys));
1611 DECLGCCALLBACKMEMBER(int, pfnGCShwModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1612 DECLGCCALLBACKMEMBER(int, pfnGCShwGetPDEByIndex,(PVM pVM, uint32_t iPD, PX86PDEPAE pPde));
1613 DECLGCCALLBACKMEMBER(int, pfnGCShwSetPDEByIndex,(PVM pVM, uint32_t iPD, X86PDEPAE Pde));
1614 DECLGCCALLBACKMEMBER(int, pfnGCShwModifyPDEByIndex,(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask));
1615#if GC_ARCH_BITS == 32 && HC_ARCH_BITS == 64
1616 RTGCPTR alignment0; /**< structure size alignment. */
1617#endif
1618
1619 DECLR0CALLBACKMEMBER(int, pfnR0ShwGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys));
1620 DECLR0CALLBACKMEMBER(int, pfnR0ShwModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1621 DECLR0CALLBACKMEMBER(int, pfnR0ShwGetPDEByIndex,(PVM pVM, uint32_t iPD, PX86PDEPAE pPde));
1622 DECLR0CALLBACKMEMBER(int, pfnR0ShwSetPDEByIndex,(PVM pVM, uint32_t iPD, X86PDEPAE Pde));
1623 DECLR0CALLBACKMEMBER(int, pfnR0ShwModifyPDEByIndex,(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask));
1624
1625 /** @} */
1626
1627 /** @name Function pointers for Guest paging.
1628 * @{
1629 */
1630 DECLR3CALLBACKMEMBER(int, pfnR3GstRelocate,(PVM pVM, RTGCUINTPTR offDelta));
1631 DECLR3CALLBACKMEMBER(int, pfnR3GstExit,(PVM pVM));
1632 DECLR3CALLBACKMEMBER(int, pfnR3GstGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTGCPHYS pGCPhys));
1633 DECLR3CALLBACKMEMBER(int, pfnR3GstModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1634 DECLR3CALLBACKMEMBER(int, pfnR3GstGetPDE,(PVM pVM, RTGCUINTPTR GCPtr, PX86PDEPAE pPde));
1635 DECLR3CALLBACKMEMBER(int, pfnR3GstMonitorCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1636 DECLR3CALLBACKMEMBER(int, pfnR3GstUnmonitorCR3,(PVM pVM));
1637 DECLR3CALLBACKMEMBER(int, pfnR3GstMapCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1638 DECLR3CALLBACKMEMBER(int, pfnR3GstUnmapCR3,(PVM pVM));
1639 R3PTRTYPE(PFNPGMR3PHYSHANDLER) pfnHCGstWriteHandlerCR3;
1640 R3PTRTYPE(const char *) pszHCGstWriteHandlerCR3;
1641
1642 DECLGCCALLBACKMEMBER(int, pfnGCGstGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTGCPHYS pGCPhys));
1643 DECLGCCALLBACKMEMBER(int, pfnGCGstModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1644 DECLGCCALLBACKMEMBER(int, pfnGCGstGetPDE,(PVM pVM, RTGCUINTPTR GCPtr, PX86PDEPAE pPde));
1645 DECLGCCALLBACKMEMBER(int, pfnGCGstMonitorCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1646 DECLGCCALLBACKMEMBER(int, pfnGCGstUnmonitorCR3,(PVM pVM));
1647 DECLGCCALLBACKMEMBER(int, pfnGCGstMapCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1648 DECLGCCALLBACKMEMBER(int, pfnGCGstUnmapCR3,(PVM pVM));
1649 GCPTRTYPE(PFNPGMGCPHYSHANDLER) pfnGCGstWriteHandlerCR3;
1650
1651 DECLR0CALLBACKMEMBER(int, pfnR0GstGetPage,(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTGCPHYS pGCPhys));
1652 DECLR0CALLBACKMEMBER(int, pfnR0GstModifyPage,(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask));
1653 DECLR0CALLBACKMEMBER(int, pfnR0GstGetPDE,(PVM pVM, RTGCUINTPTR GCPtr, PX86PDEPAE pPde));
1654 DECLR0CALLBACKMEMBER(int, pfnR0GstMonitorCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1655 DECLR0CALLBACKMEMBER(int, pfnR0GstUnmonitorCR3,(PVM pVM));
1656 DECLR0CALLBACKMEMBER(int, pfnR0GstMapCR3,(PVM pVM, RTGCPHYS GCPhysCR3));
1657 DECLR0CALLBACKMEMBER(int, pfnR0GstUnmapCR3,(PVM pVM));
1658 R0PTRTYPE(PFNPGMR0PHYSHANDLER) pfnR0GstWriteHandlerCR3;
1659 /** @} */
1660
1661 /** @name Function pointers for Both Shadow and Guest paging.
1662 * @{
1663 */
1664 DECLR3CALLBACKMEMBER(int, pfnR3BthRelocate,(PVM pVM, RTGCUINTPTR offDelta));
1665 DECLR3CALLBACKMEMBER(int, pfnR3BthTrap0eHandler,(PVM pVM, RTGCUINT uErr, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault));
1666 DECLR3CALLBACKMEMBER(int, pfnR3BthInvalidatePage,(PVM pVM, RTGCPTR GCPtrPage));
1667 DECLR3CALLBACKMEMBER(int, pfnR3BthSyncCR3,(PVM pVM, uint32_t cr0, uint32_t cr3, uint32_t cr4, bool fGlobal));
1668 DECLR3CALLBACKMEMBER(int, pfnR3BthSyncPage,(PVM pVM, VBOXPDE PdeSrc, RTGCUINTPTR GCPtrPage, unsigned cPages, unsigned uError));
1669 DECLR3CALLBACKMEMBER(int, pfnR3BthPrefetchPage,(PVM pVM, RTGCUINTPTR GCPtrPage));
1670 DECLR3CALLBACKMEMBER(int, pfnR3BthVerifyAccessSyncPage,(PVM pVM, RTGCUINTPTR GCPtrPage, unsigned fFlags, unsigned uError));
1671 DECLR3CALLBACKMEMBER(unsigned, pfnR3BthAssertCR3,(PVM pVM, uint32_t cr3, uint32_t cr4, RTGCUINTPTR GCPtr, RTGCUINTPTR cb));
1672
1673 DECLR0CALLBACKMEMBER(int, pfnR0BthTrap0eHandler,(PVM pVM, RTGCUINT uErr, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault));
1674 DECLR0CALLBACKMEMBER(int, pfnR0BthInvalidatePage,(PVM pVM, RTGCPTR GCPtrPage));
1675 DECLR0CALLBACKMEMBER(int, pfnR0BthSyncCR3,(PVM pVM, uint32_t cr0, uint32_t cr3, uint32_t cr4, bool fGlobal));
1676 DECLR0CALLBACKMEMBER(int, pfnR0BthSyncPage,(PVM pVM, VBOXPDE PdeSrc, RTGCUINTPTR GCPtrPage, unsigned cPages, unsigned uError));
1677 DECLR0CALLBACKMEMBER(int, pfnR0BthPrefetchPage,(PVM pVM, RTGCUINTPTR GCPtrPage));
1678 DECLR0CALLBACKMEMBER(int, pfnR0BthVerifyAccessSyncPage,(PVM pVM, RTGCUINTPTR GCPtrPage, unsigned fFlags, unsigned uError));
1679 DECLR0CALLBACKMEMBER(unsigned, pfnR0BthAssertCR3,(PVM pVM, uint32_t cr3, uint32_t cr4, RTGCUINTPTR GCPtr, RTGCUINTPTR cb));
1680
1681 DECLGCCALLBACKMEMBER(int, pfnGCBthTrap0eHandler,(PVM pVM, RTGCUINT uErr, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault));
1682 DECLGCCALLBACKMEMBER(int, pfnGCBthInvalidatePage,(PVM pVM, RTGCPTR GCPtrPage));
1683 DECLGCCALLBACKMEMBER(int, pfnGCBthSyncCR3,(PVM pVM, uint32_t cr0, uint32_t cr3, uint32_t cr4, bool fGlobal));
1684 DECLGCCALLBACKMEMBER(int, pfnGCBthSyncPage,(PVM pVM, VBOXPDE PdeSrc, RTGCUINTPTR GCPtrPage, unsigned cPages, unsigned uError));
1685 DECLGCCALLBACKMEMBER(int, pfnGCBthPrefetchPage,(PVM pVM, RTGCUINTPTR GCPtrPage));
1686 DECLGCCALLBACKMEMBER(int, pfnGCBthVerifyAccessSyncPage,(PVM pVM, RTGCUINTPTR GCPtrPage, unsigned fFlags, unsigned uError));
1687 DECLGCCALLBACKMEMBER(unsigned, pfnGCBthAssertCR3,(PVM pVM, uint32_t cr3, uint32_t cr4, RTGCUINTPTR GCPtr, RTGCUINTPTR cb));
1688#if GC_ARCH_BITS == 32 && HC_ARCH_BITS == 64
1689 RTGCPTR alignment2; /**< structure size alignment. */
1690#endif
1691 /** @} */
1692
1693 /** Pointer to SHW+GST mode data (function pointers).
1694 * The index into this table is made up from */
1695 R3PTRTYPE(PPGMMODEDATA) paModeData;
1696
1697
1698 /** Pointer to the list of RAM ranges (Phys GC -> Phys HC conversion) - for HC.
1699 * This is sorted by physical address and contains no overlaps.
1700 * The memory locks and other conversions are managed by MM at the moment.
1701 */
1702 R3R0PTRTYPE(PPGMRAMRANGE) pRamRangesHC;
1703 /** Pointer to the list of RAM ranges (Phys GC -> Phys HC conversion) - for GC.
1704 * This is sorted by physical address and contains no overlaps.
1705 * The memory locks and other conversions are managed by MM at the moment.
1706 */
1707 GCPTRTYPE(PPGMRAMRANGE) pRamRangesGC;
1708 /** The configured RAM size. */
1709 RTUINT cbRamSize;
1710
1711 /** PGM offset based trees - HC Ptr. */
1712 R3R0PTRTYPE(PPGMTREES) pTreesHC;
1713 /** PGM offset based trees - GC Ptr. */
1714 GCPTRTYPE(PPGMTREES) pTreesGC;
1715
1716 /** Linked list of GC mappings - for GC.
1717 * The list is sorted ascending on address.
1718 */
1719 GCPTRTYPE(PPGMMAPPING) pMappingsGC;
1720 /** Linked list of GC mappings - for HC.
1721 * The list is sorted ascending on address.
1722 */
1723 R3PTRTYPE(PPGMMAPPING) pMappingsR3;
1724 /** Linked list of GC mappings - for R0.
1725 * The list is sorted ascending on address.
1726 */
1727 R0PTRTYPE(PPGMMAPPING) pMappingsR0;
1728
1729 /** If set no conflict checks are required. (boolean) */
1730 bool fMappingsFixed;
1731 /** If set, then no mappings are put into the shadow page table. (boolean) */
1732 bool fDisableMappings;
1733 /** Size of fixed mapping */
1734 uint32_t cbMappingFixed;
1735 /** Base address (GC) of fixed mapping */
1736 RTGCPTR GCPtrMappingFixed;
1737#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32
1738 uint32_t u32Padding0; /**< alignment padding. */
1739#endif
1740
1741
1742 /** @name Intermediate Context
1743 * @{ */
1744 /** Pointer to the intermediate page directory - Normal. */
1745 R3PTRTYPE(PX86PD) pInterPD;
1746 /** Pointer to the intermedate page tables - Normal.
1747 * There are two page tables, one for the identity mapping and one for
1748 * the host context mapping (of the core code). */
1749 R3PTRTYPE(PX86PT) apInterPTs[2];
1750 /** Pointer to the intermedate page tables - PAE. */
1751 R3PTRTYPE(PX86PTPAE) apInterPaePTs[2];
1752 /** Pointer to the intermedate page directory - PAE. */
1753 R3PTRTYPE(PX86PDPAE) apInterPaePDs[4];
1754 /** Pointer to the intermedate page directory - PAE. */
1755 R3PTRTYPE(PX86PDPTR) pInterPaePDPTR;
1756 /** Pointer to the intermedate page-map level 4 - AMD64. */
1757 R3PTRTYPE(PX86PML4) pInterPaePML4;
1758 /** Pointer to the intermedate page directory - AMD64. */
1759 R3PTRTYPE(PX86PDPTR) pInterPaePDPTR64;
1760 /** The Physical Address (HC) of the intermediate Page Directory - Normal. */
1761 RTHCPHYS HCPhysInterPD;
1762 /** The Physical Address (HC) of the intermediate Page Directory Pointer Table - PAE. */
1763 RTHCPHYS HCPhysInterPaePDPTR;
1764 /** The Physical Address (HC) of the intermediate Page Map Level 4 table - AMD64. */
1765 RTHCPHYS HCPhysInterPaePML4;
1766 /** @} */
1767
1768 /** Base address of the dynamic page mapping area.
1769 * The array is MM_HYPER_DYNAMIC_SIZE bytes big.
1770 */
1771 GCPTRTYPE(uint8_t *) pbDynPageMapBaseGC;
1772 /** The index of the last entry used in the dynamic page mapping area. */
1773 RTUINT iDynPageMapLast;
1774 /** Cache containing the last entries in the dynamic page mapping area.
1775 * The cache size is covering half of the mapping area. */
1776 RTHCPHYS aHCPhysDynPageMapCache[MM_HYPER_DYNAMIC_SIZE >> (PAGE_SHIFT + 1)];
1777
1778 /** A20 gate mask.
1779 * Our current approach to A20 emulation is to let REM do it and don't bother
1780 * anywhere else. The interesting Guests will be operating with it enabled anyway.
1781 * But whould need arrise, we'll subject physical addresses to this mask. */
1782 RTGCPHYS GCPhysA20Mask;
1783 /** A20 gate state - boolean! */
1784 RTUINT fA20Enabled;
1785
1786 /** What needs syncing (PGM_SYNC_*).
1787 * This is used to queue operations for PGMSyncCR3, PGMInvalidatePage,
1788 * PGMFlushTLB, and PGMR3Load. */
1789 RTUINT fSyncFlags;
1790
1791#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32
1792 RTUINT uPadding3; /**< alignment padding. */
1793#endif
1794 /** PGM critical section.
1795 * This protects the physical & virtual access handlers, ram ranges,
1796 * and the page flag updating (some of it anyway).
1797 */
1798 PDMCRITSECT CritSect;
1799
1800 /** Shadow Page Pool - HC Ptr. */
1801 R3R0PTRTYPE(PPGMPOOL) pPoolHC;
1802 /** Shadow Page Pool - GC Ptr. */
1803 GCPTRTYPE(PPGMPOOL) pPoolGC;
1804
1805 /** We're not in a state which permits writes to guest memory.
1806 * (Only used in strict builds.) */
1807 bool fNoMorePhysWrites;
1808
1809 /** Flush the cache on the next access. */
1810 bool fPhysCacheFlushPending;
1811/** @todo r=bird: Fix member names!*/
1812 /** PGMPhysRead cache */
1813 PGMPHYSCACHE pgmphysreadcache;
1814 /** PGMPhysWrite cache */
1815 PGMPHYSCACHE pgmphyswritecache;
1816
1817 /**
1818 * Data associated with managing the ring-3 mappings of the allocation chunks.
1819 */
1820 struct
1821 {
1822 /** The chunk tree, ordered by chunk id. */
1823 R3R0PTRTYPE(PAVLU32NODECORE) pTree;
1824 /** The chunk mapping TLB. */
1825 PGMCHUNKR3MAPTLB Tlb;
1826 /** The number of mapped chunks. */
1827 uint32_t c;
1828 /** The maximum number of mapped chunks.
1829 * @cfgm PGM/MaxRing3Chunks */
1830 uint32_t cMax;
1831 /** The chunk age tree, ordered by ageing sequence number. */
1832 R3PTRTYPE(PAVLLU32NODECORE) pAgeTree;
1833 /** The current time. */
1834 uint32_t iNow;
1835 /** Number of pgmR3PhysChunkFindUnmapCandidate calls left to the next ageing. */
1836 uint32_t AgeingCountdown;
1837 } ChunkR3Map;
1838
1839 /**
1840 * The page mapping TLB for ring-3 and (for the time being) ring-0.
1841 */
1842 PGMPAGER3MAPTLB PhysTlbHC;
1843
1844 /** @name The zero page.
1845 * @{ */
1846 /** The host physical address of the zero page. */
1847 RTHCPHYS HCPhysZeroPg;
1848 /** The ring-3 mapping of the zero page. */
1849 RTR3PTR pvZeroPgR3;
1850 /** The ring-0 mapping of the zero page. */
1851 RTR0PTR pvZeroPgR0;
1852 /** The GC mapping of the zero page. */
1853 RTGCPTR pvZeroPgGC;
1854#if GC_ARCH_BITS != 32
1855 uint32_t u32ZeroAlignment; /**< Alignment padding. */
1856#endif
1857 /** @}*/
1858
1859 /** The number of handy pages. */
1860 uint32_t cHandyPages;
1861 /**
1862 * Array of handy pages.
1863 *
1864 * This array is used in a two way communication between pgmPhysAllocPage
1865 * and GMMR0AllocateHandyPages, with PGMR3PhysAllocateHandyPages serving as
1866 * an intermediary.
1867 *
1868 * The size of this array is important, see pgmPhysEnsureHandyPage for details.
1869 * (The current size of 32 pages, means 128 KB of memory.)
1870 */
1871 struct
1872 {
1873 /** The host physical address before pgmPhysAllocPage uses it,
1874 * and the guest physical address afterwards.
1875 * This is NIL_RTHCPHYS if the array entry isn't valid.
1876 * ASSUMES: sizeof(RTHCPHYS) >= sizeof(RTHCPHYS). */
1877 RTHCPHYS HCPhysGCPhys;
1878 /** The Page ID.
1879 * This is NIL_GMM_PAGEID if the array entry isn't valid. */
1880 uint32_t idPage;
1881 /** The Page ID of the shared page that pgmPageAllocPage replaced.
1882 * This is NIL_GMM_PAGEID if no shared page was replaced. */
1883 uint32_t idSharedPage;
1884 } aHandyPages[32];
1885
1886 /** @name Release Statistics
1887 * @{ */
1888 uint32_t cAllPages; /**< The total number of pages. (Should be Private + Shared + Zero.) */
1889 uint32_t cPrivatePages; /**< The number of private pages. */
1890 uint32_t cSharedPages; /**< The number of shared pages. */
1891 uint32_t cZeroPages; /**< The number of zero backed pages. */
1892 /** The number of times the guest has switched mode since last reset or statistics reset. */
1893 STAMCOUNTER cGuestModeChanges;
1894 /** @} */
1895
1896#ifdef VBOX_WITH_STATISTICS
1897 /** GC: Which statistic this \#PF should be attributed to. */
1898 GCPTRTYPE(PSTAMPROFILE) pStatTrap0eAttributionGC;
1899 RTGCPTR padding0;
1900 /** HC: Which statistic this \#PF should be attributed to. */
1901 R3R0PTRTYPE(PSTAMPROFILE) pStatTrap0eAttributionHC;
1902 RTHCPTR padding1;
1903 STAMPROFILE StatGCTrap0e; /**< GC: PGMGCTrap0eHandler() profiling. */
1904 STAMPROFILE StatTrap0eCSAM; /**< Profiling of the Trap0eHandler body when the cause is CSAM. */
1905 STAMPROFILE StatTrap0eDirtyAndAccessedBits; /**< Profiling of the Trap0eHandler body when the cause is dirty and/or accessed bit emulation. */
1906 STAMPROFILE StatTrap0eGuestTrap; /**< Profiling of the Trap0eHandler body when the cause is a guest trap. */
1907 STAMPROFILE StatTrap0eHndPhys; /**< Profiling of the Trap0eHandler body when the cause is a physical handler. */
1908 STAMPROFILE StatTrap0eHndVirt; /**< Profiling of the Trap0eHandler body when the cause is a virtual handler. */
1909 STAMPROFILE StatTrap0eHndUnhandled; /**< Profiling of the Trap0eHandler body when the cause is access outside the monitored areas of a monitored page. */
1910 STAMPROFILE StatTrap0eMisc; /**< Profiling of the Trap0eHandler body when the cause is not known. */
1911 STAMPROFILE StatTrap0eOutOfSync; /**< Profiling of the Trap0eHandler body when the cause is an out-of-sync page. */
1912 STAMPROFILE StatTrap0eOutOfSyncHndPhys; /**< Profiling of the Trap0eHandler body when the cause is an out-of-sync physical handler page. */
1913 STAMPROFILE StatTrap0eOutOfSyncHndVirt; /**< Profiling of the Trap0eHandler body when the cause is an out-of-sync virtual handler page. */
1914 STAMPROFILE StatTrap0eOutOfSyncObsHnd; /**< Profiling of the Trap0eHandler body when the cause is an obsolete handler page. */
1915 STAMPROFILE StatTrap0eSyncPT; /**< Profiling of the Trap0eHandler body when the cause is lazy syncing of a PT. */
1916
1917 STAMCOUNTER StatTrap0eMapHandler; /**< Number of traps due to access handlers in mappings. */
1918 STAMCOUNTER StatGCTrap0eConflicts; /**< GC: The number of times \#PF was caused by an undetected conflict. */
1919
1920 STAMCOUNTER StatGCTrap0eUSNotPresentRead;
1921 STAMCOUNTER StatGCTrap0eUSNotPresentWrite;
1922 STAMCOUNTER StatGCTrap0eUSWrite;
1923 STAMCOUNTER StatGCTrap0eUSReserved;
1924 STAMCOUNTER StatGCTrap0eUSRead;
1925
1926 STAMCOUNTER StatGCTrap0eSVNotPresentRead;
1927 STAMCOUNTER StatGCTrap0eSVNotPresentWrite;
1928 STAMCOUNTER StatGCTrap0eSVWrite;
1929 STAMCOUNTER StatGCTrap0eSVReserved;
1930
1931 STAMCOUNTER StatGCTrap0eUnhandled;
1932 STAMCOUNTER StatGCTrap0eMap;
1933
1934 /** GC: PGMSyncPT() profiling. */
1935 STAMPROFILE StatGCSyncPT;
1936 /** GC: The number of times PGMSyncPT() needed to allocate page tables. */
1937 STAMCOUNTER StatGCSyncPTAlloc;
1938 /** GC: The number of times PGMSyncPT() detected conflicts. */
1939 STAMCOUNTER StatGCSyncPTConflict;
1940 /** GC: The number of times PGMSyncPT() failed. */
1941 STAMCOUNTER StatGCSyncPTFailed;
1942 /** GC: PGMGCInvalidatePage() profiling. */
1943 STAMPROFILE StatGCInvalidatePage;
1944 /** GC: The number of times PGMGCInvalidatePage() was called for a 4KB page. */
1945 STAMCOUNTER StatGCInvalidatePage4KBPages;
1946 /** GC: The number of times PGMGCInvalidatePage() was called for a 4MB page. */
1947 STAMCOUNTER StatGCInvalidatePage4MBPages;
1948 /** GC: The number of times PGMGCInvalidatePage() skipped a 4MB page. */
1949 STAMCOUNTER StatGCInvalidatePage4MBPagesSkip;
1950 /** GC: The number of times PGMGCInvalidatePage() was called for a not accessed page directory. */
1951 STAMCOUNTER StatGCInvalidatePagePDNAs;
1952 /** GC: The number of times PGMGCInvalidatePage() was called for a not present page directory. */
1953 STAMCOUNTER StatGCInvalidatePagePDNPs;
1954 /** GC: The number of times PGMGCInvalidatePage() was called for a page directory containing mappings (no conflict). */
1955 STAMCOUNTER StatGCInvalidatePagePDMappings;
1956 /** GC: The number of times PGMGCInvalidatePage() was called for an out of sync page directory. */
1957 STAMCOUNTER StatGCInvalidatePagePDOutOfSync;
1958 /** HC: The number of times PGMGCInvalidatePage() was skipped due to not present shw or pending pending SyncCR3. */
1959 STAMCOUNTER StatGCInvalidatePageSkipped;
1960 /** GC: The number of times user page is out of sync was detected in GC. */
1961 STAMCOUNTER StatGCPageOutOfSyncUser;
1962 /** GC: The number of times supervisor page is out of sync was detected in GC. */
1963 STAMCOUNTER StatGCPageOutOfSyncSupervisor;
1964 /** GC: The number of dynamic page mapping cache hits */
1965 STAMCOUNTER StatDynMapCacheMisses;
1966 /** GC: The number of dynamic page mapping cache misses */
1967 STAMCOUNTER StatDynMapCacheHits;
1968 /** GC: The number of times pgmGCGuestPDWriteHandler() was successfully called. */
1969 STAMCOUNTER StatGCGuestCR3WriteHandled;
1970 /** GC: The number of times pgmGCGuestPDWriteHandler() was called and we had to fall back to the recompiler. */
1971 STAMCOUNTER StatGCGuestCR3WriteUnhandled;
1972 /** GC: The number of times pgmGCGuestPDWriteHandler() was called and a conflict was detected. */
1973 STAMCOUNTER StatGCGuestCR3WriteConflict;
1974 /** GC: Number of out-of-sync handled pages. */
1975 STAMCOUNTER StatHandlersOutOfSync;
1976 /** GC: Number of traps due to physical access handlers. */
1977 STAMCOUNTER StatHandlersPhysical;
1978 /** GC: Number of traps due to virtual access handlers. */
1979 STAMCOUNTER StatHandlersVirtual;
1980 /** GC: Number of traps due to virtual access handlers found by physical address. */
1981 STAMCOUNTER StatHandlersVirtualByPhys;
1982 /** GC: Number of traps due to virtual access handlers found by virtual address (without proper physical flags). */
1983 STAMCOUNTER StatHandlersVirtualUnmarked;
1984 /** GC: Number of traps due to access outside range of monitored page(s). */
1985 STAMCOUNTER StatHandlersUnhandled;
1986
1987 /** GC: The number of times pgmGCGuestROMWriteHandler() was successfully called. */
1988 STAMCOUNTER StatGCGuestROMWriteHandled;
1989 /** GC: The number of times pgmGCGuestROMWriteHandler() was called and we had to fall back to the recompiler */
1990 STAMCOUNTER StatGCGuestROMWriteUnhandled;
1991
1992 /** HC: PGMR3InvalidatePage() profiling. */
1993 STAMPROFILE StatHCInvalidatePage;
1994 /** HC: The number of times PGMR3InvalidatePage() was called for a 4KB page. */
1995 STAMCOUNTER StatHCInvalidatePage4KBPages;
1996 /** HC: The number of times PGMR3InvalidatePage() was called for a 4MB page. */
1997 STAMCOUNTER StatHCInvalidatePage4MBPages;
1998 /** HC: The number of times PGMR3InvalidatePage() skipped a 4MB page. */
1999 STAMCOUNTER StatHCInvalidatePage4MBPagesSkip;
2000 /** HC: The number of times PGMR3InvalidatePage() was called for a not accessed page directory. */
2001 STAMCOUNTER StatHCInvalidatePagePDNAs;
2002 /** HC: The number of times PGMR3InvalidatePage() was called for a not present page directory. */
2003 STAMCOUNTER StatHCInvalidatePagePDNPs;
2004 /** HC: The number of times PGMR3InvalidatePage() was called for a page directory containing mappings (no conflict). */
2005 STAMCOUNTER StatHCInvalidatePagePDMappings;
2006 /** HC: The number of times PGMGCInvalidatePage() was called for an out of sync page directory. */
2007 STAMCOUNTER StatHCInvalidatePagePDOutOfSync;
2008 /** HC: The number of times PGMR3InvalidatePage() was skipped due to not present shw or pending pending SyncCR3. */
2009 STAMCOUNTER StatHCInvalidatePageSkipped;
2010 /** HC: PGMR3SyncPT() profiling. */
2011 STAMPROFILE StatHCSyncPT;
2012 /** HC: pgmr3SyncPTResolveConflict() profiling (includes the entire relocation). */
2013 STAMPROFILE StatHCResolveConflict;
2014 /** HC: Number of times PGMR3CheckMappingConflicts() detected a conflict. */
2015 STAMCOUNTER StatHCDetectedConflicts;
2016 /** HC: The total number of times pgmHCGuestPDWriteHandler() was called. */
2017 STAMCOUNTER StatHCGuestPDWrite;
2018 /** HC: The number of times pgmHCGuestPDWriteHandler() detected a conflict */
2019 STAMCOUNTER StatHCGuestPDWriteConflict;
2020
2021 /** HC: The number of pages marked not present for accessed bit emulation. */
2022 STAMCOUNTER StatHCAccessedPage;
2023 /** HC: The number of pages marked read-only for dirty bit tracking. */
2024 STAMCOUNTER StatHCDirtyPage;
2025 /** HC: The number of pages marked read-only for dirty bit tracking. */
2026 STAMCOUNTER StatHCDirtyPageBig;
2027 /** HC: The number of traps generated for dirty bit tracking. */
2028 STAMCOUNTER StatHCDirtyPageTrap;
2029 /** HC: The number of pages already dirty or readonly. */
2030 STAMCOUNTER StatHCDirtyPageSkipped;
2031
2032 /** GC: The number of pages marked not present for accessed bit emulation. */
2033 STAMCOUNTER StatGCAccessedPage;
2034 /** GC: The number of pages marked read-only for dirty bit tracking. */
2035 STAMCOUNTER StatGCDirtyPage;
2036 /** GC: The number of pages marked read-only for dirty bit tracking. */
2037 STAMCOUNTER StatGCDirtyPageBig;
2038 /** GC: The number of traps generated for dirty bit tracking. */
2039 STAMCOUNTER StatGCDirtyPageTrap;
2040 /** GC: The number of pages already dirty or readonly. */
2041 STAMCOUNTER StatGCDirtyPageSkipped;
2042 /** GC: The number of pages marked dirty because of write accesses. */
2043 STAMCOUNTER StatGCDirtiedPage;
2044 /** GC: The number of pages already marked dirty because of write accesses. */
2045 STAMCOUNTER StatGCPageAlreadyDirty;
2046 /** GC: The number of real pages faults during dirty bit tracking. */
2047 STAMCOUNTER StatGCDirtyTrackRealPF;
2048
2049 /** GC: Profiling of the PGMTrackDirtyBit() body */
2050 STAMPROFILE StatGCDirtyBitTracking;
2051 /** HC: Profiling of the PGMTrackDirtyBit() body */
2052 STAMPROFILE StatHCDirtyBitTracking;
2053
2054 /** GC: Profiling of the PGMGstModifyPage() body */
2055 STAMPROFILE StatGCGstModifyPage;
2056 /** HC: Profiling of the PGMGstModifyPage() body */
2057 STAMPROFILE StatHCGstModifyPage;
2058
2059 /** GC: The number of time we've marked a PD not present from SyncPage to virtualize the accessed bit. */
2060 STAMCOUNTER StatGCSyncPagePDNAs;
2061 /** GC: The number of time we've encountered an out-of-sync PD in SyncPage. */
2062 STAMCOUNTER StatGCSyncPagePDOutOfSync;
2063 /** HC: The number of time we've marked a PD not present from SyncPage to virtualize the accessed bit. */
2064 STAMCOUNTER StatHCSyncPagePDNAs;
2065 /** HC: The number of time we've encountered an out-of-sync PD in SyncPage. */
2066 STAMCOUNTER StatHCSyncPagePDOutOfSync;
2067
2068 STAMCOUNTER StatSynPT4kGC;
2069 STAMCOUNTER StatSynPT4kHC;
2070 STAMCOUNTER StatSynPT4MGC;
2071 STAMCOUNTER StatSynPT4MHC;
2072
2073 /** Profiling of the PGMFlushTLB() body. */
2074 STAMPROFILE StatFlushTLB;
2075 /** The number of times PGMFlushTLB was called with a new CR3, non-global. (switch) */
2076 STAMCOUNTER StatFlushTLBNewCR3;
2077 /** The number of times PGMFlushTLB was called with a new CR3, global. (switch) */
2078 STAMCOUNTER StatFlushTLBNewCR3Global;
2079 /** The number of times PGMFlushTLB was called with the same CR3, non-global. (flush) */
2080 STAMCOUNTER StatFlushTLBSameCR3;
2081 /** The number of times PGMFlushTLB was called with the same CR3, global. (flush) */
2082 STAMCOUNTER StatFlushTLBSameCR3Global;
2083
2084 STAMPROFILE StatGCSyncCR3; /**< GC: PGMSyncCR3() profiling. */
2085 STAMPROFILE StatGCSyncCR3Handlers; /**< GC: Profiling of the PGMSyncCR3() update handler section. */
2086 STAMPROFILE StatGCSyncCR3HandlerVirtualReset; /**< GC: Profiling of the virtual handler resets. */
2087 STAMPROFILE StatGCSyncCR3HandlerVirtualUpdate; /**< GC: Profiling of the virtual handler updates. */
2088 STAMCOUNTER StatGCSyncCR3Global; /**< GC: The number of global CR3 syncs. */
2089 STAMCOUNTER StatGCSyncCR3NotGlobal; /**< GC: The number of non-global CR3 syncs. */
2090 STAMCOUNTER StatGCSyncCR3DstFreed; /**< GC: The number of times we've had to free a shadow entry. */
2091 STAMCOUNTER StatGCSyncCR3DstFreedSrcNP; /**< GC: The number of times we've had to free a shadow entry for which the source entry was not present. */
2092 STAMCOUNTER StatGCSyncCR3DstNotPresent; /**< GC: The number of times we've encountered a not present shadow entry for a present guest entry. */
2093 STAMCOUNTER StatGCSyncCR3DstSkippedGlobalPD; /**< GC: The number of times a global page directory wasn't flushed. */
2094 STAMCOUNTER StatGCSyncCR3DstSkippedGlobalPT; /**< GC: The number of times a page table with only global entries wasn't flushed. */
2095 STAMCOUNTER StatGCSyncCR3DstCacheHit; /**< GC: The number of times we got some kind of cache hit on a page table. */
2096
2097 STAMPROFILE StatHCSyncCR3; /**< HC: PGMSyncCR3() profiling. */
2098 STAMPROFILE StatHCSyncCR3Handlers; /**< HC: Profiling of the PGMSyncCR3() update handler section. */
2099 STAMPROFILE StatHCSyncCR3HandlerVirtualReset; /**< HC: Profiling of the virtual handler resets. */
2100 STAMPROFILE StatHCSyncCR3HandlerVirtualUpdate; /**< HC: Profiling of the virtual handler updates. */
2101 STAMCOUNTER StatHCSyncCR3Global; /**< HC: The number of global CR3 syncs. */
2102 STAMCOUNTER StatHCSyncCR3NotGlobal; /**< HC: The number of non-global CR3 syncs. */
2103 STAMCOUNTER StatHCSyncCR3DstFreed; /**< HC: The number of times we've had to free a shadow entry. */
2104 STAMCOUNTER StatHCSyncCR3DstFreedSrcNP; /**< HC: The number of times we've had to free a shadow entry for which the source entry was not present. */
2105 STAMCOUNTER StatHCSyncCR3DstNotPresent; /**< HC: The number of times we've encountered a not present shadow entry for a present guest entry. */
2106 STAMCOUNTER StatHCSyncCR3DstSkippedGlobalPD; /**< HC: The number of times a global page directory wasn't flushed. */
2107 STAMCOUNTER StatHCSyncCR3DstSkippedGlobalPT; /**< HC: The number of times a page table with only global entries wasn't flushed. */
2108 STAMCOUNTER StatHCSyncCR3DstCacheHit; /**< HC: The number of times we got some kind of cache hit on a page table. */
2109
2110 /** GC: Profiling of pgmHandlerVirtualFindByPhysAddr. */
2111 STAMPROFILE StatVirtHandleSearchByPhysGC;
2112 /** HC: Profiling of pgmHandlerVirtualFindByPhysAddr. */
2113 STAMPROFILE StatVirtHandleSearchByPhysHC;
2114 /** HC: The number of times PGMR3HandlerPhysicalReset is called. */
2115 STAMCOUNTER StatHandlePhysicalReset;
2116
2117 STAMPROFILE StatCheckPageFault;
2118 STAMPROFILE StatLazySyncPT;
2119 STAMPROFILE StatMapping;
2120 STAMPROFILE StatOutOfSync;
2121 STAMPROFILE StatHandlers;
2122 STAMPROFILE StatEIPHandlers;
2123 STAMPROFILE StatHCPrefetch;
2124
2125# ifdef PGMPOOL_WITH_GCPHYS_TRACKING
2126 /** The number of first time shadowings. */
2127 STAMCOUNTER StatTrackVirgin;
2128 /** The number of times switching to cRef2, i.e. the page is being shadowed by two PTs. */
2129 STAMCOUNTER StatTrackAliased;
2130 /** The number of times we're tracking using cRef2. */
2131 STAMCOUNTER StatTrackAliasedMany;
2132 /** The number of times we're hitting pages which has overflowed cRef2. */
2133 STAMCOUNTER StatTrackAliasedLots;
2134 /** The number of times the extent list grows to long. */
2135 STAMCOUNTER StatTrackOverflows;
2136 /** Profiling of SyncPageWorkerTrackDeref (expensive). */
2137 STAMPROFILE StatTrackDeref;
2138# endif
2139
2140 /** Ring-3/0 page mapper TLB hits. */
2141 STAMCOUNTER StatPageHCMapTlbHits;
2142 /** Ring-3/0 page mapper TLB misses. */
2143 STAMCOUNTER StatPageHCMapTlbMisses;
2144 /** Ring-3/0 chunk mapper TLB hits. */
2145 STAMCOUNTER StatChunkR3MapTlbHits;
2146 /** Ring-3/0 chunk mapper TLB misses. */
2147 STAMCOUNTER StatChunkR3MapTlbMisses;
2148 /** Times a shared page has been replaced by a private one. */
2149 STAMCOUNTER StatPageReplaceShared;
2150 /** Times the zero page has been replaced by a private one. */
2151 STAMCOUNTER StatPageReplaceZero;
2152 /** The number of times we've executed GMMR3AllocateHandyPages. */
2153 STAMCOUNTER StatPageHandyAllocs;
2154
2155 /** Allocated mbs of guest ram */
2156 STAMCOUNTER StatDynRamTotal;
2157 /** Nr of pgmr3PhysGrowRange calls. */
2158 STAMCOUNTER StatDynRamGrow;
2159
2160 STAMCOUNTER StatGCTrap0ePD[X86_PG_ENTRIES];
2161 STAMCOUNTER StatGCSyncPtPD[X86_PG_ENTRIES];
2162 STAMCOUNTER StatGCSyncPagePD[X86_PG_ENTRIES];
2163#endif
2164} PGM, *PPGM;
2165
2166
2167/** @name PGM::fSyncFlags Flags
2168 * @{
2169 */
2170/** Updates the MM_RAM_FLAGS_VIRTUAL_HANDLER page bit. */
2171#define PGM_SYNC_UPDATE_PAGE_BIT_VIRTUAL BIT(0)
2172/** Always sync CR3. */
2173#define PGM_SYNC_ALWAYS BIT(1)
2174/** Check monitoring on next CR3 (re)load and invalidate page. */
2175#define PGM_SYNC_MONITOR_CR3 BIT(2)
2176/** Clear the page pool (a light weight flush). */
2177#define PGM_SYNC_CLEAR_PGM_POOL BIT(8)
2178/** @} */
2179
2180
2181__BEGIN_DECLS
2182
2183PGMGCDECL(int) pgmGCGuestPDWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, void *pvFault, RTGCPHYS GCPhysFault, void *pvUser);
2184PGMDECL(int) pgmGuestROMWriteHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, void *pvFault, RTGCPHYS GCPhysFault, void *pvUser);
2185PGMGCDECL(int) pgmCachePTWriteGC(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, RTGCPHYS GCPhysFault, void *pvUser);
2186int pgmR3SyncPTResolveConflict(PVM pVM, PPGMMAPPING pMapping, PVBOXPD pPDSrc, int iPDOld);
2187PPGMMAPPING pgmGetMapping(PVM pVM, RTGCPTR GCPtr);
2188void pgmR3MapRelocate(PVM pVM, PPGMMAPPING pMapping, int iPDOld, int iPDNew);
2189int pgmR3ChangeMode(PVM pVM, PGMMODE enmGuestMode);
2190int pgmLock(PVM pVM);
2191void pgmUnlock(PVM pVM);
2192
2193void pgmR3HandlerPhysicalUpdateAll(PVM pVM);
2194int pgmHandlerVirtualFindByPhysAddr(PVM pVM, RTGCPHYS GCPhys, PPGMVIRTHANDLER *ppVirt, unsigned *piPage);
2195DECLCALLBACK(int) pgmHandlerVirtualResetOne(PAVLROGCPTRNODECORE pNode, void *pvUser);
2196#ifdef VBOX_STRICT
2197void pgmHandlerVirtualDumpPhysPages(PVM pVM);
2198#else
2199# define pgmHandlerVirtualDumpPhysPages(a) do { } while (0)
2200#endif
2201DECLCALLBACK(void) pgmR3InfoHandlers(PVM pVM, PCDBGFINFOHLP pHlp, const char *pszArgs);
2202
2203
2204int pgmPhysPageLoadIntoTlb(PPGM pPGM, RTGCPHYS GCPhys);
2205#ifdef IN_RING3
2206int pgmR3PhysChunkMap(PVM pVM, uint32_t idChunk, PPPGMCHUNKR3MAP ppChunk);
2207#ifndef NEW_PHYS_CODE
2208int pgmr3PhysGrowRange(PVM pVM, RTGCPHYS GCPhys);
2209#endif
2210
2211int pgmR3PoolInit(PVM pVM);
2212void pgmR3PoolRelocate(PVM pVM);
2213void pgmR3PoolReset(PVM pVM);
2214
2215#endif /* IN_RING3 */
2216#ifdef IN_GC
2217void *pgmGCPoolMapPage(PVM pVM, PPGMPOOLPAGE pPage);
2218#endif
2219int pgmPoolAlloc(PVM pVM, RTGCPHYS GCPhys, PGMPOOLKIND enmKind, uint16_t iUser, uint16_t iUserTable, PPPGMPOOLPAGE ppPage);
2220PPGMPOOLPAGE pgmPoolGetPageByHCPhys(PVM pVM, RTHCPHYS HCPhys);
2221void pgmPoolFree(PVM pVM, RTHCPHYS HCPhys, uint16_t iUser, uint16_t iUserTable);
2222void pgmPoolFreeByPage(PPGMPOOL pPool, PPGMPOOLPAGE pPage, uint16_t iUser, uint16_t iUserTable);
2223int pgmPoolFlushPage(PPGMPOOL pPool, PPGMPOOLPAGE pPage);
2224void pgmPoolFlushAll(PVM pVM);
2225void pgmPoolClearAll(PVM pVM);
2226void pgmPoolTrackFlushGCPhysPT(PVM pVM, PPGMPAGE pPhysPage, uint16_t iShw, uint16_t cRefs);
2227void pgmPoolTrackFlushGCPhysPTs(PVM pVM, PPGMPAGE pPhysPage, uint16_t iPhysExt);
2228int pgmPoolTrackFlushGCPhysPTsSlow(PVM pVM, PPGMPAGE pPhysPage);
2229PPGMPOOLPHYSEXT pgmPoolTrackPhysExtAlloc(PVM pVM, uint16_t *piPhysExt);
2230void pgmPoolTrackPhysExtFree(PVM pVM, uint16_t iPhysExt);
2231void pgmPoolTrackPhysExtFreeList(PVM pVM, uint16_t iPhysExt);
2232uint16_t pgmPoolTrackPhysExtAddref(PVM pVM, uint16_t u16, uint16_t iShwPT);
2233void pgmPoolTrackPhysExtDerefGCPhys(PPGMPOOL pPool, PPGMPOOLPAGE pPoolPage, PPGMPAGE pPhysPage);
2234#ifdef PGMPOOL_WITH_MONITORING
2235# ifdef IN_RING3
2236void pgmPoolMonitorChainChanging(PPGMPOOL pPool, PPGMPOOLPAGE pPage, RTGCPHYS GCPhysFault, RTHCPTR pvAddress, PDISCPUSTATE pCpu);
2237# else
2238void pgmPoolMonitorChainChanging(PPGMPOOL pPool, PPGMPOOLPAGE pPage, RTGCPHYS GCPhysFault, RTGCPTR pvAddress, PDISCPUSTATE pCpu);
2239# endif
2240int pgmPoolMonitorChainFlush(PPGMPOOL pPool, PPGMPOOLPAGE pPage);
2241void pgmPoolMonitorModifiedInsert(PPGMPOOL pPool, PPGMPOOLPAGE pPage);
2242void pgmPoolMonitorModifiedClearAll(PVM pVM);
2243int pgmPoolMonitorMonitorCR3(PPGMPOOL pPool, uint16_t idxRoot, RTGCPHYS GCPhysCR3);
2244int pgmPoolMonitorUnmonitorCR3(PPGMPOOL pPool, uint16_t idxRoot);
2245#endif
2246
2247__END_DECLS
2248
2249
2250/**
2251 * Gets the PGMPAGE structure for a guest page.
2252 *
2253 * @returns Pointer to the page on success.
2254 * @returns NULL on a VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS condition.
2255 *
2256 * @param pPGM PGM handle.
2257 * @param GCPhys The GC physical address.
2258 */
2259DECLINLINE(PPGMPAGE) pgmPhysGetPage(PPGM pPGM, RTGCPHYS GCPhys)
2260{
2261 /*
2262 * Optimize for the first range.
2263 */
2264 PPGMRAMRANGE pRam = CTXSUFF(pPGM->pRamRanges);
2265 RTGCPHYS off = GCPhys - pRam->GCPhys;
2266 if (RT_UNLIKELY(off >= pRam->cb))
2267 {
2268 do
2269 {
2270 pRam = CTXSUFF(pRam->pNext);
2271 if (RT_UNLIKELY(!pRam))
2272 return NULL;
2273 off = GCPhys - pRam->GCPhys;
2274 } while (off >= pRam->cb);
2275 }
2276 return &pRam->aPages[off >> PAGE_SHIFT];
2277}
2278
2279
2280/**
2281 * Gets the PGMPAGE structure for a guest page.
2282 *
2283 * Old Phys code: Will make sure the page is present.
2284 *
2285 * @returns VBox status code.
2286 * @retval VINF_SUCCESS and a valid *ppPage on success.
2287 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if the address isn't valid.
2288 *
2289 * @param pPGM PGM handle.
2290 * @param GCPhys The GC physical address.
2291 * @param ppPage Where to store the page poitner on success.
2292 */
2293DECLINLINE(int) pgmPhysGetPageEx(PPGM pPGM, RTGCPHYS GCPhys, PPPGMPAGE ppPage)
2294{
2295 /*
2296 * Optimize for the first range.
2297 */
2298 PPGMRAMRANGE pRam = CTXSUFF(pPGM->pRamRanges);
2299 RTGCPHYS off = GCPhys - pRam->GCPhys;
2300 if (RT_UNLIKELY(off >= pRam->cb))
2301 {
2302 do
2303 {
2304 pRam = CTXSUFF(pRam->pNext);
2305 if (RT_UNLIKELY(!pRam))
2306 {
2307 *ppPage = NULL; /* avoid incorrect and very annoying GCC warnings */
2308 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
2309 }
2310 off = GCPhys - pRam->GCPhys;
2311 } while (off >= pRam->cb);
2312 }
2313 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
2314#ifndef NEW_PHYS_CODE
2315
2316 /*
2317 * Make sure it's present.
2318 */
2319 if (RT_UNLIKELY( !PGM_PAGE_GET_HCPHYS(*ppPage)
2320 && (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC)))
2321 {
2322#ifdef IN_RING3
2323 int rc = pgmr3PhysGrowRange(PGM2VM(pPGM), GCPhys);
2324#else
2325 int rc = CTXALLMID(VMM, CallHost)(PGM2VM(pPGM), VMMCALLHOST_PGM_RAM_GROW_RANGE, GCPhys);
2326#endif
2327 if (VBOX_FAILURE(rc))
2328 {
2329 *ppPage = NULL; /* avoid incorrect and very annoying GCC warnings */
2330 return rc;
2331 }
2332 Assert(rc == VINF_SUCCESS);
2333 }
2334#endif
2335 return VINF_SUCCESS;
2336}
2337
2338
2339
2340
2341/**
2342 * Gets the PGMPAGE structure for a guest page.
2343 *
2344 * Old Phys code: Will make sure the page is present.
2345 *
2346 * @returns VBox status code.
2347 * @retval VINF_SUCCESS and a valid *ppPage on success.
2348 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if the address isn't valid.
2349 *
2350 * @param pPGM PGM handle.
2351 * @param GCPhys The GC physical address.
2352 * @param ppPage Where to store the page poitner on success.
2353 * @param ppRamHint Where to read and store the ram list hint.
2354 * The caller initializes this to NULL before the call.
2355 */
2356DECLINLINE(int) pgmPhysGetPageWithHintEx(PPGM pPGM, RTGCPHYS GCPhys, PPPGMPAGE ppPage, PPGMRAMRANGE *ppRamHint)
2357{
2358 RTGCPHYS off;
2359 PPGMRAMRANGE pRam = *ppRamHint;
2360 if ( !pRam
2361 || RT_UNLIKELY((off = GCPhys - pRam->GCPhys) >= pRam->cb))
2362 {
2363 pRam = CTXSUFF(pPGM->pRamRanges);
2364 off = GCPhys - pRam->GCPhys;
2365 if (RT_UNLIKELY(off >= pRam->cb))
2366 {
2367 do
2368 {
2369 pRam = CTXSUFF(pRam->pNext);
2370 if (RT_UNLIKELY(!pRam))
2371 {
2372 *ppPage = NULL; /* Kill the incorrect and extremely annoying GCC warnings. */
2373 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
2374 }
2375 off = GCPhys - pRam->GCPhys;
2376 } while (off >= pRam->cb);
2377 }
2378 *ppRamHint = pRam;
2379 }
2380 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
2381#ifndef NEW_PHYS_CODE
2382
2383 /*
2384 * Make sure it's present.
2385 */
2386 if (RT_UNLIKELY( !PGM_PAGE_GET_HCPHYS(*ppPage)
2387 && (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC)))
2388 {
2389#ifdef IN_RING3
2390 int rc = pgmr3PhysGrowRange(PGM2VM(pPGM), GCPhys);
2391#else
2392 int rc = CTXALLMID(VMM, CallHost)(PGM2VM(pPGM), VMMCALLHOST_PGM_RAM_GROW_RANGE, GCPhys);
2393#endif
2394 if (VBOX_FAILURE(rc))
2395 {
2396 *ppPage = NULL; /* Shut up annoying smart ass. */
2397 return rc;
2398 }
2399 Assert(rc == VINF_SUCCESS);
2400 }
2401#endif
2402 return VINF_SUCCESS;
2403}
2404
2405
2406/**
2407 * Gets the PGMPAGE structure for a guest page together with the PGMRAMRANGE.
2408 *
2409 * @returns Pointer to the page on success.
2410 * @returns NULL on a VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS condition.
2411 *
2412 * @param pPGM PGM handle.
2413 * @param GCPhys The GC physical address.
2414 * @param ppRam Where to store the pointer to the PGMRAMRANGE.
2415 */
2416DECLINLINE(PPGMPAGE) pgmPhysGetPageAndRange(PPGM pPGM, RTGCPHYS GCPhys, PPGMRAMRANGE *ppRam)
2417{
2418 /*
2419 * Optimize for the first range.
2420 */
2421 PPGMRAMRANGE pRam = CTXSUFF(pPGM->pRamRanges);
2422 RTGCPHYS off = GCPhys - pRam->GCPhys;
2423 if (RT_UNLIKELY(off >= pRam->cb))
2424 {
2425 do
2426 {
2427 pRam = CTXSUFF(pRam->pNext);
2428 if (RT_UNLIKELY(!pRam))
2429 return NULL;
2430 off = GCPhys - pRam->GCPhys;
2431 } while (off >= pRam->cb);
2432 }
2433 *ppRam = pRam;
2434 return &pRam->aPages[off >> PAGE_SHIFT];
2435}
2436
2437
2438
2439
2440/**
2441 * Gets the PGMPAGE structure for a guest page together with the PGMRAMRANGE.
2442 *
2443 * @returns Pointer to the page on success.
2444 * @returns NULL on a VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS condition.
2445 *
2446 * @param pPGM PGM handle.
2447 * @param GCPhys The GC physical address.
2448 * @param ppPage Where to store the pointer to the PGMPAGE structure.
2449 * @param ppRam Where to store the pointer to the PGMRAMRANGE structure.
2450 */
2451DECLINLINE(int) pgmPhysGetPageAndRangeEx(PPGM pPGM, RTGCPHYS GCPhys, PPPGMPAGE ppPage, PPGMRAMRANGE *ppRam)
2452{
2453 /*
2454 * Optimize for the first range.
2455 */
2456 PPGMRAMRANGE pRam = CTXSUFF(pPGM->pRamRanges);
2457 RTGCPHYS off = GCPhys - pRam->GCPhys;
2458 if (RT_UNLIKELY(off >= pRam->cb))
2459 {
2460 do
2461 {
2462 pRam = CTXSUFF(pRam->pNext);
2463 if (RT_UNLIKELY(!pRam))
2464 {
2465 *ppRam = NULL; /* Shut up silly GCC warnings. */
2466 *ppPage = NULL; /* ditto */
2467 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
2468 }
2469 off = GCPhys - pRam->GCPhys;
2470 } while (off >= pRam->cb);
2471 }
2472 *ppRam = pRam;
2473 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
2474#ifndef NEW_PHYS_CODE
2475
2476 /*
2477 * Make sure it's present.
2478 */
2479 if (RT_UNLIKELY( !PGM_PAGE_GET_HCPHYS(*ppPage)
2480 && (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC)))
2481 {
2482#ifdef IN_RING3
2483 int rc = pgmr3PhysGrowRange(PGM2VM(pPGM), GCPhys);
2484#else
2485 int rc = CTXALLMID(VMM, CallHost)(PGM2VM(pPGM), VMMCALLHOST_PGM_RAM_GROW_RANGE, GCPhys);
2486#endif
2487 if (VBOX_FAILURE(rc))
2488 {
2489 *ppPage = NULL; /* Shut up silly GCC warnings. */
2490 *ppPage = NULL; /* ditto */
2491 return rc;
2492 }
2493 Assert(rc == VINF_SUCCESS);
2494
2495 }
2496#endif
2497 return VINF_SUCCESS;
2498}
2499
2500
2501/**
2502 * Convert GC Phys to HC Phys.
2503 *
2504 * @returns VBox status.
2505 * @param pPGM PGM handle.
2506 * @param GCPhys The GC physical address.
2507 * @param pHCPhys Where to store the corresponding HC physical address.
2508 *
2509 * @deprecated Doesn't deal with zero, shared or write monitored pages.
2510 * Avoid when writing new code!
2511 */
2512DECLINLINE(int) pgmRamGCPhys2HCPhys(PPGM pPGM, RTGCPHYS GCPhys, PRTHCPHYS pHCPhys)
2513{
2514 PPGMPAGE pPage;
2515 int rc = pgmPhysGetPageEx(pPGM, GCPhys, &pPage);
2516 if (VBOX_FAILURE(rc))
2517 return rc;
2518 *pHCPhys = PGM_PAGE_GET_HCPHYS(pPage) | (GCPhys & PAGE_OFFSET_MASK);
2519 return VINF_SUCCESS;
2520}
2521
2522
2523#ifndef IN_GC
2524/**
2525 * Queries the Physical TLB entry for a physical guest page,
2526 * attemting to load the TLB entry if necessary.
2527 *
2528 * @returns VBox status code.
2529 * @retval VINF_SUCCESS on success
2530 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
2531 * @param pPGM The PGM instance handle.
2532 * @param GCPhys The address of the guest page.
2533 * @param ppTlbe Where to store the pointer to the TLB entry.
2534 */
2535
2536DECLINLINE(int) pgmPhysPageQueryTlbe(PPGM pPGM, RTGCPHYS GCPhys, PPPGMPAGEMAPTLBE ppTlbe)
2537{
2538 int rc;
2539 PPGMPAGEMAPTLBE pTlbe = &pPGM->CTXSUFF(PhysTlb).aEntries[PGM_PAGEMAPTLB_IDX(GCPhys)];
2540 if (pTlbe->GCPhys == (GCPhys & X86_PTE_PAE_PG_MASK))
2541 {
2542 STAM_COUNTER_INC(&pPGM->CTXMID(StatPage,MapTlbHits));
2543 rc = VINF_SUCCESS;
2544 }
2545 else
2546 rc = pgmPhysPageLoadIntoTlb(pPGM, GCPhys);
2547 *ppTlbe = pTlbe;
2548 return rc;
2549}
2550#endif /* !IN_GC */
2551
2552
2553#ifndef NEW_PHYS_CODE
2554/**
2555 * Convert GC Phys to HC Virt.
2556 *
2557 * @returns VBox status.
2558 * @param pPGM PGM handle.
2559 * @param GCPhys The GC physical address.
2560 * @param pHCPtr Where to store the corresponding HC virtual address.
2561 *
2562 * @deprecated This will be eliminated by PGMPhysGCPhys2CCPtr.
2563 */
2564DECLINLINE(int) pgmRamGCPhys2HCPtr(PPGM pPGM, RTGCPHYS GCPhys, PRTHCPTR pHCPtr)
2565{
2566 PPGMRAMRANGE pRam;
2567 PPGMPAGE pPage;
2568 int rc = pgmPhysGetPageAndRangeEx(pPGM, GCPhys, &pPage, &pRam);
2569 if (VBOX_FAILURE(rc))
2570 {
2571 *pHCPtr = 0; /* Shut up silly GCC warnings. */
2572 return rc;
2573 }
2574 RTGCPHYS off = GCPhys - pRam->GCPhys;
2575
2576 if (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC)
2577 {
2578 unsigned iChunk = off >> PGM_DYNAMIC_CHUNK_SHIFT;
2579 *pHCPtr = (RTHCPTR)((RTHCUINTPTR)CTXSUFF(pRam->pavHCChunk)[iChunk] + (off & PGM_DYNAMIC_CHUNK_OFFSET_MASK));
2580 return VINF_SUCCESS;
2581 }
2582 if (pRam->pvHC)
2583 {
2584 *pHCPtr = (RTHCPTR)((RTHCUINTPTR)pRam->pvHC + off);
2585 return VINF_SUCCESS;
2586 }
2587 *pHCPtr = 0; /* Shut up silly GCC warnings. */
2588 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
2589}
2590#endif /* !NEW_PHYS_CODE */
2591
2592
2593/**
2594 * Convert GC Phys to HC Virt.
2595 *
2596 * @returns VBox status.
2597 * @param PVM VM handle.
2598 * @param pRam Ram range
2599 * @param GCPhys The GC physical address.
2600 * @param pHCPtr Where to store the corresponding HC virtual address.
2601 *
2602 * @deprecated This will be eliminated. Don't use it.
2603 */
2604DECLINLINE(int) pgmRamGCPhys2HCPtrWithRange(PVM pVM, PPGMRAMRANGE pRam, RTGCPHYS GCPhys, PRTHCPTR pHCPtr)
2605{
2606 RTGCPHYS off = GCPhys - pRam->GCPhys;
2607 Assert(off < pRam->cb);
2608
2609 if (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC)
2610 {
2611 unsigned idx = (off >> PGM_DYNAMIC_CHUNK_SHIFT);
2612 /* Physical chunk in dynamically allocated range not present? */
2613 if (RT_UNLIKELY(!CTXSUFF(pRam->pavHCChunk)[idx]))
2614 {
2615#ifdef IN_RING3
2616 int rc = pgmr3PhysGrowRange(pVM, GCPhys);
2617#else
2618 int rc = CTXALLMID(VMM, CallHost)(pVM, VMMCALLHOST_PGM_RAM_GROW_RANGE, GCPhys);
2619#endif
2620 if (rc != VINF_SUCCESS)
2621 {
2622 *pHCPtr = 0; /* GCC crap */
2623 return rc;
2624 }
2625 }
2626 *pHCPtr = (RTHCPTR)((RTHCUINTPTR)CTXSUFF(pRam->pavHCChunk)[idx] + (off & PGM_DYNAMIC_CHUNK_OFFSET_MASK));
2627 return VINF_SUCCESS;
2628 }
2629 if (pRam->pvHC)
2630 {
2631 *pHCPtr = (RTHCPTR)((RTHCUINTPTR)pRam->pvHC + off);
2632 return VINF_SUCCESS;
2633 }
2634 *pHCPtr = 0; /* GCC crap */
2635 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
2636}
2637
2638
2639/**
2640 * Convert GC Phys to HC Virt and HC Phys.
2641 *
2642 * @returns VBox status.
2643 * @param pPGM PGM handle.
2644 * @param GCPhys The GC physical address.
2645 * @param pHCPtr Where to store the corresponding HC virtual address.
2646 * @param pHCPhys Where to store the HC Physical address and its flags.
2647 *
2648 * @deprecated Will go away or be changed. Only user is MapCR3. MapCR3 will have to do ring-3
2649 * and ring-0 locking of the CR3 in a lazy fashion I'm fear... or perhaps not. we'll see.
2650 */
2651DECLINLINE(int) pgmRamGCPhys2HCPtrAndHCPhysWithFlags(PPGM pPGM, RTGCPHYS GCPhys, PRTHCPTR pHCPtr, PRTHCPHYS pHCPhys)
2652{
2653 PPGMRAMRANGE pRam;
2654 PPGMPAGE pPage;
2655 int rc = pgmPhysGetPageAndRangeEx(pPGM, GCPhys, &pPage, &pRam);
2656 if (VBOX_FAILURE(rc))
2657 {
2658 *pHCPtr = 0; /* Shut up crappy GCC warnings */
2659 *pHCPhys = 0; /* ditto */
2660 return rc;
2661 }
2662 RTGCPHYS off = GCPhys - pRam->GCPhys;
2663
2664 *pHCPhys = pPage->HCPhys; /** @todo PAGE FLAGS */
2665 if (pRam->fFlags & MM_RAM_FLAGS_DYNAMIC_ALLOC)
2666 {
2667 unsigned idx = (off >> PGM_DYNAMIC_CHUNK_SHIFT);
2668 *pHCPtr = (RTHCPTR)((RTHCUINTPTR)CTXSUFF(pRam->pavHCChunk)[idx] + (off & PGM_DYNAMIC_CHUNK_OFFSET_MASK));
2669 return VINF_SUCCESS;
2670 }
2671 if (pRam->pvHC)
2672 {
2673 *pHCPtr = (RTHCPTR)((RTHCUINTPTR)pRam->pvHC + off);
2674 return VINF_SUCCESS;
2675 }
2676 *pHCPtr = 0;
2677 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
2678}
2679
2680
2681/**
2682 * Clears flags associated with a RAM address.
2683 *
2684 * @returns VBox status code.
2685 * @param pPGM PGM handle.
2686 * @param GCPhys Guest context physical address.
2687 * @param fFlags fFlags to clear. (Bits 0-11.)
2688 */
2689DECLINLINE(int) pgmRamFlagsClearByGCPhys(PPGM pPGM, RTGCPHYS GCPhys, unsigned fFlags)
2690{
2691 PPGMPAGE pPage;
2692 int rc = pgmPhysGetPageEx(pPGM, GCPhys, &pPage);
2693 if (VBOX_FAILURE(rc))
2694 return rc;
2695
2696 fFlags &= ~X86_PTE_PAE_PG_MASK;
2697 pPage->HCPhys &= ~(RTHCPHYS)fFlags; /** @todo PAGE FLAGS */
2698 return VINF_SUCCESS;
2699}
2700
2701
2702/**
2703 * Clears flags associated with a RAM address.
2704 *
2705 * @returns VBox status code.
2706 * @param pPGM PGM handle.
2707 * @param GCPhys Guest context physical address.
2708 * @param fFlags fFlags to clear. (Bits 0-11.)
2709 * @param ppRamHint Where to read and store the ram list hint.
2710 * The caller initializes this to NULL before the call.
2711 */
2712DECLINLINE(int) pgmRamFlagsClearByGCPhysWithHint(PPGM pPGM, RTGCPHYS GCPhys, unsigned fFlags, PPGMRAMRANGE *ppRamHint)
2713{
2714 PPGMPAGE pPage;
2715 int rc = pgmPhysGetPageWithHintEx(pPGM, GCPhys, &pPage, ppRamHint);
2716 if (VBOX_FAILURE(rc))
2717 return rc;
2718
2719 fFlags &= ~X86_PTE_PAE_PG_MASK;
2720 pPage->HCPhys &= ~(RTHCPHYS)fFlags; /** @todo PAGE FLAGS */
2721 return VINF_SUCCESS;
2722}
2723
2724/**
2725 * Sets (bitwise OR) flags associated with a RAM address.
2726 *
2727 * @returns VBox status code.
2728 * @param pPGM PGM handle.
2729 * @param GCPhys Guest context physical address.
2730 * @param fFlags fFlags to set clear. (Bits 0-11.)
2731 */
2732DECLINLINE(int) pgmRamFlagsSetByGCPhys(PPGM pPGM, RTGCPHYS GCPhys, unsigned fFlags)
2733{
2734 PPGMPAGE pPage;
2735 int rc = pgmPhysGetPageEx(pPGM, GCPhys, &pPage);
2736 if (VBOX_FAILURE(rc))
2737 return rc;
2738
2739 fFlags &= ~X86_PTE_PAE_PG_MASK;
2740 pPage->HCPhys |= fFlags; /** @todo PAGE FLAGS */
2741 return VINF_SUCCESS;
2742}
2743
2744
2745/**
2746 * Sets (bitwise OR) flags associated with a RAM address.
2747 *
2748 * @returns VBox status code.
2749 * @param pPGM PGM handle.
2750 * @param GCPhys Guest context physical address.
2751 * @param fFlags fFlags to set clear. (Bits 0-11.)
2752 * @param ppRamHint Where to read and store the ram list hint.
2753 * The caller initializes this to NULL before the call.
2754 */
2755DECLINLINE(int) pgmRamFlagsSetByGCPhysWithHint(PPGM pPGM, RTGCPHYS GCPhys, unsigned fFlags, PPGMRAMRANGE *ppRamHint)
2756{
2757 PPGMPAGE pPage;
2758 int rc = pgmPhysGetPageWithHintEx(pPGM, GCPhys, &pPage, ppRamHint);
2759 if (VBOX_FAILURE(rc))
2760 return rc;
2761
2762 fFlags &= ~X86_PTE_PAE_PG_MASK;
2763 pPage->HCPhys |= fFlags; /** @todo PAGE FLAGS */
2764 return VINF_SUCCESS;
2765}
2766
2767
2768/**
2769 * Gets the page directory for the specified address.
2770 *
2771 * @returns Pointer to the page directory in question.
2772 * @returns NULL if the page directory is not present or on an invalid page.
2773 * @param pPGM Pointer to the PGM instance data.
2774 * @param GCPtr The address.
2775 */
2776DECLINLINE(PX86PDPAE) pgmGstGetPaePD(PPGM pPGM, RTGCUINTPTR GCPtr)
2777{
2778 const unsigned iPdPtr = GCPtr >> X86_PDPTR_SHIFT;
2779 if (CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].n.u1Present)
2780 {
2781 if ((CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u & X86_PDPE_PG_MASK) == pPGM->aGCPhysGstPaePDs[iPdPtr])
2782 return CTXSUFF(pPGM->apGstPaePDs)[iPdPtr];
2783
2784 /* cache is out-of-sync. */
2785 PX86PDPAE pPD;
2786 int rc = PGM_GCPHYS_2_PTR(PGM2VM(pPGM), CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u & X86_PDPE_PG_MASK, &pPD);
2787 if (VBOX_SUCCESS(rc))
2788 return pPD;
2789 AssertMsgFailed(("Impossible! rc=%d PDPE=%#llx\n", rc, CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u));
2790 /* returning NIL_RTGCPHYS is ok if we assume it's just an invalid page of some kind emualted as all 0s. */
2791 }
2792 return NULL;
2793}
2794
2795
2796/**
2797 * Gets the page directory entry for the specified address.
2798 *
2799 * @returns Pointer to the page directory entry in question.
2800 * @returns NULL if the page directory is not present or on an invalid page.
2801 * @param pPGM Pointer to the PGM instance data.
2802 * @param GCPtr The address.
2803 */
2804DECLINLINE(PX86PDEPAE) pgmGstGetPaePDEPtr(PPGM pPGM, RTGCUINTPTR GCPtr)
2805{
2806 const unsigned iPdPtr = GCPtr >> X86_PDPTR_SHIFT;
2807 if (CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].n.u1Present)
2808 {
2809 const unsigned iPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
2810 if ((CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u & X86_PDPE_PG_MASK) == pPGM->aGCPhysGstPaePDs[iPdPtr])
2811 return &CTXSUFF(pPGM->apGstPaePDs)[iPdPtr]->a[iPD];
2812
2813 /* The cache is out-of-sync. */
2814 PX86PDPAE pPD;
2815 int rc = PGM_GCPHYS_2_PTR(PGM2VM(pPGM), CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u & X86_PDPE_PG_MASK, &pPD);
2816 if (VBOX_SUCCESS(rc))
2817 return &pPD->a[iPD];
2818 AssertMsgFailed(("Impossible! rc=%Vrc PDPE=%RX64\n", rc, CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u));
2819 /* returning NIL_RTGCPHYS is ok if we assume it's just an invalid page or something which we'll emulate as all 0s. */
2820 }
2821 return NULL;
2822}
2823
2824
2825/**
2826 * Gets the page directory entry for the specified address.
2827 *
2828 * @returns The page directory entry in question.
2829 * @returns A non-present entry if the page directory is not present or on an invalid page.
2830 * @param pPGM Pointer to the PGM instance data.
2831 * @param GCPtr The address.
2832 */
2833DECLINLINE(uint64_t) pgmGstGetPaePDE(PPGM pPGM, RTGCUINTPTR GCPtr)
2834{
2835 const unsigned iPdPtr = GCPtr >> X86_PDPTR_SHIFT;
2836 if (CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].n.u1Present)
2837 {
2838 const unsigned iPD = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
2839 if ((CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u & X86_PDPE_PG_MASK) == pPGM->aGCPhysGstPaePDs[iPdPtr])
2840 return CTXSUFF(pPGM->apGstPaePDs)[iPdPtr]->a[iPD].u;
2841
2842 /* cache is out-of-sync. */
2843 PX86PDPAE pPD;
2844 int rc = PGM_GCPHYS_2_PTR(PGM2VM(pPGM), CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u & X86_PDPE_PG_MASK, &pPD);
2845 if (VBOX_SUCCESS(rc))
2846 return pPD->a[iPD].u;
2847 AssertMsgFailed(("Impossible! rc=%d PDPE=%#llx\n", rc, CTXSUFF(pPGM->pGstPaePDPTR)->a[iPdPtr].u));
2848 }
2849 return 0;
2850}
2851
2852
2853/**
2854 * Checks if any of the specified page flags are set for the given page.
2855 *
2856 * @returns true if any of the flags are set.
2857 * @returns false if all the flags are clear.
2858 * @param pPGM PGM handle.
2859 * @param GCPhys The GC physical address.
2860 * @param fFlags The flags to check for.
2861 */
2862DECLINLINE(bool) pgmRamTestFlags(PPGM pPGM, RTGCPHYS GCPhys, uint64_t fFlags)
2863{
2864 PPGMPAGE pPage = pgmPhysGetPage(pPGM, GCPhys);
2865 return pPage
2866 && (pPage->HCPhys & fFlags) != 0; /** @todo PAGE FLAGS */
2867}
2868
2869
2870/**
2871 * Gets the ram flags for a handler.
2872 *
2873 * @returns The ram flags.
2874 * @param pCur The physical handler in question.
2875 */
2876DECLINLINE(unsigned) pgmHandlerPhysicalCalcFlags(PPGMPHYSHANDLER pCur)
2877{
2878 switch (pCur->enmType)
2879 {
2880 case PGMPHYSHANDLERTYPE_PHYSICAL:
2881 return MM_RAM_FLAGS_PHYSICAL_HANDLER;
2882
2883 case PGMPHYSHANDLERTYPE_PHYSICAL_WRITE:
2884 return MM_RAM_FLAGS_PHYSICAL_HANDLER | MM_RAM_FLAGS_PHYSICAL_WRITE;
2885
2886 case PGMPHYSHANDLERTYPE_MMIO:
2887 case PGMPHYSHANDLERTYPE_PHYSICAL_ALL:
2888 return MM_RAM_FLAGS_PHYSICAL_HANDLER | MM_RAM_FLAGS_PHYSICAL_ALL;
2889
2890 default:
2891 AssertFatalMsgFailed(("Invalid type %d\n", pCur->enmType));
2892 }
2893}
2894
2895
2896/**
2897 * Clears one physical page of a virtual handler
2898 *
2899 * @param pPGM Pointer to the PGM instance.
2900 * @param pCur Virtual handler structure
2901 * @param iPage Physical page index
2902 */
2903DECLINLINE(void) pgmHandlerVirtualClearPage(PPGM pPGM, PPGMVIRTHANDLER pCur, unsigned iPage)
2904{
2905 const PPGMPHYS2VIRTHANDLER pPhys2Virt = &pCur->aPhysToVirt[iPage];
2906
2907 /*
2908 * Remove the node from the tree (it's supposed to be in the tree if we get here!).
2909 */
2910#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
2911 AssertReleaseMsg(pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_IN_TREE,
2912 ("pPhys2Virt=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
2913 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias));
2914#endif
2915 if (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_IS_HEAD)
2916 {
2917 /* We're the head of the alias chain. */
2918 PPGMPHYS2VIRTHANDLER pRemove = (PPGMPHYS2VIRTHANDLER)RTAvlroGCPhysRemove(&pPGM->CTXSUFF(pTrees)->PhysToVirtHandlers, pPhys2Virt->Core.Key); NOREF(pRemove);
2919#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
2920 AssertReleaseMsg(pRemove != NULL,
2921 ("pPhys2Virt=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
2922 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias));
2923 AssertReleaseMsg(pRemove == pPhys2Virt,
2924 ("wanted: pPhys2Virt=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n"
2925 " got: pRemove=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
2926 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias,
2927 pRemove, pRemove->Core.Key, pRemove->Core.KeyLast, pRemove->offVirtHandler, pRemove->offNextAlias));
2928#endif
2929 if (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK)
2930 {
2931 /* Insert the next list in the alias chain into the tree. */
2932 PPGMPHYS2VIRTHANDLER pNext = (PPGMPHYS2VIRTHANDLER)((intptr_t)pPhys2Virt + (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK));
2933#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
2934 AssertReleaseMsg(pNext->offNextAlias & PGMPHYS2VIRTHANDLER_IN_TREE,
2935 ("pNext=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32}\n",
2936 pNext, pNext->Core.Key, pNext->Core.KeyLast, pNext->offVirtHandler, pNext->offNextAlias));
2937#endif
2938 pNext->offNextAlias |= PGMPHYS2VIRTHANDLER_IS_HEAD;
2939 bool fRc = RTAvlroGCPhysInsert(&pPGM->CTXSUFF(pTrees)->PhysToVirtHandlers, &pNext->Core);
2940 AssertRelease(fRc);
2941 }
2942 }
2943 else
2944 {
2945 /* Locate the previous node in the alias chain. */
2946 PPGMPHYS2VIRTHANDLER pPrev = (PPGMPHYS2VIRTHANDLER)RTAvlroGCPhysGet(&pPGM->CTXSUFF(pTrees)->PhysToVirtHandlers, pPhys2Virt->Core.Key);
2947#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
2948 AssertReleaseMsg(pPrev != pPhys2Virt,
2949 ("pPhys2Virt=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32} pPrev=%p\n",
2950 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias, pPrev));
2951#endif
2952 for (;;)
2953 {
2954 PPGMPHYS2VIRTHANDLER pNext = (PPGMPHYS2VIRTHANDLER)((intptr_t)pPrev + (pPrev->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK));
2955 if (pNext == pPhys2Virt)
2956 {
2957 /* unlink. */
2958 LogFlow(("pgmHandlerVirtualClearPage: removed %p:{.offNextAlias=%#RX32} from alias chain. prev %p:{.offNextAlias=%#RX32} [%VGp-%VGp]\n",
2959 pPhys2Virt, pPhys2Virt->offNextAlias, pPrev, pPrev->offNextAlias, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast));
2960 if (!(pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK))
2961 pPrev->offNextAlias &= ~PGMPHYS2VIRTHANDLER_OFF_MASK;
2962 else
2963 {
2964 PPGMPHYS2VIRTHANDLER pNewNext = (PPGMPHYS2VIRTHANDLER)((intptr_t)pPhys2Virt + (pPhys2Virt->offNextAlias & PGMPHYS2VIRTHANDLER_OFF_MASK));
2965 pPrev->offNextAlias = ((intptr_t)pNewNext - (intptr_t)pPrev)
2966 | (pPrev->offNextAlias & ~PGMPHYS2VIRTHANDLER_OFF_MASK);
2967 }
2968 break;
2969 }
2970
2971 /* next */
2972 if (pNext == pPrev)
2973 {
2974#ifdef VBOX_STRICT_PGM_HANDLER_VIRTUAL
2975 AssertReleaseMsg(pNext != pPrev,
2976 ("pPhys2Virt=%p:{.Core.Key=%VGp, .Core.KeyLast=%VGp, .offVirtHandler=%#RX32, .offNextAlias=%#RX32} pPrev=%p\n",
2977 pPhys2Virt, pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offVirtHandler, pPhys2Virt->offNextAlias, pPrev));
2978#endif
2979 break;
2980 }
2981 pPrev = pNext;
2982 }
2983 }
2984 Log2(("PHYS2VIRT: Removing %VGp-%VGp %#RX32 %s\n",
2985 pPhys2Virt->Core.Key, pPhys2Virt->Core.KeyLast, pPhys2Virt->offNextAlias, HCSTRING(pCur->pszDesc)));
2986 pPhys2Virt->offNextAlias = 0;
2987 pPhys2Virt->Core.KeyLast = NIL_RTGCPHYS; /* require reinsert */
2988
2989 /*
2990 * Clear the ram flags for this page.
2991 */
2992 int rc = pgmRamFlagsClearByGCPhys(pPGM, pPhys2Virt->Core.Key,
2993 MM_RAM_FLAGS_VIRTUAL_HANDLER | MM_RAM_FLAGS_VIRTUAL_ALL | MM_RAM_FLAGS_VIRTUAL_WRITE);
2994 AssertRC(rc);
2995}
2996
2997
2998/**
2999 * Internal worker for finding a 'in-use' shadow page give by it's physical address.
3000 *
3001 * @returns Pointer to the shadow page structure.
3002 * @param pPool The pool.
3003 * @param HCPhys The HC physical address of the shadow page.
3004 */
3005DECLINLINE(PPGMPOOLPAGE) pgmPoolGetPage(PPGMPOOL pPool, RTHCPHYS HCPhys)
3006{
3007 /*
3008 * Look up the page.
3009 */
3010 PPGMPOOLPAGE pPage = (PPGMPOOLPAGE)RTAvloHCPhysGet(&pPool->HCPhysTree, HCPhys & X86_PTE_PAE_PG_MASK);
3011 AssertFatalMsg(pPage && pPage->enmKind != PGMPOOLKIND_FREE, ("HCPhys=%VHp pPage=%p type=%d\n", HCPhys, pPage, (pPage) ? pPage->enmKind : 0));
3012 return pPage;
3013}
3014
3015
3016/**
3017 * Internal worker for finding a 'in-use' shadow page give by it's physical address.
3018 *
3019 * @returns Pointer to the shadow page structure.
3020 * @param pPool The pool.
3021 * @param idx The pool page index.
3022 */
3023DECLINLINE(PPGMPOOLPAGE) pgmPoolGetPageByIdx(PPGMPOOL pPool, unsigned idx)
3024{
3025 AssertFatalMsg(idx >= PGMPOOL_IDX_FIRST && idx < pPool->cCurPages, ("idx=%d\n", idx));
3026 return &pPool->aPages[idx];
3027}
3028
3029
3030#ifdef PGMPOOL_WITH_GCPHYS_TRACKING
3031/**
3032 * Clear references to guest physical memory.
3033 *
3034 * @param pPool The pool.
3035 * @param pPoolPage The pool page.
3036 * @param pPhysPage The physical guest page tracking structure.
3037 */
3038DECLINLINE(void) pgmTrackDerefGCPhys(PPGMPOOL pPool, PPGMPOOLPAGE pPoolPage, PPGMPAGE pPhysPage)
3039{
3040 /*
3041 * Just deal with the simple case here.
3042 */
3043#ifdef LOG_ENABLED
3044 const RTHCPHYS HCPhysOrg = pPhysPage->HCPhys; /** @todo PAGE FLAGS */
3045#endif
3046 const unsigned cRefs = pPhysPage->HCPhys >> MM_RAM_FLAGS_CREFS_SHIFT; /** @todo PAGE FLAGS */
3047 if (cRefs == 1)
3048 {
3049 Assert(pPoolPage->idx == ((pPhysPage->HCPhys >> MM_RAM_FLAGS_IDX_SHIFT) & MM_RAM_FLAGS_IDX_MASK));
3050 pPhysPage->HCPhys = pPhysPage->HCPhys & MM_RAM_FLAGS_NO_REFS_MASK;
3051 }
3052 else
3053 pgmPoolTrackPhysExtDerefGCPhys(pPool, pPoolPage, pPhysPage);
3054 LogFlow(("pgmTrackDerefGCPhys: HCPhys=%RHp -> %RHp\n", HCPhysOrg, pPhysPage->HCPhys));
3055}
3056#endif
3057
3058
3059#ifdef PGMPOOL_WITH_CACHE
3060/**
3061 * Moves the page to the head of the age list.
3062 *
3063 * This is done when the cached page is used in one way or another.
3064 *
3065 * @param pPool The pool.
3066 * @param pPage The cached page.
3067 * @todo inline in PGMInternal.h!
3068 */
3069DECLINLINE(void) pgmPoolCacheUsed(PPGMPOOL pPool, PPGMPOOLPAGE pPage)
3070{
3071 /*
3072 * Move to the head of the age list.
3073 */
3074 if (pPage->iAgePrev != NIL_PGMPOOL_IDX)
3075 {
3076 /* unlink */
3077 pPool->aPages[pPage->iAgePrev].iAgeNext = pPage->iAgeNext;
3078 if (pPage->iAgeNext != NIL_PGMPOOL_IDX)
3079 pPool->aPages[pPage->iAgeNext].iAgePrev = pPage->iAgePrev;
3080 else
3081 pPool->iAgeTail = pPage->iAgePrev;
3082
3083 /* insert at head */
3084 pPage->iAgePrev = NIL_PGMPOOL_IDX;
3085 pPage->iAgeNext = pPool->iAgeHead;
3086 Assert(pPage->iAgeNext != NIL_PGMPOOL_IDX); /* we would've already been head then */
3087 pPool->iAgeHead = pPage->idx;
3088 pPool->aPages[pPage->iAgeNext].iAgePrev = pPage->idx;
3089 }
3090}
3091#endif /* PGMPOOL_WITH_CACHE */
3092
3093/**
3094 * Tells if mappings are to be put into the shadow page table or not
3095 *
3096 * @returns boolean result
3097 * @param pVM VM handle.
3098 */
3099
3100DECLINLINE(bool) pgmMapAreMappingsEnabled(PPGM pPGM)
3101{
3102 return !pPGM->fDisableMappings;
3103}
3104
3105
3106#if 0
3107/** Soon to be obsolete conversion functions */
3108
3109/**
3110 * Converts a GC physical address to a HC pointer.
3111 *
3112 * @returns VINF_SUCCESS on success.
3113 * @returns VERR_PGM_PHYS_PAGE_RESERVED it it's a valid GC physical
3114 * page but has no physical backing.
3115 * @returns VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid
3116 * GC physical address.
3117 * @param pVM The VM handle.
3118 * @param GCPhys The GC physical address to convert.
3119 * @param cbRange Physical range
3120 * @param pHCPtr Where to store the HC pointer on success.
3121 */
3122PGMDECL(int) PGMPhysGCPhys2HCPtr(PVM pVM, RTGCPHYS GCPhys, RTUINT cbRange, PRTHCPTR pHCPtr);
3123
3124/**
3125 * Converts a guest pointer to a HC pointer.
3126 *
3127 * This uses the current CR3/CR0/CR4 of the guest.
3128 *
3129 * @returns VBox status code.
3130 * @param pVM The VM Handle
3131 * @param GCPtr The guest pointer to convert.
3132 * @param pHCPtr Where to store the HC virtual address.
3133 */
3134PGMDECL(int) PGMPhysGCPtr2HCPtr(PVM pVM, RTGCPTR GCPtr, PRTHCPTR pHCPtr);
3135
3136/**
3137 * Converts a guest virtual address to a HC pointer by specfied CR3 and flags.
3138 *
3139 * @returns VBox status code.
3140 * @param pVM The VM Handle
3141 * @param GCPtr The guest pointer to convert.
3142 * @param cr3 The guest CR3.
3143 * @param fFlags Flags used for interpreting the PD correctly: X86_CR4_PSE and X86_CR4_PAE
3144 * @param pHCPtr Where to store the HC pointer.
3145 *
3146 * @remark This function is used by the REM at a time where PGM could
3147 * potentially not be in sync. It could also be used by a
3148 * future DBGF API to cpu state independent conversions.
3149 */
3150PGMDECL(int) PGMPhysGCPtr2HCPtrByGstCR3(PVM pVM, RTGCPTR GCPtr, uint32_t cr3, unsigned fFlags, PRTHCPTR pHCPtr);
3151#endif
3152
3153/** @} */
3154
3155#endif
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