VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/PGMAllShw.h@ 7554

Last change on this file since 7554 was 5999, checked in by vboxsync, 17 years ago

The Giant CDDL Dual-License Header Change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 14.0 KB
Line 
1/* $Id: PGMAllShw.h 5999 2007-12-07 15:05:06Z vboxsync $ */
2/** @file
3 * VBox - Page Manager, Shadow Paging Template - All context code.
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 (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Defined Constants And Macros *
20*******************************************************************************/
21#undef SHWPT
22#undef PSHWPT
23#undef SHWPTE
24#undef PSHWPTE
25#undef SHWPD
26#undef PSHWPD
27#undef SHWPDE
28#undef PSHWPDE
29#undef SHW_PDE_PG_MASK
30#undef SHW_PD_SHIFT
31#undef SHW_PD_MASK
32#undef SHW_PTE_PG_MASK
33#undef SHW_PT_SHIFT
34#undef SHW_PT_MASK
35#undef SHW_TOTAL_PD_ENTRIES
36#undef SHW_PDPTR_SHIFT
37#undef SHW_PDPTR_MASK
38#undef SHW_POOL_ROOT_IDX
39
40#if PGM_SHW_TYPE == PGM_TYPE_32BIT
41# define SHWPT X86PT
42# define PSHWPT PX86PT
43# define SHWPTE X86PTE
44# define PSHWPTE PX86PTE
45# define SHWPD X86PD
46# define PSHWPD PX86PD
47# define SHWPDE X86PDE
48# define PSHWPDE PX86PDE
49# define SHW_PDE_PG_MASK X86_PDE_PG_MASK
50# define SHW_PD_SHIFT X86_PD_SHIFT
51# define SHW_PD_MASK X86_PD_MASK
52# define SHW_TOTAL_PD_ENTRIES X86_PG_ENTRIES
53# define SHW_PTE_PG_MASK X86_PTE_PG_MASK
54# define SHW_PT_SHIFT X86_PT_SHIFT
55# define SHW_PT_MASK X86_PT_MASK
56# define SHW_POOL_ROOT_IDX PGMPOOL_IDX_PD
57#else
58# define SHWPT X86PTPAE
59# define PSHWPT PX86PTPAE
60# define SHWPTE X86PTEPAE
61# define PSHWPTE PX86PTEPAE
62# define SHWPD X86PDPAE
63# define PSHWPD PX86PDPAE
64# define SHWPDE X86PDEPAE
65# define PSHWPDE PX86PDEPAE
66# define SHW_PDE_PG_MASK X86_PDE_PAE_PG_MASK
67# define SHW_PD_SHIFT X86_PD_PAE_SHIFT
68# define SHW_PD_MASK X86_PD_PAE_MASK
69# define SHW_PTE_PG_MASK X86_PTE_PAE_PG_MASK
70# define SHW_PT_SHIFT X86_PT_PAE_SHIFT
71# define SHW_PT_MASK X86_PT_PAE_MASK
72#if PGM_SHW_TYPE == PGM_TYPE_AMD64
73# define SHW_PDPTR_SHIFT X86_PDPTR_SHIFT
74# define SHW_PDPTR_MASK X86_PDPTR_MASK
75# define SHW_POOL_ROOT_IDX PGMPOOL_IDX_PML4
76#else /* 32 bits PAE mode */
77# define SHW_PDPTR_SHIFT X86_PDPTR_SHIFT
78# define SHW_PDPTR_MASK X86_PDPTR_MASK_32
79# define SHW_TOTAL_PD_ENTRIES (X86_PG_PAE_ENTRIES*4)
80# define SHW_POOL_ROOT_IDX PGMPOOL_IDX_PAE_PD
81#endif
82#endif
83
84
85
86/*******************************************************************************
87* Internal Functions *
88*******************************************************************************/
89__BEGIN_DECLS
90PGM_SHW_DECL(int, GetPage)(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys);
91PGM_SHW_DECL(int, ModifyPage)(PVM pVM, RTGCUINTPTR GCPtr, size_t cbPages, uint64_t fFlags, uint64_t fMask);
92PGM_SHW_DECL(int, GetPDEByIndex)(PVM pVM, uint32_t iPD, PX86PDEPAE pPde);
93PGM_SHW_DECL(int, SetPDEByIndex)(PVM pVM, uint32_t iPD, X86PDEPAE Pde);
94PGM_SHW_DECL(int, ModifyPDEByIndex)(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask);
95__END_DECLS
96
97
98
99/**
100 * Gets effective page information (from the VMM page directory).
101 *
102 * @returns VBox status.
103 * @param pVM VM Handle.
104 * @param GCPtr Guest Context virtual address of the page.
105 * @param pfFlags Where to store the flags. These are X86_PTE_*.
106 * @param pHCPhys Where to store the HC physical address of the page.
107 * This is page aligned.
108 * @remark You should use PGMMapGetPage() for pages in a mapping.
109 */
110PGM_SHW_DECL(int, GetPage)(PVM pVM, RTGCUINTPTR GCPtr, uint64_t *pfFlags, PRTHCPHYS pHCPhys)
111{
112 /*
113 * Get the PDE.
114 */
115#if PGM_SHW_TYPE == PGM_TYPE_AMD64
116 /*
117 * For the first 4G we have preallocated page directories.
118 * Since the two upper levels contains only fixed flags, we skip those when possible.
119 */
120 X86PDEPAE Pde;
121#if GC_ARCH_BITS == 64
122 if (GCPtr < _4G)
123#endif
124 {
125 const unsigned iPdptr = (GCPtr >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK;
126 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
127 Pde = CTXMID(pVM->pgm.s.ap,PaePDs)[iPdptr]->a[iPd];
128 }
129#if GC_ARCH_BITS == 64
130 else
131 {
132 /* PML4 */
133 const unsigned iPml4 = (GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
134 X86PML4E Pml4e = CTXMID(pVM->pgm.s.p,PaePML4)->a[iPml4];
135 if (!Pml4e.n.u1Present)
136 return VERR_PAGE_TABLE_NOT_PRESENT;
137
138 /* PDPTR */
139 PX86PDPTR pPdPtr;
140 int rc = PGM_HCPHYS_2_PTR(pVM, Pml4e.u & X86_PML4E_PG_MASK, &pPdPtr);
141 if (VBOX_FAILURE(rc))
142 return rc;
143 const unsigned iPdptr = (GCPtr >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK;
144 X86PDPE Pdpe = pPdPtr->a[iPdptr];
145 if (!Pdpe.n.u1Present)
146 return VERR_PAGE_TABLE_NOT_PRESENT;
147
148 /* PD */
149 PX86PDPAE pPd;
150 rc = PGM_HCPHYS_2_PTR(pVM, Pdpe.u & X86_PDPE_PG_MASK, &pPd);
151 if (VBOX_FAILURE(rc))
152 return rc;
153 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
154 Pdpe = pPdPtr->a[iPd];
155 }
156#endif /* GC_ARCH_BITS == 64 */
157
158#elif PGM_SHW_TYPE == PGM_TYPE_PAE
159 const unsigned iPdptr = (GCPtr >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK;
160 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
161 X86PDEPAE Pde = CTXMID(pVM->pgm.s.ap,PaePDs)[iPdptr]->a[iPd];
162
163#else /* PGM_TYPE_32BIT */
164 const unsigned iPd = (GCPtr >> X86_PD_SHIFT) & X86_PD_MASK;
165 X86PDE Pde = CTXMID(pVM->pgm.s.p,32BitPD)->a[iPd];
166#endif
167 if (!Pde.n.u1Present)
168 return VERR_PAGE_TABLE_NOT_PRESENT;
169
170 /*
171 * Get PT entry.
172 */
173 PSHWPT pPT;
174 if (!(Pde.u & PGM_PDFLAGS_MAPPING))
175 {
176 int rc = PGM_HCPHYS_2_PTR(pVM, Pde.u & SHW_PDE_PG_MASK, &pPT);
177 if (VBOX_FAILURE(rc))
178 return rc;
179 }
180 else /* mapping: */
181 {
182 Assert(pgmMapAreMappingsEnabled(&pVM->pgm.s));
183
184 PPGMMAPPING pMap = pgmGetMapping(pVM, (RTGCPTR)GCPtr);
185 AssertMsgReturn(pMap, ("GCPtr=%VGv\n", GCPtr), VERR_INTERNAL_ERROR);
186#if PGM_SHW_TYPE == PGM_TYPE_32BIT
187 pPT = pMap->aPTs[(GCPtr - pMap->GCPtr) >> X86_PD_SHIFT].CTXALLSUFF(pPT);
188#else /* PAE and AMD64: */
189 pPT = pMap->aPTs[(GCPtr - pMap->GCPtr) >> X86_PD_SHIFT].CTXALLSUFF(paPaePTs);
190#endif
191 }
192 const unsigned iPt = (GCPtr >> SHW_PT_SHIFT) & SHW_PT_MASK;
193 SHWPTE Pte = pPT->a[iPt];
194 if (!Pte.n.u1Present)
195 return VERR_PAGE_NOT_PRESENT;
196
197 /*
198 * Store the results.
199 * RW and US flags depend on the entire page transation hierarchy - except for
200 * legacy PAE which has a simplified PDPE.
201 */
202 if (pfFlags)
203 *pfFlags = (Pte.u & ~SHW_PTE_PG_MASK)
204 & ((Pde.u & (X86_PTE_RW | X86_PTE_US)) | ~(uint64_t)(X86_PTE_RW | X86_PTE_US));
205 if (pHCPhys)
206 *pHCPhys = Pte.u & SHW_PTE_PG_MASK;
207
208 return VINF_SUCCESS;
209}
210
211
212/**
213 * Modify page flags for a range of pages in the shadow context.
214 *
215 * The existing flags are ANDed with the fMask and ORed with the fFlags.
216 *
217 * @returns VBox status code.
218 * @param pVM VM handle.
219 * @param GCPtr Virtual address of the first page in the range. Page aligned!
220 * @param cb Size (in bytes) of the range to apply the modification to. Page aligned!
221 * @param fFlags The OR mask - page flags X86_PTE_*, excluding the page mask of course.
222 * @param fMask The AND mask - page flags X86_PTE_*.
223 * Be extremely CAREFUL with ~'ing values because they can be 32-bit!
224 * @remark You must use PGMMapModifyPage() for pages in a mapping.
225 */
226PGM_SHW_DECL(int, ModifyPage)(PVM pVM, RTGCUINTPTR GCPtr, size_t cb, uint64_t fFlags, uint64_t fMask)
227{
228 /*
229 * Walk page tables and pages till we're done.
230 */
231 for (;;)
232 {
233 /*
234 * Get the PDE.
235 */
236#if PGM_SHW_TYPE == PGM_TYPE_AMD64
237 /*
238 * For the first 4G we have preallocated page directories.
239 * Since the two upper levels contains only fixed flags, we skip those when possible.
240 */
241 X86PDEPAE Pde;
242#if GC_ARCH_BITS == 64
243 if (GCPtr < _4G)
244#endif
245 {
246 const unsigned iPdptr = (GCPtr >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK;
247 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
248 Pde = CTXMID(pVM->pgm.s.ap,PaePDs)[iPdptr]->a[iPd];
249 }
250#if GC_ARCH_BITS == 64
251 else
252 {
253 /* PML4 */
254 const unsigned iPml4 = (GCPtr >> X86_PML4_SHIFT) & X86_PML4_MASK;
255 X86PML4E Pml4e = CTXMID(pVM->pgm.s.p,PaePML4)->a[iPml4];
256 if (!Pml4e.n.u1Present)
257 return VERR_PAGE_TABLE_NOT_PRESENT;
258
259 /* PDPTR */
260 PX86PDPTR pPdPtr;
261 int rc = PGM_HCPHYS_2_PTR(pVM, Pml4e.u & X86_PML4E_PG_MASK, &pPdPtr);
262 if (VBOX_FAILURE(rc))
263 return rc;
264 const unsigned iPdptr = (GCPtr >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK;
265 X86PDPE Pdpe = pPdPtr->a[iPdptr];
266 if (!Pdpe.n.u1Present)
267 return VERR_PAGE_TABLE_NOT_PRESENT;
268
269 /* PD */
270 PX86PDPAE pPd;
271 rc = PGM_HCPHYS_2_PTR(pVM, Pdpe.u & X86_PDPE_PG_MASK, &pPd);
272 if (VBOX_FAILURE(rc))
273 return rc;
274 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
275 Pdpe = pPdPtr->a[iPd];
276 }
277#endif /* GC_ARCH_BITS == 64 */
278
279#elif PGM_SHW_TYPE == PGM_TYPE_PAE
280 const unsigned iPdptr = (GCPtr >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK;
281 const unsigned iPd = (GCPtr >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
282 X86PDEPAE Pde = CTXMID(pVM->pgm.s.ap,PaePDs)[iPdptr]->a[iPd];
283
284#else /* PGM_TYPE_32BIT */
285 const unsigned iPd = (GCPtr >> X86_PD_SHIFT) & X86_PD_MASK;
286 X86PDE Pde = CTXMID(pVM->pgm.s.p,32BitPD)->a[iPd];
287#endif
288 if (!Pde.n.u1Present)
289 return VERR_PAGE_TABLE_NOT_PRESENT;
290
291
292 /*
293 * Map the page table.
294 */
295 PSHWPT pPT;
296 int rc = PGM_HCPHYS_2_PTR(pVM, Pde.u & SHW_PDE_PG_MASK, &pPT);
297 if (VBOX_FAILURE(rc))
298 return rc;
299
300 unsigned iPTE = (GCPtr >> SHW_PT_SHIFT) & SHW_PT_MASK;
301 while (iPTE < ELEMENTS(pPT->a))
302 {
303 if (pPT->a[iPTE].n.u1Present)
304 {
305 pPT->a[iPTE].u = (pPT->a[iPTE].u & (fMask | SHW_PTE_PG_MASK)) | (fFlags & ~SHW_PTE_PG_MASK);
306 Assert(pPT->a[iPTE].n.u1Present);
307 PGM_INVL_PG(GCPtr);
308 }
309
310 /* next page */
311 cb -= PAGE_SIZE;
312 if (!cb)
313 return VINF_SUCCESS;
314 GCPtr += PAGE_SIZE;
315 iPTE++;
316 }
317 }
318}
319
320/**
321 * Retrieve shadow PDE
322 *
323 * @returns VBox status code.
324 * @param pVM The virtual machine.
325 * @param iPD Shadow PDE index.
326 * @param pPde Where to store the shadow PDE entry.
327 */
328PGM_SHW_DECL(int, GetPDEByIndex)(PVM pVM, unsigned iPD, PX86PDEPAE pPde)
329{
330#if PGM_SHW_TYPE == PGM_TYPE_32BIT || PGM_SHW_TYPE == PGM_TYPE_PAE
331 /*
332 * Get page directory addresses.
333 */
334 Assert(iPD < SHW_TOTAL_PD_ENTRIES);
335# if PGM_SHW_TYPE == PGM_TYPE_32BIT
336 PX86PDE pPdeSrc = &CTXMID(pVM->pgm.s.p,32BitPD)->a[iPD];
337# else
338 PX86PDEPAE pPdeSrc = &CTXMID(pVM->pgm.s.ap,PaePDs)[0]->a[iPD]; /* We treat this as a PD with 2048 entries. */
339# endif
340
341 pPde->u = (X86PGPAEUINT)pPdeSrc->u;
342 return VINF_SUCCESS;
343
344#else
345 AssertFailed();
346 return VERR_NOT_IMPLEMENTED;
347#endif
348}
349
350/**
351 * Set shadow PDE
352 *
353 * @returns VBox status code.
354 * @param pVM The virtual machine.
355 * @param iPD Shadow PDE index.
356 * @param Pde Shadow PDE.
357 */
358PGM_SHW_DECL(int, SetPDEByIndex)(PVM pVM, unsigned iPD, X86PDEPAE Pde)
359{
360#if PGM_SHW_TYPE == PGM_TYPE_32BIT || PGM_SHW_TYPE == PGM_TYPE_PAE
361 /*
362 * Get page directory addresses and update the specified entry.
363 */
364 Assert(iPD < SHW_TOTAL_PD_ENTRIES);
365# if PGM_SHW_TYPE == PGM_TYPE_32BIT
366 Assert(Pde.au32[1] == 0); /* First uint32_t is backwards compatible. */
367 Assert(Pde.n.u1Size == 0);
368 PX86PDE pPdeDst = &CTXMID(pVM->pgm.s.p,32BitPD)->a[iPD];
369 pPdeDst->u = Pde.au32[0];
370# else
371 PX86PDEPAE pPdeDst = &CTXMID(pVM->pgm.s.ap,PaePDs)[0]->a[iPD]; /* We treat this as a PD with 2048 entries. */
372 pPdeDst->u = Pde.u;
373# endif
374 Assert(pPdeDst->n.u1Present);
375
376 return VINF_SUCCESS;
377#else
378 AssertFailed();
379 return VERR_NOT_IMPLEMENTED;
380#endif
381}
382
383/**
384 * Modify shadow PDE
385 *
386 * @returns VBox status code.
387 * @param pVM The virtual machine.
388 * @param iPD Shadow PDE index.
389 * @param fFlags The OR mask - page flags X86_PDE_*, excluding the page mask of course.
390 * @param fMask The AND mask - page flags X86_PDE_*.
391 * Be extremely CAREFUL with ~'ing values because they can be 32-bit!
392 */
393PGM_SHW_DECL(int, ModifyPDEByIndex)(PVM pVM, uint32_t iPD, uint64_t fFlags, uint64_t fMask)
394{
395#if PGM_SHW_TYPE == PGM_TYPE_32BIT || PGM_SHW_TYPE == PGM_TYPE_PAE
396 /*
397 * Get page directory addresses and update the specified entry.
398 */
399 Assert(iPD < SHW_TOTAL_PD_ENTRIES);
400# if PGM_SHW_TYPE == PGM_TYPE_32BIT
401 PX86PDE pPdeDst = &CTXMID(pVM->pgm.s.p,32BitPD)->a[iPD];
402
403 pPdeDst->u = ((pPdeDst->u & ((X86PGUINT)fMask | SHW_PDE_PG_MASK)) | ((X86PGUINT)fFlags & ~SHW_PDE_PG_MASK));
404 Assert(!pPdeDst->n.u1Size);
405# else
406 PX86PDEPAE pPdeDst = &CTXMID(pVM->pgm.s.ap,PaePDs)[0]->a[iPD]; /* We treat this as a PD with 2048 entries. */
407
408 pPdeDst->u = (pPdeDst->u & (fMask | SHW_PDE_PG_MASK)) | (fFlags & ~SHW_PDE_PG_MASK);
409# endif
410 Assert(pPdeDst->n.u1Present);
411
412 return VINF_SUCCESS;
413#else
414 AssertFailed();
415 return VERR_NOT_IMPLEMENTED;
416#endif
417}
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