VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxServicePageSharing.cpp@ 70171

Last change on this file since 70171 was 70171, checked in by vboxsync, 7 years ago

VBoxService: Working on removing TARGET_NT4 here too...

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 27.7 KB
Line 
1/* $Id: VBoxServicePageSharing.cpp 70171 2017-12-15 22:08:07Z vboxsync $ */
2/** @file
3 * VBoxService - Guest page sharing.
4 */
5
6/*
7 * Copyright (C) 2006-2017 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/** @page pg_vgsvc_pagesharing VBoxService - Page Sharing
20 *
21 * The Page Sharing subservice is responsible for finding memory mappings
22 * suitable page fusions.
23 *
24 * It is the driving force behind the Page Fusion feature in VirtualBox.
25 * Working with PGM and GMM (ring-0) thru the VMMDev interface. Every so often
26 * it reenumerates the memory mappings (executables and shared libraries) of the
27 * guest OS and reports additions and removals to GMM. For each mapping there
28 * is a filename and version as well as and address range and subsections. GMM
29 * will match the mapping with mapping with the same name and version from other
30 * VMs and see if there are any identical pages between the two.
31 *
32 * To increase the hit rate and reduce the volatility, the service launches a
33 * child process which loads all the Windows system DLLs it can. The child
34 * process is necessary as the DLLs are loaded without running the init code,
35 * and therefore not actually callable for other VBoxService code (may crash).
36 *
37 * This is currently only implemented on Windows. There is no technical reason
38 * for it not to be doable for all the other guests too, it's just a matter of
39 * customer demand and engineering time.
40 *
41 */
42
43
44/*********************************************************************************************************************************
45* Header Files *
46*********************************************************************************************************************************/
47#undef _WIN32_WINNT /// REMOVE WHEN VBoxServiceNT IS GONE
48#define _WIN32_WINNT 0x0501 /// REMOVE WHEN VBoxServiceNT IS GONE
49#include <iprt/assert.h>
50#include <iprt/avl.h>
51#include <iprt/asm.h>
52#include <iprt/mem.h>
53#include <iprt/ldr.h>
54#include <iprt/process.h>
55#include <iprt/env.h>
56#include <iprt/stream.h>
57#include <iprt/file.h>
58#include <iprt/string.h>
59#include <iprt/semaphore.h>
60#include <iprt/string.h>
61#include <iprt/system.h>
62#include <iprt/thread.h>
63#include <iprt/time.h>
64#include <VBox/VMMDev.h>
65#include <VBox/VBoxGuestLib.h>
66
67#ifdef RT_OS_WINDOWS
68#include <iprt/nt/nt-and-windows.h>
69# include <tlhelp32.h>
70# include <psapi.h>
71# include <winternl.h>
72#endif
73
74#include "VBoxServiceInternal.h"
75#include "VBoxServiceUtils.h"
76
77
78/*********************************************************************************************************************************
79* Header Files *
80*********************************************************************************************************************************/
81typedef struct
82{
83 AVLPVNODECORE Core;
84#ifdef RT_OS_WINDOWS
85 HMODULE hModule;
86 char szFileVersion[16];
87 MODULEENTRY32 Info;
88#endif
89} VGSVCPGSHKNOWNMOD, *PVGSVCPGSHKNOWNMOD;
90
91
92/*********************************************************************************************************************************
93* Global Variables *
94*********************************************************************************************************************************/
95/** The semaphore we're blocking on. */
96static RTSEMEVENTMULTI g_PageSharingEvent = NIL_RTSEMEVENTMULTI;
97
98static PAVLPVNODECORE g_pKnownModuleTree = NULL;
99static uint64_t g_idSession = 0;
100
101
102/*********************************************************************************************************************************
103* Internal Functions *
104*********************************************************************************************************************************/
105static DECLCALLBACK(int) vgsvcPageSharingEmptyTreeCallback(PAVLPVNODECORE pNode, void *pvUser);
106
107
108#ifdef RT_OS_WINDOWS
109
110/**
111 * Registers a new module with the VMM
112 * @param pModule Module ptr
113 * @param fValidateMemory Validate/touch memory pages or not
114 */
115static void vgsvcPageSharingRegisterModule(PVGSVCPGSHKNOWNMOD pModule, bool fValidateMemory)
116{
117 VMMDEVSHAREDREGIONDESC aRegions[VMMDEVSHAREDREGIONDESC_MAX];
118 DWORD dwModuleSize = pModule->Info.modBaseSize;
119 BYTE *pBaseAddress = pModule->Info.modBaseAddr;
120
121 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule\n");
122
123 DWORD dwDummy;
124 DWORD cbVersion = GetFileVersionInfoSize(pModule->Info.szExePath, &dwDummy);
125 if (!cbVersion)
126 {
127 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: GetFileVersionInfoSize failed with %d\n", GetLastError());
128 return;
129 }
130 BYTE *pVersionInfo = (BYTE *)RTMemAllocZ(cbVersion);
131 if (!pVersionInfo)
132 return;
133
134 if (!GetFileVersionInfo(pModule->Info.szExePath, 0, cbVersion, pVersionInfo))
135 {
136 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: GetFileVersionInfo failed with %d\n", GetLastError());
137 goto end;
138 }
139
140 /* Fetch default code page. */
141 struct LANGANDCODEPAGE
142 {
143 WORD wLanguage;
144 WORD wCodePage;
145 } *lpTranslate;
146
147 UINT cbTranslate;
148 BOOL fRet = VerQueryValue(pVersionInfo, TEXT("\\VarFileInfo\\Translation"), (LPVOID *)&lpTranslate, &cbTranslate);
149 if ( !fRet
150 || cbTranslate < 4)
151 {
152 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: VerQueryValue failed with %d (cb=%d)\n", GetLastError(), cbTranslate);
153 goto end;
154 }
155
156 unsigned i;
157 UINT cbFileVersion;
158 char *pszFileVersion = NULL; /* Shut up MSC */
159 unsigned cTranslationBlocks = cbTranslate/sizeof(struct LANGANDCODEPAGE);
160
161 pModule->szFileVersion[0] = '\0';
162 for (i = 0; i < cTranslationBlocks; i++)
163 {
164 /* Fetch file version string. */
165 char szFileVersionLocation[256];
166
167/** @todo r=bird: Mixing ANSI and TCHAR crap again. This code is a mess. We
168 * always use the wide version of the API and convert to UTF-8/whatever. */
169
170 sprintf(szFileVersionLocation, TEXT("\\StringFileInfo\\%04x%04x\\FileVersion"), lpTranslate[i].wLanguage, lpTranslate[i].wCodePage);
171 fRet = VerQueryValue(pVersionInfo, szFileVersionLocation, (LPVOID *)&pszFileVersion, &cbFileVersion);
172 if (fRet)
173 {
174 RTStrCopy(pModule->szFileVersion, sizeof(pModule->szFileVersion), pszFileVersion);
175 break;
176 }
177 }
178 if (i == cTranslationBlocks)
179 {
180 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: no file version found!\n");
181 goto end;
182 }
183
184 unsigned idxRegion = 0;
185
186 if (fValidateMemory)
187 {
188 do
189 {
190 MEMORY_BASIC_INFORMATION MemInfo;
191 SIZE_T cbRet = VirtualQuery(pBaseAddress, &MemInfo, sizeof(MemInfo));
192 Assert(cbRet);
193 if (!cbRet)
194 {
195 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: VirtualQueryEx failed with %d\n", GetLastError());
196 break;
197 }
198
199 if ( MemInfo.State == MEM_COMMIT
200 && MemInfo.Type == MEM_IMAGE)
201 {
202 switch (MemInfo.Protect)
203 {
204 case PAGE_EXECUTE:
205 case PAGE_EXECUTE_READ:
206 case PAGE_READONLY:
207 {
208 char *pRegion = (char *)MemInfo.BaseAddress;
209
210 /* Skip the first region as it only contains the image file header. */
211 if (pRegion != (char *)pModule->Info.modBaseAddr)
212 {
213 /* Touch all pages. */
214 while ((uintptr_t)pRegion < (uintptr_t)MemInfo.BaseAddress + MemInfo.RegionSize)
215 {
216 /* Try to trick the optimizer to leave the page touching code in place. */
217 ASMProbeReadByte(pRegion);
218 pRegion += PAGE_SIZE;
219 }
220 }
221#ifdef RT_ARCH_X86
222 aRegions[idxRegion].GCRegionAddr = (RTGCPTR32)MemInfo.BaseAddress;
223#else
224 aRegions[idxRegion].GCRegionAddr = (RTGCPTR64)MemInfo.BaseAddress;
225#endif
226 aRegions[idxRegion].cbRegion = MemInfo.RegionSize;
227 idxRegion++;
228
229 break;
230 }
231
232 default:
233 break; /* ignore */
234 }
235 }
236
237 pBaseAddress = (BYTE *)MemInfo.BaseAddress + MemInfo.RegionSize;
238 if (dwModuleSize > MemInfo.RegionSize)
239 dwModuleSize -= MemInfo.RegionSize;
240 else
241 {
242 dwModuleSize = 0;
243 break;
244 }
245
246 if (idxRegion >= RT_ELEMENTS(aRegions))
247 break; /* out of room */
248 }
249 while (dwModuleSize);
250 }
251 else
252 {
253 /* We can't probe kernel memory ranges, so pretend it's one big region. */
254#ifdef RT_ARCH_X86
255 aRegions[idxRegion].GCRegionAddr = (RTGCPTR32)pBaseAddress;
256#else
257 aRegions[idxRegion].GCRegionAddr = (RTGCPTR64)pBaseAddress;
258#endif
259 aRegions[idxRegion].cbRegion = dwModuleSize;
260 idxRegion++;
261 }
262 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: VbglR3RegisterSharedModule %s %s base=%p size=%x cregions=%d\n", pModule->Info.szModule, pModule->szFileVersion, pModule->Info.modBaseAddr, pModule->Info.modBaseSize, idxRegion);
263 int rc = VbglR3RegisterSharedModule(pModule->Info.szModule, pModule->szFileVersion, (uintptr_t)pModule->Info.modBaseAddr,
264 pModule->Info.modBaseSize, idxRegion, aRegions);
265 if (RT_FAILURE(rc))
266 VGSvcVerbose(3, "vgsvcPageSharingRegisterModule: VbglR3RegisterSharedModule failed with %Rrc\n", rc);
267
268end:
269 RTMemFree(pVersionInfo);
270 return;
271}
272
273
274/**
275 * Inspect all loaded modules for the specified process
276 *
277 * @param dwProcessId Process id
278 * @param ppNewTree The module tree we're assembling from modules found
279 * in this process. Modules found are moved from
280 * g_pKnownModuleTree or created new.
281 */
282static void vgsvcPageSharingInspectModules(DWORD dwProcessId, PAVLPVNODECORE *ppNewTree)
283{
284 /* Get a list of all the modules in this process. */
285 HANDLE hProcess = OpenProcess(PROCESS_QUERY_INFORMATION, FALSE /* no child process handle inheritance */, dwProcessId);
286 if (hProcess == NULL)
287 {
288 VGSvcVerbose(3, "vgsvcPageSharingInspectModules: OpenProcess %x failed with %d\n", dwProcessId, GetLastError());
289 return;
290 }
291
292 HANDLE hSnapshot = g_pfnCreateToolhelp32Snapshot(TH32CS_SNAPMODULE, dwProcessId);
293 if (hSnapshot == INVALID_HANDLE_VALUE)
294 {
295 VGSvcVerbose(3, "vgsvcPageSharingInspectModules: CreateToolhelp32Snapshot failed with %d\n", GetLastError());
296 CloseHandle(hProcess);
297 return;
298 }
299
300 VGSvcVerbose(3, "vgsvcPageSharingInspectModules\n");
301
302 MODULEENTRY32 ModuleInfo;
303 BOOL bRet;
304
305 ModuleInfo.dwSize = sizeof(ModuleInfo);
306 bRet = g_pfnModule32First(hSnapshot, &ModuleInfo);
307 do
308 {
309 /** @todo when changing this make sure VBoxService.exe is excluded! */
310 char *pszDot = strrchr(ModuleInfo.szModule, '.');
311 if ( pszDot
312 && (pszDot[1] == 'e' || pszDot[1] == 'E'))
313 continue; /* ignore executables for now. */
314
315 /* Found it before? */
316 PAVLPVNODECORE pRec = RTAvlPVGet(ppNewTree, ModuleInfo.modBaseAddr);
317 if (!pRec)
318 {
319 pRec = RTAvlPVRemove(&g_pKnownModuleTree, ModuleInfo.modBaseAddr);
320 if (!pRec)
321 {
322 /* New module; register it. */
323 PVGSVCPGSHKNOWNMOD pModule = (PVGSVCPGSHKNOWNMOD)RTMemAllocZ(sizeof(*pModule));
324 Assert(pModule);
325 if (!pModule)
326 break;
327
328 pModule->Info = ModuleInfo;
329 pModule->Core.Key = ModuleInfo.modBaseAddr;
330 pModule->hModule = LoadLibraryEx(ModuleInfo.szExePath, 0, DONT_RESOLVE_DLL_REFERENCES);
331 if (pModule->hModule)
332 vgsvcPageSharingRegisterModule(pModule, true /* validate pages */);
333
334 VGSvcVerbose(3, "\n\n MODULE NAME: %s", ModuleInfo.szModule );
335 VGSvcVerbose(3, "\n executable = %s", ModuleInfo.szExePath );
336 VGSvcVerbose(3, "\n process ID = 0x%08X", ModuleInfo.th32ProcessID );
337 VGSvcVerbose(3, "\n base address = %#010p", (uintptr_t) ModuleInfo.modBaseAddr );
338 VGSvcVerbose(3, "\n base size = %d", ModuleInfo.modBaseSize );
339
340 pRec = &pModule->Core;
341 }
342 bool ret = RTAvlPVInsert(ppNewTree, pRec);
343 Assert(ret); NOREF(ret);
344 }
345 } while (g_pfnModule32Next(hSnapshot, &ModuleInfo));
346
347 CloseHandle(hSnapshot);
348 CloseHandle(hProcess);
349}
350
351
352/**
353 * Inspect all running processes for executables and dlls that might be worth sharing
354 * with other VMs.
355 *
356 */
357static void vgsvcPageSharingInspectGuest(void)
358{
359 VGSvcVerbose(3, "vgsvcPageSharingInspectGuest\n");
360 PAVLPVNODECORE pNewTree = NULL;
361
362 /*
363 * Check loaded modules for all running processes.
364 */
365 if ( g_pfnProcess32First
366 && g_pfnProcess32Next
367 && g_pfnModule32First
368 && g_pfnModule32Next
369 && g_pfnCreateToolhelp32Snapshot)
370 {
371 HANDLE hSnapshot = g_pfnCreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
372 if (hSnapshot == INVALID_HANDLE_VALUE)
373 {
374 VGSvcVerbose(3, "vgsvcPageSharingInspectGuest: CreateToolhelp32Snapshot failed with %d\n", GetLastError());
375 return;
376 }
377
378 DWORD const dwProcessId = GetCurrentProcessId();
379
380 PROCESSENTRY32 ProcessInfo;
381 ProcessInfo.dwSize = sizeof(ProcessInfo);
382 g_pfnProcess32First(hSnapshot, &ProcessInfo);
383
384 do
385 {
386 /* Skip our own process. */
387 if (ProcessInfo.th32ProcessID != dwProcessId)
388 vgsvcPageSharingInspectModules(ProcessInfo.th32ProcessID, &pNewTree);
389 }
390 while (g_pfnProcess32Next(hSnapshot, &ProcessInfo));
391
392 CloseHandle(hSnapshot);
393 }
394
395 /*
396 * Check all loaded kernel modules.
397 */
398 if (g_pfnZwQuerySystemInformation)
399 {
400 ULONG cbBuffer = 0;
401 PVOID pBuffer = NULL;
402 PRTL_PROCESS_MODULES pSystemModules;
403
404 NTSTATUS ret = g_pfnZwQuerySystemInformation(SystemModuleInformation, (PVOID)&cbBuffer, 0, &cbBuffer);
405 if (!cbBuffer)
406 {
407 VGSvcVerbose(1, "ZwQuerySystemInformation returned length 0\n");
408 goto skipkernelmodules;
409 }
410
411 pBuffer = RTMemAllocZ(cbBuffer);
412 if (!pBuffer)
413 goto skipkernelmodules;
414
415 ret = g_pfnZwQuerySystemInformation(SystemModuleInformation, pBuffer, cbBuffer, &cbBuffer);
416 if (ret != STATUS_SUCCESS)
417 {
418 VGSvcVerbose(1, "ZwQuerySystemInformation returned %x (1)\n", ret);
419 goto skipkernelmodules;
420 }
421
422 pSystemModules = (PRTL_PROCESS_MODULES)pBuffer;
423 for (unsigned i = 0; i < pSystemModules->NumberOfModules; i++)
424 {
425 VGSvcVerbose(4, "\n\n KERNEL MODULE NAME: %s", pSystemModules->Modules[i].FullPathName[pSystemModules->Modules[i].OffsetToFileName] );
426 VGSvcVerbose(4, "\n executable = %s", pSystemModules->Modules[i].FullPathName );
427 VGSvcVerbose(4, "\n flags = 0x%08X\n", pSystemModules->Modules[i].Flags);
428
429 /* User-mode modules seem to have no flags set; skip them as we detected them above. */
430 if (pSystemModules->Modules[i].Flags == 0)
431 continue;
432
433 /* Found it before? */
434 PAVLPVNODECORE pRec = RTAvlPVGet(&pNewTree, pSystemModules->Modules[i].ImageBase);
435 if (!pRec)
436 {
437 pRec = RTAvlPVRemove(&g_pKnownModuleTree, pSystemModules->Modules[i].ImageBase);
438 if (!pRec)
439 {
440 /* New module; register it. */
441 char szFullFilePath[512];
442 PVGSVCPGSHKNOWNMOD pModule = (PVGSVCPGSHKNOWNMOD)RTMemAllocZ(sizeof(*pModule));
443 Assert(pModule);
444 if (!pModule)
445 break;
446
447/** @todo FullPathName not an UTF-8 string is! An ANSI string it is. */
448 strcpy(pModule->Info.szModule,
449 (const char *)&pSystemModules->Modules[i].FullPathName[pSystemModules->Modules[i].OffsetToFileName]);
450 GetSystemDirectoryA(szFullFilePath, sizeof(szFullFilePath));
451
452 /* skip \Systemroot\system32 */
453 char *lpPath = strchr((char *)&pSystemModules->Modules[i].FullPathName[1], '\\');
454 if (!lpPath)
455 {
456 /* Seen just file names in XP; try to locate the file in the system32 and system32\drivers directories. */
457 strcat(szFullFilePath, "\\");
458 strcat(szFullFilePath, (const char *)pSystemModules->Modules[i].FullPathName);
459 VGSvcVerbose(3, "Unexpected kernel module name try %s\n", szFullFilePath);
460 if (RTFileExists(szFullFilePath) == false)
461 {
462 GetSystemDirectoryA(szFullFilePath, sizeof(szFullFilePath));
463 strcat(szFullFilePath, "\\drivers\\");
464 strcat(szFullFilePath, (const char *)pSystemModules->Modules[i].FullPathName);
465 VGSvcVerbose(3, "Unexpected kernel module name try %s\n", szFullFilePath);
466 if (RTFileExists(szFullFilePath) == false)
467 {
468 VGSvcVerbose(1, "Unexpected kernel module name %s\n", pSystemModules->Modules[i].FullPathName);
469 RTMemFree(pModule);
470 continue;
471 }
472 }
473 }
474 else
475 {
476 lpPath = strchr(lpPath + 1, '\\');
477 if (!lpPath)
478 {
479 VGSvcVerbose(1, "Unexpected kernel module name %s (2)\n", pSystemModules->Modules[i].FullPathName);
480 RTMemFree(pModule);
481 continue;
482 }
483
484 strcat(szFullFilePath, lpPath);
485 }
486
487 strcpy(pModule->Info.szExePath, szFullFilePath);
488 pModule->Info.modBaseAddr = (BYTE *)pSystemModules->Modules[i].ImageBase;
489 pModule->Info.modBaseSize = pSystemModules->Modules[i].ImageSize;
490
491 pModule->Core.Key = pSystemModules->Modules[i].ImageBase;
492 vgsvcPageSharingRegisterModule(pModule, false /* don't check memory pages */);
493
494 VGSvcVerbose(3, "\n\n KERNEL MODULE NAME: %s", pModule->Info.szModule );
495 VGSvcVerbose(3, "\n executable = %s", pModule->Info.szExePath );
496 VGSvcVerbose(3, "\n base address = %#010p", (uintptr_t)pModule->Info.modBaseAddr );
497 VGSvcVerbose(3, "\n flags = 0x%08X", pSystemModules->Modules[i].Flags);
498 VGSvcVerbose(3, "\n base size = %d", pModule->Info.modBaseSize );
499
500 pRec = &pModule->Core;
501 }
502 bool ret = RTAvlPVInsert(&pNewTree, pRec);
503 Assert(ret); NOREF(ret);
504 }
505 }
506skipkernelmodules:
507 if (pBuffer)
508 RTMemFree(pBuffer);
509 }
510
511 /* Delete leftover modules in the old tree. */
512 RTAvlPVDestroy(&g_pKnownModuleTree, vgsvcPageSharingEmptyTreeCallback, NULL);
513
514 /* Check all registered modules. */
515 VbglR3CheckSharedModules();
516
517 /* Activate new module tree. */
518 g_pKnownModuleTree = pNewTree;
519}
520
521
522/**
523 * RTAvlPVDestroy callback.
524 */
525static DECLCALLBACK(int) vgsvcPageSharingEmptyTreeCallback(PAVLPVNODECORE pNode, void *pvUser)
526{
527 PVGSVCPGSHKNOWNMOD pModule = (PVGSVCPGSHKNOWNMOD)pNode;
528 bool *pfUnregister = (bool *)pvUser;
529
530 VGSvcVerbose(3, "vgsvcPageSharingEmptyTreeCallback %s %s\n", pModule->Info.szModule, pModule->szFileVersion);
531
532 /* Dereference module in the hypervisor. */
533 if ( !pfUnregister
534 || *pfUnregister)
535 {
536 int rc = VbglR3UnregisterSharedModule(pModule->Info.szModule, pModule->szFileVersion,
537 (uintptr_t)pModule->Info.modBaseAddr, pModule->Info.modBaseSize);
538 AssertRC(rc);
539 }
540
541 if (pModule->hModule)
542 FreeLibrary(pModule->hModule);
543 RTMemFree(pNode);
544 return 0;
545}
546
547
548#else /* !RT_OS_WINDOWS */
549
550static void vgsvcPageSharingInspectGuest(void)
551{
552 /** @todo other platforms */
553}
554
555#endif /* !RT_OS_WINDOWS */
556
557/** @interface_method_impl{VBOXSERVICE,pfnInit} */
558static DECLCALLBACK(int) vgsvcPageSharingInit(void)
559{
560 VGSvcVerbose(3, "vgsvcPageSharingInit\n");
561
562 int rc = RTSemEventMultiCreate(&g_PageSharingEvent);
563 AssertRCReturn(rc, rc);
564
565#ifdef RT_OS_WINDOWS
566 rc = VbglR3GetSessionId(&g_idSession);
567 if (RT_FAILURE(rc))
568 {
569 if (rc == VERR_IO_GEN_FAILURE)
570 VGSvcVerbose(0, "PageSharing: Page sharing support is not available by the host\n");
571 else
572 VGSvcError("vgsvcPageSharingInit: Failed with rc=%Rrc\n", rc);
573
574 rc = VERR_SERVICE_DISABLED;
575
576 RTSemEventMultiDestroy(g_PageSharingEvent);
577 g_PageSharingEvent = NIL_RTSEMEVENTMULTI;
578
579 }
580#endif
581
582 return rc;
583}
584
585
586/**
587 * @interface_method_impl{VBOXSERVICE,pfnWorker}
588 */
589static DECLCALLBACK(int) vgsvcPageSharingWorker(bool volatile *pfShutdown)
590{
591 /*
592 * Tell the control thread that it can continue
593 * spawning services.
594 */
595 RTThreadUserSignal(RTThreadSelf());
596
597 /*
598 * Now enter the loop retrieving runtime data continuously.
599 */
600 for (;;)
601 {
602 bool fEnabled = VbglR3PageSharingIsEnabled();
603 VGSvcVerbose(3, "vgsvcPageSharingWorker: enabled=%d\n", fEnabled);
604
605 if (fEnabled)
606 vgsvcPageSharingInspectGuest();
607
608 /*
609 * Block for a minute.
610 *
611 * The event semaphore takes care of ignoring interruptions and it
612 * allows us to implement service wakeup later.
613 */
614 if (*pfShutdown)
615 break;
616 int rc = RTSemEventMultiWait(g_PageSharingEvent, 60000);
617 if (*pfShutdown)
618 break;
619 if (rc != VERR_TIMEOUT && RT_FAILURE(rc))
620 {
621 VGSvcError("vgsvcPageSharingWorker: RTSemEventMultiWait failed; rc=%Rrc\n", rc);
622 break;
623 }
624#ifdef RT_OS_WINDOWS
625 uint64_t idNewSession = g_idSession;
626 rc = VbglR3GetSessionId(&idNewSession);
627 AssertRC(rc);
628
629 if (idNewSession != g_idSession)
630 {
631 bool fUnregister = false;
632
633 VGSvcVerbose(3, "vgsvcPageSharingWorker: VM was restored!!\n");
634 /* The VM was restored, so reregister all modules the next time. */
635 RTAvlPVDestroy(&g_pKnownModuleTree, vgsvcPageSharingEmptyTreeCallback, &fUnregister);
636 g_pKnownModuleTree = NULL;
637
638 g_idSession = idNewSession;
639 }
640#endif
641 }
642
643 RTSemEventMultiDestroy(g_PageSharingEvent);
644 g_PageSharingEvent = NIL_RTSEMEVENTMULTI;
645
646 VGSvcVerbose(3, "vgsvcPageSharingWorker: finished thread\n");
647 return 0;
648}
649
650#ifdef RT_OS_WINDOWS
651
652/**
653 * This gets control when VBoxService is launched with "pagefusion" by
654 * vgsvcPageSharingWorkerProcess().
655 *
656 * @returns RTEXITCODE_SUCCESS.
657 *
658 * @remarks It won't normally return since the parent drops the shutdown hint
659 * via RTProcTerminate().
660 */
661RTEXITCODE VGSvcPageSharingWorkerChild(void)
662{
663 VGSvcVerbose(3, "vgsvcPageSharingInitFork\n");
664
665 bool fShutdown = false;
666 vgsvcPageSharingInit();
667 vgsvcPageSharingWorker(&fShutdown);
668
669 return RTEXITCODE_SUCCESS;
670}
671
672
673/**
674 * @interface_method_impl{VBOXSERVICE,pfnWorker}
675 */
676static DECLCALLBACK(int) vgsvcPageSharingWorkerProcess(bool volatile *pfShutdown)
677{
678 RTPROCESS hProcess = NIL_RTPROCESS;
679 int rc;
680
681 /*
682 * Tell the control thread that it can continue
683 * spawning services.
684 */
685 RTThreadUserSignal(RTThreadSelf());
686
687 /*
688 * Now enter the loop retrieving runtime data continuously.
689 */
690 for (;;)
691 {
692 bool fEnabled = VbglR3PageSharingIsEnabled();
693 VGSvcVerbose(3, "vgsvcPageSharingWorkerProcess: enabled=%d\n", fEnabled);
694
695 /*
696 * Start a 2nd VBoxService process to deal with page fusion as we do
697 * not wish to dummy load dlls into this process. (First load with
698 * DONT_RESOLVE_DLL_REFERENCES, 2nd normal -> dll init routines not called!)
699 */
700 if ( fEnabled
701 && hProcess == NIL_RTPROCESS)
702 {
703 char szExeName[256];
704 char *pszExeName = RTProcGetExecutablePath(szExeName, sizeof(szExeName));
705 if (pszExeName)
706 {
707 char const *papszArgs[3];
708 papszArgs[0] = pszExeName;
709 papszArgs[1] = "pagefusion";
710 papszArgs[2] = NULL;
711 rc = RTProcCreate(pszExeName, papszArgs, RTENV_DEFAULT, 0 /* normal child */, &hProcess);
712 if (RT_FAILURE(rc))
713 VGSvcError("vgsvcPageSharingWorkerProcess: RTProcCreate %s failed; rc=%Rrc\n", pszExeName, rc);
714 }
715 }
716
717 /*
718 * Block for a minute.
719 *
720 * The event semaphore takes care of ignoring interruptions and it
721 * allows us to implement service wakeup later.
722 */
723 if (*pfShutdown)
724 break;
725 rc = RTSemEventMultiWait(g_PageSharingEvent, 60000);
726 if (*pfShutdown)
727 break;
728 if (rc != VERR_TIMEOUT && RT_FAILURE(rc))
729 {
730 VGSvcError("vgsvcPageSharingWorkerProcess: RTSemEventMultiWait failed; rc=%Rrc\n", rc);
731 break;
732 }
733 }
734
735 if (hProcess != NIL_RTPROCESS)
736 RTProcTerminate(hProcess);
737
738 VGSvcVerbose(3, "vgsvcPageSharingWorkerProcess: finished thread\n");
739 return 0;
740}
741
742#endif /* RT_OS_WINDOWS */
743
744/**
745 * @interface_method_impl{VBOXSERVICE,pfnStop}
746 */
747static DECLCALLBACK(void) vgsvcPageSharingStop(void)
748{
749 RTSemEventMultiSignal(g_PageSharingEvent);
750}
751
752
753/**
754 * @interface_method_impl{VBOXSERVICE,pfnTerm}
755 */
756static DECLCALLBACK(void) vgsvcPageSharingTerm(void)
757{
758 if (g_PageSharingEvent != NIL_RTSEMEVENTMULTI)
759 {
760 RTSemEventMultiDestroy(g_PageSharingEvent);
761 g_PageSharingEvent = NIL_RTSEMEVENTMULTI;
762 }
763}
764
765
766/**
767 * The 'pagesharing' service description.
768 */
769VBOXSERVICE g_PageSharing =
770{
771 /* pszName. */
772 "pagesharing",
773 /* pszDescription. */
774 "Page Sharing",
775 /* pszUsage. */
776 NULL,
777 /* pszOptions. */
778 NULL,
779 /* methods */
780 VGSvcDefaultPreInit,
781 VGSvcDefaultOption,
782 vgsvcPageSharingInit,
783#ifdef RT_OS_WINDOWS
784 vgsvcPageSharingWorkerProcess,
785#else
786 vgsvcPageSharingWorker,
787#endif
788 vgsvcPageSharingStop,
789 vgsvcPageSharingTerm
790};
791
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