VirtualBox

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

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

fixed typo, it means occurred, not occured

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 96.9 KB
Line 
1/* $Id: VBoxDev.cpp 20961 2009-06-26 08:45:18Z vboxsync $ */
2/** @file
3 * VMMDev - Guest <-> VMM/Host communication device.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26/* #define LOG_ENABLED */
27/* Enable dev_vmm Log3 statements to get IRQ-related logging. */
28
29#define LOG_GROUP LOG_GROUP_DEV_VMM
30#include <VBox/VBoxDev.h>
31#include <VBox/VBoxGuest.h>
32#include <VBox/log.h>
33#include <VBox/param.h>
34#include <VBox/mm.h>
35#include <VBox/pgm.h>
36#include <VBox/err.h>
37#include <VBox/vm.h> /* for VM_IS_EMT */
38
39#include <iprt/assert.h>
40#include <iprt/string.h>
41#include <iprt/time.h>
42#ifndef IN_RC
43# include <iprt/mem.h>
44#endif
45
46#include "VMMDevState.h"
47#ifdef VBOX_WITH_HGCM
48# include "VMMDevHGCM.h"
49#endif
50
51
52/*******************************************************************************
53* Defined Constants And Macros *
54*******************************************************************************/
55#define PCIDEV_2_VMMDEVSTATE(pPciDev) ( (VMMDevState *)(pPciDev) )
56#define VMMDEVSTATE_2_DEVINS(pVMMDevState) ( (pVMMDevState)->pDevIns )
57
58#define VBOX_GUEST_ADDITIONS_VERSION_1_03(s) \
59 ((RT_HIWORD ((s)->guestInfo.additionsVersion) == 1) && \
60 (RT_LOWORD ((s)->guestInfo.additionsVersion) == 3))
61
62#define VBOX_GUEST_ADDITIONS_VERSION_OK(additionsVersion) \
63 (RT_HIWORD(additionsVersion) == RT_HIWORD(VMMDEV_VERSION) \
64 && RT_LOWORD(additionsVersion) <= RT_LOWORD(VMMDEV_VERSION))
65
66#define VBOX_GUEST_ADDITIONS_VERSION_OLD(additionsVersion) \
67 ((RT_HIWORD(additionsVersion) < RT_HIWORD(VMMDEV_VERSION) \
68 || ((RT_HIWORD(additionsVersion) == RT_HIWORD(VMMDEV_VERSION) \
69 && RT_LOWORD(additionsVersion) <= RT_LOWORD(VMMDEV_VERSION))
70
71#define VBOX_GUEST_ADDITIONS_VERSION_TOO_OLD(additionsVersion) \
72 (RT_HIWORD(additionsVersion) < RT_HIWORD(VMMDEV_VERSION))
73
74#define VBOX_GUEST_ADDITIONS_VERSION_NEW(additionsVersion) \
75 ((RT_HIWORD(additionsVersion) > RT_HIWORD(VMMDEV_VERSION) \
76 || ((RT_HIWORD(additionsVersion) == RT_HIWORD(VMMDEV_VERSION) \
77 && RT_LOWORD(additionsVersion) > RT_LOWORD(VMMDEV_VERSION))
78
79#ifndef VBOX_DEVICE_STRUCT_TESTCASE
80
81/* Whenever host wants to inform guest about something
82 * an IRQ notification will be raised.
83 *
84 * VMMDev PDM interface will contain the guest notification method.
85 *
86 * There is a 32 bit event mask which will be read
87 * by guest on an interrupt. A non zero bit in the mask
88 * means that the specific event occurred and requires
89 * processing on guest side.
90 *
91 * After reading the event mask guest must issue a
92 * generic request AcknowlegdeEvents.
93 *
94 * IRQ line is set to 1 (request) if there are unprocessed
95 * events, that is the event mask is not zero.
96 *
97 * After receiving an interrupt and checking event mask,
98 * the guest must process events using the event specific
99 * mechanism.
100 *
101 * That is if mouse capabilities were changed,
102 * guest will use VMMDev_GetMouseStatus generic request.
103 *
104 * Event mask is only a set of flags indicating that guest
105 * must proceed with a procedure.
106 *
107 * Unsupported events are therefore ignored.
108 * The guest additions must inform host which events they
109 * want to receive, to avoid unnecessary IRQ processing.
110 * By default no events are signalled to guest.
111 *
112 * This seems to be fast method. It requires
113 * only one context switch for an event processing.
114 *
115 */
116
117static void vmmdevSetIRQ_Legacy_EMT (VMMDevState *pVMMDevState)
118{
119 if (!pVMMDevState->fu32AdditionsOk)
120 {
121 Log(("vmmdevSetIRQ: IRQ is not generated, guest has not yet reported to us.\n"));
122 return;
123 }
124
125 uint32_t u32IRQLevel = 0;
126
127 /* Filter unsupported events */
128 uint32_t u32EventFlags =
129 pVMMDevState->u32HostEventFlags
130 & pVMMDevState->pVMMDevRAMR3->V.V1_03.u32GuestEventMask;
131
132 Log(("vmmdevSetIRQ: u32EventFlags = 0x%08X, "
133 "pVMMDevState->u32HostEventFlags = 0x%08X, "
134 "pVMMDevState->pVMMDevRAMR3->u32GuestEventMask = 0x%08X\n",
135 u32EventFlags,
136 pVMMDevState->u32HostEventFlags,
137 pVMMDevState->pVMMDevRAMR3->V.V1_03.u32GuestEventMask));
138
139 /* Move event flags to VMMDev RAM */
140 pVMMDevState->pVMMDevRAMR3->V.V1_03.u32HostEvents = u32EventFlags;
141
142 if (u32EventFlags)
143 {
144 /* Clear host flags which will be delivered to guest. */
145 pVMMDevState->u32HostEventFlags &= ~u32EventFlags;
146 Log(("vmmdevSetIRQ: pVMMDevState->u32HostEventFlags = 0x%08X\n",
147 pVMMDevState->u32HostEventFlags));
148 u32IRQLevel = 1;
149 }
150
151 /* Set IRQ level for pin 0 */
152 /** @todo make IRQ pin configurable, at least a symbolic constant */
153 PPDMDEVINS pDevIns = VMMDEVSTATE_2_DEVINS(pVMMDevState);
154 PDMDevHlpPCISetIrqNoWait(pDevIns, 0, u32IRQLevel);
155 Log(("vmmdevSetIRQ: IRQ set %d\n", u32IRQLevel));
156}
157
158static void vmmdevMaybeSetIRQ_EMT (VMMDevState *pVMMDevState)
159{
160 PPDMDEVINS pDevIns = VMMDEVSTATE_2_DEVINS (pVMMDevState);
161
162 Log3(("vmmdevMaybeSetIRQ_EMT: u32HostEventFlags = 0x%08X, u32GuestFilterMask = 0x%08X.\n",
163 pVMMDevState->u32HostEventFlags, pVMMDevState->u32GuestFilterMask));
164
165 if (pVMMDevState->u32HostEventFlags & pVMMDevState->u32GuestFilterMask)
166 {
167 pVMMDevState->pVMMDevRAMR3->V.V1_04.fHaveEvents = true;
168 PDMDevHlpPCISetIrqNoWait (pDevIns, 0, 1);
169 Log3(("vmmdevMaybeSetIRQ_EMT: IRQ set.\n"));
170 }
171}
172
173static void vmmdevNotifyGuest_EMT (VMMDevState *pVMMDevState, uint32_t u32EventMask)
174{
175 Log3(("VMMDevNotifyGuest_EMT: u32EventMask = 0x%08X.\n", u32EventMask));
176
177 if (VBOX_GUEST_ADDITIONS_VERSION_1_03 (pVMMDevState))
178 {
179 Log3(("VMMDevNotifyGuest_EMT: Old additions detected.\n"));
180
181 pVMMDevState->u32HostEventFlags |= u32EventMask;
182 vmmdevSetIRQ_Legacy_EMT (pVMMDevState);
183 }
184 else
185 {
186 Log3(("VMMDevNotifyGuest_EMT: New additions detected.\n"));
187
188 if (!pVMMDevState->fu32AdditionsOk)
189 {
190 pVMMDevState->u32HostEventFlags |= u32EventMask;
191 Log(("vmmdevNotifyGuest_EMT: IRQ is not generated, guest has not yet reported to us.\n"));
192 return;
193 }
194
195 const bool fHadEvents =
196 (pVMMDevState->u32HostEventFlags & pVMMDevState->u32GuestFilterMask) != 0;
197
198 Log3(("VMMDevNotifyGuest_EMT: fHadEvents = %d, u32HostEventFlags = 0x%08X, u32GuestFilterMask = 0x%08X.\n",
199 fHadEvents, pVMMDevState->u32HostEventFlags, pVMMDevState->u32GuestFilterMask));
200
201 pVMMDevState->u32HostEventFlags |= u32EventMask;
202
203 if (!fHadEvents)
204 vmmdevMaybeSetIRQ_EMT (pVMMDevState);
205 }
206}
207
208static void vmmdevCtlGuestFilterMask_EMT (VMMDevState *pVMMDevState,
209 uint32_t u32OrMask,
210 uint32_t u32NotMask)
211{
212 const bool fHadEvents =
213 (pVMMDevState->u32HostEventFlags & pVMMDevState->u32GuestFilterMask) != 0;
214
215 Log(("vmmdevCtlGuestFilterMask_EMT: u32OrMask = 0x%08X, u32NotMask = 0x%08X, fHadEvents = %d.\n", u32OrMask, u32NotMask, fHadEvents));
216 if (fHadEvents)
217 {
218 if (!pVMMDevState->fNewGuestFilterMask)
219 pVMMDevState->u32NewGuestFilterMask = pVMMDevState->u32GuestFilterMask;
220
221 pVMMDevState->u32NewGuestFilterMask |= u32OrMask;
222 pVMMDevState->u32NewGuestFilterMask &= ~u32NotMask;
223 pVMMDevState->fNewGuestFilterMask = true;
224 }
225 else
226 {
227 pVMMDevState->u32GuestFilterMask |= u32OrMask;
228 pVMMDevState->u32GuestFilterMask &= ~u32NotMask;
229 vmmdevMaybeSetIRQ_EMT (pVMMDevState);
230 }
231}
232
233void VMMDevCtlSetGuestFilterMask (VMMDevState *pVMMDevState,
234 uint32_t u32OrMask,
235 uint32_t u32NotMask)
236{
237 PPDMDEVINS pDevIns = VMMDEVSTATE_2_DEVINS(pVMMDevState);
238 PVM pVM = PDMDevHlpGetVM(pDevIns);
239
240 Log(("VMMDevCtlSetGuestFilterMask: u32OrMask = 0x%08X, u32NotMask = 0x%08X.\n", u32OrMask, u32NotMask));
241
242 if (VM_IS_EMT(pVM))
243 {
244 vmmdevCtlGuestFilterMask_EMT (pVMMDevState, u32OrMask, u32NotMask);
245 }
246 else
247 {
248 int rc;
249 PVMREQ pReq;
250
251 rc = VMR3ReqCallVoid (pVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT,
252 (PFNRT) vmmdevCtlGuestFilterMask_EMT,
253 3, pVMMDevState, u32OrMask, u32NotMask);
254 AssertReleaseRC (rc);
255 VMR3ReqFree (pReq);
256 }
257}
258
259void VMMDevNotifyGuest (VMMDevState *pVMMDevState, uint32_t u32EventMask)
260{
261 PPDMDEVINS pDevIns = VMMDEVSTATE_2_DEVINS(pVMMDevState);
262 PVM pVM = PDMDevHlpGetVM(pDevIns);
263 int rc;
264
265 Log3(("VMMDevNotifyGuest: u32EventMask = 0x%08X.\n", u32EventMask));
266
267 /*
268 * Drop notifications if the VM is not running yet/anymore.
269 */
270 if (PDMDevHlpVMState(pDevIns) != VMSTATE_RUNNING)
271 return;
272
273 /* No need to wait for the completion of this request. It is a notification
274 * about something, which has already happened.
275 */
276 rc = VMR3ReqCallEx(pVM, VMCPUID_ANY, NULL, 0, VMREQFLAGS_NO_WAIT | VMREQFLAGS_VOID,
277 (PFNRT) vmmdevNotifyGuest_EMT,
278 2, pVMMDevState, u32EventMask);
279 AssertRC(rc);
280}
281
282/**
283 * Port I/O Handler for OUT operations.
284 *
285 * @returns VBox status code.
286 *
287 * @param pDevIns The device instance.
288 * @param pvUser User argument - ignored.
289 * @param uPort Port number used for the IN operation.
290 * @param u32 The value to output.
291 * @param cb The value size in bytes.
292 */
293static DECLCALLBACK(int) vmmdevBackdoorLog(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
294{
295 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
296
297 if (!pThis->fBackdoorLogDisabled && cb == 1 && Port == RTLOG_DEBUG_PORT)
298 {
299
300 /* The raw version. */
301 switch (u32)
302 {
303 case '\r': LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: <return>\n")); break;
304 case '\n': LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: <newline>\n")); break;
305 case '\t': LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: <tab>\n")); break;
306 default: LogIt(LOG_INSTANCE, RTLOGGRPFLAGS_LEVEL_2, LOG_GROUP_DEV_VMM_BACKDOOR, ("vmmdev: %c (%02x)\n", u32, u32)); break;
307 }
308
309 /* The readable, buffered version. */
310 if (u32 == '\n' || u32 == '\r')
311 {
312 pThis->szMsg[pThis->iMsg] = '\0';
313 if (pThis->iMsg)
314 LogRelIt(LOG_REL_INSTANCE, RTLOGGRPFLAGS_LEVEL_1, LOG_GROUP_DEV_VMM_BACKDOOR, ("Guest Log: %s\n", pThis->szMsg));
315 pThis->iMsg = 0;
316 }
317 else
318 {
319 if (pThis->iMsg >= sizeof(pThis->szMsg)-1)
320 {
321 pThis->szMsg[pThis->iMsg] = '\0';
322 LogRelIt(LOG_REL_INSTANCE, RTLOGGRPFLAGS_LEVEL_1, LOG_GROUP_DEV_VMM_BACKDOOR, ("Guest Log: %s\n", pThis->szMsg));
323 pThis->iMsg = 0;
324 }
325 pThis->szMsg[pThis->iMsg] = (char )u32;
326 pThis->szMsg[++pThis->iMsg] = '\0';
327 }
328 }
329 return VINF_SUCCESS;
330}
331
332#ifdef TIMESYNC_BACKDOOR
333/**
334 * Port I/O Handler for OUT operations.
335 *
336 * @returns VBox status code.
337 *
338 * @param pDevIns The device instance.
339 * @param pvUser User argument - ignored.
340 * @param uPort Port number used for the IN operation.
341 * @param u32 The value to output.
342 * @param cb The value size in bytes.
343 */
344static DECLCALLBACK(int) vmmdevTimesyncBackdoorWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
345{
346 NOREF(pvUser);
347 if (cb == 4)
348 {
349 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
350 switch (u32)
351 {
352 case 0:
353 pThis->fTimesyncBackdoorLo = false;
354 break;
355 case 1:
356 pThis->fTimesyncBackdoorLo = true;
357 }
358 return VINF_SUCCESS;
359
360 }
361 return VINF_SUCCESS;
362}
363
364/**
365 * Port I/O Handler for backdoor timesync IN operations.
366 *
367 * @returns VBox status code.
368 *
369 * @param pDevIns The device instance.
370 * @param pvUser User argument - ignored.
371 * @param uPort Port number used for the IN operation.
372 * @param pu32 Where to store the result.
373 * @param cb Number of bytes read.
374 */
375static DECLCALLBACK(int) vmmdevTimesyncBackdoorRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
376{
377 int rc;
378 NOREF(pvUser);
379 if (cb == 4)
380 {
381 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
382 RTTIMESPEC now;
383
384 if (pThis->fTimesyncBackdoorLo)
385 *pu32 = (uint32_t)pThis->hostTime;
386 else
387 {
388 pThis->hostTime = RTTimeSpecGetMilli(PDMDevHlpUTCNow(pDevIns, &now));
389 *pu32 = (uint32_t)(pThis->hostTime >> 32);
390 }
391 rc = VINF_SUCCESS;
392 }
393 else
394 rc = VERR_IOM_IOPORT_UNUSED;
395 return rc;
396}
397#endif /* TIMESYNC_BACKDOOR */
398
399/**
400 * Port I/O Handler for the generic request interface
401 * @see FNIOMIOPORTOUT for details.
402 */
403static DECLCALLBACK(int) vmmdevRequestHandler(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
404{
405 VMMDevState *pThis = (VMMDevState*)pvUser;
406 int rcRet = VINF_SUCCESS;
407
408 /*
409 * The caller has passed the guest context physical address
410 * of the request structure. Copy the request packet.
411 */
412 VMMDevRequestHeader requestHeader = {0};
413 VMMDevRequestHeader *pRequestHeader = NULL;
414
415 PDMDevHlpPhysRead(pDevIns, (RTGCPHYS)u32, &requestHeader, sizeof(requestHeader));
416
417 /* the structure size must be greater or equal to the header size */
418 if (requestHeader.size < sizeof(VMMDevRequestHeader))
419 {
420 Log(("VMMDev request header size too small! size = %d\n", requestHeader.size));
421 rcRet = VINF_SUCCESS;
422 goto end;
423 }
424
425 /* check the version of the header structure */
426 if (requestHeader.version != VMMDEV_REQUEST_HEADER_VERSION)
427 {
428 Log(("VMMDev: guest header version (0x%08X) differs from ours (0x%08X)\n", requestHeader.version, VMMDEV_REQUEST_HEADER_VERSION));
429 rcRet = VINF_SUCCESS;
430 goto end;
431 }
432
433 Log2(("VMMDev request issued: %d\n", requestHeader.requestType));
434
435 if ( requestHeader.requestType != VMMDevReq_ReportGuestInfo
436 && !pThis->fu32AdditionsOk)
437 {
438 Log(("VMMDev: guest has not yet reported to us. Refusing operation.\n"));
439 requestHeader.rc = VERR_NOT_SUPPORTED;
440 rcRet = VINF_SUCCESS;
441 goto end;
442 }
443
444 /* Check upper limit */
445 if (requestHeader.size > VMMDEV_MAX_VMMDEVREQ_SIZE)
446 {
447 LogRel(("VMMDev: request packet too big (%x). Refusing operation.\n", requestHeader.size));
448 requestHeader.rc = VERR_NOT_SUPPORTED;
449 rcRet = VINF_SUCCESS;
450 goto end;
451 }
452
453 /* Read the entire request packet */
454 pRequestHeader = (VMMDevRequestHeader *)RTMemAlloc(requestHeader.size);
455 if (!pRequestHeader)
456 {
457 Log(("VMMDev: RTMemAlloc failed!\n"));
458 rcRet = VINF_SUCCESS;
459 requestHeader.rc = VERR_NO_MEMORY;
460 goto end;
461 }
462 PDMDevHlpPhysRead(pDevIns, (RTGCPHYS)u32, pRequestHeader, requestHeader.size);
463
464 /* which request was sent? */
465 switch (pRequestHeader->requestType)
466 {
467 /*
468 * Guest wants to give up a timeslice
469 */
470 case VMMDevReq_Idle:
471 {
472 /* just return to EMT telling it that we want to halt */
473 rcRet = VINF_EM_HALT;
474 break;
475 }
476
477 /*
478 * Guest is reporting its information
479 */
480 case VMMDevReq_ReportGuestInfo:
481 {
482 if (pRequestHeader->size < sizeof(VMMDevReportGuestInfo))
483 {
484 AssertMsgFailed(("VMMDev guest information structure has invalid size!\n"));
485 pRequestHeader->rc = VERR_INVALID_PARAMETER;
486 }
487 else
488 {
489 VMMDevReportGuestInfo *guestInfo = (VMMDevReportGuestInfo*)pRequestHeader;
490
491 if (memcmp (&pThis->guestInfo, &guestInfo->guestInfo, sizeof (guestInfo->guestInfo)) != 0)
492 {
493 /* make a copy of supplied information */
494 pThis->guestInfo = guestInfo->guestInfo;
495
496 /* Check additions version */
497 pThis->fu32AdditionsOk = VBOX_GUEST_ADDITIONS_VERSION_OK(pThis->guestInfo.additionsVersion);
498
499 LogRel(("Guest Additions information report: additionsVersion = 0x%08X osType = 0x%08X\n",
500 pThis->guestInfo.additionsVersion,
501 pThis->guestInfo.osType));
502 pThis->pDrv->pfnUpdateGuestVersion(pThis->pDrv, &pThis->guestInfo);
503 }
504
505 if (pThis->fu32AdditionsOk)
506 {
507 pRequestHeader->rc = VINF_SUCCESS;
508 }
509 else
510 {
511 pRequestHeader->rc = VERR_VERSION_MISMATCH;
512 }
513 }
514 break;
515 }
516
517 /* Report guest capabilities */
518 case VMMDevReq_ReportGuestCapabilities:
519 {
520 if (pRequestHeader->size != sizeof(VMMDevReqGuestCapabilities))
521 {
522 AssertMsgFailed(("VMMDev guest caps structure has invalid size!\n"));
523 pRequestHeader->rc = VERR_INVALID_PARAMETER;
524 }
525 else
526 {
527 VMMDevReqGuestCapabilities *guestCaps = (VMMDevReqGuestCapabilities*)pRequestHeader;
528
529 /* Enable this automatically for guests using the old
530 request to report their capabilities. */
531 /** @todo change this when we next bump the interface version */
532 guestCaps->caps |= VMMDEV_GUEST_SUPPORTS_GRAPHICS;
533 if (pThis->guestCaps != guestCaps->caps)
534 {
535 /* make a copy of supplied information */
536 pThis->guestCaps = guestCaps->caps;
537
538 LogRel(("Guest Additions capability report: (0x%x) "
539 "seamless: %s, "
540 "hostWindowMapping: %s, "
541 "graphics: %s\n",
542 guestCaps->caps,
543 guestCaps->caps & VMMDEV_GUEST_SUPPORTS_SEAMLESS ? "yes" : "no",
544 guestCaps->caps & VMMDEV_GUEST_SUPPORTS_GUEST_HOST_WINDOW_MAPPING ? "yes" : "no",
545 guestCaps->caps & VMMDEV_GUEST_SUPPORTS_GRAPHICS ? "yes" : "no"));
546
547 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, guestCaps->caps);
548 }
549 pRequestHeader->rc = VINF_SUCCESS;
550 }
551 break;
552 }
553
554 /* Change guest capabilities */
555 case VMMDevReq_SetGuestCapabilities:
556 {
557 if (pRequestHeader->size != sizeof(VMMDevReqGuestCapabilities2))
558 {
559 AssertMsgFailed(("VMMDev guest caps structure has invalid size!\n"));
560 pRequestHeader->rc = VERR_INVALID_PARAMETER;
561 }
562 else
563 {
564 VMMDevReqGuestCapabilities2 *guestCaps = (VMMDevReqGuestCapabilities2*)pRequestHeader;
565
566 pThis->guestCaps |= guestCaps->u32OrMask;
567 pThis->guestCaps &= ~guestCaps->u32NotMask;
568
569 LogRel(("Guest Additions capability report: (0x%x) "
570 "seamless: %s, "
571 "hostWindowMapping: %s, "
572 "graphics: %s\n",
573 pThis->guestCaps,
574 pThis->guestCaps & VMMDEV_GUEST_SUPPORTS_SEAMLESS ? "yes" : "no",
575 pThis->guestCaps & VMMDEV_GUEST_SUPPORTS_GUEST_HOST_WINDOW_MAPPING ? "yes" : "no",
576 pThis->guestCaps & VMMDEV_GUEST_SUPPORTS_GRAPHICS ? "yes" : "no"));
577
578 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pThis->guestCaps);
579 pRequestHeader->rc = VINF_SUCCESS;
580 }
581 break;
582 }
583
584 /*
585 * Retrieve mouse information
586 */
587 case VMMDevReq_GetMouseStatus:
588 {
589 if (pRequestHeader->size != sizeof(VMMDevReqMouseStatus))
590 {
591 AssertMsgFailed(("VMMDev mouse status structure has invalid size!\n"));
592 pRequestHeader->rc = VERR_INVALID_PARAMETER;
593 }
594 else
595 {
596 VMMDevReqMouseStatus *mouseStatus = (VMMDevReqMouseStatus*)pRequestHeader;
597 mouseStatus->mouseFeatures = 0;
598 if (pThis->mouseCapabilities & VMMDEV_MOUSEHOSTWANTSABS)
599 {
600 mouseStatus->mouseFeatures |= VBOXGUEST_MOUSE_HOST_CAN_ABSOLUTE;
601 }
602 if (pThis->mouseCapabilities & VMMDEV_MOUSEGUESTWANTSABS)
603 {
604 mouseStatus->mouseFeatures |= VBOXGUEST_MOUSE_GUEST_CAN_ABSOLUTE;
605 }
606 if (pThis->mouseCapabilities & VMMDEV_MOUSEHOSTCANNOTHWPOINTER)
607 {
608 mouseStatus->mouseFeatures |= VBOXGUEST_MOUSE_HOST_CANNOT_HWPOINTER;
609 }
610 mouseStatus->pointerXPos = pThis->mouseXAbs;
611 mouseStatus->pointerYPos = pThis->mouseYAbs;
612 Log2(("returning mouse status: features = %d, absX = %d, absY = %d\n", mouseStatus->mouseFeatures,
613 mouseStatus->pointerXPos, mouseStatus->pointerYPos));
614 pRequestHeader->rc = VINF_SUCCESS;
615 }
616 break;
617 }
618
619 /*
620 * Set mouse information
621 */
622 case VMMDevReq_SetMouseStatus:
623 {
624 if (pRequestHeader->size != sizeof(VMMDevReqMouseStatus))
625 {
626 AssertMsgFailed(("VMMDev mouse status structure has invalid size %d (%#x) version=%d!\n",
627 pRequestHeader->size, pRequestHeader->size, pRequestHeader->size, pRequestHeader->version));
628 pRequestHeader->rc = VERR_INVALID_PARAMETER;
629 }
630 else
631 {
632 bool bCapsChanged = false;
633
634 VMMDevReqMouseStatus *mouseStatus = (VMMDevReqMouseStatus*)pRequestHeader;
635
636 /* check if the guest wants absolute coordinates */
637 if (mouseStatus->mouseFeatures & VBOXGUEST_MOUSE_GUEST_CAN_ABSOLUTE)
638 {
639 /* set the capability flag and the changed flag if it's actually a change */
640 if (!(pThis->mouseCapabilities & VMMDEV_MOUSEGUESTWANTSABS))
641 {
642 pThis->mouseCapabilities |= VMMDEV_MOUSEGUESTWANTSABS;
643 bCapsChanged = true;
644 LogRel(("Guest requests mouse pointer integration\n"));
645 }
646 } else
647 {
648 if (pThis->mouseCapabilities & VMMDEV_MOUSEGUESTWANTSABS)
649 {
650 pThis->mouseCapabilities &= ~VMMDEV_MOUSEGUESTWANTSABS;
651 bCapsChanged = true;
652 LogRel(("Guest disables mouse pointer integration\n"));
653 }
654 }
655 if (mouseStatus->mouseFeatures & VBOXGUEST_MOUSE_GUEST_NEEDS_HOST_CURSOR)
656 pThis->mouseCapabilities |= VMMDEV_MOUSEGUESTNEEDSHOSTCUR;
657 else
658 pThis->mouseCapabilities &= ~VMMDEV_MOUSEGUESTNEEDSHOSTCUR;
659 if (mouseStatus->mouseFeatures & VBOXGUEST_MOUSE_GUEST_USES_VMMDEV)
660 pThis->mouseCapabilities |= VMMDEV_MOUSEGUESTUSESVMMDEV;
661 else
662 pThis->mouseCapabilities &= ~VMMDEV_MOUSEGUESTUSESVMMDEV;
663
664 /*
665 * Notify connector if something has changed
666 */
667 if (bCapsChanged)
668 {
669 Log(("VMMDevReq_SetMouseStatus: capabilities changed (%x), informing connector\n", pThis->mouseCapabilities));
670 pThis->pDrv->pfnUpdateMouseCapabilities(pThis->pDrv, pThis->mouseCapabilities);
671 }
672 pRequestHeader->rc = VINF_SUCCESS;
673 }
674
675 break;
676 }
677
678 /*
679 * Set a new mouse pointer shape
680 */
681 case VMMDevReq_SetPointerShape:
682 {
683 if (pRequestHeader->size < sizeof(VMMDevReqMousePointer))
684 {
685 AssertMsg(pRequestHeader->size == 0x10028 && pRequestHeader->version == 10000, /* don't bitch about legacy!!! */
686 ("VMMDev mouse shape structure has invalid size %d (%#x) version=%d!\n",
687 pRequestHeader->size, pRequestHeader->size, pRequestHeader->size, pRequestHeader->version));
688 pRequestHeader->rc = VERR_INVALID_PARAMETER;
689 }
690 else
691 {
692 VMMDevReqMousePointer *pointerShape = (VMMDevReqMousePointer*)pRequestHeader;
693
694 bool fVisible = (pointerShape->fFlags & VBOX_MOUSE_POINTER_VISIBLE) != 0;
695 bool fAlpha = (pointerShape->fFlags & VBOX_MOUSE_POINTER_ALPHA) != 0;
696 bool fShape = (pointerShape->fFlags & VBOX_MOUSE_POINTER_SHAPE) != 0;
697
698 Log(("VMMDevReq_SetPointerShape: visible: %d, alpha: %d, shape = %d, width: %d, height: %d\n",
699 fVisible, fAlpha, fShape, pointerShape->width, pointerShape->height));
700
701 if (pRequestHeader->size == sizeof(VMMDevReqMousePointer))
702 {
703 /* The guest did not provide the shape actually. */
704 fShape = false;
705 }
706
707 /* forward call to driver */
708 if (fShape)
709 {
710 pThis->pDrv->pfnUpdatePointerShape(pThis->pDrv,
711 fVisible,
712 fAlpha,
713 pointerShape->xHot, pointerShape->yHot,
714 pointerShape->width, pointerShape->height,
715 pointerShape->pointerData);
716 }
717 else
718 {
719 pThis->pDrv->pfnUpdatePointerShape(pThis->pDrv,
720 fVisible,
721 0,
722 0, 0,
723 0, 0,
724 NULL);
725 }
726 pRequestHeader->rc = VINF_SUCCESS;
727 }
728 break;
729 }
730
731 /*
732 * Query the system time from the host
733 */
734 case VMMDevReq_GetHostTime:
735 {
736 if (pRequestHeader->size != sizeof(VMMDevReqHostTime))
737 {
738 AssertMsgFailed(("VMMDev host time structure has invalid size!\n"));
739 pRequestHeader->rc = VERR_INVALID_PARAMETER;
740 }
741 else if (RT_UNLIKELY(pThis->fGetHostTimeDisabled))
742 pRequestHeader->rc = VERR_NOT_SUPPORTED;
743 else
744 {
745 VMMDevReqHostTime *hostTimeReq = (VMMDevReqHostTime*)pRequestHeader;
746 RTTIMESPEC now;
747 hostTimeReq->time = RTTimeSpecGetMilli(PDMDevHlpUTCNow(pDevIns, &now));
748 pRequestHeader->rc = VINF_SUCCESS;
749 }
750 break;
751 }
752
753 /*
754 * Query information about the hypervisor
755 */
756 case VMMDevReq_GetHypervisorInfo:
757 {
758 if (pRequestHeader->size != sizeof(VMMDevReqHypervisorInfo))
759 {
760 AssertMsgFailed(("VMMDev hypervisor info structure has invalid size!\n"));
761 pRequestHeader->rc = VERR_INVALID_PARAMETER;
762 }
763 else
764 {
765 VMMDevReqHypervisorInfo *hypervisorInfo = (VMMDevReqHypervisorInfo*)pRequestHeader;
766 PVM pVM = PDMDevHlpGetVM(pDevIns);
767 pRequestHeader->rc = PGMR3MappingsSize(pVM, &hypervisorInfo->hypervisorSize);
768 }
769 break;
770 }
771
772 /*
773 * Set hypervisor information
774 */
775 case VMMDevReq_SetHypervisorInfo:
776 {
777 if (pRequestHeader->size != sizeof(VMMDevReqHypervisorInfo))
778 {
779 AssertMsgFailed(("VMMDev hypervisor info structure has invalid size!\n"));
780 pRequestHeader->rc = VERR_INVALID_PARAMETER;
781 }
782 else
783 {
784 VMMDevReqHypervisorInfo *hypervisorInfo = (VMMDevReqHypervisorInfo*)pRequestHeader;
785 PVM pVM = PDMDevHlpGetVM(pDevIns);
786 if (hypervisorInfo->hypervisorStart == 0)
787 pRequestHeader->rc = PGMR3MappingsUnfix(pVM);
788 else
789 {
790 /* only if the client has queried the size before! */
791 uint32_t mappingsSize;
792 pRequestHeader->rc = PGMR3MappingsSize(pVM, &mappingsSize);
793 if (RT_SUCCESS(pRequestHeader->rc) && hypervisorInfo->hypervisorSize == mappingsSize)
794 {
795 /* new reservation */
796 pRequestHeader->rc = PGMR3MappingsFix(pVM, hypervisorInfo->hypervisorStart,
797 hypervisorInfo->hypervisorSize);
798 LogRel(("Guest reported fixed hypervisor window at 0x%p (size = 0x%x, rc = %Rrc)\n",
799 (uintptr_t)hypervisorInfo->hypervisorStart,
800 hypervisorInfo->hypervisorSize,
801 pRequestHeader->rc));
802 }
803 }
804 }
805 break;
806 }
807
808 /*
809 * Set the system power status
810 */
811 case VMMDevReq_SetPowerStatus:
812 {
813 if (pRequestHeader->size != sizeof(VMMDevPowerStateRequest))
814 {
815 AssertMsgFailed(("VMMDev power state request structure has invalid size!\n"));
816 pRequestHeader->rc = VERR_INVALID_PARAMETER;
817 }
818 else
819 {
820 VMMDevPowerStateRequest *powerStateRequest = (VMMDevPowerStateRequest*)pRequestHeader;
821 switch(powerStateRequest->powerState)
822 {
823 case VMMDevPowerState_Pause:
824 {
825 LogRel(("Guest requests the VM to be suspended (paused)\n"));
826 pRequestHeader->rc = rcRet = PDMDevHlpVMSuspend(pDevIns);
827 break;
828 }
829
830 case VMMDevPowerState_PowerOff:
831 {
832 LogRel(("Guest requests the VM to be turned off\n"));
833 pRequestHeader->rc = rcRet = PDMDevHlpVMPowerOff(pDevIns);
834 break;
835 }
836
837 case VMMDevPowerState_SaveState:
838 {
839 /** @todo no API for that yet */
840 pRequestHeader->rc = VERR_NOT_IMPLEMENTED;
841 break;
842 }
843
844 default:
845 AssertMsgFailed(("VMMDev invalid power state request: %d\n", powerStateRequest->powerState));
846 pRequestHeader->rc = VERR_INVALID_PARAMETER;
847 break;
848 }
849 }
850 break;
851 }
852
853 /*
854 * Get display change request
855 */
856 case VMMDevReq_GetDisplayChangeRequest:
857 {
858 if (pRequestHeader->size != sizeof(VMMDevDisplayChangeRequest))
859 {
860 /* Assert only if the size also not equal to a previous version size to prevent
861 * assertion with old additions.
862 */
863 AssertMsg(pRequestHeader->size == sizeof(VMMDevDisplayChangeRequest) - sizeof (uint32_t),
864 ("VMMDev display change request structure has invalid size!\n"));
865 pRequestHeader->rc = VERR_INVALID_PARAMETER;
866 }
867 else
868 {
869 VMMDevDisplayChangeRequest *displayChangeRequest = (VMMDevDisplayChangeRequest*)pRequestHeader;
870
871 if (displayChangeRequest->eventAck == VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST)
872 {
873 /* Remember which resolution the client has queried, subsequent reads will return the same values. */
874 pThis->lastReadDisplayChangeRequest = pThis->displayChangeRequest;
875 }
876
877 /* just pass on the information */
878 Log(("VMMDev: returning display change request xres = %d, yres = %d, bpp = %d\n",
879 pThis->displayChangeRequest.xres, pThis->displayChangeRequest.yres, pThis->displayChangeRequest.bpp));
880 displayChangeRequest->xres = pThis->lastReadDisplayChangeRequest.xres;
881 displayChangeRequest->yres = pThis->lastReadDisplayChangeRequest.yres;
882 displayChangeRequest->bpp = pThis->lastReadDisplayChangeRequest.bpp;
883
884 pRequestHeader->rc = VINF_SUCCESS;
885 }
886 break;
887 }
888
889 case VMMDevReq_GetDisplayChangeRequest2:
890 {
891 if (pRequestHeader->size != sizeof(VMMDevDisplayChangeRequest2))
892 {
893 pRequestHeader->rc = VERR_INVALID_PARAMETER;
894 }
895 else
896 {
897 VMMDevDisplayChangeRequest2 *displayChangeRequest = (VMMDevDisplayChangeRequest2*)pRequestHeader;
898
899 if (displayChangeRequest->eventAck == VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST)
900 {
901 /* Remember which resolution the client has queried, subsequent reads will return the same values. */
902 pThis->lastReadDisplayChangeRequest = pThis->displayChangeRequest;
903 }
904
905 /* just pass on the information */
906 Log(("VMMDev: returning display change request xres = %d, yres = %d, bpp = %d at %d\n",
907 pThis->displayChangeRequest.xres, pThis->displayChangeRequest.yres, pThis->displayChangeRequest.bpp, pThis->displayChangeRequest.display));
908 displayChangeRequest->xres = pThis->lastReadDisplayChangeRequest.xres;
909 displayChangeRequest->yres = pThis->lastReadDisplayChangeRequest.yres;
910 displayChangeRequest->bpp = pThis->lastReadDisplayChangeRequest.bpp;
911 displayChangeRequest->display = pThis->lastReadDisplayChangeRequest.display;
912
913 pRequestHeader->rc = VINF_SUCCESS;
914 }
915 break;
916 }
917
918 /*
919 * Query whether the given video mode is supported
920 */
921 case VMMDevReq_VideoModeSupported:
922 {
923 if (pRequestHeader->size != sizeof(VMMDevVideoModeSupportedRequest))
924 {
925 AssertMsgFailed(("VMMDev video mode supported request structure has invalid size!\n"));
926 pRequestHeader->rc = VERR_INVALID_PARAMETER;
927 }
928 else
929 {
930 VMMDevVideoModeSupportedRequest *videoModeSupportedRequest = (VMMDevVideoModeSupportedRequest*)pRequestHeader;
931 /* forward the call */
932 pRequestHeader->rc = pThis->pDrv->pfnVideoModeSupported(pThis->pDrv,
933 videoModeSupportedRequest->width,
934 videoModeSupportedRequest->height,
935 videoModeSupportedRequest->bpp,
936 &videoModeSupportedRequest->fSupported);
937 }
938 break;
939 }
940
941 /*
942 * Query the height reduction in pixels
943 */
944 case VMMDevReq_GetHeightReduction:
945 {
946 if (pRequestHeader->size != sizeof(VMMDevGetHeightReductionRequest))
947 {
948 AssertMsgFailed(("VMMDev height reduction request structure has invalid size!\n"));
949 pRequestHeader->rc = VERR_INVALID_PARAMETER;
950 }
951 else
952 {
953 VMMDevGetHeightReductionRequest *heightReductionRequest = (VMMDevGetHeightReductionRequest*)pRequestHeader;
954 /* forward the call */
955 pRequestHeader->rc = pThis->pDrv->pfnGetHeightReduction(pThis->pDrv,
956 &heightReductionRequest->heightReduction);
957 }
958 break;
959 }
960
961 /*
962 * Acknowledge VMMDev events
963 */
964 case VMMDevReq_AcknowledgeEvents:
965 {
966 if (pRequestHeader->size != sizeof(VMMDevEvents))
967 {
968 AssertMsgFailed(("VMMDevReq_AcknowledgeEvents structure has invalid size!\n"));
969 pRequestHeader->rc = VERR_INVALID_PARAMETER;
970 }
971 else
972 {
973 if (VBOX_GUEST_ADDITIONS_VERSION_1_03 (pThis))
974 {
975 vmmdevSetIRQ_Legacy_EMT (pThis);
976 }
977 else
978 {
979 VMMDevEvents *pAckRequest;
980
981 if (pThis->fNewGuestFilterMask)
982 {
983 pThis->fNewGuestFilterMask = false;
984 pThis->u32GuestFilterMask = pThis->u32NewGuestFilterMask;
985 }
986
987 pAckRequest = (VMMDevEvents *)pRequestHeader;
988 pAckRequest->events =
989 pThis->u32HostEventFlags & pThis->u32GuestFilterMask;
990
991 pThis->u32HostEventFlags &= ~pThis->u32GuestFilterMask;
992 pThis->pVMMDevRAMR3->V.V1_04.fHaveEvents = false;
993 PDMDevHlpPCISetIrqNoWait (pThis->pDevIns, 0, 0);
994 }
995 pRequestHeader->rc = VINF_SUCCESS;
996 }
997 break;
998 }
999
1000 /*
1001 * Change guest filter mask
1002 */
1003 case VMMDevReq_CtlGuestFilterMask:
1004 {
1005 if (pRequestHeader->size != sizeof(VMMDevCtlGuestFilterMask))
1006 {
1007 AssertMsgFailed(("VMMDevReq_AcknowledgeEvents structure has invalid size!\n"));
1008 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1009 }
1010 else
1011 {
1012 VMMDevCtlGuestFilterMask *pCtlMaskRequest;
1013
1014 pCtlMaskRequest = (VMMDevCtlGuestFilterMask *)pRequestHeader;
1015 /* The HGCM events are enabled by the VMMDev device automatically when any
1016 * HGCM command is issued. The guest then can not disable these events.
1017 */
1018 vmmdevCtlGuestFilterMask_EMT (pThis,
1019 pCtlMaskRequest->u32OrMask,
1020 pCtlMaskRequest->u32NotMask & ~VMMDEV_EVENT_HGCM);
1021 pRequestHeader->rc = VINF_SUCCESS;
1022
1023 }
1024 break;
1025 }
1026
1027#ifdef VBOX_WITH_HGCM
1028 /*
1029 * Process HGCM request
1030 */
1031 case VMMDevReq_HGCMConnect:
1032 {
1033 if (pRequestHeader->size < sizeof(VMMDevHGCMConnect))
1034 {
1035 AssertMsgFailed(("VMMDevReq_HGCMConnect structure has invalid size!\n"));
1036 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1037 }
1038 else if (!pThis->pHGCMDrv)
1039 {
1040 Log(("VMMDevReq_HGCMConnect HGCM Connector is NULL!\n"));
1041 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1042 }
1043 else
1044 {
1045 VMMDevHGCMConnect *pHGCMConnect = (VMMDevHGCMConnect *)pRequestHeader;
1046
1047 Log(("VMMDevReq_HGCMConnect\n"));
1048
1049 pRequestHeader->rc = vmmdevHGCMConnect (pThis, pHGCMConnect, (RTGCPHYS)u32);
1050 }
1051 break;
1052 }
1053
1054 case VMMDevReq_HGCMDisconnect:
1055 {
1056 if (pRequestHeader->size < sizeof(VMMDevHGCMDisconnect))
1057 {
1058 AssertMsgFailed(("VMMDevReq_HGCMDisconnect structure has invalid size!\n"));
1059 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1060 }
1061 else if (!pThis->pHGCMDrv)
1062 {
1063 Log(("VMMDevReq_HGCMDisconnect HGCM Connector is NULL!\n"));
1064 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1065 }
1066 else
1067 {
1068 VMMDevHGCMDisconnect *pHGCMDisconnect = (VMMDevHGCMDisconnect *)pRequestHeader;
1069
1070 Log(("VMMDevReq_VMMDevHGCMDisconnect\n"));
1071 pRequestHeader->rc = vmmdevHGCMDisconnect (pThis, pHGCMDisconnect, (RTGCPHYS)u32);
1072 }
1073 break;
1074 }
1075
1076#ifdef VBOX_WITH_64_BITS_GUESTS
1077 case VMMDevReq_HGCMCall32:
1078 case VMMDevReq_HGCMCall64:
1079#else
1080 case VMMDevReq_HGCMCall:
1081#endif /* VBOX_WITH_64_BITS_GUESTS */
1082 {
1083 if (pRequestHeader->size < sizeof(VMMDevHGCMCall))
1084 {
1085 AssertMsgFailed(("VMMDevReq_HGCMCall structure has invalid size!\n"));
1086 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1087 }
1088 else if (!pThis->pHGCMDrv)
1089 {
1090 Log(("VMMDevReq_HGCMCall HGCM Connector is NULL!\n"));
1091 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1092 }
1093 else
1094 {
1095 VMMDevHGCMCall *pHGCMCall = (VMMDevHGCMCall *)pRequestHeader;
1096
1097 Log2(("VMMDevReq_HGCMCall: sizeof (VMMDevHGCMRequest) = %04X\n", sizeof (VMMDevHGCMCall)));
1098 Log2(("%.*Rhxd\n", pRequestHeader->size, pRequestHeader));
1099
1100#ifdef VBOX_WITH_64_BITS_GUESTS
1101 bool f64Bits = (pRequestHeader->requestType == VMMDevReq_HGCMCall64);
1102#else
1103 bool f64Bits = false;
1104#endif /* VBOX_WITH_64_BITS_GUESTS */
1105
1106 pRequestHeader->rc = vmmdevHGCMCall (pThis, pHGCMCall, (RTGCPHYS)u32, f64Bits);
1107 }
1108 break;
1109 }
1110#endif /* VBOX_WITH_HGCM */
1111
1112 case VMMDevReq_HGCMCancel:
1113 {
1114 if (pRequestHeader->size < sizeof(VMMDevHGCMCancel))
1115 {
1116 AssertMsgFailed(("VMMDevReq_HGCMCancel structure has invalid size!\n"));
1117 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1118 }
1119 else if (!pThis->pHGCMDrv)
1120 {
1121 Log(("VMMDevReq_HGCMCancel HGCM Connector is NULL!\n"));
1122 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1123 }
1124 else
1125 {
1126 VMMDevHGCMCancel *pHGCMCancel = (VMMDevHGCMCancel *)pRequestHeader;
1127
1128 Log(("VMMDevReq_VMMDevHGCMCancel\n"));
1129 pRequestHeader->rc = vmmdevHGCMCancel (pThis, pHGCMCancel, (RTGCPHYS)u32);
1130 }
1131 break;
1132 }
1133
1134 case VMMDevReq_VideoAccelEnable:
1135 {
1136 if (pRequestHeader->size < sizeof(VMMDevVideoAccelEnable))
1137 {
1138 Log(("VMMDevReq_VideoAccelEnable request size too small!!!\n"));
1139 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1140 }
1141 else if (!pThis->pDrv)
1142 {
1143 Log(("VMMDevReq_VideoAccelEnable Connector is NULL!!!\n"));
1144 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1145 }
1146 else
1147 {
1148 VMMDevVideoAccelEnable *ptr = (VMMDevVideoAccelEnable *)pRequestHeader;
1149
1150 if (ptr->cbRingBuffer != VBVA_RING_BUFFER_SIZE)
1151 {
1152 /* The guest driver seems compiled with another headers. */
1153 Log(("VMMDevReq_VideoAccelEnable guest ring buffer size %d, should be %d!!!\n", ptr->cbRingBuffer, VBVA_RING_BUFFER_SIZE));
1154 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1155 }
1156 else
1157 {
1158 /* The request is correct. */
1159 ptr->fu32Status |= VBVA_F_STATUS_ACCEPTED;
1160
1161 LogFlow(("VMMDevReq_VideoAccelEnable ptr->u32Enable = %d\n", ptr->u32Enable));
1162
1163 pRequestHeader->rc = ptr->u32Enable?
1164 pThis->pDrv->pfnVideoAccelEnable (pThis->pDrv, true, &pThis->pVMMDevRAMR3->vbvaMemory):
1165 pThis->pDrv->pfnVideoAccelEnable (pThis->pDrv, false, NULL);
1166
1167 if ( ptr->u32Enable
1168 && RT_SUCCESS (pRequestHeader->rc))
1169 {
1170 ptr->fu32Status |= VBVA_F_STATUS_ENABLED;
1171
1172 /* Remember that guest successfully enabled acceleration.
1173 * We need to reestablish it on restoring the VM from saved state.
1174 */
1175 pThis->u32VideoAccelEnabled = 1;
1176 }
1177 else
1178 {
1179 /* The acceleration was not enabled. Remember that. */
1180 pThis->u32VideoAccelEnabled = 0;
1181 }
1182 }
1183 }
1184 break;
1185 }
1186
1187 case VMMDevReq_VideoAccelFlush:
1188 {
1189 if (pRequestHeader->size < sizeof(VMMDevVideoAccelFlush))
1190 {
1191 AssertMsgFailed(("VMMDevReq_VideoAccelFlush request size too small.\n"));
1192 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1193 }
1194 else if (!pThis->pDrv)
1195 {
1196 Log(("VMMDevReq_VideoAccelFlush Connector is NULL!\n"));
1197 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1198 }
1199 else
1200 {
1201 pThis->pDrv->pfnVideoAccelFlush (pThis->pDrv);
1202
1203 pRequestHeader->rc = VINF_SUCCESS;
1204 }
1205 break;
1206 }
1207
1208 case VMMDevReq_VideoSetVisibleRegion:
1209 {
1210 if (pRequestHeader->size < sizeof(VMMDevVideoSetVisibleRegion))
1211 {
1212 Log(("VMMDevReq_VideoSetVisibleRegion request size too small!!!\n"));
1213 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1214 }
1215 else if (!pThis->pDrv)
1216 {
1217 Log(("VMMDevReq_VideoSetVisibleRegion Connector is NULL!!!\n"));
1218 pRequestHeader->rc = VERR_NOT_SUPPORTED;
1219 }
1220 else
1221 {
1222 VMMDevVideoSetVisibleRegion *ptr = (VMMDevVideoSetVisibleRegion *)pRequestHeader;
1223
1224 if (!ptr->cRect)
1225 {
1226 Log(("VMMDevReq_VideoSetVisibleRegion no rectangles!!!\n"));
1227 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1228 }
1229 else
1230 if (pRequestHeader->size != sizeof(VMMDevVideoSetVisibleRegion) + (ptr->cRect-1)*sizeof(RTRECT))
1231 {
1232 Log(("VMMDevReq_VideoSetVisibleRegion request size too small!!!\n"));
1233 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1234 }
1235 else
1236 {
1237 Log(("VMMDevReq_VideoSetVisibleRegion %d rectangles\n", ptr->cRect));
1238 /* forward the call */
1239 pRequestHeader->rc = pThis->pDrv->pfnSetVisibleRegion(pThis->pDrv, ptr->cRect, &ptr->Rect);
1240 }
1241 }
1242 break;
1243 }
1244
1245 case VMMDevReq_GetSeamlessChangeRequest:
1246 {
1247 if (pRequestHeader->size != sizeof(VMMDevSeamlessChangeRequest))
1248 {
1249 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1250 }
1251 else
1252 {
1253 VMMDevSeamlessChangeRequest *seamlessChangeRequest = (VMMDevSeamlessChangeRequest*)pRequestHeader;
1254 /* just pass on the information */
1255 Log(("VMMDev: returning seamless change request mode=%d\n", pThis->fSeamlessEnabled));
1256 if (pThis->fSeamlessEnabled)
1257 seamlessChangeRequest->mode = VMMDev_Seamless_Visible_Region;
1258 else
1259 seamlessChangeRequest->mode = VMMDev_Seamless_Disabled;
1260
1261 if (seamlessChangeRequest->eventAck == VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST)
1262 {
1263 /* Remember which mode the client has queried. */
1264 pThis->fLastSeamlessEnabled = pThis->fSeamlessEnabled;
1265 }
1266
1267 pRequestHeader->rc = VINF_SUCCESS;
1268 }
1269 break;
1270 }
1271
1272 case VMMDevReq_GetVRDPChangeRequest:
1273 {
1274 if (pRequestHeader->size != sizeof(VMMDevVRDPChangeRequest))
1275 {
1276 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1277 }
1278 else
1279 {
1280 VMMDevVRDPChangeRequest *vrdpChangeRequest = (VMMDevVRDPChangeRequest*)pRequestHeader;
1281 /* just pass on the information */
1282 Log(("VMMDev: returning VRDP status %d level %d\n", pThis->fVRDPEnabled, pThis->u32VRDPExperienceLevel));
1283
1284 vrdpChangeRequest->u8VRDPActive = pThis->fVRDPEnabled;
1285 vrdpChangeRequest->u32VRDPExperienceLevel = pThis->u32VRDPExperienceLevel;
1286
1287 pRequestHeader->rc = VINF_SUCCESS;
1288 }
1289 break;
1290 }
1291
1292 case VMMDevReq_GetMemBalloonChangeRequest:
1293 {
1294 Log(("VMMDevReq_GetMemBalloonChangeRequest\n"));
1295 if (pRequestHeader->size != sizeof(VMMDevGetMemBalloonChangeRequest))
1296 {
1297 AssertFailed();
1298 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1299 }
1300 else
1301 {
1302 VMMDevGetMemBalloonChangeRequest *memBalloonChangeRequest = (VMMDevGetMemBalloonChangeRequest*)pRequestHeader;
1303 /* just pass on the information */
1304 Log(("VMMDev: returning memory balloon size =%d\n", pThis->u32MemoryBalloonSize));
1305 memBalloonChangeRequest->u32BalloonSize = pThis->u32MemoryBalloonSize;
1306 memBalloonChangeRequest->u32PhysMemSize = pThis->cbGuestRAM / (uint64_t)_1M;
1307
1308 if (memBalloonChangeRequest->eventAck == VMMDEV_EVENT_BALLOON_CHANGE_REQUEST)
1309 {
1310 /* Remember which mode the client has queried. */
1311 pThis->u32LastMemoryBalloonSize = pThis->u32MemoryBalloonSize;
1312 }
1313
1314 pRequestHeader->rc = VINF_SUCCESS;
1315 }
1316 break;
1317 }
1318
1319 case VMMDevReq_ChangeMemBalloon:
1320 {
1321 VMMDevChangeMemBalloon *memBalloonChange = (VMMDevChangeMemBalloon*)pRequestHeader;
1322
1323 Log(("VMMDevReq_ChangeMemBalloon\n"));
1324 if ( pRequestHeader->size < sizeof(VMMDevChangeMemBalloon)
1325 || memBalloonChange->cPages != VMMDEV_MEMORY_BALLOON_CHUNK_PAGES
1326 || pRequestHeader->size != (uint32_t)RT_OFFSETOF(VMMDevChangeMemBalloon, aPhysPage[memBalloonChange->cPages]))
1327 {
1328 AssertFailed();
1329 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1330 }
1331 else
1332 {
1333 pRequestHeader->rc = pThis->pDrv->pfnChangeMemoryBalloon(pThis->pDrv, !!memBalloonChange->fInflate, memBalloonChange->cPages, memBalloonChange->aPhysPage);
1334 }
1335 break;
1336 }
1337
1338 case VMMDevReq_GetStatisticsChangeRequest:
1339 {
1340 Log(("VMMDevReq_GetStatisticsChangeRequest\n"));
1341 if (pRequestHeader->size != sizeof(VMMDevGetStatisticsChangeRequest))
1342 {
1343 AssertFailed();
1344 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1345 }
1346 else
1347 {
1348 VMMDevGetStatisticsChangeRequest *statIntervalChangeRequest = (VMMDevGetStatisticsChangeRequest*)pRequestHeader;
1349 /* just pass on the information */
1350 Log(("VMMDev: returning statistics interval %d seconds\n", pThis->u32StatIntervalSize));
1351 statIntervalChangeRequest->u32StatInterval = pThis->u32StatIntervalSize;
1352
1353 if (statIntervalChangeRequest->eventAck == VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST)
1354 {
1355 /* Remember which mode the client has queried. */
1356 pThis->u32LastStatIntervalSize= pThis->u32StatIntervalSize;
1357 }
1358
1359 pRequestHeader->rc = VINF_SUCCESS;
1360 }
1361 break;
1362 }
1363
1364 case VMMDevReq_ReportGuestStats:
1365 {
1366 Log(("VMMDevReq_ReportGuestStats\n"));
1367 if (pRequestHeader->size != sizeof(VMMDevReportGuestStats))
1368 {
1369 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1370 }
1371 else
1372 {
1373 VMMDevReportGuestStats *stats = (VMMDevReportGuestStats*)pRequestHeader;
1374
1375#ifdef DEBUG
1376 VBoxGuestStatistics *pGuestStats = &stats->guestStats;
1377
1378 Log(("Current statistics:\n"));
1379 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_CPU_LOAD_IDLE)
1380 Log(("CPU%d: CPU Load Idle %-3d%%\n", pGuestStats->u32CpuId, pGuestStats->u32CpuLoad_Idle));
1381
1382 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_CPU_LOAD_KERNEL)
1383 Log(("CPU%d: CPU Load Kernel %-3d%%\n", pGuestStats->u32CpuId, pGuestStats->u32CpuLoad_Kernel));
1384
1385 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_CPU_LOAD_USER)
1386 Log(("CPU%d: CPU Load User %-3d%%\n", pGuestStats->u32CpuId, pGuestStats->u32CpuLoad_User));
1387
1388 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_THREADS)
1389 Log(("CPU%d: Thread %d\n", pGuestStats->u32CpuId, pGuestStats->u32Threads));
1390
1391 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PROCESSES)
1392 Log(("CPU%d: Processes %d\n", pGuestStats->u32CpuId, pGuestStats->u32Processes));
1393
1394 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_HANDLES)
1395 Log(("CPU%d: Handles %d\n", pGuestStats->u32CpuId, pGuestStats->u32Handles));
1396
1397 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEMORY_LOAD)
1398 Log(("CPU%d: Memory Load %d%%\n", pGuestStats->u32CpuId, pGuestStats->u32MemoryLoad));
1399
1400 /* Note that reported values are in pages; upper layers expect them in megabytes */
1401 Assert(pGuestStats->u32PageSize == 4096);
1402 if (pGuestStats->u32PageSize != 4096)
1403 pGuestStats->u32PageSize = 4096;
1404
1405 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PHYS_MEM_TOTAL)
1406 Log(("CPU%d: Total physical memory %-4d MB\n", pGuestStats->u32CpuId, (pGuestStats->u32PhysMemTotal + (_1M/pGuestStats->u32PageSize)-1)/ (_1M/pGuestStats->u32PageSize)));
1407
1408 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PHYS_MEM_AVAIL)
1409 Log(("CPU%d: Free physical memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32PhysMemAvail / (_1M/pGuestStats->u32PageSize)));
1410
1411 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PHYS_MEM_BALLOON)
1412 Log(("CPU%d: Memory balloon size %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32PhysMemBalloon / (_1M/pGuestStats->u32PageSize)));
1413
1414 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_COMMIT_TOTAL)
1415 Log(("CPU%d: Committed memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemCommitTotal / (_1M/pGuestStats->u32PageSize)));
1416
1417 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_KERNEL_TOTAL)
1418 Log(("CPU%d: Total kernel memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemKernelTotal / (_1M/pGuestStats->u32PageSize)));
1419
1420 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_KERNEL_PAGED)
1421 Log(("CPU%d: Paged kernel memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemKernelPaged / (_1M/pGuestStats->u32PageSize)));
1422
1423 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_KERNEL_NONPAGED)
1424 Log(("CPU%d: Nonpaged kernel memory %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemKernelNonPaged / (_1M/pGuestStats->u32PageSize)));
1425
1426 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_MEM_SYSTEM_CACHE)
1427 Log(("CPU%d: System cache size %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32MemSystemCache / (_1M/pGuestStats->u32PageSize)));
1428
1429 if (pGuestStats->u32StatCaps & VBOX_GUEST_STAT_PAGE_FILE_SIZE)
1430 Log(("CPU%d: Page file size %-4d MB\n", pGuestStats->u32CpuId, pGuestStats->u32PageFileSize / (_1M/pGuestStats->u32PageSize)));
1431 Log(("Statistics end *******************\n"));
1432#endif
1433
1434 /* forward the call */
1435 pRequestHeader->rc = pThis->pDrv->pfnReportStatistics(pThis->pDrv, &stats->guestStats);
1436 }
1437 break;
1438 }
1439
1440 case VMMDevReq_QueryCredentials:
1441 {
1442 if (pRequestHeader->size != sizeof(VMMDevCredentials))
1443 {
1444 AssertMsgFailed(("VMMDevReq_QueryCredentials request size too small.\n"));
1445 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1446 }
1447 else
1448 {
1449 VMMDevCredentials *credentials = (VMMDevCredentials*)pRequestHeader;
1450
1451 /* let's start by nulling out the data */
1452 memset(credentials->szUserName, '\0', VMMDEV_CREDENTIALS_STRLEN);
1453 memset(credentials->szPassword, '\0', VMMDEV_CREDENTIALS_STRLEN);
1454 memset(credentials->szDomain, '\0', VMMDEV_CREDENTIALS_STRLEN);
1455
1456 /* should we return whether we got credentials for a logon? */
1457 if (credentials->u32Flags & VMMDEV_CREDENTIALS_QUERYPRESENCE)
1458 {
1459 if ( pThis->credentialsLogon.szUserName[0]
1460 || pThis->credentialsLogon.szPassword[0]
1461 || pThis->credentialsLogon.szDomain[0])
1462 {
1463 credentials->u32Flags |= VMMDEV_CREDENTIALS_PRESENT;
1464 }
1465 else
1466 {
1467 credentials->u32Flags &= ~VMMDEV_CREDENTIALS_PRESENT;
1468 }
1469 }
1470
1471 /* does the guest want to read logon credentials? */
1472 if (credentials->u32Flags & VMMDEV_CREDENTIALS_READ)
1473 {
1474 if (pThis->credentialsLogon.szUserName[0])
1475 strcpy(credentials->szUserName, pThis->credentialsLogon.szUserName);
1476 if (pThis->credentialsLogon.szPassword[0])
1477 strcpy(credentials->szPassword, pThis->credentialsLogon.szPassword);
1478 if (pThis->credentialsLogon.szDomain[0])
1479 strcpy(credentials->szDomain, pThis->credentialsLogon.szDomain);
1480 if (!pThis->credentialsLogon.fAllowInteractiveLogon)
1481 credentials->u32Flags |= VMMDEV_CREDENTIALS_NOLOCALLOGON;
1482 else
1483 credentials->u32Flags &= ~VMMDEV_CREDENTIALS_NOLOCALLOGON;
1484 }
1485
1486 if (!pThis->fKeepCredentials)
1487 {
1488 /* does the caller want us to destroy the logon credentials? */
1489 if (credentials->u32Flags & VMMDEV_CREDENTIALS_CLEAR)
1490 {
1491 memset(pThis->credentialsLogon.szUserName, '\0', VMMDEV_CREDENTIALS_STRLEN);
1492 memset(pThis->credentialsLogon.szPassword, '\0', VMMDEV_CREDENTIALS_STRLEN);
1493 memset(pThis->credentialsLogon.szDomain, '\0', VMMDEV_CREDENTIALS_STRLEN);
1494 }
1495 }
1496
1497 /* does the guest want to read credentials for verification? */
1498 if (credentials->u32Flags & VMMDEV_CREDENTIALS_READJUDGE)
1499 {
1500 if (pThis->credentialsJudge.szUserName[0])
1501 strcpy(credentials->szUserName, pThis->credentialsJudge.szUserName);
1502 if (pThis->credentialsJudge.szPassword[0])
1503 strcpy(credentials->szPassword, pThis->credentialsJudge.szPassword);
1504 if (pThis->credentialsJudge.szDomain[0])
1505 strcpy(credentials->szDomain, pThis->credentialsJudge.szDomain);
1506 }
1507
1508 /* does the caller want us to destroy the judgement credentials? */
1509 if (credentials->u32Flags & VMMDEV_CREDENTIALS_CLEARJUDGE)
1510 {
1511 memset(pThis->credentialsJudge.szUserName, '\0', VMMDEV_CREDENTIALS_STRLEN);
1512 memset(pThis->credentialsJudge.szPassword, '\0', VMMDEV_CREDENTIALS_STRLEN);
1513 memset(pThis->credentialsJudge.szDomain, '\0', VMMDEV_CREDENTIALS_STRLEN);
1514 }
1515
1516 pRequestHeader->rc = VINF_SUCCESS;
1517 }
1518 break;
1519 }
1520
1521 case VMMDevReq_ReportCredentialsJudgement:
1522 {
1523 if (pRequestHeader->size != sizeof(VMMDevCredentials))
1524 {
1525 AssertMsgFailed(("VMMDevReq_ReportCredentialsJudgement request size too small.\n"));
1526 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1527 }
1528 else
1529 {
1530 VMMDevCredentials *credentials = (VMMDevCredentials*)pRequestHeader;
1531
1532 /* what does the guest think about the credentials? (note: the order is important here!) */
1533 if (credentials->u32Flags & VMMDEV_CREDENTIALS_JUDGE_DENY)
1534 {
1535 pThis->pDrv->pfnSetCredentialsJudgementResult(pThis->pDrv, VMMDEV_CREDENTIALS_JUDGE_DENY);
1536 }
1537 else if (credentials->u32Flags & VMMDEV_CREDENTIALS_JUDGE_NOJUDGEMENT)
1538 {
1539 pThis->pDrv->pfnSetCredentialsJudgementResult(pThis->pDrv, VMMDEV_CREDENTIALS_JUDGE_NOJUDGEMENT);
1540 }
1541 else if (credentials->u32Flags & VMMDEV_CREDENTIALS_JUDGE_OK)
1542 {
1543 pThis->pDrv->pfnSetCredentialsJudgementResult(pThis->pDrv, VMMDEV_CREDENTIALS_JUDGE_OK);
1544 }
1545 else
1546 Log(("VMMDevReq_ReportCredentialsJudgement: invalid flags: %d!!!\n", credentials->u32Flags));
1547
1548 pRequestHeader->rc = VINF_SUCCESS;
1549 }
1550 break;
1551 }
1552
1553#ifdef DEBUG
1554 case VMMDevReq_LogString:
1555 {
1556 if (pRequestHeader->size < sizeof(VMMDevReqLogString))
1557 {
1558 AssertMsgFailed(("VMMDevReq_LogString request size too small.\n"));
1559 pRequestHeader->rc = VERR_INVALID_PARAMETER;
1560 }
1561 else
1562 {
1563 VMMDevReqLogString *pReqLogString = (VMMDevReqLogString*)pRequestHeader;
1564#undef LOG_GROUP
1565#define LOG_GROUP LOG_GROUP_DEV_VMM_BACKDOOR
1566// Log(("Guest Log: %s", pReqLogString->szString));
1567 Log(("DEBUG LOG: %s", pReqLogString->szString));
1568
1569#undef LOG_GROUP
1570#define LOG_GROUP LOG_GROUP_DEV_VMM
1571 pRequestHeader->rc = VINF_SUCCESS;
1572 }
1573 break;
1574 }
1575#endif
1576 default:
1577 {
1578 pRequestHeader->rc = VERR_NOT_IMPLEMENTED;
1579
1580 Log(("VMMDev unknown request type %d\n", pRequestHeader->requestType));
1581
1582 break;
1583 }
1584 }
1585
1586end:
1587 /* Write the result back to guest memory */
1588 if (pRequestHeader)
1589 {
1590 PDMDevHlpPhysWrite(pDevIns, (RTGCPHYS)u32, pRequestHeader, pRequestHeader->size);
1591 RTMemFree(pRequestHeader);
1592 }
1593 else
1594 {
1595 /* early error case; write back header only */
1596 PDMDevHlpPhysWrite(pDevIns, (RTGCPHYS)u32, &requestHeader, sizeof(requestHeader));
1597 }
1598 return rcRet;
1599}
1600
1601/**
1602 * Callback function for mapping an PCI I/O region.
1603 *
1604 * @return VBox status code.
1605 * @param pPciDev Pointer to PCI device. Use pPciDev->pDevIns to get the device instance.
1606 * @param iRegion The region number.
1607 * @param GCPhysAddress Physical address of the region. If iType is PCI_ADDRESS_SPACE_IO, this is an
1608 * I/O port, else it's a physical address.
1609 * This address is *NOT* relative to pci_mem_base like earlier!
1610 * @param enmType One of the PCI_ADDRESS_SPACE_* values.
1611 */
1612static DECLCALLBACK(int) vmmdevIORAMRegionMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion, RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
1613{
1614 LogFlow(("vmmdevR3IORAMRegionMap: iRegion=%d GCPhysAddress=%RGp cb=%#x enmType=%d\n", iRegion, GCPhysAddress, cb, enmType));
1615 VMMDevState *pThis = PCIDEV_2_VMMDEVSTATE(pPciDev);
1616 int rc;
1617
1618 if (iRegion == 1)
1619 {
1620 AssertReturn(enmType == PCI_ADDRESS_SPACE_MEM, VERR_INTERNAL_ERROR);
1621 Assert(pThis->pVMMDevRAMR3 != NULL);
1622 if (GCPhysAddress != NIL_RTGCPHYS)
1623 {
1624 /*
1625 * Map the MMIO2 memory.
1626 */
1627 pThis->GCPhysVMMDevRAM = GCPhysAddress;
1628 Assert(pThis->GCPhysVMMDevRAM == GCPhysAddress);
1629 rc = PDMDevHlpMMIO2Map(pPciDev->pDevIns, iRegion, GCPhysAddress);
1630 }
1631 else
1632 {
1633 /*
1634 * It is about to be unmapped, just clean up.
1635 */
1636 pThis->GCPhysVMMDevRAM = NIL_RTGCPHYS32;
1637 rc = VINF_SUCCESS;
1638 }
1639 }
1640 else if (iRegion == 2)
1641 {
1642 AssertReturn(enmType == PCI_ADDRESS_SPACE_MEM_PREFETCH, VERR_INTERNAL_ERROR);
1643 Assert(pThis->pVMMDevHeapR3 != NULL);
1644 if (GCPhysAddress != NIL_RTGCPHYS)
1645 {
1646 /*
1647 * Map the MMIO2 memory.
1648 */
1649 pThis->GCPhysVMMDevHeap = GCPhysAddress;
1650 Assert(pThis->GCPhysVMMDevHeap == GCPhysAddress);
1651 rc = PDMDevHlpMMIO2Map(pPciDev->pDevIns, iRegion, GCPhysAddress);
1652 if (RT_SUCCESS(rc))
1653 rc = PDMDevHlpRegisterVMMDevHeap(pPciDev->pDevIns, GCPhysAddress, pThis->pVMMDevHeapR3, VMMDEV_HEAP_SIZE);
1654 }
1655 else
1656 {
1657 /*
1658 * It is about to be unmapped, just clean up.
1659 */
1660 PDMDevHlpUnregisterVMMDevHeap(pPciDev->pDevIns, pThis->GCPhysVMMDevHeap);
1661 pThis->GCPhysVMMDevHeap = NIL_RTGCPHYS32;
1662 rc = VINF_SUCCESS;
1663 }
1664 }
1665 else
1666 {
1667 AssertMsgFailed(("%d\n", iRegion));
1668 rc = VERR_INVALID_PARAMETER;
1669 }
1670
1671 return rc;
1672}
1673
1674
1675/**
1676 * Callback function for mapping a PCI I/O region.
1677 *
1678 * @return VBox status code.
1679 * @param pPciDev Pointer to PCI device. Use pPciDev->pDevIns to get the device instance.
1680 * @param iRegion The region number.
1681 * @param GCPhysAddress Physical address of the region. If iType is PCI_ADDRESS_SPACE_IO, this is an
1682 * I/O port, else it's a physical address.
1683 * This address is *NOT* relative to pci_mem_base like earlier!
1684 * @param enmType One of the PCI_ADDRESS_SPACE_* values.
1685 */
1686static DECLCALLBACK(int) vmmdevIOPortRegionMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion, RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
1687{
1688 VMMDevState *pThis = PCIDEV_2_VMMDEVSTATE(pPciDev);
1689 int rc = VINF_SUCCESS;
1690
1691 Assert(enmType == PCI_ADDRESS_SPACE_IO);
1692 Assert(iRegion == 0);
1693 AssertMsg(RT_ALIGN(GCPhysAddress, 8) == GCPhysAddress, ("Expected 8 byte alignment. GCPhysAddress=%#x\n", GCPhysAddress));
1694
1695 /*
1696 * Save the base port address to simplify Port offset calculations.
1697 */
1698 pThis->PortBase = (RTIOPORT)GCPhysAddress;
1699
1700 /*
1701 * Register our port IO handlers.
1702 */
1703 rc = PDMDevHlpIOPortRegister(pPciDev->pDevIns,
1704 (RTIOPORT)GCPhysAddress + PORT_VMMDEV_REQUEST_OFFSET, 1,
1705 (void*)pThis, vmmdevRequestHandler,
1706 NULL, NULL, NULL, "VMMDev Request Handler");
1707 AssertRC(rc);
1708 return rc;
1709}
1710
1711/**
1712 * Queries an interface to the driver.
1713 *
1714 * @returns Pointer to interface.
1715 * @returns NULL if the interface was not supported by the driver.
1716 * @param pInterface Pointer to this interface structure.
1717 * @param enmInterface The requested interface identification.
1718 * @thread Any thread.
1719 */
1720static DECLCALLBACK(void *) vmmdevPortQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
1721{
1722 VMMDevState *pThis = (VMMDevState*)((uintptr_t)pInterface - RT_OFFSETOF(VMMDevState, Base));
1723 switch (enmInterface)
1724 {
1725 case PDMINTERFACE_BASE:
1726 return &pThis->Base;
1727 case PDMINTERFACE_VMMDEV_PORT:
1728 return &pThis->Port;
1729#ifdef VBOX_WITH_HGCM
1730 case PDMINTERFACE_HGCM_PORT:
1731 return &pThis->HGCMPort;
1732#endif
1733 case PDMINTERFACE_LED_PORTS:
1734 /* Currently only for shared folders */
1735 return &pThis->SharedFolders.ILeds;
1736 default:
1737 return NULL;
1738 }
1739}
1740
1741/**
1742 * Gets the pointer to the status LED of a unit.
1743 *
1744 * @returns VBox status code.
1745 * @param pInterface Pointer to the interface structure containing the called function pointer.
1746 * @param iLUN The unit which status LED we desire.
1747 * @param ppLed Where to store the LED pointer.
1748 */
1749static DECLCALLBACK(int) vmmdevQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
1750{
1751 VMMDevState *pThis = (VMMDevState *)( (uintptr_t)pInterface - RT_OFFSETOF(VMMDevState, SharedFolders.ILeds) );
1752 if (iLUN == 0) /* LUN 0 is shared folders */
1753 {
1754 *ppLed = &pThis->SharedFolders.Led;
1755 return VINF_SUCCESS;
1756 }
1757 return VERR_PDM_LUN_NOT_FOUND;
1758}
1759
1760/* -=-=-=-=-=- IVMMDevPort -=-=-=-=-=- */
1761
1762/** Converts a VMMDev port interface pointer to a VMMDev state pointer. */
1763#define IVMMDEVPORT_2_VMMDEVSTATE(pInterface) ( (VMMDevState*)((uintptr_t)pInterface - RT_OFFSETOF(VMMDevState, Port)) )
1764
1765
1766/**
1767 * Return the current absolute mouse position in pixels
1768 *
1769 * @returns VBox status code
1770 * @param pAbsX Pointer of result value, can be NULL
1771 * @param pAbsY Pointer of result value, can be NULL
1772 */
1773static DECLCALLBACK(int) vmmdevQueryAbsoluteMouse(PPDMIVMMDEVPORT pInterface, uint32_t *pAbsX, uint32_t *pAbsY)
1774{
1775 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1776 if (pAbsX)
1777 *pAbsX = pThis->mouseXAbs;
1778 if (pAbsY)
1779 *pAbsY = pThis->mouseYAbs;
1780 return VINF_SUCCESS;
1781}
1782
1783/**
1784 * Set the new absolute mouse position in pixels
1785 *
1786 * @returns VBox status code
1787 * @param absX New absolute X position
1788 * @param absY New absolute Y position
1789 */
1790static DECLCALLBACK(int) vmmdevSetAbsoluteMouse(PPDMIVMMDEVPORT pInterface, uint32_t absX, uint32_t absY)
1791{
1792 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1793 Log2(("vmmdevSetAbsoluteMouse: settings absolute position to x = %d, y = %d\n", absX, absY));
1794 pThis->mouseXAbs = absX;
1795 pThis->mouseYAbs = absY;
1796 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
1797 return VINF_SUCCESS;
1798}
1799
1800/**
1801 * Return the current mouse capability flags
1802 *
1803 * @returns VBox status code
1804 * @param pCapabilities Pointer of result value
1805 */
1806static DECLCALLBACK(int) vmmdevQueryMouseCapabilities(PPDMIVMMDEVPORT pInterface, uint32_t *pCapabilities)
1807{
1808 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1809 if (!pCapabilities)
1810 return VERR_INVALID_PARAMETER;
1811 *pCapabilities = pThis->mouseCapabilities;
1812 return VINF_SUCCESS;
1813}
1814
1815/**
1816 * Set the current mouse capability flag (host side)
1817 *
1818 * @returns VBox status code
1819 * @param capabilities Capability mask
1820 */
1821static DECLCALLBACK(int) vmmdevSetMouseCapabilities(PPDMIVMMDEVPORT pInterface, uint32_t capabilities)
1822{
1823 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1824
1825 bool bCapsChanged = ((capabilities & VMMDEV_MOUSEHOSTWANTSABS)
1826 != (pThis->mouseCapabilities & VMMDEV_MOUSEHOSTWANTSABS));
1827
1828 Log(("vmmdevSetMouseCapabilities: bCapsChanged %d\n", bCapsChanged));
1829
1830 if (capabilities & VMMDEV_MOUSEHOSTCANNOTHWPOINTER)
1831 pThis->mouseCapabilities |= VMMDEV_MOUSEHOSTCANNOTHWPOINTER;
1832 else
1833 pThis->mouseCapabilities &= ~VMMDEV_MOUSEHOSTCANNOTHWPOINTER;
1834
1835 if (capabilities & VMMDEV_MOUSEHOSTWANTSABS)
1836 pThis->mouseCapabilities |= VMMDEV_MOUSEHOSTWANTSABS;
1837 else
1838 pThis->mouseCapabilities &= ~VMMDEV_MOUSEHOSTWANTSABS;
1839
1840 if (bCapsChanged)
1841 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_MOUSE_CAPABILITIES_CHANGED);
1842
1843 return VINF_SUCCESS;
1844}
1845
1846
1847static DECLCALLBACK(int) vmmdevRequestDisplayChange(PPDMIVMMDEVPORT pInterface, uint32_t xres, uint32_t yres, uint32_t bpp, uint32_t display)
1848{
1849 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1850
1851 /* Verify that the new resolution is different and that guest does not yet know about it. */
1852 bool fSameResolution = (!xres || (pThis->lastReadDisplayChangeRequest.xres == xres)) &&
1853 (!yres || (pThis->lastReadDisplayChangeRequest.yres == yres)) &&
1854 (!bpp || (pThis->lastReadDisplayChangeRequest.bpp == bpp)) &&
1855 pThis->lastReadDisplayChangeRequest.display == display;
1856
1857 if (!xres && !yres && !bpp)
1858 {
1859 /* Special case of reset video mode. */
1860 fSameResolution = false;
1861 }
1862
1863 Log3(("vmmdevRequestDisplayChange: same=%d. new: xres=%d, yres=%d, bpp=%d, display=%d. old: xres=%d, yres=%d, bpp=%d, display=%d.\n",
1864 fSameResolution, xres, yres, bpp, display, pThis->lastReadDisplayChangeRequest.xres, pThis->lastReadDisplayChangeRequest.yres, pThis->lastReadDisplayChangeRequest.bpp, pThis->lastReadDisplayChangeRequest.display));
1865
1866 if (!fSameResolution)
1867 {
1868 LogRel(("VMMDev::SetVideoModeHint: got a video mode hint (%dx%dx%d) at %d\n",
1869 xres, yres, bpp, display));
1870
1871 /* we could validate the information here but hey, the guest can do that as well! */
1872 pThis->displayChangeRequest.xres = xres;
1873 pThis->displayChangeRequest.yres = yres;
1874 pThis->displayChangeRequest.bpp = bpp;
1875 pThis->displayChangeRequest.display = display;
1876
1877 /* IRQ so the guest knows what's going on */
1878 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST);
1879 }
1880
1881 return VINF_SUCCESS;
1882}
1883
1884static DECLCALLBACK(int) vmmdevRequestSeamlessChange(PPDMIVMMDEVPORT pInterface, bool fEnabled)
1885{
1886 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1887
1888 /* Verify that the new resolution is different and that guest does not yet know about it. */
1889 bool fSameMode = (pThis->fLastSeamlessEnabled == fEnabled);
1890
1891 Log(("vmmdevRequestSeamlessChange: same=%d. new=%d\n", fSameMode, fEnabled));
1892
1893 if (!fSameMode)
1894 {
1895 /* we could validate the information here but hey, the guest can do that as well! */
1896 pThis->fSeamlessEnabled = fEnabled;
1897
1898 /* IRQ so the guest knows what's going on */
1899 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_SEAMLESS_MODE_CHANGE_REQUEST);
1900 }
1901
1902 return VINF_SUCCESS;
1903}
1904
1905static DECLCALLBACK(int) vmmdevSetMemoryBalloon(PPDMIVMMDEVPORT pInterface, uint32_t ulBalloonSize)
1906{
1907 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1908
1909 /* Verify that the new resolution is different and that guest does not yet know about it. */
1910 bool fSame = (pThis->u32LastMemoryBalloonSize == ulBalloonSize);
1911
1912 Log(("vmmdevSetMemoryBalloon: old=%d. new=%d\n", pThis->u32LastMemoryBalloonSize, ulBalloonSize));
1913
1914 if (!fSame)
1915 {
1916 /* we could validate the information here but hey, the guest can do that as well! */
1917 pThis->u32MemoryBalloonSize = ulBalloonSize;
1918
1919 /* IRQ so the guest knows what's going on */
1920 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_BALLOON_CHANGE_REQUEST);
1921 }
1922
1923 return VINF_SUCCESS;
1924}
1925
1926static DECLCALLBACK(int) vmmdevVRDPChange(PPDMIVMMDEVPORT pInterface, bool fVRDPEnabled, uint32_t u32VRDPExperienceLevel)
1927{
1928 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1929
1930 bool fSame = (pThis->fVRDPEnabled == fVRDPEnabled);
1931
1932 Log(("vmmdevVRDPChange: old=%d. new=%d\n", pThis->fVRDPEnabled, fVRDPEnabled));
1933
1934 if (!fSame)
1935 {
1936 pThis->fVRDPEnabled = fVRDPEnabled;
1937 pThis->u32VRDPExperienceLevel = u32VRDPExperienceLevel;
1938
1939 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_VRDP);
1940 }
1941
1942 return VINF_SUCCESS;
1943}
1944
1945static DECLCALLBACK(int) vmmdevSetStatisticsInterval(PPDMIVMMDEVPORT pInterface, uint32_t ulStatInterval)
1946{
1947 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1948
1949 /* Verify that the new resolution is different and that guest does not yet know about it. */
1950 bool fSame = (pThis->u32LastStatIntervalSize == ulStatInterval);
1951
1952 Log(("vmmdevSetStatisticsInterval: old=%d. new=%d\n", pThis->u32LastStatIntervalSize, ulStatInterval));
1953
1954 if (!fSame)
1955 {
1956 /* we could validate the information here but hey, the guest can do that as well! */
1957 pThis->u32StatIntervalSize = ulStatInterval;
1958
1959 /* IRQ so the guest knows what's going on */
1960 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_STATISTICS_INTERVAL_CHANGE_REQUEST);
1961 }
1962
1963 return VINF_SUCCESS;
1964}
1965
1966
1967static DECLCALLBACK(int) vmmdevSetCredentials(PPDMIVMMDEVPORT pInterface, const char *pszUsername,
1968 const char *pszPassword, const char *pszDomain,
1969 uint32_t u32Flags)
1970{
1971 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
1972
1973 /* logon mode? */
1974 if (u32Flags & VMMDEV_SETCREDENTIALS_GUESTLOGON)
1975 {
1976 /* memorize the data */
1977 strcpy(pThis->credentialsLogon.szUserName, pszUsername);
1978 strcpy(pThis->credentialsLogon.szPassword, pszPassword);
1979 strcpy(pThis->credentialsLogon.szDomain, pszDomain);
1980 pThis->credentialsLogon.fAllowInteractiveLogon = !(u32Flags & VMMDEV_SETCREDENTIALS_NOLOCALLOGON);
1981 }
1982 /* credentials verification mode? */
1983 else if (u32Flags & VMMDEV_SETCREDENTIALS_JUDGE)
1984 {
1985 /* memorize the data */
1986 strcpy(pThis->credentialsJudge.szUserName, pszUsername);
1987 strcpy(pThis->credentialsJudge.szPassword, pszPassword);
1988 strcpy(pThis->credentialsJudge.szDomain, pszDomain);
1989
1990 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_JUDGE_CREDENTIALS);
1991 }
1992 else
1993 return VERR_INVALID_PARAMETER;
1994
1995 return VINF_SUCCESS;
1996}
1997
1998/**
1999 * Notification from the Display. Especially useful when
2000 * acceleration is disabled after a video mode change.
2001 *
2002 * @param fEnable Current acceleration status.
2003 */
2004static DECLCALLBACK(void) vmmdevVBVAChange(PPDMIVMMDEVPORT pInterface, bool fEnabled)
2005{
2006 VMMDevState *pThis = IVMMDEVPORT_2_VMMDEVSTATE(pInterface);
2007
2008 Log(("vmmdevVBVAChange: fEnabled = %d\n", fEnabled));
2009
2010 if (pThis)
2011 {
2012 pThis->u32VideoAccelEnabled = fEnabled;
2013 }
2014
2015 return;
2016}
2017
2018
2019/* -=-=-=-=-=- IHGCMPort -=-=-=-=-=- */
2020
2021/** Converts a VMMDev port interface pointer to a VMMDev state pointer. */
2022#define IHGCMPORT_2_VMMDEVSTATE(pInterface) ( (VMMDevState*)((uintptr_t)pInterface - RT_OFFSETOF(VMMDevState, HGCMPort)) )
2023
2024
2025
2026#define VMMDEV_SSM_VERSION 9
2027
2028/**
2029 * Saves a state of the VMM device.
2030 *
2031 * @returns VBox status code.
2032 * @param pDevIns The device instance.
2033 * @param pSSMHandle The handle to save the state to.
2034 */
2035static DECLCALLBACK(int) vmmdevSaveState(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle)
2036{
2037 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState*);
2038 SSMR3PutU32(pSSMHandle, pThis->hypervisorSize);
2039 SSMR3PutU32(pSSMHandle, pThis->mouseCapabilities);
2040 SSMR3PutU32(pSSMHandle, pThis->mouseXAbs);
2041 SSMR3PutU32(pSSMHandle, pThis->mouseYAbs);
2042
2043 SSMR3PutBool(pSSMHandle, pThis->fNewGuestFilterMask);
2044 SSMR3PutU32(pSSMHandle, pThis->u32NewGuestFilterMask);
2045 SSMR3PutU32(pSSMHandle, pThis->u32GuestFilterMask);
2046 SSMR3PutU32(pSSMHandle, pThis->u32HostEventFlags);
2047 // here be dragons (probably)
2048// SSMR3PutBool(pSSMHandle, pThis->pVMMDevRAMR3->V.V1_04.fHaveEvents);
2049 SSMR3PutMem(pSSMHandle, &pThis->pVMMDevRAMR3->V, sizeof (pThis->pVMMDevRAMR3->V));
2050
2051 SSMR3PutMem(pSSMHandle, &pThis->guestInfo, sizeof (pThis->guestInfo));
2052 SSMR3PutU32(pSSMHandle, pThis->fu32AdditionsOk);
2053 SSMR3PutU32(pSSMHandle, pThis->u32VideoAccelEnabled);
2054
2055 SSMR3PutU32(pSSMHandle, pThis->guestCaps);
2056
2057#ifdef VBOX_WITH_HGCM
2058 vmmdevHGCMSaveState (pThis, pSSMHandle);
2059#endif /* VBOX_WITH_HGCM */
2060
2061 return VINF_SUCCESS;
2062}
2063
2064/**
2065 * Loads the saved VMM device state.
2066 *
2067 * @returns VBox status code.
2068 * @param pDevIns The device instance.
2069 * @param pSSMHandle The handle to the saved state.
2070 * @param u32Version The data unit version number.
2071 */
2072static DECLCALLBACK(int) vmmdevLoadState(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle, uint32_t u32Version)
2073{
2074 /** @todo The code load code is assuming we're always loaded into a fresh VM. */
2075 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState*);
2076 if ( SSM_VERSION_MAJOR_CHANGED(u32Version, VMMDEV_SSM_VERSION)
2077 || (SSM_VERSION_MINOR(u32Version) < 6))
2078 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
2079
2080 SSMR3GetU32(pSSMHandle, &pThis->hypervisorSize);
2081 SSMR3GetU32(pSSMHandle, &pThis->mouseCapabilities);
2082 SSMR3GetU32(pSSMHandle, &pThis->mouseXAbs);
2083 SSMR3GetU32(pSSMHandle, &pThis->mouseYAbs);
2084
2085 SSMR3GetBool(pSSMHandle, &pThis->fNewGuestFilterMask);
2086 SSMR3GetU32(pSSMHandle, &pThis->u32NewGuestFilterMask);
2087 SSMR3GetU32(pSSMHandle, &pThis->u32GuestFilterMask);
2088 SSMR3GetU32(pSSMHandle, &pThis->u32HostEventFlags);
2089// SSMR3GetBool(pSSMHandle, &pThis->pVMMDevRAMR3->fHaveEvents);
2090 // here be dragons (probably)
2091 SSMR3GetMem(pSSMHandle, &pThis->pVMMDevRAMR3->V, sizeof (pThis->pVMMDevRAMR3->V));
2092
2093 SSMR3GetMem(pSSMHandle, &pThis->guestInfo, sizeof (pThis->guestInfo));
2094 SSMR3GetU32(pSSMHandle, &pThis->fu32AdditionsOk);
2095 SSMR3GetU32(pSSMHandle, &pThis->u32VideoAccelEnabled);
2096
2097 SSMR3GetU32(pSSMHandle, &pThis->guestCaps);
2098
2099 /* Attributes which were temporarily introduced in r30072 */
2100 if ( SSM_VERSION_MAJOR(u32Version) == 0
2101 && SSM_VERSION_MINOR(u32Version) == 7)
2102 {
2103 uint32_t temp;
2104 SSMR3GetU32(pSSMHandle, &temp);
2105 SSMR3GetU32(pSSMHandle, &temp);
2106 }
2107
2108#ifdef VBOX_WITH_HGCM
2109 vmmdevHGCMLoadState (pThis, pSSMHandle, u32Version);
2110#endif /* VBOX_WITH_HGCM */
2111
2112 /*
2113 * On a resume, we send the capabilities changed message so
2114 * that listeners can sync their state again
2115 */
2116 Log(("vmmdevLoadState: capabilities changed (%x), informing connector\n", pThis->mouseCapabilities));
2117 if (pThis->pDrv)
2118 pThis->pDrv->pfnUpdateMouseCapabilities(pThis->pDrv, pThis->mouseCapabilities);
2119
2120 /* Reestablish the acceleration status. */
2121 if ( pThis->u32VideoAccelEnabled
2122 && pThis->pDrv)
2123 {
2124 pThis->pDrv->pfnVideoAccelEnable (pThis->pDrv, !!pThis->u32VideoAccelEnabled, &pThis->pVMMDevRAMR3->vbvaMemory);
2125 }
2126
2127 if (pThis->fu32AdditionsOk)
2128 {
2129 LogRel(("Guest Additions information report: additionsVersion = 0x%08X, osType = 0x%08X\n",
2130 pThis->guestInfo.additionsVersion,
2131 pThis->guestInfo.osType));
2132 if (pThis->pDrv)
2133 pThis->pDrv->pfnUpdateGuestVersion(pThis->pDrv, &pThis->guestInfo);
2134 }
2135 if (pThis->pDrv)
2136 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pThis->guestCaps);
2137
2138 return VINF_SUCCESS;
2139}
2140
2141/**
2142 * Load state done callback. Notify guest of restore event.
2143 *
2144 * @returns VBox status code.
2145 * @param pDevIns The device instance.
2146 * @param pSSMHandle The handle to the saved state.
2147 */
2148static DECLCALLBACK(int) vmmdevLoadStateDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle)
2149{
2150 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState*);
2151
2152#ifdef VBOX_WITH_HGCM
2153 vmmdevHGCMLoadStateDone (pThis, pSSMHandle);
2154#endif /* VBOX_WITH_HGCM */
2155
2156 VMMDevNotifyGuest (pThis, VMMDEV_EVENT_RESTORED);
2157
2158 return VINF_SUCCESS;
2159}
2160
2161/**
2162 * (Re-)initializes the MMIO2 data.
2163 *
2164 * @param pThis Pointer to the VMMDev instance data.
2165 */
2166static void vmmdevInitRam(VMMDevState *pThis)
2167{
2168 memset(pThis->pVMMDevRAMR3, 0, sizeof(VMMDevMemory));
2169 pThis->pVMMDevRAMR3->u32Size = sizeof(VMMDevMemory);
2170 pThis->pVMMDevRAMR3->u32Version = VMMDEV_MEMORY_VERSION;
2171}
2172
2173/**
2174 * Construct a device instance for a VM.
2175 *
2176 * @returns VBox status.
2177 * @param pDevIns The device instance data.
2178 * If the registration structure is needed, pDevIns->pDevReg points to it.
2179 * @param iInstance Instance number. Use this to figure out which registers and such to use.
2180 * The device number is also found in pDevIns->iInstance, but since it's
2181 * likely to be freqently used PDM passes it as parameter.
2182 * @param pCfgHandle Configuration node handle for the device. Use this to obtain the configuration
2183 * of the device instance. It's also found in pDevIns->pCfgHandle, but like
2184 * iInstance it's expected to be used a bit in this function.
2185 */
2186static DECLCALLBACK(int) vmmdevConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfgHandle)
2187{
2188 int rc;
2189 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState *);
2190
2191 Assert(iInstance == 0);
2192
2193 /*
2194 * Validate and read the configuration.
2195 */
2196 if (!CFGMR3AreValuesValid(pCfgHandle,
2197 "GetHostTimeDisabled\0"
2198 "BackdoorLogDisabled\0"
2199 "KeepCredentials\0"
2200 "HeapEnabled\0"
2201 ))
2202 return VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
2203
2204 rc = CFGMR3QueryBoolDef(pCfgHandle, "GetHostTimeDisabled", &pThis->fGetHostTimeDisabled, false);
2205 if (RT_FAILURE(rc))
2206 return PDMDEV_SET_ERROR(pDevIns, rc,
2207 N_("Configuration error: Failed querying \"GetHostTimeDisabled\" as a boolean"));
2208
2209 rc = CFGMR3QueryBoolDef(pCfgHandle, "BackdoorLogDisabled", &pThis->fBackdoorLogDisabled, false);
2210 if (RT_FAILURE(rc))
2211 return PDMDEV_SET_ERROR(pDevIns, rc,
2212 N_("Configuration error: Failed querying \"BackdoorLogDisabled\" as a boolean"));
2213
2214 rc = CFGMR3QueryBoolDef(pCfgHandle, "KeepCredentials", &pThis->fKeepCredentials, false);
2215 if (RT_FAILURE(rc))
2216 return PDMDEV_SET_ERROR(pDevIns, rc,
2217 N_("Configuration error: Failed querying \"KeepCredentials\" as a boolean"));
2218
2219 bool fHeapEnabled;
2220 rc = CFGMR3QueryBoolDef(pCfgHandle, "HeapEnabled", &fHeapEnabled, true);
2221 if (RT_FAILURE(rc))
2222 return PDMDEV_SET_ERROR(pDevIns, rc,
2223 N_("Configuration error: Failed querying \"HeapEnabled\" as a boolean"));
2224
2225 /*
2226 * Initialize data (most of it anyway).
2227 */
2228 /* Save PDM device instance data for future reference. */
2229 pThis->pDevIns = pDevIns;
2230
2231 /* PCI vendor, just a free bogus value */
2232 pThis->dev.config[0x00] = 0xee;
2233 pThis->dev.config[0x01] = 0x80;
2234 /* device ID */
2235 pThis->dev.config[0x02] = 0xfe;
2236 pThis->dev.config[0x03] = 0xca;
2237 /* class sub code (other type of system peripheral) */
2238 pThis->dev.config[0x0a] = 0x80;
2239 /* class base code (base system peripheral) */
2240 pThis->dev.config[0x0b] = 0x08;
2241 /* header type */
2242 pThis->dev.config[0x0e] = 0x00;
2243 /* interrupt on pin 0 */
2244 pThis->dev.config[0x3d] = 0x01;
2245
2246 /*
2247 * Interfaces
2248 */
2249 /* Base */
2250 pThis->Base.pfnQueryInterface = vmmdevPortQueryInterface;
2251
2252 /* VMMDev port */
2253 pThis->Port.pfnQueryAbsoluteMouse = vmmdevQueryAbsoluteMouse;
2254 pThis->Port.pfnSetAbsoluteMouse = vmmdevSetAbsoluteMouse;
2255 pThis->Port.pfnQueryMouseCapabilities = vmmdevQueryMouseCapabilities;
2256 pThis->Port.pfnSetMouseCapabilities = vmmdevSetMouseCapabilities;
2257 pThis->Port.pfnRequestDisplayChange = vmmdevRequestDisplayChange;
2258 pThis->Port.pfnSetCredentials = vmmdevSetCredentials;
2259 pThis->Port.pfnVBVAChange = vmmdevVBVAChange;
2260 pThis->Port.pfnRequestSeamlessChange = vmmdevRequestSeamlessChange;
2261 pThis->Port.pfnSetMemoryBalloon = vmmdevSetMemoryBalloon;
2262 pThis->Port.pfnSetStatisticsInterval = vmmdevSetStatisticsInterval;
2263 pThis->Port.pfnVRDPChange = vmmdevVRDPChange;
2264
2265 /* Shared folder LED */
2266 pThis->SharedFolders.Led.u32Magic = PDMLED_MAGIC;
2267 pThis->SharedFolders.ILeds.pfnQueryStatusLed = vmmdevQueryStatusLed;
2268
2269#ifdef VBOX_WITH_HGCM
2270 /* HGCM port */
2271 pThis->HGCMPort.pfnCompleted = hgcmCompleted;
2272#endif
2273
2274 /** @todo convert this into a config parameter like we do everywhere else! */
2275 pThis->cbGuestRAM = MMR3PhysGetRamSize(PDMDevHlpGetVM(pDevIns));
2276
2277 /*
2278 * Register the backdoor logging port
2279 */
2280 rc = PDMDevHlpIOPortRegister(pDevIns, RTLOG_DEBUG_PORT, 1, NULL, vmmdevBackdoorLog, NULL, NULL, NULL, "VMMDev backdoor logging");
2281 AssertRCReturn(rc, rc);
2282
2283#ifdef TIMESYNC_BACKDOOR
2284 /*
2285 * Alternative timesync source (temporary!)
2286 */
2287 rc = PDMDevHlpIOPortRegister(pDevIns, 0x505, 1, NULL, vmmdevTimesyncBackdoorWrite, vmmdevTimesyncBackdoorRead, NULL, NULL, "VMMDev timesync backdoor");
2288 AssertRCReturn(rc, rc);
2289#endif
2290
2291 /*
2292 * Allocate and initialize the MMIO2 memory.
2293 */
2294 rc = PDMDevHlpMMIO2Register(pDevIns, 1 /*iRegion*/, VMMDEV_RAM_SIZE, 0 /*fFlags*/, (void **)&pThis->pVMMDevRAMR3, "VMMDev");
2295 if (RT_FAILURE(rc))
2296 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2297 N_("Failed to allocate %u bytes of memory for the VMM device"), VMMDEV_RAM_SIZE);
2298 vmmdevInitRam(pThis);
2299
2300 if (fHeapEnabled)
2301 {
2302 rc = PDMDevHlpMMIO2Register(pDevIns, 2 /*iRegion*/, VMMDEV_HEAP_SIZE, 0 /*fFlags*/, (void **)&pThis->pVMMDevHeapR3, "VMMDev Heap");
2303 if (RT_FAILURE(rc))
2304 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2305 N_("Failed to allocate %u bytes of memory for the VMM device heap"), PAGE_SIZE);
2306 }
2307
2308 /*
2309 * Register the PCI device.
2310 */
2311 rc = PDMDevHlpPCIRegister(pDevIns, &pThis->dev);
2312 if (RT_FAILURE(rc))
2313 return rc;
2314 if (pThis->dev.devfn == 32 || iInstance != 0)
2315 Log(("!!WARNING!!: pThis->dev.devfn=%d (ignore if testcase or no started by Main)\n", pThis->dev.devfn));
2316 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, 0x20, PCI_ADDRESS_SPACE_IO, vmmdevIOPortRegionMap);
2317 if (RT_FAILURE(rc))
2318 return rc;
2319 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, VMMDEV_RAM_SIZE, PCI_ADDRESS_SPACE_MEM, vmmdevIORAMRegionMap);
2320 if (RT_FAILURE(rc))
2321 return rc;
2322 if (fHeapEnabled)
2323 {
2324 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 2, VMMDEV_HEAP_SIZE, PCI_ADDRESS_SPACE_MEM_PREFETCH, vmmdevIORAMRegionMap);
2325 if (RT_FAILURE(rc))
2326 return rc;
2327 }
2328
2329 /*
2330 * Get the corresponding connector interface
2331 */
2332 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThis->Base, &pThis->pDrvBase, "VMM Driver Port");
2333 if (RT_SUCCESS(rc))
2334 {
2335 pThis->pDrv = (PPDMIVMMDEVCONNECTOR)pThis->pDrvBase->pfnQueryInterface(pThis->pDrvBase, PDMINTERFACE_VMMDEV_CONNECTOR);
2336 if (!pThis->pDrv)
2337 AssertMsgFailedReturn(("LUN #0 doesn't have a VMMDev connector interface!\n"), VERR_PDM_MISSING_INTERFACE);
2338#ifdef VBOX_WITH_HGCM
2339 pThis->pHGCMDrv = (PPDMIHGCMCONNECTOR)pThis->pDrvBase->pfnQueryInterface(pThis->pDrvBase, PDMINTERFACE_HGCM_CONNECTOR);
2340 if (!pThis->pHGCMDrv)
2341 {
2342 Log(("LUN #0 doesn't have a HGCM connector interface, HGCM is not supported. rc=%Rrc\n", rc));
2343 /* this is not actually an error, just means that there is no support for HGCM */
2344 }
2345#endif
2346 }
2347 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
2348 {
2349 Log(("%s/%d: warning: no driver attached to LUN #0!\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
2350 rc = VINF_SUCCESS;
2351 }
2352 else
2353 AssertMsgFailedReturn(("Failed to attach LUN #0! rc=%Rrc\n", rc), rc);
2354
2355 /*
2356 * Attach status driver for shared folders (optional).
2357 */
2358 PPDMIBASE pBase;
2359 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->Base, &pBase, "Status Port");
2360 if (RT_SUCCESS(rc))
2361 pThis->SharedFolders.pLedsConnector = (PPDMILEDCONNECTORS)
2362 pBase->pfnQueryInterface(pBase, PDMINTERFACE_LED_CONNECTORS);
2363 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
2364 {
2365 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
2366 return rc;
2367 }
2368
2369 /*
2370 * Register saved state and init the HGCM CmdList critsect.
2371 */
2372 rc = PDMDevHlpSSMRegister(pDevIns, "VMMDev", iInstance, VMMDEV_SSM_VERSION, sizeof(*pThis),
2373 NULL, vmmdevSaveState, NULL,
2374 NULL, vmmdevLoadState, vmmdevLoadStateDone);
2375 AssertRCReturn(rc, rc);
2376
2377#ifdef VBOX_WITH_HGCM
2378 pThis->pHGCMCmdList = NULL;
2379 rc = RTCritSectInit(&pThis->critsectHGCMCmdList);
2380 AssertRCReturn(rc, rc);
2381 pThis->u32HGCMEnabled = 0;
2382#endif /* VBOX_WITH_HGCM */
2383
2384 return rc;
2385}
2386
2387/**
2388 * Reset notification.
2389 *
2390 * @returns VBox status.
2391 * @param pDrvIns The driver instance data.
2392 */
2393static DECLCALLBACK(void) vmmdevReset(PPDMDEVINS pDevIns)
2394{
2395 VMMDevState *pThis = PDMINS_2_DATA(pDevIns, VMMDevState*);
2396
2397 /*
2398 * Reset the mouse integration feature bit
2399 */
2400 if (pThis->mouseCapabilities & (VMMDEV_MOUSEGUESTWANTSABS|VMMDEV_MOUSEGUESTNEEDSHOSTCUR))
2401 {
2402 pThis->mouseCapabilities &= ~VMMDEV_MOUSEGUESTWANTSABS;
2403 /* notify the connector */
2404 Log(("vmmdevReset: capabilities changed (%x), informing connector\n", pThis->mouseCapabilities));
2405 pThis->pDrv->pfnUpdateMouseCapabilities(pThis->pDrv, pThis->mouseCapabilities);
2406 }
2407
2408 pThis->hypervisorSize = 0;
2409
2410 pThis->u32HostEventFlags = 0;
2411
2412 /* re-initialize the VMMDev memory */
2413 if (pThis->pVMMDevRAMR3)
2414 vmmdevInitRam(pThis);
2415
2416 /* credentials have to go away (by default) */
2417 if (!pThis->fKeepCredentials)
2418 {
2419 memset(pThis->credentialsLogon.szUserName, '\0', VMMDEV_CREDENTIALS_STRLEN);
2420 memset(pThis->credentialsLogon.szPassword, '\0', VMMDEV_CREDENTIALS_STRLEN);
2421 memset(pThis->credentialsLogon.szDomain, '\0', VMMDEV_CREDENTIALS_STRLEN);
2422 }
2423 memset(pThis->credentialsJudge.szUserName, '\0', VMMDEV_CREDENTIALS_STRLEN);
2424 memset(pThis->credentialsJudge.szPassword, '\0', VMMDEV_CREDENTIALS_STRLEN);
2425 memset(pThis->credentialsJudge.szDomain, '\0', VMMDEV_CREDENTIALS_STRLEN);
2426
2427 /* Reset means that additions will report again. */
2428 const bool fVersionChanged = pThis->fu32AdditionsOk
2429 || pThis->guestInfo.additionsVersion
2430 || pThis->guestInfo.osType != VBOXOSTYPE_Unknown;
2431 if (fVersionChanged)
2432 Log(("vmmdevReset: fu32AdditionsOk=%d additionsVersion=%x osType=%#x\n",
2433 pThis->fu32AdditionsOk, pThis->guestInfo.additionsVersion, pThis->guestInfo.osType));
2434 pThis->fu32AdditionsOk = false;
2435 memset (&pThis->guestInfo, 0, sizeof (pThis->guestInfo));
2436
2437 /* clear pending display change request. */
2438 memset (&pThis->lastReadDisplayChangeRequest, 0, sizeof (pThis->lastReadDisplayChangeRequest));
2439
2440 /* disable seamless mode */
2441 pThis->fLastSeamlessEnabled = false;
2442
2443 /* disabled memory ballooning */
2444 pThis->u32LastMemoryBalloonSize = 0;
2445
2446 /* disabled statistics updating */
2447 pThis->u32LastStatIntervalSize = 0;
2448
2449 /* Clear the "HGCM event enabled" flag so the event can be automatically reenabled. */
2450 pThis->u32HGCMEnabled = 0;
2451
2452 /*
2453 * Clear the event variables.
2454 *
2455 * Note: The pThis->u32HostEventFlags is not cleared.
2456 * It is designed that way so host events do not
2457 * depend on guest resets.
2458 */
2459 pThis->u32GuestFilterMask = 0;
2460 pThis->u32NewGuestFilterMask = 0;
2461 pThis->fNewGuestFilterMask = 0;
2462
2463 /* This is the default, as Windows and OS/2 guests take this for granted. (Actually, neither does...) */
2464 /** @todo change this when we next bump the interface version */
2465 const bool fCapsChanged = pThis->guestCaps != VMMDEV_GUEST_SUPPORTS_GRAPHICS;
2466 if (fCapsChanged)
2467 Log(("vmmdevReset: fCapsChanged=%#x -> %#x\n", pThis->guestCaps, VMMDEV_GUEST_SUPPORTS_GRAPHICS));
2468 pThis->guestCaps = VMMDEV_GUEST_SUPPORTS_GRAPHICS; /** @todo r=bird: why? I cannot see this being done at construction?*/
2469
2470 /*
2471 * Call the update functions as required.
2472 */
2473 if (fVersionChanged)
2474 pThis->pDrv->pfnUpdateGuestVersion(pThis->pDrv, &pThis->guestInfo);
2475 if (fCapsChanged)
2476 pThis->pDrv->pfnUpdateGuestCapabilities(pThis->pDrv, pThis->guestCaps);
2477}
2478
2479
2480/**
2481 * The device registration structure.
2482 */
2483extern "C" const PDMDEVREG g_DeviceVMMDev =
2484{
2485 /* u32Version */
2486 PDM_DEVREG_VERSION,
2487 /* szDeviceName */
2488 "VMMDev",
2489 /* szRCMod */
2490 "",
2491 /* szR0Mod */
2492 "",
2493 /* pszDescription */
2494 "VirtualBox VMM Device\n",
2495 /* fFlags */
2496 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_32,
2497 /* fClass */
2498 PDM_DEVREG_CLASS_VMM_DEV,
2499 /* cMaxInstances */
2500 1,
2501 /* cbInstance */
2502 sizeof(VMMDevState),
2503 /* pfnConstruct */
2504 vmmdevConstruct,
2505 /* pfnDestruct */
2506 NULL,
2507 /* pfnRelocate */
2508 NULL,
2509 /* pfnIOCtl */
2510 NULL,
2511 /* pfnPowerOn */
2512 NULL,
2513 /* pfnReset */
2514 vmmdevReset,
2515 /* pfnSuspend */
2516 NULL,
2517 /* pfnResume */
2518 NULL,
2519 /* pfnAttach */
2520 NULL,
2521 /* pfnDetach */
2522 NULL,
2523 /* pfnQueryInterface. */
2524 NULL,
2525 /* pfnInitComplete */
2526 NULL,
2527 /* pfnPowerOff */
2528 NULL,
2529 /* pfnSoftReset */
2530 NULL,
2531 /* u32VersionEnd */
2532 PDM_DEVREG_VERSION
2533};
2534#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
2535
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