VirtualBox

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

Last change on this file since 55705 was 55275, checked in by vboxsync, 10 years ago

heartbeat service: document it, sanity check and fixed logging

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