VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR0/PGMR0SharedPage.cpp@ 35333

Last change on this file since 35333 was 35333, checked in by vboxsync, 14 years ago

VMM source reorg.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 6.1 KB
Line 
1/* $Id: PGMR0SharedPage.cpp 35333 2010-12-27 12:10:56Z vboxsync $ */
2/** @file
3 * PGM - Page Manager and Monitor, Page Sharing, Ring-0.
4 */
5
6/*
7 * Copyright (C) 2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_PGM_SHARED
22#include <VBox/pgm.h>
23#include <VBox/gmm.h>
24#include "PGMInternal.h"
25#include <VBox/vm.h>
26#include "PGMInline.h"
27#include <VBox/log.h>
28#include <VBox/err.h>
29#include <iprt/assert.h>
30#include <iprt/mem.h>
31
32
33#ifdef VBOX_WITH_PAGE_SHARING
34/**
35 * Check a registered module for shared page changes
36 *
37 * @returns The following VBox status codes.
38 *
39 * @param pVM The VM handle.
40 * @param pGVM Pointer to the GVM instance data.
41 * @param idCpu VCPU id
42 * @param pModule Module description
43 * @param cRegions Number of regions
44 * @param pRegions Region array
45 */
46VMMR0DECL(int) PGMR0SharedModuleCheck(PVM pVM, PGVM pGVM, VMCPUID idCpu, PGMMSHAREDMODULE pModule, uint32_t cRegions, PGMMSHAREDREGIONDESC pRegions)
47{
48 int rc = VINF_SUCCESS;
49 GMMSHAREDPAGEDESC PageDesc;
50 bool fFlushTLBs = false;
51 PVMCPU pVCpu = &pVM->aCpus[idCpu];
52
53 Log(("PGMR0SharedModuleCheck: check %s %s base=%RGv size=%x\n", pModule->szName, pModule->szVersion, pModule->Core.Key, pModule->cbModule));
54
55 Assert(PGMIsLockOwner(pVM)); /* This cannot fail as we grab the lock in pgmR3SharedModuleRegRendezvous before calling into ring-0. */
56
57 /* Check every region of the shared module. */
58 for (unsigned idxRegion = 0; idxRegion < cRegions; idxRegion++)
59 {
60 Assert((pRegions[idxRegion].cbRegion & 0xfff) == 0);
61 Assert((pRegions[idxRegion].GCRegionAddr & 0xfff) == 0);
62
63 RTGCPTR GCRegion = pRegions[idxRegion].GCRegionAddr;
64 unsigned cbRegion = pRegions[idxRegion].cbRegion & ~0xfff;
65 unsigned idxPage = 0;
66
67 while (cbRegion)
68 {
69 RTGCPHYS GCPhys;
70 uint64_t fFlags;
71
72 /** @todo inefficient to fetch each guest page like this... */
73 rc = PGMGstGetPage(pVCpu, GCRegion, &fFlags, &GCPhys);
74 if ( rc == VINF_SUCCESS
75 && !(fFlags & X86_PTE_RW)) /* important as we make assumptions about this below! */
76 {
77 PPGMPAGE pPage = pgmPhysGetPage(&pVM->pgm.s, GCPhys);
78 Assert(!pPage || !PGM_PAGE_IS_BALLOONED(pPage));
79 if ( pPage
80 && PGM_PAGE_GET_STATE(pPage) == PGM_PAGE_STATE_ALLOCATED)
81 {
82 PageDesc.uHCPhysPageId = PGM_PAGE_GET_PAGEID(pPage);
83 PageDesc.HCPhys = PGM_PAGE_GET_HCPHYS(pPage);
84 PageDesc.GCPhys = GCPhys;
85
86 rc = GMMR0SharedModuleCheckPage(pGVM, pModule, idxRegion, idxPage, &PageDesc);
87 if (rc == VINF_SUCCESS)
88 {
89 /* Any change for this page? */
90 if (PageDesc.uHCPhysPageId != NIL_GMM_PAGEID)
91 {
92 Assert(PGM_PAGE_GET_STATE(pPage) == PGM_PAGE_STATE_ALLOCATED);
93
94 Log(("PGMR0SharedModuleCheck: shared page gc virt=%RGv phys %RGp host %RHp->%RHp\n", pRegions[idxRegion].GCRegionAddr + idxPage * PAGE_SIZE, PageDesc.GCPhys, PGM_PAGE_GET_HCPHYS(pPage), PageDesc.HCPhys));
95 if (PageDesc.HCPhys != PGM_PAGE_GET_HCPHYS(pPage))
96 {
97 bool fFlush = false;
98
99 /* Page was replaced by an existing shared version of it; clear all references first. */
100 rc = pgmPoolTrackUpdateGCPhys(pVM, PageDesc.GCPhys, pPage, true /* clear the entries */, &fFlush);
101 Assert(rc == VINF_SUCCESS || (VMCPU_FF_ISSET(pVCpu, VMCPU_FF_PGM_SYNC_CR3) && (pVCpu->pgm.s.fSyncFlags & PGM_SYNC_CLEAR_PGM_POOL)));
102 if (rc == VINF_SUCCESS)
103 fFlushTLBs |= fFlush;
104
105 /* Update the physical address and page id now. */
106 PGM_PAGE_SET_HCPHYS(pPage, PageDesc.HCPhys);
107 PGM_PAGE_SET_PAGEID(pPage, PageDesc.uHCPhysPageId);
108
109 /* Invalidate page map TLB entry for this page too. */
110 PGMPhysInvalidatePageMapTLBEntry(pVM, PageDesc.GCPhys);
111 pVM->pgm.s.cReusedSharedPages++;
112 }
113 /* else nothing changed (== this page is now a shared page), so no need to flush anything. */
114
115 pVM->pgm.s.cSharedPages++;
116 pVM->pgm.s.cPrivatePages--;
117 PGM_PAGE_SET_STATE(pPage, PGM_PAGE_STATE_SHARED);
118 }
119 }
120 else
121 break;
122 }
123 }
124 else
125 {
126 Assert( rc == VINF_SUCCESS
127 || rc == VERR_PAGE_NOT_PRESENT
128 || rc == VERR_PAGE_MAP_LEVEL4_NOT_PRESENT
129 || rc == VERR_PAGE_DIRECTORY_PTR_NOT_PRESENT
130 || rc == VERR_PAGE_TABLE_NOT_PRESENT);
131 rc = VINF_SUCCESS; /* ignore error */
132 }
133
134 idxPage++;
135 GCRegion += PAGE_SIZE;
136 cbRegion -= PAGE_SIZE;
137 }
138 }
139
140 if (fFlushTLBs)
141 PGM_INVL_ALL_VCPU_TLBS(pVM);
142
143 return rc;
144}
145#endif
146
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