VirtualBox

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

Last change on this file since 5085 was 5040, checked in by vboxsync, 17 years ago

GC phys/virt to HC virt functions are no longer accessible in our PDM interface.
Rewrote disassembly functions to use the mapping functions.

Code that runs in EMT (like CSAM/PATM) can still use the old conversion functions. Easier to use
and (far) less overhead.

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