VirtualBox

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

Last change on this file since 67075 was 66220, checked in by vboxsync, 8 years ago

Windows 2016 OS type

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