VirtualBox

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

Last change on this file since 7449 was 7436, checked in by vboxsync, 17 years ago

Devices/DevVMM: spacing

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

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette