VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxGuest/VBoxGuest.cpp@ 20369

Last change on this file since 20369 was 20369, checked in by vboxsync, 16 years ago

Additions: BEGIN_DECLS -> RT_BEGIN_DECLS; END_DECLS -> RT_END_DECLS.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 63.7 KB
Line 
1/** @file
2 *
3 * VBoxGuest -- VirtualBox Win32 guest support driver
4 *
5 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
6 *
7 * This file is part of VirtualBox Open Source Edition (OSE), as
8 * available from http://www.virtualbox.org. This file is free software;
9 * you can redistribute it and/or modify it under the terms of the GNU
10 * General Public License (GPL) as published by the Free Software
11 * Foundation, in version 2 as it comes in the "COPYING" file of the
12 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
13 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
14 *
15 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
16 * Clara, CA 95054 USA or visit http://www.sun.com if you need
17 * additional information or have any questions.
18 */
19
20// enable backdoor logging
21//#define LOG_ENABLED
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#include "VBoxGuest_Internal.h"
27#ifdef TARGET_NT4
28#include "NTLegacy.h"
29#else
30#include "VBoxGuestPnP.h"
31#endif
32#include "Helper.h"
33#include <excpt.h>
34#include <VBox/err.h>
35#include <VBox/log.h>
36#include <iprt/assert.h>
37#include <iprt/asm.h>
38#include <iprt/mem.h>
39#include <stdio.h>
40#include <VBox/VBoxGuestLib.h>
41#include <VBoxGuestInternal.h>
42
43#ifdef TARGET_NT4
44/*
45 * XP DDK #defines ExFreePool to ExFreePoolWithTag. The latter does not exist
46 * on NT4, so... The same for ExAllocatePool.
47 */
48#undef ExAllocatePool
49#undef ExFreePool
50#endif
51
52/*******************************************************************************
53* Defined Constants And Macros *
54*******************************************************************************/
55
56
57/*******************************************************************************
58* Internal Functions *
59*******************************************************************************/
60extern "C"
61{
62static NTSTATUS VBoxGuestAddDevice(PDRIVER_OBJECT pDrvObj, PDEVICE_OBJECT pDevObj);
63static void VBoxGuestUnload(PDRIVER_OBJECT pDrvObj);
64static NTSTATUS VBoxGuestCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp);
65static NTSTATUS VBoxGuestClose(PDEVICE_OBJECT pDevObj, PIRP pIrp);
66static NTSTATUS VBoxGuestDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
67static NTSTATUS VBoxGuestSystemControl(PDEVICE_OBJECT pDevObj, PIRP pIrp);
68static NTSTATUS VBoxGuestShutdown(PDEVICE_OBJECT pDevObj, PIRP pIrp);
69static NTSTATUS VBoxGuestNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp);
70static VOID vboxWorkerThread(PVOID context);
71static VOID reserveHypervisorMemory(PVBOXGUESTDEVEXT pDevExt);
72static VOID vboxIdleThread(PVOID context);
73}
74
75#ifdef VBOX_WITH_HGCM
76DECLVBGL(void) VBoxHGCMCallback(VMMDevHGCMRequestHeader *pHeader, void *pvData, uint32_t u32Data);
77#endif
78
79/*******************************************************************************
80* Exported Functions *
81*******************************************************************************/
82RT_BEGIN_DECLS
83ULONG DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath);
84RT_END_DECLS
85
86#ifdef ALLOC_PRAGMA
87#pragma alloc_text (INIT, DriverEntry)
88#pragma alloc_text (PAGE, createThreads)
89#pragma alloc_text (PAGE, unreserveHypervisorMemory)
90#pragma alloc_text (PAGE, VBoxGuestAddDevice)
91#pragma alloc_text (PAGE, VBoxGuestUnload)
92#pragma alloc_text (PAGE, VBoxGuestCreate)
93#pragma alloc_text (PAGE, VBoxGuestClose)
94#pragma alloc_text (PAGE, VBoxGuestDeviceControl)
95#pragma alloc_text (PAGE, VBoxGuestShutdown)
96#pragma alloc_text (PAGE, VBoxGuestNotSupportedStub)
97/* Note: at least the isr handler should be in non-pageable memory! */
98/*#pragma alloc_text (PAGE, VBoxGuestDpcHandler)
99 #pragma alloc_text (PAGE, VBoxGuestIsrHandler) */
100#pragma alloc_text (PAGE, vboxWorkerThread)
101#pragma alloc_text (PAGE, reserveHypervisorMemory)
102#pragma alloc_text (PAGE, vboxIdleThread)
103#endif
104
105winVersion_t winVersion;
106
107/**
108 * Driver entry point.
109 *
110 * @returns appropriate status code.
111 * @param pDrvObj Pointer to driver object.
112 * @param pRegPath Registry base path.
113 */
114ULONG DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING pRegPath)
115{
116 NTSTATUS rc = STATUS_SUCCESS;
117
118 dprintf(("VBoxGuest::DriverEntry. Driver built: %s %s\n", __DATE__, __TIME__));
119
120 ULONG majorVersion;
121 ULONG minorVersion;
122 ULONG buildNumber;
123 PsGetVersion(&majorVersion, &minorVersion, &buildNumber, NULL);
124 dprintf(("VBoxGuest::DriverEntry: running on Windows NT version %d.%d, build %d\n", majorVersion, minorVersion, buildNumber));
125 switch (majorVersion)
126 {
127 case 6:
128 winVersion = WINVISTA;
129 break;
130 case 5:
131 switch (minorVersion)
132 {
133 case 2:
134 winVersion = WIN2K3;
135 break;
136 case 1:
137 winVersion = WINXP;
138 break;
139 case 0:
140 winVersion = WIN2K;
141 break;
142 default:
143 dprintf(("VBoxGuest::DriverEntry: unknown version of Windows, refusing!\n"));
144 return STATUS_DRIVER_UNABLE_TO_LOAD;
145 }
146 break;
147 case 4:
148 winVersion = WINNT4;
149 break;
150 default:
151 dprintf(("VBoxGuest::DriverEntry: NT4 required!\n"));
152 return STATUS_DRIVER_UNABLE_TO_LOAD;
153 }
154
155 /*
156 * Setup the driver entry points in pDrvObj.
157 */
158 pDrvObj->DriverUnload = VBoxGuestUnload;
159 pDrvObj->MajorFunction[IRP_MJ_CREATE] = VBoxGuestCreate;
160 pDrvObj->MajorFunction[IRP_MJ_CLOSE] = VBoxGuestClose;
161 pDrvObj->MajorFunction[IRP_MJ_DEVICE_CONTROL] = VBoxGuestDeviceControl;
162 pDrvObj->MajorFunction[IRP_MJ_INTERNAL_DEVICE_CONTROL] = VBoxGuestDeviceControl;
163 pDrvObj->MajorFunction[IRP_MJ_SHUTDOWN] = VBoxGuestShutdown;
164 pDrvObj->MajorFunction[IRP_MJ_READ] = VBoxGuestNotSupportedStub;
165 pDrvObj->MajorFunction[IRP_MJ_WRITE] = VBoxGuestNotSupportedStub;
166#ifdef TARGET_NT4
167 rc = ntCreateDevice(pDrvObj, NULL, pRegPath);
168#else
169 pDrvObj->MajorFunction[IRP_MJ_PNP] = VBoxGuestPnP;
170 pDrvObj->MajorFunction[IRP_MJ_POWER] = VBoxGuestPower;
171 pDrvObj->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = VBoxGuestSystemControl;
172 pDrvObj->DriverExtension->AddDevice = (PDRIVER_ADD_DEVICE)VBoxGuestAddDevice;
173#endif
174
175 dprintf(("VBoxGuest::DriverEntry returning %#x\n", rc));
176 return rc;
177}
178
179#ifndef TARGET_NT4
180/**
181 * Handle request from the Plug & Play subsystem
182 *
183 * @returns NT status code
184 * @param pDrvObj Driver object
185 * @param pDevObj Device object
186 */
187static NTSTATUS VBoxGuestAddDevice(PDRIVER_OBJECT pDrvObj, PDEVICE_OBJECT pDevObj)
188{
189 NTSTATUS rc;
190 dprintf(("VBoxGuest::VBoxGuestAddDevice\n"));
191
192 /*
193 * Create device.
194 */
195 PDEVICE_OBJECT deviceObject = NULL;
196 UNICODE_STRING devName;
197 RtlInitUnicodeString(&devName, VBOXGUEST_DEVICE_NAME_NT);
198 rc = IoCreateDevice(pDrvObj, sizeof(VBOXGUESTDEVEXT), &devName, FILE_DEVICE_UNKNOWN, 0, FALSE, &deviceObject);
199 if (!NT_SUCCESS(rc))
200 {
201 dprintf(("VBoxGuest::VBoxGuestAddDevice: IoCreateDevice failed with rc=%#x!\n", rc));
202 return rc;
203 }
204 UNICODE_STRING win32Name;
205 RtlInitUnicodeString(&win32Name, VBOXGUEST_DEVICE_NAME_DOS);
206 rc = IoCreateSymbolicLink(&win32Name, &devName);
207 if (!NT_SUCCESS(rc))
208 {
209 dprintf(("VBoxGuest::VBoxGuestAddDevice: IoCreateSymbolicLink failed with rc=%#x!\n", rc));
210 IoDeleteDevice(deviceObject);
211 return rc;
212 }
213
214 /*
215 * Setup the device extension.
216 */
217 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)deviceObject->DeviceExtension;
218 RtlZeroMemory(pDevExt, sizeof(VBOXGUESTDEVEXT));
219
220 pDevExt->deviceObject = deviceObject;
221 pDevExt->devState = STOPPED;
222
223 pDevExt->nextLowerDriver = IoAttachDeviceToDeviceStack(deviceObject, pDevObj);
224 if (pDevExt->nextLowerDriver == NULL)
225 {
226 dprintf(("VBoxGuest::VBoxGuestAddDevice: IoAttachDeviceToDeviceStack did not give a nextLowerDrive\n"));
227 IoDeleteSymbolicLink(&win32Name);
228 IoDeleteDevice(deviceObject);
229 return STATUS_DEVICE_NOT_CONNECTED;
230 }
231
232#ifdef VBOX_WITH_HGCM
233 int rc2 = RTSpinlockCreate(&pDevExt->SessionSpinlock);
234 if (RT_FAILURE(rc2))
235 {
236 dprintf(("VBoxGuest::VBoxGuestAddDevice: RTSpinlockCreate failed\n"));
237 IoDetachDevice(pDevExt->nextLowerDriver);
238 IoDeleteSymbolicLink(&win32Name);
239 IoDeleteDevice(deviceObject);
240 return STATUS_DRIVER_UNABLE_TO_LOAD;
241 }
242#endif
243
244#ifdef VBOX_WITH_GUEST_BUGCHECK_DETECTION
245 hlpRegisterBugCheckCallback(pDevExt); /* ignore failure! */
246#endif
247
248 /* Driver is ready now. */
249 deviceObject->Flags &= ~DO_DEVICE_INITIALIZING;
250
251 dprintf(("VBoxGuest::VBoxGuestAddDevice: returning with rc = 0x%x\n", rc));
252 return rc;
253}
254#endif
255
256
257/**
258 * Unload the driver.
259 *
260 * @param pDrvObj Driver object.
261 */
262void VBoxGuestUnload(PDRIVER_OBJECT pDrvObj)
263{
264 dprintf(("VBoxGuest::VBoxGuestUnload\n"));
265#ifdef TARGET_NT4
266 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDrvObj->DeviceObject->DeviceExtension;
267 unreserveHypervisorMemory(pDevExt);
268 if (pDevExt->workerThread)
269 {
270 dprintf(("VBoxGuest::VBoxGuestUnload: waiting for the worker thread to terminate...\n"));
271 pDevExt->stopThread = TRUE;
272 KeSetEvent(&pDevExt->workerThreadRequest, 0, FALSE);
273 KeWaitForSingleObject(pDevExt->workerThread,
274 Executive, KernelMode, FALSE, NULL);
275 dprintf(("VBoxGuest::VBoxGuestUnload: returned from KeWaitForSingleObject for worker thread\n"));
276 }
277 if (pDevExt->idleThread)
278 {
279 dprintf(("VBoxGuest::VBoxGuestUnload: waiting for the idle thread to terminate...\n"));
280 pDevExt->stopThread = TRUE;
281 KeWaitForSingleObject(pDevExt->idleThread,
282 Executive, KernelMode, FALSE, NULL);
283 dprintf(("VBoxGuest::VBoxGuestUnload: returned from KeWaitForSingleObject for idle thread\n"));
284 }
285
286 hlpVBoxUnmapVMMDevMemory (pDevExt);
287
288 VBoxCleanupMemBalloon(pDevExt);
289
290#ifdef VBOX_WITH_GUEST_BUGCHECK_DETECTION
291 hlpDeregisterBugCheckCallback(pDevExt); /* ignore failure! */
292#endif
293
294 /*
295 * I don't think it's possible to unload a driver which processes have
296 * opened, at least we'll blindly assume that here.
297 */
298 UNICODE_STRING win32Name;
299 RtlInitUnicodeString(&win32Name, VBOXGUEST_DEVICE_NAME_DOS);
300 NTSTATUS rc = IoDeleteSymbolicLink(&win32Name);
301
302#ifdef VBOX_WITH_HGCM
303 if (pDevExt->SessionSpinlock != NIL_RTSPINLOCK)
304 RTSpinlockDestroy(pDevExt->SessionSpinlock);
305#endif
306 IoDeleteDevice(pDrvObj->DeviceObject);
307#endif
308
309 dprintf(("VBoxGuest::VBoxGuestUnload: returning\n"));
310}
311
312/**
313 * Create (i.e. Open) file entry point.
314 *
315 * @param pDevObj Device object.
316 * @param pIrp Request packet.
317 */
318NTSTATUS VBoxGuestCreate(PDEVICE_OBJECT pDevObj, PIRP pIrp)
319{
320 dprintf(("VBoxGuest::VBoxGuestCreate\n"));
321
322 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
323 PFILE_OBJECT pFileObj = pStack->FileObject;
324 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDevObj->DeviceExtension;
325
326 /*
327 * We are not remotely similar to a directory...
328 * (But this is possible.)
329 */
330 if (pStack->Parameters.Create.Options & FILE_DIRECTORY_FILE)
331 {
332 dprintf(("VBoxGuest::VBoxGuestCreate: we're not a directory!\n"));
333 pIrp->IoStatus.Status = STATUS_NOT_A_DIRECTORY;
334 pIrp->IoStatus.Information = 0;
335 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
336 return STATUS_NOT_A_DIRECTORY;
337 }
338
339#ifdef VBOX_WITH_HGCM
340 if (pFileObj)
341 {
342 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)RTMemAllocZ(sizeof(*pSession));
343 if (RT_UNLIKELY(!pSession))
344 {
345 dprintf(("VBoxGuestCreate: no memory!\n"));
346 pIrp->IoStatus.Status = STATUS_NO_MEMORY;
347 pIrp->IoStatus.Information = 0;
348 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
349 return STATUS_NO_MEMORY;
350 }
351
352 pFileObj->FsContext = pSession;
353 dprintf(("VBoxGuestCreate: pDevExt=%p pFileObj=%p pSession=%p\n",
354 pDevExt, pFileObj, pFileObj->FsContext));
355 }
356#endif
357
358 NTSTATUS rcNt = pIrp->IoStatus.Status = STATUS_SUCCESS;
359 pIrp->IoStatus.Information = 0;
360 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
361
362 dprintf(("VBoxGuest::VBoxGuestCreate: returning 0x%x\n", rcNt));
363 return rcNt;
364}
365
366
367/**
368 * Close file entry point.
369 *
370 * @param pDevObj Device object.
371 * @param pIrp Request packet.
372 */
373NTSTATUS VBoxGuestClose(PDEVICE_OBJECT pDevObj, PIRP pIrp)
374{
375 dprintf(("VBoxGuest::VBoxGuestClose\n"));
376
377 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDevObj->DeviceExtension;
378 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
379 PFILE_OBJECT pFileObj = pStack->FileObject;
380 dprintf(("VBoxGuest::VBoxGuestClose: pDevExt=%p pFileObj=%p pSession=%p\n",
381 pDevExt, pFileObj, pFileObj->FsContext));
382
383#ifdef VBOX_WITH_HGCM
384 if (pFileObj)
385 {
386 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pFileObj->FsContext;
387 if (RT_UNLIKELY(!pSession))
388 {
389 dprintf(("VBoxGuestClose: no FsContext!\n"));
390 }
391 else
392 {
393 for (unsigned i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
394 if (pSession->aHGCMClientIds[i])
395 {
396 VBoxGuestHGCMDisconnectInfo Info;
397 Info.result = 0;
398 Info.u32ClientID = pSession->aHGCMClientIds[i];
399 pSession->aHGCMClientIds[i] = 0;
400 dprintf(("VBoxGuestClose: disconnecting HGCM client id %#RX32\n", Info.u32ClientID));
401 VbglHGCMDisconnect(&Info, VBoxHGCMCallback, pDevExt, RT_INDEFINITE_WAIT);
402 }
403 RTMemFree(pSession);
404 }
405 }
406#endif
407
408 pFileObj->FsContext = NULL;
409 pIrp->IoStatus.Information = 0;
410 pIrp->IoStatus.Status = STATUS_SUCCESS;
411 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
412
413 return STATUS_SUCCESS;
414}
415
416#ifdef VBOX_WITH_HGCM
417void VBoxHGCMCallbackWorker (VMMDevHGCMRequestHeader *pHeader, PVBOXGUESTDEVEXT pDevExt,
418 uint32_t u32Timeout, bool fInterruptible)
419{
420 /* Possible problem with request completion right between the fu32Flags check and KeWaitForSingleObject
421 * call; introduce a timeout to make sure we don't wait indefinitely.
422 */
423
424 LARGE_INTEGER timeout;
425 if (u32Timeout == RT_INDEFINITE_WAIT)
426 timeout.QuadPart = pDevExt->HGCMWaitTimeout.QuadPart;
427 else
428 {
429 timeout.QuadPart = u32Timeout;
430 timeout.QuadPart *= -10000; /* relative in 100ns units */
431 }
432 while ((pHeader->fu32Flags & VBOX_HGCM_REQ_DONE) == 0)
433 {
434 /* Specifying UserMode so killing the user process will abort the wait. */
435 NTSTATUS rc = KeWaitForSingleObject (&pDevExt->keventNotification, Executive,
436 UserMode,
437 fInterruptible ? TRUE : FALSE, /* Alertable */
438 &timeout
439 );
440 dprintf(("VBoxHGCMCallback: Wait returned %d fu32Flags=%x\n", rc, pHeader->fu32Flags));
441
442 if (rc == STATUS_TIMEOUT && u32Timeout == RT_INDEFINITE_WAIT)
443 continue;
444
445 if (rc != STATUS_WAIT_0)
446 {
447 dprintf(("VBoxHGCMCallback: The external event was signalled or the wait timed out or terminated rc = 0x%08X.\n", rc));
448 break;
449 }
450
451 dprintf(("VBoxHGCMCallback: fu32Flags = %08X\n", pHeader->fu32Flags));
452 }
453 return;
454}
455
456DECLVBGL(void) VBoxHGCMCallback (VMMDevHGCMRequestHeader *pHeader, void *pvData, uint32_t u32Data)
457{
458 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pvData;
459
460 dprintf(("VBoxHGCMCallback\n"));
461 VBoxHGCMCallbackWorker (pHeader, pDevExt, u32Data, false);
462}
463
464DECLVBGL(void) VBoxHGCMCallbackInterruptible (VMMDevHGCMRequestHeader *pHeader, void *pvData,
465 uint32_t u32Data)
466{
467 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pvData;
468
469 dprintf(("VBoxHGCMCallbackInterruptible\n"));
470 VBoxHGCMCallbackWorker (pHeader, pDevExt, u32Data, true);
471}
472
473NTSTATUS vboxHGCMVerifyIOBuffers (PIO_STACK_LOCATION pStack, unsigned cb)
474{
475 if (pStack->Parameters.DeviceIoControl.OutputBufferLength < cb)
476 {
477 dprintf(("VBoxGuest::vboxHGCMVerifyIOBuffers: OutputBufferLength %d < %d\n",
478 pStack->Parameters.DeviceIoControl.OutputBufferLength, cb));
479 return STATUS_INVALID_PARAMETER;
480 }
481
482 if (pStack->Parameters.DeviceIoControl.InputBufferLength < cb)
483 {
484 dprintf(("VBoxGuest::vboxHGCMVerifyIOBuffers: InputBufferLength %d < %d\n",
485 pStack->Parameters.DeviceIoControl.InputBufferLength, cb));
486 return STATUS_INVALID_PARAMETER;
487 }
488
489 return STATUS_SUCCESS;
490}
491
492#endif /* VBOX_WITH_HGCM */
493
494static bool IsPowerOfTwo (uint32_t val)
495{
496 return (val & (val - 1)) == 0;
497}
498
499static bool CtlGuestFilterMask (uint32_t u32OrMask, uint32_t u32NotMask)
500{
501 bool result = false;
502 VMMDevCtlGuestFilterMask *req;
503 int rc = VbglGRAlloc ((VMMDevRequestHeader **) &req, sizeof (*req),
504 VMMDevReq_CtlGuestFilterMask);
505
506 if (RT_SUCCESS (rc))
507 {
508 req->u32OrMask = u32OrMask;
509 req->u32NotMask = u32NotMask;
510
511 rc = VbglGRPerform (&req->header);
512 if (RT_FAILURE (rc) || RT_FAILURE (req->header.rc))
513 {
514 dprintf (("VBoxGuest::VBoxGuestDeviceControl: error issuing request to VMMDev! "
515 "rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
516 }
517 else
518 {
519 result = true;
520 }
521 VbglGRFree (&req->header);
522 }
523
524 return result;
525}
526
527#ifdef VBOX_WITH_MANAGEMENT
528static int VBoxGuestSetBalloonSize(PVBOXGUESTDEVEXT pDevExt, uint32_t u32BalloonSize)
529{
530 VMMDevChangeMemBalloon *req = NULL;
531 int rc = VINF_SUCCESS;
532
533 if (u32BalloonSize > pDevExt->MemBalloon.cMaxBalloons)
534 {
535 AssertMsgFailed(("VBoxGuestSetBalloonSize illegal balloon size %d (max=%d)\n", u32BalloonSize, pDevExt->MemBalloon.cMaxBalloons));
536 return VERR_INVALID_PARAMETER;
537 }
538
539 if (u32BalloonSize == pDevExt->MemBalloon.cBalloons)
540 return VINF_SUCCESS; /* nothing to do */
541
542 /* Allocate request packet */
543 rc = VbglGRAlloc((VMMDevRequestHeader **)&req, RT_OFFSETOF(VMMDevChangeMemBalloon, aPhysPage[VMMDEV_MEMORY_BALLOON_CHUNK_PAGES]), VMMDevReq_ChangeMemBalloon);
544 if (RT_FAILURE(rc))
545 return rc;
546
547 vmmdevInitRequest(&req->header, VMMDevReq_ChangeMemBalloon);
548
549 if (u32BalloonSize > pDevExt->MemBalloon.cBalloons)
550 {
551 /* inflate */
552 for (uint32_t i=pDevExt->MemBalloon.cBalloons;i<u32BalloonSize;i++)
553 {
554#ifndef TARGET_NT4
555 /*
556 * Use MmAllocatePagesForMdl to specify the range of physical addresses we wish to use.
557 */
558 PHYSICAL_ADDRESS Zero;
559 PHYSICAL_ADDRESS HighAddr;
560 Zero.QuadPart = 0;
561 HighAddr.QuadPart = _4G - 1;
562 PMDL pMdl = MmAllocatePagesForMdl(Zero, HighAddr, Zero, VMMDEV_MEMORY_BALLOON_CHUNK_SIZE);
563 if (pMdl)
564 {
565 if (MmGetMdlByteCount(pMdl) < VMMDEV_MEMORY_BALLOON_CHUNK_SIZE)
566 {
567 MmFreePagesFromMdl(pMdl);
568 ExFreePool(pMdl);
569 rc = VERR_NO_MEMORY;
570 goto end;
571 }
572 }
573#else
574 PVOID pvBalloon;
575 pvBalloon = ExAllocatePool(PagedPool, VMMDEV_MEMORY_BALLOON_CHUNK_SIZE);
576 if (!pvBalloon)
577 {
578 rc = VERR_NO_MEMORY;
579 goto end;
580 }
581
582 PMDL pMdl = IoAllocateMdl (pvBalloon, VMMDEV_MEMORY_BALLOON_CHUNK_SIZE, FALSE, FALSE, NULL);
583 if (pMdl == NULL)
584 {
585 rc = VERR_NO_MEMORY;
586 ExFreePool(pvBalloon);
587 AssertMsgFailed(("IoAllocateMdl %p %x failed!!\n", pvBalloon, VMMDEV_MEMORY_BALLOON_CHUNK_SIZE));
588 goto end;
589 }
590 else
591 {
592 __try {
593 /* Calls to MmProbeAndLockPages must be enclosed in a try/except block. */
594 MmProbeAndLockPages (pMdl, KernelMode, IoModifyAccess);
595 }
596 __except(EXCEPTION_EXECUTE_HANDLER)
597 {
598 dprintf(("MmProbeAndLockPages failed!\n"));
599 rc = VERR_NO_MEMORY;
600 IoFreeMdl (pMdl);
601 ExFreePool(pvBalloon);
602 goto end;
603 }
604 }
605#endif
606
607 PPFN_NUMBER pPageDesc = MmGetMdlPfnArray(pMdl);
608
609 /* Copy manually as RTGCPHYS is always 64 bits */
610 for (uint32_t j=0;j<VMMDEV_MEMORY_BALLOON_CHUNK_PAGES;j++)
611 req->aPhysPage[j] = pPageDesc[j];
612
613 req->header.size = RT_OFFSETOF(VMMDevChangeMemBalloon, aPhysPage[VMMDEV_MEMORY_BALLOON_CHUNK_PAGES]);
614 req->cPages = VMMDEV_MEMORY_BALLOON_CHUNK_PAGES;
615 req->fInflate = true;
616
617 rc = VbglGRPerform(&req->header);
618 if (RT_FAILURE(rc) || RT_FAILURE(req->header.rc))
619 {
620 dprintf(("VBoxGuest::VBoxGuestSetBalloonSize: error issuing request to VMMDev!"
621 "rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
622
623#ifndef TARGET_NT4
624 MmFreePagesFromMdl(pMdl);
625 ExFreePool(pMdl);
626#else
627 IoFreeMdl (pMdl);
628 ExFreePool(pvBalloon);
629#endif
630 goto end;
631 }
632 else
633 {
634#ifndef TARGET_NT4
635 dprintf(("VBoxGuest::VBoxGuestSetBalloonSize %d MB added chunk at %x\n", i, pMdl));
636#else
637 dprintf(("VBoxGuest::VBoxGuestSetBalloonSize %d MB added chunk at %x\n", i, pvBalloon));
638#endif
639 pDevExt->MemBalloon.paMdlMemBalloon[i] = pMdl;
640 pDevExt->MemBalloon.cBalloons++;
641 }
642 }
643 }
644 else
645 {
646 /* deflate */
647 for (uint32_t _i=pDevExt->MemBalloon.cBalloons;_i>u32BalloonSize;_i--)
648 {
649 uint32_t index = _i - 1;
650 PMDL pMdl = pDevExt->MemBalloon.paMdlMemBalloon[index];
651
652 Assert(pMdl);
653 if (pMdl)
654 {
655#ifdef TARGET_NT4
656 PVOID pvBalloon = MmGetMdlVirtualAddress(pMdl);
657#endif
658
659 PPFN_NUMBER pPageDesc = MmGetMdlPfnArray(pMdl);
660
661 /* Copy manually as RTGCPHYS is always 64 bits */
662 for (uint32_t j=0;j<VMMDEV_MEMORY_BALLOON_CHUNK_PAGES;j++)
663 req->aPhysPage[j] = pPageDesc[j];
664
665 req->header.size = RT_OFFSETOF(VMMDevChangeMemBalloon, aPhysPage[VMMDEV_MEMORY_BALLOON_CHUNK_PAGES]);
666 req->cPages = VMMDEV_MEMORY_BALLOON_CHUNK_PAGES;
667 req->fInflate = false;
668
669 rc = VbglGRPerform(&req->header);
670 if (RT_FAILURE(rc) || RT_FAILURE(req->header.rc))
671 {
672 AssertMsgFailed(("VBoxGuest::VBoxGuestSetBalloonSize: error issuing request to VMMDev! rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
673 break;
674 }
675
676 /* Free the ballooned memory */
677#ifndef TARGET_NT4
678 dprintf(("VBoxGuest::VBoxGuestSetBalloonSize %d MB free chunk at %x\n", index, pMdl));
679 MmFreePagesFromMdl(pMdl);
680 ExFreePool(pMdl);
681#else
682 dprintf(("VBoxGuest::VBoxGuestSetBalloonSize %d MB free chunk at %x\n", index, pvBalloon));
683 MmUnlockPages (pMdl);
684 IoFreeMdl (pMdl);
685 ExFreePool(pvBalloon);
686#endif
687
688 pDevExt->MemBalloon.paMdlMemBalloon[index] = NULL;
689 pDevExt->MemBalloon.cBalloons--;
690 }
691 }
692 }
693 Assert(pDevExt->MemBalloon.cBalloons <= pDevExt->MemBalloon.cMaxBalloons);
694
695end:
696 VbglGRFree(&req->header);
697 return rc;
698}
699
700static int VBoxGuestQueryMemoryBalloon(PVBOXGUESTDEVEXT pDevExt, ULONG *pMemBalloonSize)
701{
702 /* just perform the request */
703 VMMDevGetMemBalloonChangeRequest *req = NULL;
704
705 dprintf(("VBoxGuestQueryMemoryBalloon\n"));
706
707 int rc = VbglGRAlloc((VMMDevRequestHeader **)&req, sizeof(VMMDevGetMemBalloonChangeRequest), VMMDevReq_GetMemBalloonChangeRequest);
708 vmmdevInitRequest(&req->header, VMMDevReq_GetMemBalloonChangeRequest);
709 req->eventAck = VMMDEV_EVENT_BALLOON_CHANGE_REQUEST;
710
711 if (RT_SUCCESS(rc))
712 {
713 rc = VbglGRPerform(&req->header);
714
715 if (RT_FAILURE(rc) || RT_FAILURE(req->header.rc))
716 {
717 dprintf(("VBoxGuest::VBoxGuestDeviceControl VBOXGUEST_IOCTL_CTL_CHECK_BALLOON: error issuing request to VMMDev!"
718 "rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
719 }
720 else
721 {
722 if (!pDevExt->MemBalloon.paMdlMemBalloon)
723 {
724 pDevExt->MemBalloon.cMaxBalloons = req->u32PhysMemSize;
725 pDevExt->MemBalloon.paMdlMemBalloon = (PMDL *)ExAllocatePool(PagedPool, req->u32PhysMemSize * sizeof(PMDL));
726 Assert(pDevExt->MemBalloon.paMdlMemBalloon);
727 if (!pDevExt->MemBalloon.paMdlMemBalloon)
728 return VERR_NO_MEMORY;
729 }
730 Assert(pDevExt->MemBalloon.cMaxBalloons == req->u32PhysMemSize);
731
732 rc = VBoxGuestSetBalloonSize(pDevExt, req->u32BalloonSize);
733 /* ignore out of memory failures */
734 if (rc == VERR_NO_MEMORY)
735 rc = VINF_SUCCESS;
736
737 if (pMemBalloonSize)
738 *pMemBalloonSize = pDevExt->MemBalloon.cBalloons;
739 }
740
741 VbglGRFree(&req->header);
742 }
743 return rc;
744}
745#endif
746
747void VBoxInitMemBalloon(PVBOXGUESTDEVEXT pDevExt)
748{
749#ifdef VBOX_WITH_MANAGEMENT
750 ULONG dummy;
751
752 pDevExt->MemBalloon.cBalloons = 0;
753 pDevExt->MemBalloon.cMaxBalloons = 0;
754 pDevExt->MemBalloon.paMdlMemBalloon = NULL;
755
756 VBoxGuestQueryMemoryBalloon(pDevExt, &dummy);
757#endif
758}
759
760void VBoxCleanupMemBalloon(PVBOXGUESTDEVEXT pDevExt)
761{
762#ifdef VBOX_WITH_MANAGEMENT
763 if (pDevExt->MemBalloon.paMdlMemBalloon)
764 {
765 /* Clean up the memory balloon leftovers */
766 VBoxGuestSetBalloonSize(pDevExt, 0);
767 ExFreePool(pDevExt->MemBalloon.paMdlMemBalloon);
768 pDevExt->MemBalloon.paMdlMemBalloon = NULL;
769 }
770 Assert(pDevExt->MemBalloon.cBalloons == 0);
771#endif
772}
773
774/**
775 * Device I/O Control entry point.
776 *
777 * @param pDevObj Device object.
778 * @param pIrp Request packet.
779 */
780NTSTATUS VBoxGuestDeviceControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
781{
782 dprintf(("VBoxGuest::VBoxGuestDeviceControl\n"));
783
784 NTSTATUS Status = STATUS_SUCCESS;
785
786 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDevObj->DeviceExtension;
787
788 PIO_STACK_LOCATION pStack = IoGetCurrentIrpStackLocation(pIrp);
789
790 char *pBuf = (char *)pIrp->AssociatedIrp.SystemBuffer; /* all requests are buffered. */
791
792 unsigned cbOut = 0;
793
794 switch (pStack->Parameters.DeviceIoControl.IoControlCode)
795 {
796 case VBOXGUEST_IOCTL_GETVMMDEVPORT:
797 {
798 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_GETVMMDEVPORT\n"));
799
800 if (pStack->Parameters.DeviceIoControl.OutputBufferLength < sizeof (VBoxGuestPortInfo))
801 {
802 Status = STATUS_BUFFER_TOO_SMALL;
803 break;
804 }
805
806 VBoxGuestPortInfo *portInfo = (VBoxGuestPortInfo*)pBuf;
807
808 portInfo->portAddress = pDevExt->startPortAddress;
809 portInfo->pVMMDevMemory = pDevExt->pVMMDevMemory;
810
811 cbOut = sizeof(VBoxGuestPortInfo);
812
813 break;
814 }
815
816 case VBOXGUEST_IOCTL_WAITEVENT:
817 {
818 /* Need to be extended to support multiple waiters for an event,
819 * array of counters for each event, event mask is computed, each
820 * time a wait event is arrived.
821 */
822 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_WAITEVENT\n"));
823
824 if (pStack->Parameters.DeviceIoControl.OutputBufferLength < sizeof(VBoxGuestWaitEventInfo))
825 {
826 dprintf(("VBoxGuest::VBoxGuestDeviceControl: OutputBufferLength %d < sizeof(VBoxGuestWaitEventInfo)\n",
827 pStack->Parameters.DeviceIoControl.OutputBufferLength, sizeof(VBoxGuestWaitEventInfo)));
828 Status = STATUS_BUFFER_TOO_SMALL;
829 break;
830 }
831
832 if (pStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(VBoxGuestWaitEventInfo)) {
833 dprintf(("VBoxGuest::VBoxGuestDeviceControl: InputBufferLength %d < sizeof(VBoxGuestWaitEventInfo)\n",
834 pStack->Parameters.DeviceIoControl.InputBufferLength, sizeof(VBoxGuestWaitEventInfo)));
835 Status = STATUS_BUFFER_TOO_SMALL;
836 break;
837 }
838
839 VBoxGuestWaitEventInfo *eventInfo = (VBoxGuestWaitEventInfo *)pBuf;
840
841 if (!eventInfo->u32EventMaskIn || !IsPowerOfTwo (eventInfo->u32EventMaskIn)) {
842 dprintf (("VBoxGuest::VBoxGuestDeviceControl: Invalid input mask %#x\n",
843 eventInfo->u32EventMaskIn));
844 Status = STATUS_INVALID_PARAMETER;
845 break;
846 }
847
848 eventInfo->u32EventFlagsOut = 0;
849 int iBitOffset = ASMBitFirstSetU32 (eventInfo->u32EventMaskIn) - 1;
850 bool fTimeout = (eventInfo->u32TimeoutIn != ~0L);
851
852 dprintf (("mask = %d, iBitOffset = %d\n", iBitOffset, eventInfo->u32EventMaskIn));
853
854 /* Possible problem with request completion right between the pending event check and KeWaitForSingleObject
855 * call; introduce a timeout (if none was specified) to make sure we don't wait indefinitely.
856 */
857 LARGE_INTEGER timeout;
858 timeout.QuadPart = (fTimeout) ? eventInfo->u32TimeoutIn : 250;
859 timeout.QuadPart *= -10000;
860
861 NTSTATUS rc = STATUS_SUCCESS;
862
863 for (;;)
864 {
865 bool fEventPending = ASMAtomicBitTestAndClear(&pDevExt->u32Events, iBitOffset);
866 if (fEventPending)
867 {
868 eventInfo->u32EventFlagsOut = 1 << iBitOffset;
869 break;
870 }
871
872 rc = KeWaitForSingleObject (&pDevExt->keventNotification, Executive /** @todo UserRequest? */,
873 KernelMode, TRUE, &timeout);
874 dprintf(("VBOXGUEST_IOCTL_WAITEVENT: Wait returned %d -> event %x\n", rc, eventInfo->u32EventFlagsOut));
875
876 if (!fTimeout && rc == STATUS_TIMEOUT)
877 continue;
878
879 if (rc != STATUS_SUCCESS)
880 {
881 /* There was a timeout or wait was interrupted, etc. */
882 break;
883 }
884 }
885
886 dprintf (("u32EventFlagsOut = %#x\n", eventInfo->u32EventFlagsOut));
887 cbOut = sizeof(VBoxGuestWaitEventInfo);
888 break;
889 }
890
891 case VBOXGUEST_IOCTL_VMMREQUEST(0): /* (The size isn't relevant on NT.)*/
892 {
893 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_VMMREQUEST\n"));
894
895#define CHECK_SIZE(s) \
896 if (pStack->Parameters.DeviceIoControl.OutputBufferLength < s) \
897 { \
898 dprintf(("VBoxGuest::VBoxGuestDeviceControl: OutputBufferLength %d < %d\n", \
899 pStack->Parameters.DeviceIoControl.OutputBufferLength, s)); \
900 Status = STATUS_BUFFER_TOO_SMALL; \
901 break; \
902 } \
903 if (pStack->Parameters.DeviceIoControl.InputBufferLength < s) { \
904 dprintf(("VBoxGuest::VBoxGuestDeviceControl: InputBufferLength %d < %d\n", \
905 pStack->Parameters.DeviceIoControl.InputBufferLength, s)); \
906 Status = STATUS_BUFFER_TOO_SMALL; \
907 break; \
908 }
909
910 /* get the request header */
911 CHECK_SIZE(sizeof(VMMDevRequestHeader));
912 VMMDevRequestHeader *requestHeader = (VMMDevRequestHeader *)pBuf;
913 if (!vmmdevGetRequestSize(requestHeader->requestType))
914 {
915 Status = STATUS_INVALID_PARAMETER;
916 break;
917 }
918 /* make sure the buffers suit the request */
919 CHECK_SIZE(vmmdevGetRequestSize(requestHeader->requestType));
920
921 /* just perform the request */
922 VMMDevRequestHeader *req = NULL;
923
924 int rc = VbglGRAlloc((VMMDevRequestHeader **)&req, requestHeader->size, requestHeader->requestType);
925
926 if (RT_SUCCESS(rc))
927 {
928 /* copy the request information */
929 memcpy((void*)req, (void*)pBuf, requestHeader->size);
930 rc = VbglGRPerform(req);
931
932 if (RT_FAILURE(rc) || RT_FAILURE(req->rc))
933 {
934 dprintf(("VBoxGuest::VBoxGuestDeviceControl VBOXGUEST_IOCTL_VMMREQUEST: Error issuing request to VMMDev! "
935 "rc = %d, VMMDev rc = %Rrc\n", rc, req->rc));
936 Status = STATUS_UNSUCCESSFUL;
937 }
938 else
939 {
940 /* copy result */
941 memcpy((void*)pBuf, (void*)req, requestHeader->size);
942 cbOut = requestHeader->size;
943 }
944
945 VbglGRFree(req);
946 }
947 else
948 {
949 Status = STATUS_UNSUCCESSFUL;
950 }
951#undef CHECK_SIZE
952 break;
953 }
954
955 case VBOXGUEST_IOCTL_CTL_FILTER_MASK:
956 {
957 VBoxGuestFilterMaskInfo *maskInfo;
958
959 if (pStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(VBoxGuestFilterMaskInfo)) {
960 dprintf (("VBoxGuest::VBoxGuestDeviceControl: InputBufferLength %d < %d\n",
961 pStack->Parameters.DeviceIoControl.InputBufferLength,
962 sizeof (VBoxGuestFilterMaskInfo)));
963 Status = STATUS_BUFFER_TOO_SMALL;
964 break;
965
966 }
967
968 maskInfo = (VBoxGuestFilterMaskInfo *) pBuf;
969 if (!CtlGuestFilterMask (maskInfo->u32OrMask, maskInfo->u32NotMask))
970 {
971 Status = STATUS_UNSUCCESSFUL;
972 }
973 break;
974 }
975
976#ifdef VBOX_WITH_HGCM
977 /* HGCM offers blocking IOCTLSs just like waitevent and actually
978 * uses the same waiting code.
979 */
980#ifdef RT_ARCH_AMD64
981 case VBOXGUEST_IOCTL_HGCM_CONNECT_32:
982#endif /* RT_ARCH_AMD64 */
983 case VBOXGUEST_IOCTL_HGCM_CONNECT:
984 {
985 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_HGCM_CONNECT\n"));
986
987 if (pStack->Parameters.DeviceIoControl.OutputBufferLength != sizeof(VBoxGuestHGCMConnectInfo))
988 {
989 dprintf(("VBoxGuest::VBoxGuestDeviceControl: OutputBufferLength %d != sizeof(VBoxGuestHGCMConnectInfo) %d\n",
990 pStack->Parameters.DeviceIoControl.OutputBufferLength, sizeof(VBoxGuestHGCMConnectInfo)));
991 Status = STATUS_INVALID_PARAMETER;
992 break;
993 }
994
995 if (pStack->Parameters.DeviceIoControl.InputBufferLength != sizeof(VBoxGuestHGCMConnectInfo)) {
996 dprintf(("VBoxGuest::VBoxGuestDeviceControl: InputBufferLength %d != sizeof(VBoxGuestHGCMConnectInfo) %d\n",
997 pStack->Parameters.DeviceIoControl.InputBufferLength, sizeof(VBoxGuestHGCMConnectInfo)));
998 Status = STATUS_INVALID_PARAMETER;
999 break;
1000 }
1001
1002 VBoxGuestHGCMConnectInfo *ptr = (VBoxGuestHGCMConnectInfo *)pBuf;
1003
1004 /* If request will be processed asynchronously, execution will
1005 * go to VBoxHGCMCallback. There it will wait for the request event, signalled from IRQ.
1006 * On IRQ arrival, the VBoxHGCMCallback(s) will check the request memory and, if completion
1007 * flag is set, returns.
1008 */
1009
1010 dprintf(("a) ptr->u32ClientID = %d\n", ptr->u32ClientID));
1011
1012 int rc = VbglHGCMConnect (ptr, VBoxHGCMCallback, pDevExt, RT_INDEFINITE_WAIT);
1013
1014 dprintf(("b) ptr->u32ClientID = %d\n", ptr->u32ClientID));
1015
1016 if (RT_FAILURE(rc))
1017 {
1018 dprintf(("VBOXGUEST_IOCTL_HGCM_CONNECT: vbox rc = %Rrc\n", rc));
1019 Status = STATUS_UNSUCCESSFUL;
1020 }
1021 else
1022 {
1023 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1024
1025 if (RT_SUCCESS(ptr->result) && pStack->FileObject)
1026 {
1027 dprintf(("VBOXGUEST_IOCTL_HGCM_CONNECT: pDevExt=%p pFileObj=%p pSession=%p\n",
1028 pDevExt, pStack->FileObject, pStack->FileObject->FsContext));
1029
1030 /*
1031 * Append the client id to the client id table.
1032 * If the table has somehow become filled up, we'll disconnect the session.
1033 */
1034 unsigned i;
1035 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pStack->FileObject->FsContext;
1036 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
1037
1038 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
1039 for (i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
1040 if (!pSession->aHGCMClientIds[i])
1041 {
1042 pSession->aHGCMClientIds[i] = ptr->u32ClientID;
1043 break;
1044 }
1045 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
1046
1047 if (i >= RT_ELEMENTS(pSession->aHGCMClientIds))
1048 {
1049 static unsigned s_cErrors = 0;
1050 if (s_cErrors++ < 32)
1051 dprintf(("VBoxGuestCommonIOCtl: HGCM_CONNECT: too many HGCMConnect calls for one session!\n"));
1052
1053 VBoxGuestHGCMDisconnectInfo Info;
1054 Info.result = 0;
1055 Info.u32ClientID = ptr->u32ClientID;
1056 VbglHGCMDisconnect(&Info, VBoxHGCMCallback, pDevExt, RT_INDEFINITE_WAIT);
1057 Status = STATUS_UNSUCCESSFUL;
1058 break;
1059 }
1060 }
1061 else
1062 {
1063 /* @fixme, r=Leonid. I have no clue what to do in cases where
1064 * pStack->FileObject==NULL. Can't populate list of HGCM ID's...
1065 * But things worked before, so do nothing for now.
1066 */
1067 dprintf(("VBOXGUEST_IOCTL_HGCM_CONNECT: pDevExt=%p, pStack->FileObject=%p\n", pDevExt, pStack->FileObject));
1068 }
1069 }
1070
1071 } break;
1072
1073#ifdef RT_ARCH_AMD64
1074 case VBOXGUEST_IOCTL_HGCM_DISCONNECT_32:
1075#endif /* RT_ARCH_AMD64 */
1076 case VBOXGUEST_IOCTL_HGCM_DISCONNECT:
1077 {
1078 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_HGCM_DISCONNECT\n"));
1079
1080 if (pStack->Parameters.DeviceIoControl.OutputBufferLength != sizeof(VBoxGuestHGCMDisconnectInfo))
1081 {
1082 dprintf(("VBoxGuest::VBoxGuestDeviceControl: OutputBufferLength %d != sizeof(VBoxGuestHGCMDisconnectInfo) %d\n",
1083 pStack->Parameters.DeviceIoControl.OutputBufferLength, sizeof(VBoxGuestHGCMDisconnectInfo)));
1084 Status = STATUS_INVALID_PARAMETER;
1085 break;
1086 }
1087
1088 if (pStack->Parameters.DeviceIoControl.InputBufferLength != sizeof(VBoxGuestHGCMDisconnectInfo)) {
1089 dprintf(("VBoxGuest::VBoxGuestDeviceControl: InputBufferLength %d != sizeof(VBoxGuestHGCMDisconnectInfo) %d\n",
1090 pStack->Parameters.DeviceIoControl.InputBufferLength, sizeof(VBoxGuestHGCMDisconnectInfo)));
1091 Status = STATUS_INVALID_PARAMETER;
1092 break;
1093 }
1094
1095 VBoxGuestHGCMDisconnectInfo *ptr = (VBoxGuestHGCMDisconnectInfo *)pBuf;
1096
1097 uint32_t u32ClientId=0;
1098 unsigned i=0;
1099 PVBOXGUESTSESSION pSession=0;
1100 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
1101
1102 /* See comment in VBOXGUEST_IOCTL_HGCM_CONNECT */
1103 if (pStack->FileObject)
1104 {
1105 dprintf(("VBOXGUEST_IOCTL_HGCM_DISCONNECT: pDevExt=%p pFileObj=%p pSession=%p\n",
1106 pDevExt, pStack->FileObject, pStack->FileObject->FsContext));
1107
1108 u32ClientId = ptr->u32ClientID;
1109 pSession = (PVBOXGUESTSESSION)pStack->FileObject->FsContext;
1110
1111 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
1112 for (i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
1113 if (pSession->aHGCMClientIds[i] == u32ClientId)
1114 {
1115 pSession->aHGCMClientIds[i] = UINT32_MAX;
1116 break;
1117 }
1118 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
1119 if (i >= RT_ELEMENTS(pSession->aHGCMClientIds))
1120 {
1121 static unsigned s_cErrors = 0;
1122 if (s_cErrors++ > 32)
1123 dprintf(("VBoxGuestCommonIOCtl: HGCM_DISCONNECT: u32Client=%RX32\n", u32ClientId));
1124 Status = STATUS_INVALID_PARAMETER;
1125 break;
1126 }
1127 }
1128
1129 /* If request will be processed asynchronously, execution will
1130 * go to VBoxHGCMCallback. There it will wait for the request event, signalled from IRQ.
1131 * On IRQ arrival, the VBoxHGCMCallback(s) will check the request memory and, if completion
1132 * flag is set, returns.
1133 */
1134
1135 int rc = VbglHGCMDisconnect (ptr, VBoxHGCMCallback, pDevExt, RT_INDEFINITE_WAIT);
1136
1137 if (RT_FAILURE(rc))
1138 {
1139 dprintf(("VBOXGUEST_IOCTL_HGCM_DISCONNECT: vbox rc = %Rrc\n", rc));
1140 Status = STATUS_UNSUCCESSFUL;
1141 }
1142 else
1143 {
1144 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1145 }
1146
1147 if (pStack->FileObject)
1148 {
1149 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
1150 if (pSession->aHGCMClientIds[i] == UINT32_MAX)
1151 pSession->aHGCMClientIds[i] = RT_SUCCESS(rc) && RT_SUCCESS(ptr->result) ? 0 : u32ClientId;
1152 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
1153 }
1154 } break;
1155
1156#ifdef RT_ARCH_AMD64
1157 case VBOXGUEST_IOCTL_HGCM_CALL_32(0): /* (The size isn't relevant on NT.) */
1158 {
1159 /* A 32 bit application call. */
1160 int rc;
1161
1162 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_HGCM_CALL_32\n"));
1163
1164 Status = vboxHGCMVerifyIOBuffers (pStack,
1165 sizeof (VBoxGuestHGCMCallInfo));
1166
1167 if (Status != STATUS_SUCCESS)
1168 {
1169 dprintf(("VBoxGuest::VBoxGuestDeviceControl: invalid parameter. Status: %p\n", Status));
1170 break;
1171 }
1172
1173 /* @todo: Old guest OpenGL driver used the same IOCtl code for both 32 and 64 bit binaries.
1174 * This is a protection, and can be removed if there were no 64 bit driver.
1175 */
1176 if (!IoIs32bitProcess(pIrp))
1177 {
1178 Status = STATUS_UNSUCCESSFUL;
1179 break;
1180 }
1181
1182 VBoxGuestHGCMCallInfo *ptr = (VBoxGuestHGCMCallInfo *)pBuf;
1183
1184 rc = VbglHGCMCall32(ptr, VBoxHGCMCallback, pDevExt, RT_INDEFINITE_WAIT);
1185
1186 if (RT_FAILURE(rc))
1187 {
1188 dprintf(("VBOXGUEST_IOCTL_HGCM_CALL_32: vbox rc = %Rrc\n", rc));
1189 Status = STATUS_UNSUCCESSFUL;
1190 }
1191 else
1192 {
1193 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1194 }
1195
1196 } break;
1197#endif /* RT_ARCH_AMD64 */
1198
1199 case VBOXGUEST_IOCTL_HGCM_CALL(0): /* (The size isn't relevant on NT.) */
1200 {
1201 int rc;
1202
1203 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_HGCM_CALL\n"));
1204
1205 Status = vboxHGCMVerifyIOBuffers (pStack,
1206 sizeof (VBoxGuestHGCMCallInfo));
1207
1208 if (Status != STATUS_SUCCESS)
1209 {
1210 dprintf(("VBoxGuest::VBoxGuestDeviceControl: invalid parameter. Status: %p\n", Status));
1211 break;
1212 }
1213
1214 VBoxGuestHGCMCallInfo *ptr = (VBoxGuestHGCMCallInfo *)pBuf;
1215
1216 rc = VbglHGCMCall (ptr, VBoxHGCMCallback, pDevExt, RT_INDEFINITE_WAIT);
1217
1218 if (RT_FAILURE(rc))
1219 {
1220 dprintf(("VBOXGUEST_IOCTL_HGCM_CALL: vbox rc = %Rrc\n", rc));
1221 Status = STATUS_UNSUCCESSFUL;
1222 }
1223 else
1224 {
1225 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1226 }
1227
1228 } break;
1229
1230 case VBOXGUEST_IOCTL_HGCM_CALL_TIMED(0): /* (The size isn't relevant on NT.) */
1231 {
1232 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_HGCM_CALL_TIMED\n"));
1233
1234 Status = vboxHGCMVerifyIOBuffers (pStack,
1235 sizeof (VBoxGuestHGCMCallInfoTimed));
1236
1237 if (Status != STATUS_SUCCESS)
1238 {
1239 dprintf(("nvalid parameter. Status: %p\n", Status));
1240 break;
1241 }
1242
1243 VBoxGuestHGCMCallInfoTimed *pInfo = (VBoxGuestHGCMCallInfoTimed *)pBuf;
1244 VBoxGuestHGCMCallInfo *ptr = &pInfo->info;
1245
1246 int rc;
1247 if (pInfo->fInterruptible)
1248 {
1249 dprintf(("VBoxGuest::VBoxGuestDeviceControl: calling VBoxHGCMCall interruptible, timeout %lu ms\n",
1250 pInfo->u32Timeout));
1251 rc = VbglHGCMCall (ptr, VBoxHGCMCallbackInterruptible, pDevExt, pInfo->u32Timeout);
1252 }
1253 else
1254 {
1255 dprintf(("VBoxGuest::VBoxGuestDeviceControl: calling VBoxHGCMCall, timeout %lu ms\n",
1256 pInfo->u32Timeout));
1257 rc = VbglHGCMCall (ptr, VBoxHGCMCallback, pDevExt, pInfo->u32Timeout);
1258 }
1259
1260 if (RT_FAILURE(rc))
1261 {
1262 dprintf(("VBOXGUEST_IOCTL_HGCM_CALL_TIMED: vbox rc = %Rrc\n", rc));
1263 Status = STATUS_UNSUCCESSFUL;
1264 }
1265 else
1266 {
1267 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1268 }
1269
1270 } break;
1271#endif /* VBOX_WITH_HGCM */
1272
1273#ifdef VBOX_WITH_VRDP_SESSION_HANDLING
1274 case VBOXGUEST_IOCTL_ENABLE_VRDP_SESSION:
1275 {
1276 LogRel(("VRDP_SESSION: Enable. Currently: %sabled\n", pDevExt->fVRDPEnabled? "en": "dis"));
1277 if (!pDevExt->fVRDPEnabled)
1278 {
1279 KUSER_SHARED_DATA *pSharedUserData = (KUSER_SHARED_DATA *)KI_USER_SHARED_DATA;
1280
1281 pDevExt->fVRDPEnabled = TRUE;
1282 LogRel(("VRDP_SESSION: Current active console id: 0x%08X\n", pSharedUserData->ActiveConsoleId));
1283 pDevExt->ulOldActiveConsoleId = pSharedUserData->ActiveConsoleId;
1284 pSharedUserData->ActiveConsoleId = 2;
1285 }
1286 break;
1287 }
1288
1289 case VBOXGUEST_IOCTL_DISABLE_VRDP_SESSION:
1290 {
1291 LogRel(("VRDP_SESSION: Disable. Currently: %sabled\n", pDevExt->fVRDPEnabled? "en": "dis"));
1292 if (pDevExt->fVRDPEnabled)
1293 {
1294 KUSER_SHARED_DATA *pSharedUserData = (KUSER_SHARED_DATA *)KI_USER_SHARED_DATA;
1295
1296 pDevExt->fVRDPEnabled = FALSE;
1297 LogRel(("VRDP_SESSION: Current active console id: 0x%08X\n", pSharedUserData->ActiveConsoleId));
1298 pSharedUserData->ActiveConsoleId = pDevExt->ulOldActiveConsoleId;
1299 pDevExt->ulOldActiveConsoleId = 0;
1300 }
1301 break;
1302 }
1303#endif
1304
1305#ifdef VBOX_WITH_MANAGEMENT
1306 case VBOXGUEST_IOCTL_CTL_CHECK_BALLOON_MASK:
1307 {
1308 ULONG *pMemBalloonSize = (ULONG *) pBuf;
1309
1310 if (pStack->Parameters.DeviceIoControl.OutputBufferLength != sizeof(ULONG))
1311 {
1312 dprintf(("VBoxGuest::VBoxGuestDeviceControl: OutputBufferLength %d != sizeof(ULONG) %d\n",
1313 pStack->Parameters.DeviceIoControl.OutputBufferLength, sizeof(ULONG)));
1314 Status = STATUS_INVALID_PARAMETER;
1315 break;
1316 }
1317
1318 int rc = VBoxGuestQueryMemoryBalloon(pDevExt, pMemBalloonSize);
1319 if (RT_FAILURE(rc))
1320 {
1321 dprintf(("VBOXGUEST_IOCTL_CTL_CHECK_BALLOON: vbox rc = %Rrc\n", rc));
1322 Status = STATUS_UNSUCCESSFUL;
1323 }
1324 else
1325 {
1326 cbOut = pStack->Parameters.DeviceIoControl.OutputBufferLength;
1327 }
1328 break;
1329 }
1330#endif
1331
1332 case VBOXGUEST_IOCTL_LOG(0): /* The size isn't relevant on NT. */
1333 {
1334 /* Enable this only for debugging:
1335 dprintf(("VBoxGuest::VBoxGuestDeviceControl: VBOXGUEST_IOCTL_LOG %.*s\n", (int)pStack->Parameters.DeviceIoControl.InputBufferLength, pBuf));
1336 */
1337 LogRel(("%.*s", (int)pStack->Parameters.DeviceIoControl.InputBufferLength, pBuf));
1338 cbOut = 0;
1339 break;
1340 }
1341
1342 default:
1343 Status = STATUS_INVALID_PARAMETER;
1344 break;
1345 }
1346
1347 pIrp->IoStatus.Status = Status;
1348 pIrp->IoStatus.Information = cbOut;
1349
1350 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
1351
1352 dprintf(("VBoxGuest::VBoxGuestDeviceControl: returned cbOut=%d rc=%#x\n", cbOut, Status));
1353
1354 return Status;
1355}
1356
1357
1358/**
1359 * IRP_MJ_SYSTEM_CONTROL handler
1360 *
1361 * @returns NT status code
1362 * @param pDevObj Device object.
1363 * @param pIrp IRP.
1364 */
1365NTSTATUS VBoxGuestSystemControl(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1366{
1367 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDevObj->DeviceExtension;
1368
1369 dprintf(("VBoxGuest::VBoxGuestSystemControl\n"));
1370
1371 /* Always pass it on to the next driver. */
1372 IoSkipCurrentIrpStackLocation(pIrp);
1373
1374 return IoCallDriver(pDevExt->nextLowerDriver, pIrp);
1375}
1376
1377/**
1378 * IRP_MJ_SHUTDOWN handler
1379 *
1380 * @returns NT status code
1381 * @param pDevObj Device object.
1382 * @param pIrp IRP.
1383 */
1384NTSTATUS VBoxGuestShutdown(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1385{
1386 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDevObj->DeviceExtension;
1387
1388 dprintf(("VBoxGuest::VBoxGuestShutdown\n"));
1389
1390 if (pDevExt && pDevExt->powerStateRequest)
1391 {
1392 VMMDevPowerStateRequest *req = pDevExt->powerStateRequest;
1393
1394 req->header.requestType = VMMDevReq_SetPowerStatus;
1395 req->powerState = VMMDevPowerState_PowerOff;
1396
1397 int rc = VbglGRPerform (&req->header);
1398
1399 if (RT_FAILURE(rc) || RT_FAILURE(req->header.rc))
1400 {
1401 dprintf(("VBoxGuest::PowerStateRequest: error performing request to VMMDev."
1402 "rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
1403 }
1404 }
1405
1406 return STATUS_SUCCESS;
1407}
1408
1409/**
1410 * Stub function for functions we don't implemented.
1411 *
1412 * @returns STATUS_NOT_SUPPORTED
1413 * @param pDevObj Device object.
1414 * @param pIrp IRP.
1415 */
1416NTSTATUS VBoxGuestNotSupportedStub(PDEVICE_OBJECT pDevObj, PIRP pIrp)
1417{
1418 dprintf(("VBoxGuest::VBoxGuestNotSupportedStub\n"));
1419 pDevObj = pDevObj;
1420
1421 pIrp->IoStatus.Information = 0;
1422 pIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;
1423 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
1424
1425 return STATUS_NOT_SUPPORTED;
1426}
1427
1428/**
1429 * DPC handler
1430 *
1431 * @param dpc DPC descriptor.
1432 * @param pDevObj Device object.
1433 * @param irp Interrupt request packet.
1434 * @param context Context specific pointer.
1435 */
1436VOID VBoxGuestDpcHandler(PKDPC dpc, PDEVICE_OBJECT pDevObj,
1437 PIRP irp, PVOID context)
1438{
1439 /* Unblock handlers waiting for arrived events.
1440 *
1441 * Events are very low things, there is one event flag (1 or more bit)
1442 * for each event. Each event is processed by exactly one handler.
1443 *
1444 * Assume that we trust additions and that other drivers will
1445 * handle its respective events without trying to fetch all events.
1446 *
1447 * Anyway design assures that wrong event processing will affect only guest.
1448 *
1449 * Event handler calls VMMDev IOCTL for waiting an event.
1450 * It supplies event mask. IOCTL blocks on EventNotification.
1451 * Here we just signal an the EventNotification to all waiting
1452 * threads, the IOCTL handler analyzes events and either
1453 * return to caller or blocks again.
1454 *
1455 * If we do not have too many events this is a simple and good
1456 * approach. Other way is to have as many Event objects as the callers
1457 * and wake up only callers waiting for the specific event.
1458 *
1459 * Now with the 'wake up all' appoach we probably do not need the DPC
1460 * handler and can signal event directly from ISR.
1461 *
1462 */
1463
1464 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pDevObj->DeviceExtension;
1465
1466 dprintf(("VBoxGuest::VBoxGuestDpcHandler\n"));
1467
1468 KePulseEvent(&pDevExt->keventNotification, 0, FALSE);
1469
1470}
1471
1472/**
1473 * ISR handler
1474 *
1475 * @return BOOLEAN indicates whether the IRQ came from us (TRUE) or not (FALSE)
1476 * @param interrupt Interrupt that was triggered.
1477 * @param serviceContext Context specific pointer.
1478 */
1479BOOLEAN VBoxGuestIsrHandler(PKINTERRUPT interrupt, PVOID serviceContext)
1480{
1481 NTSTATUS rc;
1482 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)serviceContext;
1483 BOOLEAN fIRQTaken = FALSE;
1484
1485 dprintf(("VBoxGuest::VBoxGuestIsrHandler haveEvents = %d\n",
1486 pDevExt->pVMMDevMemory->V.V1_04.fHaveEvents));
1487
1488 /*
1489 * now we have to find out whether it was our IRQ. Read the event mask
1490 * from our device to see if there are any pending events
1491 */
1492 if (pDevExt->pVMMDevMemory->V.V1_04.fHaveEvents)
1493 {
1494 /* Acknowlegde events. */
1495 VMMDevEvents *req = pDevExt->irqAckEvents;
1496
1497 rc = VbglGRPerform (&req->header);
1498 if (RT_SUCCESS(rc) && RT_SUCCESS(req->header.rc))
1499 {
1500 dprintf(("VBoxGuest::VBoxGuestIsrHandler: acknowledge events succeeded %#x\n",
1501 req->events));
1502
1503 ASMAtomicOrU32((uint32_t *)&pDevExt->u32Events, req->events);
1504 IoRequestDpc(pDevExt->deviceObject, pDevExt->currentIrp, NULL);
1505 }
1506 else
1507 {
1508 /* This can't be actually. This is sign of a serious problem. */
1509 dprintf(("VBoxGuest::VBoxGuestIsrHandler: "
1510 "acknowledge events failed rc = %d, header rc = %d\n",
1511 rc, req->header.rc));
1512 }
1513
1514 /* Mark IRQ as taken, there were events for us. */
1515 fIRQTaken = TRUE;
1516 }
1517
1518 return fIRQTaken;
1519}
1520
1521/**
1522 * Worker thread to do periodic things such as notify other
1523 * drivers of events.
1524 *
1525 * @param pDevExt device extension pointer
1526 */
1527VOID vboxWorkerThread(PVOID context)
1528{
1529 PVBOXGUESTDEVEXT pDevExt;
1530
1531 pDevExt = (PVBOXGUESTDEVEXT)context;
1532 dprintf(("VBoxGuest::vboxWorkerThread entered\n"));
1533
1534 /* perform the hypervisor address space reservation */
1535 reserveHypervisorMemory(pDevExt);
1536
1537 do
1538 {
1539 /* Nothing to do here yet. */
1540
1541 /*
1542 * Go asleep unless we're supposed to terminate
1543 */
1544 if (!pDevExt->stopThread)
1545 {
1546 ULONG secWait = 60;
1547 dprintf(("VBoxGuest::vboxWorkerThread: waiting for %u seconds...\n", secWait));
1548 LARGE_INTEGER dueTime;
1549 dueTime.QuadPart = -10000 * 1000 * (int)secWait;
1550 if (KeWaitForSingleObject(&pDevExt->workerThreadRequest, Executive,
1551 KernelMode, FALSE, &dueTime) == STATUS_SUCCESS)
1552 {
1553 KeResetEvent(&pDevExt->workerThreadRequest);
1554 }
1555 }
1556 } while (!pDevExt->stopThread);
1557
1558 dprintf(("VBoxGuest::vboxWorkerThread: we've been asked to terminate!\n"));
1559
1560 if (pDevExt->workerThread)
1561 {
1562 ObDereferenceObject(pDevExt->workerThread);
1563 pDevExt->workerThread = NULL;
1564 }
1565 dprintf(("VBoxGuest::vboxWorkerThread: now really gone!\n"));
1566}
1567
1568/**
1569 * Create driver worker threads
1570 *
1571 * @returns NTSTATUS NT status code
1572 * @param pDevExt VBoxGuest device extension
1573 */
1574NTSTATUS createThreads(PVBOXGUESTDEVEXT pDevExt)
1575{
1576 NTSTATUS rc;
1577 HANDLE threadHandle;
1578 OBJECT_ATTRIBUTES objAttributes;
1579
1580 dprintf(("VBoxGuest::createThreads\n"));
1581
1582 // first setup the request semaphore
1583 KeInitializeEvent(&pDevExt->workerThreadRequest, SynchronizationEvent, FALSE);
1584
1585// the API has slightly changed after NT4
1586#ifdef TARGET_NT4
1587#ifdef OBJ_KERNEL_HANDLE
1588#undef OBJ_KERNEL_HANDLE
1589#endif
1590#define OBJ_KERNEL_HANDLE 0
1591#endif
1592
1593 /*
1594 * The worker thread
1595 */
1596 InitializeObjectAttributes(&objAttributes,
1597 NULL,
1598 OBJ_KERNEL_HANDLE,
1599 NULL,
1600 NULL);
1601
1602 rc = PsCreateSystemThread(&threadHandle,
1603 THREAD_ALL_ACCESS,
1604 &objAttributes,
1605 (HANDLE)0L,
1606 NULL,
1607 vboxWorkerThread,
1608 pDevExt);
1609 dprintf(("VBoxGuest::createThreads: PsCreateSystemThread for worker thread returned: 0x%x\n", rc));
1610 rc = ObReferenceObjectByHandle(threadHandle,
1611 THREAD_ALL_ACCESS,
1612 NULL,
1613 KernelMode,
1614 (PVOID*)&pDevExt->workerThread,
1615 NULL);
1616 ZwClose(threadHandle);
1617
1618 /*
1619 * The idle thread
1620 */
1621#if 0 /// @todo Windows "sees" that time is lost and reports 100% usage
1622 rc = PsCreateSystemThread(&threadHandle,
1623 THREAD_ALL_ACCESS,
1624 &objAttributes,
1625 (HANDLE)0L,
1626 NULL,
1627 vboxIdleThread,
1628 pDevExt);
1629 dprintf(("VBoxGuest::createThreads: PsCreateSystemThread for idle thread returned: 0x%x\n", rc));
1630 rc = ObReferenceObjectByHandle(threadHandle,
1631 THREAD_ALL_ACCESS,
1632 NULL,
1633 KernelMode,
1634 (PVOID*)&pDevExt->idleThread,
1635 NULL);
1636 ZwClose(threadHandle);
1637#endif
1638
1639 return rc;
1640}
1641
1642/**
1643 * Helper routine to reserve address space for the hypervisor
1644 * and communicate its position.
1645 *
1646 * @param pDevExt Device extension structure.
1647 */
1648VOID reserveHypervisorMemory(PVBOXGUESTDEVEXT pDevExt)
1649{
1650 // @todo rc handling
1651 uint32_t hypervisorSize;
1652
1653 VMMDevReqHypervisorInfo *req = NULL;
1654
1655 int rc = VbglGRAlloc ((VMMDevRequestHeader **)&req, sizeof (VMMDevReqHypervisorInfo), VMMDevReq_GetHypervisorInfo);
1656
1657 if (RT_SUCCESS(rc))
1658 {
1659 req->hypervisorStart = 0;
1660 req->hypervisorSize = 0;
1661
1662 rc = VbglGRPerform (&req->header);
1663
1664 if (RT_SUCCESS(rc) && RT_SUCCESS(req->header.rc))
1665 {
1666 hypervisorSize = req->hypervisorSize;
1667
1668 if (!hypervisorSize)
1669 {
1670 dprintf(("VBoxGuest::reserveHypervisorMemory: host returned 0, not doing anything\n"));
1671 return;
1672 }
1673
1674 dprintf(("VBoxGuest::reserveHypervisorMemory: host wants %u bytes of hypervisor address space\n", hypervisorSize));
1675
1676 // Map fictive physical memory into the kernel address space to reserve virtual
1677 // address space. This API does not perform any checks but just allocate the
1678 // PTEs (which we don't really need/want but there isn't any other clean method).
1679 // The hypervisor only likes 4MB aligned virtual addresses, so we have to allocate
1680 // 4MB more than we are actually supposed to in order to guarantee that. Maybe we
1681 // can come up with a less lavish algorithm lateron.
1682 PHYSICAL_ADDRESS physAddr;
1683 physAddr.QuadPart = HYPERVISOR_PHYSICAL_START;
1684 pDevExt->hypervisorMappingSize = hypervisorSize + 0x400000;
1685 pDevExt->hypervisorMapping = MmMapIoSpace(physAddr,
1686 pDevExt->hypervisorMappingSize,
1687 MmNonCached);
1688 if (!pDevExt->hypervisorMapping)
1689 {
1690 dprintf(("VBoxGuest::reserveHypervisorMemory: MmMapIoSpace returned NULL!\n"));
1691 return;
1692 }
1693
1694 dprintf(("VBoxGuest::reserveHypervisorMemory: MmMapIoSpace returned %p\n", pDevExt->hypervisorMapping));
1695 dprintf(("VBoxGuest::reserveHypervisorMemory: communicating %p to host\n",
1696 RT_ALIGN_P(pDevExt->hypervisorMapping, 0x400000)));
1697
1698 /* align at 4MB */
1699 req->hypervisorStart = (uintptr_t)RT_ALIGN_P(pDevExt->hypervisorMapping, 0x400000);
1700
1701 req->header.requestType = VMMDevReq_SetHypervisorInfo;
1702 req->header.rc = VERR_GENERAL_FAILURE;
1703
1704 /* issue request */
1705 rc = VbglGRPerform (&req->header);
1706
1707 if (RT_FAILURE(rc) || RT_FAILURE(req->header.rc))
1708 {
1709 dprintf(("VBoxGuest::reserveHypervisorMemory: error communicating physical address to VMMDev!"
1710 "rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
1711 }
1712 }
1713 else
1714 {
1715 dprintf(("VBoxGuest::reserveHypervisorMemory: request failed with rc %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
1716 }
1717 VbglGRFree (&req->header);
1718 }
1719
1720 return;
1721}
1722
1723/**
1724 * Helper function to unregister a virtual address space mapping
1725 *
1726 * @param pDevExt Device extension
1727 */
1728VOID unreserveHypervisorMemory(PVBOXGUESTDEVEXT pDevExt)
1729{
1730 VMMDevReqHypervisorInfo *req = NULL;
1731
1732 int rc = VbglGRAlloc ((VMMDevRequestHeader **)&req, sizeof (VMMDevReqHypervisorInfo), VMMDevReq_SetHypervisorInfo);
1733
1734 if (RT_SUCCESS(rc))
1735 {
1736 /* tell the hypervisor that the mapping is no longer available */
1737
1738 req->hypervisorStart = 0;
1739 req->hypervisorSize = 0;
1740
1741 rc = VbglGRPerform (&req->header);
1742
1743 if (RT_FAILURE(rc) || RT_FAILURE(req->header.rc))
1744 {
1745 dprintf(("VBoxGuest::unreserveHypervisorMemory: error communicating physical address to VMMDev!"
1746 "rc = %d, VMMDev rc = %Rrc\n", rc, req->header.rc));
1747 }
1748
1749 VbglGRFree (&req->header);
1750 }
1751
1752 if (!pDevExt->hypervisorMapping)
1753 {
1754 dprintf(("VBoxGuest::unreserveHypervisorMemory: there is no mapping, returning\n"));
1755 return;
1756 }
1757
1758 // unmap fictive IO space
1759 MmUnmapIoSpace(pDevExt->hypervisorMapping, pDevExt->hypervisorMappingSize);
1760 dprintf(("VBoxGuest::unreserveHypervisorMemmory: done\n"));
1761}
1762
1763/**
1764 * Idle thread that runs at the lowest priority possible
1765 * and whenever scheduled, makes a VMMDev call to give up
1766 * timeslices. This is so prevent Windows from thinking that
1767 * nothing is happening on the machine and doing stupid things
1768 * that would steal time from other VMs it doesn't know of.
1769 *
1770 * @param pDevExt device extension pointer
1771 */
1772VOID vboxIdleThread(PVOID context)
1773{
1774 PVBOXGUESTDEVEXT pDevExt;
1775
1776 pDevExt = (PVBOXGUESTDEVEXT)context;
1777 dprintf(("VBoxGuest::vboxIdleThread entered\n"));
1778
1779 /* set priority as low as possible */
1780 KeSetPriorityThread(KeGetCurrentThread(), LOW_PRIORITY);
1781
1782 /* allocate VMMDev request structure */
1783 VMMDevReqIdle *req;
1784 int rc = VbglGRAlloc((VMMDevRequestHeader **)&req, sizeof (VMMDevReqHypervisorInfo), VMMDevReq_Idle);
1785 if (RT_FAILURE(rc))
1786 {
1787 dprintf(("VBoxGuest::vboxIdleThread: error %Rrc allocating request structure!\n"));
1788 return;
1789 }
1790
1791 do
1792 {
1793 //dprintf(("VBoxGuest: performing idle request..\n"));
1794 /* perform idle request */
1795 VbglGRPerform(&req->header);
1796
1797 } while (!pDevExt->stopThread);
1798
1799 VbglGRFree(&req->header);
1800
1801 dprintf(("VBoxGuest::vboxIdleThread leaving\n"));
1802}
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