VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/PGMAllPhys.cpp@ 81786

Last change on this file since 81786 was 81624, checked in by vboxsync, 5 years ago

PDM,PGM: Added handled based MMIO2 interface. Made some adjustments to the PCI I/O region registrations. (Preps for VMMDev.) bugref:9218

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 165.1 KB
Line 
1/* $Id: PGMAllPhys.cpp 81624 2019-11-01 20:46:49Z vboxsync $ */
2/** @file
3 * PGM - Page Manager and Monitor, Physical Memory Addressing.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_PGM_PHYS
23#include <VBox/vmm/pgm.h>
24#include <VBox/vmm/trpm.h>
25#include <VBox/vmm/vmm.h>
26#include <VBox/vmm/iom.h>
27#include <VBox/vmm/em.h>
28#include <VBox/vmm/nem.h>
29#include "PGMInternal.h"
30#include <VBox/vmm/vmcc.h>
31#include "PGMInline.h"
32#include <VBox/param.h>
33#include <VBox/err.h>
34#include <iprt/assert.h>
35#include <iprt/string.h>
36#include <iprt/asm-amd64-x86.h>
37#include <VBox/log.h>
38#ifdef IN_RING3
39# include <iprt/thread.h>
40#endif
41
42
43/*********************************************************************************************************************************
44* Defined Constants And Macros *
45*********************************************************************************************************************************/
46/** Enable the physical TLB. */
47#define PGM_WITH_PHYS_TLB
48
49/** @def PGM_HANDLER_PHYS_IS_VALID_STATUS
50 * Checks if valid physical access handler return code (normal handler, not PF).
51 *
52 * Checks if the given strict status code is one of the expected ones for a
53 * physical access handler in the current context.
54 *
55 * @returns true or false.
56 * @param a_rcStrict The status code.
57 * @param a_fWrite Whether it is a write or read being serviced.
58 *
59 * @remarks We wish to keep the list of statuses here as short as possible.
60 * When changing, please make sure to update the PGMPhysRead,
61 * PGMPhysWrite, PGMPhysReadGCPtr and PGMPhysWriteGCPtr docs too.
62 */
63#ifdef IN_RING3
64# define PGM_HANDLER_PHYS_IS_VALID_STATUS(a_rcStrict, a_fWrite) \
65 ( (a_rcStrict) == VINF_SUCCESS \
66 || (a_rcStrict) == VINF_PGM_HANDLER_DO_DEFAULT)
67#elif defined(IN_RING0)
68#define PGM_HANDLER_PHYS_IS_VALID_STATUS(a_rcStrict, a_fWrite) \
69 ( (a_rcStrict) == VINF_SUCCESS \
70 || (a_rcStrict) == VINF_PGM_HANDLER_DO_DEFAULT \
71 \
72 || (a_rcStrict) == ((a_fWrite) ? VINF_IOM_R3_MMIO_WRITE : VINF_IOM_R3_MMIO_READ) \
73 || (a_rcStrict) == VINF_IOM_R3_MMIO_READ_WRITE \
74 || ((a_rcStrict) == VINF_IOM_R3_MMIO_COMMIT_WRITE && (a_fWrite)) \
75 \
76 || (a_rcStrict) == VINF_EM_RAW_EMULATE_INSTR \
77 || (a_rcStrict) == VINF_EM_DBG_STOP \
78 || (a_rcStrict) == VINF_EM_DBG_EVENT \
79 || (a_rcStrict) == VINF_EM_DBG_BREAKPOINT \
80 || (a_rcStrict) == VINF_EM_OFF \
81 || (a_rcStrict) == VINF_EM_SUSPEND \
82 || (a_rcStrict) == VINF_EM_RESET \
83 )
84#else
85# error "Context?"
86#endif
87
88/** @def PGM_HANDLER_VIRT_IS_VALID_STATUS
89 * Checks if valid virtual access handler return code (normal handler, not PF).
90 *
91 * Checks if the given strict status code is one of the expected ones for a
92 * virtual access handler in the current context.
93 *
94 * @returns true or false.
95 * @param a_rcStrict The status code.
96 * @param a_fWrite Whether it is a write or read being serviced.
97 *
98 * @remarks We wish to keep the list of statuses here as short as possible.
99 * When changing, please make sure to update the PGMPhysRead,
100 * PGMPhysWrite, PGMPhysReadGCPtr and PGMPhysWriteGCPtr docs too.
101 */
102#ifdef IN_RING3
103# define PGM_HANDLER_VIRT_IS_VALID_STATUS(a_rcStrict, a_fWrite) \
104 ( (a_rcStrict) == VINF_SUCCESS \
105 || (a_rcStrict) == VINF_PGM_HANDLER_DO_DEFAULT)
106#elif defined(IN_RING0)
107# define PGM_HANDLER_VIRT_IS_VALID_STATUS(a_rcStrict, a_fWrite) \
108 (false /* no virtual handlers in ring-0! */ )
109#else
110# error "Context?"
111#endif
112
113
114
115#ifndef IN_RING3
116
117/**
118 * @callback_method_impl{FNPGMPHYSHANDLER,
119 * Dummy for forcing ring-3 handling of the access.}
120 */
121DECLEXPORT(VBOXSTRICTRC)
122pgmPhysHandlerRedirectToHC(PVMCC pVM, PVMCPUCC pVCpu, RTGCPHYS GCPhys, void *pvPhys, void *pvBuf, size_t cbBuf,
123 PGMACCESSTYPE enmAccessType, PGMACCESSORIGIN enmOrigin, void *pvUser)
124{
125 NOREF(pVM); NOREF(pVCpu); NOREF(GCPhys); NOREF(pvPhys); NOREF(pvBuf); NOREF(cbBuf);
126 NOREF(enmAccessType); NOREF(enmOrigin); NOREF(pvUser);
127 return VINF_EM_RAW_EMULATE_INSTR;
128}
129
130
131/**
132 * @callback_method_impl{FNPGMRZPHYSPFHANDLER,
133 * Dummy for forcing ring-3 handling of the access.}
134 */
135VMMDECL(VBOXSTRICTRC) pgmPhysPfHandlerRedirectToHC(PVMCC pVM, PVMCPUCC pVCpu, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame,
136 RTGCPTR pvFault, RTGCPHYS GCPhysFault, void *pvUser)
137{
138 NOREF(pVM); NOREF(pVCpu); NOREF(uErrorCode); NOREF(pRegFrame); NOREF(pvFault); NOREF(GCPhysFault); NOREF(pvUser);
139 return VINF_EM_RAW_EMULATE_INSTR;
140}
141
142
143/**
144 * @callback_method_impl{FNPGMRZPHYSPFHANDLER,
145 * \#PF access handler callback for guest ROM range write access.}
146 *
147 * @remarks The @a pvUser argument points to the PGMROMRANGE.
148 */
149DECLEXPORT(VBOXSTRICTRC) pgmPhysRomWritePfHandler(PVMCC pVM, PVMCPUCC pVCpu, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame,
150 RTGCPTR pvFault, RTGCPHYS GCPhysFault, void *pvUser)
151{
152 int rc;
153 PPGMROMRANGE pRom = (PPGMROMRANGE)pvUser;
154 uint32_t iPage = (GCPhysFault - pRom->GCPhys) >> PAGE_SHIFT;
155 NOREF(uErrorCode); NOREF(pvFault);
156
157 Assert(uErrorCode & X86_TRAP_PF_RW); /* This shall not be used for read access! */
158
159 Assert(iPage < (pRom->cb >> PAGE_SHIFT));
160 switch (pRom->aPages[iPage].enmProt)
161 {
162 case PGMROMPROT_READ_ROM_WRITE_IGNORE:
163 case PGMROMPROT_READ_RAM_WRITE_IGNORE:
164 {
165 /*
166 * If it's a simple instruction which doesn't change the cpu state
167 * we will simply skip it. Otherwise we'll have to defer it to REM.
168 */
169 uint32_t cbOp;
170 PDISCPUSTATE pDis = &pVCpu->pgm.s.DisState;
171 rc = EMInterpretDisasCurrent(pVM, pVCpu, pDis, &cbOp);
172 if ( RT_SUCCESS(rc)
173 && pDis->uCpuMode == DISCPUMODE_32BIT /** @todo why does this matter? */
174 && !(pDis->fPrefix & (DISPREFIX_REPNE | DISPREFIX_REP | DISPREFIX_SEG)))
175 {
176 switch (pDis->bOpCode)
177 {
178 /** @todo Find other instructions we can safely skip, possibly
179 * adding this kind of detection to DIS or EM. */
180 case OP_MOV:
181 pRegFrame->rip += cbOp;
182 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZGuestROMWriteHandled);
183 return VINF_SUCCESS;
184 }
185 }
186 break;
187 }
188
189 case PGMROMPROT_READ_RAM_WRITE_RAM:
190 pRom->aPages[iPage].LiveSave.fWrittenTo = true;
191 rc = PGMHandlerPhysicalPageTempOff(pVM, pRom->GCPhys, GCPhysFault & X86_PTE_PG_MASK);
192 AssertRC(rc);
193 break; /** @todo Must edit the shadow PT and restart the instruction, not use the interpreter! */
194
195 case PGMROMPROT_READ_ROM_WRITE_RAM:
196 /* Handle it in ring-3 because it's *way* easier there. */
197 pRom->aPages[iPage].LiveSave.fWrittenTo = true;
198 break;
199
200 default:
201 AssertMsgFailedReturn(("enmProt=%d iPage=%d GCPhysFault=%RGp\n",
202 pRom->aPages[iPage].enmProt, iPage, GCPhysFault),
203 VERR_IPE_NOT_REACHED_DEFAULT_CASE);
204 }
205
206 STAM_COUNTER_INC(&pVCpu->pgm.s.CTX_SUFF(pStats)->StatRZGuestROMWriteUnhandled);
207 return VINF_EM_RAW_EMULATE_INSTR;
208}
209
210#endif /* !IN_RING3 */
211
212
213/**
214 * @callback_method_impl{FNPGMPHYSHANDLER,
215 * Access handler callback for ROM write accesses.}
216 *
217 * @remarks The @a pvUser argument points to the PGMROMRANGE.
218 */
219PGM_ALL_CB2_DECL(VBOXSTRICTRC)
220pgmPhysRomWriteHandler(PVMCC pVM, PVMCPUCC pVCpu, RTGCPHYS GCPhys, void *pvPhys, void *pvBuf, size_t cbBuf,
221 PGMACCESSTYPE enmAccessType, PGMACCESSORIGIN enmOrigin, void *pvUser)
222{
223 PPGMROMRANGE pRom = (PPGMROMRANGE)pvUser;
224 const uint32_t iPage = (GCPhys - pRom->GCPhys) >> PAGE_SHIFT;
225 Assert(iPage < (pRom->cb >> PAGE_SHIFT));
226 PPGMROMPAGE pRomPage = &pRom->aPages[iPage];
227 Log5(("pgmPhysRomWriteHandler: %d %c %#08RGp %#04zx\n", pRomPage->enmProt, enmAccessType == PGMACCESSTYPE_READ ? 'R' : 'W', GCPhys, cbBuf));
228 NOREF(pVCpu); NOREF(pvPhys); NOREF(enmOrigin);
229
230 if (enmAccessType == PGMACCESSTYPE_READ)
231 {
232 switch (pRomPage->enmProt)
233 {
234 /*
235 * Take the default action.
236 */
237 case PGMROMPROT_READ_ROM_WRITE_IGNORE:
238 case PGMROMPROT_READ_RAM_WRITE_IGNORE:
239 case PGMROMPROT_READ_ROM_WRITE_RAM:
240 case PGMROMPROT_READ_RAM_WRITE_RAM:
241 return VINF_PGM_HANDLER_DO_DEFAULT;
242
243 default:
244 AssertMsgFailedReturn(("enmProt=%d iPage=%d GCPhys=%RGp\n",
245 pRom->aPages[iPage].enmProt, iPage, GCPhys),
246 VERR_IPE_NOT_REACHED_DEFAULT_CASE);
247 }
248 }
249 else
250 {
251 Assert(enmAccessType == PGMACCESSTYPE_WRITE);
252 switch (pRomPage->enmProt)
253 {
254 /*
255 * Ignore writes.
256 */
257 case PGMROMPROT_READ_ROM_WRITE_IGNORE:
258 case PGMROMPROT_READ_RAM_WRITE_IGNORE:
259 return VINF_SUCCESS;
260
261 /*
262 * Write to the RAM page.
263 */
264 case PGMROMPROT_READ_ROM_WRITE_RAM:
265 case PGMROMPROT_READ_RAM_WRITE_RAM: /* yes this will get here too, it's *way* simpler that way. */
266 {
267 /* This should be impossible now, pvPhys doesn't work cross page anylonger. */
268 Assert(((GCPhys - pRom->GCPhys + cbBuf - 1) >> PAGE_SHIFT) == iPage);
269
270 /*
271 * Take the lock, do lazy allocation, map the page and copy the data.
272 *
273 * Note that we have to bypass the mapping TLB since it works on
274 * guest physical addresses and entering the shadow page would
275 * kind of screw things up...
276 */
277 int rc = pgmLock(pVM);
278 AssertRC(rc);
279
280 PPGMPAGE pShadowPage = &pRomPage->Shadow;
281 if (!PGMROMPROT_IS_ROM(pRomPage->enmProt))
282 {
283 pShadowPage = pgmPhysGetPage(pVM, GCPhys);
284 AssertLogRelReturn(pShadowPage, VERR_PGM_PHYS_PAGE_GET_IPE);
285 }
286
287 void *pvDstPage;
288 rc = pgmPhysPageMakeWritableAndMap(pVM, pShadowPage, GCPhys & X86_PTE_PG_MASK, &pvDstPage);
289 if (RT_SUCCESS(rc))
290 {
291 memcpy((uint8_t *)pvDstPage + (GCPhys & PAGE_OFFSET_MASK), pvBuf, cbBuf);
292 pRomPage->LiveSave.fWrittenTo = true;
293
294 AssertMsg( rc == VINF_SUCCESS
295 || ( rc == VINF_PGM_SYNC_CR3
296 && VMCPU_FF_IS_ANY_SET(pVCpu, VMCPU_FF_PGM_SYNC_CR3 | VMCPU_FF_PGM_SYNC_CR3_NON_GLOBAL))
297 , ("%Rrc\n", rc));
298 rc = VINF_SUCCESS;
299 }
300
301 pgmUnlock(pVM);
302 return rc;
303 }
304
305 default:
306 AssertMsgFailedReturn(("enmProt=%d iPage=%d GCPhys=%RGp\n",
307 pRom->aPages[iPage].enmProt, iPage, GCPhys),
308 VERR_IPE_NOT_REACHED_DEFAULT_CASE);
309 }
310 }
311}
312
313
314/**
315 * Invalidates the RAM range TLBs.
316 *
317 * @param pVM The cross context VM structure.
318 */
319void pgmPhysInvalidRamRangeTlbs(PVMCC pVM)
320{
321 pgmLock(pVM);
322 RT_ZERO(pVM->pgm.s.apRamRangesTlbR3);
323 RT_ZERO(pVM->pgm.s.apRamRangesTlbR0);
324 pgmUnlock(pVM);
325}
326
327
328/**
329 * Tests if a value of type RTGCPHYS is negative if the type had been signed
330 * instead of unsigned.
331 *
332 * @returns @c true if negative, @c false if positive or zero.
333 * @param a_GCPhys The value to test.
334 * @todo Move me to iprt/types.h.
335 */
336#define RTGCPHYS_IS_NEGATIVE(a_GCPhys) ((a_GCPhys) & ((RTGCPHYS)1 << (sizeof(RTGCPHYS)*8 - 1)))
337
338
339/**
340 * Slow worker for pgmPhysGetRange.
341 *
342 * @copydoc pgmPhysGetRange
343 */
344PPGMRAMRANGE pgmPhysGetRangeSlow(PVM pVM, RTGCPHYS GCPhys)
345{
346 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbMisses));
347
348 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(pRamRangeTree);
349 while (pRam)
350 {
351 RTGCPHYS off = GCPhys - pRam->GCPhys;
352 if (off < pRam->cb)
353 {
354 pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)] = pRam;
355 return pRam;
356 }
357 if (RTGCPHYS_IS_NEGATIVE(off))
358 pRam = pRam->CTX_SUFF(pLeft);
359 else
360 pRam = pRam->CTX_SUFF(pRight);
361 }
362 return NULL;
363}
364
365
366/**
367 * Slow worker for pgmPhysGetRangeAtOrAbove.
368 *
369 * @copydoc pgmPhysGetRangeAtOrAbove
370 */
371PPGMRAMRANGE pgmPhysGetRangeAtOrAboveSlow(PVM pVM, RTGCPHYS GCPhys)
372{
373 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbMisses));
374
375 PPGMRAMRANGE pLastLeft = NULL;
376 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(pRamRangeTree);
377 while (pRam)
378 {
379 RTGCPHYS off = GCPhys - pRam->GCPhys;
380 if (off < pRam->cb)
381 {
382 pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)] = pRam;
383 return pRam;
384 }
385 if (RTGCPHYS_IS_NEGATIVE(off))
386 {
387 pLastLeft = pRam;
388 pRam = pRam->CTX_SUFF(pLeft);
389 }
390 else
391 pRam = pRam->CTX_SUFF(pRight);
392 }
393 return pLastLeft;
394}
395
396
397/**
398 * Slow worker for pgmPhysGetPage.
399 *
400 * @copydoc pgmPhysGetPage
401 */
402PPGMPAGE pgmPhysGetPageSlow(PVM pVM, RTGCPHYS GCPhys)
403{
404 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbMisses));
405
406 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(pRamRangeTree);
407 while (pRam)
408 {
409 RTGCPHYS off = GCPhys - pRam->GCPhys;
410 if (off < pRam->cb)
411 {
412 pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)] = pRam;
413 return &pRam->aPages[off >> PAGE_SHIFT];
414 }
415
416 if (RTGCPHYS_IS_NEGATIVE(off))
417 pRam = pRam->CTX_SUFF(pLeft);
418 else
419 pRam = pRam->CTX_SUFF(pRight);
420 }
421 return NULL;
422}
423
424
425/**
426 * Slow worker for pgmPhysGetPageEx.
427 *
428 * @copydoc pgmPhysGetPageEx
429 */
430int pgmPhysGetPageExSlow(PVM pVM, RTGCPHYS GCPhys, PPPGMPAGE ppPage)
431{
432 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbMisses));
433
434 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(pRamRangeTree);
435 while (pRam)
436 {
437 RTGCPHYS off = GCPhys - pRam->GCPhys;
438 if (off < pRam->cb)
439 {
440 pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)] = pRam;
441 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
442 return VINF_SUCCESS;
443 }
444
445 if (RTGCPHYS_IS_NEGATIVE(off))
446 pRam = pRam->CTX_SUFF(pLeft);
447 else
448 pRam = pRam->CTX_SUFF(pRight);
449 }
450
451 *ppPage = NULL;
452 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
453}
454
455
456/**
457 * Slow worker for pgmPhysGetPageAndRangeEx.
458 *
459 * @copydoc pgmPhysGetPageAndRangeEx
460 */
461int pgmPhysGetPageAndRangeExSlow(PVM pVM, RTGCPHYS GCPhys, PPPGMPAGE ppPage, PPGMRAMRANGE *ppRam)
462{
463 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,RamRangeTlbMisses));
464
465 PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(pRamRangeTree);
466 while (pRam)
467 {
468 RTGCPHYS off = GCPhys - pRam->GCPhys;
469 if (off < pRam->cb)
470 {
471 pVM->pgm.s.CTX_SUFF(apRamRangesTlb)[PGM_RAMRANGE_TLB_IDX(GCPhys)] = pRam;
472 *ppRam = pRam;
473 *ppPage = &pRam->aPages[off >> PAGE_SHIFT];
474 return VINF_SUCCESS;
475 }
476
477 if (RTGCPHYS_IS_NEGATIVE(off))
478 pRam = pRam->CTX_SUFF(pLeft);
479 else
480 pRam = pRam->CTX_SUFF(pRight);
481 }
482
483 *ppRam = NULL;
484 *ppPage = NULL;
485 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
486}
487
488
489/**
490 * Checks if Address Gate 20 is enabled or not.
491 *
492 * @returns true if enabled.
493 * @returns false if disabled.
494 * @param pVCpu The cross context virtual CPU structure.
495 */
496VMMDECL(bool) PGMPhysIsA20Enabled(PVMCPU pVCpu)
497{
498 LogFlow(("PGMPhysIsA20Enabled %d\n", pVCpu->pgm.s.fA20Enabled));
499 return pVCpu->pgm.s.fA20Enabled;
500}
501
502
503/**
504 * Validates a GC physical address.
505 *
506 * @returns true if valid.
507 * @returns false if invalid.
508 * @param pVM The cross context VM structure.
509 * @param GCPhys The physical address to validate.
510 */
511VMMDECL(bool) PGMPhysIsGCPhysValid(PVMCC pVM, RTGCPHYS GCPhys)
512{
513 PPGMPAGE pPage = pgmPhysGetPage(pVM, GCPhys);
514 return pPage != NULL;
515}
516
517
518/**
519 * Checks if a GC physical address is a normal page,
520 * i.e. not ROM, MMIO or reserved.
521 *
522 * @returns true if normal.
523 * @returns false if invalid, ROM, MMIO or reserved page.
524 * @param pVM The cross context VM structure.
525 * @param GCPhys The physical address to check.
526 */
527VMMDECL(bool) PGMPhysIsGCPhysNormal(PVMCC pVM, RTGCPHYS GCPhys)
528{
529 PPGMPAGE pPage = pgmPhysGetPage(pVM, GCPhys);
530 return pPage
531 && PGM_PAGE_GET_TYPE(pPage) == PGMPAGETYPE_RAM;
532}
533
534
535/**
536 * Converts a GC physical address to a HC physical address.
537 *
538 * @returns VINF_SUCCESS on success.
539 * @returns VERR_PGM_PHYS_PAGE_RESERVED it it's a valid GC physical
540 * page but has no physical backing.
541 * @returns VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid
542 * GC physical address.
543 *
544 * @param pVM The cross context VM structure.
545 * @param GCPhys The GC physical address to convert.
546 * @param pHCPhys Where to store the HC physical address on success.
547 */
548VMM_INT_DECL(int) PGMPhysGCPhys2HCPhys(PVMCC pVM, RTGCPHYS GCPhys, PRTHCPHYS pHCPhys)
549{
550 pgmLock(pVM);
551 PPGMPAGE pPage;
552 int rc = pgmPhysGetPageEx(pVM, GCPhys, &pPage);
553 if (RT_SUCCESS(rc))
554 *pHCPhys = PGM_PAGE_GET_HCPHYS(pPage) | (GCPhys & PAGE_OFFSET_MASK);
555 pgmUnlock(pVM);
556 return rc;
557}
558
559
560/**
561 * Invalidates all page mapping TLBs.
562 *
563 * @param pVM The cross context VM structure.
564 */
565void pgmPhysInvalidatePageMapTLB(PVMCC pVM)
566{
567 pgmLock(pVM);
568 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->StatPageMapTlbFlushes);
569
570 /* Clear the R3 & R0 TLBs completely. */
571 for (unsigned i = 0; i < RT_ELEMENTS(pVM->pgm.s.PhysTlbR0.aEntries); i++)
572 {
573 pVM->pgm.s.PhysTlbR0.aEntries[i].GCPhys = NIL_RTGCPHYS;
574 pVM->pgm.s.PhysTlbR0.aEntries[i].pPage = 0;
575 pVM->pgm.s.PhysTlbR0.aEntries[i].pMap = 0;
576 pVM->pgm.s.PhysTlbR0.aEntries[i].pv = 0;
577 }
578
579 for (unsigned i = 0; i < RT_ELEMENTS(pVM->pgm.s.PhysTlbR3.aEntries); i++)
580 {
581 pVM->pgm.s.PhysTlbR3.aEntries[i].GCPhys = NIL_RTGCPHYS;
582 pVM->pgm.s.PhysTlbR3.aEntries[i].pPage = 0;
583 pVM->pgm.s.PhysTlbR3.aEntries[i].pMap = 0;
584 pVM->pgm.s.PhysTlbR3.aEntries[i].pv = 0;
585 }
586
587 pgmUnlock(pVM);
588}
589
590
591/**
592 * Invalidates a page mapping TLB entry
593 *
594 * @param pVM The cross context VM structure.
595 * @param GCPhys GCPhys entry to flush
596 */
597void pgmPhysInvalidatePageMapTLBEntry(PVM pVM, RTGCPHYS GCPhys)
598{
599 PGM_LOCK_ASSERT_OWNER(pVM);
600
601 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->StatPageMapTlbFlushEntry);
602
603 unsigned const idx = PGM_PAGER3MAPTLB_IDX(GCPhys);
604
605 pVM->pgm.s.PhysTlbR0.aEntries[idx].GCPhys = NIL_RTGCPHYS;
606 pVM->pgm.s.PhysTlbR0.aEntries[idx].pPage = 0;
607 pVM->pgm.s.PhysTlbR0.aEntries[idx].pMap = 0;
608 pVM->pgm.s.PhysTlbR0.aEntries[idx].pv = 0;
609
610 pVM->pgm.s.PhysTlbR3.aEntries[idx].GCPhys = NIL_RTGCPHYS;
611 pVM->pgm.s.PhysTlbR3.aEntries[idx].pPage = 0;
612 pVM->pgm.s.PhysTlbR3.aEntries[idx].pMap = 0;
613 pVM->pgm.s.PhysTlbR3.aEntries[idx].pv = 0;
614}
615
616
617/**
618 * Makes sure that there is at least one handy page ready for use.
619 *
620 * This will also take the appropriate actions when reaching water-marks.
621 *
622 * @returns VBox status code.
623 * @retval VINF_SUCCESS on success.
624 * @retval VERR_EM_NO_MEMORY if we're really out of memory.
625 *
626 * @param pVM The cross context VM structure.
627 *
628 * @remarks Must be called from within the PGM critical section. It may
629 * nip back to ring-3/0 in some cases.
630 */
631static int pgmPhysEnsureHandyPage(PVMCC pVM)
632{
633 AssertMsg(pVM->pgm.s.cHandyPages <= RT_ELEMENTS(pVM->pgm.s.aHandyPages), ("%d\n", pVM->pgm.s.cHandyPages));
634
635 /*
636 * Do we need to do anything special?
637 */
638#ifdef IN_RING3
639 if (pVM->pgm.s.cHandyPages <= RT_MAX(PGM_HANDY_PAGES_SET_FF, PGM_HANDY_PAGES_R3_ALLOC))
640#else
641 if (pVM->pgm.s.cHandyPages <= RT_MAX(PGM_HANDY_PAGES_SET_FF, PGM_HANDY_PAGES_RZ_TO_R3))
642#endif
643 {
644 /*
645 * Allocate pages only if we're out of them, or in ring-3, almost out.
646 */
647#ifdef IN_RING3
648 if (pVM->pgm.s.cHandyPages <= PGM_HANDY_PAGES_R3_ALLOC)
649#else
650 if (pVM->pgm.s.cHandyPages <= PGM_HANDY_PAGES_RZ_ALLOC)
651#endif
652 {
653 Log(("PGM: cHandyPages=%u out of %u -> allocate more; VM_FF_PGM_NO_MEMORY=%RTbool\n",
654 pVM->pgm.s.cHandyPages, RT_ELEMENTS(pVM->pgm.s.aHandyPages), VM_FF_IS_SET(pVM, VM_FF_PGM_NO_MEMORY) ));
655#ifdef IN_RING3
656 int rc = PGMR3PhysAllocateHandyPages(pVM);
657#else
658 int rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_PGM_ALLOCATE_HANDY_PAGES, 0);
659#endif
660 if (RT_UNLIKELY(rc != VINF_SUCCESS))
661 {
662 if (RT_FAILURE(rc))
663 return rc;
664 AssertMsgReturn(rc == VINF_EM_NO_MEMORY, ("%Rrc\n", rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
665 if (!pVM->pgm.s.cHandyPages)
666 {
667 LogRel(("PGM: no more handy pages!\n"));
668 return VERR_EM_NO_MEMORY;
669 }
670 Assert(VM_FF_IS_SET(pVM, VM_FF_PGM_NEED_HANDY_PAGES));
671 Assert(VM_FF_IS_SET(pVM, VM_FF_PGM_NO_MEMORY));
672#ifndef IN_RING3
673 VMCPU_FF_SET(VMMGetCpu(pVM), VMCPU_FF_TO_R3); /* paranoia */
674#endif
675 }
676 AssertMsgReturn( pVM->pgm.s.cHandyPages > 0
677 && pVM->pgm.s.cHandyPages <= RT_ELEMENTS(pVM->pgm.s.aHandyPages),
678 ("%u\n", pVM->pgm.s.cHandyPages),
679 VERR_PGM_HANDY_PAGE_IPE);
680 }
681 else
682 {
683 if (pVM->pgm.s.cHandyPages <= PGM_HANDY_PAGES_SET_FF)
684 VM_FF_SET(pVM, VM_FF_PGM_NEED_HANDY_PAGES);
685#ifndef IN_RING3
686 if (pVM->pgm.s.cHandyPages <= PGM_HANDY_PAGES_RZ_TO_R3)
687 {
688 Log(("PGM: VM_FF_TO_R3 - cHandyPages=%u out of %u\n", pVM->pgm.s.cHandyPages, RT_ELEMENTS(pVM->pgm.s.aHandyPages)));
689 VMCPU_FF_SET(VMMGetCpu(pVM), VMCPU_FF_TO_R3);
690 }
691#endif
692 }
693 }
694
695 return VINF_SUCCESS;
696}
697
698
699
700/**
701 * Replace a zero or shared page with new page that we can write to.
702 *
703 * @returns The following VBox status codes.
704 * @retval VINF_SUCCESS on success, pPage is modified.
705 * @retval VINF_PGM_SYNC_CR3 on success and a page pool flush is pending.
706 * @retval VERR_EM_NO_MEMORY if we're totally out of memory.
707 *
708 * @todo Propagate VERR_EM_NO_MEMORY up the call tree.
709 *
710 * @param pVM The cross context VM structure.
711 * @param pPage The physical page tracking structure. This will
712 * be modified on success.
713 * @param GCPhys The address of the page.
714 *
715 * @remarks Must be called from within the PGM critical section. It may
716 * nip back to ring-3/0 in some cases.
717 *
718 * @remarks This function shouldn't really fail, however if it does
719 * it probably means we've screwed up the size of handy pages and/or
720 * the low-water mark. Or, that some device I/O is causing a lot of
721 * pages to be allocated while while the host is in a low-memory
722 * condition. This latter should be handled elsewhere and in a more
723 * controlled manner, it's on the @bugref{3170} todo list...
724 */
725int pgmPhysAllocPage(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys)
726{
727 LogFlow(("pgmPhysAllocPage: %R[pgmpage] %RGp\n", pPage, GCPhys));
728
729 /*
730 * Prereqs.
731 */
732 PGM_LOCK_ASSERT_OWNER(pVM);
733 AssertMsg(PGM_PAGE_IS_ZERO(pPage) || PGM_PAGE_IS_SHARED(pPage), ("%R[pgmpage] %RGp\n", pPage, GCPhys));
734 Assert(!PGM_PAGE_IS_MMIO_OR_ALIAS(pPage));
735
736# ifdef PGM_WITH_LARGE_PAGES
737 /*
738 * Try allocate a large page if applicable.
739 */
740 if ( PGMIsUsingLargePages(pVM)
741 && PGM_PAGE_GET_TYPE(pPage) == PGMPAGETYPE_RAM
742 && !VM_IS_NEM_ENABLED(pVM)) /** @todo NEM: Implement large pages support. */
743 {
744 RTGCPHYS GCPhysBase = GCPhys & X86_PDE2M_PAE_PG_MASK;
745 PPGMPAGE pBasePage;
746
747 int rc = pgmPhysGetPageEx(pVM, GCPhysBase, &pBasePage);
748 AssertRCReturn(rc, rc); /* paranoia; can't happen. */
749 if (PGM_PAGE_GET_PDE_TYPE(pBasePage) == PGM_PAGE_PDE_TYPE_DONTCARE)
750 {
751 rc = pgmPhysAllocLargePage(pVM, GCPhys);
752 if (rc == VINF_SUCCESS)
753 return rc;
754 }
755 /* Mark the base as type page table, so we don't check over and over again. */
756 PGM_PAGE_SET_PDE_TYPE(pVM, pBasePage, PGM_PAGE_PDE_TYPE_PT);
757
758 /* fall back to 4KB pages. */
759 }
760# endif
761
762 /*
763 * Flush any shadow page table mappings of the page.
764 * When VBOX_WITH_NEW_LAZY_PAGE_ALLOC isn't defined, there shouldn't be any.
765 */
766 bool fFlushTLBs = false;
767 int rc = pgmPoolTrackUpdateGCPhys(pVM, GCPhys, pPage, true /*fFlushTLBs*/, &fFlushTLBs);
768 AssertMsgReturn(rc == VINF_SUCCESS || rc == VINF_PGM_SYNC_CR3, ("%Rrc\n", rc), RT_FAILURE(rc) ? rc : VERR_IPE_UNEXPECTED_STATUS);
769
770 /*
771 * Ensure that we've got a page handy, take it and use it.
772 */
773 int rc2 = pgmPhysEnsureHandyPage(pVM);
774 if (RT_FAILURE(rc2))
775 {
776 if (fFlushTLBs)
777 PGM_INVL_ALL_VCPU_TLBS(pVM);
778 Assert(rc2 == VERR_EM_NO_MEMORY);
779 return rc2;
780 }
781 /* re-assert preconditions since pgmPhysEnsureHandyPage may do a context switch. */
782 PGM_LOCK_ASSERT_OWNER(pVM);
783 AssertMsg(PGM_PAGE_IS_ZERO(pPage) || PGM_PAGE_IS_SHARED(pPage), ("%R[pgmpage] %RGp\n", pPage, GCPhys));
784 Assert(!PGM_PAGE_IS_MMIO_OR_ALIAS(pPage));
785
786 uint32_t iHandyPage = --pVM->pgm.s.cHandyPages;
787 AssertMsg(iHandyPage < RT_ELEMENTS(pVM->pgm.s.aHandyPages), ("%d\n", iHandyPage));
788 Assert(pVM->pgm.s.aHandyPages[iHandyPage].HCPhysGCPhys != NIL_RTHCPHYS);
789 Assert(!(pVM->pgm.s.aHandyPages[iHandyPage].HCPhysGCPhys & ~X86_PTE_PAE_PG_MASK));
790 Assert(pVM->pgm.s.aHandyPages[iHandyPage].idPage != NIL_GMM_PAGEID);
791 Assert(pVM->pgm.s.aHandyPages[iHandyPage].idSharedPage == NIL_GMM_PAGEID);
792
793 /*
794 * There are one or two action to be taken the next time we allocate handy pages:
795 * - Tell the GMM (global memory manager) what the page is being used for.
796 * (Speeds up replacement operations - sharing and defragmenting.)
797 * - If the current backing is shared, it must be freed.
798 */
799 const RTHCPHYS HCPhys = pVM->pgm.s.aHandyPages[iHandyPage].HCPhysGCPhys;
800 pVM->pgm.s.aHandyPages[iHandyPage].HCPhysGCPhys = GCPhys & ~(RTGCPHYS)PAGE_OFFSET_MASK;
801
802 void const *pvSharedPage = NULL;
803 if (PGM_PAGE_IS_SHARED(pPage))
804 {
805 /* Mark this shared page for freeing/dereferencing. */
806 pVM->pgm.s.aHandyPages[iHandyPage].idSharedPage = PGM_PAGE_GET_PAGEID(pPage);
807 Assert(PGM_PAGE_GET_PAGEID(pPage) != NIL_GMM_PAGEID);
808
809 Log(("PGM: Replaced shared page %#x at %RGp with %#x / %RHp\n", PGM_PAGE_GET_PAGEID(pPage),
810 GCPhys, pVM->pgm.s.aHandyPages[iHandyPage].idPage, HCPhys));
811 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PageReplaceShared));
812 pVM->pgm.s.cSharedPages--;
813
814 /* Grab the address of the page so we can make a copy later on. (safe) */
815 rc = pgmPhysPageMapReadOnly(pVM, pPage, GCPhys, &pvSharedPage);
816 AssertRC(rc);
817 }
818 else
819 {
820 Log2(("PGM: Replaced zero page %RGp with %#x / %RHp\n", GCPhys, pVM->pgm.s.aHandyPages[iHandyPage].idPage, HCPhys));
821 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->StatRZPageReplaceZero);
822 pVM->pgm.s.cZeroPages--;
823 }
824
825 /*
826 * Do the PGMPAGE modifications.
827 */
828 pVM->pgm.s.cPrivatePages++;
829 PGM_PAGE_SET_HCPHYS(pVM, pPage, HCPhys);
830 PGM_PAGE_SET_PAGEID(pVM, pPage, pVM->pgm.s.aHandyPages[iHandyPage].idPage);
831 PGM_PAGE_SET_STATE(pVM, pPage, PGM_PAGE_STATE_ALLOCATED);
832 PGM_PAGE_SET_PDE_TYPE(pVM, pPage, PGM_PAGE_PDE_TYPE_PT);
833 pgmPhysInvalidatePageMapTLBEntry(pVM, GCPhys);
834
835 /* Copy the shared page contents to the replacement page. */
836 if (pvSharedPage)
837 {
838 /* Get the virtual address of the new page. */
839 PGMPAGEMAPLOCK PgMpLck;
840 void *pvNewPage;
841 rc = pgmPhysGCPhys2CCPtrInternal(pVM, pPage, GCPhys, &pvNewPage, &PgMpLck); AssertRC(rc);
842 if (RT_SUCCESS(rc))
843 {
844 memcpy(pvNewPage, pvSharedPage, PAGE_SIZE); /** @todo todo write ASMMemCopyPage */
845 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
846 }
847 }
848
849 if ( fFlushTLBs
850 && rc != VINF_PGM_GCPHYS_ALIASED)
851 PGM_INVL_ALL_VCPU_TLBS(pVM);
852
853 /*
854 * Notify NEM about the mapping change for this page.
855 *
856 * Note! Shadow ROM pages are complicated as they can definitely be
857 * allocated while not visible, so play safe.
858 */
859 if (VM_IS_NEM_ENABLED(pVM))
860 {
861 PGMPAGETYPE enmType = (PGMPAGETYPE)PGM_PAGE_GET_TYPE(pPage);
862 if ( enmType != PGMPAGETYPE_ROM_SHADOW
863 || pgmPhysGetPage(pVM, GCPhys) == pPage)
864 {
865 uint8_t u2State = PGM_PAGE_GET_NEM_STATE(pPage);
866 rc2 = NEMHCNotifyPhysPageAllocated(pVM, GCPhys & ~(RTGCPHYS)X86_PAGE_OFFSET_MASK, HCPhys,
867 pgmPhysPageCalcNemProtection(pPage, enmType), enmType, &u2State);
868 if (RT_SUCCESS(rc))
869 PGM_PAGE_SET_NEM_STATE(pPage, u2State);
870 else
871 rc = rc2;
872 }
873 }
874
875 return rc;
876}
877
878#ifdef PGM_WITH_LARGE_PAGES
879
880/**
881 * Replace a 2 MB range of zero pages with new pages that we can write to.
882 *
883 * @returns The following VBox status codes.
884 * @retval VINF_SUCCESS on success, pPage is modified.
885 * @retval VINF_PGM_SYNC_CR3 on success and a page pool flush is pending.
886 * @retval VERR_EM_NO_MEMORY if we're totally out of memory.
887 *
888 * @todo Propagate VERR_EM_NO_MEMORY up the call tree.
889 *
890 * @param pVM The cross context VM structure.
891 * @param GCPhys The address of the page.
892 *
893 * @remarks Must be called from within the PGM critical section. It may
894 * nip back to ring-3/0 in some cases.
895 */
896int pgmPhysAllocLargePage(PVMCC pVM, RTGCPHYS GCPhys)
897{
898 RTGCPHYS GCPhysBase = GCPhys & X86_PDE2M_PAE_PG_MASK;
899 LogFlow(("pgmPhysAllocLargePage: %RGp base %RGp\n", GCPhys, GCPhysBase));
900 Assert(!VM_IS_NEM_ENABLED(pVM)); /** @todo NEM: Large page support. */
901
902 /*
903 * Prereqs.
904 */
905 PGM_LOCK_ASSERT_OWNER(pVM);
906 Assert(PGMIsUsingLargePages(pVM));
907
908 PPGMPAGE pFirstPage;
909 int rc = pgmPhysGetPageEx(pVM, GCPhysBase, &pFirstPage);
910 if ( RT_SUCCESS(rc)
911 && PGM_PAGE_GET_TYPE(pFirstPage) == PGMPAGETYPE_RAM)
912 {
913 unsigned uPDEType = PGM_PAGE_GET_PDE_TYPE(pFirstPage);
914
915 /* Don't call this function for already allocated pages. */
916 Assert(uPDEType != PGM_PAGE_PDE_TYPE_PDE);
917
918 if ( uPDEType == PGM_PAGE_PDE_TYPE_DONTCARE
919 && PGM_PAGE_GET_STATE(pFirstPage) == PGM_PAGE_STATE_ZERO)
920 {
921 /* Lazy approach: check all pages in the 2 MB range.
922 * The whole range must be ram and unallocated. */
923 GCPhys = GCPhysBase;
924 unsigned iPage;
925 for (iPage = 0; iPage < _2M/PAGE_SIZE; iPage++)
926 {
927 PPGMPAGE pSubPage;
928 rc = pgmPhysGetPageEx(pVM, GCPhys, &pSubPage);
929 if ( RT_FAILURE(rc)
930 || PGM_PAGE_GET_TYPE(pSubPage) != PGMPAGETYPE_RAM /* Anything other than ram implies monitoring. */
931 || PGM_PAGE_GET_STATE(pSubPage) != PGM_PAGE_STATE_ZERO) /* Allocated, monitored or shared means we can't use a large page here */
932 {
933 LogFlow(("Found page %RGp with wrong attributes (type=%d; state=%d); cancel check. rc=%d\n", GCPhys, PGM_PAGE_GET_TYPE(pSubPage), PGM_PAGE_GET_STATE(pSubPage), rc));
934 break;
935 }
936 Assert(PGM_PAGE_GET_PDE_TYPE(pSubPage) == PGM_PAGE_PDE_TYPE_DONTCARE);
937 GCPhys += PAGE_SIZE;
938 }
939 if (iPage != _2M/PAGE_SIZE)
940 {
941 /* Failed. Mark as requiring a PT so we don't check the whole thing again in the future. */
942 STAM_REL_COUNTER_INC(&pVM->pgm.s.StatLargePageRefused);
943 PGM_PAGE_SET_PDE_TYPE(pVM, pFirstPage, PGM_PAGE_PDE_TYPE_PT);
944 return VERR_PGM_INVALID_LARGE_PAGE_RANGE;
945 }
946
947 /*
948 * Do the allocation.
949 */
950# ifdef IN_RING3
951 rc = PGMR3PhysAllocateLargeHandyPage(pVM, GCPhysBase);
952# else
953 rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_PGM_ALLOCATE_LARGE_HANDY_PAGE, GCPhysBase);
954# endif
955 if (RT_SUCCESS(rc))
956 {
957 Assert(PGM_PAGE_GET_STATE(pFirstPage) == PGM_PAGE_STATE_ALLOCATED);
958 pVM->pgm.s.cLargePages++;
959 return VINF_SUCCESS;
960 }
961
962 /* If we fail once, it most likely means the host's memory is too
963 fragmented; don't bother trying again. */
964 LogFlow(("pgmPhysAllocLargePage failed with %Rrc\n", rc));
965 PGMSetLargePageUsage(pVM, false);
966 return rc;
967 }
968 }
969 return VERR_PGM_INVALID_LARGE_PAGE_RANGE;
970}
971
972
973/**
974 * Recheck the entire 2 MB range to see if we can use it again as a large page.
975 *
976 * @returns The following VBox status codes.
977 * @retval VINF_SUCCESS on success, the large page can be used again
978 * @retval VERR_PGM_INVALID_LARGE_PAGE_RANGE if it can't be reused
979 *
980 * @param pVM The cross context VM structure.
981 * @param GCPhys The address of the page.
982 * @param pLargePage Page structure of the base page
983 */
984int pgmPhysRecheckLargePage(PVMCC pVM, RTGCPHYS GCPhys, PPGMPAGE pLargePage)
985{
986 STAM_REL_COUNTER_INC(&pVM->pgm.s.StatLargePageRecheck);
987
988 Assert(!VM_IS_NEM_ENABLED(pVM)); /** @todo NEM: Large page support. */
989
990 GCPhys &= X86_PDE2M_PAE_PG_MASK;
991
992 /* Check the base page. */
993 Assert(PGM_PAGE_GET_PDE_TYPE(pLargePage) == PGM_PAGE_PDE_TYPE_PDE_DISABLED);
994 if ( PGM_PAGE_GET_STATE(pLargePage) != PGM_PAGE_STATE_ALLOCATED
995 || PGM_PAGE_GET_TYPE(pLargePage) != PGMPAGETYPE_RAM
996 || PGM_PAGE_GET_HNDL_PHYS_STATE(pLargePage) != PGM_PAGE_HNDL_PHYS_STATE_NONE)
997 {
998 LogFlow(("pgmPhysRecheckLargePage: checks failed for base page %x %x %x\n", PGM_PAGE_GET_STATE(pLargePage), PGM_PAGE_GET_TYPE(pLargePage), PGM_PAGE_GET_HNDL_PHYS_STATE(pLargePage)));
999 return VERR_PGM_INVALID_LARGE_PAGE_RANGE;
1000 }
1001
1002 STAM_PROFILE_START(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,IsValidLargePage), a);
1003 /* Check all remaining pages in the 2 MB range. */
1004 unsigned i;
1005 GCPhys += PAGE_SIZE;
1006 for (i = 1; i < _2M/PAGE_SIZE; i++)
1007 {
1008 PPGMPAGE pPage;
1009 int rc = pgmPhysGetPageEx(pVM, GCPhys, &pPage);
1010 AssertRCBreak(rc);
1011
1012 if ( PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_ALLOCATED
1013 || PGM_PAGE_GET_PDE_TYPE(pPage) != PGM_PAGE_PDE_TYPE_PDE
1014 || PGM_PAGE_GET_TYPE(pPage) != PGMPAGETYPE_RAM
1015 || PGM_PAGE_GET_HNDL_PHYS_STATE(pPage) != PGM_PAGE_HNDL_PHYS_STATE_NONE)
1016 {
1017 LogFlow(("pgmPhysRecheckLargePage: checks failed for page %d; %x %x %x\n", i, PGM_PAGE_GET_STATE(pPage), PGM_PAGE_GET_TYPE(pPage), PGM_PAGE_GET_HNDL_PHYS_STATE(pPage)));
1018 break;
1019 }
1020
1021 GCPhys += PAGE_SIZE;
1022 }
1023 STAM_PROFILE_STOP(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,IsValidLargePage), a);
1024
1025 if (i == _2M/PAGE_SIZE)
1026 {
1027 PGM_PAGE_SET_PDE_TYPE(pVM, pLargePage, PGM_PAGE_PDE_TYPE_PDE);
1028 pVM->pgm.s.cLargePagesDisabled--;
1029 Log(("pgmPhysRecheckLargePage: page %RGp can be reused!\n", GCPhys - _2M));
1030 return VINF_SUCCESS;
1031 }
1032
1033 return VERR_PGM_INVALID_LARGE_PAGE_RANGE;
1034}
1035
1036#endif /* PGM_WITH_LARGE_PAGES */
1037
1038
1039/**
1040 * Deal with a write monitored page.
1041 *
1042 * @returns VBox strict status code.
1043 *
1044 * @param pVM The cross context VM structure.
1045 * @param pPage The physical page tracking structure.
1046 * @param GCPhys The guest physical address of the page.
1047 * PGMPhysReleasePageMappingLock() passes NIL_RTGCPHYS in a
1048 * very unlikely situation where it is okay that we let NEM
1049 * fix the page access in a lazy fasion.
1050 *
1051 * @remarks Called from within the PGM critical section.
1052 */
1053void pgmPhysPageMakeWriteMonitoredWritable(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys)
1054{
1055 Assert(PGM_PAGE_GET_STATE(pPage) == PGM_PAGE_STATE_WRITE_MONITORED);
1056 PGM_PAGE_SET_WRITTEN_TO(pVM, pPage);
1057 PGM_PAGE_SET_STATE(pVM, pPage, PGM_PAGE_STATE_ALLOCATED);
1058 Assert(pVM->pgm.s.cMonitoredPages > 0);
1059 pVM->pgm.s.cMonitoredPages--;
1060 pVM->pgm.s.cWrittenToPages++;
1061
1062 /*
1063 * Notify NEM about the protection change so we won't spin forever.
1064 *
1065 * Note! NEM need to be handle to lazily correct page protection as we cannot
1066 * really get it 100% right here it seems. The page pool does this too.
1067 */
1068 if (VM_IS_NEM_ENABLED(pVM) && GCPhys != NIL_RTGCPHYS)
1069 {
1070 uint8_t u2State = PGM_PAGE_GET_NEM_STATE(pPage);
1071 PGMPAGETYPE enmType = (PGMPAGETYPE)PGM_PAGE_GET_TYPE(pPage);
1072 NEMHCNotifyPhysPageProtChanged(pVM, GCPhys, PGM_PAGE_GET_HCPHYS(pPage),
1073 pgmPhysPageCalcNemProtection(pPage, enmType), enmType, &u2State);
1074 PGM_PAGE_SET_NEM_STATE(pPage, u2State);
1075 }
1076}
1077
1078
1079/**
1080 * Deal with pages that are not writable, i.e. not in the ALLOCATED state.
1081 *
1082 * @returns VBox strict status code.
1083 * @retval VINF_SUCCESS on success.
1084 * @retval VINF_PGM_SYNC_CR3 on success and a page pool flush is pending.
1085 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1086 *
1087 * @param pVM The cross context VM structure.
1088 * @param pPage The physical page tracking structure.
1089 * @param GCPhys The address of the page.
1090 *
1091 * @remarks Called from within the PGM critical section.
1092 */
1093int pgmPhysPageMakeWritable(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys)
1094{
1095 PGM_LOCK_ASSERT_OWNER(pVM);
1096 switch (PGM_PAGE_GET_STATE(pPage))
1097 {
1098 case PGM_PAGE_STATE_WRITE_MONITORED:
1099 pgmPhysPageMakeWriteMonitoredWritable(pVM, pPage, GCPhys);
1100 RT_FALL_THRU();
1101 default: /* to shut up GCC */
1102 case PGM_PAGE_STATE_ALLOCATED:
1103 return VINF_SUCCESS;
1104
1105 /*
1106 * Zero pages can be dummy pages for MMIO or reserved memory,
1107 * so we need to check the flags before joining cause with
1108 * shared page replacement.
1109 */
1110 case PGM_PAGE_STATE_ZERO:
1111 if (PGM_PAGE_IS_MMIO(pPage))
1112 return VERR_PGM_PHYS_PAGE_RESERVED;
1113 RT_FALL_THRU();
1114 case PGM_PAGE_STATE_SHARED:
1115 return pgmPhysAllocPage(pVM, pPage, GCPhys);
1116
1117 /* Not allowed to write to ballooned pages. */
1118 case PGM_PAGE_STATE_BALLOONED:
1119 return VERR_PGM_PHYS_PAGE_BALLOONED;
1120 }
1121}
1122
1123
1124/**
1125 * Internal usage: Map the page specified by its GMM ID.
1126 *
1127 * This is similar to pgmPhysPageMap
1128 *
1129 * @returns VBox status code.
1130 *
1131 * @param pVM The cross context VM structure.
1132 * @param idPage The Page ID.
1133 * @param HCPhys The physical address (for RC).
1134 * @param ppv Where to store the mapping address.
1135 *
1136 * @remarks Called from within the PGM critical section. The mapping is only
1137 * valid while you are inside this section.
1138 */
1139int pgmPhysPageMapByPageID(PVMCC pVM, uint32_t idPage, RTHCPHYS HCPhys, void **ppv)
1140{
1141 /*
1142 * Validation.
1143 */
1144 PGM_LOCK_ASSERT_OWNER(pVM);
1145 AssertReturn(HCPhys && !(HCPhys & PAGE_OFFSET_MASK), VERR_INVALID_PARAMETER);
1146 const uint32_t idChunk = idPage >> GMM_CHUNKID_SHIFT;
1147 AssertReturn(idChunk != NIL_GMM_CHUNKID, VERR_INVALID_PARAMETER);
1148
1149#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1150 /*
1151 * Map it by HCPhys.
1152 */
1153 return pgmRZDynMapHCPageInlined(VMMGetCpu(pVM), HCPhys, ppv RTLOG_COMMA_SRC_POS);
1154
1155#else
1156 /*
1157 * Find/make Chunk TLB entry for the mapping chunk.
1158 */
1159 PPGMCHUNKR3MAP pMap;
1160 PPGMCHUNKR3MAPTLBE pTlbe = &pVM->pgm.s.ChunkR3Map.Tlb.aEntries[PGM_CHUNKR3MAPTLB_IDX(idChunk)];
1161 if (pTlbe->idChunk == idChunk)
1162 {
1163 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,ChunkR3MapTlbHits));
1164 pMap = pTlbe->pChunk;
1165 }
1166 else
1167 {
1168 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,ChunkR3MapTlbMisses));
1169
1170 /*
1171 * Find the chunk, map it if necessary.
1172 */
1173 pMap = (PPGMCHUNKR3MAP)RTAvlU32Get(&pVM->pgm.s.ChunkR3Map.pTree, idChunk);
1174 if (pMap)
1175 pMap->iLastUsed = pVM->pgm.s.ChunkR3Map.iNow;
1176 else
1177 {
1178# ifdef IN_RING0
1179 int rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_PGM_MAP_CHUNK, idChunk);
1180 AssertRCReturn(rc, rc);
1181 pMap = (PPGMCHUNKR3MAP)RTAvlU32Get(&pVM->pgm.s.ChunkR3Map.pTree, idChunk);
1182 Assert(pMap);
1183# else
1184 int rc = pgmR3PhysChunkMap(pVM, idChunk, &pMap);
1185 if (RT_FAILURE(rc))
1186 return rc;
1187# endif
1188 }
1189
1190 /*
1191 * Enter it into the Chunk TLB.
1192 */
1193 pTlbe->idChunk = idChunk;
1194 pTlbe->pChunk = pMap;
1195 }
1196
1197 *ppv = (uint8_t *)pMap->pv + ((idPage &GMM_PAGEID_IDX_MASK) << PAGE_SHIFT);
1198 return VINF_SUCCESS;
1199#endif
1200}
1201
1202
1203/**
1204 * Maps a page into the current virtual address space so it can be accessed.
1205 *
1206 * @returns VBox status code.
1207 * @retval VINF_SUCCESS on success.
1208 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1209 *
1210 * @param pVM The cross context VM structure.
1211 * @param pPage The physical page tracking structure.
1212 * @param GCPhys The address of the page.
1213 * @param ppMap Where to store the address of the mapping tracking structure.
1214 * @param ppv Where to store the mapping address of the page. The page
1215 * offset is masked off!
1216 *
1217 * @remarks Called from within the PGM critical section.
1218 */
1219static int pgmPhysPageMapCommon(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, PPPGMPAGEMAP ppMap, void **ppv)
1220{
1221 PGM_LOCK_ASSERT_OWNER(pVM);
1222 NOREF(GCPhys);
1223
1224#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1225 /*
1226 * Just some sketchy GC/R0-darwin code.
1227 */
1228 *ppMap = NULL;
1229 RTHCPHYS HCPhys = PGM_PAGE_GET_HCPHYS(pPage);
1230 Assert(HCPhys != pVM->pgm.s.HCPhysZeroPg);
1231 pgmRZDynMapHCPageInlined(VMMGetCpu(pVM), HCPhys, ppv RTLOG_COMMA_SRC_POS);
1232 return VINF_SUCCESS;
1233
1234#else /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1235
1236
1237 /*
1238 * Special cases: MMIO2, ZERO and specially aliased MMIO pages.
1239 */
1240 if ( PGM_PAGE_GET_TYPE(pPage) == PGMPAGETYPE_MMIO2
1241 || PGM_PAGE_GET_TYPE(pPage) == PGMPAGETYPE_MMIO2_ALIAS_MMIO)
1242 {
1243 /* Decode the page id to a page in a MMIO2 ram range. */
1244 uint8_t idMmio2 = PGM_MMIO2_PAGEID_GET_MMIO2_ID(PGM_PAGE_GET_PAGEID(pPage));
1245 uint32_t iPage = PGM_MMIO2_PAGEID_GET_IDX(PGM_PAGE_GET_PAGEID(pPage));
1246 AssertLogRelMsgReturn((uint8_t)(idMmio2 - 1U) < RT_ELEMENTS(pVM->pgm.s.CTX_SUFF(apMmio2Ranges)),
1247 ("idMmio2=%u size=%u type=%u GCPHys=%#RGp Id=%u State=%u", idMmio2,
1248 RT_ELEMENTS(pVM->pgm.s.CTX_SUFF(apMmio2Ranges)), PGM_PAGE_GET_TYPE(pPage), GCPhys,
1249 pPage->s.idPage, pPage->s.uStateY),
1250 VERR_PGM_PHYS_PAGE_MAP_MMIO2_IPE);
1251 PPGMREGMMIO2RANGE pMmio2Range = pVM->pgm.s.CTX_SUFF(apMmio2Ranges)[idMmio2 - 1];
1252 AssertLogRelReturn(pMmio2Range, VERR_PGM_PHYS_PAGE_MAP_MMIO2_IPE);
1253 AssertLogRelReturn(pMmio2Range->idMmio2 == idMmio2, VERR_PGM_PHYS_PAGE_MAP_MMIO2_IPE);
1254 AssertLogRelReturn(iPage < (pMmio2Range->RamRange.cb >> PAGE_SHIFT), VERR_PGM_PHYS_PAGE_MAP_MMIO2_IPE);
1255 *ppv = (uint8_t *)pMmio2Range->RamRange.pvR3 + ((uintptr_t)iPage << PAGE_SHIFT);
1256 *ppMap = NULL;
1257 return VINF_SUCCESS;
1258 }
1259
1260 const uint32_t idChunk = PGM_PAGE_GET_CHUNKID(pPage);
1261 if (idChunk == NIL_GMM_CHUNKID)
1262 {
1263 AssertMsgReturn(PGM_PAGE_GET_PAGEID(pPage) == NIL_GMM_PAGEID, ("pPage=%R[pgmpage]\n", pPage),
1264 VERR_PGM_PHYS_PAGE_MAP_IPE_1);
1265 if (!PGM_PAGE_IS_SPECIAL_ALIAS_MMIO(pPage))
1266 {
1267 AssertMsgReturn(PGM_PAGE_IS_ZERO(pPage), ("pPage=%R[pgmpage]\n", pPage),
1268 VERR_PGM_PHYS_PAGE_MAP_IPE_3);
1269 AssertMsgReturn(PGM_PAGE_GET_HCPHYS(pPage)== pVM->pgm.s.HCPhysZeroPg, ("pPage=%R[pgmpage]\n", pPage),
1270 VERR_PGM_PHYS_PAGE_MAP_IPE_4);
1271 *ppv = pVM->pgm.s.CTXALLSUFF(pvZeroPg);
1272 }
1273 else
1274 *ppv = pVM->pgm.s.CTXALLSUFF(pvZeroPg);
1275 *ppMap = NULL;
1276 return VINF_SUCCESS;
1277 }
1278
1279 /*
1280 * Find/make Chunk TLB entry for the mapping chunk.
1281 */
1282 PPGMCHUNKR3MAP pMap;
1283 PPGMCHUNKR3MAPTLBE pTlbe = &pVM->pgm.s.ChunkR3Map.Tlb.aEntries[PGM_CHUNKR3MAPTLB_IDX(idChunk)];
1284 if (pTlbe->idChunk == idChunk)
1285 {
1286 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,ChunkR3MapTlbHits));
1287 pMap = pTlbe->pChunk;
1288 AssertPtr(pMap->pv);
1289 }
1290 else
1291 {
1292 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,ChunkR3MapTlbMisses));
1293
1294 /*
1295 * Find the chunk, map it if necessary.
1296 */
1297 pMap = (PPGMCHUNKR3MAP)RTAvlU32Get(&pVM->pgm.s.ChunkR3Map.pTree, idChunk);
1298 if (pMap)
1299 {
1300 AssertPtr(pMap->pv);
1301 pMap->iLastUsed = pVM->pgm.s.ChunkR3Map.iNow;
1302 }
1303 else
1304 {
1305# ifdef IN_RING0
1306 int rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_PGM_MAP_CHUNK, idChunk);
1307 AssertRCReturn(rc, rc);
1308 pMap = (PPGMCHUNKR3MAP)RTAvlU32Get(&pVM->pgm.s.ChunkR3Map.pTree, idChunk);
1309 Assert(pMap);
1310# else
1311 int rc = pgmR3PhysChunkMap(pVM, idChunk, &pMap);
1312 if (RT_FAILURE(rc))
1313 return rc;
1314# endif
1315 AssertPtr(pMap->pv);
1316 }
1317
1318 /*
1319 * Enter it into the Chunk TLB.
1320 */
1321 pTlbe->idChunk = idChunk;
1322 pTlbe->pChunk = pMap;
1323 }
1324
1325 *ppv = (uint8_t *)pMap->pv + (PGM_PAGE_GET_PAGE_IN_CHUNK(pPage) << PAGE_SHIFT);
1326 *ppMap = pMap;
1327 return VINF_SUCCESS;
1328#endif /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1329}
1330
1331
1332/**
1333 * Combination of pgmPhysPageMakeWritable and pgmPhysPageMapWritable.
1334 *
1335 * This is typically used is paths where we cannot use the TLB methods (like ROM
1336 * pages) or where there is no point in using them since we won't get many hits.
1337 *
1338 * @returns VBox strict status code.
1339 * @retval VINF_SUCCESS on success.
1340 * @retval VINF_PGM_SYNC_CR3 on success and a page pool flush is pending.
1341 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1342 *
1343 * @param pVM The cross context VM structure.
1344 * @param pPage The physical page tracking structure.
1345 * @param GCPhys The address of the page.
1346 * @param ppv Where to store the mapping address of the page. The page
1347 * offset is masked off!
1348 *
1349 * @remarks Called from within the PGM critical section. The mapping is only
1350 * valid while you are inside section.
1351 */
1352int pgmPhysPageMakeWritableAndMap(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void **ppv)
1353{
1354 int rc = pgmPhysPageMakeWritable(pVM, pPage, GCPhys);
1355 if (RT_SUCCESS(rc))
1356 {
1357 AssertMsg(rc == VINF_SUCCESS || rc == VINF_PGM_SYNC_CR3 /* returned */, ("%Rrc\n", rc));
1358 PPGMPAGEMAP pMapIgnore;
1359 int rc2 = pgmPhysPageMapCommon(pVM, pPage, GCPhys, &pMapIgnore, ppv);
1360 if (RT_FAILURE(rc2)) /* preserve rc */
1361 rc = rc2;
1362 }
1363 return rc;
1364}
1365
1366
1367/**
1368 * Maps a page into the current virtual address space so it can be accessed for
1369 * both writing and reading.
1370 *
1371 * This is typically used is paths where we cannot use the TLB methods (like ROM
1372 * pages) or where there is no point in using them since we won't get many hits.
1373 *
1374 * @returns VBox status code.
1375 * @retval VINF_SUCCESS on success.
1376 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1377 *
1378 * @param pVM The cross context VM structure.
1379 * @param pPage The physical page tracking structure. Must be in the
1380 * allocated state.
1381 * @param GCPhys The address of the page.
1382 * @param ppv Where to store the mapping address of the page. The page
1383 * offset is masked off!
1384 *
1385 * @remarks Called from within the PGM critical section. The mapping is only
1386 * valid while you are inside section.
1387 */
1388int pgmPhysPageMap(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void **ppv)
1389{
1390 Assert(PGM_PAGE_GET_STATE(pPage) == PGM_PAGE_STATE_ALLOCATED);
1391 PPGMPAGEMAP pMapIgnore;
1392 return pgmPhysPageMapCommon(pVM, pPage, GCPhys, &pMapIgnore, ppv);
1393}
1394
1395
1396/**
1397 * Maps a page into the current virtual address space so it can be accessed for
1398 * reading.
1399 *
1400 * This is typically used is paths where we cannot use the TLB methods (like ROM
1401 * pages) or where there is no point in using them since we won't get many hits.
1402 *
1403 * @returns VBox status code.
1404 * @retval VINF_SUCCESS on success.
1405 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1406 *
1407 * @param pVM The cross context VM structure.
1408 * @param pPage The physical page tracking structure.
1409 * @param GCPhys The address of the page.
1410 * @param ppv Where to store the mapping address of the page. The page
1411 * offset is masked off!
1412 *
1413 * @remarks Called from within the PGM critical section. The mapping is only
1414 * valid while you are inside this section.
1415 */
1416int pgmPhysPageMapReadOnly(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void const **ppv)
1417{
1418 PPGMPAGEMAP pMapIgnore;
1419 return pgmPhysPageMapCommon(pVM, pPage, GCPhys, &pMapIgnore, (void **)ppv);
1420}
1421
1422#ifndef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1423
1424/**
1425 * Load a guest page into the ring-3 physical TLB.
1426 *
1427 * @returns VBox status code.
1428 * @retval VINF_SUCCESS on success
1429 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1430 * @param pVM The cross context VM structure.
1431 * @param GCPhys The guest physical address in question.
1432 */
1433int pgmPhysPageLoadIntoTlb(PVMCC pVM, RTGCPHYS GCPhys)
1434{
1435 PGM_LOCK_ASSERT_OWNER(pVM);
1436
1437 /*
1438 * Find the ram range and page and hand it over to the with-page function.
1439 * 99.8% of requests are expected to be in the first range.
1440 */
1441 PPGMPAGE pPage = pgmPhysGetPage(pVM, GCPhys);
1442 if (!pPage)
1443 {
1444 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PageMapTlbMisses));
1445 return VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS;
1446 }
1447
1448 return pgmPhysPageLoadIntoTlbWithPage(pVM, pPage, GCPhys);
1449}
1450
1451
1452/**
1453 * Load a guest page into the ring-3 physical TLB.
1454 *
1455 * @returns VBox status code.
1456 * @retval VINF_SUCCESS on success
1457 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1458 *
1459 * @param pVM The cross context VM structure.
1460 * @param pPage Pointer to the PGMPAGE structure corresponding to
1461 * GCPhys.
1462 * @param GCPhys The guest physical address in question.
1463 */
1464int pgmPhysPageLoadIntoTlbWithPage(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys)
1465{
1466 PGM_LOCK_ASSERT_OWNER(pVM);
1467 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PageMapTlbMisses));
1468
1469 /*
1470 * Map the page.
1471 * Make a special case for the zero page as it is kind of special.
1472 */
1473 PPGMPAGEMAPTLBE pTlbe = &pVM->pgm.s.CTX_SUFF(PhysTlb).aEntries[PGM_PAGEMAPTLB_IDX(GCPhys)];
1474 if ( !PGM_PAGE_IS_ZERO(pPage)
1475 && !PGM_PAGE_IS_BALLOONED(pPage))
1476 {
1477 void *pv;
1478 PPGMPAGEMAP pMap;
1479 int rc = pgmPhysPageMapCommon(pVM, pPage, GCPhys, &pMap, &pv);
1480 if (RT_FAILURE(rc))
1481 return rc;
1482 pTlbe->pMap = pMap;
1483 pTlbe->pv = pv;
1484 Assert(!((uintptr_t)pTlbe->pv & PAGE_OFFSET_MASK));
1485 }
1486 else
1487 {
1488 AssertMsg(PGM_PAGE_GET_HCPHYS(pPage) == pVM->pgm.s.HCPhysZeroPg, ("%RGp/%R[pgmpage]\n", GCPhys, pPage));
1489 pTlbe->pMap = NULL;
1490 pTlbe->pv = pVM->pgm.s.CTXALLSUFF(pvZeroPg);
1491 }
1492# ifdef PGM_WITH_PHYS_TLB
1493 if ( PGM_PAGE_GET_TYPE(pPage) < PGMPAGETYPE_ROM_SHADOW
1494 || PGM_PAGE_GET_TYPE(pPage) > PGMPAGETYPE_ROM)
1495 pTlbe->GCPhys = GCPhys & X86_PTE_PAE_PG_MASK;
1496 else
1497 pTlbe->GCPhys = NIL_RTGCPHYS; /* ROM: Problematic because of the two pages. :-/ */
1498# else
1499 pTlbe->GCPhys = NIL_RTGCPHYS;
1500# endif
1501 pTlbe->pPage = pPage;
1502 return VINF_SUCCESS;
1503}
1504
1505#endif /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1506
1507/**
1508 * Internal version of PGMPhysGCPhys2CCPtr that expects the caller to
1509 * own the PGM lock and therefore not need to lock the mapped page.
1510 *
1511 * @returns VBox status code.
1512 * @retval VINF_SUCCESS on success.
1513 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1514 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1515 *
1516 * @param pVM The cross context VM structure.
1517 * @param GCPhys The guest physical address of the page that should be mapped.
1518 * @param pPage Pointer to the PGMPAGE structure for the page.
1519 * @param ppv Where to store the address corresponding to GCPhys.
1520 *
1521 * @internal
1522 * @deprecated Use pgmPhysGCPhys2CCPtrInternalEx.
1523 */
1524int pgmPhysGCPhys2CCPtrInternalDepr(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void **ppv)
1525{
1526 int rc;
1527 AssertReturn(pPage, VERR_PGM_PHYS_NULL_PAGE_PARAM);
1528 PGM_LOCK_ASSERT_OWNER(pVM);
1529 pVM->pgm.s.cDeprecatedPageLocks++;
1530
1531 /*
1532 * Make sure the page is writable.
1533 */
1534 if (RT_UNLIKELY(PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_ALLOCATED))
1535 {
1536 rc = pgmPhysPageMakeWritable(pVM, pPage, GCPhys);
1537 if (RT_FAILURE(rc))
1538 return rc;
1539 AssertMsg(rc == VINF_SUCCESS || rc == VINF_PGM_SYNC_CR3 /* not returned */, ("%Rrc\n", rc));
1540 }
1541 Assert(PGM_PAGE_GET_HCPHYS(pPage) != 0);
1542
1543 /*
1544 * Get the mapping address.
1545 */
1546#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1547 void *pv;
1548 rc = pgmRZDynMapHCPageInlined(VMMGetCpu(pVM),
1549 PGM_PAGE_GET_HCPHYS(pPage),
1550 &pv
1551 RTLOG_COMMA_SRC_POS);
1552 if (RT_FAILURE(rc))
1553 return rc;
1554 *ppv = (void *)((uintptr_t)pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1555#else
1556 PPGMPAGEMAPTLBE pTlbe;
1557 rc = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
1558 if (RT_FAILURE(rc))
1559 return rc;
1560 *ppv = (void *)((uintptr_t)pTlbe->pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1561#endif
1562 return VINF_SUCCESS;
1563}
1564
1565#ifndef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1566
1567/**
1568 * Locks a page mapping for writing.
1569 *
1570 * @param pVM The cross context VM structure.
1571 * @param pPage The page.
1572 * @param pTlbe The mapping TLB entry for the page.
1573 * @param pLock The lock structure (output).
1574 */
1575DECLINLINE(void) pgmPhysPageMapLockForWriting(PVM pVM, PPGMPAGE pPage, PPGMPAGEMAPTLBE pTlbe, PPGMPAGEMAPLOCK pLock)
1576{
1577 PPGMPAGEMAP pMap = pTlbe->pMap;
1578 if (pMap)
1579 pMap->cRefs++;
1580
1581 unsigned cLocks = PGM_PAGE_GET_WRITE_LOCKS(pPage);
1582 if (RT_LIKELY(cLocks < PGM_PAGE_MAX_LOCKS - 1))
1583 {
1584 if (cLocks == 0)
1585 pVM->pgm.s.cWriteLockedPages++;
1586 PGM_PAGE_INC_WRITE_LOCKS(pPage);
1587 }
1588 else if (cLocks != PGM_PAGE_MAX_LOCKS)
1589 {
1590 PGM_PAGE_INC_WRITE_LOCKS(pPage);
1591 AssertMsgFailed(("%R[pgmpage] is entering permanent write locked state!\n", pPage));
1592 if (pMap)
1593 pMap->cRefs++; /* Extra ref to prevent it from going away. */
1594 }
1595
1596 pLock->uPageAndType = (uintptr_t)pPage | PGMPAGEMAPLOCK_TYPE_WRITE;
1597 pLock->pvMap = pMap;
1598}
1599
1600/**
1601 * Locks a page mapping for reading.
1602 *
1603 * @param pVM The cross context VM structure.
1604 * @param pPage The page.
1605 * @param pTlbe The mapping TLB entry for the page.
1606 * @param pLock The lock structure (output).
1607 */
1608DECLINLINE(void) pgmPhysPageMapLockForReading(PVM pVM, PPGMPAGE pPage, PPGMPAGEMAPTLBE pTlbe, PPGMPAGEMAPLOCK pLock)
1609{
1610 PPGMPAGEMAP pMap = pTlbe->pMap;
1611 if (pMap)
1612 pMap->cRefs++;
1613
1614 unsigned cLocks = PGM_PAGE_GET_READ_LOCKS(pPage);
1615 if (RT_LIKELY(cLocks < PGM_PAGE_MAX_LOCKS - 1))
1616 {
1617 if (cLocks == 0)
1618 pVM->pgm.s.cReadLockedPages++;
1619 PGM_PAGE_INC_READ_LOCKS(pPage);
1620 }
1621 else if (cLocks != PGM_PAGE_MAX_LOCKS)
1622 {
1623 PGM_PAGE_INC_READ_LOCKS(pPage);
1624 AssertMsgFailed(("%R[pgmpage] is entering permanent read locked state!\n", pPage));
1625 if (pMap)
1626 pMap->cRefs++; /* Extra ref to prevent it from going away. */
1627 }
1628
1629 pLock->uPageAndType = (uintptr_t)pPage | PGMPAGEMAPLOCK_TYPE_READ;
1630 pLock->pvMap = pMap;
1631}
1632
1633#endif /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1634
1635
1636/**
1637 * Internal version of PGMPhysGCPhys2CCPtr that expects the caller to
1638 * own the PGM lock and have access to the page structure.
1639 *
1640 * @returns VBox status code.
1641 * @retval VINF_SUCCESS on success.
1642 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1643 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1644 *
1645 * @param pVM The cross context VM structure.
1646 * @param GCPhys The guest physical address of the page that should be mapped.
1647 * @param pPage Pointer to the PGMPAGE structure for the page.
1648 * @param ppv Where to store the address corresponding to GCPhys.
1649 * @param pLock Where to store the lock information that
1650 * pgmPhysReleaseInternalPageMappingLock needs.
1651 *
1652 * @internal
1653 */
1654int pgmPhysGCPhys2CCPtrInternal(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void **ppv, PPGMPAGEMAPLOCK pLock)
1655{
1656 int rc;
1657 AssertReturn(pPage, VERR_PGM_PHYS_NULL_PAGE_PARAM);
1658 PGM_LOCK_ASSERT_OWNER(pVM);
1659
1660 /*
1661 * Make sure the page is writable.
1662 */
1663 if (RT_UNLIKELY(PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_ALLOCATED))
1664 {
1665 rc = pgmPhysPageMakeWritable(pVM, pPage, GCPhys);
1666 if (RT_FAILURE(rc))
1667 return rc;
1668 AssertMsg(rc == VINF_SUCCESS || rc == VINF_PGM_SYNC_CR3 /* not returned */, ("%Rrc\n", rc));
1669 }
1670 Assert(PGM_PAGE_GET_HCPHYS(pPage) != 0);
1671
1672 /*
1673 * Do the job.
1674 */
1675#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1676 void *pv;
1677 PVMCPU pVCpu = VMMGetCpu(pVM);
1678 rc = pgmRZDynMapHCPageInlined(pVCpu,
1679 PGM_PAGE_GET_HCPHYS(pPage),
1680 &pv
1681 RTLOG_COMMA_SRC_POS);
1682 if (RT_FAILURE(rc))
1683 return rc;
1684 *ppv = (void *)((uintptr_t)pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1685 pLock->pvPage = pv;
1686 pLock->pVCpu = pVCpu;
1687
1688#else
1689 PPGMPAGEMAPTLBE pTlbe;
1690 rc = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
1691 if (RT_FAILURE(rc))
1692 return rc;
1693 pgmPhysPageMapLockForWriting(pVM, pPage, pTlbe, pLock);
1694 *ppv = (void *)((uintptr_t)pTlbe->pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1695#endif
1696 return VINF_SUCCESS;
1697}
1698
1699
1700/**
1701 * Internal version of PGMPhysGCPhys2CCPtrReadOnly that expects the caller to
1702 * own the PGM lock and have access to the page structure.
1703 *
1704 * @returns VBox status code.
1705 * @retval VINF_SUCCESS on success.
1706 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1707 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1708 *
1709 * @param pVM The cross context VM structure.
1710 * @param GCPhys The guest physical address of the page that should be mapped.
1711 * @param pPage Pointer to the PGMPAGE structure for the page.
1712 * @param ppv Where to store the address corresponding to GCPhys.
1713 * @param pLock Where to store the lock information that
1714 * pgmPhysReleaseInternalPageMappingLock needs.
1715 *
1716 * @internal
1717 */
1718int pgmPhysGCPhys2CCPtrInternalReadOnly(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, const void **ppv, PPGMPAGEMAPLOCK pLock)
1719{
1720 AssertReturn(pPage, VERR_PGM_PHYS_NULL_PAGE_PARAM);
1721 PGM_LOCK_ASSERT_OWNER(pVM);
1722 Assert(PGM_PAGE_GET_HCPHYS(pPage) != 0);
1723
1724 /*
1725 * Do the job.
1726 */
1727#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1728 void *pv;
1729 PVMCPU pVCpu = VMMGetCpu(pVM);
1730 int rc = pgmRZDynMapHCPageInlined(pVCpu,
1731 PGM_PAGE_GET_HCPHYS(pPage),
1732 &pv
1733 RTLOG_COMMA_SRC_POS); /** @todo add a read only flag? */
1734 if (RT_FAILURE(rc))
1735 return rc;
1736 *ppv = (void *)((uintptr_t)pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1737 pLock->pvPage = pv;
1738 pLock->pVCpu = pVCpu;
1739
1740#else
1741 PPGMPAGEMAPTLBE pTlbe;
1742 int rc = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
1743 if (RT_FAILURE(rc))
1744 return rc;
1745 pgmPhysPageMapLockForReading(pVM, pPage, pTlbe, pLock);
1746 *ppv = (void *)((uintptr_t)pTlbe->pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1747#endif
1748 return VINF_SUCCESS;
1749}
1750
1751
1752/**
1753 * Requests the mapping of a guest page into the current context.
1754 *
1755 * This API should only be used for very short term, as it will consume scarse
1756 * resources (R0 and GC) in the mapping cache. When you're done with the page,
1757 * call PGMPhysReleasePageMappingLock() ASAP to release it.
1758 *
1759 * This API will assume your intention is to write to the page, and will
1760 * therefore replace shared and zero pages. If you do not intend to modify
1761 * the page, use the PGMPhysGCPhys2CCPtrReadOnly() API.
1762 *
1763 * @returns VBox status code.
1764 * @retval VINF_SUCCESS on success.
1765 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1766 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1767 *
1768 * @param pVM The cross context VM structure.
1769 * @param GCPhys The guest physical address of the page that should be
1770 * mapped.
1771 * @param ppv Where to store the address corresponding to GCPhys.
1772 * @param pLock Where to store the lock information that
1773 * PGMPhysReleasePageMappingLock needs.
1774 *
1775 * @remarks The caller is responsible for dealing with access handlers.
1776 * @todo Add an informational return code for pages with access handlers?
1777 *
1778 * @remark Avoid calling this API from within critical sections (other than
1779 * the PGM one) because of the deadlock risk. External threads may
1780 * need to delegate jobs to the EMTs.
1781 * @remarks Only one page is mapped! Make no assumption about what's after or
1782 * before the returned page!
1783 * @thread Any thread.
1784 */
1785VMM_INT_DECL(int) PGMPhysGCPhys2CCPtr(PVMCC pVM, RTGCPHYS GCPhys, void **ppv, PPGMPAGEMAPLOCK pLock)
1786{
1787 int rc = pgmLock(pVM);
1788 AssertRCReturn(rc, rc);
1789
1790#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1791 /*
1792 * Find the page and make sure it's writable.
1793 */
1794 PPGMPAGE pPage;
1795 rc = pgmPhysGetPageEx(pVM, GCPhys, &pPage);
1796 if (RT_SUCCESS(rc))
1797 {
1798 if (RT_UNLIKELY(PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_ALLOCATED))
1799 rc = pgmPhysPageMakeWritable(pVM, pPage, GCPhys);
1800 if (RT_SUCCESS(rc))
1801 {
1802 AssertMsg(rc == VINF_SUCCESS || rc == VINF_PGM_SYNC_CR3 /* not returned */, ("%Rrc\n", rc));
1803
1804 PVMCPU pVCpu = VMMGetCpu(pVM);
1805 void *pv;
1806 rc = pgmRZDynMapHCPageInlined(pVCpu,
1807 PGM_PAGE_GET_HCPHYS(pPage),
1808 &pv
1809 RTLOG_COMMA_SRC_POS);
1810 if (RT_SUCCESS(rc))
1811 {
1812 AssertRCSuccess(rc);
1813
1814 pv = (void *)((uintptr_t)pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1815 *ppv = pv;
1816 pLock->pvPage = pv;
1817 pLock->pVCpu = pVCpu;
1818 }
1819 }
1820 }
1821
1822#else /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1823 /*
1824 * Query the Physical TLB entry for the page (may fail).
1825 */
1826 PPGMPAGEMAPTLBE pTlbe;
1827 rc = pgmPhysPageQueryTlbe(pVM, GCPhys, &pTlbe);
1828 if (RT_SUCCESS(rc))
1829 {
1830 /*
1831 * If the page is shared, the zero page, or being write monitored
1832 * it must be converted to a page that's writable if possible.
1833 */
1834 PPGMPAGE pPage = pTlbe->pPage;
1835 if (RT_UNLIKELY(PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_ALLOCATED))
1836 {
1837 rc = pgmPhysPageMakeWritable(pVM, pPage, GCPhys);
1838 if (RT_SUCCESS(rc))
1839 {
1840 AssertMsg(rc == VINF_SUCCESS || rc == VINF_PGM_SYNC_CR3 /* not returned */, ("%Rrc\n", rc));
1841 rc = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
1842 }
1843 }
1844 if (RT_SUCCESS(rc))
1845 {
1846 /*
1847 * Now, just perform the locking and calculate the return address.
1848 */
1849 pgmPhysPageMapLockForWriting(pVM, pPage, pTlbe, pLock);
1850 *ppv = (void *)((uintptr_t)pTlbe->pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1851 }
1852 }
1853
1854#endif /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1855 pgmUnlock(pVM);
1856 return rc;
1857}
1858
1859
1860/**
1861 * Requests the mapping of a guest page into the current context.
1862 *
1863 * This API should only be used for very short term, as it will consume scarse
1864 * resources (R0 and GC) in the mapping cache. When you're done with the page,
1865 * call PGMPhysReleasePageMappingLock() ASAP to release it.
1866 *
1867 * @returns VBox status code.
1868 * @retval VINF_SUCCESS on success.
1869 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1870 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1871 *
1872 * @param pVM The cross context VM structure.
1873 * @param GCPhys The guest physical address of the page that should be
1874 * mapped.
1875 * @param ppv Where to store the address corresponding to GCPhys.
1876 * @param pLock Where to store the lock information that
1877 * PGMPhysReleasePageMappingLock needs.
1878 *
1879 * @remarks The caller is responsible for dealing with access handlers.
1880 * @todo Add an informational return code for pages with access handlers?
1881 *
1882 * @remarks Avoid calling this API from within critical sections (other than
1883 * the PGM one) because of the deadlock risk.
1884 * @remarks Only one page is mapped! Make no assumption about what's after or
1885 * before the returned page!
1886 * @thread Any thread.
1887 */
1888VMM_INT_DECL(int) PGMPhysGCPhys2CCPtrReadOnly(PVMCC pVM, RTGCPHYS GCPhys, void const **ppv, PPGMPAGEMAPLOCK pLock)
1889{
1890 int rc = pgmLock(pVM);
1891 AssertRCReturn(rc, rc);
1892
1893#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
1894 /*
1895 * Find the page and make sure it's readable.
1896 */
1897 PPGMPAGE pPage;
1898 rc = pgmPhysGetPageEx(pVM, GCPhys, &pPage);
1899 if (RT_SUCCESS(rc))
1900 {
1901 if (RT_UNLIKELY(PGM_PAGE_IS_MMIO_OR_SPECIAL_ALIAS(pPage)))
1902 rc = VERR_PGM_PHYS_PAGE_RESERVED;
1903 else
1904 {
1905 PVMCPU pVCpu = VMMGetCpu(pVM);
1906 void *pv;
1907 rc = pgmRZDynMapHCPageInlined(pVCpu,
1908 PGM_PAGE_GET_HCPHYS(pPage),
1909 &pv
1910 RTLOG_COMMA_SRC_POS); /** @todo add a read only flag? */
1911 if (RT_SUCCESS(rc))
1912 {
1913 AssertRCSuccess(rc);
1914
1915 pv = (void *)((uintptr_t)pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1916 *ppv = pv;
1917 pLock->pvPage = pv;
1918 pLock->pVCpu = pVCpu;
1919 }
1920 }
1921 }
1922
1923#else /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1924 /*
1925 * Query the Physical TLB entry for the page (may fail).
1926 */
1927 PPGMPAGEMAPTLBE pTlbe;
1928 rc = pgmPhysPageQueryTlbe(pVM, GCPhys, &pTlbe);
1929 if (RT_SUCCESS(rc))
1930 {
1931 /* MMIO pages doesn't have any readable backing. */
1932 PPGMPAGE pPage = pTlbe->pPage;
1933 if (RT_UNLIKELY(PGM_PAGE_IS_MMIO_OR_SPECIAL_ALIAS(pPage)))
1934 rc = VERR_PGM_PHYS_PAGE_RESERVED;
1935 else
1936 {
1937 /*
1938 * Now, just perform the locking and calculate the return address.
1939 */
1940 pgmPhysPageMapLockForReading(pVM, pPage, pTlbe, pLock);
1941 *ppv = (void *)((uintptr_t)pTlbe->pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
1942 }
1943 }
1944
1945#endif /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
1946 pgmUnlock(pVM);
1947 return rc;
1948}
1949
1950
1951/**
1952 * Requests the mapping of a guest page given by virtual address into the current context.
1953 *
1954 * This API should only be used for very short term, as it will consume
1955 * scarse resources (R0 and GC) in the mapping cache. When you're done
1956 * with the page, call PGMPhysReleasePageMappingLock() ASAP to release it.
1957 *
1958 * This API will assume your intention is to write to the page, and will
1959 * therefore replace shared and zero pages. If you do not intend to modify
1960 * the page, use the PGMPhysGCPtr2CCPtrReadOnly() API.
1961 *
1962 * @returns VBox status code.
1963 * @retval VINF_SUCCESS on success.
1964 * @retval VERR_PAGE_TABLE_NOT_PRESENT if the page directory for the virtual address isn't present.
1965 * @retval VERR_PAGE_NOT_PRESENT if the page at the virtual address isn't present.
1966 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
1967 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
1968 *
1969 * @param pVCpu The cross context virtual CPU structure.
1970 * @param GCPtr The guest physical address of the page that should be
1971 * mapped.
1972 * @param ppv Where to store the address corresponding to GCPhys.
1973 * @param pLock Where to store the lock information that PGMPhysReleasePageMappingLock needs.
1974 *
1975 * @remark Avoid calling this API from within critical sections (other than
1976 * the PGM one) because of the deadlock risk.
1977 * @thread EMT
1978 */
1979VMM_INT_DECL(int) PGMPhysGCPtr2CCPtr(PVMCPUCC pVCpu, RTGCPTR GCPtr, void **ppv, PPGMPAGEMAPLOCK pLock)
1980{
1981 VM_ASSERT_EMT(pVCpu->CTX_SUFF(pVM));
1982 RTGCPHYS GCPhys;
1983 int rc = PGMPhysGCPtr2GCPhys(pVCpu, GCPtr, &GCPhys);
1984 if (RT_SUCCESS(rc))
1985 rc = PGMPhysGCPhys2CCPtr(pVCpu->CTX_SUFF(pVM), GCPhys, ppv, pLock);
1986 return rc;
1987}
1988
1989
1990/**
1991 * Requests the mapping of a guest page given by virtual address into the current context.
1992 *
1993 * This API should only be used for very short term, as it will consume
1994 * scarse resources (R0 and GC) in the mapping cache. When you're done
1995 * with the page, call PGMPhysReleasePageMappingLock() ASAP to release it.
1996 *
1997 * @returns VBox status code.
1998 * @retval VINF_SUCCESS on success.
1999 * @retval VERR_PAGE_TABLE_NOT_PRESENT if the page directory for the virtual address isn't present.
2000 * @retval VERR_PAGE_NOT_PRESENT if the page at the virtual address isn't present.
2001 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical backing.
2002 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
2003 *
2004 * @param pVCpu The cross context virtual CPU structure.
2005 * @param GCPtr The guest physical address of the page that should be
2006 * mapped.
2007 * @param ppv Where to store the address corresponding to GCPtr.
2008 * @param pLock Where to store the lock information that PGMPhysReleasePageMappingLock needs.
2009 *
2010 * @remark Avoid calling this API from within critical sections (other than
2011 * the PGM one) because of the deadlock risk.
2012 * @thread EMT
2013 */
2014VMM_INT_DECL(int) PGMPhysGCPtr2CCPtrReadOnly(PVMCPUCC pVCpu, RTGCPTR GCPtr, void const **ppv, PPGMPAGEMAPLOCK pLock)
2015{
2016 VM_ASSERT_EMT(pVCpu->CTX_SUFF(pVM));
2017 RTGCPHYS GCPhys;
2018 int rc = PGMPhysGCPtr2GCPhys(pVCpu, GCPtr, &GCPhys);
2019 if (RT_SUCCESS(rc))
2020 rc = PGMPhysGCPhys2CCPtrReadOnly(pVCpu->CTX_SUFF(pVM), GCPhys, ppv, pLock);
2021 return rc;
2022}
2023
2024
2025/**
2026 * Release the mapping of a guest page.
2027 *
2028 * This is the counter part of PGMPhysGCPhys2CCPtr, PGMPhysGCPhys2CCPtrReadOnly
2029 * PGMPhysGCPtr2CCPtr and PGMPhysGCPtr2CCPtrReadOnly.
2030 *
2031 * @param pVM The cross context VM structure.
2032 * @param pLock The lock structure initialized by the mapping function.
2033 */
2034VMMDECL(void) PGMPhysReleasePageMappingLock(PVMCC pVM, PPGMPAGEMAPLOCK pLock)
2035{
2036#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
2037 Assert(pLock->pvPage != NULL);
2038 Assert(pLock->pVCpu == VMMGetCpu(pVM)); RT_NOREF_PV(pVM);
2039 PGM_DYNMAP_UNUSED_HINT(pLock->pVCpu, pLock->pvPage);
2040 pLock->pVCpu = NULL;
2041 pLock->pvPage = NULL;
2042
2043#else
2044 PPGMPAGEMAP pMap = (PPGMPAGEMAP)pLock->pvMap;
2045 PPGMPAGE pPage = (PPGMPAGE)(pLock->uPageAndType & ~PGMPAGEMAPLOCK_TYPE_MASK);
2046 bool fWriteLock = (pLock->uPageAndType & PGMPAGEMAPLOCK_TYPE_MASK) == PGMPAGEMAPLOCK_TYPE_WRITE;
2047
2048 pLock->uPageAndType = 0;
2049 pLock->pvMap = NULL;
2050
2051 pgmLock(pVM);
2052 if (fWriteLock)
2053 {
2054 unsigned cLocks = PGM_PAGE_GET_WRITE_LOCKS(pPage);
2055 Assert(cLocks > 0);
2056 if (RT_LIKELY(cLocks > 0 && cLocks < PGM_PAGE_MAX_LOCKS))
2057 {
2058 if (cLocks == 1)
2059 {
2060 Assert(pVM->pgm.s.cWriteLockedPages > 0);
2061 pVM->pgm.s.cWriteLockedPages--;
2062 }
2063 PGM_PAGE_DEC_WRITE_LOCKS(pPage);
2064 }
2065
2066 if (PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_WRITE_MONITORED)
2067 { /* probably extremely likely */ }
2068 else
2069 pgmPhysPageMakeWriteMonitoredWritable(pVM, pPage, NIL_RTGCPHYS);
2070 }
2071 else
2072 {
2073 unsigned cLocks = PGM_PAGE_GET_READ_LOCKS(pPage);
2074 Assert(cLocks > 0);
2075 if (RT_LIKELY(cLocks > 0 && cLocks < PGM_PAGE_MAX_LOCKS))
2076 {
2077 if (cLocks == 1)
2078 {
2079 Assert(pVM->pgm.s.cReadLockedPages > 0);
2080 pVM->pgm.s.cReadLockedPages--;
2081 }
2082 PGM_PAGE_DEC_READ_LOCKS(pPage);
2083 }
2084 }
2085
2086 if (pMap)
2087 {
2088 Assert(pMap->cRefs >= 1);
2089 pMap->cRefs--;
2090 }
2091 pgmUnlock(pVM);
2092#endif /* !VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
2093}
2094
2095
2096#ifdef IN_RING3
2097/**
2098 * Release the mapping of multiple guest pages.
2099 *
2100 * This is the counter part to PGMR3PhysBulkGCPhys2CCPtrExternal() and
2101 * PGMR3PhysBulkGCPhys2CCPtrReadOnlyExternal().
2102 *
2103 * @param pVM The cross context VM structure.
2104 * @param cPages Number of pages to unlock.
2105 * @param paLocks Array of locks lock structure initialized by the mapping
2106 * function.
2107 */
2108VMMDECL(void) PGMPhysBulkReleasePageMappingLocks(PVMCC pVM, uint32_t cPages, PPGMPAGEMAPLOCK paLocks)
2109{
2110 Assert(cPages > 0);
2111 bool const fWriteLock = (paLocks[0].uPageAndType & PGMPAGEMAPLOCK_TYPE_MASK) == PGMPAGEMAPLOCK_TYPE_WRITE;
2112#ifdef VBOX_STRICT
2113 for (uint32_t i = 1; i < cPages; i++)
2114 {
2115 Assert(fWriteLock == ((paLocks[i].uPageAndType & PGMPAGEMAPLOCK_TYPE_MASK) == PGMPAGEMAPLOCK_TYPE_WRITE));
2116 AssertPtr(paLocks[i].uPageAndType);
2117 }
2118#endif
2119
2120 pgmLock(pVM);
2121 if (fWriteLock)
2122 {
2123 /*
2124 * Write locks:
2125 */
2126 for (uint32_t i = 0; i < cPages; i++)
2127 {
2128 PPGMPAGE pPage = (PPGMPAGE)(paLocks[i].uPageAndType & ~PGMPAGEMAPLOCK_TYPE_MASK);
2129 unsigned cLocks = PGM_PAGE_GET_WRITE_LOCKS(pPage);
2130 Assert(cLocks > 0);
2131 if (RT_LIKELY(cLocks > 0 && cLocks < PGM_PAGE_MAX_LOCKS))
2132 {
2133 if (cLocks == 1)
2134 {
2135 Assert(pVM->pgm.s.cWriteLockedPages > 0);
2136 pVM->pgm.s.cWriteLockedPages--;
2137 }
2138 PGM_PAGE_DEC_WRITE_LOCKS(pPage);
2139 }
2140
2141 if (PGM_PAGE_GET_STATE(pPage) != PGM_PAGE_STATE_WRITE_MONITORED)
2142 { /* probably extremely likely */ }
2143 else
2144 pgmPhysPageMakeWriteMonitoredWritable(pVM, pPage, NIL_RTGCPHYS);
2145
2146 PPGMPAGEMAP pMap = (PPGMPAGEMAP)paLocks[i].pvMap;
2147 if (pMap)
2148 {
2149 Assert(pMap->cRefs >= 1);
2150 pMap->cRefs--;
2151 }
2152
2153 /* Yield the lock: */
2154 if ((i & 1023) == 1023)
2155 {
2156 pgmLock(pVM);
2157 pgmUnlock(pVM);
2158 }
2159 }
2160 }
2161 else
2162 {
2163 /*
2164 * Read locks:
2165 */
2166 for (uint32_t i = 0; i < cPages; i++)
2167 {
2168 PPGMPAGE pPage = (PPGMPAGE)(paLocks[i].uPageAndType & ~PGMPAGEMAPLOCK_TYPE_MASK);
2169 unsigned cLocks = PGM_PAGE_GET_READ_LOCKS(pPage);
2170 Assert(cLocks > 0);
2171 if (RT_LIKELY(cLocks > 0 && cLocks < PGM_PAGE_MAX_LOCKS))
2172 {
2173 if (cLocks == 1)
2174 {
2175 Assert(pVM->pgm.s.cReadLockedPages > 0);
2176 pVM->pgm.s.cReadLockedPages--;
2177 }
2178 PGM_PAGE_DEC_READ_LOCKS(pPage);
2179 }
2180
2181 PPGMPAGEMAP pMap = (PPGMPAGEMAP)paLocks[i].pvMap;
2182 if (pMap)
2183 {
2184 Assert(pMap->cRefs >= 1);
2185 pMap->cRefs--;
2186 }
2187
2188 /* Yield the lock: */
2189 if ((i & 1023) == 1023)
2190 {
2191 pgmLock(pVM);
2192 pgmUnlock(pVM);
2193 }
2194 }
2195 }
2196 pgmUnlock(pVM);
2197
2198 RT_BZERO(paLocks, sizeof(paLocks[0]) * cPages);
2199}
2200#endif /* IN_RING3 */
2201
2202
2203/**
2204 * Release the internal mapping of a guest page.
2205 *
2206 * This is the counter part of pgmPhysGCPhys2CCPtrInternalEx and
2207 * pgmPhysGCPhys2CCPtrInternalReadOnly.
2208 *
2209 * @param pVM The cross context VM structure.
2210 * @param pLock The lock structure initialized by the mapping function.
2211 *
2212 * @remarks Caller must hold the PGM lock.
2213 */
2214void pgmPhysReleaseInternalPageMappingLock(PVMCC pVM, PPGMPAGEMAPLOCK pLock)
2215{
2216 PGM_LOCK_ASSERT_OWNER(pVM);
2217 PGMPhysReleasePageMappingLock(pVM, pLock); /* lazy for now */
2218}
2219
2220
2221/**
2222 * Converts a GC physical address to a HC ring-3 pointer.
2223 *
2224 * @returns VINF_SUCCESS on success.
2225 * @returns VERR_PGM_PHYS_PAGE_RESERVED it it's a valid GC physical
2226 * page but has no physical backing.
2227 * @returns VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid
2228 * GC physical address.
2229 * @returns VERR_PGM_GCPHYS_RANGE_CROSSES_BOUNDARY if the range crosses
2230 * a dynamic ram chunk boundary
2231 *
2232 * @param pVM The cross context VM structure.
2233 * @param GCPhys The GC physical address to convert.
2234 * @param pR3Ptr Where to store the R3 pointer on success.
2235 *
2236 * @deprecated Avoid when possible!
2237 */
2238int pgmPhysGCPhys2R3Ptr(PVMCC pVM, RTGCPHYS GCPhys, PRTR3PTR pR3Ptr)
2239{
2240/** @todo this is kind of hacky and needs some more work. */
2241#ifndef DEBUG_sandervl
2242 VM_ASSERT_EMT(pVM); /* no longer safe for use outside the EMT thread! */
2243#endif
2244
2245 Log(("pgmPhysGCPhys2R3Ptr(,%RGp,): dont use this API!\n", GCPhys)); /** @todo eliminate this API! */
2246#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
2247 NOREF(pVM); NOREF(pR3Ptr); RT_NOREF_PV(GCPhys);
2248 AssertFailedReturn(VERR_NOT_IMPLEMENTED);
2249#else
2250 pgmLock(pVM);
2251
2252 PPGMRAMRANGE pRam;
2253 PPGMPAGE pPage;
2254 int rc = pgmPhysGetPageAndRangeEx(pVM, GCPhys, &pPage, &pRam);
2255 if (RT_SUCCESS(rc))
2256 rc = pgmPhysGCPhys2CCPtrInternalDepr(pVM, pPage, GCPhys, (void **)pR3Ptr);
2257
2258 pgmUnlock(pVM);
2259 Assert(rc <= VINF_SUCCESS);
2260 return rc;
2261#endif
2262}
2263
2264#if 0 /*def VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 */
2265
2266/**
2267 * Maps and locks a guest CR3 or PD (PAE) page.
2268 *
2269 * @returns VINF_SUCCESS on success.
2270 * @returns VERR_PGM_PHYS_PAGE_RESERVED it it's a valid GC physical
2271 * page but has no physical backing.
2272 * @returns VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid
2273 * GC physical address.
2274 * @returns VERR_PGM_GCPHYS_RANGE_CROSSES_BOUNDARY if the range crosses
2275 * a dynamic ram chunk boundary
2276 *
2277 * @param pVM The cross context VM structure.
2278 * @param GCPhys The GC physical address to convert.
2279 * @param pR3Ptr Where to store the R3 pointer on success. This may or
2280 * may not be valid in ring-0 depending on the
2281 * VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0 build option.
2282 *
2283 * @remarks The caller must own the PGM lock.
2284 */
2285int pgmPhysCr3ToHCPtr(PVM pVM, RTGCPHYS GCPhys, PRTR3PTR pR3Ptr)
2286{
2287
2288 PPGMRAMRANGE pRam;
2289 PPGMPAGE pPage;
2290 int rc = pgmPhysGetPageAndRangeEx(pVM, GCPhys, &pPage, &pRam);
2291 if (RT_SUCCESS(rc))
2292 rc = pgmPhysGCPhys2CCPtrInternalDepr(pVM, pPage, GCPhys, (void **)pR3Ptr);
2293 Assert(rc <= VINF_SUCCESS);
2294 return rc;
2295}
2296
2297
2298int pgmPhysCr3ToHCPtr(PVM pVM, RTGCPHYS GCPhys, PRTR3PTR pR3Ptr)
2299{
2300
2301}
2302
2303#endif
2304
2305/**
2306 * Converts a guest pointer to a GC physical address.
2307 *
2308 * This uses the current CR3/CR0/CR4 of the guest.
2309 *
2310 * @returns VBox status code.
2311 * @param pVCpu The cross context virtual CPU structure.
2312 * @param GCPtr The guest pointer to convert.
2313 * @param pGCPhys Where to store the GC physical address.
2314 */
2315VMMDECL(int) PGMPhysGCPtr2GCPhys(PVMCPUCC pVCpu, RTGCPTR GCPtr, PRTGCPHYS pGCPhys)
2316{
2317 int rc = PGMGstGetPage(pVCpu, (RTGCUINTPTR)GCPtr, NULL, pGCPhys);
2318 if (pGCPhys && RT_SUCCESS(rc))
2319 *pGCPhys |= (RTGCUINTPTR)GCPtr & PAGE_OFFSET_MASK;
2320 return rc;
2321}
2322
2323
2324/**
2325 * Converts a guest pointer to a HC physical address.
2326 *
2327 * This uses the current CR3/CR0/CR4 of the guest.
2328 *
2329 * @returns VBox status code.
2330 * @param pVCpu The cross context virtual CPU structure.
2331 * @param GCPtr The guest pointer to convert.
2332 * @param pHCPhys Where to store the HC physical address.
2333 */
2334VMM_INT_DECL(int) PGMPhysGCPtr2HCPhys(PVMCPUCC pVCpu, RTGCPTR GCPtr, PRTHCPHYS pHCPhys)
2335{
2336 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
2337 RTGCPHYS GCPhys;
2338 int rc = PGMGstGetPage(pVCpu, (RTGCUINTPTR)GCPtr, NULL, &GCPhys);
2339 if (RT_SUCCESS(rc))
2340 rc = PGMPhysGCPhys2HCPhys(pVM, GCPhys | ((RTGCUINTPTR)GCPtr & PAGE_OFFSET_MASK), pHCPhys);
2341 return rc;
2342}
2343
2344
2345
2346#undef LOG_GROUP
2347#define LOG_GROUP LOG_GROUP_PGM_PHYS_ACCESS
2348
2349
2350#if defined(IN_RING3) && defined(SOME_UNUSED_FUNCTION)
2351/**
2352 * Cache PGMPhys memory access
2353 *
2354 * @param pVM The cross context VM structure.
2355 * @param pCache Cache structure pointer
2356 * @param GCPhys GC physical address
2357 * @param pbHC HC pointer corresponding to physical page
2358 *
2359 * @thread EMT.
2360 */
2361static void pgmPhysCacheAdd(PVM pVM, PGMPHYSCACHE *pCache, RTGCPHYS GCPhys, uint8_t *pbR3)
2362{
2363 uint32_t iCacheIndex;
2364
2365 Assert(VM_IS_EMT(pVM));
2366
2367 GCPhys = PHYS_PAGE_ADDRESS(GCPhys);
2368 pbR3 = (uint8_t *)PAGE_ADDRESS(pbR3);
2369
2370 iCacheIndex = ((GCPhys >> PAGE_SHIFT) & PGM_MAX_PHYSCACHE_ENTRIES_MASK);
2371
2372 ASMBitSet(&pCache->aEntries, iCacheIndex);
2373
2374 pCache->Entry[iCacheIndex].GCPhys = GCPhys;
2375 pCache->Entry[iCacheIndex].pbR3 = pbR3;
2376}
2377#endif /* IN_RING3 */
2378
2379
2380/**
2381 * Deals with reading from a page with one or more ALL access handlers.
2382 *
2383 * @returns Strict VBox status code in ring-0 and raw-mode, ignorable in ring-3.
2384 * See PGM_HANDLER_PHYS_IS_VALID_STATUS and
2385 * PGM_HANDLER_VIRT_IS_VALID_STATUS for details.
2386 *
2387 * @param pVM The cross context VM structure.
2388 * @param pPage The page descriptor.
2389 * @param GCPhys The physical address to start reading at.
2390 * @param pvBuf Where to put the bits we read.
2391 * @param cb How much to read - less or equal to a page.
2392 * @param enmOrigin The origin of this call.
2393 */
2394static VBOXSTRICTRC pgmPhysReadHandler(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void *pvBuf, size_t cb,
2395 PGMACCESSORIGIN enmOrigin)
2396{
2397 /*
2398 * The most frequent access here is MMIO and shadowed ROM.
2399 * The current code ASSUMES all these access handlers covers full pages!
2400 */
2401
2402 /*
2403 * Whatever we do we need the source page, map it first.
2404 */
2405 PGMPAGEMAPLOCK PgMpLck;
2406 const void *pvSrc = NULL;
2407 int rc = pgmPhysGCPhys2CCPtrInternalReadOnly(pVM, pPage, GCPhys, &pvSrc, &PgMpLck);
2408/** @todo Check how this can work for MMIO pages? */
2409 if (RT_FAILURE(rc))
2410 {
2411 AssertLogRelMsgFailed(("pgmPhysGCPhys2CCPtrInternalReadOnly failed on %RGp / %R[pgmpage] -> %Rrc\n",
2412 GCPhys, pPage, rc));
2413 memset(pvBuf, 0xff, cb);
2414 return VINF_SUCCESS;
2415 }
2416
2417 VBOXSTRICTRC rcStrict = VINF_PGM_HANDLER_DO_DEFAULT;
2418
2419 /*
2420 * Deal with any physical handlers.
2421 */
2422 PVMCPUCC pVCpu = VMMGetCpu(pVM);
2423 PPGMPHYSHANDLER pPhys = NULL;
2424 if ( PGM_PAGE_GET_HNDL_PHYS_STATE(pPage) == PGM_PAGE_HNDL_PHYS_STATE_ALL
2425 || PGM_PAGE_IS_MMIO_OR_SPECIAL_ALIAS(pPage))
2426 {
2427 pPhys = pgmHandlerPhysicalLookup(pVM, GCPhys);
2428 AssertReleaseMsg(pPhys, ("GCPhys=%RGp cb=%#x\n", GCPhys, cb));
2429 Assert(GCPhys >= pPhys->Core.Key && GCPhys <= pPhys->Core.KeyLast);
2430 Assert((pPhys->Core.Key & PAGE_OFFSET_MASK) == 0);
2431 Assert((pPhys->Core.KeyLast & PAGE_OFFSET_MASK) == PAGE_OFFSET_MASK);
2432#ifndef IN_RING3
2433 if (enmOrigin != PGMACCESSORIGIN_IEM)
2434 {
2435 /* Cannot reliably handle informational status codes in this context */
2436 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2437 return VERR_PGM_PHYS_WR_HIT_HANDLER;
2438 }
2439#endif
2440 PFNPGMPHYSHANDLER pfnHandler = PGMPHYSHANDLER_GET_TYPE(pVM, pPhys)->CTX_SUFF(pfnHandler); Assert(pfnHandler);
2441 void *pvUser = pPhys->CTX_SUFF(pvUser);
2442
2443 Log5(("pgmPhysReadHandler: GCPhys=%RGp cb=%#x pPage=%R[pgmpage] phys %s\n", GCPhys, cb, pPage, R3STRING(pPhys->pszDesc) ));
2444 STAM_PROFILE_START(&pPhys->Stat, h);
2445 PGM_LOCK_ASSERT_OWNER(pVM);
2446
2447 /* Release the PGM lock as MMIO handlers take the IOM lock. (deadlock prevention) */
2448 pgmUnlock(pVM);
2449 rcStrict = pfnHandler(pVM, pVCpu, GCPhys, (void *)pvSrc, pvBuf, cb, PGMACCESSTYPE_READ, enmOrigin, pvUser);
2450 pgmLock(pVM);
2451
2452#ifdef VBOX_WITH_STATISTICS
2453 pPhys = pgmHandlerPhysicalLookup(pVM, GCPhys);
2454 if (pPhys)
2455 STAM_PROFILE_STOP(&pPhys->Stat, h);
2456#else
2457 pPhys = NULL; /* might not be valid anymore. */
2458#endif
2459 AssertLogRelMsg(PGM_HANDLER_PHYS_IS_VALID_STATUS(rcStrict, false),
2460 ("rcStrict=%Rrc GCPhys=%RGp\n", VBOXSTRICTRC_VAL(rcStrict), GCPhys));
2461 if ( rcStrict != VINF_PGM_HANDLER_DO_DEFAULT
2462 && !PGM_PHYS_RW_IS_SUCCESS(rcStrict))
2463 {
2464 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2465 return rcStrict;
2466 }
2467 }
2468
2469 /*
2470 * Take the default action.
2471 */
2472 if (rcStrict == VINF_PGM_HANDLER_DO_DEFAULT)
2473 {
2474 memcpy(pvBuf, pvSrc, cb);
2475 rcStrict = VINF_SUCCESS;
2476 }
2477 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2478 return rcStrict;
2479}
2480
2481
2482/**
2483 * Read physical memory.
2484 *
2485 * This API respects access handlers and MMIO. Use PGMPhysSimpleReadGCPhys() if you
2486 * want to ignore those.
2487 *
2488 * @returns Strict VBox status code in raw-mode and ring-0, normal VBox status
2489 * code in ring-3. Use PGM_PHYS_RW_IS_SUCCESS to check.
2490 * @retval VINF_SUCCESS in all context - read completed.
2491 *
2492 * @retval VINF_EM_OFF in RC and R0 - read completed.
2493 * @retval VINF_EM_SUSPEND in RC and R0 - read completed.
2494 * @retval VINF_EM_RESET in RC and R0 - read completed.
2495 * @retval VINF_EM_HALT in RC and R0 - read completed.
2496 * @retval VINF_SELM_SYNC_GDT in RC only - read completed.
2497 *
2498 * @retval VINF_EM_DBG_STOP in RC and R0 - read completed.
2499 * @retval VINF_EM_DBG_BREAKPOINT in RC and R0 - read completed.
2500 * @retval VINF_EM_RAW_EMULATE_INSTR in RC and R0 only.
2501 *
2502 * @retval VINF_IOM_R3_MMIO_READ in RC and R0.
2503 * @retval VINF_IOM_R3_MMIO_READ_WRITE in RC and R0.
2504 *
2505 * @retval VINF_PATM_CHECK_PATCH_PAGE in RC only.
2506 *
2507 * @retval VERR_PGM_PHYS_WR_HIT_HANDLER in RC and R0 for access origins that
2508 * haven't been cleared for strict status codes yet.
2509 *
2510 * @param pVM The cross context VM structure.
2511 * @param GCPhys Physical address start reading from.
2512 * @param pvBuf Where to put the read bits.
2513 * @param cbRead How many bytes to read.
2514 * @param enmOrigin The origin of this call.
2515 */
2516VMMDECL(VBOXSTRICTRC) PGMPhysRead(PVMCC pVM, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead, PGMACCESSORIGIN enmOrigin)
2517{
2518 AssertMsgReturn(cbRead > 0, ("don't even think about reading zero bytes!\n"), VINF_SUCCESS);
2519 LogFlow(("PGMPhysRead: %RGp %d\n", GCPhys, cbRead));
2520
2521 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysRead));
2522 STAM_COUNTER_ADD(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysReadBytes), cbRead);
2523
2524 pgmLock(pVM);
2525
2526 /*
2527 * Copy loop on ram ranges.
2528 */
2529 VBOXSTRICTRC rcStrict = VINF_SUCCESS;
2530 PPGMRAMRANGE pRam = pgmPhysGetRangeAtOrAbove(pVM, GCPhys);
2531 for (;;)
2532 {
2533 /* Inside range or not? */
2534 if (pRam && GCPhys >= pRam->GCPhys)
2535 {
2536 /*
2537 * Must work our way thru this page by page.
2538 */
2539 RTGCPHYS off = GCPhys - pRam->GCPhys;
2540 while (off < pRam->cb)
2541 {
2542 unsigned iPage = off >> PAGE_SHIFT;
2543 PPGMPAGE pPage = &pRam->aPages[iPage];
2544 size_t cb = PAGE_SIZE - (off & PAGE_OFFSET_MASK);
2545 if (cb > cbRead)
2546 cb = cbRead;
2547
2548 /*
2549 * Normal page? Get the pointer to it.
2550 */
2551 if ( !PGM_PAGE_HAS_ACTIVE_ALL_HANDLERS(pPage)
2552 && !PGM_PAGE_IS_SPECIAL_ALIAS_MMIO(pPage))
2553 {
2554 /*
2555 * Get the pointer to the page.
2556 */
2557 PGMPAGEMAPLOCK PgMpLck;
2558 const void *pvSrc;
2559 int rc = pgmPhysGCPhys2CCPtrInternalReadOnly(pVM, pPage, pRam->GCPhys + off, &pvSrc, &PgMpLck);
2560 if (RT_SUCCESS(rc))
2561 {
2562 memcpy(pvBuf, pvSrc, cb);
2563 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2564 }
2565 else
2566 {
2567 AssertLogRelMsgFailed(("pgmPhysGCPhys2CCPtrInternalReadOnly failed on %RGp / %R[pgmpage] -> %Rrc\n",
2568 pRam->GCPhys + off, pPage, rc));
2569 memset(pvBuf, 0xff, cb);
2570 }
2571 }
2572 /*
2573 * Have ALL/MMIO access handlers.
2574 */
2575 else
2576 {
2577 VBOXSTRICTRC rcStrict2 = pgmPhysReadHandler(pVM, pPage, pRam->GCPhys + off, pvBuf, cb, enmOrigin);
2578 if (PGM_PHYS_RW_IS_SUCCESS(rcStrict2))
2579 PGM_PHYS_RW_DO_UPDATE_STRICT_RC(rcStrict, rcStrict2);
2580 else
2581 {
2582 memset(pvBuf, 0xff, cb);
2583 pgmUnlock(pVM);
2584 return rcStrict2;
2585 }
2586 }
2587
2588 /* next page */
2589 if (cb >= cbRead)
2590 {
2591 pgmUnlock(pVM);
2592 return rcStrict;
2593 }
2594 cbRead -= cb;
2595 off += cb;
2596 pvBuf = (char *)pvBuf + cb;
2597 } /* walk pages in ram range. */
2598
2599 GCPhys = pRam->GCPhysLast + 1;
2600 }
2601 else
2602 {
2603 LogFlow(("PGMPhysRead: Unassigned %RGp size=%u\n", GCPhys, cbRead));
2604
2605 /*
2606 * Unassigned address space.
2607 */
2608 size_t cb = pRam ? pRam->GCPhys - GCPhys : ~(size_t)0;
2609 if (cb >= cbRead)
2610 {
2611 memset(pvBuf, 0xff, cbRead);
2612 break;
2613 }
2614 memset(pvBuf, 0xff, cb);
2615
2616 cbRead -= cb;
2617 pvBuf = (char *)pvBuf + cb;
2618 GCPhys += cb;
2619 }
2620
2621 /* Advance range if necessary. */
2622 while (pRam && GCPhys > pRam->GCPhysLast)
2623 pRam = pRam->CTX_SUFF(pNext);
2624 } /* Ram range walk */
2625
2626 pgmUnlock(pVM);
2627 return rcStrict;
2628}
2629
2630
2631/**
2632 * Deals with writing to a page with one or more WRITE or ALL access handlers.
2633 *
2634 * @returns Strict VBox status code in ring-0 and raw-mode, ignorable in ring-3.
2635 * See PGM_HANDLER_PHYS_IS_VALID_STATUS and
2636 * PGM_HANDLER_VIRT_IS_VALID_STATUS for details.
2637 *
2638 * @param pVM The cross context VM structure.
2639 * @param pPage The page descriptor.
2640 * @param GCPhys The physical address to start writing at.
2641 * @param pvBuf What to write.
2642 * @param cbWrite How much to write - less or equal to a page.
2643 * @param enmOrigin The origin of this call.
2644 */
2645static VBOXSTRICTRC pgmPhysWriteHandler(PVMCC pVM, PPGMPAGE pPage, RTGCPHYS GCPhys, void const *pvBuf, size_t cbWrite,
2646 PGMACCESSORIGIN enmOrigin)
2647{
2648 PGMPAGEMAPLOCK PgMpLck;
2649 void *pvDst = NULL;
2650 VBOXSTRICTRC rcStrict;
2651
2652 /*
2653 * Give priority to physical handlers (like #PF does).
2654 *
2655 * Hope for a lonely physical handler first that covers the whole
2656 * write area. This should be a pretty frequent case with MMIO and
2657 * the heavy usage of full page handlers in the page pool.
2658 */
2659 PVMCPUCC pVCpu = VMMGetCpu(pVM);
2660 PPGMPHYSHANDLER pCur = pgmHandlerPhysicalLookup(pVM, GCPhys);
2661 if (pCur)
2662 {
2663 Assert(GCPhys >= pCur->Core.Key && GCPhys <= pCur->Core.KeyLast);
2664#ifndef IN_RING3
2665 if (enmOrigin != PGMACCESSORIGIN_IEM)
2666 /* Cannot reliably handle informational status codes in this context */
2667 return VERR_PGM_PHYS_WR_HIT_HANDLER;
2668#endif
2669 size_t cbRange = pCur->Core.KeyLast - GCPhys + 1;
2670 if (cbRange > cbWrite)
2671 cbRange = cbWrite;
2672
2673 Assert(PGMPHYSHANDLER_GET_TYPE(pVM, pCur)->CTX_SUFF(pfnHandler));
2674 Log5(("pgmPhysWriteHandler: GCPhys=%RGp cbRange=%#x pPage=%R[pgmpage] phys %s\n",
2675 GCPhys, cbRange, pPage, R3STRING(pCur->pszDesc) ));
2676 if (!PGM_PAGE_IS_MMIO_OR_SPECIAL_ALIAS(pPage))
2677 rcStrict = pgmPhysGCPhys2CCPtrInternal(pVM, pPage, GCPhys, &pvDst, &PgMpLck);
2678 else
2679 rcStrict = VINF_SUCCESS;
2680 if (RT_SUCCESS(rcStrict))
2681 {
2682 PFNPGMPHYSHANDLER pfnHandler = PGMPHYSHANDLER_GET_TYPE(pVM, pCur)->CTX_SUFF(pfnHandler);
2683 void *pvUser = pCur->CTX_SUFF(pvUser);
2684 STAM_PROFILE_START(&pCur->Stat, h);
2685
2686 /* Release the PGM lock as MMIO handlers take the IOM lock. (deadlock prevention) */
2687 PGM_LOCK_ASSERT_OWNER(pVM);
2688 pgmUnlock(pVM);
2689 rcStrict = pfnHandler(pVM, pVCpu, GCPhys, pvDst, (void *)pvBuf, cbRange, PGMACCESSTYPE_WRITE, enmOrigin, pvUser);
2690 pgmLock(pVM);
2691
2692#ifdef VBOX_WITH_STATISTICS
2693 pCur = pgmHandlerPhysicalLookup(pVM, GCPhys);
2694 if (pCur)
2695 STAM_PROFILE_STOP(&pCur->Stat, h);
2696#else
2697 pCur = NULL; /* might not be valid anymore. */
2698#endif
2699 if (rcStrict == VINF_PGM_HANDLER_DO_DEFAULT)
2700 {
2701 if (pvDst)
2702 memcpy(pvDst, pvBuf, cbRange);
2703 rcStrict = VINF_SUCCESS;
2704 }
2705 else
2706 AssertLogRelMsg(PGM_HANDLER_PHYS_IS_VALID_STATUS(rcStrict, true),
2707 ("rcStrict=%Rrc GCPhys=%RGp pPage=%R[pgmpage] %s\n",
2708 VBOXSTRICTRC_VAL(rcStrict), GCPhys, pPage, pCur ? R3STRING(pCur->pszDesc) : ""));
2709 }
2710 else
2711 AssertLogRelMsgFailedReturn(("pgmPhysGCPhys2CCPtrInternal failed on %RGp / %R[pgmpage] -> %Rrc\n",
2712 GCPhys, pPage, VBOXSTRICTRC_VAL(rcStrict)), rcStrict);
2713 if (RT_LIKELY(cbRange == cbWrite) || !PGM_PHYS_RW_IS_SUCCESS(rcStrict))
2714 {
2715 if (pvDst)
2716 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2717 return rcStrict;
2718 }
2719
2720 /* more fun to be had below */
2721 cbWrite -= cbRange;
2722 GCPhys += cbRange;
2723 pvBuf = (uint8_t *)pvBuf + cbRange;
2724 pvDst = (uint8_t *)pvDst + cbRange;
2725 }
2726 else /* The handler is somewhere else in the page, deal with it below. */
2727 rcStrict = VINF_SUCCESS;
2728 Assert(!PGM_PAGE_IS_MMIO_OR_ALIAS(pPage)); /* MMIO handlers are all PAGE_SIZEed! */
2729
2730 /*
2731 * Deal with all the odd ends (used to be deal with virt+phys).
2732 */
2733 Assert(rcStrict != VINF_PGM_HANDLER_DO_DEFAULT);
2734
2735 /* We need a writable destination page. */
2736 if (!pvDst)
2737 {
2738 int rc2 = pgmPhysGCPhys2CCPtrInternal(pVM, pPage, GCPhys, &pvDst, &PgMpLck);
2739 AssertLogRelMsgReturn(RT_SUCCESS(rc2),
2740 ("pgmPhysGCPhys2CCPtrInternal failed on %RGp / %R[pgmpage] -> %Rrc\n", GCPhys, pPage, rc2),
2741 rc2);
2742 }
2743
2744 /* The loop state (big + ugly). */
2745 PPGMPHYSHANDLER pPhys = NULL;
2746 uint32_t offPhys = PAGE_SIZE;
2747 uint32_t offPhysLast = PAGE_SIZE;
2748 bool fMorePhys = PGM_PAGE_HAS_ACTIVE_PHYSICAL_HANDLERS(pPage);
2749
2750 /* The loop. */
2751 for (;;)
2752 {
2753 if (fMorePhys && !pPhys)
2754 {
2755 pPhys = pgmHandlerPhysicalLookup(pVM, GCPhys);
2756 if (pPhys)
2757 {
2758 offPhys = 0;
2759 offPhysLast = pPhys->Core.KeyLast - GCPhys; /* ASSUMES < 4GB handlers... */
2760 }
2761 else
2762 {
2763 pPhys = (PPGMPHYSHANDLER)RTAvlroGCPhysGetBestFit(&pVM->pgm.s.CTX_SUFF(pTrees)->PhysHandlers,
2764 GCPhys, true /* fAbove */);
2765 if ( pPhys
2766 && pPhys->Core.Key <= GCPhys + (cbWrite - 1))
2767 {
2768 offPhys = pPhys->Core.Key - GCPhys;
2769 offPhysLast = pPhys->Core.KeyLast - GCPhys; /* ASSUMES < 4GB handlers... */
2770 }
2771 else
2772 {
2773 pPhys = NULL;
2774 fMorePhys = false;
2775 offPhys = offPhysLast = PAGE_SIZE;
2776 }
2777 }
2778 }
2779
2780 /*
2781 * Handle access to space without handlers (that's easy).
2782 */
2783 VBOXSTRICTRC rcStrict2 = VINF_PGM_HANDLER_DO_DEFAULT;
2784 uint32_t cbRange = (uint32_t)cbWrite;
2785
2786 /*
2787 * Physical handler.
2788 */
2789 if (!offPhys)
2790 {
2791#ifndef IN_RING3
2792 if (enmOrigin != PGMACCESSORIGIN_IEM)
2793 /* Cannot reliably handle informational status codes in this context */
2794 return VERR_PGM_PHYS_WR_HIT_HANDLER;
2795#endif
2796 if (cbRange > offPhysLast + 1)
2797 cbRange = offPhysLast + 1;
2798
2799 PFNPGMPHYSHANDLER pfnHandler = PGMPHYSHANDLER_GET_TYPE(pVM, pPhys)->CTX_SUFF(pfnHandler);
2800 void *pvUser = pPhys->CTX_SUFF(pvUser);
2801
2802 Log5(("pgmPhysWriteHandler: GCPhys=%RGp cbRange=%#x pPage=%R[pgmpage] phys %s\n", GCPhys, cbRange, pPage, R3STRING(pPhys->pszDesc) ));
2803 STAM_PROFILE_START(&pPhys->Stat, h);
2804
2805 /* Release the PGM lock as MMIO handlers take the IOM lock. (deadlock prevention) */
2806 PGM_LOCK_ASSERT_OWNER(pVM);
2807 pgmUnlock(pVM);
2808 rcStrict2 = pfnHandler(pVM, pVCpu, GCPhys, pvDst, (void *)pvBuf, cbRange, PGMACCESSTYPE_WRITE, enmOrigin, pvUser);
2809 pgmLock(pVM);
2810
2811#ifdef VBOX_WITH_STATISTICS
2812 pPhys = pgmHandlerPhysicalLookup(pVM, GCPhys);
2813 if (pPhys)
2814 STAM_PROFILE_STOP(&pPhys->Stat, h);
2815#else
2816 pPhys = NULL; /* might not be valid anymore. */
2817#endif
2818 AssertLogRelMsg(PGM_HANDLER_PHYS_IS_VALID_STATUS(rcStrict2, true),
2819 ("rcStrict2=%Rrc (rcStrict=%Rrc) GCPhys=%RGp pPage=%R[pgmpage] %s\n", VBOXSTRICTRC_VAL(rcStrict2),
2820 VBOXSTRICTRC_VAL(rcStrict), GCPhys, pPage, pPhys ? R3STRING(pPhys->pszDesc) : ""));
2821 }
2822
2823 /*
2824 * Execute the default action and merge the status codes.
2825 */
2826 if (rcStrict2 == VINF_PGM_HANDLER_DO_DEFAULT)
2827 {
2828 memcpy(pvDst, pvBuf, cbRange);
2829 rcStrict2 = VINF_SUCCESS;
2830 }
2831 else if (!PGM_PHYS_RW_IS_SUCCESS(rcStrict2))
2832 {
2833 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2834 return rcStrict2;
2835 }
2836 else
2837 PGM_PHYS_RW_DO_UPDATE_STRICT_RC(rcStrict, rcStrict2);
2838
2839 /*
2840 * Advance if we've got more stuff to do.
2841 */
2842 if (cbRange >= cbWrite)
2843 {
2844 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2845 return rcStrict;
2846 }
2847
2848
2849 cbWrite -= cbRange;
2850 GCPhys += cbRange;
2851 pvBuf = (uint8_t *)pvBuf + cbRange;
2852 pvDst = (uint8_t *)pvDst + cbRange;
2853
2854 offPhys -= cbRange;
2855 offPhysLast -= cbRange;
2856 }
2857}
2858
2859
2860/**
2861 * Write to physical memory.
2862 *
2863 * This API respects access handlers and MMIO. Use PGMPhysSimpleWriteGCPhys() if you
2864 * want to ignore those.
2865 *
2866 * @returns Strict VBox status code in raw-mode and ring-0, normal VBox status
2867 * code in ring-3. Use PGM_PHYS_RW_IS_SUCCESS to check.
2868 * @retval VINF_SUCCESS in all context - write completed.
2869 *
2870 * @retval VINF_EM_OFF in RC and R0 - write completed.
2871 * @retval VINF_EM_SUSPEND in RC and R0 - write completed.
2872 * @retval VINF_EM_RESET in RC and R0 - write completed.
2873 * @retval VINF_EM_HALT in RC and R0 - write completed.
2874 * @retval VINF_SELM_SYNC_GDT in RC only - write completed.
2875 *
2876 * @retval VINF_EM_DBG_STOP in RC and R0 - write completed.
2877 * @retval VINF_EM_DBG_BREAKPOINT in RC and R0 - write completed.
2878 * @retval VINF_EM_RAW_EMULATE_INSTR in RC and R0 only.
2879 *
2880 * @retval VINF_IOM_R3_MMIO_WRITE in RC and R0.
2881 * @retval VINF_IOM_R3_MMIO_READ_WRITE in RC and R0.
2882 * @retval VINF_IOM_R3_MMIO_COMMIT_WRITE in RC and R0.
2883 *
2884 * @retval VINF_EM_RAW_EMULATE_INSTR_GDT_FAULT in RC only - write completed.
2885 * @retval VINF_EM_RAW_EMULATE_INSTR_LDT_FAULT in RC only.
2886 * @retval VINF_EM_RAW_EMULATE_INSTR_TSS_FAULT in RC only.
2887 * @retval VINF_EM_RAW_EMULATE_INSTR_IDT_FAULT in RC only.
2888 * @retval VINF_CSAM_PENDING_ACTION in RC only.
2889 * @retval VINF_PATM_CHECK_PATCH_PAGE in RC only.
2890 *
2891 * @retval VERR_PGM_PHYS_WR_HIT_HANDLER in RC and R0 for access origins that
2892 * haven't been cleared for strict status codes yet.
2893 *
2894 *
2895 * @param pVM The cross context VM structure.
2896 * @param GCPhys Physical address to write to.
2897 * @param pvBuf What to write.
2898 * @param cbWrite How many bytes to write.
2899 * @param enmOrigin Who is calling.
2900 */
2901VMMDECL(VBOXSTRICTRC) PGMPhysWrite(PVMCC pVM, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite, PGMACCESSORIGIN enmOrigin)
2902{
2903 AssertMsg(!pVM->pgm.s.fNoMorePhysWrites, ("Calling PGMPhysWrite after pgmR3Save()! enmOrigin=%d\n", enmOrigin));
2904 AssertMsgReturn(cbWrite > 0, ("don't even think about writing zero bytes!\n"), VINF_SUCCESS);
2905 LogFlow(("PGMPhysWrite: %RGp %d\n", GCPhys, cbWrite));
2906
2907 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysWrite));
2908 STAM_COUNTER_ADD(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysWriteBytes), cbWrite);
2909
2910 pgmLock(pVM);
2911
2912 /*
2913 * Copy loop on ram ranges.
2914 */
2915 VBOXSTRICTRC rcStrict = VINF_SUCCESS;
2916 PPGMRAMRANGE pRam = pgmPhysGetRangeAtOrAbove(pVM, GCPhys);
2917 for (;;)
2918 {
2919 /* Inside range or not? */
2920 if (pRam && GCPhys >= pRam->GCPhys)
2921 {
2922 /*
2923 * Must work our way thru this page by page.
2924 */
2925 RTGCPTR off = GCPhys - pRam->GCPhys;
2926 while (off < pRam->cb)
2927 {
2928 RTGCPTR iPage = off >> PAGE_SHIFT;
2929 PPGMPAGE pPage = &pRam->aPages[iPage];
2930 size_t cb = PAGE_SIZE - (off & PAGE_OFFSET_MASK);
2931 if (cb > cbWrite)
2932 cb = cbWrite;
2933
2934 /*
2935 * Normal page? Get the pointer to it.
2936 */
2937 if ( !PGM_PAGE_HAS_ACTIVE_HANDLERS(pPage)
2938 && !PGM_PAGE_IS_SPECIAL_ALIAS_MMIO(pPage))
2939 {
2940 PGMPAGEMAPLOCK PgMpLck;
2941 void *pvDst;
2942 int rc = pgmPhysGCPhys2CCPtrInternal(pVM, pPage, pRam->GCPhys + off, &pvDst, &PgMpLck);
2943 if (RT_SUCCESS(rc))
2944 {
2945 Assert(!PGM_PAGE_IS_BALLOONED(pPage));
2946 memcpy(pvDst, pvBuf, cb);
2947 pgmPhysReleaseInternalPageMappingLock(pVM, &PgMpLck);
2948 }
2949 /* Ignore writes to ballooned pages. */
2950 else if (!PGM_PAGE_IS_BALLOONED(pPage))
2951 AssertLogRelMsgFailed(("pgmPhysGCPhys2CCPtrInternal failed on %RGp / %R[pgmpage] -> %Rrc\n",
2952 pRam->GCPhys + off, pPage, rc));
2953 }
2954 /*
2955 * Active WRITE or ALL access handlers.
2956 */
2957 else
2958 {
2959 VBOXSTRICTRC rcStrict2 = pgmPhysWriteHandler(pVM, pPage, pRam->GCPhys + off, pvBuf, cb, enmOrigin);
2960 if (PGM_PHYS_RW_IS_SUCCESS(rcStrict2))
2961 PGM_PHYS_RW_DO_UPDATE_STRICT_RC(rcStrict, rcStrict2);
2962 else
2963 {
2964 pgmUnlock(pVM);
2965 return rcStrict2;
2966 }
2967 }
2968
2969 /* next page */
2970 if (cb >= cbWrite)
2971 {
2972 pgmUnlock(pVM);
2973 return rcStrict;
2974 }
2975
2976 cbWrite -= cb;
2977 off += cb;
2978 pvBuf = (const char *)pvBuf + cb;
2979 } /* walk pages in ram range */
2980
2981 GCPhys = pRam->GCPhysLast + 1;
2982 }
2983 else
2984 {
2985 /*
2986 * Unassigned address space, skip it.
2987 */
2988 if (!pRam)
2989 break;
2990 size_t cb = pRam->GCPhys - GCPhys;
2991 if (cb >= cbWrite)
2992 break;
2993 cbWrite -= cb;
2994 pvBuf = (const char *)pvBuf + cb;
2995 GCPhys += cb;
2996 }
2997
2998 /* Advance range if necessary. */
2999 while (pRam && GCPhys > pRam->GCPhysLast)
3000 pRam = pRam->CTX_SUFF(pNext);
3001 } /* Ram range walk */
3002
3003 pgmUnlock(pVM);
3004 return rcStrict;
3005}
3006
3007
3008/**
3009 * Read from guest physical memory by GC physical address, bypassing
3010 * MMIO and access handlers.
3011 *
3012 * @returns VBox status code.
3013 * @param pVM The cross context VM structure.
3014 * @param pvDst The destination address.
3015 * @param GCPhysSrc The source address (GC physical address).
3016 * @param cb The number of bytes to read.
3017 */
3018VMMDECL(int) PGMPhysSimpleReadGCPhys(PVMCC pVM, void *pvDst, RTGCPHYS GCPhysSrc, size_t cb)
3019{
3020 /*
3021 * Treat the first page as a special case.
3022 */
3023 if (!cb)
3024 return VINF_SUCCESS;
3025
3026 /* map the 1st page */
3027 void const *pvSrc;
3028 PGMPAGEMAPLOCK Lock;
3029 int rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhysSrc, &pvSrc, &Lock);
3030 if (RT_FAILURE(rc))
3031 return rc;
3032
3033 /* optimize for the case where access is completely within the first page. */
3034 size_t cbPage = PAGE_SIZE - (GCPhysSrc & PAGE_OFFSET_MASK);
3035 if (RT_LIKELY(cb <= cbPage))
3036 {
3037 memcpy(pvDst, pvSrc, cb);
3038 PGMPhysReleasePageMappingLock(pVM, &Lock);
3039 return VINF_SUCCESS;
3040 }
3041
3042 /* copy to the end of the page. */
3043 memcpy(pvDst, pvSrc, cbPage);
3044 PGMPhysReleasePageMappingLock(pVM, &Lock);
3045 GCPhysSrc += cbPage;
3046 pvDst = (uint8_t *)pvDst + cbPage;
3047 cb -= cbPage;
3048
3049 /*
3050 * Page by page.
3051 */
3052 for (;;)
3053 {
3054 /* map the page */
3055 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhysSrc, &pvSrc, &Lock);
3056 if (RT_FAILURE(rc))
3057 return rc;
3058
3059 /* last page? */
3060 if (cb <= PAGE_SIZE)
3061 {
3062 memcpy(pvDst, pvSrc, cb);
3063 PGMPhysReleasePageMappingLock(pVM, &Lock);
3064 return VINF_SUCCESS;
3065 }
3066
3067 /* copy the entire page and advance */
3068 memcpy(pvDst, pvSrc, PAGE_SIZE);
3069 PGMPhysReleasePageMappingLock(pVM, &Lock);
3070 GCPhysSrc += PAGE_SIZE;
3071 pvDst = (uint8_t *)pvDst + PAGE_SIZE;
3072 cb -= PAGE_SIZE;
3073 }
3074 /* won't ever get here. */
3075}
3076
3077
3078/**
3079 * Write to guest physical memory referenced by GC pointer.
3080 * Write memory to GC physical address in guest physical memory.
3081 *
3082 * This will bypass MMIO and access handlers.
3083 *
3084 * @returns VBox status code.
3085 * @param pVM The cross context VM structure.
3086 * @param GCPhysDst The GC physical address of the destination.
3087 * @param pvSrc The source buffer.
3088 * @param cb The number of bytes to write.
3089 */
3090VMMDECL(int) PGMPhysSimpleWriteGCPhys(PVMCC pVM, RTGCPHYS GCPhysDst, const void *pvSrc, size_t cb)
3091{
3092 LogFlow(("PGMPhysSimpleWriteGCPhys: %RGp %zu\n", GCPhysDst, cb));
3093
3094 /*
3095 * Treat the first page as a special case.
3096 */
3097 if (!cb)
3098 return VINF_SUCCESS;
3099
3100 /* map the 1st page */
3101 void *pvDst;
3102 PGMPAGEMAPLOCK Lock;
3103 int rc = PGMPhysGCPhys2CCPtr(pVM, GCPhysDst, &pvDst, &Lock);
3104 if (RT_FAILURE(rc))
3105 return rc;
3106
3107 /* optimize for the case where access is completely within the first page. */
3108 size_t cbPage = PAGE_SIZE - (GCPhysDst & PAGE_OFFSET_MASK);
3109 if (RT_LIKELY(cb <= cbPage))
3110 {
3111 memcpy(pvDst, pvSrc, cb);
3112 PGMPhysReleasePageMappingLock(pVM, &Lock);
3113 return VINF_SUCCESS;
3114 }
3115
3116 /* copy to the end of the page. */
3117 memcpy(pvDst, pvSrc, cbPage);
3118 PGMPhysReleasePageMappingLock(pVM, &Lock);
3119 GCPhysDst += cbPage;
3120 pvSrc = (const uint8_t *)pvSrc + cbPage;
3121 cb -= cbPage;
3122
3123 /*
3124 * Page by page.
3125 */
3126 for (;;)
3127 {
3128 /* map the page */
3129 rc = PGMPhysGCPhys2CCPtr(pVM, GCPhysDst, &pvDst, &Lock);
3130 if (RT_FAILURE(rc))
3131 return rc;
3132
3133 /* last page? */
3134 if (cb <= PAGE_SIZE)
3135 {
3136 memcpy(pvDst, pvSrc, cb);
3137 PGMPhysReleasePageMappingLock(pVM, &Lock);
3138 return VINF_SUCCESS;
3139 }
3140
3141 /* copy the entire page and advance */
3142 memcpy(pvDst, pvSrc, PAGE_SIZE);
3143 PGMPhysReleasePageMappingLock(pVM, &Lock);
3144 GCPhysDst += PAGE_SIZE;
3145 pvSrc = (const uint8_t *)pvSrc + PAGE_SIZE;
3146 cb -= PAGE_SIZE;
3147 }
3148 /* won't ever get here. */
3149}
3150
3151
3152/**
3153 * Read from guest physical memory referenced by GC pointer.
3154 *
3155 * This function uses the current CR3/CR0/CR4 of the guest and will
3156 * bypass access handlers and not set any accessed bits.
3157 *
3158 * @returns VBox status code.
3159 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3160 * @param pvDst The destination address.
3161 * @param GCPtrSrc The source address (GC pointer).
3162 * @param cb The number of bytes to read.
3163 */
3164VMMDECL(int) PGMPhysSimpleReadGCPtr(PVMCPUCC pVCpu, void *pvDst, RTGCPTR GCPtrSrc, size_t cb)
3165{
3166 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3167/** @todo fix the macro / state handling: VMCPU_ASSERT_EMT_OR_GURU(pVCpu); */
3168
3169 /*
3170 * Treat the first page as a special case.
3171 */
3172 if (!cb)
3173 return VINF_SUCCESS;
3174
3175 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysSimpleRead));
3176 STAM_COUNTER_ADD(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysSimpleReadBytes), cb);
3177
3178 /* Take the PGM lock here, because many called functions take the lock for a very short period. That's counter-productive
3179 * when many VCPUs are fighting for the lock.
3180 */
3181 pgmLock(pVM);
3182
3183 /* map the 1st page */
3184 void const *pvSrc;
3185 PGMPAGEMAPLOCK Lock;
3186 int rc = PGMPhysGCPtr2CCPtrReadOnly(pVCpu, GCPtrSrc, &pvSrc, &Lock);
3187 if (RT_FAILURE(rc))
3188 {
3189 pgmUnlock(pVM);
3190 return rc;
3191 }
3192
3193 /* optimize for the case where access is completely within the first page. */
3194 size_t cbPage = PAGE_SIZE - ((RTGCUINTPTR)GCPtrSrc & PAGE_OFFSET_MASK);
3195 if (RT_LIKELY(cb <= cbPage))
3196 {
3197 memcpy(pvDst, pvSrc, cb);
3198 PGMPhysReleasePageMappingLock(pVM, &Lock);
3199 pgmUnlock(pVM);
3200 return VINF_SUCCESS;
3201 }
3202
3203 /* copy to the end of the page. */
3204 memcpy(pvDst, pvSrc, cbPage);
3205 PGMPhysReleasePageMappingLock(pVM, &Lock);
3206 GCPtrSrc = (RTGCPTR)((RTGCUINTPTR)GCPtrSrc + cbPage);
3207 pvDst = (uint8_t *)pvDst + cbPage;
3208 cb -= cbPage;
3209
3210 /*
3211 * Page by page.
3212 */
3213 for (;;)
3214 {
3215 /* map the page */
3216 rc = PGMPhysGCPtr2CCPtrReadOnly(pVCpu, GCPtrSrc, &pvSrc, &Lock);
3217 if (RT_FAILURE(rc))
3218 {
3219 pgmUnlock(pVM);
3220 return rc;
3221 }
3222
3223 /* last page? */
3224 if (cb <= PAGE_SIZE)
3225 {
3226 memcpy(pvDst, pvSrc, cb);
3227 PGMPhysReleasePageMappingLock(pVM, &Lock);
3228 pgmUnlock(pVM);
3229 return VINF_SUCCESS;
3230 }
3231
3232 /* copy the entire page and advance */
3233 memcpy(pvDst, pvSrc, PAGE_SIZE);
3234 PGMPhysReleasePageMappingLock(pVM, &Lock);
3235 GCPtrSrc = (RTGCPTR)((RTGCUINTPTR)GCPtrSrc + PAGE_SIZE);
3236 pvDst = (uint8_t *)pvDst + PAGE_SIZE;
3237 cb -= PAGE_SIZE;
3238 }
3239 /* won't ever get here. */
3240}
3241
3242
3243/**
3244 * Write to guest physical memory referenced by GC pointer.
3245 *
3246 * This function uses the current CR3/CR0/CR4 of the guest and will
3247 * bypass access handlers and not set dirty or accessed bits.
3248 *
3249 * @returns VBox status code.
3250 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3251 * @param GCPtrDst The destination address (GC pointer).
3252 * @param pvSrc The source address.
3253 * @param cb The number of bytes to write.
3254 */
3255VMMDECL(int) PGMPhysSimpleWriteGCPtr(PVMCPUCC pVCpu, RTGCPTR GCPtrDst, const void *pvSrc, size_t cb)
3256{
3257 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3258 VMCPU_ASSERT_EMT(pVCpu);
3259
3260 /*
3261 * Treat the first page as a special case.
3262 */
3263 if (!cb)
3264 return VINF_SUCCESS;
3265
3266 STAM_COUNTER_INC(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysSimpleWrite));
3267 STAM_COUNTER_ADD(&pVM->pgm.s.CTX_SUFF(pStats)->CTX_MID_Z(Stat,PhysSimpleWriteBytes), cb);
3268
3269 /* map the 1st page */
3270 void *pvDst;
3271 PGMPAGEMAPLOCK Lock;
3272 int rc = PGMPhysGCPtr2CCPtr(pVCpu, GCPtrDst, &pvDst, &Lock);
3273 if (RT_FAILURE(rc))
3274 return rc;
3275
3276 /* optimize for the case where access is completely within the first page. */
3277 size_t cbPage = PAGE_SIZE - ((RTGCUINTPTR)GCPtrDst & PAGE_OFFSET_MASK);
3278 if (RT_LIKELY(cb <= cbPage))
3279 {
3280 memcpy(pvDst, pvSrc, cb);
3281 PGMPhysReleasePageMappingLock(pVM, &Lock);
3282 return VINF_SUCCESS;
3283 }
3284
3285 /* copy to the end of the page. */
3286 memcpy(pvDst, pvSrc, cbPage);
3287 PGMPhysReleasePageMappingLock(pVM, &Lock);
3288 GCPtrDst = (RTGCPTR)((RTGCUINTPTR)GCPtrDst + cbPage);
3289 pvSrc = (const uint8_t *)pvSrc + cbPage;
3290 cb -= cbPage;
3291
3292 /*
3293 * Page by page.
3294 */
3295 for (;;)
3296 {
3297 /* map the page */
3298 rc = PGMPhysGCPtr2CCPtr(pVCpu, GCPtrDst, &pvDst, &Lock);
3299 if (RT_FAILURE(rc))
3300 return rc;
3301
3302 /* last page? */
3303 if (cb <= PAGE_SIZE)
3304 {
3305 memcpy(pvDst, pvSrc, cb);
3306 PGMPhysReleasePageMappingLock(pVM, &Lock);
3307 return VINF_SUCCESS;
3308 }
3309
3310 /* copy the entire page and advance */
3311 memcpy(pvDst, pvSrc, PAGE_SIZE);
3312 PGMPhysReleasePageMappingLock(pVM, &Lock);
3313 GCPtrDst = (RTGCPTR)((RTGCUINTPTR)GCPtrDst + PAGE_SIZE);
3314 pvSrc = (const uint8_t *)pvSrc + PAGE_SIZE;
3315 cb -= PAGE_SIZE;
3316 }
3317 /* won't ever get here. */
3318}
3319
3320
3321/**
3322 * Write to guest physical memory referenced by GC pointer and update the PTE.
3323 *
3324 * This function uses the current CR3/CR0/CR4 of the guest and will
3325 * bypass access handlers but will set any dirty and accessed bits in the PTE.
3326 *
3327 * If you don't want to set the dirty bit, use PGMPhysSimpleWriteGCPtr().
3328 *
3329 * @returns VBox status code.
3330 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3331 * @param GCPtrDst The destination address (GC pointer).
3332 * @param pvSrc The source address.
3333 * @param cb The number of bytes to write.
3334 */
3335VMMDECL(int) PGMPhysSimpleDirtyWriteGCPtr(PVMCPUCC pVCpu, RTGCPTR GCPtrDst, const void *pvSrc, size_t cb)
3336{
3337 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3338 VMCPU_ASSERT_EMT(pVCpu);
3339
3340 /*
3341 * Treat the first page as a special case.
3342 * Btw. this is the same code as in PGMPhyssimpleWriteGCPtr excep for the PGMGstModifyPage.
3343 */
3344 if (!cb)
3345 return VINF_SUCCESS;
3346
3347 /* map the 1st page */
3348 void *pvDst;
3349 PGMPAGEMAPLOCK Lock;
3350 int rc = PGMPhysGCPtr2CCPtr(pVCpu, GCPtrDst, &pvDst, &Lock);
3351 if (RT_FAILURE(rc))
3352 return rc;
3353
3354 /* optimize for the case where access is completely within the first page. */
3355 size_t cbPage = PAGE_SIZE - ((RTGCUINTPTR)GCPtrDst & PAGE_OFFSET_MASK);
3356 if (RT_LIKELY(cb <= cbPage))
3357 {
3358 memcpy(pvDst, pvSrc, cb);
3359 PGMPhysReleasePageMappingLock(pVM, &Lock);
3360 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D)); AssertRC(rc);
3361 return VINF_SUCCESS;
3362 }
3363
3364 /* copy to the end of the page. */
3365 memcpy(pvDst, pvSrc, cbPage);
3366 PGMPhysReleasePageMappingLock(pVM, &Lock);
3367 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D)); AssertRC(rc);
3368 GCPtrDst = (RTGCPTR)((RTGCUINTPTR)GCPtrDst + cbPage);
3369 pvSrc = (const uint8_t *)pvSrc + cbPage;
3370 cb -= cbPage;
3371
3372 /*
3373 * Page by page.
3374 */
3375 for (;;)
3376 {
3377 /* map the page */
3378 rc = PGMPhysGCPtr2CCPtr(pVCpu, GCPtrDst, &pvDst, &Lock);
3379 if (RT_FAILURE(rc))
3380 return rc;
3381
3382 /* last page? */
3383 if (cb <= PAGE_SIZE)
3384 {
3385 memcpy(pvDst, pvSrc, cb);
3386 PGMPhysReleasePageMappingLock(pVM, &Lock);
3387 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D)); AssertRC(rc);
3388 return VINF_SUCCESS;
3389 }
3390
3391 /* copy the entire page and advance */
3392 memcpy(pvDst, pvSrc, PAGE_SIZE);
3393 PGMPhysReleasePageMappingLock(pVM, &Lock);
3394 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D)); AssertRC(rc);
3395 GCPtrDst = (RTGCPTR)((RTGCUINTPTR)GCPtrDst + PAGE_SIZE);
3396 pvSrc = (const uint8_t *)pvSrc + PAGE_SIZE;
3397 cb -= PAGE_SIZE;
3398 }
3399 /* won't ever get here. */
3400}
3401
3402
3403/**
3404 * Read from guest physical memory referenced by GC pointer.
3405 *
3406 * This function uses the current CR3/CR0/CR4 of the guest and will
3407 * respect access handlers and set accessed bits.
3408 *
3409 * @returns Strict VBox status, see PGMPhysRead for details.
3410 * @retval VERR_PAGE_TABLE_NOT_PRESENT if there is no page mapped at the
3411 * specified virtual address.
3412 *
3413 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3414 * @param pvDst The destination address.
3415 * @param GCPtrSrc The source address (GC pointer).
3416 * @param cb The number of bytes to read.
3417 * @param enmOrigin Who is calling.
3418 * @thread EMT(pVCpu)
3419 */
3420VMMDECL(VBOXSTRICTRC) PGMPhysReadGCPtr(PVMCPUCC pVCpu, void *pvDst, RTGCPTR GCPtrSrc, size_t cb, PGMACCESSORIGIN enmOrigin)
3421{
3422 RTGCPHYS GCPhys;
3423 uint64_t fFlags;
3424 int rc;
3425 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3426 VMCPU_ASSERT_EMT(pVCpu);
3427
3428 /*
3429 * Anything to do?
3430 */
3431 if (!cb)
3432 return VINF_SUCCESS;
3433
3434 LogFlow(("PGMPhysReadGCPtr: %RGv %zu\n", GCPtrSrc, cb));
3435
3436 /*
3437 * Optimize reads within a single page.
3438 */
3439 if (((RTGCUINTPTR)GCPtrSrc & PAGE_OFFSET_MASK) + cb <= PAGE_SIZE)
3440 {
3441 /* Convert virtual to physical address + flags */
3442 rc = PGMGstGetPage(pVCpu, (RTGCUINTPTR)GCPtrSrc, &fFlags, &GCPhys);
3443 AssertMsgRCReturn(rc, ("GetPage failed with %Rrc for %RGv\n", rc, GCPtrSrc), rc);
3444 GCPhys |= (RTGCUINTPTR)GCPtrSrc & PAGE_OFFSET_MASK;
3445
3446 /* mark the guest page as accessed. */
3447 if (!(fFlags & X86_PTE_A))
3448 {
3449 rc = PGMGstModifyPage(pVCpu, GCPtrSrc, 1, X86_PTE_A, ~(uint64_t)(X86_PTE_A));
3450 AssertRC(rc);
3451 }
3452
3453 return PGMPhysRead(pVM, GCPhys, pvDst, cb, enmOrigin);
3454 }
3455
3456 /*
3457 * Page by page.
3458 */
3459 for (;;)
3460 {
3461 /* Convert virtual to physical address + flags */
3462 rc = PGMGstGetPage(pVCpu, (RTGCUINTPTR)GCPtrSrc, &fFlags, &GCPhys);
3463 AssertMsgRCReturn(rc, ("GetPage failed with %Rrc for %RGv\n", rc, GCPtrSrc), rc);
3464 GCPhys |= (RTGCUINTPTR)GCPtrSrc & PAGE_OFFSET_MASK;
3465
3466 /* mark the guest page as accessed. */
3467 if (!(fFlags & X86_PTE_A))
3468 {
3469 rc = PGMGstModifyPage(pVCpu, GCPtrSrc, 1, X86_PTE_A, ~(uint64_t)(X86_PTE_A));
3470 AssertRC(rc);
3471 }
3472
3473 /* copy */
3474 size_t cbRead = PAGE_SIZE - ((RTGCUINTPTR)GCPtrSrc & PAGE_OFFSET_MASK);
3475 if (cbRead < cb)
3476 {
3477 VBOXSTRICTRC rcStrict = PGMPhysRead(pVM, GCPhys, pvDst, cbRead, enmOrigin);
3478 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
3479 { /* likely */ }
3480 else
3481 return rcStrict;
3482 }
3483 else /* Last page (cbRead is PAGE_SIZE, we only need cb!) */
3484 return PGMPhysRead(pVM, GCPhys, pvDst, cb, enmOrigin);
3485
3486 /* next */
3487 Assert(cb > cbRead);
3488 cb -= cbRead;
3489 pvDst = (uint8_t *)pvDst + cbRead;
3490 GCPtrSrc += cbRead;
3491 }
3492}
3493
3494
3495/**
3496 * Write to guest physical memory referenced by GC pointer.
3497 *
3498 * This function uses the current CR3/CR0/CR4 of the guest and will
3499 * respect access handlers and set dirty and accessed bits.
3500 *
3501 * @returns Strict VBox status, see PGMPhysWrite for details.
3502 * @retval VERR_PAGE_TABLE_NOT_PRESENT if there is no page mapped at the
3503 * specified virtual address.
3504 *
3505 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3506 * @param GCPtrDst The destination address (GC pointer).
3507 * @param pvSrc The source address.
3508 * @param cb The number of bytes to write.
3509 * @param enmOrigin Who is calling.
3510 */
3511VMMDECL(VBOXSTRICTRC) PGMPhysWriteGCPtr(PVMCPUCC pVCpu, RTGCPTR GCPtrDst, const void *pvSrc, size_t cb, PGMACCESSORIGIN enmOrigin)
3512{
3513 RTGCPHYS GCPhys;
3514 uint64_t fFlags;
3515 int rc;
3516 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3517 VMCPU_ASSERT_EMT(pVCpu);
3518
3519 /*
3520 * Anything to do?
3521 */
3522 if (!cb)
3523 return VINF_SUCCESS;
3524
3525 LogFlow(("PGMPhysWriteGCPtr: %RGv %zu\n", GCPtrDst, cb));
3526
3527 /*
3528 * Optimize writes within a single page.
3529 */
3530 if (((RTGCUINTPTR)GCPtrDst & PAGE_OFFSET_MASK) + cb <= PAGE_SIZE)
3531 {
3532 /* Convert virtual to physical address + flags */
3533 rc = PGMGstGetPage(pVCpu, (RTGCUINTPTR)GCPtrDst, &fFlags, &GCPhys);
3534 AssertMsgRCReturn(rc, ("GetPage failed with %Rrc for %RGv\n", rc, GCPtrDst), rc);
3535 GCPhys |= (RTGCUINTPTR)GCPtrDst & PAGE_OFFSET_MASK;
3536
3537 /* Mention when we ignore X86_PTE_RW... */
3538 if (!(fFlags & X86_PTE_RW))
3539 Log(("PGMPhysWriteGCPtr: Writing to RO page %RGv %#x\n", GCPtrDst, cb));
3540
3541 /* Mark the guest page as accessed and dirty if necessary. */
3542 if ((fFlags & (X86_PTE_A | X86_PTE_D)) != (X86_PTE_A | X86_PTE_D))
3543 {
3544 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D));
3545 AssertRC(rc);
3546 }
3547
3548 return PGMPhysWrite(pVM, GCPhys, pvSrc, cb, enmOrigin);
3549 }
3550
3551 /*
3552 * Page by page.
3553 */
3554 for (;;)
3555 {
3556 /* Convert virtual to physical address + flags */
3557 rc = PGMGstGetPage(pVCpu, (RTGCUINTPTR)GCPtrDst, &fFlags, &GCPhys);
3558 AssertMsgRCReturn(rc, ("GetPage failed with %Rrc for %RGv\n", rc, GCPtrDst), rc);
3559 GCPhys |= (RTGCUINTPTR)GCPtrDst & PAGE_OFFSET_MASK;
3560
3561 /* Mention when we ignore X86_PTE_RW... */
3562 if (!(fFlags & X86_PTE_RW))
3563 Log(("PGMPhysWriteGCPtr: Writing to RO page %RGv %#x\n", GCPtrDst, cb));
3564
3565 /* Mark the guest page as accessed and dirty if necessary. */
3566 if ((fFlags & (X86_PTE_A | X86_PTE_D)) != (X86_PTE_A | X86_PTE_D))
3567 {
3568 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D));
3569 AssertRC(rc);
3570 }
3571
3572 /* copy */
3573 size_t cbWrite = PAGE_SIZE - ((RTGCUINTPTR)GCPtrDst & PAGE_OFFSET_MASK);
3574 if (cbWrite < cb)
3575 {
3576 VBOXSTRICTRC rcStrict = PGMPhysWrite(pVM, GCPhys, pvSrc, cbWrite, enmOrigin);
3577 if (RT_LIKELY(rcStrict == VINF_SUCCESS))
3578 { /* likely */ }
3579 else
3580 return rcStrict;
3581 }
3582 else /* Last page (cbWrite is PAGE_SIZE, we only need cb!) */
3583 return PGMPhysWrite(pVM, GCPhys, pvSrc, cb, enmOrigin);
3584
3585 /* next */
3586 Assert(cb > cbWrite);
3587 cb -= cbWrite;
3588 pvSrc = (uint8_t *)pvSrc + cbWrite;
3589 GCPtrDst += cbWrite;
3590 }
3591}
3592
3593
3594/**
3595 * Performs a read of guest virtual memory for instruction emulation.
3596 *
3597 * This will check permissions, raise exceptions and update the access bits.
3598 *
3599 * The current implementation will bypass all access handlers. It may later be
3600 * changed to at least respect MMIO.
3601 *
3602 *
3603 * @returns VBox status code suitable to scheduling.
3604 * @retval VINF_SUCCESS if the read was performed successfully.
3605 * @retval VINF_EM_RAW_GUEST_TRAP if an exception was raised but not dispatched yet.
3606 * @retval VINF_TRPM_XCPT_DISPATCHED if an exception was raised and dispatched.
3607 *
3608 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3609 * @param pCtxCore The context core.
3610 * @param pvDst Where to put the bytes we've read.
3611 * @param GCPtrSrc The source address.
3612 * @param cb The number of bytes to read. Not more than a page.
3613 *
3614 * @remark This function will dynamically map physical pages in GC. This may unmap
3615 * mappings done by the caller. Be careful!
3616 */
3617VMMDECL(int) PGMPhysInterpretedRead(PVMCPUCC pVCpu, PCPUMCTXCORE pCtxCore, void *pvDst, RTGCUINTPTR GCPtrSrc, size_t cb)
3618{
3619 NOREF(pCtxCore);
3620 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3621 Assert(cb <= PAGE_SIZE);
3622 VMCPU_ASSERT_EMT(pVCpu);
3623
3624/** @todo r=bird: This isn't perfect!
3625 * -# It's not checking for reserved bits being 1.
3626 * -# It's not correctly dealing with the access bit.
3627 * -# It's not respecting MMIO memory or any other access handlers.
3628 */
3629 /*
3630 * 1. Translate virtual to physical. This may fault.
3631 * 2. Map the physical address.
3632 * 3. Do the read operation.
3633 * 4. Set access bits if required.
3634 */
3635 int rc;
3636 unsigned cb1 = PAGE_SIZE - (GCPtrSrc & PAGE_OFFSET_MASK);
3637 if (cb <= cb1)
3638 {
3639 /*
3640 * Not crossing pages.
3641 */
3642 RTGCPHYS GCPhys;
3643 uint64_t fFlags;
3644 rc = PGMGstGetPage(pVCpu, GCPtrSrc, &fFlags, &GCPhys);
3645 if (RT_SUCCESS(rc))
3646 {
3647 /** @todo we should check reserved bits ... */
3648 PGMPAGEMAPLOCK PgMpLck;
3649 void const *pvSrc;
3650 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhys, &pvSrc, &PgMpLck);
3651 switch (rc)
3652 {
3653 case VINF_SUCCESS:
3654 Log(("PGMPhysInterpretedRead: pvDst=%p pvSrc=%p cb=%d\n", pvDst, (uint8_t *)pvSrc + (GCPtrSrc & PAGE_OFFSET_MASK), cb));
3655 memcpy(pvDst, (uint8_t *)pvSrc + (GCPtrSrc & PAGE_OFFSET_MASK), cb);
3656 PGMPhysReleasePageMappingLock(pVM, &PgMpLck);
3657 break;
3658 case VERR_PGM_PHYS_PAGE_RESERVED:
3659 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
3660 memset(pvDst, 0xff, cb);
3661 break;
3662 default:
3663 Assert(RT_FAILURE_NP(rc));
3664 return rc;
3665 }
3666
3667 /** @todo access bit emulation isn't 100% correct. */
3668 if (!(fFlags & X86_PTE_A))
3669 {
3670 rc = PGMGstModifyPage(pVCpu, GCPtrSrc, 1, X86_PTE_A, ~(uint64_t)X86_PTE_A);
3671 AssertRC(rc);
3672 }
3673 return VINF_SUCCESS;
3674 }
3675 }
3676 else
3677 {
3678 /*
3679 * Crosses pages.
3680 */
3681 size_t cb2 = cb - cb1;
3682 uint64_t fFlags1;
3683 RTGCPHYS GCPhys1;
3684 uint64_t fFlags2;
3685 RTGCPHYS GCPhys2;
3686 rc = PGMGstGetPage(pVCpu, GCPtrSrc, &fFlags1, &GCPhys1);
3687 if (RT_SUCCESS(rc))
3688 {
3689 rc = PGMGstGetPage(pVCpu, GCPtrSrc + cb1, &fFlags2, &GCPhys2);
3690 if (RT_SUCCESS(rc))
3691 {
3692 /** @todo we should check reserved bits ... */
3693 AssertMsgFailed(("cb=%d cb1=%d cb2=%d GCPtrSrc=%RGv\n", cb, cb1, cb2, GCPtrSrc));
3694 PGMPAGEMAPLOCK PgMpLck;
3695 void const *pvSrc1;
3696 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhys1, &pvSrc1, &PgMpLck);
3697 switch (rc)
3698 {
3699 case VINF_SUCCESS:
3700 memcpy(pvDst, (uint8_t *)pvSrc1 + (GCPtrSrc & PAGE_OFFSET_MASK), cb1);
3701 PGMPhysReleasePageMappingLock(pVM, &PgMpLck);
3702 break;
3703 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
3704 memset(pvDst, 0xff, cb1);
3705 break;
3706 default:
3707 Assert(RT_FAILURE_NP(rc));
3708 return rc;
3709 }
3710
3711 void const *pvSrc2;
3712 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhys2, &pvSrc2, &PgMpLck);
3713 switch (rc)
3714 {
3715 case VINF_SUCCESS:
3716 memcpy((uint8_t *)pvDst + cb1, pvSrc2, cb2);
3717 PGMPhysReleasePageMappingLock(pVM, &PgMpLck);
3718 break;
3719 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
3720 memset((uint8_t *)pvDst + cb1, 0xff, cb2);
3721 break;
3722 default:
3723 Assert(RT_FAILURE_NP(rc));
3724 return rc;
3725 }
3726
3727 if (!(fFlags1 & X86_PTE_A))
3728 {
3729 rc = PGMGstModifyPage(pVCpu, GCPtrSrc, 1, X86_PTE_A, ~(uint64_t)X86_PTE_A);
3730 AssertRC(rc);
3731 }
3732 if (!(fFlags2 & X86_PTE_A))
3733 {
3734 rc = PGMGstModifyPage(pVCpu, GCPtrSrc + cb1, 1, X86_PTE_A, ~(uint64_t)X86_PTE_A);
3735 AssertRC(rc);
3736 }
3737 return VINF_SUCCESS;
3738 }
3739 }
3740 }
3741
3742 /*
3743 * Raise a #PF.
3744 */
3745 uint32_t uErr;
3746
3747 /* Get the current privilege level. */
3748 uint32_t cpl = CPUMGetGuestCPL(pVCpu);
3749 switch (rc)
3750 {
3751 case VINF_SUCCESS:
3752 uErr = (cpl >= 2) ? X86_TRAP_PF_RSVD | X86_TRAP_PF_US : X86_TRAP_PF_RSVD;
3753 break;
3754
3755 case VERR_PAGE_NOT_PRESENT:
3756 case VERR_PAGE_TABLE_NOT_PRESENT:
3757 uErr = (cpl >= 2) ? X86_TRAP_PF_US : 0;
3758 break;
3759
3760 default:
3761 AssertMsgFailed(("rc=%Rrc GCPtrSrc=%RGv cb=%#x\n", rc, GCPtrSrc, cb));
3762 return rc;
3763 }
3764 Log(("PGMPhysInterpretedRead: GCPtrSrc=%RGv cb=%#x -> #PF(%#x)\n", GCPtrSrc, cb, uErr));
3765 rc = TRPMAssertXcptPF(pVCpu, GCPtrSrc, uErr);
3766 if (RT_SUCCESS(rc))
3767 return VINF_EM_RAW_GUEST_TRAP;
3768 return rc;
3769}
3770
3771
3772/**
3773 * Performs a read of guest virtual memory for instruction emulation.
3774 *
3775 * This will check permissions, raise exceptions and update the access bits.
3776 *
3777 * The current implementation will bypass all access handlers. It may later be
3778 * changed to at least respect MMIO.
3779 *
3780 *
3781 * @returns VBox status code suitable to scheduling.
3782 * @retval VINF_SUCCESS if the read was performed successfully.
3783 * @retval VINF_EM_RAW_GUEST_TRAP if an exception was raised but not dispatched yet.
3784 * @retval VINF_TRPM_XCPT_DISPATCHED if an exception was raised and dispatched.
3785 *
3786 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3787 * @param pCtxCore The context core.
3788 * @param pvDst Where to put the bytes we've read.
3789 * @param GCPtrSrc The source address.
3790 * @param cb The number of bytes to read. Not more than a page.
3791 * @param fRaiseTrap If set the trap will be raised on as per spec, if clear
3792 * an appropriate error status will be returned (no
3793 * informational at all).
3794 *
3795 *
3796 * @remarks Takes the PGM lock.
3797 * @remarks A page fault on the 2nd page of the access will be raised without
3798 * writing the bits on the first page since we're ASSUMING that the
3799 * caller is emulating an instruction access.
3800 * @remarks This function will dynamically map physical pages in GC. This may
3801 * unmap mappings done by the caller. Be careful!
3802 */
3803VMMDECL(int) PGMPhysInterpretedReadNoHandlers(PVMCPUCC pVCpu, PCPUMCTXCORE pCtxCore, void *pvDst, RTGCUINTPTR GCPtrSrc, size_t cb,
3804 bool fRaiseTrap)
3805{
3806 NOREF(pCtxCore);
3807 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
3808 Assert(cb <= PAGE_SIZE);
3809 VMCPU_ASSERT_EMT(pVCpu);
3810
3811 /*
3812 * 1. Translate virtual to physical. This may fault.
3813 * 2. Map the physical address.
3814 * 3. Do the read operation.
3815 * 4. Set access bits if required.
3816 */
3817 int rc;
3818 unsigned cb1 = PAGE_SIZE - (GCPtrSrc & PAGE_OFFSET_MASK);
3819 if (cb <= cb1)
3820 {
3821 /*
3822 * Not crossing pages.
3823 */
3824 RTGCPHYS GCPhys;
3825 uint64_t fFlags;
3826 rc = PGMGstGetPage(pVCpu, GCPtrSrc, &fFlags, &GCPhys);
3827 if (RT_SUCCESS(rc))
3828 {
3829 if (1) /** @todo we should check reserved bits ... */
3830 {
3831 const void *pvSrc;
3832 PGMPAGEMAPLOCK Lock;
3833 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhys, &pvSrc, &Lock);
3834 switch (rc)
3835 {
3836 case VINF_SUCCESS:
3837 Log(("PGMPhysInterpretedReadNoHandlers: pvDst=%p pvSrc=%p (%RGv) cb=%d\n",
3838 pvDst, (const uint8_t *)pvSrc + (GCPtrSrc & PAGE_OFFSET_MASK), GCPtrSrc, cb));
3839 memcpy(pvDst, (const uint8_t *)pvSrc + (GCPtrSrc & PAGE_OFFSET_MASK), cb);
3840 PGMPhysReleasePageMappingLock(pVM, &Lock);
3841 break;
3842 case VERR_PGM_PHYS_PAGE_RESERVED:
3843 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
3844 memset(pvDst, 0xff, cb);
3845 break;
3846 default:
3847 AssertMsgFailed(("%Rrc\n", rc));
3848 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
3849 return rc;
3850 }
3851
3852 if (!(fFlags & X86_PTE_A))
3853 {
3854 /** @todo access bit emulation isn't 100% correct. */
3855 rc = PGMGstModifyPage(pVCpu, GCPtrSrc, 1, X86_PTE_A, ~(uint64_t)X86_PTE_A);
3856 AssertRC(rc);
3857 }
3858 return VINF_SUCCESS;
3859 }
3860 }
3861 }
3862 else
3863 {
3864 /*
3865 * Crosses pages.
3866 */
3867 size_t cb2 = cb - cb1;
3868 uint64_t fFlags1;
3869 RTGCPHYS GCPhys1;
3870 uint64_t fFlags2;
3871 RTGCPHYS GCPhys2;
3872 rc = PGMGstGetPage(pVCpu, GCPtrSrc, &fFlags1, &GCPhys1);
3873 if (RT_SUCCESS(rc))
3874 {
3875 rc = PGMGstGetPage(pVCpu, GCPtrSrc + cb1, &fFlags2, &GCPhys2);
3876 if (RT_SUCCESS(rc))
3877 {
3878 if (1) /** @todo we should check reserved bits ... */
3879 {
3880 const void *pvSrc;
3881 PGMPAGEMAPLOCK Lock;
3882 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhys1, &pvSrc, &Lock);
3883 switch (rc)
3884 {
3885 case VINF_SUCCESS:
3886 Log(("PGMPhysInterpretedReadNoHandlers: pvDst=%p pvSrc=%p (%RGv) cb=%d [2]\n",
3887 pvDst, (const uint8_t *)pvSrc + (GCPtrSrc & PAGE_OFFSET_MASK), GCPtrSrc, cb1));
3888 memcpy(pvDst, (const uint8_t *)pvSrc + (GCPtrSrc & PAGE_OFFSET_MASK), cb1);
3889 PGMPhysReleasePageMappingLock(pVM, &Lock);
3890 break;
3891 case VERR_PGM_PHYS_PAGE_RESERVED:
3892 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
3893 memset(pvDst, 0xff, cb1);
3894 break;
3895 default:
3896 AssertMsgFailed(("%Rrc\n", rc));
3897 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
3898 return rc;
3899 }
3900
3901 rc = PGMPhysGCPhys2CCPtrReadOnly(pVM, GCPhys2, &pvSrc, &Lock);
3902 switch (rc)
3903 {
3904 case VINF_SUCCESS:
3905 memcpy((uint8_t *)pvDst + cb1, pvSrc, cb2);
3906 PGMPhysReleasePageMappingLock(pVM, &Lock);
3907 break;
3908 case VERR_PGM_PHYS_PAGE_RESERVED:
3909 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
3910 memset((uint8_t *)pvDst + cb1, 0xff, cb2);
3911 break;
3912 default:
3913 AssertMsgFailed(("%Rrc\n", rc));
3914 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
3915 return rc;
3916 }
3917
3918 if (!(fFlags1 & X86_PTE_A))
3919 {
3920 rc = PGMGstModifyPage(pVCpu, GCPtrSrc, 1, X86_PTE_A, ~(uint64_t)X86_PTE_A);
3921 AssertRC(rc);
3922 }
3923 if (!(fFlags2 & X86_PTE_A))
3924 {
3925 rc = PGMGstModifyPage(pVCpu, GCPtrSrc + cb1, 1, X86_PTE_A, ~(uint64_t)X86_PTE_A);
3926 AssertRC(rc);
3927 }
3928 return VINF_SUCCESS;
3929 }
3930 /* sort out which page */
3931 }
3932 else
3933 GCPtrSrc += cb1; /* fault on 2nd page */
3934 }
3935 }
3936
3937 /*
3938 * Raise a #PF if we're allowed to do that.
3939 */
3940 /* Calc the error bits. */
3941 uint32_t cpl = CPUMGetGuestCPL(pVCpu);
3942 uint32_t uErr;
3943 switch (rc)
3944 {
3945 case VINF_SUCCESS:
3946 uErr = (cpl >= 2) ? X86_TRAP_PF_RSVD | X86_TRAP_PF_US : X86_TRAP_PF_RSVD;
3947 rc = VERR_ACCESS_DENIED;
3948 break;
3949
3950 case VERR_PAGE_NOT_PRESENT:
3951 case VERR_PAGE_TABLE_NOT_PRESENT:
3952 uErr = (cpl >= 2) ? X86_TRAP_PF_US : 0;
3953 break;
3954
3955 default:
3956 AssertMsgFailed(("rc=%Rrc GCPtrSrc=%RGv cb=%#x\n", rc, GCPtrSrc, cb));
3957 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
3958 return rc;
3959 }
3960 if (fRaiseTrap)
3961 {
3962 Log(("PGMPhysInterpretedReadNoHandlers: GCPtrSrc=%RGv cb=%#x -> Raised #PF(%#x)\n", GCPtrSrc, cb, uErr));
3963 rc = TRPMAssertXcptPF(pVCpu, GCPtrSrc, uErr);
3964 if (RT_SUCCESS(rc))
3965 return VINF_EM_RAW_GUEST_TRAP;
3966 return rc;
3967 }
3968 Log(("PGMPhysInterpretedReadNoHandlers: GCPtrSrc=%RGv cb=%#x -> #PF(%#x) [!raised]\n", GCPtrSrc, cb, uErr));
3969 return rc;
3970}
3971
3972
3973/**
3974 * Performs a write to guest virtual memory for instruction emulation.
3975 *
3976 * This will check permissions, raise exceptions and update the dirty and access
3977 * bits.
3978 *
3979 * @returns VBox status code suitable to scheduling.
3980 * @retval VINF_SUCCESS if the read was performed successfully.
3981 * @retval VINF_EM_RAW_GUEST_TRAP if an exception was raised but not dispatched yet.
3982 * @retval VINF_TRPM_XCPT_DISPATCHED if an exception was raised and dispatched.
3983 *
3984 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
3985 * @param pCtxCore The context core.
3986 * @param GCPtrDst The destination address.
3987 * @param pvSrc What to write.
3988 * @param cb The number of bytes to write. Not more than a page.
3989 * @param fRaiseTrap If set the trap will be raised on as per spec, if clear
3990 * an appropriate error status will be returned (no
3991 * informational at all).
3992 *
3993 * @remarks Takes the PGM lock.
3994 * @remarks A page fault on the 2nd page of the access will be raised without
3995 * writing the bits on the first page since we're ASSUMING that the
3996 * caller is emulating an instruction access.
3997 * @remarks This function will dynamically map physical pages in GC. This may
3998 * unmap mappings done by the caller. Be careful!
3999 */
4000VMMDECL(int) PGMPhysInterpretedWriteNoHandlers(PVMCPUCC pVCpu, PCPUMCTXCORE pCtxCore, RTGCPTR GCPtrDst, const void *pvSrc,
4001 size_t cb, bool fRaiseTrap)
4002{
4003 NOREF(pCtxCore);
4004 Assert(cb <= PAGE_SIZE);
4005 PVMCC pVM = pVCpu->CTX_SUFF(pVM);
4006 VMCPU_ASSERT_EMT(pVCpu);
4007
4008 /*
4009 * 1. Translate virtual to physical. This may fault.
4010 * 2. Map the physical address.
4011 * 3. Do the write operation.
4012 * 4. Set access bits if required.
4013 */
4014 /** @todo Since this method is frequently used by EMInterpret or IOM
4015 * upon a write fault to an write access monitored page, we can
4016 * reuse the guest page table walking from the \#PF code. */
4017 int rc;
4018 unsigned cb1 = PAGE_SIZE - (GCPtrDst & PAGE_OFFSET_MASK);
4019 if (cb <= cb1)
4020 {
4021 /*
4022 * Not crossing pages.
4023 */
4024 RTGCPHYS GCPhys;
4025 uint64_t fFlags;
4026 rc = PGMGstGetPage(pVCpu, GCPtrDst, &fFlags, &GCPhys);
4027 if (RT_SUCCESS(rc))
4028 {
4029 if ( (fFlags & X86_PTE_RW) /** @todo Also check reserved bits. */
4030 || ( !(CPUMGetGuestCR0(pVCpu) & X86_CR0_WP)
4031 && CPUMGetGuestCPL(pVCpu) <= 2) ) /** @todo it's 2, right? Check cpl check below as well. */
4032 {
4033 void *pvDst;
4034 PGMPAGEMAPLOCK Lock;
4035 rc = PGMPhysGCPhys2CCPtr(pVM, GCPhys, &pvDst, &Lock);
4036 switch (rc)
4037 {
4038 case VINF_SUCCESS:
4039 Log(("PGMPhysInterpretedWriteNoHandlers: pvDst=%p (%RGv) pvSrc=%p cb=%d\n",
4040 (uint8_t *)pvDst + (GCPtrDst & PAGE_OFFSET_MASK), GCPtrDst, pvSrc, cb));
4041 memcpy((uint8_t *)pvDst + (GCPtrDst & PAGE_OFFSET_MASK), pvSrc, cb);
4042 PGMPhysReleasePageMappingLock(pVM, &Lock);
4043 break;
4044 case VERR_PGM_PHYS_PAGE_RESERVED:
4045 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
4046 /* bit bucket */
4047 break;
4048 default:
4049 AssertMsgFailed(("%Rrc\n", rc));
4050 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
4051 return rc;
4052 }
4053
4054 if (!(fFlags & (X86_PTE_A | X86_PTE_D)))
4055 {
4056 /** @todo dirty & access bit emulation isn't 100% correct. */
4057 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, X86_PTE_A | X86_PTE_D, ~(uint64_t)(X86_PTE_A | X86_PTE_D));
4058 AssertRC(rc);
4059 }
4060 return VINF_SUCCESS;
4061 }
4062 rc = VERR_ACCESS_DENIED;
4063 }
4064 }
4065 else
4066 {
4067 /*
4068 * Crosses pages.
4069 */
4070 size_t cb2 = cb - cb1;
4071 uint64_t fFlags1;
4072 RTGCPHYS GCPhys1;
4073 uint64_t fFlags2;
4074 RTGCPHYS GCPhys2;
4075 rc = PGMGstGetPage(pVCpu, GCPtrDst, &fFlags1, &GCPhys1);
4076 if (RT_SUCCESS(rc))
4077 {
4078 rc = PGMGstGetPage(pVCpu, GCPtrDst + cb1, &fFlags2, &GCPhys2);
4079 if (RT_SUCCESS(rc))
4080 {
4081 if ( ( (fFlags1 & X86_PTE_RW) /** @todo Also check reserved bits. */
4082 && (fFlags2 & X86_PTE_RW))
4083 || ( !(CPUMGetGuestCR0(pVCpu) & X86_CR0_WP)
4084 && CPUMGetGuestCPL(pVCpu) <= 2) )
4085 {
4086 void *pvDst;
4087 PGMPAGEMAPLOCK Lock;
4088 rc = PGMPhysGCPhys2CCPtr(pVM, GCPhys1, &pvDst, &Lock);
4089 switch (rc)
4090 {
4091 case VINF_SUCCESS:
4092 Log(("PGMPhysInterpretedWriteNoHandlers: pvDst=%p (%RGv) pvSrc=%p cb=%d\n",
4093 (uint8_t *)pvDst + (GCPtrDst & PAGE_OFFSET_MASK), GCPtrDst, pvSrc, cb1));
4094 memcpy((uint8_t *)pvDst + (GCPtrDst & PAGE_OFFSET_MASK), pvSrc, cb1);
4095 PGMPhysReleasePageMappingLock(pVM, &Lock);
4096 break;
4097 case VERR_PGM_PHYS_PAGE_RESERVED:
4098 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
4099 /* bit bucket */
4100 break;
4101 default:
4102 AssertMsgFailed(("%Rrc\n", rc));
4103 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
4104 return rc;
4105 }
4106
4107 rc = PGMPhysGCPhys2CCPtr(pVM, GCPhys2, &pvDst, &Lock);
4108 switch (rc)
4109 {
4110 case VINF_SUCCESS:
4111 memcpy(pvDst, (const uint8_t *)pvSrc + cb1, cb2);
4112 PGMPhysReleasePageMappingLock(pVM, &Lock);
4113 break;
4114 case VERR_PGM_PHYS_PAGE_RESERVED:
4115 case VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS:
4116 /* bit bucket */
4117 break;
4118 default:
4119 AssertMsgFailed(("%Rrc\n", rc));
4120 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
4121 return rc;
4122 }
4123
4124 if (!(fFlags1 & (X86_PTE_A | X86_PTE_RW)))
4125 {
4126 rc = PGMGstModifyPage(pVCpu, GCPtrDst, 1, (X86_PTE_A | X86_PTE_RW), ~(uint64_t)(X86_PTE_A | X86_PTE_RW));
4127 AssertRC(rc);
4128 }
4129 if (!(fFlags2 & (X86_PTE_A | X86_PTE_RW)))
4130 {
4131 rc = PGMGstModifyPage(pVCpu, GCPtrDst + cb1, 1, (X86_PTE_A | X86_PTE_RW), ~(uint64_t)(X86_PTE_A | X86_PTE_RW));
4132 AssertRC(rc);
4133 }
4134 return VINF_SUCCESS;
4135 }
4136 if ((fFlags1 & (X86_PTE_RW)) == X86_PTE_RW)
4137 GCPtrDst += cb1; /* fault on the 2nd page. */
4138 rc = VERR_ACCESS_DENIED;
4139 }
4140 else
4141 GCPtrDst += cb1; /* fault on the 2nd page. */
4142 }
4143 }
4144
4145 /*
4146 * Raise a #PF if we're allowed to do that.
4147 */
4148 /* Calc the error bits. */
4149 uint32_t uErr;
4150 uint32_t cpl = CPUMGetGuestCPL(pVCpu);
4151 switch (rc)
4152 {
4153 case VINF_SUCCESS:
4154 uErr = (cpl >= 2) ? X86_TRAP_PF_RSVD | X86_TRAP_PF_US : X86_TRAP_PF_RSVD;
4155 rc = VERR_ACCESS_DENIED;
4156 break;
4157
4158 case VERR_ACCESS_DENIED:
4159 uErr = (cpl >= 2) ? X86_TRAP_PF_RW | X86_TRAP_PF_US : X86_TRAP_PF_RW;
4160 break;
4161
4162 case VERR_PAGE_NOT_PRESENT:
4163 case VERR_PAGE_TABLE_NOT_PRESENT:
4164 uErr = (cpl >= 2) ? X86_TRAP_PF_US : 0;
4165 break;
4166
4167 default:
4168 AssertMsgFailed(("rc=%Rrc GCPtrDst=%RGv cb=%#x\n", rc, GCPtrDst, cb));
4169 AssertReturn(RT_FAILURE(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
4170 return rc;
4171 }
4172 if (fRaiseTrap)
4173 {
4174 Log(("PGMPhysInterpretedWriteNoHandlers: GCPtrDst=%RGv cb=%#x -> Raised #PF(%#x)\n", GCPtrDst, cb, uErr));
4175 rc = TRPMAssertXcptPF(pVCpu, GCPtrDst, uErr);
4176 if (RT_SUCCESS(rc))
4177 return VINF_EM_RAW_GUEST_TRAP;
4178 return rc;
4179 }
4180 Log(("PGMPhysInterpretedWriteNoHandlers: GCPtrDst=%RGv cb=%#x -> #PF(%#x) [!raised]\n", GCPtrDst, cb, uErr));
4181 return rc;
4182}
4183
4184
4185/**
4186 * Return the page type of the specified physical address.
4187 *
4188 * @returns The page type.
4189 * @param pVM The cross context VM structure.
4190 * @param GCPhys Guest physical address
4191 */
4192VMM_INT_DECL(PGMPAGETYPE) PGMPhysGetPageType(PVMCC pVM, RTGCPHYS GCPhys)
4193{
4194 pgmLock(pVM);
4195 PPGMPAGE pPage = pgmPhysGetPage(pVM, GCPhys);
4196 PGMPAGETYPE enmPgType = pPage ? (PGMPAGETYPE)PGM_PAGE_GET_TYPE(pPage) : PGMPAGETYPE_INVALID;
4197 pgmUnlock(pVM);
4198
4199 return enmPgType;
4200}
4201
4202
4203/**
4204 * Converts a GC physical address to a HC ring-3 pointer, with some
4205 * additional checks.
4206 *
4207 * @returns VBox status code (no informational statuses).
4208 *
4209 * @param pVM The cross context VM structure.
4210 * @param pVCpu The cross context virtual CPU structure of the
4211 * calling EMT.
4212 * @param GCPhys The GC physical address to convert. This API mask
4213 * the A20 line when necessary.
4214 * @param puTlbPhysRev Where to read the physical TLB revision. Needs to
4215 * be done while holding the PGM lock.
4216 * @param ppb Where to store the pointer corresponding to GCPhys
4217 * on success.
4218 * @param pfTlb The TLB flags and revision. We only add stuff.
4219 *
4220 * @remarks This is more or a less a copy of PGMR3PhysTlbGCPhys2Ptr and
4221 * PGMPhysIemGCPhys2Ptr.
4222 *
4223 * @thread EMT(pVCpu).
4224 */
4225VMM_INT_DECL(int) PGMPhysIemGCPhys2PtrNoLock(PVMCC pVM, PVMCPUCC pVCpu, RTGCPHYS GCPhys, uint64_t const volatile *puTlbPhysRev,
4226#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
4227 R3PTRTYPE(uint8_t *) *ppb,
4228#else
4229 R3R0PTRTYPE(uint8_t *) *ppb,
4230#endif
4231 uint64_t *pfTlb)
4232{
4233 PGM_A20_APPLY_TO_VAR(pVCpu, GCPhys);
4234 Assert(!(GCPhys & X86_PAGE_OFFSET_MASK));
4235
4236 pgmLock(pVM);
4237
4238 PPGMRAMRANGE pRam;
4239 PPGMPAGE pPage;
4240 int rc = pgmPhysGetPageAndRangeEx(pVM, GCPhys, &pPage, &pRam);
4241 if (RT_SUCCESS(rc))
4242 {
4243 if (!PGM_PAGE_IS_BALLOONED(pPage))
4244 {
4245 if (!PGM_PAGE_IS_SPECIAL_ALIAS_MMIO(pPage))
4246 {
4247 if (!PGM_PAGE_HAS_ANY_HANDLERS(pPage))
4248 {
4249 /*
4250 * No access handler.
4251 */
4252 switch (PGM_PAGE_GET_STATE(pPage))
4253 {
4254 case PGM_PAGE_STATE_ALLOCATED:
4255 *pfTlb |= *puTlbPhysRev;
4256 break;
4257 case PGM_PAGE_STATE_BALLOONED:
4258 AssertFailed();
4259 RT_FALL_THRU();
4260 case PGM_PAGE_STATE_ZERO:
4261 case PGM_PAGE_STATE_SHARED:
4262 case PGM_PAGE_STATE_WRITE_MONITORED:
4263 *pfTlb |= *puTlbPhysRev | PGMIEMGCPHYS2PTR_F_NO_WRITE;
4264 break;
4265 }
4266#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
4267 *pfTlb |= PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3;
4268 *ppb = NULL;
4269#else
4270 PPGMPAGER3MAPTLBE pTlbe;
4271 rc = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
4272 AssertLogRelRCReturn(rc, rc);
4273 *ppb = (uint8_t *)pTlbe->pv;
4274#endif
4275 }
4276 else if (PGM_PAGE_HAS_ACTIVE_ALL_HANDLERS(pPage))
4277 {
4278 /*
4279 * MMIO or similar all access handler: Catch all access.
4280 */
4281 *pfTlb |= *puTlbPhysRev
4282 | PGMIEMGCPHYS2PTR_F_NO_WRITE | PGMIEMGCPHYS2PTR_F_NO_READ | PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3;
4283 *ppb = NULL;
4284 }
4285 else
4286 {
4287 /*
4288 * Write access handler: Catch write accesses if active.
4289 */
4290 if (PGM_PAGE_HAS_ACTIVE_HANDLERS(pPage))
4291 *pfTlb |= *puTlbPhysRev | PGMIEMGCPHYS2PTR_F_NO_WRITE;
4292 else
4293 switch (PGM_PAGE_GET_STATE(pPage))
4294 {
4295 case PGM_PAGE_STATE_ALLOCATED:
4296 *pfTlb |= *puTlbPhysRev;
4297 break;
4298 case PGM_PAGE_STATE_BALLOONED:
4299 AssertFailed();
4300 RT_FALL_THRU();
4301 case PGM_PAGE_STATE_ZERO:
4302 case PGM_PAGE_STATE_SHARED:
4303 case PGM_PAGE_STATE_WRITE_MONITORED:
4304 *pfTlb |= *puTlbPhysRev | PGMIEMGCPHYS2PTR_F_NO_WRITE;
4305 break;
4306 }
4307#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
4308 *pfTlb |= PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3;
4309 *ppb = NULL;
4310#else
4311 PPGMPAGER3MAPTLBE pTlbe;
4312 rc = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
4313 AssertLogRelRCReturn(rc, rc);
4314 *ppb = (uint8_t *)pTlbe->pv;
4315#endif
4316 }
4317 }
4318 else
4319 {
4320 /* Alias MMIO: For now, we catch all access. */
4321 *pfTlb |= *puTlbPhysRev
4322 | PGMIEMGCPHYS2PTR_F_NO_WRITE | PGMIEMGCPHYS2PTR_F_NO_READ | PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3;
4323 *ppb = NULL;
4324 }
4325 }
4326 else
4327 {
4328 /* Ballooned: Shouldn't get here, but we read zero page via PGMPhysRead and writes goes to /dev/null. */
4329 *pfTlb |= *puTlbPhysRev | PGMIEMGCPHYS2PTR_F_NO_WRITE | PGMIEMGCPHYS2PTR_F_NO_READ | PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3;
4330 *ppb = NULL;
4331 }
4332 Log6(("PGMPhysIemGCPhys2PtrNoLock: GCPhys=%RGp *ppb=%p *pfTlb=%#RX64 pPage=%R[pgmpage]\n", GCPhys, *ppb, *pfTlb, pPage));
4333 }
4334 else
4335 {
4336 *pfTlb |= *puTlbPhysRev | PGMIEMGCPHYS2PTR_F_NO_WRITE | PGMIEMGCPHYS2PTR_F_NO_READ | PGMIEMGCPHYS2PTR_F_NO_MAPPINGR3;
4337 *ppb = NULL;
4338 Log6(("PGMPhysIemGCPhys2PtrNoLock: GCPhys=%RGp *ppb=%p *pfTlb=%#RX64 (rc=%Rrc)\n", GCPhys, *ppb, *pfTlb, rc));
4339 }
4340
4341 pgmUnlock(pVM);
4342 return VINF_SUCCESS;
4343}
4344
4345
4346/**
4347 * Converts a GC physical address to a HC ring-3 pointer, with some
4348 * additional checks.
4349 *
4350 * @returns VBox status code (no informational statuses).
4351 * @retval VINF_SUCCESS on success.
4352 * @retval VERR_PGM_PHYS_TLB_CATCH_WRITE and *ppv set if the page has a write
4353 * access handler of some kind.
4354 * @retval VERR_PGM_PHYS_TLB_CATCH_ALL if the page has a handler catching all
4355 * accesses or is odd in any way.
4356 * @retval VERR_PGM_PHYS_TLB_UNASSIGNED if the page doesn't exist.
4357 *
4358 * @param pVM The cross context VM structure.
4359 * @param pVCpu The cross context virtual CPU structure of the
4360 * calling EMT.
4361 * @param GCPhys The GC physical address to convert. This API mask
4362 * the A20 line when necessary.
4363 * @param fWritable Whether write access is required.
4364 * @param fByPassHandlers Whether to bypass access handlers.
4365 * @param ppv Where to store the pointer corresponding to GCPhys
4366 * on success.
4367 * @param pLock
4368 *
4369 * @remarks This is more or a less a copy of PGMR3PhysTlbGCPhys2Ptr.
4370 * @thread EMT(pVCpu).
4371 */
4372VMM_INT_DECL(int) PGMPhysIemGCPhys2Ptr(PVMCC pVM, PVMCPUCC pVCpu, RTGCPHYS GCPhys, bool fWritable, bool fByPassHandlers,
4373 void **ppv, PPGMPAGEMAPLOCK pLock)
4374{
4375 PGM_A20_APPLY_TO_VAR(pVCpu, GCPhys);
4376
4377 pgmLock(pVM);
4378
4379 PPGMRAMRANGE pRam;
4380 PPGMPAGE pPage;
4381 int rc = pgmPhysGetPageAndRangeEx(pVM, GCPhys, &pPage, &pRam);
4382 if (RT_SUCCESS(rc))
4383 {
4384 if (PGM_PAGE_IS_BALLOONED(pPage))
4385 rc = VERR_PGM_PHYS_TLB_CATCH_WRITE;
4386 else if (PGM_PAGE_IS_SPECIAL_ALIAS_MMIO(pPage))
4387 rc = VERR_PGM_PHYS_TLB_CATCH_ALL;
4388 else if ( !PGM_PAGE_HAS_ANY_HANDLERS(pPage)
4389 || (fByPassHandlers && !PGM_PAGE_IS_MMIO(pPage)) )
4390 rc = VINF_SUCCESS;
4391 else
4392 {
4393 if (PGM_PAGE_HAS_ACTIVE_ALL_HANDLERS(pPage)) /* catches MMIO */
4394 {
4395 Assert(!fByPassHandlers || PGM_PAGE_IS_MMIO(pPage));
4396 rc = VERR_PGM_PHYS_TLB_CATCH_ALL;
4397 }
4398 else if (PGM_PAGE_HAS_ACTIVE_HANDLERS(pPage) && fWritable)
4399 {
4400 Assert(!fByPassHandlers);
4401 rc = VERR_PGM_PHYS_TLB_CATCH_WRITE;
4402 }
4403 }
4404 if (RT_SUCCESS(rc))
4405 {
4406 int rc2;
4407
4408 /* Make sure what we return is writable. */
4409 if (fWritable)
4410 switch (PGM_PAGE_GET_STATE(pPage))
4411 {
4412 case PGM_PAGE_STATE_ALLOCATED:
4413 break;
4414 case PGM_PAGE_STATE_BALLOONED:
4415 AssertFailed();
4416 break;
4417 case PGM_PAGE_STATE_ZERO:
4418 case PGM_PAGE_STATE_SHARED:
4419 case PGM_PAGE_STATE_WRITE_MONITORED:
4420 rc2 = pgmPhysPageMakeWritable(pVM, pPage, GCPhys & ~(RTGCPHYS)PAGE_OFFSET_MASK);
4421 AssertLogRelRCReturn(rc2, rc2);
4422 break;
4423 }
4424
4425#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE_IN_R0
4426 void *pv;
4427 rc = pgmRZDynMapHCPageInlined(pVCpu,
4428 PGM_PAGE_GET_HCPHYS(pPage),
4429 &pv
4430 RTLOG_COMMA_SRC_POS);
4431 if (RT_FAILURE(rc))
4432 return rc;
4433 *ppv = (void *)((uintptr_t)pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
4434 pLock->pvPage = pv;
4435 pLock->pVCpu = pVCpu;
4436
4437#else
4438 /* Get a ring-3 mapping of the address. */
4439 PPGMPAGER3MAPTLBE pTlbe;
4440 rc2 = pgmPhysPageQueryTlbeWithPage(pVM, pPage, GCPhys, &pTlbe);
4441 AssertLogRelRCReturn(rc2, rc2);
4442
4443 /* Lock it and calculate the address. */
4444 if (fWritable)
4445 pgmPhysPageMapLockForWriting(pVM, pPage, pTlbe, pLock);
4446 else
4447 pgmPhysPageMapLockForReading(pVM, pPage, pTlbe, pLock);
4448 *ppv = (void *)((uintptr_t)pTlbe->pv | (uintptr_t)(GCPhys & PAGE_OFFSET_MASK));
4449#endif
4450
4451 Log6(("PGMPhysIemGCPhys2Ptr: GCPhys=%RGp rc=%Rrc pPage=%R[pgmpage] *ppv=%p\n", GCPhys, rc, pPage, *ppv));
4452 }
4453 else
4454 Log6(("PGMPhysIemGCPhys2Ptr: GCPhys=%RGp rc=%Rrc pPage=%R[pgmpage]\n", GCPhys, rc, pPage));
4455
4456 /* else: handler catching all access, no pointer returned. */
4457 }
4458 else
4459 rc = VERR_PGM_PHYS_TLB_UNASSIGNED;
4460
4461 pgmUnlock(pVM);
4462 return rc;
4463}
4464
4465
4466/**
4467 * Checks if the give GCPhys page requires special handling for the given access
4468 * because it's MMIO or otherwise monitored.
4469 *
4470 * @returns VBox status code (no informational statuses).
4471 * @retval VINF_SUCCESS on success.
4472 * @retval VERR_PGM_PHYS_TLB_CATCH_WRITE and *ppv set if the page has a write
4473 * access handler of some kind.
4474 * @retval VERR_PGM_PHYS_TLB_CATCH_ALL if the page has a handler catching all
4475 * accesses or is odd in any way.
4476 * @retval VERR_PGM_PHYS_TLB_UNASSIGNED if the page doesn't exist.
4477 *
4478 * @param pVM The cross context VM structure.
4479 * @param GCPhys The GC physical address to convert. Since this is
4480 * only used for filling the REM TLB, the A20 mask must
4481 * be applied before calling this API.
4482 * @param fWritable Whether write access is required.
4483 * @param fByPassHandlers Whether to bypass access handlers.
4484 *
4485 * @remarks This is a watered down version PGMPhysIemGCPhys2Ptr and really just
4486 * a stop gap thing that should be removed once there is a better TLB
4487 * for virtual address accesses.
4488 */
4489VMM_INT_DECL(int) PGMPhysIemQueryAccess(PVMCC pVM, RTGCPHYS GCPhys, bool fWritable, bool fByPassHandlers)
4490{
4491 pgmLock(pVM);
4492 PGM_A20_ASSERT_MASKED(VMMGetCpu(pVM), GCPhys);
4493
4494 PPGMRAMRANGE pRam;
4495 PPGMPAGE pPage;
4496 int rc = pgmPhysGetPageAndRangeEx(pVM, GCPhys, &pPage, &pRam);
4497 if (RT_SUCCESS(rc))
4498 {
4499 if (PGM_PAGE_IS_BALLOONED(pPage))
4500 rc = VERR_PGM_PHYS_TLB_CATCH_WRITE;
4501 else if (PGM_PAGE_IS_SPECIAL_ALIAS_MMIO(pPage))
4502 rc = VERR_PGM_PHYS_TLB_CATCH_ALL;
4503 else if ( !PGM_PAGE_HAS_ANY_HANDLERS(pPage)
4504 || (fByPassHandlers && !PGM_PAGE_IS_MMIO(pPage)) )
4505 rc = VINF_SUCCESS;
4506 else
4507 {
4508 if (PGM_PAGE_HAS_ACTIVE_ALL_HANDLERS(pPage)) /* catches MMIO */
4509 {
4510 Assert(!fByPassHandlers || PGM_PAGE_IS_MMIO(pPage));
4511 rc = VERR_PGM_PHYS_TLB_CATCH_ALL;
4512 }
4513 else if (PGM_PAGE_HAS_ACTIVE_HANDLERS(pPage) && fWritable)
4514 {
4515 Assert(!fByPassHandlers);
4516 rc = VERR_PGM_PHYS_TLB_CATCH_WRITE;
4517 }
4518 }
4519 }
4520
4521 pgmUnlock(pVM);
4522 return rc;
4523}
4524
4525
4526/**
4527 * Interface used by NEM to check what to do on a memory access exit.
4528 *
4529 * @returns VBox status code.
4530 * @param pVM The cross context VM structure.
4531 * @param pVCpu The cross context per virtual CPU structure.
4532 * Optional.
4533 * @param GCPhys The guest physical address.
4534 * @param fMakeWritable Whether to try make the page writable or not. If it
4535 * cannot be made writable, NEM_PAGE_PROT_WRITE won't
4536 * be returned and the return code will be unaffected
4537 * @param pInfo Where to return the page information. This is
4538 * initialized even on failure.
4539 * @param pfnChecker Page in-sync checker callback. Optional.
4540 * @param pvUser User argument to pass to pfnChecker.
4541 */
4542VMM_INT_DECL(int) PGMPhysNemPageInfoChecker(PVMCC pVM, PVMCPUCC pVCpu, RTGCPHYS GCPhys, bool fMakeWritable, PPGMPHYSNEMPAGEINFO pInfo,
4543 PFNPGMPHYSNEMCHECKPAGE pfnChecker, void *pvUser)
4544{
4545 pgmLock(pVM);
4546
4547 PPGMPAGE pPage;
4548 int rc = pgmPhysGetPageEx(pVM, GCPhys, &pPage);
4549 if (RT_SUCCESS(rc))
4550 {
4551 /* Try make it writable if requested. */
4552 pInfo->u2OldNemState = PGM_PAGE_GET_NEM_STATE(pPage);
4553 if (fMakeWritable)
4554 switch (PGM_PAGE_GET_STATE(pPage))
4555 {
4556 case PGM_PAGE_STATE_SHARED:
4557 case PGM_PAGE_STATE_WRITE_MONITORED:
4558 case PGM_PAGE_STATE_ZERO:
4559 rc = pgmPhysPageMakeWritable(pVM, pPage, GCPhys);
4560 if (rc == VERR_PGM_PHYS_PAGE_RESERVED)
4561 rc = VINF_SUCCESS;
4562 break;
4563 }
4564
4565 /* Fill in the info. */
4566 pInfo->HCPhys = PGM_PAGE_GET_HCPHYS(pPage);
4567 pInfo->u2NemState = PGM_PAGE_GET_NEM_STATE(pPage);
4568 pInfo->fHasHandlers = PGM_PAGE_HAS_ACTIVE_HANDLERS(pPage) ? 1 : 0;
4569 PGMPAGETYPE const enmType = (PGMPAGETYPE)PGM_PAGE_GET_TYPE(pPage);
4570 pInfo->enmType = enmType;
4571 pInfo->fNemProt = pgmPhysPageCalcNemProtection(pPage, enmType);
4572 switch (PGM_PAGE_GET_STATE(pPage))
4573 {
4574 case PGM_PAGE_STATE_ALLOCATED:
4575 pInfo->fZeroPage = 0;
4576 break;
4577
4578 case PGM_PAGE_STATE_ZERO:
4579 pInfo->fZeroPage = 1;
4580 break;
4581
4582 case PGM_PAGE_STATE_WRITE_MONITORED:
4583 pInfo->fZeroPage = 0;
4584 break;
4585
4586 case PGM_PAGE_STATE_SHARED:
4587 pInfo->fZeroPage = 0;
4588 break;
4589
4590 case PGM_PAGE_STATE_BALLOONED:
4591 pInfo->fZeroPage = 1;
4592 break;
4593
4594 default:
4595 pInfo->fZeroPage = 1;
4596 AssertFailedStmt(rc = VERR_PGM_PHYS_PAGE_GET_IPE);
4597 }
4598
4599 /* Call the checker and update NEM state. */
4600 if (pfnChecker)
4601 {
4602 rc = pfnChecker(pVM, pVCpu, GCPhys, pInfo, pvUser);
4603 PGM_PAGE_SET_NEM_STATE(pPage, pInfo->u2NemState);
4604 }
4605
4606 /* Done. */
4607 pgmUnlock(pVM);
4608 }
4609 else
4610 {
4611 pgmUnlock(pVM);
4612
4613 pInfo->HCPhys = NIL_RTHCPHYS;
4614 pInfo->fNemProt = NEM_PAGE_PROT_NONE;
4615 pInfo->u2NemState = 0;
4616 pInfo->fHasHandlers = 0;
4617 pInfo->fZeroPage = 0;
4618 pInfo->enmType = PGMPAGETYPE_INVALID;
4619 }
4620
4621 return rc;
4622}
4623
4624
4625/**
4626 * NEM helper that performs @a pfnCallback on pages with NEM state @a uMinState
4627 * or higher.
4628 *
4629 * @returns VBox status code from callback.
4630 * @param pVM The cross context VM structure.
4631 * @param pVCpu The cross context per CPU structure. This is
4632 * optional as its only for passing to callback.
4633 * @param uMinState The minimum NEM state value to call on.
4634 * @param pfnCallback The callback function.
4635 * @param pvUser User argument for the callback.
4636 */
4637VMM_INT_DECL(int) PGMPhysNemEnumPagesByState(PVMCC pVM, PVMCPUCC pVCpu, uint8_t uMinState,
4638 PFNPGMPHYSNEMENUMCALLBACK pfnCallback, void *pvUser)
4639{
4640 /*
4641 * Just brute force this problem.
4642 */
4643 pgmLock(pVM);
4644 int rc = VINF_SUCCESS;
4645 for (PPGMRAMRANGE pRam = pVM->pgm.s.CTX_SUFF(pRamRangesX); pRam; pRam = pRam->CTX_SUFF(pNext))
4646 {
4647 uint32_t const cPages = pRam->cb >> X86_PAGE_SHIFT;
4648 for (uint32_t iPage = 0; iPage < cPages; iPage++)
4649 {
4650 uint8_t u2State = PGM_PAGE_GET_NEM_STATE(&pRam->aPages[iPage]);
4651 if (u2State < uMinState)
4652 { /* likely */ }
4653 else
4654 {
4655 rc = pfnCallback(pVM, pVCpu, pRam->GCPhys + ((RTGCPHYS)iPage << X86_PAGE_SHIFT), &u2State, pvUser);
4656 if (RT_SUCCESS(rc))
4657 PGM_PAGE_SET_NEM_STATE(&pRam->aPages[iPage], u2State);
4658 else
4659 break;
4660 }
4661 }
4662 }
4663 pgmUnlock(pVM);
4664
4665 return rc;
4666
4667}
4668
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