VirtualBox

source: vbox/trunk/src/VBox/Devices/VMMDev/VMMDev.cpp@ 48786

Last change on this file since 48786 was 48070, checked in by vboxsync, 11 years ago

wddm,vboxtray: forward-port autoresize, multimon, and seamless fixes from 4.2 r87071, r87353, r87356, r87528, r87568, r87581, r87584, r87608, r87673, r87678, r87708, r87629, r87529; additional fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 147.9 KB
Line 
1/* $Id: VMMDev.cpp 48070 2013-08-26 18:13:22Z vboxsync $ */
2/** @file
3 * VMMDev - Guest <-> VMM/Host communication device.
4 */
5
6/*
7 * Copyright (C) 2006-2013 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/* Enable dev_vmm Log3 statements to get IRQ-related logging. */
23#define LOG_GROUP LOG_GROUP_DEV_VMM
24#include <VBox/VMMDev.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/log.h>
27#include <VBox/param.h>
28#include <iprt/path.h>
29#include <iprt/dir.h>
30#include <iprt/file.h>
31#include <VBox/vmm/pgm.h>
32#include <VBox/err.h>
33#include <VBox/vmm/vm.h> /* for VM_IS_EMT */
34#include <VBox/dbg.h>
35#include <VBox/version.h>
36
37#include <iprt/asm.h>
38#include <iprt/asm-amd64-x86.h>
39#include <iprt/assert.h>
40#include <iprt/buildconfig.h>
41#include <iprt/string.h>
42#include <iprt/time.h>
43#ifndef IN_RC
44# include <iprt/mem.h>
45#endif
46#ifdef IN_RING3
47# include <iprt/uuid.h>
48#endif
49
50#include "VMMDevState.h"
51#ifdef VBOX_WITH_HGCM
52# include "VMMDevHGCM.h"
53#endif
54#ifndef VBOX_WITHOUT_TESTING_FEATURES
55# include "VMMDevTesting.h"
56#endif
57
58
59/*******************************************************************************
60* Defined Constants And Macros *
61*******************************************************************************/
62#define VBOX_GUEST_INTERFACE_VERSION_1_03(s) \
63 ( RT_HIWORD((s)->guestInfo.interfaceVersion) == 1 \
64 && RT_LOWORD((s)->guestInfo.interfaceVersion) == 3 )
65
66#define VBOX_GUEST_INTERFACE_VERSION_OK(additionsVersion) \
67 ( RT_HIWORD(additionsVersion) == RT_HIWORD(VMMDEV_VERSION) \
68 && RT_LOWORD(additionsVersion) <= RT_LOWORD(VMMDEV_VERSION) )
69
70#define VBOX_GUEST_INTERFACE_VERSION_OLD(additionsVersion) \
71 ( (RT_HIWORD(additionsVersion) < RT_HIWORD(VMMDEV_VERSION) \
72 || ( RT_HIWORD(additionsVersion) == RT_HIWORD(VMMDEV_VERSION) \
73 && RT_LOWORD(additionsVersion) <= RT_LOWORD(VMMDEV_VERSION) ) )
74
75#define VBOX_GUEST_INTERFACE_VERSION_TOO_OLD(additionsVersion) \
76 ( RT_HIWORD(additionsVersion) < RT_HIWORD(VMMDEV_VERSION) )
77
78#define VBOX_GUEST_INTERFACE_VERSION_NEW(additionsVersion) \
79 ( RT_HIWORD(additionsVersion) > RT_HIWORD(VMMDEV_VERSION) \
80 || ( RT_HIWORD(additionsVersion) == RT_HIWORD(VMMDEV_VERSION) \
81 && RT_LOWORD(additionsVersion) > RT_LOWORD(VMMDEV_VERSION) ) )
82
83/** The saved state version. */
84#define VMMDEV_SAVED_STATE_VERSION 15
85/** The saved state version which is missing the guest facility statuses. */
86#define VMMDEV_SAVED_STATE_VERSION_MISSING_FACILITY_STATUSES 14
87/** The saved state version which is missing the guestInfo2 bits. */
88#define VMMDEV_SAVED_STATE_VERSION_MISSING_GUEST_INFO_2 13
89/** The saved state version used by VirtualBox 3.0.
90 * This doesn't have the config part. */
91#define VMMDEV_SAVED_STATE_VERSION_VBOX_30 11
92
93
94#ifndef VBOX_DEVICE_STRUCT_TESTCASE
95
96/** @page pg_vmmdev VMMDev
97 *
98 * Whenever host wants to inform guest about something an IRQ notification will
99 * be raised.
100 *
101 * VMMDev PDM interface will contain the guest notification method.
102 *
103 * There is a 32 bit event mask which will be read by guest on an interrupt. A
104 * non zero bit in the mask means that the specific event occurred and requires
105 * processing on guest side.
106 *
107 * After reading the event mask guest must issue a generic request
108 * AcknowlegdeEvents.
109 *
110 * IRQ line is set to 1 (request) if there are unprocessed events, that is the
111 * event mask is not zero.
112 *
113 * After receiving an interrupt and checking event mask, the guest must process
114 * events using the event specific mechanism.
115 *
116 * That is if mouse capabilities were changed, guest will use
117 * VMMDev_GetMouseStatus generic request.
118 *
119 * Event mask is only a set of flags indicating that guest must proceed with a
120 * procedure.
121 *
122 * Unsupported events are therefore ignored. The guest additions must inform
123 * host which events they want to receive, to avoid unnecessary IRQ processing.
124 * By default no events are signalled to guest.
125 *
126 * This seems to be fast method. It requires only one context switch for an
127 * event processing.
128 *
129 */
130
131
132/* -=-=-=-=- Misc Helpers -=-=-=-=- */
133
134
135/**
136 * Sets the IRQ (raise it or lower it) for 1.03 additions.
137 *
138 * @param pThis The VMMDev state.
139 * @thread Any.
140 * @remarks Must be called owning the critical section.
141 */
142static void vmmdevSetIRQ_Legacy(PVMMDEV pThis)
143{
144 if (!pThis->fu32AdditionsOk)
145 {
146 Log(("vmmdevSetIRQ: IRQ is not generated, guest has not yet reported to us.\n"));
147 return;
148 }
149
150 /* Filter unsupported events */
151 uint32_t u32EventFlags = pThis->u32HostEventFlags
152 & pThis->pVMMDevRAMR3->V.V1_03.u32GuestEventMask;
153
154 Log(("vmmdevSetIRQ: u32EventFlags=%#010x, u32HostEventFlags=%#010x, u32GuestEventMask=%#010x.\n",
155 u32EventFlags, pThis->u32HostEventFlags, pThis->pVMMDevRAMR3->V.V1_03.u32GuestEventMask));
156
157 /* Move event flags to VMMDev RAM */
158 pThis->pVMMDevRAMR3->V.V1_03.u32HostEvents = u32EventFlags;
159
160 uint32_t u32IRQLevel = 0;
161 if (u32EventFlags)
162 {
163 /* Clear host flags which will be delivered to guest. */
164 pThis->u32HostEventFlags &= ~u32EventFlags;
165 Log(("vmmdevSetIRQ: u32HostEventFlags=%#010x\n", pThis->u32HostEventFlags));
166 u32IRQLevel = 1;
167 }
168
169 /* Set IRQ level for pin 0 (see NoWait comment in vmmdevMaybeSetIRQ). */
170 /** @todo make IRQ pin configurable, at least a symbolic constant */
171 PPDMDEVINS pDevIns = pThis->pDevIns;
172 PDMDevHlpPCISetIrqNoWait(pDevIns, 0, u32IRQLevel);
173 Log(("vmmdevSetIRQ: IRQ set %d\n", u32IRQLevel));
174}
175
176/**
177 * Sets the IRQ if there are events to be delivered.
178 *
179 * @param pThis The VMMDev state.
180 * @thread Any.
181 * @remarks Must be called owning the critical section.
182 */
183static void vmmdevMaybeSetIRQ(PVMMDEV pThis)
184{
185 Log3(("vmmdevMaybeSetIRQ: u32HostEventFlags=%#010x, u32GuestFilterMask=%#010x.\n",
186 pThis->u32HostEventFlags, pThis->u32GuestFilterMask));
187
188 if (pThis->u32HostEventFlags & pThis->u32GuestFilterMask)
189 {
190 /*
191 * Note! No need to wait for the IRQs to be set (if we're not luck
192 * with the locks, etc). It is a notification about something,
193 * which has already happened.
194 */
195 pThis->pVMMDevRAMR3->V.V1_04.fHaveEvents = true;
196 PDMDevHlpPCISetIrqNoWait(pThis->pDevIns, 0, 1);
197 Log3(("vmmdevMaybeSetIRQ: IRQ set.\n"));
198 }
199}
200
201/**
202 * Notifies the guest about new events (@a fAddEvents).
203 *
204 * @param pThis The VMMDev state.
205 * @param fAddEvents New events to add.
206 * @thread Any.
207 * @remarks Must be called owning the critical section.
208 */
209static void vmmdevNotifyGuestWorker(PVMMDEV pThis, uint32_t fAddEvents)
210{
211 Log3(("vmmdevNotifyGuestWorker: fAddEvents=%#010x.\n", fAddEvents));
212 Assert(PDMCritSectIsOwner(&pThis->CritSect));
213
214 if (VBOX_GUEST_INTERFACE_VERSION_1_03(pThis))
215 {
216 Log3(("vmmdevNotifyGuestWorker: Old additions detected.\n"));
217
218 pThis->u32HostEventFlags |= fAddEvents;
219 vmmdevSetIRQ_Legacy(pThis);
220 }
221 else
222 {
223 Log3(("vmmdevNotifyGuestWorker: New additions detected.\n"));
224
225 if (!pThis->fu32AdditionsOk)
226 {
227 pThis->u32HostEventFlags |= fAddEvents;
228 Log(("vmmdevNotifyGuestWorker: IRQ is not generated, guest has not yet reported to us.\n"));
229 return;
230 }
231
232 const bool fHadEvents = (pThis->u32HostEventFlags & pThis->u32GuestFilterMask) != 0;
233
234 Log3(("vmmdevNotifyGuestWorker: fHadEvents=%d, u32HostEventFlags=%#010x, u32GuestFilterMask=%#010x.\n",
235 fHadEvents, pThis->u32HostEventFlags, pThis->u32GuestFilterMask));
236
237 pThis->u32HostEventFlags |= fAddEvents;
238
239 if (!fHadEvents)
240 vmmdevMaybeSetIRQ(pThis);
241 }
242}
243
244
245
246/* -=-=-=-=- Interfaces shared with VMMDevHGCM.cpp -=-=-=-=- */
247
248/**
249 * Notifies the guest about new events (@a fAddEvents).
250 *
251 * This is used by VMMDev.cpp as well as VMMDevHGCM.cpp.
252 *
253 * @param pThis The VMMDev state.
254 * @param fAddEvents New events to add.
255 * @thread Any.
256 */
257void VMMDevNotifyGuest(PVMMDEV pThis, uint32_t fAddEvents)
258{
259 Log3(("VMMDevNotifyGuest: fAddEvents=%#010x\n", fAddEvents));
260
261 /*
262 * Drop notifications if the VM is not running yet/anymore.
263 */
264 VMSTATE enmVMState = PDMDevHlpVMState(pThis->pDevIns);
265 if ( enmVMState != VMSTATE_RUNNING
266 && enmVMState != VMSTATE_RUNNING_LS)
267 return;
268
269 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
270 vmmdevNotifyGuestWorker(pThis, fAddEvents);
271 PDMCritSectLeave(&pThis->CritSect);
272}
273
274/**
275 * Code shared by VMMDevReq_CtlGuestFilterMask and HGCM for controlling the
276 * events the guest are interested in.
277 *
278 * @param pThis The VMMDev state.
279 * @param fOrMask Events to add (VMMDEV_EVENT_XXX). Pass 0 for no
280 * change.
281 * @param fNotMask Events to remove (VMMDEV_EVENT_XXX). Pass 0 for no
282 * change.
283 *
284 * @remarks When HGCM will automatically enable VMMDEV_EVENT_HGCM when the guest
285 * starts submitting HGCM requests. Otherwise, the events are
286 * controlled by the guest.
287 */
288void VMMDevCtlSetGuestFilterMask(PVMMDEV pThis, uint32_t fOrMask, uint32_t fNotMask)
289{
290 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
291
292 const bool fHadEvents = (pThis->u32HostEventFlags & pThis->u32GuestFilterMask) != 0;
293
294 Log(("VMMDevCtlSetGuestFilterMask: fOrMask=%#010x, u32NotMask=%#010x, fHadEvents=%d.\n", fOrMask, fNotMask, fHadEvents));
295 if (fHadEvents)
296 {
297 if (!pThis->fNewGuestFilterMask)
298 pThis->u32NewGuestFilterMask = pThis->u32GuestFilterMask;
299
300 pThis->u32NewGuestFilterMask |= fOrMask;
301 pThis->u32NewGuestFilterMask &= ~fNotMask;
302 pThis->fNewGuestFilterMask = true;
303 }
304 else
305 {
306 pThis->u32GuestFilterMask |= fOrMask;
307 pThis->u32GuestFilterMask &= ~fNotMask;
308 vmmdevMaybeSetIRQ(pThis);
309 }
310
311 PDMCritSectLeave(&pThis->CritSect);
312}
313
314
315
316/* -=-=-=-=- Request processing functions. -=-=-=-=- */
317
318/**
319 * Handles VMMDevReq_ReportGuestInfo.
320 *
321 * @returns VBox status code that the guest should see.
322 * @param pThis The VMMDev instance data.
323 * @param pRequestHeader The header of the request to handle.
324 */
325static int vmmdevReqHandler_ReportGuestInfo(PVMMDEV pThis, VMMDevRequestHeader *pRequestHeader)
326{
327 AssertMsgReturn(pRequestHeader->size == sizeof(VMMDevReportGuestInfo), ("%u\n", pRequestHeader->size), VERR_INVALID_PARAMETER);
328 VBoxGuestInfo const *pInfo = &((VMMDevReportGuestInfo *)pRequestHeader)->guestInfo;
329
330 if (memcmp(&pThis->guestInfo, pInfo, sizeof(*pInfo)) != 0)
331 {
332 /* Make a copy of supplied information. */
333 pThis->guestInfo = *pInfo;
334
335 /* Check additions interface version. */
336 pThis->fu32AdditionsOk = VBOX_GUEST_INTERFACE_VERSION_OK(pThis->guestInfo.interfaceVersion);
337
338 LogRel(("Guest Additions information report: Interface = 0x%08X osType = 0x%08X\n",
339 pThis->guestInfo.interfaceVersion, pThis->guestInfo.osType));
340
341 if (pThis->pDrv && pThis->pDrv->pfnUpdateGuestInfo)
342 pThis->pDrv->pfnUpdateGuestInfo(pThis->pDrv, &pThis->guestInfo);
343 }
344
345 if (!pThis->fu32AdditionsOk)
346 return VERR_VERSION_MISMATCH;
347
348 /* Clear our IRQ in case it was high for whatever reason. */
349 PDMDevHlpPCISetIrqNoWait(pThis->pDevIns, 0, 0);
350
351 return VINF_SUCCESS;
352}
353
354
355/**
356 * Validates a publisher tag.
357 *
358 * @returns true / false.
359 * @param pszTag Tag to validate.
360 */
361static bool vmmdevReqIsValidPublisherTag(const char *pszTag)
362{
363 /* Note! This character set is also found in Config.kmk. */
364 static char const s_szValidChars[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz()[]{}+-.,";
365
366 while (*pszTag != '\0')
367 {
368 if (!strchr(s_szValidChars, *pszTag))
369 return false;
370 pszTag++;
371 }
372 return true;
373}
374
375
376/**
377 * Validates a build tag.
378 *
379 * @returns true / false.
380 * @param pszTag Tag to validate.
381 */
382static bool vmmdevReqIsValidBuildTag(const char *pszTag)
383{
384 int cchPrefix;
385 if (!strncmp(pszTag, "RC", 2))
386 cchPrefix = 2;
387 else if (!strncmp(pszTag, "BETA", 4))
388 cchPrefix = 4;
389 else if (!strncmp(pszTag, "ALPHA", 5))
390 cchPrefix = 5;
391 else
392 return false;
393
394 if (pszTag[cchPrefix] == '\0')
395 return true;
396
397 uint8_t u8;
398 int rc = RTStrToUInt8Full(&pszTag[cchPrefix], 10, &u8);
399 return rc == VINF_SUCCESS;
400}
401
402
403/**
404 * Handles VMMDevReq_ReportGuestInfo2.
405 *
406 * @returns VBox status code that the guest should see.
407 * @param pThis The VMMDev instance data.
408 * @param pReqHdr The header of the request to handle.
409 */
410static int vmmdevReqHandler_ReportGuestInfo2(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
411{
412 AssertMsgReturn(pReqHdr->size == sizeof(VMMDevReportGuestInfo2), ("%u\n", pReqHdr->size), VERR_INVALID_PARAMETER);
413 VBoxGuestInfo2 const *pInfo2 = &((VMMDevReportGuestInfo2 *)pReqHdr)->guestInfo;
414
415 LogRel(("Guest Additions information report: Version %d.%d.%d r%d '%.*s'\n",
416 pInfo2->additionsMajor, pInfo2->additionsMinor, pInfo2->additionsBuild,
417 pInfo2->additionsRevision, sizeof(pInfo2->szName), pInfo2->szName));
418
419 /* The interface was introduced in 3.2 and will definitely not be
420 backported beyond 3.0 (bird). */
421 AssertMsgReturn(pInfo2->additionsMajor >= 3,
422 ("%u.%u.%u\n", pInfo2->additionsMajor, pInfo2->additionsMinor, pInfo2->additionsBuild),
423 VERR_INVALID_PARAMETER);
424
425 /* The version must fit in a full version compression. */
426 uint32_t uFullVersion = VBOX_FULL_VERSION_MAKE(pInfo2->additionsMajor, pInfo2->additionsMinor, pInfo2->additionsBuild);
427 AssertMsgReturn( VBOX_FULL_VERSION_GET_MAJOR(uFullVersion) == pInfo2->additionsMajor
428 && VBOX_FULL_VERSION_GET_MINOR(uFullVersion) == pInfo2->additionsMinor
429 && VBOX_FULL_VERSION_GET_BUILD(uFullVersion) == pInfo2->additionsBuild,
430 ("%u.%u.%u\n", pInfo2->additionsMajor, pInfo2->additionsMinor, pInfo2->additionsBuild),
431 VERR_OUT_OF_RANGE);
432
433 /*
434 * Validate the name.
435 * Be less strict towards older additions (< v4.1.50).
436 */
437 AssertCompile(sizeof(pThis->guestInfo2.szName) == sizeof(pInfo2->szName));
438 AssertReturn(memchr(pInfo2->szName, '\0', sizeof(pInfo2->szName)) != NULL, VERR_INVALID_PARAMETER);
439 const char *pszName = pInfo2->szName;
440
441 /* The version number which shouldn't be there. */
442 char szTmp[sizeof(pInfo2->szName)];
443 size_t cchStart = RTStrPrintf(szTmp, sizeof(szTmp), "%u.%u.%u", pInfo2->additionsMajor, pInfo2->additionsMinor, pInfo2->additionsBuild);
444 AssertMsgReturn(!strncmp(pszName, szTmp, cchStart), ("%s != %s\n", pszName, szTmp), VERR_INVALID_PARAMETER);
445 pszName += cchStart;
446
447 /* Now we can either have nothing or a build tag or/and a publisher tag. */
448 if (*pszName != '\0')
449 {
450 const char *pszRelaxedName = "";
451 bool const fStrict = pInfo2->additionsMajor > 4
452 || (pInfo2->additionsMajor == 4 && pInfo2->additionsMinor > 1)
453 || (pInfo2->additionsMajor == 4 && pInfo2->additionsMinor == 1 && pInfo2->additionsBuild >= 50);
454 bool fOk = false;
455 if (*pszName == '_')
456 {
457 pszName++;
458 strcpy(szTmp, pszName);
459 char *pszTag2 = strchr(szTmp, '_');
460 if (!pszTag2)
461 {
462 fOk = vmmdevReqIsValidBuildTag(szTmp)
463 || vmmdevReqIsValidPublisherTag(szTmp);
464 }
465 else
466 {
467 *pszTag2++ = '\0';
468 fOk = vmmdevReqIsValidBuildTag(szTmp);
469 if (fOk)
470 {
471 fOk = vmmdevReqIsValidPublisherTag(pszTag2);
472 if (!fOk)
473 pszRelaxedName = szTmp;
474 }
475 }
476 }
477
478 if (!fOk)
479 {
480 AssertLogRelMsgReturn(!fStrict, ("%s", pszName), VERR_INVALID_PARAMETER);
481
482 /* non-strict mode, just zap the extra stuff. */
483 LogRel(("ReportGuestInfo2: Ignoring unparsable version name bits: '%s' -> '%s'.\n", pszName, pszRelaxedName));
484 pszName = pszRelaxedName;
485 }
486 }
487
488 /*
489 * Save the info and tell Main or whoever is listening.
490 */
491 pThis->guestInfo2.uFullVersion = uFullVersion;
492 pThis->guestInfo2.uRevision = pInfo2->additionsRevision;
493 pThis->guestInfo2.fFeatures = pInfo2->additionsFeatures;
494 strcpy(pThis->guestInfo2.szName, pszName);
495
496 if (pThis->pDrv && pThis->pDrv->pfnUpdateGuestInfo2)
497 pThis->pDrv->pfnUpdateGuestInfo2(pThis->pDrv, uFullVersion, pszName, pInfo2->additionsRevision, pInfo2->additionsFeatures);
498
499 /* Clear our IRQ in case it was high for whatever reason. */
500 PDMDevHlpPCISetIrqNoWait (pThis->pDevIns, 0, 0);
501
502 return VINF_SUCCESS;
503}
504
505
506/**
507 * Allocates a new facility status entry, initializing it to inactive.
508 *
509 * @returns Pointer to a facility status entry on success, NULL on failure
510 * (table full).
511 * @param pThis The VMMDev instance data.
512 * @param uFacility The facility type code - VBoxGuestFacilityType.
513 * @param fFixed This is set when allocating the standard entries
514 * from the constructor.
515 * @param pTimeSpecNow Optionally giving the entry timestamp to use (ctor).
516 */
517static PVMMDEVFACILITYSTATUSENTRY
518vmmdevAllocFacilityStatusEntry(PVMMDEV pThis, uint32_t uFacility, bool fFixed, PCRTTIMESPEC pTimeSpecNow)
519{
520 /* If full, expunge one inactive entry. */
521 if (pThis->cFacilityStatuses == RT_ELEMENTS(pThis->aFacilityStatuses))
522 {
523 uint32_t i = pThis->cFacilityStatuses;
524 while (i-- > 0)
525 {
526 if ( pThis->aFacilityStatuses[i].uStatus == VBoxGuestFacilityStatus_Inactive
527 && !pThis->aFacilityStatuses[i].fFixed)
528 {
529 pThis->cFacilityStatuses--;
530 int cToMove = pThis->cFacilityStatuses - i;
531 if (cToMove)
532 memmove(&pThis->aFacilityStatuses[i], &pThis->aFacilityStatuses[i + 1],
533 cToMove * sizeof(pThis->aFacilityStatuses[i]));
534 RT_ZERO(pThis->aFacilityStatuses[pThis->cFacilityStatuses]);
535 break;
536 }
537 }
538
539 if (pThis->cFacilityStatuses == RT_ELEMENTS(pThis->aFacilityStatuses))
540 return NULL;
541 }
542
543 /* Find location in array (it's sorted). */
544 uint32_t i = pThis->cFacilityStatuses;
545 while (i-- > 0)
546 if (pThis->aFacilityStatuses[i].uFacility < uFacility)
547 break;
548 i++;
549
550 /* Move. */
551 int cToMove = pThis->cFacilityStatuses - i;
552 if (cToMove > 0)
553 memmove(&pThis->aFacilityStatuses[i + 1], &pThis->aFacilityStatuses[i],
554 cToMove * sizeof(pThis->aFacilityStatuses[i]));
555 pThis->cFacilityStatuses++;
556
557 /* Initialize. */
558 pThis->aFacilityStatuses[i].uFacility = uFacility;
559 pThis->aFacilityStatuses[i].uStatus = VBoxGuestFacilityStatus_Inactive;
560 pThis->aFacilityStatuses[i].fFixed = fFixed;
561 pThis->aFacilityStatuses[i].fPadding = 0;
562 pThis->aFacilityStatuses[i].fFlags = 0;
563 pThis->aFacilityStatuses[i].uPadding = 0;
564 if (pTimeSpecNow)
565 pThis->aFacilityStatuses[i].TimeSpecTS = *pTimeSpecNow;
566 else
567 RTTimeSpecSetNano(&pThis->aFacilityStatuses[i].TimeSpecTS, 0);
568
569 return &pThis->aFacilityStatuses[i];
570}
571
572
573/**
574 * Gets a facility status entry, allocating a new one if not already present.
575 *
576 * @returns Pointer to a facility status entry on success, NULL on failure
577 * (table full).
578 * @param pThis The VMMDev instance data.
579 * @param uFacility The facility type code - VBoxGuestFacilityType.
580 */
581static PVMMDEVFACILITYSTATUSENTRY vmmdevGetFacilityStatusEntry(PVMMDEV pThis, uint32_t uFacility)
582{
583 /** @todo change to binary search. */
584 uint32_t i = pThis->cFacilityStatuses;
585 while (i-- > 0)
586 {
587 if (pThis->aFacilityStatuses[i].uFacility == uFacility)
588 return &pThis->aFacilityStatuses[i];
589 if (pThis->aFacilityStatuses[i].uFacility < uFacility)
590 break;
591 }
592 return vmmdevAllocFacilityStatusEntry(pThis, uFacility, false /*fFixed*/, NULL);
593}
594
595
596/**
597 * Handles VMMDevReq_ReportGuestStatus.
598 *
599 * @returns VBox status code that the guest should see.
600 * @param pThis The VMMDev instance data.
601 * @param pReqHdr The header of the request to handle.
602 */
603static int vmmdevReqHandler_ReportGuestStatus(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
604{
605 /*
606 * Validate input.
607 */
608 AssertMsgReturn(pReqHdr->size == sizeof(VMMDevReportGuestStatus), ("%u\n", pReqHdr->size), VERR_INVALID_PARAMETER);
609 VBoxGuestStatus *pStatus = &((VMMDevReportGuestStatus *)pReqHdr)->guestStatus;
610 AssertMsgReturn( pStatus->facility > VBoxGuestFacilityType_Unknown
611 && pStatus->facility <= VBoxGuestFacilityType_All,
612 ("%d\n", pStatus->facility),
613 VERR_INVALID_PARAMETER);
614 AssertMsgReturn(pStatus->status == (VBoxGuestFacilityStatus)(uint16_t)pStatus->status,
615 ("%#x (%u)\n", pStatus->status, pStatus->status),
616 VERR_OUT_OF_RANGE);
617
618 /*
619 * Do the update.
620 */
621 RTTIMESPEC Now;
622 RTTimeNow(&Now);
623 if (pStatus->facility == VBoxGuestFacilityType_All)
624 {
625 uint32_t i = pThis->cFacilityStatuses;
626 while (i-- > 0)
627 {
628 pThis->aFacilityStatuses[i].TimeSpecTS = Now;
629 pThis->aFacilityStatuses[i].uStatus = (uint16_t)pStatus->status;
630 pThis->aFacilityStatuses[i].fFlags = pStatus->flags;
631 }
632 }
633 else
634 {
635 PVMMDEVFACILITYSTATUSENTRY pEntry = vmmdevGetFacilityStatusEntry(pThis, pStatus->facility);
636 if (!pEntry)
637 {
638 static int g_cLogEntries = 0;
639 if (g_cLogEntries++ < 10)
640 LogRel(("VMM: Facility table is full - facility=%u status=%u.\n", pStatus->facility, pStatus->status));
641 return VERR_OUT_OF_RESOURCES;
642 }
643
644 pEntry->TimeSpecTS = Now;
645 pEntry->uStatus = (uint16_t)pStatus->status; /** @todo r=andy uint16_t vs. 32-bit enum. */
646 pEntry->fFlags = pStatus->flags;
647 }
648
649 if (pThis->pDrv && pThis->pDrv->pfnUpdateGuestStatus)
650 pThis->pDrv->pfnUpdateGuestStatus(pThis->pDrv, pStatus->facility, pStatus->status, pStatus->flags, &Now);
651
652 return VINF_SUCCESS;
653}
654
655
656/**
657 * Handles VMMDevReq_ReportGuestUserState.
658 *
659 * @returns VBox status code that the guest should see.
660 * @param pThis The VMMDev instance data.
661 * @param pReqHdr The header of the request to handle.
662 */
663static int vmmdevReqHandler_ReportGuestUserState(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
664{
665 /*
666 * Validate input.
667 */
668 AssertMsgReturn(pReqHdr->size >= sizeof(VMMDevReportGuestUserState), ("%u\n", pReqHdr->size), VERR_INVALID_PARAMETER);
669 VBoxGuestUserStatus *pStatus = &((VMMDevReportGuestUserState *)pReqHdr)->status;
670
671 if ( pThis->pDrv
672 && pThis->pDrv->pfnUpdateGuestUserState)
673 {
674 AssertPtr(pStatus);
675
676 if ( pReqHdr->size > _2K
677 || pStatus->cbUser > 256
678 || pStatus->cbDomain > 256
679 || pStatus->cbDetails > _1K) /* Play safe. */
680 {
681 return VERR_INVALID_PARAMETER;
682 }
683
684 /* pyDynamic marks the beginning of the struct's dynamically
685 * allocated data area. */
686 uint8_t *pvDynamic = (uint8_t *)pStatus + RT_OFFSETOF(VBoxGuestUserStatus, szUser);
687 AssertPtr(pvDynamic);
688
689 if (!pStatus->cbUser) /* User name is required. */
690 return VERR_INVALID_PARAMETER;
691 const char *pszUser = (const char *)pvDynamic;
692 AssertPtrReturn(pszUser, VERR_INVALID_POINTER);
693
694 pvDynamic += pStatus->cbUser; /* Advance to next field. */
695 const char *pszDomain = pStatus->cbDomain
696 ? (const char *)pvDynamic : NULL;
697 /* Note: pszDomain can be NULL. */
698
699 pvDynamic += pStatus->cbDomain; /* Advance to next field. */
700 const uint8_t *puDetails = pStatus->cbDetails
701 ? pvDynamic : NULL;
702 /* Note: puDetails can be NULL. */
703
704 pThis->pDrv->pfnUpdateGuestUserState(pThis->pDrv, pszUser, pszDomain,
705 /* State */
706 (uint32_t)pStatus->state,
707 /* State details */
708 puDetails,
709 pStatus->cbDetails);
710 }
711
712 return VINF_SUCCESS;
713}
714
715
716/**
717 * Handles VMMDevReq_ReportGuestCapabilities.
718 *
719 * @returns VBox status code that the guest should see.
720 * @param pThis The VMMDev instance data.
721 * @param pReqHdr The header of the request to handle.
722 */
723static int vmmdevReqHandler_ReportGuestCapabilities(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
724{
725 VMMDevReqGuestCapabilities *pReq = (VMMDevReqGuestCapabilities *)pReqHdr;
726 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
727
728 /* Enable this automatically for guests using the old
729 request to report their capabilities. */
730 /** @todo change this when we next bump the interface version */
731 pReq->caps |= VMMDEV_GUEST_SUPPORTS_GRAPHICS;
732 if (pThis->guestCaps != pReq->caps)
733 {
734 /* make a copy of supplied information */
735 pThis->guestCaps = pReq->caps;
736
737 LogRel(("Guest Additions capability report: (0x%x) seamless: %s, hostWindowMapping: %s, graphics: %s\n",
738 pReq->caps,
739 pReq->caps & VMMDEV_GUEST_SUPPORTS_SEAMLESS ? "yes" : "no",
740 pReq->caps & VMMDEV_GUEST_SUPPORTS_GUEST_HOST_WINDOW_MAPPING ? "yes" : "no",
741 pReq->caps & VMMDEV_GUEST_SUPPORTS_GRAPHICS ? "yes" : "no"));
742
743 if (pThis->pDrv && pThis->pDrv->pfnUpdateGuestCapabilities)
744 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pReq->caps);
745 }
746 return VINF_SUCCESS;
747}
748
749
750/**
751 * Handles VMMDevReq_SetGuestCapabilities.
752 *
753 * @returns VBox status code that the guest should see.
754 * @param pThis The VMMDev instance data.
755 * @param pReqHdr The header of the request to handle.
756 */
757static int vmmdevReqHandler_SetGuestCapabilities(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
758{
759 VMMDevReqGuestCapabilities2 *pReq = (VMMDevReqGuestCapabilities2 *)pReqHdr;
760 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
761
762 uint32_t const fOldCaps = pThis->guestCaps; NOREF(fOldCaps);
763 pThis->guestCaps |= pReq->u32OrMask;
764 pThis->guestCaps &= ~pReq->u32NotMask;
765
766 LogRel(("Guest Additions capability report: (%#x -> %#x) seamless: %s, hostWindowMapping: %s, graphics: %s\n",
767 fOldCaps, pThis->guestCaps,
768 pThis->guestCaps & VMMDEV_GUEST_SUPPORTS_SEAMLESS ? "yes" : "no",
769 pThis->guestCaps & VMMDEV_GUEST_SUPPORTS_GUEST_HOST_WINDOW_MAPPING ? "yes" : "no",
770 pThis->guestCaps & VMMDEV_GUEST_SUPPORTS_GRAPHICS ? "yes" : "no"));
771
772 if (pThis->pDrv && pThis->pDrv->pfnUpdateGuestCapabilities)
773 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pThis->guestCaps);
774
775 return VINF_SUCCESS;
776}
777
778
779/**
780 * Handles VMMDevReq_GetMouseStatus.
781 *
782 * @returns VBox status code that the guest should see.
783 * @param pThis The VMMDev instance data.
784 * @param pReqHdr The header of the request to handle.
785 */
786static int vmmdevReqHandler_GetMouseStatus(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
787{
788 VMMDevReqMouseStatus *pReq = (VMMDevReqMouseStatus *)pReqHdr;
789 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
790
791 pReq->mouseFeatures = pThis->mouseCapabilities
792 & VMMDEV_MOUSE_MASK;
793 pReq->pointerXPos = pThis->mouseXAbs;
794 pReq->pointerYPos = pThis->mouseYAbs;
795 LogRel2(("VMMDevReq_GetMouseStatus: features = %#x, xAbs = %d, yAbs = %d\n",
796 pReq->mouseFeatures, pReq->pointerXPos, pReq->pointerYPos));
797 return VINF_SUCCESS;
798}
799
800
801/**
802 * Handles VMMDevReq_SetMouseStatus.
803 *
804 * @returns VBox status code that the guest should see.
805 * @param pThis The VMMDev instance data.
806 * @param pReqHdr The header of the request to handle.
807 */
808static int vmmdevReqHandler_SetMouseStatus(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
809{
810 VMMDevReqMouseStatus *pReq = (VMMDevReqMouseStatus *)pReqHdr;
811 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
812
813 LogRelFlowFunc(("mouseFeatures = %#x\n", pReq->mouseFeatures));
814
815 bool fNotify = false;
816 if ( (pReq->mouseFeatures & VMMDEV_MOUSE_NOTIFY_HOST_MASK)
817 != ( pThis->mouseCapabilities
818 & VMMDEV_MOUSE_NOTIFY_HOST_MASK))
819 fNotify = true;
820
821 pThis->mouseCapabilities &= ~VMMDEV_MOUSE_GUEST_MASK;
822 pThis->mouseCapabilities |= (pReq->mouseFeatures & VMMDEV_MOUSE_GUEST_MASK);
823
824 LogRelFlowFunc(("new host capabilities: %#x\n", pThis->mouseCapabilities));
825
826 /*
827 * Notify connector if something changed.
828 */
829 if (fNotify)
830 {
831 LogRelFlowFunc(("notifying connector\n"));
832 pThis->pDrv->pfnUpdateMouseCapabilities(pThis->pDrv, pThis->mouseCapabilities);
833 }
834
835 return VINF_SUCCESS;
836}
837
838
839/**
840 * Handles VMMDevReq_SetPointerShape.
841 *
842 * @returns VBox status code that the guest should see.
843 * @param pThis The VMMDev instance data.
844 * @param pReqHdr The header of the request to handle.
845 */
846static int vmmdevReqHandler_SetPointerShape(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
847{
848 VMMDevReqMousePointer *pReq = (VMMDevReqMousePointer *)pReqHdr;
849 if (pReq->header.size < sizeof(*pReq))
850 {
851 AssertMsg(pReq->header.size == 0x10028 && pReq->header.version == 10000, /* don't complain about legacy!!! */
852 ("VMMDev mouse shape structure has invalid size %d (%#x) version=%d!\n",
853 pReq->header.size, pReq->header.size, pReq->header.size, pReq->header.version));
854 return VERR_INVALID_PARAMETER;
855 }
856
857 bool fVisible = (pReq->fFlags & VBOX_MOUSE_POINTER_VISIBLE) != 0;
858 bool fAlpha = (pReq->fFlags & VBOX_MOUSE_POINTER_ALPHA) != 0;
859 bool fShape = (pReq->fFlags & VBOX_MOUSE_POINTER_SHAPE) != 0;
860
861 Log(("VMMDevReq_SetPointerShape: visible: %d, alpha: %d, shape = %d, width: %d, height: %d\n",
862 fVisible, fAlpha, fShape, pReq->width, pReq->height));
863
864 if (pReq->header.size == sizeof(VMMDevReqMousePointer))
865 {
866 /* The guest did not provide the shape actually. */
867 fShape = false;
868 }
869
870 /* forward call to driver */
871 if (fShape)
872 {
873 pThis->pDrv->pfnUpdatePointerShape(pThis->pDrv,
874 fVisible,
875 fAlpha,
876 pReq->xHot, pReq->yHot,
877 pReq->width, pReq->height,
878 pReq->pointerData);
879 }
880 else
881 {
882 pThis->pDrv->pfnUpdatePointerShape(pThis->pDrv,
883 fVisible,
884 0,
885 0, 0,
886 0, 0,
887 NULL);
888 }
889
890 pThis->fHostCursorRequested = fVisible;
891 return VINF_SUCCESS;
892}
893
894
895/**
896 * Handles VMMDevReq_GetHostTime.
897 *
898 * @returns VBox status code that the guest should see.
899 * @param pThis The VMMDev instance data.
900 * @param pReqHdr The header of the request to handle.
901 */
902static int vmmdevReqHandler_GetHostTime(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
903{
904 VMMDevReqHostTime *pReq = (VMMDevReqHostTime *)pReqHdr;
905 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
906
907 if (RT_UNLIKELY(pThis->fGetHostTimeDisabled))
908 return VERR_NOT_SUPPORTED;
909
910 RTTIMESPEC now;
911 pReq->time = RTTimeSpecGetMilli(PDMDevHlpTMUtcNow(pThis->pDevIns, &now));
912 return VINF_SUCCESS;
913}
914
915
916/**
917 * Handles VMMDevReq_GetHypervisorInfo.
918 *
919 * @returns VBox status code that the guest should see.
920 * @param pThis The VMMDev instance data.
921 * @param pReqHdr The header of the request to handle.
922 */
923static int vmmdevReqHandler_GetHypervisorInfo(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
924{
925 VMMDevReqHypervisorInfo *pReq = (VMMDevReqHypervisorInfo *)pReqHdr;
926 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
927
928 return PGMR3MappingsSize(PDMDevHlpGetVM(pThis->pDevIns), &pReq->hypervisorSize);
929}
930
931
932/**
933 * Handles VMMDevReq_SetHypervisorInfo.
934 *
935 * @returns VBox status code that the guest should see.
936 * @param pThis The VMMDev instance data.
937 * @param pReqHdr The header of the request to handle.
938 */
939static int vmmdevReqHandler_SetHypervisorInfo(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
940{
941 VMMDevReqHypervisorInfo *pReq = (VMMDevReqHypervisorInfo *)pReqHdr;
942 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
943
944 int rc;
945 PVM pVM = PDMDevHlpGetVM(pThis->pDevIns);
946 if (pReq->hypervisorStart == 0)
947 rc = PGMR3MappingsUnfix(pVM);
948 else
949 {
950 /* only if the client has queried the size before! */
951 uint32_t cbMappings;
952 rc = PGMR3MappingsSize(pVM, &cbMappings);
953 if (RT_SUCCESS(rc) && pReq->hypervisorSize == cbMappings)
954 {
955 /* new reservation */
956 rc = PGMR3MappingsFix(pVM, pReq->hypervisorStart, pReq->hypervisorSize);
957 LogRel(("Guest reported fixed hypervisor window at 0%010x (size = %#x, rc = %Rrc)\n",
958 pReq->hypervisorStart, pReq->hypervisorSize, rc));
959 }
960 else if (RT_FAILURE(rc))
961 rc = VERR_TRY_AGAIN;
962 }
963 return rc;
964}
965
966
967/**
968 * Handles VMMDevReq_RegisterPatchMemory.
969 *
970 * @returns VBox status code that the guest should see.
971 * @param pThis The VMMDev instance data.
972 * @param pReqHdr The header of the request to handle.
973 */
974static int vmmdevReqHandler_RegisterPatchMemory(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
975{
976 VMMDevReqPatchMemory *pReq = (VMMDevReqPatchMemory *)pReqHdr;
977 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
978
979 return VMMR3RegisterPatchMemory(PDMDevHlpGetVM(pThis->pDevIns), pReq->pPatchMem, pReq->cbPatchMem);
980}
981
982
983/**
984 * Handles VMMDevReq_DeregisterPatchMemory.
985 *
986 * @returns VBox status code that the guest should see.
987 * @param pThis The VMMDev instance data.
988 * @param pReqHdr The header of the request to handle.
989 */
990static int vmmdevReqHandler_DeregisterPatchMemory(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
991{
992 VMMDevReqPatchMemory *pReq = (VMMDevReqPatchMemory *)pReqHdr;
993 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
994
995 return VMMR3DeregisterPatchMemory(PDMDevHlpGetVM(pThis->pDevIns), pReq->pPatchMem, pReq->cbPatchMem);
996}
997
998
999/**
1000 * Handles VMMDevReq_SetPowerStatus.
1001 *
1002 * @returns VBox status code that the guest should see.
1003 * @param pThis The VMMDev instance data.
1004 * @param pReqHdr The header of the request to handle.
1005 */
1006static int vmmdevReqHandler_SetPowerStatus(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1007{
1008 VMMDevPowerStateRequest *pReq = (VMMDevPowerStateRequest *)pReqHdr;
1009 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1010
1011 switch (pReq->powerState)
1012 {
1013 case VMMDevPowerState_Pause:
1014 {
1015 LogRel(("Guest requests the VM to be suspended (paused)\n"));
1016 return PDMDevHlpVMSuspend(pThis->pDevIns);
1017 }
1018
1019 case VMMDevPowerState_PowerOff:
1020 {
1021 LogRel(("Guest requests the VM to be turned off\n"));
1022 return PDMDevHlpVMPowerOff(pThis->pDevIns);
1023 }
1024
1025 case VMMDevPowerState_SaveState:
1026 {
1027 if (true /*pThis->fAllowGuestToSaveState*/)
1028 {
1029 LogRel(("Guest requests the VM to be saved and powered off\n"));
1030 return PDMDevHlpVMSuspendSaveAndPowerOff(pThis->pDevIns);
1031 }
1032 LogRel(("Guest requests the VM to be saved and powered off, declined\n"));
1033 return VERR_ACCESS_DENIED;
1034 }
1035
1036 default:
1037 AssertMsgFailed(("VMMDev invalid power state request: %d\n", pReq->powerState));
1038 return VERR_INVALID_PARAMETER;
1039 }
1040}
1041
1042
1043/**
1044 * Handles VMMDevReq_GetDisplayChangeRequest
1045 *
1046 * @returns VBox status code that the guest should see.
1047 * @param pThis The VMMDev instance data.
1048 * @param pReqHdr The header of the request to handle.
1049 * @remarks Deprecated.
1050 */
1051static int vmmdevReqHandler_GetDisplayChangeRequest(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1052{
1053 VMMDevDisplayChangeRequest *pReq = (VMMDevDisplayChangeRequest *)pReqHdr;
1054 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1055
1056/**
1057 * @todo It looks like a multi-monitor guest which only uses
1058 * @a VMMDevReq_GetDisplayChangeRequest (not the *2 version)
1059 * will get into a @a VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST event
1060 * loop if it tries to acknowlege host requests for additional
1061 * monitors. Should the loop which checks for those requests
1062 * be removed?
1063 */
1064
1065 DISPLAYCHANGEREQUEST *pDispRequest = &pThis->displayChangeData.aRequests[0];
1066
1067 if (pReq->eventAck == VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST)
1068 {
1069 /* Current request has been read at least once. */
1070 pDispRequest->fPending = false;
1071
1072 /* Check if there are more pending requests. */
1073 for (unsigned i = 1; i < RT_ELEMENTS(pThis->displayChangeData.aRequests); i++)
1074 {
1075 if (pThis->displayChangeData.aRequests[i].fPending)
1076 {
1077 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST);
1078 break;
1079 }
1080 }
1081
1082 /* Remember which resolution the client has queried, subsequent reads
1083 * will return the same values. */
1084 pDispRequest->lastReadDisplayChangeRequest = pDispRequest->displayChangeRequest;
1085 pThis->displayChangeData.fGuestSentChangeEventAck = true;
1086 }
1087
1088 if (pThis->displayChangeData.fGuestSentChangeEventAck)
1089 {
1090 pReq->xres = pDispRequest->lastReadDisplayChangeRequest.xres;
1091 pReq->yres = pDispRequest->lastReadDisplayChangeRequest.yres;
1092 pReq->bpp = pDispRequest->lastReadDisplayChangeRequest.bpp;
1093 }
1094 else
1095 {
1096 /* This is not a response to a VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST, just
1097 * read the last valid video mode hint. This happens when the guest X server
1098 * determines the initial mode. */
1099 pReq->xres = pDispRequest->displayChangeRequest.xres;
1100 pReq->yres = pDispRequest->displayChangeRequest.yres;
1101 pReq->bpp = pDispRequest->displayChangeRequest.bpp;
1102 }
1103 Log(("VMMDev: returning display change request xres = %d, yres = %d, bpp = %d\n", pReq->xres, pReq->yres, pReq->bpp));
1104
1105 return VINF_SUCCESS;
1106}
1107
1108
1109/**
1110 * Handles VMMDevReq_GetDisplayChangeRequest2.
1111 *
1112 * @returns VBox status code that the guest should see.
1113 * @param pThis The VMMDev instance data.
1114 * @param pReqHdr The header of the request to handle.
1115 */
1116static int vmmdevReqHandler_GetDisplayChangeRequest2(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1117{
1118 VMMDevDisplayChangeRequest2 *pReq = (VMMDevDisplayChangeRequest2 *)pReqHdr;
1119 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1120
1121 DISPLAYCHANGEREQUEST *pDispRequest = NULL;
1122
1123 if (pReq->eventAck == VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST)
1124 {
1125 /* Select a pending request to report. */
1126 unsigned i;
1127 for (i = 0; i < RT_ELEMENTS(pThis->displayChangeData.aRequests); i++)
1128 {
1129 if (pThis->displayChangeData.aRequests[i].fPending)
1130 {
1131 pDispRequest = &pThis->displayChangeData.aRequests[i];
1132 /* Remember which request should be reported. */
1133 pThis->displayChangeData.iCurrentMonitor = i;
1134 Log3(("VMMDev: will report pending request for %u\n", i));
1135 break;
1136 }
1137 }
1138
1139 /* Check if there are more pending requests. */
1140 i++;
1141 for (; i < RT_ELEMENTS(pThis->displayChangeData.aRequests); i++)
1142 {
1143 if (pThis->displayChangeData.aRequests[i].fPending)
1144 {
1145 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST);
1146 Log3(("VMMDev: another pending at %u\n", i));
1147 break;
1148 }
1149 }
1150
1151 if (pDispRequest)
1152 {
1153 /* Current request has been read at least once. */
1154 pDispRequest->fPending = false;
1155
1156 /* Remember which resolution the client has queried, subsequent reads
1157 * will return the same values. */
1158 pDispRequest->lastReadDisplayChangeRequest = pDispRequest->displayChangeRequest;
1159 pThis->displayChangeData.fGuestSentChangeEventAck = true;
1160 }
1161 else
1162 {
1163 Log3(("VMMDev: no pending request!!!\n"));
1164 }
1165 }
1166
1167 if (!pDispRequest)
1168 {
1169 Log3(("VMMDev: default to %d\n", pThis->displayChangeData.iCurrentMonitor));
1170 pDispRequest = &pThis->displayChangeData.aRequests[pThis->displayChangeData.iCurrentMonitor];
1171 }
1172
1173 if (pThis->displayChangeData.fGuestSentChangeEventAck)
1174 {
1175 pReq->xres = pDispRequest->lastReadDisplayChangeRequest.xres;
1176 pReq->yres = pDispRequest->lastReadDisplayChangeRequest.yres;
1177 pReq->bpp = pDispRequest->lastReadDisplayChangeRequest.bpp;
1178 pReq->display = pDispRequest->lastReadDisplayChangeRequest.display;
1179 }
1180 else
1181 {
1182 /* This is not a response to a VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST, just
1183 * read the last valid video mode hint. This happens when the guest X server
1184 * determines the initial video mode. */
1185 pReq->xres = pDispRequest->displayChangeRequest.xres;
1186 pReq->yres = pDispRequest->displayChangeRequest.yres;
1187 pReq->bpp = pDispRequest->displayChangeRequest.bpp;
1188 pReq->display = pDispRequest->displayChangeRequest.display;
1189 }
1190 Log(("VMMDev: returning display change request xres = %d, yres = %d, bpp = %d at %d\n",
1191 pReq->xres, pReq->yres, pReq->bpp, pReq->display));
1192
1193 return VINF_SUCCESS;
1194}
1195
1196
1197/**
1198 * Handles VMMDevReq_GetDisplayChangeRequestEx.
1199 *
1200 * @returns VBox status code that the guest should see.
1201 * @param pThis The VMMDev instance data.
1202 * @param pReqHdr The header of the request to handle.
1203 */
1204static int vmmdevReqHandler_GetDisplayChangeRequestEx(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1205{
1206 VMMDevDisplayChangeRequestEx *pReq = (VMMDevDisplayChangeRequestEx *)pReqHdr;
1207 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1208
1209 DISPLAYCHANGEREQUEST *pDispRequest = NULL;
1210
1211 if (pReq->eventAck == VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST)
1212 {
1213 /* Select a pending request to report. */
1214 unsigned i;
1215 for (i = 0; i < RT_ELEMENTS(pThis->displayChangeData.aRequests); i++)
1216 {
1217 if (pThis->displayChangeData.aRequests[i].fPending)
1218 {
1219 pDispRequest = &pThis->displayChangeData.aRequests[i];
1220 /* Remember which request should be reported. */
1221 pThis->displayChangeData.iCurrentMonitor = i;
1222 Log3(("VMMDev: will report pending request for %d\n",
1223 i));
1224 break;
1225 }
1226 }
1227
1228 /* Check if there are more pending requests. */
1229 i++;
1230 for (; i < RT_ELEMENTS(pThis->displayChangeData.aRequests); i++)
1231 {
1232 if (pThis->displayChangeData.aRequests[i].fPending)
1233 {
1234 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST);
1235 Log3(("VMMDev: another pending at %d\n",
1236 i));
1237 break;
1238 }
1239 }
1240
1241 if (pDispRequest)
1242 {
1243 /* Current request has been read at least once. */
1244 pDispRequest->fPending = false;
1245
1246 /* Remember which resolution the client has queried, subsequent reads
1247 * will return the same values. */
1248 pDispRequest->lastReadDisplayChangeRequest = pDispRequest->displayChangeRequest;
1249 pThis->displayChangeData.fGuestSentChangeEventAck = true;
1250 }
1251 else
1252 {
1253 Log3(("VMMDev: no pending request!!!\n"));
1254 }
1255 }
1256
1257 if (!pDispRequest)
1258 {
1259 Log3(("VMMDev: default to %d\n",
1260 pThis->displayChangeData.iCurrentMonitor));
1261 pDispRequest = &pThis->displayChangeData.aRequests[pThis->displayChangeData.iCurrentMonitor];
1262 }
1263
1264 if (pThis->displayChangeData.fGuestSentChangeEventAck)
1265 {
1266 pReq->xres = pDispRequest->lastReadDisplayChangeRequest.xres;
1267 pReq->yres = pDispRequest->lastReadDisplayChangeRequest.yres;
1268 pReq->bpp = pDispRequest->lastReadDisplayChangeRequest.bpp;
1269 pReq->display = pDispRequest->lastReadDisplayChangeRequest.display;
1270 pReq->cxOrigin = pDispRequest->lastReadDisplayChangeRequest.xOrigin;
1271 pReq->cyOrigin = pDispRequest->lastReadDisplayChangeRequest.yOrigin;
1272 pReq->fEnabled = pDispRequest->lastReadDisplayChangeRequest.fEnabled;
1273 pReq->fChangeOrigin = pDispRequest->lastReadDisplayChangeRequest.fChangeOrigin;
1274 }
1275 else
1276 {
1277 /* This is not a response to a VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST, just
1278 * read the last valid video mode hint. This happens when the guest X server
1279 * determines the initial video mode. */
1280 pReq->xres = pDispRequest->displayChangeRequest.xres;
1281 pReq->yres = pDispRequest->displayChangeRequest.yres;
1282 pReq->bpp = pDispRequest->displayChangeRequest.bpp;
1283 pReq->display = pDispRequest->displayChangeRequest.display;
1284 pReq->cxOrigin = pDispRequest->displayChangeRequest.xOrigin;
1285 pReq->cyOrigin = pDispRequest->displayChangeRequest.yOrigin;
1286 pReq->fEnabled = pDispRequest->displayChangeRequest.fEnabled;
1287 pReq->fChangeOrigin = pDispRequest->displayChangeRequest.fChangeOrigin;
1288
1289 }
1290 Log(("VMMDevEx: returning display change request xres = %d, yres = %d, bpp = %d id %d xPos = %d, yPos = %d & Enabled=%d\n",
1291 pReq->xres, pReq->yres, pReq->bpp, pReq->display, pReq->cxOrigin, pReq->cyOrigin, pReq->fEnabled));
1292
1293 return VINF_SUCCESS;
1294}
1295
1296
1297/**
1298 * Handles VMMDevReq_VideoModeSupported.
1299 *
1300 * Query whether the given video mode is supported.
1301 *
1302 * @returns VBox status code that the guest should see.
1303 * @param pThis The VMMDev instance data.
1304 * @param pReqHdr The header of the request to handle.
1305 */
1306static int vmmdevReqHandler_VideoModeSupported(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1307{
1308 VMMDevVideoModeSupportedRequest *pReq = (VMMDevVideoModeSupportedRequest *)pReqHdr;
1309 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1310
1311 /* forward the call */
1312 return pThis->pDrv->pfnVideoModeSupported(pThis->pDrv,
1313 0, /* primary screen. */
1314 pReq->width,
1315 pReq->height,
1316 pReq->bpp,
1317 &pReq->fSupported);
1318}
1319
1320
1321/**
1322 * Handles VMMDevReq_VideoModeSupported2.
1323 *
1324 * Query whether the given video mode is supported for a specific display
1325 *
1326 * @returns VBox status code that the guest should see.
1327 * @param pThis The VMMDev instance data.
1328 * @param pReqHdr The header of the request to handle.
1329 */
1330static int vmmdevReqHandler_VideoModeSupported2(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1331{
1332 VMMDevVideoModeSupportedRequest2 *pReq = (VMMDevVideoModeSupportedRequest2 *)pReqHdr;
1333 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1334
1335 /* forward the call */
1336 return pThis->pDrv->pfnVideoModeSupported(pThis->pDrv,
1337 pReq->display,
1338 pReq->width,
1339 pReq->height,
1340 pReq->bpp,
1341 &pReq->fSupported);
1342}
1343
1344
1345
1346/**
1347 * Handles VMMDevReq_GetHeightReduction.
1348 *
1349 * @returns VBox status code that the guest should see.
1350 * @param pThis The VMMDev instance data.
1351 * @param pReqHdr The header of the request to handle.
1352 */
1353static int vmmdevReqHandler_GetHeightReduction(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1354{
1355 VMMDevGetHeightReductionRequest *pReq = (VMMDevGetHeightReductionRequest *)pReqHdr;
1356 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1357
1358 /* forward the call */
1359 return pThis->pDrv->pfnGetHeightReduction(pThis->pDrv, &pReq->heightReduction);
1360}
1361
1362
1363/**
1364 * Handles VMMDevReq_AcknowledgeEvents.
1365 *
1366 * @returns VBox status code that the guest should see.
1367 * @param pThis The VMMDev instance data.
1368 * @param pReqHdr The header of the request to handle.
1369 */
1370static int vmmdevReqHandler_AcknowledgeEvents(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1371{
1372 VMMDevEvents *pReq = (VMMDevEvents *)pReqHdr;
1373 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1374
1375 if (VBOX_GUEST_INTERFACE_VERSION_1_03(pThis))
1376 {
1377 vmmdevSetIRQ_Legacy(pThis);
1378 }
1379 else
1380 {
1381 if (pThis->fNewGuestFilterMask)
1382 {
1383 pThis->fNewGuestFilterMask = false;
1384 pThis->u32GuestFilterMask = pThis->u32NewGuestFilterMask;
1385 }
1386
1387 pReq->events = pThis->u32HostEventFlags & pThis->u32GuestFilterMask;
1388
1389 pThis->u32HostEventFlags &= ~pThis->u32GuestFilterMask;
1390 pThis->pVMMDevRAMR3->V.V1_04.fHaveEvents = false;
1391 PDMDevHlpPCISetIrqNoWait(pThis->pDevIns, 0, 0);
1392 }
1393 return VINF_SUCCESS;
1394}
1395
1396
1397/**
1398 * Handles VMMDevReq_CtlGuestFilterMask.
1399 *
1400 * @returns VBox status code that the guest should see.
1401 * @param pThis The VMMDev instance data.
1402 * @param pReqHdr The header of the request to handle.
1403 */
1404static int vmmdevReqHandler_CtlGuestFilterMask(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1405{
1406 VMMDevCtlGuestFilterMask *pReq = (VMMDevCtlGuestFilterMask *)pReqHdr;
1407 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1408
1409 LogRelFlowFunc(("VMMDevCtlGuestFilterMask: or mask: %#x, not mask: %#x\n", pReq->u32OrMask, pReq->u32NotMask));
1410
1411 /* HGCM event notification is enabled by the VMMDev device
1412 * automatically when any HGCM command is issued. The guest
1413 * cannot disable these notifications. */
1414 VMMDevCtlSetGuestFilterMask(pThis, pReq->u32OrMask, pReq->u32NotMask & ~VMMDEV_EVENT_HGCM);
1415 return VINF_SUCCESS;
1416}
1417
1418#ifdef VBOX_WITH_HGCM
1419
1420/**
1421 * Handles VMMDevReq_HGCMConnect.
1422 *
1423 * @returns VBox status code that the guest should see.
1424 * @param pThis The VMMDev instance data.
1425 * @param pReqHdr The header of the request to handle.
1426 * @param GCPhysReqHdr The guest physical address of the request header.
1427 */
1428static int vmmdevReqHandler_HGCMConnect(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr, RTGCPHYS GCPhysReqHdr)
1429{
1430 VMMDevHGCMConnect *pReq = (VMMDevHGCMConnect *)pReqHdr;
1431 AssertMsgReturn(pReq->header.header.size >= sizeof(*pReq), ("%u\n", pReq->header.header.size), VERR_INVALID_PARAMETER); /** @todo Not sure why this is >= ... */
1432
1433 if (!pThis->pHGCMDrv)
1434 {
1435 Log(("VMMDevReq_HGCMConnect: HGCM Connector is NULL!\n"));
1436 return VERR_NOT_SUPPORTED;
1437 }
1438
1439 Log(("VMMDevReq_HGCMConnect\n"));
1440 return vmmdevHGCMConnect(pThis, pReq, GCPhysReqHdr);
1441}
1442
1443
1444/**
1445 * Handles VMMDevReq_HGCMDisconnect.
1446 *
1447 * @returns VBox status code that the guest should see.
1448 * @param pThis The VMMDev instance data.
1449 * @param pReqHdr The header of the request to handle.
1450 * @param GCPhysReqHdr The guest physical address of the request header.
1451 */
1452static int vmmdevReqHandler_HGCMDisconnect(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr, RTGCPHYS GCPhysReqHdr)
1453{
1454 VMMDevHGCMDisconnect *pReq = (VMMDevHGCMDisconnect *)pReqHdr;
1455 AssertMsgReturn(pReq->header.header.size >= sizeof(*pReq), ("%u\n", pReq->header.header.size), VERR_INVALID_PARAMETER); /** @todo Not sure why this >= ... */
1456
1457 if (!pThis->pHGCMDrv)
1458 {
1459 Log(("VMMDevReq_VMMDevHGCMDisconnect: HGCM Connector is NULL!\n"));
1460 return VERR_NOT_SUPPORTED;
1461 }
1462
1463 Log(("VMMDevReq_VMMDevHGCMDisconnect\n"));
1464 return vmmdevHGCMDisconnect(pThis, pReq, GCPhysReqHdr);
1465}
1466
1467
1468/**
1469 * Handles VMMDevReq_HGCMCall.
1470 *
1471 * @returns VBox status code that the guest should see.
1472 * @param pThis The VMMDev instance data.
1473 * @param pReqHdr The header of the request to handle.
1474 * @param GCPhysReqHdr The guest physical address of the request header.
1475 */
1476static int vmmdevReqHandler_HGCMCall(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr, RTGCPHYS GCPhysReqHdr)
1477{
1478 VMMDevHGCMCall *pReq = (VMMDevHGCMCall *)pReqHdr;
1479 AssertMsgReturn(pReq->header.header.size >= sizeof(*pReq), ("%u\n", pReq->header.header.size), VERR_INVALID_PARAMETER);
1480
1481 if (!pThis->pHGCMDrv)
1482 {
1483 Log(("VMMDevReq_HGCMCall: HGCM Connector is NULL!\n"));
1484 return VERR_NOT_SUPPORTED;
1485 }
1486
1487 Log2(("VMMDevReq_HGCMCall: sizeof(VMMDevHGCMRequest) = %04X\n", sizeof(VMMDevHGCMCall)));
1488 Log2(("%.*Rhxd\n", pReq->header.header.size, pReq));
1489
1490#ifdef VBOX_WITH_64_BITS_GUESTS
1491 bool f64Bits = (pReq->header.header.requestType == VMMDevReq_HGCMCall64);
1492#else
1493 bool f64Bits = false;
1494#endif /* VBOX_WITH_64_BITS_GUESTS */
1495
1496 return vmmdevHGCMCall(pThis, pReq, pReq->header.header.size, GCPhysReqHdr, f64Bits);
1497}
1498
1499#endif /* VBOX_WITH_HGCM */
1500
1501/**
1502 * Handles VMMDevReq_HGCMCancel.
1503 *
1504 * @returns VBox status code that the guest should see.
1505 * @param pThis The VMMDev instance data.
1506 * @param pReqHdr The header of the request to handle.
1507 * @param GCPhysReqHdr The guest physical address of the request header.
1508 */
1509static int vmmdevReqHandler_HGCMCancel(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr, RTGCPHYS GCPhysReqHdr)
1510{
1511 VMMDevHGCMCancel *pReq = (VMMDevHGCMCancel *)pReqHdr;
1512 AssertMsgReturn(pReq->header.header.size >= sizeof(*pReq), ("%u\n", pReq->header.header.size), VERR_INVALID_PARAMETER); /** @todo Not sure why this >= ... */
1513
1514 if (!pThis->pHGCMDrv)
1515 {
1516 Log(("VMMDevReq_VMMDevHGCMCancel: HGCM Connector is NULL!\n"));
1517 return VERR_NOT_SUPPORTED;
1518 }
1519
1520 Log(("VMMDevReq_VMMDevHGCMCancel\n"));
1521 return vmmdevHGCMCancel(pThis, pReq, GCPhysReqHdr);
1522}
1523
1524
1525/**
1526 * Handles VMMDevReq_HGCMCancel2.
1527 *
1528 * @returns VBox status code that the guest should see.
1529 * @param pThis The VMMDev instance data.
1530 * @param pReqHdr The header of the request to handle.
1531 */
1532static int vmmdevReqHandler_HGCMCancel2(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1533{
1534 VMMDevHGCMCancel2 *pReq = (VMMDevHGCMCancel2 *)pReqHdr;
1535 AssertMsgReturn(pReq->header.size >= sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER); /** @todo Not sure why this >= ... */
1536
1537 if (!pThis->pHGCMDrv)
1538 {
1539 Log(("VMMDevReq_HGCMConnect2: HGCM Connector is NULL!\n"));
1540 return VERR_NOT_SUPPORTED;
1541 }
1542
1543 Log(("VMMDevReq_VMMDevHGCMCancel\n"));
1544 return vmmdevHGCMCancel2(pThis, pReq->physReqToCancel);
1545}
1546
1547
1548/**
1549 * Handles VMMDevReq_VideoAccelEnable.
1550 *
1551 * @returns VBox status code that the guest should see.
1552 * @param pThis The VMMDev instance data.
1553 * @param pReqHdr The header of the request to handle.
1554 */
1555static int vmmdevReqHandler_VideoAccelEnable(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1556{
1557 VMMDevVideoAccelEnable *pReq = (VMMDevVideoAccelEnable *)pReqHdr;
1558 AssertMsgReturn(pReq->header.size >= sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER); /** @todo Not sure why this >= ... */
1559
1560 if (!pThis->pDrv)
1561 {
1562 Log(("VMMDevReq_VideoAccelEnable Connector is NULL!!!\n"));
1563 return VERR_NOT_SUPPORTED;
1564 }
1565
1566 if (pReq->cbRingBuffer != VBVA_RING_BUFFER_SIZE)
1567 {
1568 /* The guest driver seems compiled with another headers. */
1569 Log(("VMMDevReq_VideoAccelEnable guest ring buffer size %d, should be %d!!!\n", pReq->cbRingBuffer, VBVA_RING_BUFFER_SIZE));
1570 return VERR_INVALID_PARAMETER;
1571 }
1572
1573 /* The request is correct. */
1574 pReq->fu32Status |= VBVA_F_STATUS_ACCEPTED;
1575
1576 LogFlow(("VMMDevReq_VideoAccelEnable pReq->u32Enable = %d\n", pReq->u32Enable));
1577
1578 int rc = pReq->u32Enable
1579 ? pThis->pDrv->pfnVideoAccelEnable(pThis->pDrv, true, &pThis->pVMMDevRAMR3->vbvaMemory)
1580 : pThis->pDrv->pfnVideoAccelEnable(pThis->pDrv, false, NULL);
1581
1582 if ( pReq->u32Enable
1583 && RT_SUCCESS(rc))
1584 {
1585 pReq->fu32Status |= VBVA_F_STATUS_ENABLED;
1586
1587 /* Remember that guest successfully enabled acceleration.
1588 * We need to reestablish it on restoring the VM from saved state.
1589 */
1590 pThis->u32VideoAccelEnabled = 1;
1591 }
1592 else
1593 {
1594 /* The acceleration was not enabled. Remember that. */
1595 pThis->u32VideoAccelEnabled = 0;
1596 }
1597 return VINF_SUCCESS;
1598}
1599
1600
1601/**
1602 * Handles VMMDevReq_VideoAccelFlush.
1603 *
1604 * @returns VBox status code that the guest should see.
1605 * @param pThis The VMMDev instance data.
1606 * @param pReqHdr The header of the request to handle.
1607 */
1608static int vmmdevReqHandler_VideoAccelFlush(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1609{
1610 VMMDevVideoAccelFlush *pReq = (VMMDevVideoAccelFlush *)pReqHdr;
1611 AssertMsgReturn(pReq->header.size >= sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER); /** @todo Not sure why this >= ... */
1612
1613 if (!pThis->pDrv)
1614 {
1615 Log(("VMMDevReq_VideoAccelFlush: Connector is NULL!!!\n"));
1616 return VERR_NOT_SUPPORTED;
1617 }
1618
1619 pThis->pDrv->pfnVideoAccelFlush(pThis->pDrv);
1620 return VINF_SUCCESS;
1621}
1622
1623
1624/**
1625 * Handles VMMDevReq_VideoSetVisibleRegion.
1626 *
1627 * @returns VBox status code that the guest should see.
1628 * @param pThis The VMMDev instance data.
1629 * @param pReqHdr The header of the request to handle.
1630 */
1631static int vmmdevReqHandler_VideoSetVisibleRegion(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1632{
1633 VMMDevVideoSetVisibleRegion *pReq = (VMMDevVideoSetVisibleRegion *)pReqHdr;
1634 AssertMsgReturn(pReq->header.size + sizeof(RTRECT) >= sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1635
1636 if (!pThis->pDrv)
1637 {
1638 Log(("VMMDevReq_VideoSetVisibleRegion: Connector is NULL!!!\n"));
1639 return VERR_NOT_SUPPORTED;
1640 }
1641
1642 if ( pReq->cRect > _1M /* restrict to sane range */
1643 || pReq->header.size != sizeof(VMMDevVideoSetVisibleRegion) + pReq->cRect * sizeof(RTRECT) - sizeof(RTRECT))
1644 {
1645 Log(("VMMDevReq_VideoSetVisibleRegion: cRects=%#x doesn't match size=%#x or is out of bounds\n",
1646 pReq->cRect, pReq->header.size));
1647 return VERR_INVALID_PARAMETER;
1648 }
1649
1650 Log(("VMMDevReq_VideoSetVisibleRegion %d rectangles\n", pReq->cRect));
1651 /* forward the call */
1652 return pThis->pDrv->pfnSetVisibleRegion(pThis->pDrv, pReq->cRect, &pReq->Rect);
1653}
1654
1655
1656/**
1657 * Handles VMMDevReq_GetSeamlessChangeRequest.
1658 *
1659 * @returns VBox status code that the guest should see.
1660 * @param pThis The VMMDev instance data.
1661 * @param pReqHdr The header of the request to handle.
1662 */
1663static int vmmdevReqHandler_GetSeamlessChangeRequest(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1664{
1665 VMMDevSeamlessChangeRequest *pReq = (VMMDevSeamlessChangeRequest *)pReqHdr;
1666 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1667
1668 /* just pass on the information */
1669 Log(("VMMDev: returning seamless change request mode=%d\n", pThis->fSeamlessEnabled));
1670 if (pThis->fSeamlessEnabled)
1671 pReq->mode = VMMDev_Seamless_Visible_Region;
1672 else
1673 pReq->mode = VMMDev_Seamless_Disabled;
1674
1675 if (pReq->eventAck == VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST)
1676 {
1677 /* Remember which mode the client has queried. */
1678 pThis->fLastSeamlessEnabled = pThis->fSeamlessEnabled;
1679 }
1680
1681 return VINF_SUCCESS;
1682}
1683
1684
1685/**
1686 * Handles VMMDevReq_GetVRDPChangeRequest.
1687 *
1688 * @returns VBox status code that the guest should see.
1689 * @param pThis The VMMDev instance data.
1690 * @param pReqHdr The header of the request to handle.
1691 */
1692static int vmmdevReqHandler_GetVRDPChangeRequest(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1693{
1694 VMMDevVRDPChangeRequest *pReq = (VMMDevVRDPChangeRequest *)pReqHdr;
1695 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1696
1697 /* just pass on the information */
1698 Log(("VMMDev: returning VRDP status %d level %d\n", pThis->fVRDPEnabled, pThis->uVRDPExperienceLevel));
1699
1700 pReq->u8VRDPActive = pThis->fVRDPEnabled;
1701 pReq->u32VRDPExperienceLevel = pThis->uVRDPExperienceLevel;
1702
1703 return VINF_SUCCESS;
1704}
1705
1706
1707/**
1708 * Handles VMMDevReq_GetMemBalloonChangeRequest.
1709 *
1710 * @returns VBox status code that the guest should see.
1711 * @param pThis The VMMDev instance data.
1712 * @param pReqHdr The header of the request to handle.
1713 */
1714static int vmmdevReqHandler_GetMemBalloonChangeRequest(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1715{
1716 VMMDevGetMemBalloonChangeRequest *pReq = (VMMDevGetMemBalloonChangeRequest *)pReqHdr;
1717 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1718
1719 /* just pass on the information */
1720 Log(("VMMDev: returning memory balloon size =%d\n", pThis->cMbMemoryBalloon));
1721 pReq->cBalloonChunks = pThis->cMbMemoryBalloon;
1722 pReq->cPhysMemChunks = pThis->cbGuestRAM / (uint64_t)_1M;
1723
1724 if (pReq->eventAck == VMMDEV_EVENT_BALLOON_CHANGE_REQUEST)
1725 {
1726 /* Remember which mode the client has queried. */
1727 pThis->cMbMemoryBalloonLast = pThis->cMbMemoryBalloon;
1728 }
1729
1730 return VINF_SUCCESS;
1731}
1732
1733
1734/**
1735 * Handles VMMDevReq_ChangeMemBalloon.
1736 *
1737 * @returns VBox status code that the guest should see.
1738 * @param pThis The VMMDev instance data.
1739 * @param pReqHdr The header of the request to handle.
1740 */
1741static int vmmdevReqHandler_ChangeMemBalloon(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1742{
1743 VMMDevChangeMemBalloon *pReq = (VMMDevChangeMemBalloon *)pReqHdr;
1744 AssertMsgReturn(pReq->header.size >= sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1745 AssertMsgReturn(pReq->cPages == VMMDEV_MEMORY_BALLOON_CHUNK_PAGES, ("%u\n", pReq->cPages), VERR_INVALID_PARAMETER);
1746 AssertMsgReturn(pReq->header.size == (uint32_t)RT_OFFSETOF(VMMDevChangeMemBalloon, aPhysPage[pReq->cPages]),
1747 ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1748
1749 Log(("VMMDevReq_ChangeMemBalloon\n"));
1750 int rc = PGMR3PhysChangeMemBalloon(PDMDevHlpGetVM(pThis->pDevIns), !!pReq->fInflate, pReq->cPages, pReq->aPhysPage);
1751 if (pReq->fInflate)
1752 STAM_REL_U32_INC(&pThis->StatMemBalloonChunks);
1753 else
1754 STAM_REL_U32_DEC(&pThis->StatMemBalloonChunks);
1755 return rc;
1756}
1757
1758
1759/**
1760 * Handles VMMDevReq_GetStatisticsChangeRequest.
1761 *
1762 * @returns VBox status code that the guest should see.
1763 * @param pThis The VMMDev instance data.
1764 * @param pReqHdr The header of the request to handle.
1765 */
1766static int vmmdevReqHandler_GetStatisticsChangeRequest(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1767{
1768 VMMDevGetStatisticsChangeRequest *pReq = (VMMDevGetStatisticsChangeRequest *)pReqHdr;
1769 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1770
1771 Log(("VMMDevReq_GetStatisticsChangeRequest\n"));
1772 /* just pass on the information */
1773 Log(("VMMDev: returning statistics interval %d seconds\n", pThis->u32StatIntervalSize));
1774 pReq->u32StatInterval = pThis->u32StatIntervalSize;
1775
1776 if (pReq->eventAck == VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST)
1777 {
1778 /* Remember which mode the client has queried. */
1779 pThis->u32LastStatIntervalSize= pThis->u32StatIntervalSize;
1780 }
1781
1782 return VINF_SUCCESS;
1783}
1784
1785
1786/**
1787 * Handles VMMDevReq_ReportGuestStats.
1788 *
1789 * @returns VBox status code that the guest should see.
1790 * @param pThis The VMMDev instance data.
1791 * @param pReqHdr The header of the request to handle.
1792 */
1793static int vmmdevReqHandler_ReportGuestStats(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1794{
1795 VMMDevReportGuestStats *pReq = (VMMDevReportGuestStats *)pReqHdr;
1796 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1797
1798 Log(("VMMDevReq_ReportGuestStats\n"));
1799#ifdef LOG_ENABLED
1800 VBoxGuestStatistics *pGuestStats = &pReq->guestStats;
1801
1802 Log(("Current statistics:\n"));
1803 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_CPU_LOAD_IDLE)
1804 Log(("CPU%u: CPU Load Idle %-3d%%\n", pGuestStats->u32CpuId, pGuestStats->u32CpuLoad_Idle));
1805
1806 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_CPU_LOAD_KERNEL)
1807 Log(("CPU%u: CPU Load Kernel %-3d%%\n", pGuestStats->u32CpuId, pGuestStats->u32CpuLoad_Kernel));
1808
1809 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_CPU_LOAD_USER)
1810 Log(("CPU%u: CPU Load User %-3d%%\n", pGuestStats->u32CpuId, pGuestStats->u32CpuLoad_User));
1811
1812 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_THREADS)
1813 Log(("CPU%u: Thread %d\n", pGuestStats->u32CpuId, pGuestStats->u32Threads));
1814
1815 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PROCESSES)
1816 Log(("CPU%u: Processes %d\n", pGuestStats->u32CpuId, pGuestStats->u32Processes));
1817
1818 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_HANDLES)
1819 Log(("CPU%u: Handles %d\n", pGuestStats->u32CpuId, pGuestStats->u32Handles));
1820
1821 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEMORY_LOAD)
1822 Log(("CPU%u: Memory Load %d%%\n", pGuestStats->u32CpuId, pGuestStats->u32MemoryLoad));
1823
1824 /* Note that reported values are in pages; upper layers expect them in megabytes */
1825 Log(("CPU%u: Page size %-4d bytes\n", pGuestStats->u32CpuId, pGuestStats->u32PageSize));
1826 Assert(pGuestStats->u32PageSize == 4096);
1827
1828 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PHYS_MEM_TOTAL)
1829 Log(("CPU%u: Total physical memory %-4d MB\n", pGuestStats->u32CpuId, (pGuestStats->u32PhysMemTotal + (_1M/_4K)-1) / (_1M/_4K)));
1830
1831 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PHYS_MEM_AVAIL)
1832 Log(("CPU%u: Free physical memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32PhysMemAvail / (_1M/_4K)));
1833
1834 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PHYS_MEM_BALLOON)
1835 Log(("CPU%u: Memory balloon size %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32PhysMemBalloon / (_1M/_4K)));
1836
1837 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_COMMIT_TOTAL)
1838 Log(("CPU%u: Committed memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemCommitTotal / (_1M/_4K)));
1839
1840 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_KERNEL_TOTAL)
1841 Log(("CPU%u: Total kernel memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemKernelTotal / (_1M/_4K)));
1842
1843 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_KERNEL_PAGED)
1844 Log(("CPU%u: Paged kernel memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemKernelPaged / (_1M/_4K)));
1845
1846 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_KERNEL_NONPAGED)
1847 Log(("CPU%u: Nonpaged kernel memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemKernelNonPaged / (_1M/_4K)));
1848
1849 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_SYSTEM_CACHE)
1850 Log(("CPU%u: System cache size %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemSystemCache / (_1M/_4K)));
1851
1852 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PAGE_FILE_SIZE)
1853 Log(("CPU%u: Page file size %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32PageFileSize / (_1M/_4K)));
1854 Log(("Statistics end *******************\n"));
1855#endif /* LOG_ENABLED */
1856
1857 /* forward the call */
1858 return pThis->pDrv->pfnReportStatistics(pThis->pDrv, &pReq->guestStats);
1859}
1860
1861
1862/**
1863 * Handles VMMDevReq_QueryCredentials.
1864 *
1865 * @returns VBox status code that the guest should see.
1866 * @param pThis The VMMDev instance data.
1867 * @param pReqHdr The header of the request to handle.
1868 */
1869static int vmmdevReqHandler_QueryCredentials(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1870{
1871 VMMDevCredentials *pReq = (VMMDevCredentials *)pReqHdr;
1872 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1873
1874 /* let's start by nulling out the data */
1875 memset(pReq->szUserName, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1876 memset(pReq->szPassword, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1877 memset(pReq->szDomain, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1878
1879 /* should we return whether we got credentials for a logon? */
1880 if (pReq->u32Flags & VMMDEV_CREDENTIALS_QUERYPRESENCE)
1881 {
1882 if ( pThis->pCredentials->Logon.szUserName[0]
1883 || pThis->pCredentials->Logon.szPassword[0]
1884 || pThis->pCredentials->Logon.szDomain[0])
1885 pReq->u32Flags |= VMMDEV_CREDENTIALS_PRESENT;
1886 else
1887 pReq->u32Flags &= ~VMMDEV_CREDENTIALS_PRESENT;
1888 }
1889
1890 /* does the guest want to read logon credentials? */
1891 if (pReq->u32Flags & VMMDEV_CREDENTIALS_READ)
1892 {
1893 if (pThis->pCredentials->Logon.szUserName[0])
1894 strcpy(pReq->szUserName, pThis->pCredentials->Logon.szUserName);
1895 if (pThis->pCredentials->Logon.szPassword[0])
1896 strcpy(pReq->szPassword, pThis->pCredentials->Logon.szPassword);
1897 if (pThis->pCredentials->Logon.szDomain[0])
1898 strcpy(pReq->szDomain, pThis->pCredentials->Logon.szDomain);
1899 if (!pThis->pCredentials->Logon.fAllowInteractiveLogon)
1900 pReq->u32Flags |= VMMDEV_CREDENTIALS_NOLOCALLOGON;
1901 else
1902 pReq->u32Flags &= ~VMMDEV_CREDENTIALS_NOLOCALLOGON;
1903 }
1904
1905 if (!pThis->fKeepCredentials)
1906 {
1907 /* does the caller want us to destroy the logon credentials? */
1908 if (pReq->u32Flags & VMMDEV_CREDENTIALS_CLEAR)
1909 {
1910 memset(pThis->pCredentials->Logon.szUserName, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1911 memset(pThis->pCredentials->Logon.szPassword, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1912 memset(pThis->pCredentials->Logon.szDomain, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1913 }
1914 }
1915
1916 /* does the guest want to read credentials for verification? */
1917 if (pReq->u32Flags & VMMDEV_CREDENTIALS_READJUDGE)
1918 {
1919 if (pThis->pCredentials->Judge.szUserName[0])
1920 strcpy(pReq->szUserName, pThis->pCredentials->Judge.szUserName);
1921 if (pThis->pCredentials->Judge.szPassword[0])
1922 strcpy(pReq->szPassword, pThis->pCredentials->Judge.szPassword);
1923 if (pThis->pCredentials->Judge.szDomain[0])
1924 strcpy(pReq->szDomain, pThis->pCredentials->Judge.szDomain);
1925 }
1926
1927 /* does the caller want us to destroy the judgement credentials? */
1928 if (pReq->u32Flags & VMMDEV_CREDENTIALS_CLEARJUDGE)
1929 {
1930 memset(pThis->pCredentials->Judge.szUserName, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1931 memset(pThis->pCredentials->Judge.szPassword, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1932 memset(pThis->pCredentials->Judge.szDomain, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
1933 }
1934
1935 return VINF_SUCCESS;
1936}
1937
1938
1939/**
1940 * Handles VMMDevReq_ReportCredentialsJudgement.
1941 *
1942 * @returns VBox status code that the guest should see.
1943 * @param pThis The VMMDev instance data.
1944 * @param pReqHdr The header of the request to handle.
1945 */
1946static int vmmdevReqHandler_ReportCredentialsJudgement(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1947{
1948 VMMDevCredentials *pReq = (VMMDevCredentials *)pReqHdr;
1949 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1950
1951 /* what does the guest think about the credentials? (note: the order is important here!) */
1952 if (pReq->u32Flags & VMMDEV_CREDENTIALS_JUDGE_DENY)
1953 pThis->pDrv->pfnSetCredentialsJudgementResult(pThis->pDrv, VMMDEV_CREDENTIALS_JUDGE_DENY);
1954 else if (pReq->u32Flags & VMMDEV_CREDENTIALS_JUDGE_NOJUDGEMENT)
1955 pThis->pDrv->pfnSetCredentialsJudgementResult(pThis->pDrv, VMMDEV_CREDENTIALS_JUDGE_NOJUDGEMENT);
1956 else if (pReq->u32Flags & VMMDEV_CREDENTIALS_JUDGE_OK)
1957 pThis->pDrv->pfnSetCredentialsJudgementResult(pThis->pDrv, VMMDEV_CREDENTIALS_JUDGE_OK);
1958 else
1959 {
1960 Log(("VMMDevReq_ReportCredentialsJudgement: invalid flags: %d!!!\n", pReq->u32Flags));
1961 /** @todo why don't we return VERR_INVALID_PARAMETER to the guest? */
1962 }
1963
1964 return VINF_SUCCESS;
1965}
1966
1967
1968/**
1969 * Handles VMMDevReq_GetHostVersion.
1970 *
1971 * @returns VBox status code that the guest should see.
1972 * @param pThis The VMMDev instance data.
1973 * @param pReqHdr The header of the request to handle.
1974 * @since 3.1.0
1975 * @note The ring-0 VBoxGuestLib uses this to check whether
1976 * VMMDevHGCMParmType_PageList is supported.
1977 */
1978static int vmmdevReqHandler_GetHostVersion(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
1979{
1980 VMMDevReqHostVersion *pReq = (VMMDevReqHostVersion *)pReqHdr;
1981 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
1982
1983 pReq->major = RTBldCfgVersionMajor();
1984 pReq->minor = RTBldCfgVersionMinor();
1985 pReq->build = RTBldCfgVersionBuild();
1986 pReq->revision = RTBldCfgRevision();
1987 pReq->features = VMMDEV_HVF_HGCM_PHYS_PAGE_LIST;
1988 return VINF_SUCCESS;
1989}
1990
1991
1992/**
1993 * Handles VMMDevReq_GetCpuHotPlugRequest.
1994 *
1995 * @returns VBox status code that the guest should see.
1996 * @param pThis The VMMDev instance data.
1997 * @param pReqHdr The header of the request to handle.
1998 */
1999static int vmmdevReqHandler_GetCpuHotPlugRequest(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2000{
2001 VMMDevGetCpuHotPlugRequest *pReq = (VMMDevGetCpuHotPlugRequest *)pReqHdr;
2002 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2003
2004 pReq->enmEventType = pThis->enmCpuHotPlugEvent;
2005 pReq->idCpuCore = pThis->idCpuCore;
2006 pReq->idCpuPackage = pThis->idCpuPackage;
2007
2008 /* Clear the event */
2009 pThis->enmCpuHotPlugEvent = VMMDevCpuEventType_None;
2010 pThis->idCpuCore = UINT32_MAX;
2011 pThis->idCpuPackage = UINT32_MAX;
2012
2013 return VINF_SUCCESS;
2014}
2015
2016
2017/**
2018 * Handles VMMDevReq_SetCpuHotPlugStatus.
2019 *
2020 * @returns VBox status code that the guest should see.
2021 * @param pThis The VMMDev instance data.
2022 * @param pReqHdr The header of the request to handle.
2023 */
2024static int vmmdevReqHandler_SetCpuHotPlugStatus(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2025{
2026 VMMDevCpuHotPlugStatusRequest *pReq = (VMMDevCpuHotPlugStatusRequest *)pReqHdr;
2027 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2028
2029 if (pReq->enmStatusType == VMMDevCpuStatusType_Disable)
2030 pThis->fCpuHotPlugEventsEnabled = false;
2031 else if (pReq->enmStatusType == VMMDevCpuStatusType_Enable)
2032 pThis->fCpuHotPlugEventsEnabled = true;
2033 else
2034 return VERR_INVALID_PARAMETER;
2035 return VINF_SUCCESS;
2036}
2037
2038
2039#ifdef DEBUG
2040/**
2041 * Handles VMMDevReq_LogString.
2042 *
2043 * @returns VBox status code that the guest should see.
2044 * @param pThis The VMMDev instance data.
2045 * @param pReqHdr The header of the request to handle.
2046 */
2047static int vmmdevReqHandler_LogString(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2048{
2049 VMMDevReqLogString *pReq = (VMMDevReqLogString *)pReqHdr;
2050 AssertMsgReturn(pReq->header.size >= sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2051 AssertMsgReturn(pReq->szString[pReq->header.size - RT_OFFSETOF(VMMDevReqLogString, szString) - 1] == '\0',
2052 ("not null terminated\n"), VERR_INVALID_PARAMETER);
2053
2054 LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_1, LOG_GROUP_DEV_VMM_BACKDOOR, ("DEBUG LOG: %s", pReq->szString));
2055 return VINF_SUCCESS;
2056}
2057#endif /* DEBUG */
2058
2059/**
2060 * Handles VMMDevReq_GetSessionId.
2061 *
2062 * Get a unique "session" ID for this VM, where the ID will be different after each
2063 * start, reset or restore of the VM. This can be used for restore detection
2064 * inside the guest.
2065 *
2066 * @returns VBox status code that the guest should see.
2067 * @param pThis The VMMDev instance data.
2068 * @param pReqHdr The header of the request to handle.
2069 */
2070static int vmmdevReqHandler_GetSessionId(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2071{
2072 VMMDevReqSessionId *pReq = (VMMDevReqSessionId *)pReqHdr;
2073 AssertMsgReturn(pReq->header.size == sizeof(*pReq), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2074
2075 pReq->idSession = pThis->idSession;
2076 return VINF_SUCCESS;
2077}
2078
2079
2080#ifdef VBOX_WITH_PAGE_SHARING
2081
2082/**
2083 * Handles VMMDevReq_RegisterSharedModule.
2084 *
2085 * @returns VBox status code that the guest should see.
2086 * @param pThis The VMMDev instance data.
2087 * @param pReqHdr The header of the request to handle.
2088 */
2089static int vmmdevReqHandler_RegisterSharedModule(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2090{
2091 /*
2092 * Basic input validation (more done by GMM).
2093 */
2094 VMMDevSharedModuleRegistrationRequest *pReq = (VMMDevSharedModuleRegistrationRequest *)pReqHdr;
2095 AssertMsgReturn(pReq->header.size >= sizeof(VMMDevSharedModuleRegistrationRequest),
2096 ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2097 AssertMsgReturn(pReq->header.size == RT_UOFFSETOF(VMMDevSharedModuleRegistrationRequest, aRegions[pReq->cRegions]),
2098 ("%u cRegions=%u\n", pReq->header.size, pReq->cRegions), VERR_INVALID_PARAMETER);
2099
2100 AssertReturn(memchr(pReq->szName, '\0', sizeof(pReq->szName)), VERR_INVALID_PARAMETER);
2101 AssertReturn(memchr(pReq->szVersion, '\0', sizeof(pReq->szVersion)), VERR_INVALID_PARAMETER);
2102
2103 /*
2104 * Forward the request to the VMM.
2105 */
2106 return PGMR3SharedModuleRegister(PDMDevHlpGetVM(pThis->pDevIns), pReq->enmGuestOS, pReq->szName, pReq->szVersion,
2107 pReq->GCBaseAddr, pReq->cbModule, pReq->cRegions, pReq->aRegions);
2108}
2109
2110/**
2111 * Handles VMMDevReq_UnregisterSharedModule.
2112 *
2113 * @returns VBox status code that the guest should see.
2114 * @param pThis The VMMDev instance data.
2115 * @param pReqHdr The header of the request to handle.
2116 */
2117static int vmmdevReqHandler_UnregisterSharedModule(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2118{
2119 /*
2120 * Basic input validation.
2121 */
2122 VMMDevSharedModuleUnregistrationRequest *pReq = (VMMDevSharedModuleUnregistrationRequest *)pReqHdr;
2123 AssertMsgReturn(pReq->header.size == sizeof(VMMDevSharedModuleUnregistrationRequest),
2124 ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2125
2126 AssertReturn(memchr(pReq->szName, '\0', sizeof(pReq->szName)), VERR_INVALID_PARAMETER);
2127 AssertReturn(memchr(pReq->szVersion, '\0', sizeof(pReq->szVersion)), VERR_INVALID_PARAMETER);
2128
2129 /*
2130 * Forward the request to the VMM.
2131 */
2132 return PGMR3SharedModuleUnregister(PDMDevHlpGetVM(pThis->pDevIns), pReq->szName, pReq->szVersion,
2133 pReq->GCBaseAddr, pReq->cbModule);
2134}
2135
2136/**
2137 * Handles VMMDevReq_CheckSharedModules.
2138 *
2139 * @returns VBox status code that the guest should see.
2140 * @param pThis The VMMDev instance data.
2141 * @param pReqHdr The header of the request to handle.
2142 */
2143static int vmmdevReqHandler_CheckSharedModules(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2144{
2145 VMMDevSharedModuleCheckRequest *pReq = (VMMDevSharedModuleCheckRequest *)pReqHdr;
2146 AssertMsgReturn(pReq->header.size == sizeof(VMMDevSharedModuleCheckRequest),
2147 ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2148 return PGMR3SharedModuleCheckAll(PDMDevHlpGetVM(pThis->pDevIns));
2149}
2150
2151/**
2152 * Handles VMMDevReq_GetPageSharingStatus.
2153 *
2154 * @returns VBox status code that the guest should see.
2155 * @param pThis The VMMDev instance data.
2156 * @param pReqHdr The header of the request to handle.
2157 */
2158static int vmmdevReqHandler_GetPageSharingStatus(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2159{
2160 VMMDevPageSharingStatusRequest *pReq = (VMMDevPageSharingStatusRequest *)pReqHdr;
2161 AssertMsgReturn(pReq->header.size == sizeof(VMMDevPageSharingStatusRequest),
2162 ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2163
2164 pReq->fEnabled = false;
2165 int rc = pThis->pDrv->pfnIsPageFusionEnabled(pThis->pDrv, &pReq->fEnabled);
2166 if (RT_FAILURE(rc))
2167 pReq->fEnabled = false;
2168 return VINF_SUCCESS;
2169}
2170
2171
2172/**
2173 * Handles VMMDevReq_DebugIsPageShared.
2174 *
2175 * @returns VBox status code that the guest should see.
2176 * @param pThis The VMMDev instance data.
2177 * @param pReqHdr The header of the request to handle.
2178 */
2179static int vmmdevReqHandler_DebugIsPageShared(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2180{
2181 VMMDevPageIsSharedRequest *pReq = (VMMDevPageIsSharedRequest *)pReqHdr;
2182 AssertMsgReturn(pReq->header.size == sizeof(VMMDevPageIsSharedRequest),
2183 ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2184
2185# ifdef DEBUG
2186 return PGMR3SharedModuleGetPageState(PDMDevHlpGetVM(pThis->pDevIns), pReq->GCPtrPage, &pReq->fShared, &pReq->uPageFlags);
2187# else
2188 return VERR_NOT_IMPLEMENTED;
2189# endif
2190}
2191
2192#endif /* VBOX_WITH_PAGE_SHARING */
2193
2194
2195/**
2196 * Handles VMMDevReq_WriteCoreDumpe
2197 *
2198 * @returns VBox status code that the guest should see.
2199 * @param pThis The VMMDev instance data.
2200 * @param pReqHdr Pointer to the request header.
2201 */
2202static int vmmdevReqHandler_WriteCoreDump(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr)
2203{
2204 VMMDevReqWriteCoreDump *pReq = (VMMDevReqWriteCoreDump *)pReqHdr;
2205 AssertMsgReturn(pReq->header.size == sizeof(VMMDevReqWriteCoreDump), ("%u\n", pReq->header.size), VERR_INVALID_PARAMETER);
2206
2207 /*
2208 * Only available if explicitly enabled by the user.
2209 */
2210 if (!pThis->fGuestCoreDumpEnabled)
2211 return VERR_ACCESS_DENIED;
2212
2213 /*
2214 * User makes sure the directory exists before composing the path.
2215 */
2216 if (!RTDirExists(pThis->szGuestCoreDumpDir))
2217 return VERR_PATH_NOT_FOUND;
2218
2219 char szCorePath[RTPATH_MAX];
2220 RTStrCopy(szCorePath, sizeof(szCorePath), pThis->szGuestCoreDumpDir);
2221 RTPathAppend(szCorePath, sizeof(szCorePath), "VBox.core");
2222
2223 /*
2224 * Rotate existing cores based on number of additional cores to keep around.
2225 */
2226 if (pThis->cGuestCoreDumps > 0)
2227 for (int64_t i = pThis->cGuestCoreDumps - 1; i >= 0; i--)
2228 {
2229 char szFilePathOld[RTPATH_MAX];
2230 if (i == 0)
2231 RTStrCopy(szFilePathOld, sizeof(szFilePathOld), szCorePath);
2232 else
2233 RTStrPrintf(szFilePathOld, sizeof(szFilePathOld), "%s.%lld", szCorePath, i);
2234
2235 char szFilePathNew[RTPATH_MAX];
2236 RTStrPrintf(szFilePathNew, sizeof(szFilePathNew), "%s.%lld", szCorePath, i + 1);
2237 int vrc = RTFileMove(szFilePathOld, szFilePathNew, RTFILEMOVE_FLAGS_REPLACE);
2238 if (vrc == VERR_FILE_NOT_FOUND)
2239 RTFileDelete(szFilePathNew);
2240 }
2241
2242 /*
2243 * Write the core file.
2244 */
2245 PUVM pUVM = PDMDevHlpGetUVM(pThis->pDevIns);
2246 return DBGFR3CoreWrite(pUVM, szCorePath, true /*fReplaceFile*/);
2247}
2248
2249
2250/**
2251 * Dispatch the request to the appropriate handler function.
2252 *
2253 * @returns Port I/O handler exit code.
2254 * @param pThis The VMM device instance data.
2255 * @param pReqHdr The request header (cached in host memory).
2256 * @param GCPhysReqHdr The guest physical address of the request (for
2257 * HGCM).
2258 * @param pfDelayedUnlock Where to indicate whether the critical section exit
2259 * needs to be delayed till after the request has been
2260 * written back. This is a HGCM kludge, see critsect
2261 * work in hgcmCompletedWorker for more details.
2262 */
2263static int vmmdevReqDispatcher(PVMMDEV pThis, VMMDevRequestHeader *pReqHdr, RTGCPHYS GCPhysReqHdr, bool *pfDelayedUnlock)
2264{
2265 int rcRet = VINF_SUCCESS;
2266 *pfDelayedUnlock = false;
2267
2268 switch (pReqHdr->requestType)
2269 {
2270 case VMMDevReq_ReportGuestInfo:
2271 pReqHdr->rc = vmmdevReqHandler_ReportGuestInfo(pThis, pReqHdr);
2272 break;
2273
2274 case VMMDevReq_ReportGuestInfo2:
2275 pReqHdr->rc = vmmdevReqHandler_ReportGuestInfo2(pThis, pReqHdr);
2276 break;
2277
2278 case VMMDevReq_ReportGuestStatus:
2279 pReqHdr->rc = vmmdevReqHandler_ReportGuestStatus(pThis, pReqHdr);
2280 break;
2281
2282 case VMMDevReq_ReportGuestUserState:
2283 pReqHdr->rc = vmmdevReqHandler_ReportGuestUserState(pThis, pReqHdr);
2284 break;
2285
2286 case VMMDevReq_ReportGuestCapabilities:
2287 pReqHdr->rc = vmmdevReqHandler_ReportGuestCapabilities(pThis, pReqHdr);
2288 break;
2289
2290 case VMMDevReq_SetGuestCapabilities:
2291 pReqHdr->rc = vmmdevReqHandler_SetGuestCapabilities(pThis, pReqHdr);
2292 break;
2293
2294 case VMMDevReq_WriteCoreDump:
2295 pReqHdr->rc = vmmdevReqHandler_WriteCoreDump(pThis, pReqHdr);
2296 break;
2297
2298 case VMMDevReq_GetMouseStatus:
2299 pReqHdr->rc = vmmdevReqHandler_GetMouseStatus(pThis, pReqHdr);
2300 break;
2301
2302 case VMMDevReq_SetMouseStatus:
2303 pReqHdr->rc = vmmdevReqHandler_SetMouseStatus(pThis, pReqHdr);
2304 break;
2305
2306 case VMMDevReq_SetPointerShape:
2307 pReqHdr->rc = vmmdevReqHandler_SetPointerShape(pThis, pReqHdr);
2308 break;
2309
2310 case VMMDevReq_GetHostTime:
2311 pReqHdr->rc = vmmdevReqHandler_GetHostTime(pThis, pReqHdr);
2312 break;
2313
2314 case VMMDevReq_GetHypervisorInfo:
2315 pReqHdr->rc = vmmdevReqHandler_GetHypervisorInfo(pThis, pReqHdr);
2316 break;
2317
2318 case VMMDevReq_SetHypervisorInfo:
2319 pReqHdr->rc = vmmdevReqHandler_SetHypervisorInfo(pThis, pReqHdr);
2320 break;
2321
2322 case VMMDevReq_RegisterPatchMemory:
2323 pReqHdr->rc = vmmdevReqHandler_RegisterPatchMemory(pThis, pReqHdr);
2324 break;
2325
2326 case VMMDevReq_DeregisterPatchMemory:
2327 pReqHdr->rc = vmmdevReqHandler_DeregisterPatchMemory(pThis, pReqHdr);
2328 break;
2329
2330 case VMMDevReq_SetPowerStatus:
2331 {
2332 int rc = pReqHdr->rc = vmmdevReqHandler_SetPowerStatus(pThis, pReqHdr);
2333 if (rc != VINF_SUCCESS && RT_SUCCESS(rc))
2334 rcRet = rc;
2335 break;
2336 }
2337
2338 case VMMDevReq_GetDisplayChangeRequest:
2339 pReqHdr->rc = vmmdevReqHandler_GetDisplayChangeRequest(pThis, pReqHdr);
2340 break;
2341
2342 case VMMDevReq_GetDisplayChangeRequest2:
2343 pReqHdr->rc = vmmdevReqHandler_GetDisplayChangeRequest2(pThis, pReqHdr);
2344 break;
2345
2346 case VMMDevReq_GetDisplayChangeRequestEx:
2347 pReqHdr->rc = vmmdevReqHandler_GetDisplayChangeRequestEx(pThis, pReqHdr);
2348 break;
2349
2350 case VMMDevReq_VideoModeSupported:
2351 pReqHdr->rc = vmmdevReqHandler_VideoModeSupported(pThis, pReqHdr);
2352 break;
2353
2354 case VMMDevReq_VideoModeSupported2:
2355 pReqHdr->rc = vmmdevReqHandler_VideoModeSupported2(pThis, pReqHdr);
2356 break;
2357
2358 case VMMDevReq_GetHeightReduction:
2359 pReqHdr->rc = vmmdevReqHandler_GetHeightReduction(pThis, pReqHdr);
2360 break;
2361
2362 case VMMDevReq_AcknowledgeEvents:
2363 pReqHdr->rc = vmmdevReqHandler_AcknowledgeEvents(pThis, pReqHdr);
2364 break;
2365
2366 case VMMDevReq_CtlGuestFilterMask:
2367 pReqHdr->rc = vmmdevReqHandler_CtlGuestFilterMask(pThis, pReqHdr);
2368 break;
2369
2370#ifdef VBOX_WITH_HGCM
2371 case VMMDevReq_HGCMConnect:
2372 pReqHdr->rc = vmmdevReqHandler_HGCMConnect(pThis, pReqHdr, GCPhysReqHdr);
2373 *pfDelayedUnlock = true;
2374 break;
2375
2376 case VMMDevReq_HGCMDisconnect:
2377 pReqHdr->rc = vmmdevReqHandler_HGCMDisconnect(pThis, pReqHdr, GCPhysReqHdr);
2378 *pfDelayedUnlock = true;
2379 break;
2380
2381# ifdef VBOX_WITH_64_BITS_GUESTS
2382 case VMMDevReq_HGCMCall32:
2383 case VMMDevReq_HGCMCall64:
2384# else
2385 case VMMDevReq_HGCMCall:
2386# endif /* VBOX_WITH_64_BITS_GUESTS */
2387 pReqHdr->rc = vmmdevReqHandler_HGCMCall(pThis, pReqHdr, GCPhysReqHdr);
2388 *pfDelayedUnlock = true;
2389 break;
2390#endif /* VBOX_WITH_HGCM */
2391
2392 case VMMDevReq_HGCMCancel:
2393 pReqHdr->rc = vmmdevReqHandler_HGCMCancel(pThis, pReqHdr, GCPhysReqHdr);
2394 *pfDelayedUnlock = true;
2395 break;
2396
2397 case VMMDevReq_HGCMCancel2:
2398 pReqHdr->rc = vmmdevReqHandler_HGCMCancel2(pThis, pReqHdr);
2399 break;
2400
2401 case VMMDevReq_VideoAccelEnable:
2402 pReqHdr->rc = vmmdevReqHandler_VideoAccelEnable(pThis, pReqHdr);
2403 break;
2404
2405 case VMMDevReq_VideoAccelFlush:
2406 pReqHdr->rc = vmmdevReqHandler_VideoAccelFlush(pThis, pReqHdr);
2407 break;
2408
2409 case VMMDevReq_VideoSetVisibleRegion:
2410 pReqHdr->rc = vmmdevReqHandler_VideoSetVisibleRegion(pThis, pReqHdr);
2411 break;
2412
2413 case VMMDevReq_GetSeamlessChangeRequest:
2414 pReqHdr->rc = vmmdevReqHandler_GetSeamlessChangeRequest(pThis, pReqHdr);
2415 break;
2416
2417 case VMMDevReq_GetVRDPChangeRequest:
2418 pReqHdr->rc = vmmdevReqHandler_GetVRDPChangeRequest(pThis, pReqHdr);
2419 break;
2420
2421 case VMMDevReq_GetMemBalloonChangeRequest:
2422 pReqHdr->rc = vmmdevReqHandler_GetMemBalloonChangeRequest(pThis, pReqHdr);
2423 break;
2424
2425 case VMMDevReq_ChangeMemBalloon:
2426 pReqHdr->rc = vmmdevReqHandler_ChangeMemBalloon(pThis, pReqHdr);
2427 break;
2428
2429 case VMMDevReq_GetStatisticsChangeRequest:
2430 pReqHdr->rc = vmmdevReqHandler_GetStatisticsChangeRequest(pThis, pReqHdr);
2431 break;
2432
2433 case VMMDevReq_ReportGuestStats:
2434 pReqHdr->rc = vmmdevReqHandler_ReportGuestStats(pThis, pReqHdr);
2435 break;
2436
2437 case VMMDevReq_QueryCredentials:
2438 pReqHdr->rc = vmmdevReqHandler_QueryCredentials(pThis, pReqHdr);
2439 break;
2440
2441 case VMMDevReq_ReportCredentialsJudgement:
2442 pReqHdr->rc = vmmdevReqHandler_ReportCredentialsJudgement(pThis, pReqHdr);
2443 break;
2444
2445 case VMMDevReq_GetHostVersion:
2446 pReqHdr->rc = vmmdevReqHandler_GetHostVersion(pThis, pReqHdr);
2447 break;
2448
2449 case VMMDevReq_GetCpuHotPlugRequest:
2450 pReqHdr->rc = vmmdevReqHandler_GetCpuHotPlugRequest(pThis, pReqHdr);
2451 break;
2452
2453 case VMMDevReq_SetCpuHotPlugStatus:
2454 pReqHdr->rc = vmmdevReqHandler_SetCpuHotPlugStatus(pThis, pReqHdr);
2455 break;
2456
2457#ifdef VBOX_WITH_PAGE_SHARING
2458 case VMMDevReq_RegisterSharedModule:
2459 pReqHdr->rc = vmmdevReqHandler_RegisterSharedModule(pThis, pReqHdr);
2460 break;
2461
2462 case VMMDevReq_UnregisterSharedModule:
2463 pReqHdr->rc = vmmdevReqHandler_UnregisterSharedModule(pThis, pReqHdr);
2464 break;
2465
2466 case VMMDevReq_CheckSharedModules:
2467 pReqHdr->rc = vmmdevReqHandler_CheckSharedModules(pThis, pReqHdr);
2468 break;
2469
2470 case VMMDevReq_GetPageSharingStatus:
2471 pReqHdr->rc = vmmdevReqHandler_GetPageSharingStatus(pThis, pReqHdr);
2472 break;
2473
2474 case VMMDevReq_DebugIsPageShared:
2475 pReqHdr->rc = vmmdevReqHandler_DebugIsPageShared(pThis, pReqHdr);
2476 break;
2477
2478#endif /* VBOX_WITH_PAGE_SHARING */
2479
2480#ifdef DEBUG
2481 case VMMDevReq_LogString:
2482 pReqHdr->rc = vmmdevReqHandler_LogString(pThis, pReqHdr);
2483 break;
2484#endif
2485
2486 case VMMDevReq_GetSessionId:
2487 pReqHdr->rc = vmmdevReqHandler_GetSessionId(pThis, pReqHdr);
2488 break;
2489
2490 /*
2491 * Guest wants to give up a timeslice.
2492 * Note! This was only ever used by experimental GAs!
2493 */
2494 /** @todo maybe we could just remove this? */
2495 case VMMDevReq_Idle:
2496 {
2497 /* just return to EMT telling it that we want to halt */
2498 rcRet = VINF_EM_HALT;
2499 break;
2500 }
2501
2502 default:
2503 {
2504 pReqHdr->rc = VERR_NOT_IMPLEMENTED;
2505 Log(("VMMDev unknown request type %d\n", pReqHdr->requestType));
2506 break;
2507 }
2508 }
2509 return rcRet;
2510}
2511
2512
2513/**
2514 * @callback_method_impl{FNIOMIOPORTOUT, Port I/O Handler for the generic
2515 * request interface.}
2516 */
2517static DECLCALLBACK(int) vmmdevRequestHandler(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2518{
2519 PVMMDEV pThis = (VMMDevState*)pvUser;
2520
2521 /*
2522 * The caller has passed the guest context physical address of the request
2523 * structure. We'll copy all of it into a heap buffer eventually, but we
2524 * will have to start off with the header.
2525 */
2526 VMMDevRequestHeader requestHeader;
2527 RT_ZERO(requestHeader);
2528 PDMDevHlpPhysRead(pDevIns, (RTGCPHYS)u32, &requestHeader, sizeof(requestHeader));
2529
2530 /* The structure size must be greater or equal to the header size. */
2531 if (requestHeader.size < sizeof(VMMDevRequestHeader))
2532 {
2533 Log(("VMMDev request header size too small! size = %d\n", requestHeader.size));
2534 return VINF_SUCCESS;
2535 }
2536
2537 /* Check the version of the header structure. */
2538 if (requestHeader.version != VMMDEV_REQUEST_HEADER_VERSION)
2539 {
2540 Log(("VMMDev: guest header version (0x%08X) differs from ours (0x%08X)\n", requestHeader.version, VMMDEV_REQUEST_HEADER_VERSION));
2541 return VINF_SUCCESS;
2542 }
2543
2544 Log2(("VMMDev request issued: %d\n", requestHeader.requestType));
2545
2546 int rcRet = VINF_SUCCESS;
2547 bool fDelayedUnlock = false;
2548 VMMDevRequestHeader *pRequestHeader = NULL;
2549
2550 /* Check that is doesn't exceed the max packet size. */
2551 if (requestHeader.size <= VMMDEV_MAX_VMMDEVREQ_SIZE)
2552 {
2553 /*
2554 * We require the GAs to report it's information before we let it have
2555 * access to all the functions. The VMMDevReq_ReportGuestInfo request
2556 * is the one which unlocks the access. Newer additions will first
2557 * issue VMMDevReq_ReportGuestInfo2, older ones doesn't know this one.
2558 * Two exceptions: VMMDevReq_GetHostVersion and VMMDevReq_WriteCoreDump.
2559 */
2560 if ( pThis->fu32AdditionsOk
2561 || requestHeader.requestType == VMMDevReq_ReportGuestInfo2
2562 || requestHeader.requestType == VMMDevReq_ReportGuestInfo
2563 || requestHeader.requestType == VMMDevReq_WriteCoreDump
2564 || requestHeader.requestType == VMMDevReq_GetHostVersion
2565 )
2566 {
2567 /*
2568 * The request looks fine. Allocate a heap block for it, read the
2569 * entire package from guest memory and feed it to the dispatcher.
2570 */
2571 pRequestHeader = (VMMDevRequestHeader *)RTMemAlloc(requestHeader.size);
2572 if (pRequestHeader)
2573 {
2574 memcpy(pRequestHeader, &requestHeader, sizeof(VMMDevRequestHeader));
2575 size_t cbLeft = requestHeader.size - sizeof(VMMDevRequestHeader);
2576 if (cbLeft)
2577 PDMDevHlpPhysRead(pDevIns,
2578 (RTGCPHYS)u32 + sizeof(VMMDevRequestHeader),
2579 (uint8_t *)pRequestHeader + sizeof(VMMDevRequestHeader),
2580 cbLeft);
2581
2582 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
2583 rcRet = vmmdevReqDispatcher(pThis, pRequestHeader, u32, &fDelayedUnlock);
2584 if (!fDelayedUnlock)
2585 PDMCritSectLeave(&pThis->CritSect);
2586 }
2587 else
2588 {
2589 Log(("VMMDev: RTMemAlloc failed!\n"));
2590 requestHeader.rc = VERR_NO_MEMORY;
2591 }
2592 }
2593 else
2594 {
2595 static int s_cRelWarn;
2596 if (s_cRelWarn < 10)
2597 {
2598 s_cRelWarn++;
2599 LogRel(("VMMDev: the guest has not yet reported to us -- refusing operation of request #%d\n",
2600 requestHeader.requestType));
2601 }
2602 requestHeader.rc = VERR_NOT_SUPPORTED;
2603 }
2604 }
2605 else
2606 {
2607 static int s_cRelWarn;
2608 if (s_cRelWarn < 50)
2609 {
2610 s_cRelWarn++;
2611 LogRel(("VMMDev: request packet too big (%x). Refusing operation.\n", requestHeader.size));
2612 }
2613 requestHeader.rc = VERR_NOT_SUPPORTED;
2614 }
2615
2616 /*
2617 * Write the result back to guest memory
2618 */
2619 if (pRequestHeader)
2620 {
2621 PDMDevHlpPhysWrite(pDevIns, u32, pRequestHeader, pRequestHeader->size);
2622 if (fDelayedUnlock)
2623 PDMCritSectLeave(&pThis->CritSect);
2624 RTMemFree(pRequestHeader);
2625 }
2626 else
2627 {
2628 /* early error case; write back header only */
2629 PDMDevHlpPhysWrite(pDevIns, u32, &requestHeader, sizeof(requestHeader));
2630 Assert(!fDelayedUnlock);
2631 }
2632
2633 return rcRet;
2634}
2635
2636
2637/* -=-=-=-=-=- PCI Device -=-=-=-=-=- */
2638
2639
2640/**
2641 * @interface_method_impl{FNPCIIOREGIONMAP, MMIO/MMIO2 regions}
2642 */
2643static DECLCALLBACK(int)
2644vmmdevIORAMRegionMap(PPCIDEVICE pPciDev, int iRegion, RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
2645{
2646 LogFlow(("vmmdevR3IORAMRegionMap: iRegion=%d GCPhysAddress=%RGp cb=%#x enmType=%d\n", iRegion, GCPhysAddress, cb, enmType));
2647 PVMMDEV pThis = RT_FROM_MEMBER(pPciDev, VMMDEV, PciDev);
2648 int rc;
2649
2650 if (iRegion == 1)
2651 {
2652 AssertReturn(enmType == PCI_ADDRESS_SPACE_MEM, VERR_INTERNAL_ERROR);
2653 Assert(pThis->pVMMDevRAMR3 != NULL);
2654 if (GCPhysAddress != NIL_RTGCPHYS)
2655 {
2656 /*
2657 * Map the MMIO2 memory.
2658 */
2659 pThis->GCPhysVMMDevRAM = GCPhysAddress;
2660 Assert(pThis->GCPhysVMMDevRAM == GCPhysAddress);
2661 rc = PDMDevHlpMMIO2Map(pPciDev->pDevIns, iRegion, GCPhysAddress);
2662 }
2663 else
2664 {
2665 /*
2666 * It is about to be unmapped, just clean up.
2667 */
2668 pThis->GCPhysVMMDevRAM = NIL_RTGCPHYS32;
2669 rc = VINF_SUCCESS;
2670 }
2671 }
2672 else if (iRegion == 2)
2673 {
2674 AssertReturn(enmType == PCI_ADDRESS_SPACE_MEM_PREFETCH, VERR_INTERNAL_ERROR);
2675 Assert(pThis->pVMMDevHeapR3 != NULL);
2676 if (GCPhysAddress != NIL_RTGCPHYS)
2677 {
2678 /*
2679 * Map the MMIO2 memory.
2680 */
2681 pThis->GCPhysVMMDevHeap = GCPhysAddress;
2682 Assert(pThis->GCPhysVMMDevHeap == GCPhysAddress);
2683 rc = PDMDevHlpMMIO2Map(pPciDev->pDevIns, iRegion, GCPhysAddress);
2684 if (RT_SUCCESS(rc))
2685 rc = PDMDevHlpRegisterVMMDevHeap(pPciDev->pDevIns, GCPhysAddress, pThis->pVMMDevHeapR3, VMMDEV_HEAP_SIZE);
2686 }
2687 else
2688 {
2689 /*
2690 * It is about to be unmapped, just clean up.
2691 */
2692 PDMDevHlpUnregisterVMMDevHeap(pPciDev->pDevIns, pThis->GCPhysVMMDevHeap);
2693 pThis->GCPhysVMMDevHeap = NIL_RTGCPHYS32;
2694 rc = VINF_SUCCESS;
2695 }
2696 }
2697 else
2698 {
2699 AssertMsgFailed(("%d\n", iRegion));
2700 rc = VERR_INVALID_PARAMETER;
2701 }
2702
2703 return rc;
2704}
2705
2706
2707/**
2708 * @interface_method_impl{FNPCIIOREGIONMAP, I/O Port Region}
2709 */
2710static DECLCALLBACK(int)
2711vmmdevIOPortRegionMap(PPCIDEVICE pPciDev, int iRegion, RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
2712{
2713 PVMMDEV pThis = RT_FROM_MEMBER(pPciDev, VMMDEV, PciDev);
2714
2715 Assert(enmType == PCI_ADDRESS_SPACE_IO);
2716 Assert(iRegion == 0);
2717 AssertMsg(RT_ALIGN(GCPhysAddress, 8) == GCPhysAddress, ("Expected 8 byte alignment. GCPhysAddress=%#x\n", GCPhysAddress));
2718
2719 /*
2720 * Register our port IO handlers.
2721 */
2722 int rc = PDMDevHlpIOPortRegister(pPciDev->pDevIns, (RTIOPORT)GCPhysAddress + VMMDEV_PORT_OFF_REQUEST, 1,
2723 pThis, vmmdevRequestHandler, NULL, NULL, NULL, "VMMDev Request Handler");
2724 AssertRC(rc);
2725 return rc;
2726}
2727
2728
2729
2730/* -=-=-=-=-=- Backdoor Logging and Time Sync. -=-=-=-=-=- */
2731
2732/**
2733 * @callback_method_impl{FNIOMIOPORTOUT, Backdoor Logging.}
2734 */
2735static DECLCALLBACK(int) vmmdevBackdoorLog(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2736{
2737 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
2738
2739 if (!pThis->fBackdoorLogDisabled && cb == 1 && Port == RTLOG_DEBUG_PORT)
2740 {
2741
2742 /* The raw version. */
2743 switch (u32)
2744 {
2745 case '\r': LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: <return>\n")); break;
2746 case '\n': LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: <newline>\n")); break;
2747 case '\t': LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: <tab>\n")); break;
2748 default: LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: %c (%02x)\n", u32, u32)); break;
2749 }
2750
2751 /* The readable, buffered version. */
2752 if (u32 == '\n' || u32 == '\r')
2753 {
2754 pThis->szMsg[pThis->iMsg] = '\0';
2755 if (pThis->iMsg)
2756 LogRelIt(LOG_REL_INSTANCE, RTLOGGRPFLAGS_LEVEL_1, LOG_GROUP_DEV_VMM_BACKDOOR, ("Guest Log: %s\n", pThis->szMsg));
2757 pThis->iMsg = 0;
2758 }
2759 else
2760 {
2761 if (pThis->iMsg >= sizeof(pThis->szMsg)-1)
2762 {
2763 pThis->szMsg[pThis->iMsg] = '\0';
2764 LogRelIt(LOG_REL_INSTANCE, RTLOGGRPFLAGS_LEVEL_1, LOG_GROUP_DEV_VMM_BACKDOOR, ("Guest Log: %s\n", pThis->szMsg));
2765 pThis->iMsg = 0;
2766 }
2767 pThis->szMsg[pThis->iMsg] = (char )u32;
2768 pThis->szMsg[++pThis->iMsg] = '\0';
2769 }
2770 }
2771 return VINF_SUCCESS;
2772}
2773
2774#ifdef VMMDEV_WITH_ALT_TIMESYNC
2775
2776/**
2777 * @callback_method_impl{FNIOMIOPORTOUT, Alternative time synchronization.}
2778 */
2779static DECLCALLBACK(int) vmmdevAltTimeSyncWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
2780{
2781 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
2782 if (cb == 4)
2783 {
2784 /* Selects high (0) or low (1) DWORD. The high has to be read first. */
2785 switch (u32)
2786 {
2787 case 0:
2788 pThis->fTimesyncBackdoorLo = false;
2789 break;
2790 case 1:
2791 pThis->fTimesyncBackdoorLo = true;
2792 break;
2793 default:
2794 Log(("vmmdevAltTimeSyncWrite: Invalid access cb=%#x u32=%#x\n", cb, u32));
2795 break;
2796 }
2797 }
2798 else
2799 Log(("vmmdevAltTimeSyncWrite: Invalid access cb=%#x u32=%#x\n", cb, u32));
2800 return VINF_SUCCESS;
2801}
2802
2803/**
2804 * @callback_method_impl{FNIOMIOPORTOUT, Alternative time synchronization.}
2805 */
2806static DECLCALLBACK(int) vmmdevAltTimeSyncRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
2807{
2808 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
2809 int rc;
2810 if (cb == 4)
2811 {
2812 if (pThis->fTimesyncBackdoorLo)
2813 *pu32 = (uint32_t)pThis->hostTime;
2814 else
2815 {
2816 /* Reading the high dword gets and saves the current time. */
2817 RTTIMESPEC Now;
2818 pThis->hostTime = RTTimeSpecGetMilli(PDMDevHlpTMUtcNow(pDevIns, &Now));
2819 *pu32 = (uint32_t)(pThis->hostTime >> 32);
2820 }
2821 rc = VINF_SUCCESS;
2822 }
2823 else
2824 {
2825 Log(("vmmdevAltTimeSyncRead: Invalid access cb=%#x\n", cb));
2826 rc = VERR_IOM_IOPORT_UNUSED;
2827 }
2828 return rc;
2829}
2830
2831#endif /* VMMDEV_WITH_ALT_TIMESYNC */
2832
2833
2834/* -=-=-=-=-=- IBase -=-=-=-=-=- */
2835
2836/**
2837 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
2838 */
2839static DECLCALLBACK(void *) vmmdevPortQueryInterface(PPDMIBASE pInterface, const char *pszIID)
2840{
2841 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IBase);
2842
2843 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
2844 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIVMMDEVPORT, &pThis->IPort);
2845#ifdef VBOX_WITH_HGCM
2846 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHGCMPORT, &pThis->IHGCMPort);
2847#endif
2848 /* Currently only for shared folders. */
2849 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->SharedFolders.ILeds);
2850 return NULL;
2851}
2852
2853
2854/* -=-=-=-=-=- ILeds -=-=-=-=-=- */
2855
2856/**
2857 * Gets the pointer to the status LED of a unit.
2858 *
2859 * @returns VBox status code.
2860 * @param pInterface Pointer to the interface structure containing the called function pointer.
2861 * @param iLUN The unit which status LED we desire.
2862 * @param ppLed Where to store the LED pointer.
2863 */
2864static DECLCALLBACK(int) vmmdevQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
2865{
2866 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, SharedFolders.ILeds);
2867 if (iLUN == 0) /* LUN 0 is shared folders */
2868 {
2869 *ppLed = &pThis->SharedFolders.Led;
2870 return VINF_SUCCESS;
2871 }
2872 return VERR_PDM_LUN_NOT_FOUND;
2873}
2874
2875
2876/* -=-=-=-=-=- PDMIVMMDEVPORT (VMMDEV::IPort) -=-=-=-=-=- */
2877
2878/**
2879 * @interface_method_impl{PDMIVMMDEVPORT, pfnQueryAbsoluteMouse}
2880 */
2881static DECLCALLBACK(int) vmmdevIPort_QueryAbsoluteMouse(PPDMIVMMDEVPORT pInterface, int32_t *pxAbs, int32_t *pyAbs)
2882{
2883 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
2884
2885 /** @todo at the first sign of trouble in this area, just enter the critsect.
2886 * As indicated by the comment below, the atomic reads serves no real purpose
2887 * here since we can assume cache coherency protocoles and int32_t alignment
2888 * rules making sure we won't see a halfwritten value. */
2889 if (pxAbs)
2890 *pxAbs = ASMAtomicReadS32(&pThis->mouseXAbs); /* why the atomic read? */
2891 if (pyAbs)
2892 *pyAbs = ASMAtomicReadS32(&pThis->mouseYAbs);
2893
2894 return VINF_SUCCESS;
2895}
2896
2897/**
2898 * @interface_method_impl{PDMIVMMDEVPORT, pfnSetAbsoluteMouse}
2899 */
2900static DECLCALLBACK(int) vmmdevIPort_SetAbsoluteMouse(PPDMIVMMDEVPORT pInterface, int32_t xAbs, int32_t yAbs)
2901{
2902 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
2903 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
2904
2905 if ( pThis->mouseXAbs != xAbs
2906 || pThis->mouseYAbs != yAbs)
2907 {
2908 Log2(("vmmdevIPort_SetAbsoluteMouse : settings absolute position to x = %d, y = %d\n", xAbs, yAbs));
2909 pThis->mouseXAbs = xAbs;
2910 pThis->mouseYAbs = yAbs;
2911 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
2912 }
2913
2914 PDMCritSectLeave(&pThis->CritSect);
2915 return VINF_SUCCESS;
2916}
2917
2918/**
2919 * @interface_method_impl{PDMIVMMDEVPORT, pfnQueryMouseCapabilities}
2920 */
2921static DECLCALLBACK(int) vmmdevIPort_QueryMouseCapabilities(PPDMIVMMDEVPORT pInterface, uint32_t *pfCapabilities)
2922{
2923 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
2924 AssertPtrReturn(pfCapabilities, VERR_INVALID_PARAMETER);
2925
2926 *pfCapabilities = pThis->mouseCapabilities;
2927 return VINF_SUCCESS;
2928}
2929
2930/**
2931 * @interface_method_impl{PDMIVMMDEVPORT, pfnUpdateMouseCapabilities}
2932 */
2933static DECLCALLBACK(int)
2934vmmdevIPort_UpdateMouseCapabilities(PPDMIVMMDEVPORT pInterface, uint32_t fCapsAdded, uint32_t fCapsRemoved)
2935{
2936 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
2937 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
2938
2939 uint32_t fOldCaps = pThis->mouseCapabilities;
2940 pThis->mouseCapabilities &= ~(fCapsRemoved & VMMDEV_MOUSE_HOST_MASK);
2941 pThis->mouseCapabilities |= (fCapsAdded & VMMDEV_MOUSE_HOST_MASK)
2942 | VMMDEV_MOUSE_HOST_RECHECKS_NEEDS_HOST_CURSOR;
2943 bool fNotify = fOldCaps != pThis->mouseCapabilities;
2944
2945 LogRelFlowFunc(("fCapsAdded=0x%x, fCapsRemoved=0x%x, fNotify=%RTbool\n", fCapsAdded, fCapsRemoved, fNotify));
2946
2947 if (fNotify)
2948 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_MOUSE_CAPABILITIES_CHANGED);
2949
2950 PDMCritSectLeave(&pThis->CritSect);
2951 return VINF_SUCCESS;
2952}
2953
2954/**
2955 * @interface_method_impl{PDMIVMMDEVPORT, pfnRequestDisplayChange}
2956 */
2957static DECLCALLBACK(int)
2958vmmdevIPort_RequestDisplayChange(PPDMIVMMDEVPORT pInterface, uint32_t cx, uint32_t cy, uint32_t cBits, uint32_t idxDisplay,
2959 int32_t xOrigin, int32_t yOrigin, bool fEnabled, bool fChangeOrigin)
2960{
2961 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
2962
2963 if (idxDisplay >= RT_ELEMENTS(pThis->displayChangeData.aRequests))
2964 return VERR_INVALID_PARAMETER;
2965
2966 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
2967
2968 DISPLAYCHANGEREQUEST *pRequest = &pThis->displayChangeData.aRequests[idxDisplay];
2969
2970 /* Verify that the new resolution is different and that guest does not yet know about it. */
2971 bool fSameResolution = (!cx || pRequest->lastReadDisplayChangeRequest.xres == cx)
2972 && (!cy || pRequest->lastReadDisplayChangeRequest.yres == cy)
2973 && (!cBits || pRequest->lastReadDisplayChangeRequest.bpp == cBits)
2974 && pRequest->lastReadDisplayChangeRequest.xOrigin == xOrigin
2975 && pRequest->lastReadDisplayChangeRequest.yOrigin == yOrigin
2976 && pRequest->lastReadDisplayChangeRequest.fEnabled == fEnabled
2977 && pRequest->lastReadDisplayChangeRequest.display == idxDisplay;
2978
2979 if (!cx && !cy && !cBits)
2980 {
2981 /* Special case of reset video mode. */
2982 fSameResolution = false;
2983 }
2984
2985 Log3(("vmmdevIPort_RequestDisplayChange: same=%d. new: cx=%d, cy=%d, cBits=%d, idxDisplay=%d.\
2986 old: cx=%d, cy=%d, cBits=%d, idxDisplay=%d. \n \
2987 ,OriginX = %d , OriginY=%d, Enabled=%d, ChangeOrigin=%d\n",
2988 fSameResolution, cx, cy, cBits, idxDisplay, pRequest->lastReadDisplayChangeRequest.xres,
2989 pRequest->lastReadDisplayChangeRequest.yres, pRequest->lastReadDisplayChangeRequest.bpp,
2990 pRequest->lastReadDisplayChangeRequest.display,
2991 xOrigin, yOrigin, fEnabled, fChangeOrigin));
2992
2993 if (!fSameResolution)
2994 {
2995 LogRel(("VMMDev::SetVideoModeHint: got a video mode hint (%dx%dx%d)@(%dx%d),(%d;%d) at %d\n",
2996 cx, cy, cBits, xOrigin, yOrigin, fEnabled, fChangeOrigin, idxDisplay));
2997
2998 /* we could validate the information here but hey, the guest can do that as well! */
2999 pRequest->displayChangeRequest.xres = cx;
3000 pRequest->displayChangeRequest.yres = cy;
3001 pRequest->displayChangeRequest.bpp = cBits;
3002 pRequest->displayChangeRequest.display = idxDisplay;
3003 pRequest->displayChangeRequest.xOrigin = xOrigin;
3004 pRequest->displayChangeRequest.yOrigin = yOrigin;
3005 pRequest->displayChangeRequest.fEnabled = fEnabled;
3006 pRequest->displayChangeRequest.fChangeOrigin = fChangeOrigin;
3007
3008 pRequest->fPending = true;
3009
3010 /* IRQ so the guest knows what's going on */
3011 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST);
3012 }
3013
3014 PDMCritSectLeave(&pThis->CritSect);
3015 return VINF_SUCCESS;
3016}
3017
3018/**
3019 * @interface_method_impl{PDMIVMMDEVPORT, pfnRequestSeamlessChange}
3020 */
3021static DECLCALLBACK(int) vmmdevIPort_RequestSeamlessChange(PPDMIVMMDEVPORT pInterface, bool fEnabled)
3022{
3023 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3024 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3025
3026 /* Verify that the new resolution is different and that guest does not yet know about it. */
3027 bool fSameMode = (pThis->fLastSeamlessEnabled == fEnabled);
3028
3029 Log(("vmmdevIPort_RequestSeamlessChange: same=%d. new=%d\n", fSameMode, fEnabled));
3030
3031 if (!fSameMode)
3032 {
3033 /* we could validate the information here but hey, the guest can do that as well! */
3034 pThis->fSeamlessEnabled = fEnabled;
3035
3036 /* IRQ so the guest knows what's going on */
3037 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST);
3038 }
3039
3040 PDMCritSectLeave(&pThis->CritSect);
3041 return VINF_SUCCESS;
3042}
3043
3044/**
3045 * @interface_method_impl{PDMIVMMDEVPORT, pfnSetMemoryBalloon}
3046 */
3047static DECLCALLBACK(int) vmmdevIPort_SetMemoryBalloon(PPDMIVMMDEVPORT pInterface, uint32_t cMbBalloon)
3048{
3049 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3050 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3051
3052 /* Verify that the new resolution is different and that guest does not yet know about it. */
3053 Log(("vmmdevIPort_SetMemoryBalloon: old=%u new=%u\n", pThis->cMbMemoryBalloonLast, cMbBalloon));
3054 if (pThis->cMbMemoryBalloonLast != cMbBalloon)
3055 {
3056 /* we could validate the information here but hey, the guest can do that as well! */
3057 pThis->cMbMemoryBalloon = cMbBalloon;
3058
3059 /* IRQ so the guest knows what's going on */
3060 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_BALLOON_CHANGE_REQUEST);
3061 }
3062
3063 PDMCritSectLeave(&pThis->CritSect);
3064 return VINF_SUCCESS;
3065}
3066
3067/**
3068 * @interface_method_impl{PDMIVMMDEVPORT, pfnVRDPChange}
3069 */
3070static DECLCALLBACK(int) vmmdevIPort_VRDPChange(PPDMIVMMDEVPORT pInterface, bool fVRDPEnabled, uint32_t uVRDPExperienceLevel)
3071{
3072 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3073 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3074
3075 bool fSame = (pThis->fVRDPEnabled == fVRDPEnabled);
3076
3077 Log(("vmmdevIPort_VRDPChange: old=%d. new=%d\n", pThis->fVRDPEnabled, fVRDPEnabled));
3078
3079 if (!fSame)
3080 {
3081 pThis->fVRDPEnabled = fVRDPEnabled;
3082 pThis->uVRDPExperienceLevel = uVRDPExperienceLevel;
3083
3084 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_VRDP);
3085 }
3086
3087 PDMCritSectLeave(&pThis->CritSect);
3088 return VINF_SUCCESS;
3089}
3090
3091/**
3092 * @interface_method_impl{PDMIVMMDEVPORT, pfnSetStatisticsInterval}
3093 */
3094static DECLCALLBACK(int) vmmdevIPort_SetStatisticsInterval(PPDMIVMMDEVPORT pInterface, uint32_t cSecsStatInterval)
3095{
3096 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3097 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3098
3099 /* Verify that the new resolution is different and that guest does not yet know about it. */
3100 bool fSame = (pThis->u32LastStatIntervalSize == cSecsStatInterval);
3101
3102 Log(("vmmdevIPort_SetStatisticsInterval: old=%d. new=%d\n", pThis->u32LastStatIntervalSize, cSecsStatInterval));
3103
3104 if (!fSame)
3105 {
3106 /* we could validate the information here but hey, the guest can do that as well! */
3107 pThis->u32StatIntervalSize = cSecsStatInterval;
3108
3109 /* IRQ so the guest knows what's going on */
3110 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST);
3111 }
3112
3113 PDMCritSectLeave(&pThis->CritSect);
3114 return VINF_SUCCESS;
3115}
3116
3117/**
3118 * @interface_method_impl{PDMIVMMDEVPORT, pfnSetCredentials}
3119 */
3120static DECLCALLBACK(int) vmmdevIPort_SetCredentials(PPDMIVMMDEVPORT pInterface, const char *pszUsername,
3121 const char *pszPassword, const char *pszDomain, uint32_t fFlags)
3122{
3123 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3124 AssertReturn(fFlags & (VMMDEV_SETCREDENTIALS_GUESTLOGON | VMMDEV_SETCREDENTIALS_JUDGE), VERR_INVALID_PARAMETER);
3125
3126 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3127
3128 /*
3129 * Logon mode
3130 */
3131 if (fFlags & VMMDEV_SETCREDENTIALS_GUESTLOGON)
3132 {
3133 /* memorize the data */
3134 strcpy(pThis->pCredentials->Logon.szUserName, pszUsername);
3135 strcpy(pThis->pCredentials->Logon.szPassword, pszPassword);
3136 strcpy(pThis->pCredentials->Logon.szDomain, pszDomain);
3137 pThis->pCredentials->Logon.fAllowInteractiveLogon = !(fFlags & VMMDEV_SETCREDENTIALS_NOLOCALLOGON);
3138 }
3139 /*
3140 * Credentials verification mode?
3141 */
3142 else
3143 {
3144 /* memorize the data */
3145 strcpy(pThis->pCredentials->Judge.szUserName, pszUsername);
3146 strcpy(pThis->pCredentials->Judge.szPassword, pszPassword);
3147 strcpy(pThis->pCredentials->Judge.szDomain, pszDomain);
3148
3149 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_JUDGE_CREDENTIALS);
3150 }
3151
3152 PDMCritSectLeave(&pThis->CritSect);
3153 return VINF_SUCCESS;
3154}
3155
3156/**
3157 * @interface_method_impl{PDMIVMMDEVPORT, pfnVBVAChange}
3158 *
3159 * Notification from the Display. Especially useful when acceleration is
3160 * disabled after a video mode change.
3161 */
3162static DECLCALLBACK(void) vmmdevIPort_VBVAChange(PPDMIVMMDEVPORT pInterface, bool fEnabled)
3163{
3164 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3165 Log(("vmmdevIPort_VBVAChange: fEnabled = %d\n", fEnabled));
3166
3167 /* Only used by saved state, which I guess is why we don't bother with locking here. */
3168 pThis->u32VideoAccelEnabled = fEnabled;
3169}
3170
3171/**
3172 * @interface_method_impl{PDMIVMMDEVPORT, pfnCpuHotUnplug}
3173 */
3174static DECLCALLBACK(int) vmmdevIPort_CpuHotUnplug(PPDMIVMMDEVPORT pInterface, uint32_t idCpuCore, uint32_t idCpuPackage)
3175{
3176 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3177 int rc = VINF_SUCCESS;
3178
3179 Log(("vmmdevIPort_CpuHotUnplug: idCpuCore=%u idCpuPackage=%u\n", idCpuCore, idCpuPackage));
3180
3181 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3182
3183 if (pThis->fCpuHotPlugEventsEnabled)
3184 {
3185 pThis->enmCpuHotPlugEvent = VMMDevCpuEventType_Unplug;
3186 pThis->idCpuCore = idCpuCore;
3187 pThis->idCpuPackage = idCpuPackage;
3188 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_CPU_HOTPLUG);
3189 }
3190 else
3191 rc = VERR_CPU_HOTPLUG_NOT_MONITORED_BY_GUEST;
3192
3193 PDMCritSectLeave(&pThis->CritSect);
3194 return rc;
3195}
3196
3197/**
3198 * @interface_method_impl{PDMIVMMDEVPORT, pfnCpuHotPlug}
3199 */
3200static DECLCALLBACK(int) vmmdevIPort_CpuHotPlug(PPDMIVMMDEVPORT pInterface, uint32_t idCpuCore, uint32_t idCpuPackage)
3201{
3202 PVMMDEV pThis = RT_FROM_MEMBER(pInterface, VMMDEV, IPort);
3203 int rc = VINF_SUCCESS;
3204
3205 Log(("vmmdevCpuPlug: idCpuCore=%u idCpuPackage=%u\n", idCpuCore, idCpuPackage));
3206
3207 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3208
3209 if (pThis->fCpuHotPlugEventsEnabled)
3210 {
3211 pThis->enmCpuHotPlugEvent = VMMDevCpuEventType_Plug;
3212 pThis->idCpuCore = idCpuCore;
3213 pThis->idCpuPackage = idCpuPackage;
3214 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_CPU_HOTPLUG);
3215 }
3216 else
3217 rc = VERR_CPU_HOTPLUG_NOT_MONITORED_BY_GUEST;
3218
3219 PDMCritSectLeave(&pThis->CritSect);
3220 return rc;
3221}
3222
3223
3224/* -=-=-=-=-=- Saved State -=-=-=-=-=- */
3225
3226/**
3227 * @callback_method_impl{NSSMDEVLIVEEXEC}
3228 */
3229static DECLCALLBACK(int) vmmdevLiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
3230{
3231 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3232
3233 SSMR3PutBool(pSSM, pThis->fGetHostTimeDisabled);
3234 SSMR3PutBool(pSSM, pThis->fBackdoorLogDisabled);
3235 SSMR3PutBool(pSSM, pThis->fKeepCredentials);
3236 SSMR3PutBool(pSSM, pThis->fHeapEnabled);
3237
3238 return VINF_SSM_DONT_CALL_AGAIN;
3239}
3240
3241
3242/**
3243 * @callback_method_impl{FNSSMDEVSAVEEXEC}
3244 */
3245static DECLCALLBACK(int) vmmdevSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3246{
3247 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3248 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3249
3250 vmmdevLiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
3251
3252 SSMR3PutU32(pSSM, pThis->hypervisorSize);
3253 SSMR3PutU32(pSSM, pThis->mouseCapabilities);
3254 SSMR3PutS32(pSSM, pThis->mouseXAbs);
3255 SSMR3PutS32(pSSM, pThis->mouseYAbs);
3256
3257 SSMR3PutBool(pSSM, pThis->fNewGuestFilterMask);
3258 SSMR3PutU32(pSSM, pThis->u32NewGuestFilterMask);
3259 SSMR3PutU32(pSSM, pThis->u32GuestFilterMask);
3260 SSMR3PutU32(pSSM, pThis->u32HostEventFlags);
3261 /* The following is not strictly necessary as PGM restores MMIO2, keeping it for historical reasons. */
3262 SSMR3PutMem(pSSM, &pThis->pVMMDevRAMR3->V, sizeof(pThis->pVMMDevRAMR3->V));
3263
3264 SSMR3PutMem(pSSM, &pThis->guestInfo, sizeof(pThis->guestInfo));
3265 SSMR3PutU32(pSSM, pThis->fu32AdditionsOk);
3266 SSMR3PutU32(pSSM, pThis->u32VideoAccelEnabled);
3267 SSMR3PutBool(pSSM, pThis->displayChangeData.fGuestSentChangeEventAck);
3268
3269 SSMR3PutU32(pSSM, pThis->guestCaps);
3270
3271#ifdef VBOX_WITH_HGCM
3272 vmmdevHGCMSaveState(pThis, pSSM);
3273#endif /* VBOX_WITH_HGCM */
3274
3275 SSMR3PutU32(pSSM, pThis->fHostCursorRequested);
3276
3277 SSMR3PutU32(pSSM, pThis->guestInfo2.uFullVersion);
3278 SSMR3PutU32(pSSM, pThis->guestInfo2.uRevision);
3279 SSMR3PutU32(pSSM, pThis->guestInfo2.fFeatures);
3280 SSMR3PutStrZ(pSSM, pThis->guestInfo2.szName);
3281 SSMR3PutU32(pSSM, pThis->cFacilityStatuses);
3282 for (uint32_t i = 0; i < pThis->cFacilityStatuses; i++)
3283 {
3284 SSMR3PutU32(pSSM, pThis->aFacilityStatuses[i].uFacility);
3285 SSMR3PutU32(pSSM, pThis->aFacilityStatuses[i].fFlags);
3286 SSMR3PutU16(pSSM, pThis->aFacilityStatuses[i].uStatus);
3287 SSMR3PutS64(pSSM, RTTimeSpecGetNano(&pThis->aFacilityStatuses[i].TimeSpecTS));
3288 }
3289
3290 PDMCritSectLeave(&pThis->CritSect);
3291 return VINF_SUCCESS;
3292}
3293
3294/**
3295 * @callback_method_impl{FNSSMDEVLOADEXEC}
3296 */
3297static DECLCALLBACK(int) vmmdevLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
3298{
3299 /** @todo The code load code is assuming we're always loaded into a freshly
3300 * constructed VM. */
3301 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3302 int rc;
3303
3304 if ( uVersion > VMMDEV_SAVED_STATE_VERSION
3305 || uVersion < 6)
3306 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
3307
3308 /* config */
3309 if (uVersion > VMMDEV_SAVED_STATE_VERSION_VBOX_30)
3310 {
3311 bool f;
3312 rc = SSMR3GetBool(pSSM, &f); AssertRCReturn(rc, rc);
3313 if (pThis->fGetHostTimeDisabled != f)
3314 LogRel(("VMMDev: Config mismatch - fGetHostTimeDisabled: config=%RTbool saved=%RTbool\n", pThis->fGetHostTimeDisabled, f));
3315
3316 rc = SSMR3GetBool(pSSM, &f); AssertRCReturn(rc, rc);
3317 if (pThis->fBackdoorLogDisabled != f)
3318 LogRel(("VMMDev: Config mismatch - fBackdoorLogDisabled: config=%RTbool saved=%RTbool\n", pThis->fBackdoorLogDisabled, f));
3319
3320 rc = SSMR3GetBool(pSSM, &f); AssertRCReturn(rc, rc);
3321 if (pThis->fKeepCredentials != f)
3322 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - fKeepCredentials: config=%RTbool saved=%RTbool"),
3323 pThis->fKeepCredentials, f);
3324 rc = SSMR3GetBool(pSSM, &f); AssertRCReturn(rc, rc);
3325 if (pThis->fHeapEnabled != f)
3326 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - fHeapEnabled: config=%RTbool saved=%RTbool"),
3327 pThis->fHeapEnabled, f);
3328 }
3329
3330 if (uPass != SSM_PASS_FINAL)
3331 return VINF_SUCCESS;
3332
3333 /* state */
3334 SSMR3GetU32(pSSM, &pThis->hypervisorSize);
3335 SSMR3GetU32(pSSM, &pThis->mouseCapabilities);
3336 SSMR3GetS32(pSSM, &pThis->mouseXAbs);
3337 SSMR3GetS32(pSSM, &pThis->mouseYAbs);
3338
3339 SSMR3GetBool(pSSM, &pThis->fNewGuestFilterMask);
3340 SSMR3GetU32(pSSM, &pThis->u32NewGuestFilterMask);
3341 SSMR3GetU32(pSSM, &pThis->u32GuestFilterMask);
3342 SSMR3GetU32(pSSM, &pThis->u32HostEventFlags);
3343
3344// SSMR3GetBool(pSSM, &pThis->pVMMDevRAMR3->fHaveEvents);
3345 // here be dragons (probably)
3346 SSMR3GetMem(pSSM, &pThis->pVMMDevRAMR3->V, sizeof (pThis->pVMMDevRAMR3->V));
3347
3348 SSMR3GetMem(pSSM, &pThis->guestInfo, sizeof (pThis->guestInfo));
3349 SSMR3GetU32(pSSM, &pThis->fu32AdditionsOk);
3350 SSMR3GetU32(pSSM, &pThis->u32VideoAccelEnabled);
3351 if (uVersion > 10)
3352 SSMR3GetBool(pSSM, &pThis->displayChangeData.fGuestSentChangeEventAck);
3353
3354 rc = SSMR3GetU32(pSSM, &pThis->guestCaps);
3355
3356 /* Attributes which were temporarily introduced in r30072 */
3357 if (uVersion == 7)
3358 {
3359 uint32_t temp;
3360 SSMR3GetU32(pSSM, &temp);
3361 rc = SSMR3GetU32(pSSM, &temp);
3362 }
3363 AssertRCReturn(rc, rc);
3364
3365#ifdef VBOX_WITH_HGCM
3366 rc = vmmdevHGCMLoadState(pThis, pSSM, uVersion);
3367 AssertRCReturn(rc, rc);
3368#endif /* VBOX_WITH_HGCM */
3369
3370 if (uVersion >= 10)
3371 rc = SSMR3GetU32(pSSM, &pThis->fHostCursorRequested);
3372 AssertRCReturn(rc, rc);
3373
3374 if (uVersion > VMMDEV_SAVED_STATE_VERSION_MISSING_GUEST_INFO_2)
3375 {
3376 SSMR3GetU32(pSSM, &pThis->guestInfo2.uFullVersion);
3377 SSMR3GetU32(pSSM, &pThis->guestInfo2.uRevision);
3378 SSMR3GetU32(pSSM, &pThis->guestInfo2.fFeatures);
3379 rc = SSMR3GetStrZ(pSSM, &pThis->guestInfo2.szName[0], sizeof(pThis->guestInfo2.szName));
3380 AssertRCReturn(rc, rc);
3381 }
3382
3383 if (uVersion > VMMDEV_SAVED_STATE_VERSION_MISSING_FACILITY_STATUSES)
3384 {
3385 uint32_t cFacilityStatuses;
3386 rc = SSMR3GetU32(pSSM, &cFacilityStatuses);
3387 AssertRCReturn(rc, rc);
3388
3389 for (uint32_t i = 0; i < cFacilityStatuses; i++)
3390 {
3391 uint32_t uFacility, fFlags;
3392 uint16_t uStatus;
3393 int64_t iTimeStampNano;
3394
3395 SSMR3GetU32(pSSM, &uFacility);
3396 SSMR3GetU32(pSSM, &fFlags);
3397 SSMR3GetU16(pSSM, &uStatus);
3398 rc = SSMR3GetS64(pSSM, &iTimeStampNano);
3399 AssertRCReturn(rc, rc);
3400
3401 PVMMDEVFACILITYSTATUSENTRY pEntry = vmmdevGetFacilityStatusEntry(pThis, uFacility);
3402 AssertLogRelMsgReturn(pEntry,
3403 ("VMMDev: Ran out of entries restoring the guest facility statuses. Saved state has %u.\n", cFacilityStatuses),
3404 VERR_OUT_OF_RESOURCES);
3405 pEntry->uStatus = uStatus;
3406 pEntry->fFlags = fFlags;
3407 RTTimeSpecSetNano(&pEntry->TimeSpecTS, iTimeStampNano);
3408 }
3409 }
3410
3411
3412 /*
3413 * On a resume, we send the capabilities changed message so
3414 * that listeners can sync their state again
3415 */
3416 Log(("vmmdevLoadState: capabilities changed (%x), informing connector\n", pThis->mouseCapabilities));
3417 if (pThis->pDrv)
3418 {
3419 pThis->pDrv->pfnUpdateMouseCapabilities(pThis->pDrv, pThis->mouseCapabilities);
3420 if (uVersion >= 10)
3421 pThis->pDrv->pfnUpdatePointerShape(pThis->pDrv,
3422 /*fVisible=*/!!pThis->fHostCursorRequested,
3423 /*fAlpha=*/false,
3424 /*xHot=*/0, /*yHot=*/0,
3425 /*cx=*/0, /*cy=*/0,
3426 /*pvShape=*/NULL);
3427 }
3428
3429 /* Reestablish the acceleration status. */
3430 if ( pThis->u32VideoAccelEnabled
3431 && pThis->pDrv)
3432 {
3433 pThis->pDrv->pfnVideoAccelEnable(pThis->pDrv, !!pThis->u32VideoAccelEnabled, &pThis->pVMMDevRAMR3->vbvaMemory);
3434 }
3435
3436 if (pThis->fu32AdditionsOk)
3437 {
3438 LogRel(("Guest Additions information report: additionsVersion = 0x%08X, osType = 0x%08X\n",
3439 pThis->guestInfo.interfaceVersion, pThis->guestInfo.osType));
3440 if (pThis->pDrv)
3441 {
3442 if (pThis->guestInfo2.uFullVersion && pThis->pDrv->pfnUpdateGuestInfo2)
3443 pThis->pDrv->pfnUpdateGuestInfo2(pThis->pDrv, pThis->guestInfo2.uFullVersion, pThis->guestInfo2.szName,
3444 pThis->guestInfo2.uRevision, pThis->guestInfo2.fFeatures);
3445 if (pThis->pDrv->pfnUpdateGuestInfo)
3446 pThis->pDrv->pfnUpdateGuestInfo(pThis->pDrv, &pThis->guestInfo);
3447
3448 if (pThis->pDrv->pfnUpdateGuestStatus)
3449 {
3450 for (uint32_t i = 0; i < pThis->cFacilityStatuses; i++) /* ascending order! */
3451 if ( pThis->aFacilityStatuses[i].uStatus != VBoxGuestFacilityStatus_Inactive
3452 || !pThis->aFacilityStatuses[i].fFixed)
3453 pThis->pDrv->pfnUpdateGuestStatus(pThis->pDrv,
3454 pThis->aFacilityStatuses[i].uFacility,
3455 pThis->aFacilityStatuses[i].uStatus,
3456 pThis->aFacilityStatuses[i].fFlags,
3457 &pThis->aFacilityStatuses[i].TimeSpecTS);
3458 }
3459 }
3460 }
3461 if (pThis->pDrv && pThis->pDrv->pfnUpdateGuestCapabilities)
3462 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pThis->guestCaps);
3463
3464 return VINF_SUCCESS;
3465}
3466
3467/**
3468 * Load state done callback. Notify guest of restore event.
3469 *
3470 * @returns VBox status code.
3471 * @param pDevIns The device instance.
3472 * @param pSSM The handle to the saved state.
3473 */
3474static DECLCALLBACK(int) vmmdevLoadStateDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3475{
3476 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3477
3478#ifdef VBOX_WITH_HGCM
3479 int rc = vmmdevHGCMLoadStateDone(pThis, pSSM);
3480 AssertLogRelRCReturn(rc, rc);
3481#endif /* VBOX_WITH_HGCM */
3482
3483 VMMDevNotifyGuest(pThis, VMMDEV_EVENT_RESTORED);
3484
3485 return VINF_SUCCESS;
3486}
3487
3488
3489/* -=-=-=-=- PDMDEVREG -=-=-=-=- */
3490
3491/**
3492 * (Re-)initializes the MMIO2 data.
3493 *
3494 * @param pThis Pointer to the VMMDev instance data.
3495 */
3496static void vmmdevInitRam(PVMMDEV pThis)
3497{
3498 memset(pThis->pVMMDevRAMR3, 0, sizeof(VMMDevMemory));
3499 pThis->pVMMDevRAMR3->u32Size = sizeof(VMMDevMemory);
3500 pThis->pVMMDevRAMR3->u32Version = VMMDEV_MEMORY_VERSION;
3501}
3502
3503
3504/**
3505 * @interface_method_impl{PDMDEVREG,pfnReset}
3506 */
3507static DECLCALLBACK(void) vmmdevReset(PPDMDEVINS pDevIns)
3508{
3509 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3510 PDMCritSectEnter(&pThis->CritSect, VERR_IGNORED);
3511
3512 /*
3513 * Reset the mouse integration feature bits
3514 */
3515 if (pThis->mouseCapabilities & VMMDEV_MOUSE_GUEST_MASK)
3516 {
3517 pThis->mouseCapabilities &= ~VMMDEV_MOUSE_GUEST_MASK;
3518 /* notify the connector */
3519 Log(("vmmdevReset: capabilities changed (%x), informing connector\n", pThis->mouseCapabilities));
3520 pThis->pDrv->pfnUpdateMouseCapabilities(pThis->pDrv, pThis->mouseCapabilities);
3521 }
3522 pThis->fHostCursorRequested = false;
3523
3524 pThis->hypervisorSize = 0;
3525
3526 /* re-initialize the VMMDev memory */
3527 if (pThis->pVMMDevRAMR3)
3528 vmmdevInitRam(pThis);
3529
3530 /* credentials have to go away (by default) */
3531 if (!pThis->fKeepCredentials)
3532 {
3533 memset(pThis->pCredentials->Logon.szUserName, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
3534 memset(pThis->pCredentials->Logon.szPassword, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
3535 memset(pThis->pCredentials->Logon.szDomain, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
3536 }
3537 memset(pThis->pCredentials->Judge.szUserName, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
3538 memset(pThis->pCredentials->Judge.szPassword, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
3539 memset(pThis->pCredentials->Judge.szDomain, '\0', VMMDEV_CREDENTIALS_SZ_SIZE);
3540
3541 /* Reset means that additions will report again. */
3542 const bool fVersionChanged = pThis->fu32AdditionsOk
3543 || pThis->guestInfo.interfaceVersion
3544 || pThis->guestInfo.osType != VBOXOSTYPE_Unknown;
3545 if (fVersionChanged)
3546 Log(("vmmdevReset: fu32AdditionsOk=%d additionsVersion=%x osType=%#x\n",
3547 pThis->fu32AdditionsOk, pThis->guestInfo.interfaceVersion, pThis->guestInfo.osType));
3548 pThis->fu32AdditionsOk = false;
3549 memset (&pThis->guestInfo, 0, sizeof (pThis->guestInfo));
3550 RT_ZERO(pThis->guestInfo2);
3551
3552 /* Clear facilities. No need to tell Main as it will get a
3553 pfnUpdateGuestInfo callback. */
3554 RTTIMESPEC TimeStampNow;
3555 RTTimeNow(&TimeStampNow);
3556 uint32_t iFacility = pThis->cFacilityStatuses;
3557 while (iFacility-- > 0)
3558 {
3559 pThis->aFacilityStatuses[iFacility].uStatus = VBoxGuestFacilityStatus_Inactive;
3560 pThis->aFacilityStatuses[iFacility].TimeSpecTS = TimeStampNow;
3561 }
3562
3563 /* clear pending display change request. */
3564 for (unsigned i = 0; i < RT_ELEMENTS(pThis->displayChangeData.aRequests); i++)
3565 {
3566 DISPLAYCHANGEREQUEST *pRequest = &pThis->displayChangeData.aRequests[i];
3567 memset (&pRequest->lastReadDisplayChangeRequest, 0, sizeof (pRequest->lastReadDisplayChangeRequest));
3568 }
3569 pThis->displayChangeData.iCurrentMonitor = 0;
3570 pThis->displayChangeData.fGuestSentChangeEventAck = false;
3571
3572 /* disable seamless mode */
3573 pThis->fLastSeamlessEnabled = false;
3574
3575 /* disabled memory ballooning */
3576 pThis->cMbMemoryBalloonLast = 0;
3577
3578 /* disabled statistics updating */
3579 pThis->u32LastStatIntervalSize = 0;
3580
3581 /* Clear the "HGCM event enabled" flag so the event can be automatically reenabled. */
3582 pThis->u32HGCMEnabled = 0;
3583
3584 /*
3585 * Clear the event variables.
3586 *
3587 * XXX By design we should NOT clear pThis->u32HostEventFlags because it is designed
3588 * that way so host events do not depend on guest resets. However, the pending
3589 * event flags actually _were_ cleared since ages so we mask out events from
3590 * clearing which we really need to survive the reset. See xtracker 5767.
3591 */
3592 pThis->u32HostEventFlags &= VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST;
3593 pThis->u32GuestFilterMask = 0;
3594 pThis->u32NewGuestFilterMask = 0;
3595 pThis->fNewGuestFilterMask = 0;
3596
3597 /* This is the default, as Windows and OS/2 guests take this for granted. (Actually, neither does...) */
3598 /** @todo change this when we next bump the interface version */
3599 const bool fCapsChanged = pThis->guestCaps != VMMDEV_GUEST_SUPPORTS_GRAPHICS;
3600 if (fCapsChanged)
3601 Log(("vmmdevReset: fCapsChanged=%#x -> %#x\n", pThis->guestCaps, VMMDEV_GUEST_SUPPORTS_GRAPHICS));
3602 pThis->guestCaps = VMMDEV_GUEST_SUPPORTS_GRAPHICS; /** @todo r=bird: why? I cannot see this being done at construction?*/
3603
3604 /*
3605 * Call the update functions as required.
3606 */
3607 if (fVersionChanged && pThis->pDrv && pThis->pDrv->pfnUpdateGuestInfo)
3608 pThis->pDrv->pfnUpdateGuestInfo(pThis->pDrv, &pThis->guestInfo);
3609 if (fCapsChanged && pThis->pDrv && pThis->pDrv->pfnUpdateGuestCapabilities)
3610 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pThis->guestCaps);
3611
3612 /* Generate a unique session id for this VM; it will be changed for each start, reset or restore.
3613 * This can be used for restore detection inside the guest.
3614 */
3615 pThis->idSession = ASMReadTSC();
3616
3617 PDMCritSectLeave(&pThis->CritSect);
3618}
3619
3620
3621/**
3622 * @interface_method_impl{PDMDEVREG,pfnRelocate}
3623 */
3624static DECLCALLBACK(void) vmmdevRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
3625{
3626 NOREF(pDevIns);
3627 NOREF(offDelta);
3628}
3629
3630
3631/**
3632 * @interface_method_impl{PDMDEVREG,pfnDestruct}
3633 */
3634static DECLCALLBACK(int) vmmdevDestruct(PPDMDEVINS pDevIns)
3635{
3636 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3637 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
3638
3639 /*
3640 * Wipe and free the credentials.
3641 */
3642 if (pThis->pCredentials)
3643 {
3644 RTMemWipeThoroughly(pThis->pCredentials, sizeof(*pThis->pCredentials), 10);
3645 RTMemFree(pThis->pCredentials);
3646 pThis->pCredentials = NULL;
3647 }
3648
3649#ifndef VBOX_WITHOUT_TESTING_FEATURES
3650 /*
3651 * Clean up the testing device.
3652 */
3653 vmmdevTestingTerminate(pDevIns);
3654#endif
3655
3656 return VINF_SUCCESS;
3657}
3658
3659
3660/**
3661 * @interface_method_impl{PDMDEVREG,pfnConstruct}
3662 */
3663static DECLCALLBACK(int) vmmdevConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
3664{
3665 PVMMDEV pThis = PDMINS_2_DATA(pDevIns, PVMMDEV);
3666 int rc;
3667
3668 Assert(iInstance == 0);
3669 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
3670
3671 /*
3672 * Initialize data (most of it anyway).
3673 */
3674 /* Save PDM device instance data for future reference. */
3675 pThis->pDevIns = pDevIns;
3676
3677 /* PCI vendor, just a free bogus value */
3678 PCIDevSetVendorId(&pThis->PciDev, 0x80ee);
3679 /* device ID */
3680 PCIDevSetDeviceId(&pThis->PciDev, 0xcafe);
3681 /* class sub code (other type of system peripheral) */
3682 PCIDevSetClassSub(&pThis->PciDev, 0x80);
3683 /* class base code (base system peripheral) */
3684 PCIDevSetClassBase(&pThis->PciDev, 0x08);
3685 /* header type */
3686 PCIDevSetHeaderType(&pThis->PciDev, 0x00);
3687 /* interrupt on pin 0 */
3688 PCIDevSetInterruptPin(&pThis->PciDev, 0x01);
3689
3690 RTTIMESPEC TimeStampNow;
3691 RTTimeNow(&TimeStampNow);
3692 vmmdevAllocFacilityStatusEntry(pThis, VBoxGuestFacilityType_VBoxGuestDriver, true /*fFixed*/, &TimeStampNow);
3693 vmmdevAllocFacilityStatusEntry(pThis, VBoxGuestFacilityType_VBoxService, true /*fFixed*/, &TimeStampNow);
3694 vmmdevAllocFacilityStatusEntry(pThis, VBoxGuestFacilityType_VBoxTrayClient, true /*fFixed*/, &TimeStampNow);
3695 vmmdevAllocFacilityStatusEntry(pThis, VBoxGuestFacilityType_Seamless, true /*fFixed*/, &TimeStampNow);
3696 vmmdevAllocFacilityStatusEntry(pThis, VBoxGuestFacilityType_Graphics, true /*fFixed*/, &TimeStampNow);
3697 Assert(pThis->cFacilityStatuses == 5);
3698
3699 /*
3700 * Interfaces
3701 */
3702 /* IBase */
3703 pThis->IBase.pfnQueryInterface = vmmdevPortQueryInterface;
3704
3705 /* VMMDev port */
3706 pThis->IPort.pfnQueryAbsoluteMouse = vmmdevIPort_QueryAbsoluteMouse;
3707 pThis->IPort.pfnSetAbsoluteMouse = vmmdevIPort_SetAbsoluteMouse ;
3708 pThis->IPort.pfnQueryMouseCapabilities = vmmdevIPort_QueryMouseCapabilities;
3709 pThis->IPort.pfnUpdateMouseCapabilities = vmmdevIPort_UpdateMouseCapabilities;
3710 pThis->IPort.pfnRequestDisplayChange = vmmdevIPort_RequestDisplayChange;
3711 pThis->IPort.pfnSetCredentials = vmmdevIPort_SetCredentials;
3712 pThis->IPort.pfnVBVAChange = vmmdevIPort_VBVAChange;
3713 pThis->IPort.pfnRequestSeamlessChange = vmmdevIPort_RequestSeamlessChange;
3714 pThis->IPort.pfnSetMemoryBalloon = vmmdevIPort_SetMemoryBalloon;
3715 pThis->IPort.pfnSetStatisticsInterval = vmmdevIPort_SetStatisticsInterval;
3716 pThis->IPort.pfnVRDPChange = vmmdevIPort_VRDPChange;
3717 pThis->IPort.pfnCpuHotUnplug = vmmdevIPort_CpuHotUnplug;
3718 pThis->IPort.pfnCpuHotPlug = vmmdevIPort_CpuHotPlug;
3719
3720 /* Shared folder LED */
3721 pThis->SharedFolders.Led.u32Magic = PDMLED_MAGIC;
3722 pThis->SharedFolders.ILeds.pfnQueryStatusLed = vmmdevQueryStatusLed;
3723
3724#ifdef VBOX_WITH_HGCM
3725 /* HGCM port */
3726 pThis->IHGCMPort.pfnCompleted = hgcmCompleted;
3727#endif
3728
3729 pThis->pCredentials = (VMMDEVCREDS *)RTMemAllocZ(sizeof(*pThis->pCredentials));
3730 if (!pThis->pCredentials)
3731 return VERR_NO_MEMORY;
3732
3733
3734 /*
3735 * Validate and read the configuration.
3736 */
3737 PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns,
3738 "GetHostTimeDisabled|"
3739 "BackdoorLogDisabled|"
3740 "KeepCredentials|"
3741 "HeapEnabled|"
3742 "RamSize|"
3743 "RZEnabled|"
3744 "GuestCoreDumpEnabled|"
3745 "GuestCoreDumpDir|"
3746 "GuestCoreDumpCount|"
3747 "TestingEnabled|"
3748 "TestingMMIO|"
3749 "TestintXmlOutputFile"
3750 ,
3751 "");
3752
3753 rc = CFGMR3QueryU64(pCfg, "RamSize", &pThis->cbGuestRAM);
3754 if (RT_FAILURE(rc))
3755 return PDMDEV_SET_ERROR(pDevIns, rc,
3756 N_("Configuration error: Failed querying \"RamSize\" as a 64-bit unsigned integer"));
3757
3758 rc = CFGMR3QueryBoolDef(pCfg, "GetHostTimeDisabled", &pThis->fGetHostTimeDisabled, false);
3759 if (RT_FAILURE(rc))
3760 return PDMDEV_SET_ERROR(pDevIns, rc,
3761 N_("Configuration error: Failed querying \"GetHostTimeDisabled\" as a boolean"));
3762
3763 rc = CFGMR3QueryBoolDef(pCfg, "BackdoorLogDisabled", &pThis->fBackdoorLogDisabled, false);
3764 if (RT_FAILURE(rc))
3765 return PDMDEV_SET_ERROR(pDevIns, rc,
3766 N_("Configuration error: Failed querying \"BackdoorLogDisabled\" as a boolean"));
3767
3768 rc = CFGMR3QueryBoolDef(pCfg, "KeepCredentials", &pThis->fKeepCredentials, false);
3769 if (RT_FAILURE(rc))
3770 return PDMDEV_SET_ERROR(pDevIns, rc,
3771 N_("Configuration error: Failed querying \"KeepCredentials\" as a boolean"));
3772
3773 rc = CFGMR3QueryBoolDef(pCfg, "HeapEnabled", &pThis->fHeapEnabled, true);
3774 if (RT_FAILURE(rc))
3775 return PDMDEV_SET_ERROR(pDevIns, rc,
3776 N_("Configuration error: Failed querying \"HeapEnabled\" as a boolean"));
3777
3778 rc = CFGMR3QueryBoolDef(pCfg, "RZEnabled", &pThis->fRZEnabled, true);
3779 if (RT_FAILURE(rc))
3780 return PDMDEV_SET_ERROR(pDevIns, rc,
3781 N_("Configuration error: Failed querying \"RZEnabled\" as a boolean"));
3782
3783 rc = CFGMR3QueryBoolDef(pCfg, "GuestCoreDumpEnabled", &pThis->fGuestCoreDumpEnabled, false);
3784 if (RT_FAILURE(rc))
3785 return PDMDEV_SET_ERROR(pDevIns, rc,
3786 N_("Configuration error: Failed querying \"GuestCoreDumpEnabled\" as a boolean"));
3787
3788 char *pszGuestCoreDumpDir = NULL;
3789 rc = CFGMR3QueryStringAllocDef(pCfg, "GuestCoreDumpDir", &pszGuestCoreDumpDir, "");
3790 if (RT_FAILURE(rc))
3791 return PDMDEV_SET_ERROR(pDevIns, rc,
3792 N_("Configuration error: Failed querying \"GuestCoreDumpDir\" as a string"));
3793
3794 RTStrCopy(pThis->szGuestCoreDumpDir, sizeof(pThis->szGuestCoreDumpDir), pszGuestCoreDumpDir);
3795 MMR3HeapFree(pszGuestCoreDumpDir);
3796
3797 rc = CFGMR3QueryU32Def(pCfg, "GuestCoreDumpCount", &pThis->cGuestCoreDumps, 3);
3798 if (RT_FAILURE(rc))
3799 return PDMDEV_SET_ERROR(pDevIns, rc,
3800 N_("Configuration error: Failed querying \"GuestCoreDumpCount\" as a 32-bit unsigned integer"));
3801
3802#ifndef VBOX_WITHOUT_TESTING_FEATURES
3803 rc = CFGMR3QueryBoolDef(pCfg, "TestingEnabled", &pThis->fTestingEnabled, false);
3804 if (RT_FAILURE(rc))
3805 return PDMDEV_SET_ERROR(pDevIns, rc,
3806 N_("Configuration error: Failed querying \"TestingEnabled\" as a boolean"));
3807 rc = CFGMR3QueryBoolDef(pCfg, "TestingMMIO", &pThis->fTestingMMIO, false);
3808 if (RT_FAILURE(rc))
3809 return PDMDEV_SET_ERROR(pDevIns, rc,
3810 N_("Configuration error: Failed querying \"TestingMMIO\" as a boolean"));
3811 rc = CFGMR3QueryStringAllocDef(pCfg, "TestintXmlOutputFile", &pThis->pszTestingXmlOutput, NULL);
3812 if (RT_FAILURE(rc))
3813 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Failed querying \"TestintXmlOutputFile\" as a string"));
3814
3815 /** @todo image-to-load-filename? */
3816#endif
3817
3818 /*
3819 * We do our own locking entirely. So, install NOP critsect for the device
3820 * and create our own critsect for use where it really matters (++).
3821 */
3822 rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
3823 AssertRCReturn(rc, rc);
3824 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->CritSect, RT_SRC_POS, "VMMDev#%u", iInstance);
3825 AssertRCReturn(rc, rc);
3826
3827 /*
3828 * Register the backdoor logging port
3829 */
3830 rc = PDMDevHlpIOPortRegister(pDevIns, RTLOG_DEBUG_PORT, 1, NULL, vmmdevBackdoorLog,
3831 NULL, NULL, NULL, "VMMDev backdoor logging");
3832 AssertRCReturn(rc, rc);
3833
3834#ifdef VMMDEV_WITH_ALT_TIMESYNC
3835 /*
3836 * Alternative timesync source.
3837 *
3838 * This was orignally added for creating a simple time sync service in an
3839 * OpenBSD guest without requiring VBoxGuest and VBoxService to be ported
3840 * first. We keep it in case it comes in handy.
3841 */
3842 rc = PDMDevHlpIOPortRegister(pDevIns, 0x505, 1, NULL,
3843 vmmdevAltTimeSyncWrite, vmmdevAltTimeSyncRead,
3844 NULL, NULL, "VMMDev timesync backdoor");
3845 AssertRCReturn(rc, rc);
3846#endif
3847
3848 /*
3849 * Allocate and initialize the MMIO2 memory.
3850 */
3851 rc = PDMDevHlpMMIO2Register(pDevIns, 1 /*iRegion*/, VMMDEV_RAM_SIZE, 0 /*fFlags*/, (void **)&pThis->pVMMDevRAMR3, "VMMDev");
3852 if (RT_FAILURE(rc))
3853 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
3854 N_("Failed to allocate %u bytes of memory for the VMM device"), VMMDEV_RAM_SIZE);
3855 vmmdevInitRam(pThis);
3856
3857 if (pThis->fHeapEnabled)
3858 {
3859 rc = PDMDevHlpMMIO2Register(pDevIns, 2 /*iRegion*/, VMMDEV_HEAP_SIZE, 0 /*fFlags*/, (void **)&pThis->pVMMDevHeapR3, "VMMDev Heap");
3860 if (RT_FAILURE(rc))
3861 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
3862 N_("Failed to allocate %u bytes of memory for the VMM device heap"), PAGE_SIZE);
3863 }
3864
3865 /*
3866 * Register the PCI device.
3867 */
3868 rc = PDMDevHlpPCIRegister(pDevIns, &pThis->PciDev);
3869 if (RT_FAILURE(rc))
3870 return rc;
3871 if (pThis->PciDev.devfn != 32 || iInstance != 0)
3872 Log(("!!WARNING!!: pThis->PciDev.devfn=%d (ignore if testcase or no started by Main)\n", pThis->PciDev.devfn));
3873 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, 0x20, PCI_ADDRESS_SPACE_IO, vmmdevIOPortRegionMap);
3874 if (RT_FAILURE(rc))
3875 return rc;
3876 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, VMMDEV_RAM_SIZE, PCI_ADDRESS_SPACE_MEM, vmmdevIORAMRegionMap);
3877 if (RT_FAILURE(rc))
3878 return rc;
3879 if (pThis->fHeapEnabled)
3880 {
3881 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 2, VMMDEV_HEAP_SIZE, PCI_ADDRESS_SPACE_MEM_PREFETCH, vmmdevIORAMRegionMap);
3882 if (RT_FAILURE(rc))
3883 return rc;
3884 }
3885
3886#ifndef VBOX_WITHOUT_TESTING_FEATURES
3887 /*
3888 * Initialize testing.
3889 */
3890 rc = vmmdevTestingInitialize(pDevIns);
3891 if (RT_FAILURE(rc))
3892 return rc;
3893#endif
3894
3895 /*
3896 * Get the corresponding connector interface
3897 */
3898 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThis->IBase, &pThis->pDrvBase, "VMM Driver Port");
3899 if (RT_SUCCESS(rc))
3900 {
3901 pThis->pDrv = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIVMMDEVCONNECTOR);
3902 AssertMsgReturn(pThis->pDrv, ("LUN #0 doesn't have a VMMDev connector interface!\n"), VERR_PDM_MISSING_INTERFACE);
3903#ifdef VBOX_WITH_HGCM
3904 pThis->pHGCMDrv = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIHGCMCONNECTOR);
3905 if (!pThis->pHGCMDrv)
3906 {
3907 Log(("LUN #0 doesn't have a HGCM connector interface, HGCM is not supported. rc=%Rrc\n", rc));
3908 /* this is not actually an error, just means that there is no support for HGCM */
3909 }
3910#endif
3911 /* Query the initial balloon size. */
3912 AssertPtr(pThis->pDrv->pfnQueryBalloonSize);
3913 rc = pThis->pDrv->pfnQueryBalloonSize(pThis->pDrv, &pThis->cMbMemoryBalloon);
3914 AssertRC(rc);
3915
3916 Log(("Initial balloon size %x\n", pThis->cMbMemoryBalloon));
3917 }
3918 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
3919 {
3920 Log(("%s/%d: warning: no driver attached to LUN #0!\n", pDevIns->pReg->szName, pDevIns->iInstance));
3921 rc = VINF_SUCCESS;
3922 }
3923 else
3924 AssertMsgFailedReturn(("Failed to attach LUN #0! rc=%Rrc\n", rc), rc);
3925
3926 /*
3927 * Attach status driver for shared folders (optional).
3928 */
3929 PPDMIBASE pBase;
3930 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
3931 if (RT_SUCCESS(rc))
3932 pThis->SharedFolders.pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
3933 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
3934 {
3935 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
3936 return rc;
3937 }
3938
3939 /*
3940 * Register saved state and init the HGCM CmdList critsect.
3941 */
3942 rc = PDMDevHlpSSMRegisterEx(pDevIns, VMMDEV_SAVED_STATE_VERSION, sizeof(*pThis), NULL,
3943 NULL, vmmdevLiveExec, NULL,
3944 NULL, vmmdevSaveExec, NULL,
3945 NULL, vmmdevLoadExec, vmmdevLoadStateDone);
3946 AssertRCReturn(rc, rc);
3947
3948#ifdef VBOX_WITH_HGCM
3949 pThis->pHGCMCmdList = NULL;
3950 rc = RTCritSectInit(&pThis->critsectHGCMCmdList);
3951 AssertRCReturn(rc, rc);
3952 pThis->u32HGCMEnabled = 0;
3953#endif /* VBOX_WITH_HGCM */
3954
3955 /*
3956 * In this version of VirtualBox the GUI checks whether "needs host cursor"
3957 * changes.
3958 */
3959 pThis->mouseCapabilities |= VMMDEV_MOUSE_HOST_RECHECKS_NEEDS_HOST_CURSOR;
3960
3961 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatMemBalloonChunks, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Memory balloon size", "/Devices/VMMDev/BalloonChunks");
3962
3963 /*
3964 * Generate a unique session id for this VM; it will be changed for each
3965 * start, reset or restore. This can be used for restore detection inside
3966 * the guest.
3967 */
3968 pThis->idSession = ASMReadTSC();
3969 return rc;
3970}
3971
3972/**
3973 * The device registration structure.
3974 */
3975extern "C" const PDMDEVREG g_DeviceVMMDev =
3976{
3977 /* u32Version */
3978 PDM_DEVREG_VERSION,
3979 /* szName */
3980 "VMMDev",
3981 /* szRCMod */
3982 "VBoxDDGC.gc",
3983 /* szR0Mod */
3984 "VBoxDDR0.r0",
3985 /* pszDescription */
3986 "VirtualBox VMM Device\n",
3987 /* fFlags */
3988 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
3989 /* fClass */
3990 PDM_DEVREG_CLASS_VMM_DEV,
3991 /* cMaxInstances */
3992 1,
3993 /* cbInstance */
3994 sizeof(VMMDevState),
3995 /* pfnConstruct */
3996 vmmdevConstruct,
3997 /* pfnDestruct */
3998 vmmdevDestruct,
3999 /* pfnRelocate */
4000 vmmdevRelocate,
4001 /* pfnMemSetup */
4002 NULL,
4003 /* pfnPowerOn */
4004 NULL,
4005 /* pfnReset */
4006 vmmdevReset,
4007 /* pfnSuspend */
4008 NULL,
4009 /* pfnResume */
4010 NULL,
4011 /* pfnAttach */
4012 NULL,
4013 /* pfnDetach */
4014 NULL,
4015 /* pfnQueryInterface. */
4016 NULL,
4017 /* pfnInitComplete */
4018 NULL,
4019 /* pfnPowerOff */
4020 NULL,
4021 /* pfnSoftReset */
4022 NULL,
4023 /* u32VersionEnd */
4024 PDM_DEVREG_VERSION
4025};
4026#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
4027
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