VirtualBox

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

Last change on this file since 27452 was 25985, checked in by vboxsync, 15 years ago

pdmifs.h: the final batch of refactored interface ID code.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 103.2 KB
Line 
1/* $Id: VMMDevHGCM.cpp 25985 2010-01-23 00:51:04Z vboxsync $ */
2/** @file
3 * VMMDev - HGCM - Host-Guest Communication Manager Device.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23#define LOG_GROUP LOG_GROUP_DEV_VMM
24#include <iprt/alloc.h>
25#include <iprt/asm.h>
26#include <iprt/assert.h>
27#include <iprt/param.h>
28#include <iprt/string.h>
29
30#include <VBox/err.h>
31#include <VBox/hgcmsvc.h>
32
33#include <VBox/log.h>
34
35#include "VMMDevHGCM.h"
36
37#ifdef VBOX_WITH_DTRACE
38# include "VBoxDD-dtrace.h"
39#else
40# define VBOXDD_HGCMCALL_ENTER(a,b,c,d) do { } while (0)
41# define VBOXDD_HGCMCALL_COMPLETED_REQ(a,b) do { } while (0)
42# define VBOXDD_HGCMCALL_COMPLETED_EMT(a,b) do { } while (0)
43# define VBOXDD_HGCMCALL_COMPLETED_DONE(a,b,c,d) do { } while (0)
44#endif
45
46typedef enum _VBOXHGCMCMDTYPE
47{
48 VBOXHGCMCMDTYPE_LOADSTATE = 0,
49 VBOXHGCMCMDTYPE_CONNECT,
50 VBOXHGCMCMDTYPE_DISCONNECT,
51 VBOXHGCMCMDTYPE_CALL,
52 VBOXHGCMCMDTYPE_SizeHack = 0x7fffffff
53} VBOXHGCMCMDTYPE;
54
55/* Information about a linear ptr parameter. */
56typedef struct _VBOXHGCMLINPTR
57{
58 /* Index of the parameter. */
59 uint32_t iParm;
60
61 /* Offset in the first physical page of the region. */
62 uint32_t offFirstPage;
63
64 /* How many pages. */
65 uint32_t cPages;
66
67 /* Pointer to array of the GC physical addresses for these pages.
68 * It is assumed that the physical address of the locked resident
69 * guest page does not change.
70 */
71 RTGCPHYS *paPages;
72
73} VBOXHGCMLINPTR;
74
75struct VBOXHGCMCMD
76{
77 /* Active commands, list is protected by critsectHGCMCmdList. */
78 struct VBOXHGCMCMD *pNext;
79 struct VBOXHGCMCMD *pPrev;
80
81 /* The type of the command. */
82 VBOXHGCMCMDTYPE enmCmdType;
83
84 /* Whether the command was cancelled by the guest. */
85 bool fCancelled;
86
87 /* GC physical address of the guest request. */
88 RTGCPHYS GCPhys;
89
90 /* Request packet size */
91 uint32_t cbSize;
92
93 /* Pointer to converted host parameters in case of a Call request.
94 * Parameters follow this structure in the same memory block.
95 */
96 VBOXHGCMSVCPARM *paHostParms;
97
98 /* Linear pointer parameters information. */
99 int cLinPtrs;
100
101 /* How many pages for all linptrs of this command.
102 * Only valid if cLinPtrs > 0. This field simplifies loading of saved state.
103 */
104 int cLinPtrPages;
105
106 /* Pointer to descriptions of linear pointers. */
107 VBOXHGCMLINPTR *paLinPtrs;
108};
109
110static int vmmdevHGCMCmdListLock (VMMDevState *pVMMDevState)
111{
112 int rc = RTCritSectEnter (&pVMMDevState->critsectHGCMCmdList);
113 AssertRC (rc);
114 return rc;
115}
116
117static void vmmdevHGCMCmdListUnlock (VMMDevState *pVMMDevState)
118{
119 int rc = RTCritSectLeave (&pVMMDevState->critsectHGCMCmdList);
120 AssertRC (rc);
121}
122
123static int vmmdevHGCMAddCommand (VMMDevState *pVMMDevState, PVBOXHGCMCMD pCmd, RTGCPHYS GCPhys, uint32_t cbSize, VBOXHGCMCMDTYPE enmCmdType)
124{
125 /* PPDMDEVINS pDevIns = pVMMDevState->pDevIns; */
126
127 int rc = vmmdevHGCMCmdListLock (pVMMDevState);
128
129 if (RT_SUCCESS (rc))
130 {
131 LogFlowFunc(("%p type %d\n", pCmd, enmCmdType));
132
133 /* Insert at the head of the list. The vmmdevHGCMLoadStateDone depends on this. */
134 pCmd->pNext = pVMMDevState->pHGCMCmdList;
135 pCmd->pPrev = NULL;
136
137 if (pVMMDevState->pHGCMCmdList)
138 {
139 pVMMDevState->pHGCMCmdList->pPrev = pCmd;
140 }
141
142 pVMMDevState->pHGCMCmdList = pCmd;
143
144 if (enmCmdType != VBOXHGCMCMDTYPE_LOADSTATE)
145 {
146 /* Loaded commands already have the right type. */
147 pCmd->enmCmdType = enmCmdType;
148 }
149 pCmd->GCPhys = GCPhys;
150 pCmd->cbSize = cbSize;
151
152 /* Automatically enable HGCM events, if there are HGCM commands. */
153 if ( enmCmdType == VBOXHGCMCMDTYPE_CONNECT
154 || enmCmdType == VBOXHGCMCMDTYPE_DISCONNECT
155 || enmCmdType == VBOXHGCMCMDTYPE_CALL)
156 {
157 Log(("vmmdevHGCMAddCommand: u32HGCMEnabled = %d\n", pVMMDevState->u32HGCMEnabled));
158 if (ASMAtomicCmpXchgU32(&pVMMDevState->u32HGCMEnabled, 1, 0))
159 {
160 VMMDevCtlSetGuestFilterMask (pVMMDevState, VMMDEV_EVENT_HGCM, 0);
161 }
162 }
163
164 vmmdevHGCMCmdListUnlock (pVMMDevState);
165 }
166
167 return rc;
168}
169
170static int vmmdevHGCMRemoveCommand (VMMDevState *pVMMDevState, PVBOXHGCMCMD pCmd)
171{
172 /* PPDMDEVINS pDevIns = pVMMDevState->pDevIns; */
173
174 int rc = vmmdevHGCMCmdListLock (pVMMDevState);
175
176 if (RT_SUCCESS (rc))
177 {
178 LogFlowFunc(("%p\n", pCmd));
179
180 if (pCmd->pNext)
181 {
182 pCmd->pNext->pPrev = pCmd->pPrev;
183 }
184 else
185 {
186 /* Tail, do nothing. */
187 }
188
189 if (pCmd->pPrev)
190 {
191 pCmd->pPrev->pNext = pCmd->pNext;
192 }
193 else
194 {
195 pVMMDevState->pHGCMCmdList = pCmd->pNext;
196 }
197
198 vmmdevHGCMCmdListUnlock (pVMMDevState);
199 }
200
201 return rc;
202}
203
204
205/**
206 * Find a HGCM command by its physical address.
207 *
208 * The caller is responsible for taking the command list lock before calling
209 * this function.
210 *
211 * @returns Pointer to the command on success, NULL otherwise.
212 * @param pThis The VMMDev instance data.
213 * @param GCPhys The physical address of the command we're looking
214 * for.
215 */
216DECLINLINE(PVBOXHGCMCMD) vmmdevHGCMFindCommandLocked (VMMDevState *pThis, RTGCPHYS GCPhys)
217{
218 for (PVBOXHGCMCMD pCmd = pThis->pHGCMCmdList;
219 pCmd;
220 pCmd = pCmd->pNext)
221 {
222 if (pCmd->GCPhys == GCPhys)
223 return pCmd;
224 }
225 return NULL;
226}
227
228static int vmmdevHGCMSaveLinPtr (PPDMDEVINS pDevIns,
229 uint32_t iParm,
230 RTGCPTR GCPtr,
231 uint32_t u32Size,
232 uint32_t iLinPtr,
233 VBOXHGCMLINPTR *paLinPtrs,
234 RTGCPHYS **ppPages)
235{
236 int rc = VINF_SUCCESS;
237
238 AssertRelease (u32Size > 0);
239
240 VBOXHGCMLINPTR *pLinPtr = &paLinPtrs[iLinPtr];
241
242 /* Take the offset into the current page also into account! */
243 u32Size += GCPtr & PAGE_OFFSET_MASK;
244
245 uint32_t cPages = (u32Size + PAGE_SIZE - 1) / PAGE_SIZE;
246
247 Log(("vmmdevHGCMSaveLinPtr: parm %d: %RGv %d = %d pages\n", iParm, GCPtr, u32Size, cPages));
248
249 pLinPtr->iParm = iParm;
250 pLinPtr->offFirstPage = GCPtr & PAGE_OFFSET_MASK;
251 pLinPtr->cPages = cPages;
252 pLinPtr->paPages = *ppPages;
253
254 *ppPages += cPages;
255
256 uint32_t iPage = 0;
257
258 GCPtr &= PAGE_BASE_GC_MASK;
259
260 /* Gonvert the guest linear pointers of pages to HC addresses. */
261 while (iPage < cPages)
262 {
263 /* convert */
264 RTGCPHYS GCPhys;
265
266 rc = PDMDevHlpPhysGCPtr2GCPhys(pDevIns, GCPtr, &GCPhys);
267
268 Log(("vmmdevHGCMSaveLinPtr: Page %d: %RGv -> %RGp. %Rrc\n", iPage, GCPtr, GCPhys, rc));
269
270 if (RT_FAILURE (rc))
271 {
272 break;
273 }
274
275 /* store */
276 pLinPtr->paPages[iPage++] = GCPhys;
277
278 /* next */
279 GCPtr += PAGE_SIZE;
280 }
281
282 AssertRelease (iPage == cPages);
283
284 return rc;
285}
286
287static int vmmdevHGCMWriteLinPtr (PPDMDEVINS pDevIns,
288 uint32_t iParm,
289 void *pvHost,
290 uint32_t u32Size,
291 uint32_t iLinPtr,
292 VBOXHGCMLINPTR *paLinPtrs)
293{
294 int rc = VINF_SUCCESS;
295
296 VBOXHGCMLINPTR *pLinPtr = &paLinPtrs[iLinPtr];
297
298 AssertRelease (u32Size > 0 && iParm == (uint32_t)pLinPtr->iParm);
299
300 RTGCPHYS GCPhysDst = pLinPtr->paPages[0] + pLinPtr->offFirstPage;
301 uint8_t *pu8Src = (uint8_t *)pvHost;
302
303 Log(("vmmdevHGCMWriteLinPtr: parm %d: size %d, cPages = %d\n", iParm, u32Size, pLinPtr->cPages));
304
305 uint32_t iPage = 0;
306
307 while (iPage < pLinPtr->cPages)
308 {
309 /* copy */
310 uint32_t cbWrite = iPage == 0?
311 PAGE_SIZE - pLinPtr->offFirstPage:
312 PAGE_SIZE;
313
314 Log(("vmmdevHGCMWriteLinPtr: page %d: dst %RGp, src %p, cbWrite %d\n", iPage, GCPhysDst, pu8Src, cbWrite));
315
316 iPage++;
317
318 if (cbWrite >= u32Size)
319 {
320 PDMDevHlpPhysWrite(pDevIns, GCPhysDst, pu8Src, u32Size);
321 u32Size = 0;
322 break;
323 }
324
325 PDMDevHlpPhysWrite(pDevIns, GCPhysDst, pu8Src, cbWrite);
326
327 /* next */
328 u32Size -= cbWrite;
329 pu8Src += cbWrite;
330
331 GCPhysDst = pLinPtr->paPages[iPage];
332 }
333
334 AssertRelease (iPage == pLinPtr->cPages);
335 Assert(u32Size == 0);
336
337 return rc;
338}
339
340DECLINLINE(bool) vmmdevHGCMPageListIsContiguous(const HGCMPageListInfo *pPgLst)
341{
342 if (pPgLst->cPages == 1)
343 return true;
344 RTGCPHYS64 Phys = pPgLst->aPages[0] + PAGE_SIZE;
345 if (Phys != pPgLst->aPages[1])
346 return false;
347 if (pPgLst->cPages > 2)
348 {
349 uint32_t iPage = 2;
350 do
351 {
352 Phys += PAGE_SIZE;
353 if (Phys != pPgLst->aPages[iPage])
354 return false;
355 iPage++;
356 } while (iPage < pPgLst->cPages);
357 }
358 return true;
359}
360
361static int vmmdevHGCMPageListRead(PPDMDEVINSR3 pDevIns, void *pvDst, uint32_t cbDst, const HGCMPageListInfo *pPageListInfo)
362{
363 /*
364 * Try detect contiguous buffers.
365 */
366 /** @todo We need a flag for indicating this. */
367 if (vmmdevHGCMPageListIsContiguous(pPageListInfo))
368 return PDMDevHlpPhysRead(pDevIns, pPageListInfo->aPages[0] | pPageListInfo->offFirstPage, pvDst, cbDst);
369
370 /*
371 * Page by page fallback
372 */
373 int rc = VINF_SUCCESS;
374
375 uint8_t *pu8Dst = (uint8_t *)pvDst;
376 uint32_t offPage = pPageListInfo->offFirstPage;
377 size_t cbRemaining = (size_t)cbDst;
378
379 uint32_t iPage;
380
381 for (iPage = 0; iPage < pPageListInfo->cPages; iPage++)
382 {
383 if (cbRemaining == 0)
384 {
385 break;
386 }
387
388 size_t cbChunk = PAGE_SIZE - offPage;
389
390 if (cbChunk > cbRemaining)
391 {
392 cbChunk = cbRemaining;
393 }
394
395 rc = PDMDevHlpPhysRead(pDevIns,
396 pPageListInfo->aPages[iPage] + offPage,
397 pu8Dst, cbChunk);
398
399 AssertRCBreak(rc);
400
401 offPage = 0; /* A next page is read from 0 offset. */
402 cbRemaining -= cbChunk;
403 pu8Dst += cbChunk;
404 }
405
406 return rc;
407}
408
409static int vmmdevHGCMPageListWrite(PPDMDEVINSR3 pDevIns, const HGCMPageListInfo *pPageListInfo, const void *pvSrc, uint32_t cbSrc)
410{
411 int rc = VINF_SUCCESS;
412
413 uint8_t *pu8Src = (uint8_t *)pvSrc;
414 uint32_t offPage = pPageListInfo->offFirstPage;
415 size_t cbRemaining = (size_t)cbSrc;
416
417 uint32_t iPage;
418 for (iPage = 0; iPage < pPageListInfo->cPages; iPage++)
419 {
420 if (cbRemaining == 0)
421 {
422 break;
423 }
424
425 size_t cbChunk = PAGE_SIZE - offPage;
426
427 if (cbChunk > cbRemaining)
428 {
429 cbChunk = cbRemaining;
430 }
431
432 rc = PDMDevHlpPhysWrite(pDevIns,
433 pPageListInfo->aPages[iPage] + offPage,
434 pu8Src, cbChunk);
435
436 AssertRCBreak(rc);
437
438 offPage = 0; /* A next page is read from 0 offset. */
439 cbRemaining -= cbChunk;
440 pu8Src += cbChunk;
441 }
442
443 return rc;
444}
445
446static void vmmdevRestoreSavedCommand(VBOXHGCMCMD *pCmd, VBOXHGCMCMD *pSavedCmd)
447{
448 /* Copy relevant saved command information to the new allocated structure. */
449 pCmd->enmCmdType = pSavedCmd->enmCmdType;
450 pCmd->fCancelled = pSavedCmd->fCancelled;
451 pCmd->GCPhys = pSavedCmd->GCPhys;
452 pCmd->cbSize = pSavedCmd->cbSize;
453 pCmd->cLinPtrs = pSavedCmd->cLinPtrs;
454 pCmd->cLinPtrPages = pSavedCmd->cLinPtrPages;
455 pCmd->paLinPtrs = pSavedCmd->paLinPtrs;
456
457 /* The new allocated command owns the 'paLinPtrs' pointer. */
458 pSavedCmd->paLinPtrs = NULL;
459}
460
461int vmmdevHGCMConnect (VMMDevState *pVMMDevState, VMMDevHGCMConnect *pHGCMConnect, RTGCPHYS GCPhys)
462{
463 int rc = VINF_SUCCESS;
464
465 uint32_t cbCmdSize = sizeof (struct VBOXHGCMCMD) + pHGCMConnect->header.header.size;
466
467 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (cbCmdSize);
468
469 if (pCmd)
470 {
471 VMMDevHGCMConnect *pHGCMConnectCopy = (VMMDevHGCMConnect *)(pCmd+1);
472
473 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, pHGCMConnect->header.header.size, VBOXHGCMCMDTYPE_CONNECT);
474
475 memcpy(pHGCMConnectCopy, pHGCMConnect, pHGCMConnect->header.header.size);
476
477 pCmd->paHostParms = NULL;
478 pCmd->cLinPtrs = 0;
479 pCmd->paLinPtrs = NULL;
480
481 /* Only allow the guest to use existing services! */
482 Assert(pHGCMConnectCopy->loc.type == VMMDevHGCMLoc_LocalHost_Existing);
483 pHGCMConnectCopy->loc.type = VMMDevHGCMLoc_LocalHost_Existing;
484
485 rc = pVMMDevState->pHGCMDrv->pfnConnect (pVMMDevState->pHGCMDrv, pCmd, &pHGCMConnectCopy->loc, &pHGCMConnectCopy->u32ClientID);
486 }
487 else
488 {
489 rc = VERR_NO_MEMORY;
490 }
491
492 return rc;
493}
494
495static int vmmdevHGCMConnectSaved (VMMDevState *pVMMDevState, VMMDevHGCMConnect *pHGCMConnect, RTGCPHYS GCPhys, bool *pfHGCMCalled, VBOXHGCMCMD *pSavedCmd, VBOXHGCMCMD **ppCmd)
496{
497 int rc = VINF_SUCCESS;
498
499 /* Allocate buffer for the new command. */
500 uint32_t cbCmdSize = sizeof (struct VBOXHGCMCMD) + pHGCMConnect->header.header.size;
501
502 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (cbCmdSize);
503
504 if (pCmd)
505 {
506 vmmdevRestoreSavedCommand(pCmd, pSavedCmd);
507 *ppCmd = pCmd;
508
509 VMMDevHGCMConnect *pHGCMConnectCopy = (VMMDevHGCMConnect *)(pCmd+1);
510
511 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, pHGCMConnect->header.header.size, VBOXHGCMCMDTYPE_CONNECT);
512
513 memcpy(pHGCMConnectCopy, pHGCMConnect, pHGCMConnect->header.header.size);
514
515 /* Only allow the guest to use existing services! */
516 Assert(pHGCMConnectCopy->loc.type == VMMDevHGCMLoc_LocalHost_Existing);
517 pHGCMConnectCopy->loc.type = VMMDevHGCMLoc_LocalHost_Existing;
518
519 rc = pVMMDevState->pHGCMDrv->pfnConnect (pVMMDevState->pHGCMDrv, pCmd, &pHGCMConnectCopy->loc, &pHGCMConnectCopy->u32ClientID);
520
521 if (RT_SUCCESS (rc))
522 {
523 *pfHGCMCalled = true;
524 }
525 }
526 else
527 {
528 rc = VERR_NO_MEMORY;
529 }
530
531 return rc;
532}
533
534int vmmdevHGCMDisconnect (VMMDevState *pVMMDevState, VMMDevHGCMDisconnect *pHGCMDisconnect, RTGCPHYS GCPhys)
535{
536 int rc = VINF_SUCCESS;
537
538 uint32_t cbCmdSize = sizeof (struct VBOXHGCMCMD);
539
540 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (cbCmdSize);
541
542 if (pCmd)
543 {
544 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, pHGCMDisconnect->header.header.size, VBOXHGCMCMDTYPE_DISCONNECT);
545
546 pCmd->paHostParms = NULL;
547 pCmd->cLinPtrs = 0;
548 pCmd->paLinPtrs = NULL;
549
550 rc = pVMMDevState->pHGCMDrv->pfnDisconnect (pVMMDevState->pHGCMDrv, pCmd, pHGCMDisconnect->u32ClientID);
551 }
552 else
553 {
554 rc = VERR_NO_MEMORY;
555 }
556
557 return rc;
558}
559
560static int vmmdevHGCMDisconnectSaved (VMMDevState *pVMMDevState, VMMDevHGCMDisconnect *pHGCMDisconnect, RTGCPHYS GCPhys, bool *pfHGCMCalled, VBOXHGCMCMD *pSavedCmd, VBOXHGCMCMD **ppCmd)
561{
562 int rc = VINF_SUCCESS;
563
564 uint32_t cbCmdSize = sizeof (struct VBOXHGCMCMD);
565
566 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (cbCmdSize);
567
568 if (pCmd)
569 {
570 vmmdevRestoreSavedCommand(pCmd, pSavedCmd);
571 *ppCmd = pCmd;
572
573 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, pHGCMDisconnect->header.header.size, VBOXHGCMCMDTYPE_DISCONNECT);
574
575 pCmd->paHostParms = NULL;
576 pCmd->cLinPtrs = 0;
577 pCmd->paLinPtrs = NULL;
578
579 rc = pVMMDevState->pHGCMDrv->pfnDisconnect (pVMMDevState->pHGCMDrv, pCmd, pHGCMDisconnect->u32ClientID);
580
581 if (RT_SUCCESS (rc))
582 {
583 *pfHGCMCalled = true;
584 }
585 }
586 else
587 {
588 rc = VERR_NO_MEMORY;
589 }
590
591 return rc;
592}
593
594int vmmdevHGCMCall (VMMDevState *pVMMDevState, VMMDevHGCMCall *pHGCMCall, uint32_t cbHGCMCall, RTGCPHYS GCPhys, bool f64Bits)
595{
596 int rc = VINF_SUCCESS;
597
598 Log(("vmmdevHGCMCall: client id = %d, function = %d, %s bit\n", pHGCMCall->u32ClientID, pHGCMCall->u32Function, f64Bits? "64": "32"));
599
600 /* Compute size and allocate memory block to hold:
601 * struct VBOXHGCMCMD
602 * VBOXHGCMSVCPARM[cParms]
603 * memory buffers for pointer parameters.
604 */
605
606 uint32_t cParms = pHGCMCall->cParms;
607
608 Log(("vmmdevHGCMCall: cParms = %d\n", cParms));
609
610 /*
611 * Compute size of required memory buffer.
612 */
613
614 uint32_t cbCmdSize = sizeof (struct VBOXHGCMCMD) + cParms * sizeof (VBOXHGCMSVCPARM);
615
616 uint32_t i;
617
618 uint32_t cLinPtrs = 0;
619 uint32_t cLinPtrPages = 0;
620
621 if (f64Bits)
622 {
623#ifdef VBOX_WITH_64_BITS_GUESTS
624 HGCMFunctionParameter64 *pGuestParm = VMMDEV_HGCM_CALL_PARMS64(pHGCMCall);
625#else
626 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
627 AssertFailed (); /* This code should not be called in this case */
628#endif /* VBOX_WITH_64_BITS_GUESTS */
629
630 /* Look for pointer parameters, which require a host buffer. */
631 for (i = 0; i < cParms && RT_SUCCESS(rc); i++, pGuestParm++)
632 {
633 switch (pGuestParm->type)
634 {
635 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
636 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
637 case VMMDevHGCMParmType_LinAddr: /* In & Out */
638 {
639 if (pGuestParm->u.Pointer.size > 0)
640 {
641 /* Only pointers with some actual data are counted. */
642 cbCmdSize += pGuestParm->u.Pointer.size;
643
644 cLinPtrs++;
645 /* Take the offset into the current page also into account! */
646 cLinPtrPages += ((pGuestParm->u.Pointer.u.linearAddr & PAGE_OFFSET_MASK)
647 + pGuestParm->u.Pointer.size + PAGE_SIZE - 1) / PAGE_SIZE;
648 }
649
650 Log(("vmmdevHGCMCall: linptr size = %d\n", pGuestParm->u.Pointer.size));
651 } break;
652
653 case VMMDevHGCMParmType_PageList:
654 {
655 cbCmdSize += pGuestParm->u.PageList.size;
656 Log(("vmmdevHGCMCall: pagelist size = %d\n", pGuestParm->u.PageList.size));
657 } break;
658
659 case VMMDevHGCMParmType_32bit:
660 case VMMDevHGCMParmType_64bit:
661 {
662 } break;
663
664 default:
665 case VMMDevHGCMParmType_PhysAddr:
666 {
667 AssertMsgFailed(("vmmdevHGCMCall: invalid parameter type %x\n", pGuestParm->type));
668 rc = VERR_INVALID_PARAMETER;
669 break;
670 }
671 }
672 }
673 }
674 else
675 {
676#ifdef VBOX_WITH_64_BITS_GUESTS
677 HGCMFunctionParameter32 *pGuestParm = VMMDEV_HGCM_CALL_PARMS32(pHGCMCall);
678#else
679 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
680#endif /* VBOX_WITH_64_BITS_GUESTS */
681
682 /* Look for pointer parameters, which require a host buffer. */
683 for (i = 0; i < cParms && RT_SUCCESS(rc); i++, pGuestParm++)
684 {
685 switch (pGuestParm->type)
686 {
687 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
688 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
689 case VMMDevHGCMParmType_LinAddr: /* In & Out */
690 {
691 if (pGuestParm->u.Pointer.size > 0)
692 {
693 /* Only pointers with some actual data are counted. */
694 cbCmdSize += pGuestParm->u.Pointer.size;
695
696 cLinPtrs++;
697 /* Take the offset into the current page also into account! */
698 cLinPtrPages += ((pGuestParm->u.Pointer.u.linearAddr & PAGE_OFFSET_MASK)
699 + pGuestParm->u.Pointer.size + PAGE_SIZE - 1) / PAGE_SIZE;
700 }
701
702 Log(("vmmdevHGCMCall: linptr size = %d\n", pGuestParm->u.Pointer.size));
703 } break;
704
705 case VMMDevHGCMParmType_PageList:
706 {
707 cbCmdSize += pGuestParm->u.PageList.size;
708 Log(("vmmdevHGCMCall: pagelist size = %d\n", pGuestParm->u.PageList.size));
709 } break;
710
711 case VMMDevHGCMParmType_32bit:
712 case VMMDevHGCMParmType_64bit:
713 {
714 } break;
715
716 default:
717 {
718 AssertMsgFailed(("vmmdevHGCMCall: invalid parameter type %x\n", pGuestParm->type));
719 rc = VERR_INVALID_PARAMETER;
720 break;
721 }
722 }
723 }
724 }
725
726 if (RT_FAILURE (rc))
727 {
728 return rc;
729 }
730
731 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAlloc (cbCmdSize);
732
733 if (pCmd == NULL)
734 {
735 return VERR_NO_MEMORY;
736 }
737
738 memset (pCmd, 0, sizeof (*pCmd));
739
740 pCmd->paHostParms = NULL;
741 pCmd->cLinPtrs = cLinPtrs;
742 pCmd->cLinPtrPages = cLinPtrPages;
743
744 if (cLinPtrs > 0)
745 {
746 pCmd->paLinPtrs = (VBOXHGCMLINPTR *)RTMemAlloc ( sizeof (VBOXHGCMLINPTR) * cLinPtrs
747 + sizeof (RTGCPHYS) * cLinPtrPages);
748
749 if (pCmd->paLinPtrs == NULL)
750 {
751 RTMemFree (pCmd);
752 return VERR_NO_MEMORY;
753 }
754 }
755 else
756 {
757 pCmd->paLinPtrs = NULL;
758 }
759
760 VBOXDD_HGCMCALL_ENTER(pCmd, pHGCMCall->u32Function, pHGCMCall->u32ClientID, cbCmdSize);
761
762 /* Process parameters, changing them to host context pointers for easy
763 * processing by connector. Guest must insure that the pointed data is actually
764 * in the guest RAM and remains locked there for entire request processing.
765 */
766
767 if (cParms != 0)
768 {
769 /* Compute addresses of host parms array and first memory buffer. */
770 VBOXHGCMSVCPARM *pHostParm = (VBOXHGCMSVCPARM *)((char *)pCmd + sizeof (struct VBOXHGCMCMD));
771
772 uint8_t *pcBuf = (uint8_t *)pHostParm + cParms * sizeof (VBOXHGCMSVCPARM);
773
774 pCmd->paHostParms = pHostParm;
775
776 uint32_t iLinPtr = 0;
777 RTGCPHYS *pPages = (RTGCPHYS *)((uint8_t *)pCmd->paLinPtrs + sizeof (VBOXHGCMLINPTR) *cLinPtrs);
778
779 if (f64Bits)
780 {
781#ifdef VBOX_WITH_64_BITS_GUESTS
782 HGCMFunctionParameter64 *pGuestParm = VMMDEV_HGCM_CALL_PARMS64(pHGCMCall);
783#else
784 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
785 AssertFailed (); /* This code should not be called in this case */
786#endif /* VBOX_WITH_64_BITS_GUESTS */
787
788
789 for (i = 0; i < cParms && RT_SUCCESS(rc); i++, pGuestParm++, pHostParm++)
790 {
791 switch (pGuestParm->type)
792 {
793 case VMMDevHGCMParmType_32bit:
794 {
795 uint32_t u32 = pGuestParm->u.value32;
796
797 pHostParm->type = VBOX_HGCM_SVC_PARM_32BIT;
798 pHostParm->u.uint32 = u32;
799
800 Log(("vmmdevHGCMCall: uint32 guest parameter %u\n", u32));
801 break;
802 }
803
804 case VMMDevHGCMParmType_64bit:
805 {
806 uint64_t u64 = pGuestParm->u.value64;
807
808 pHostParm->type = VBOX_HGCM_SVC_PARM_64BIT;
809 pHostParm->u.uint64 = u64;
810
811 Log(("vmmdevHGCMCall: uint64 guest parameter %llu\n", u64));
812 break;
813 }
814
815 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
816 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
817 case VMMDevHGCMParmType_LinAddr: /* In & Out */
818 {
819 uint32_t size = pGuestParm->u.Pointer.size;
820 RTGCPTR linearAddr = pGuestParm->u.Pointer.u.linearAddr;
821
822 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
823 pHostParm->u.pointer.size = size;
824
825 /* Copy guest data to an allocated buffer, so
826 * services can use the data.
827 */
828
829 if (size == 0)
830 {
831 pHostParm->u.pointer.addr = NULL;
832 }
833 else
834 {
835 /* Don't overdo it */
836 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_Out)
837 rc = PDMDevHlpPhysReadGCVirt(pVMMDevState->pDevIns, pcBuf, linearAddr, size);
838 else
839 rc = VINF_SUCCESS;
840
841 if (RT_SUCCESS(rc))
842 {
843 pHostParm->u.pointer.addr = pcBuf;
844 pcBuf += size;
845
846 /* Remember the guest physical pages that belong to the virtual address region.
847 * Do it for all linear pointers because load state will require In pointer info too.
848 */
849 rc = vmmdevHGCMSaveLinPtr (pVMMDevState->pDevIns, i, linearAddr, size, iLinPtr, pCmd->paLinPtrs, &pPages);
850
851 iLinPtr++;
852 }
853 }
854
855 Log(("vmmdevHGCMCall: LinAddr guest parameter %RGv, rc = %Rrc\n", linearAddr, rc));
856 break;
857 }
858
859 case VMMDevHGCMParmType_PageList:
860 {
861 uint32_t size = pGuestParm->u.PageList.size;
862
863 /* Check that the page list info is within the request. */
864 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
865 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
866 {
867 rc = VERR_INVALID_PARAMETER;
868 break;
869 }
870
871 /* At least the structure is within. */
872 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
873
874 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
875
876 if ( pPageListInfo->cPages == 0
877 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
878 {
879 rc = VERR_INVALID_PARAMETER;
880 break;
881 }
882
883 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
884 pHostParm->u.pointer.size = size;
885
886 /* Copy guest data to an allocated buffer, so
887 * services can use the data.
888 */
889
890 if (size == 0)
891 {
892 pHostParm->u.pointer.addr = NULL;
893 }
894 else
895 {
896 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_TO_HOST)
897 {
898 /* Copy pages to the pcBuf[size]. */
899 rc = vmmdevHGCMPageListRead(pVMMDevState->pDevIns, pcBuf, size, pPageListInfo);
900 }
901 else
902 rc = VINF_SUCCESS;
903
904 if (RT_SUCCESS(rc))
905 {
906 pHostParm->u.pointer.addr = pcBuf;
907 pcBuf += size;
908 }
909 }
910
911 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
912 break;
913 }
914
915 /* just to shut up gcc */
916 default:
917 AssertFailed();
918 break;
919 }
920 }
921 }
922 else
923 {
924#ifdef VBOX_WITH_64_BITS_GUESTS
925 HGCMFunctionParameter32 *pGuestParm = VMMDEV_HGCM_CALL_PARMS32(pHGCMCall);
926#else
927 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
928#endif /* VBOX_WITH_64_BITS_GUESTS */
929
930 for (i = 0; i < cParms && RT_SUCCESS(rc); i++, pGuestParm++, pHostParm++)
931 {
932 switch (pGuestParm->type)
933 {
934 case VMMDevHGCMParmType_32bit:
935 {
936 uint32_t u32 = pGuestParm->u.value32;
937
938 pHostParm->type = VBOX_HGCM_SVC_PARM_32BIT;
939 pHostParm->u.uint32 = u32;
940
941 Log(("vmmdevHGCMCall: uint32 guest parameter %u\n", u32));
942 break;
943 }
944
945 case VMMDevHGCMParmType_64bit:
946 {
947 uint64_t u64 = pGuestParm->u.value64;
948
949 pHostParm->type = VBOX_HGCM_SVC_PARM_64BIT;
950 pHostParm->u.uint64 = u64;
951
952 Log(("vmmdevHGCMCall: uint64 guest parameter %llu\n", u64));
953 break;
954 }
955
956 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
957 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
958 case VMMDevHGCMParmType_LinAddr: /* In & Out */
959 {
960 uint32_t size = pGuestParm->u.Pointer.size;
961 RTGCPTR linearAddr = pGuestParm->u.Pointer.u.linearAddr;
962
963 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
964 pHostParm->u.pointer.size = size;
965
966 /* Copy guest data to an allocated buffer, so
967 * services can use the data.
968 */
969
970 if (size == 0)
971 {
972 pHostParm->u.pointer.addr = NULL;
973 }
974 else
975 {
976 /* Don't overdo it */
977 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_Out)
978 rc = PDMDevHlpPhysReadGCVirt(pVMMDevState->pDevIns, pcBuf, linearAddr, size);
979 else
980 rc = VINF_SUCCESS;
981
982 if (RT_SUCCESS(rc))
983 {
984 pHostParm->u.pointer.addr = pcBuf;
985 pcBuf += size;
986
987 /* Remember the guest physical pages that belong to the virtual address region.
988 * Do it for all linear pointers because load state will require In pointer info too.
989 */
990 rc = vmmdevHGCMSaveLinPtr (pVMMDevState->pDevIns, i, linearAddr, size, iLinPtr, pCmd->paLinPtrs, &pPages);
991
992 iLinPtr++;
993 }
994 }
995
996 Log(("vmmdevHGCMCall: LinAddr guest parameter %RGv, rc = %Rrc\n", linearAddr, rc));
997 break;
998 }
999
1000 case VMMDevHGCMParmType_PageList:
1001 {
1002 uint32_t size = pGuestParm->u.PageList.size;
1003
1004 /* Check that the page list info is within the request. */
1005 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
1006 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
1007 {
1008 rc = VERR_INVALID_PARAMETER;
1009 break;
1010 }
1011
1012 /* At least the structure is within. */
1013 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
1014
1015 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
1016
1017 if ( pPageListInfo->cPages == 0
1018 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
1019 {
1020 rc = VERR_INVALID_PARAMETER;
1021 break;
1022 }
1023
1024 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
1025 pHostParm->u.pointer.size = size;
1026
1027 /* Copy guest data to an allocated buffer, so
1028 * services can use the data.
1029 */
1030
1031 if (size == 0)
1032 {
1033 pHostParm->u.pointer.addr = NULL;
1034 }
1035 else
1036 {
1037 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_TO_HOST)
1038 {
1039 /* Copy pages to the pcBuf[size]. */
1040 rc = vmmdevHGCMPageListRead(pVMMDevState->pDevIns, pcBuf, size, pPageListInfo);
1041 }
1042 else
1043 rc = VINF_SUCCESS;
1044
1045 if (RT_SUCCESS(rc))
1046 {
1047 pHostParm->u.pointer.addr = pcBuf;
1048 pcBuf += size;
1049 }
1050 }
1051
1052 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
1053 break;
1054 }
1055
1056 /* just to shut up gcc */
1057 default:
1058 AssertFailed();
1059 break;
1060 }
1061 }
1062 }
1063 }
1064
1065 if (RT_SUCCESS (rc))
1066 {
1067 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, pHGCMCall->header.header.size, VBOXHGCMCMDTYPE_CALL);
1068
1069 /* Pass the function call to HGCM connector for actual processing */
1070 rc = pVMMDevState->pHGCMDrv->pfnCall (pVMMDevState->pHGCMDrv, pCmd, pHGCMCall->u32ClientID,
1071 pHGCMCall->u32Function, cParms, pCmd->paHostParms);
1072 }
1073 else
1074 {
1075 if (pCmd->paLinPtrs)
1076 {
1077 RTMemFree (pCmd->paLinPtrs);
1078 }
1079
1080 RTMemFree (pCmd);
1081 }
1082
1083 return rc;
1084}
1085
1086static void logRelLoadStatePointerIndexMismatch (uint32_t iParm, uint32_t iSavedParm, int iLinPtr, int cLinPtrs)
1087{
1088 LogRel(("Warning: VMMDev load state: a pointer parameter index mismatch %d (expected %d) (%d/%d)\n",
1089 (int)iParm, (int)iSavedParm, iLinPtr, cLinPtrs));
1090}
1091
1092static void logRelLoadStateBufferSizeMismatch (uint32_t size, uint32_t iPage, uint32_t cPages)
1093{
1094 LogRel(("Warning: VMMDev load state: buffer size mismatch: size %d, page %d/%d\n",
1095 (int)size, (int)iPage, (int)cPages));
1096}
1097
1098
1099static int vmmdevHGCMCallSaved (VMMDevState *pVMMDevState, VMMDevHGCMCall *pHGCMCall, RTGCPHYS GCPhys, uint32_t cbHGCMCall, bool f64Bits, bool *pfHGCMCalled, VBOXHGCMCMD *pSavedCmd, VBOXHGCMCMD **ppCmd)
1100{
1101 int rc = VINF_SUCCESS;
1102
1103 Log(("vmmdevHGCMCallSaved: client id = %d, function = %d, %s bit\n", pHGCMCall->u32ClientID, pHGCMCall->u32Function, f64Bits? "64": "32"));
1104
1105 /* Compute size and allocate memory block to hold:
1106 * struct VBOXHGCMCMD
1107 * VBOXHGCMSVCPARM[cParms]
1108 * memory buffers for pointer parameters.
1109 */
1110
1111 uint32_t cParms = pHGCMCall->cParms;
1112
1113 Log(("vmmdevHGCMCall: cParms = %d\n", cParms));
1114
1115 /*
1116 * Compute size of required memory buffer.
1117 */
1118
1119 uint32_t cbCmdSize = sizeof (struct VBOXHGCMCMD) + cParms * sizeof (VBOXHGCMSVCPARM);
1120
1121 uint32_t i;
1122
1123 int32_t cLinPtrs = 0;
1124 int32_t cLinPtrPages = 0;
1125
1126 if (f64Bits)
1127 {
1128#ifdef VBOX_WITH_64_BITS_GUESTS
1129 HGCMFunctionParameter64 *pGuestParm = VMMDEV_HGCM_CALL_PARMS64(pHGCMCall);
1130#else
1131 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
1132 AssertFailed (); /* This code should not be called in this case */
1133#endif /* VBOX_WITH_64_BITS_GUESTS */
1134
1135 /* Look for pointer parameters, which require a host buffer. */
1136 for (i = 0; i < cParms && RT_SUCCESS(rc); i++, pGuestParm++)
1137 {
1138 switch (pGuestParm->type)
1139 {
1140 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
1141 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
1142 case VMMDevHGCMParmType_LinAddr: /* In & Out */
1143 {
1144 if (pGuestParm->u.Pointer.size > 0)
1145 {
1146 /* Only pointers with some actual data are counted. */
1147 cbCmdSize += pGuestParm->u.Pointer.size;
1148
1149 cLinPtrs++;
1150 /* Take the offset into the current page also into account! */
1151 cLinPtrPages += ((pGuestParm->u.Pointer.u.linearAddr & PAGE_OFFSET_MASK)
1152 + pGuestParm->u.Pointer.size + PAGE_SIZE - 1) / PAGE_SIZE;
1153 }
1154
1155 Log(("vmmdevHGCMCall: linptr size = %d\n", pGuestParm->u.Pointer.size));
1156 } break;
1157
1158 case VMMDevHGCMParmType_PageList:
1159 {
1160 cbCmdSize += pGuestParm->u.PageList.size;
1161 Log(("vmmdevHGCMCall: pagelist size = %d\n", pGuestParm->u.PageList.size));
1162 } break;
1163
1164 case VMMDevHGCMParmType_32bit:
1165 case VMMDevHGCMParmType_64bit:
1166 {
1167 } break;
1168
1169 default:
1170 case VMMDevHGCMParmType_PhysAddr:
1171 {
1172 AssertMsgFailed(("vmmdevHGCMCall: invalid parameter type %x\n", pGuestParm->type));
1173 rc = VERR_INVALID_PARAMETER;
1174 break;
1175 }
1176 }
1177 }
1178 }
1179 else
1180 {
1181#ifdef VBOX_WITH_64_BITS_GUESTS
1182 HGCMFunctionParameter32 *pGuestParm = VMMDEV_HGCM_CALL_PARMS32(pHGCMCall);
1183#else
1184 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
1185#endif /* VBOX_WITH_64_BITS_GUESTS */
1186
1187 /* Look for pointer parameters, which require a host buffer. */
1188 for (i = 0; i < cParms && RT_SUCCESS(rc); i++, pGuestParm++)
1189 {
1190 switch (pGuestParm->type)
1191 {
1192 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
1193 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
1194 case VMMDevHGCMParmType_LinAddr: /* In & Out */
1195 {
1196 if (pGuestParm->u.Pointer.size > 0)
1197 {
1198 /* Only pointers with some actual data are counted. */
1199 cbCmdSize += pGuestParm->u.Pointer.size;
1200
1201 cLinPtrs++;
1202 /* Take the offset into the current page also into account! */
1203 cLinPtrPages += ((pGuestParm->u.Pointer.u.linearAddr & PAGE_OFFSET_MASK)
1204 + pGuestParm->u.Pointer.size + PAGE_SIZE - 1) / PAGE_SIZE;
1205 }
1206
1207 Log(("vmmdevHGCMCall: linptr size = %d\n", pGuestParm->u.Pointer.size));
1208 } break;
1209
1210 case VMMDevHGCMParmType_PageList:
1211 {
1212 cbCmdSize += pGuestParm->u.PageList.size;
1213 Log(("vmmdevHGCMCall: pagelist size = %d\n", pGuestParm->u.PageList.size));
1214 } break;
1215
1216 case VMMDevHGCMParmType_32bit:
1217 case VMMDevHGCMParmType_64bit:
1218 {
1219 } break;
1220
1221 default:
1222 {
1223 AssertMsgFailed(("vmmdevHGCMCall: invalid parameter type %x\n", pGuestParm->type));
1224 rc = VERR_INVALID_PARAMETER;
1225 break;
1226 }
1227 }
1228 }
1229 }
1230
1231 if (RT_FAILURE (rc))
1232 {
1233 return rc;
1234 }
1235
1236 if ( pSavedCmd->cLinPtrs != cLinPtrs
1237 || pSavedCmd->cLinPtrPages != cLinPtrPages)
1238 {
1239 LogRel(("VMMDev: invalid saved command ptrs: %d/%d, pages %d/%d\n",
1240 pSavedCmd->cLinPtrs, cLinPtrs, pSavedCmd->cLinPtrPages, cLinPtrPages));
1241 AssertFailed();
1242 return VERR_INVALID_PARAMETER;
1243 }
1244
1245 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (cbCmdSize);
1246
1247 if (pCmd == NULL)
1248 {
1249 return VERR_NO_MEMORY;
1250 }
1251
1252 vmmdevRestoreSavedCommand(pCmd, pSavedCmd);
1253 *ppCmd = pCmd;
1254
1255 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, pHGCMCall->header.header.size, VBOXHGCMCMDTYPE_CALL);
1256
1257 /* Process parameters, changing them to host context pointers for easy
1258 * processing by connector. Guest must insure that the pointed data is actually
1259 * in the guest RAM and remains locked there for entire request processing.
1260 */
1261
1262 if (cParms != 0)
1263 {
1264 /* Compute addresses of host parms array and first memory buffer. */
1265 VBOXHGCMSVCPARM *pHostParm = (VBOXHGCMSVCPARM *)((uint8_t *)pCmd + sizeof (struct VBOXHGCMCMD));
1266
1267 uint8_t *pu8Buf = (uint8_t *)pHostParm + cParms * sizeof (VBOXHGCMSVCPARM);
1268
1269 pCmd->paHostParms = pHostParm;
1270
1271 uint32_t iParm;
1272 int iLinPtr = 0;
1273
1274 if (f64Bits)
1275 {
1276#ifdef VBOX_WITH_64_BITS_GUESTS
1277 HGCMFunctionParameter64 *pGuestParm = VMMDEV_HGCM_CALL_PARMS64(pHGCMCall);
1278#else
1279 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
1280 AssertFailed (); /* This code should not be called in this case */
1281#endif /* VBOX_WITH_64_BITS_GUESTS */
1282
1283 for (iParm = 0; iParm < cParms && RT_SUCCESS(rc); iParm++, pGuestParm++, pHostParm++)
1284 {
1285 switch (pGuestParm->type)
1286 {
1287 case VMMDevHGCMParmType_32bit:
1288 {
1289 uint32_t u32 = pGuestParm->u.value32;
1290
1291 pHostParm->type = VBOX_HGCM_SVC_PARM_32BIT;
1292 pHostParm->u.uint32 = u32;
1293
1294 Log(("vmmdevHGCMCall: uint32 guest parameter %u\n", u32));
1295 break;
1296 }
1297
1298 case VMMDevHGCMParmType_64bit:
1299 {
1300 uint64_t u64 = pGuestParm->u.value64;
1301
1302 pHostParm->type = VBOX_HGCM_SVC_PARM_64BIT;
1303 pHostParm->u.uint64 = u64;
1304
1305 Log(("vmmdevHGCMCall: uint64 guest parameter %llu\n", u64));
1306 break;
1307 }
1308
1309 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
1310 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
1311 case VMMDevHGCMParmType_LinAddr: /* In & Out */
1312 {
1313 uint32_t size = pGuestParm->u.Pointer.size;
1314
1315 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
1316 pHostParm->u.pointer.size = size;
1317
1318 /* Copy guest data to an allocated buffer, so
1319 * services can use the data.
1320 */
1321
1322 if (size == 0)
1323 {
1324 pHostParm->u.pointer.addr = NULL;
1325 }
1326 else
1327 {
1328 /* The saved command already have the page list in pCmd->paLinPtrs.
1329 * Read data from guest pages.
1330 */
1331 /* Don't overdo it */
1332 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_Out)
1333 {
1334 if ( iLinPtr >= pCmd->cLinPtrs
1335 || pCmd->paLinPtrs[iLinPtr].iParm != iParm)
1336 {
1337 logRelLoadStatePointerIndexMismatch (iParm, pCmd->paLinPtrs[iLinPtr].iParm, iLinPtr, pCmd->cLinPtrs);
1338 rc = VERR_INVALID_PARAMETER;
1339 }
1340 else
1341 {
1342 VBOXHGCMLINPTR *pLinPtr = &pCmd->paLinPtrs[iLinPtr];
1343
1344 uint32_t iPage;
1345 uint32_t offPage = pLinPtr->offFirstPage;
1346 size_t cbRemaining = size;
1347 uint8_t *pu8Dst = pu8Buf;
1348 for (iPage = 0; iPage < pLinPtr->cPages; iPage++)
1349 {
1350 if (cbRemaining == 0)
1351 {
1352 logRelLoadStateBufferSizeMismatch (size, iPage, pLinPtr->cPages);
1353 break;
1354 }
1355
1356 size_t cbChunk = PAGE_SIZE - offPage;
1357
1358 if (cbChunk > cbRemaining)
1359 {
1360 cbChunk = cbRemaining;
1361 }
1362
1363 rc = PDMDevHlpPhysRead(pVMMDevState->pDevIns,
1364 pLinPtr->paPages[iPage] + offPage,
1365 pu8Dst, cbChunk);
1366
1367 AssertRCBreak(rc);
1368
1369 offPage = 0; /* A next page is read from 0 offset. */
1370 cbRemaining -= cbChunk;
1371 pu8Dst += cbChunk;
1372 }
1373 }
1374 }
1375 else
1376 rc = VINF_SUCCESS;
1377
1378 if (RT_SUCCESS(rc))
1379 {
1380 pHostParm->u.pointer.addr = pu8Buf;
1381 pu8Buf += size;
1382
1383 iLinPtr++;
1384 }
1385 }
1386
1387 Log(("vmmdevHGCMCall: LinAddr guest parameter %RGv, rc = %Rrc\n",
1388 pGuestParm->u.Pointer.u.linearAddr, rc));
1389 break;
1390 }
1391
1392 case VMMDevHGCMParmType_PageList:
1393 {
1394 uint32_t size = pGuestParm->u.PageList.size;
1395
1396 /* Check that the page list info is within the request. */
1397 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
1398 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
1399 {
1400 rc = VERR_INVALID_PARAMETER;
1401 break;
1402 }
1403
1404 /* At least the structure is within. */
1405 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
1406
1407 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
1408
1409 if ( pPageListInfo->cPages == 0
1410 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
1411 {
1412 rc = VERR_INVALID_PARAMETER;
1413 break;
1414 }
1415
1416 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
1417 pHostParm->u.pointer.size = size;
1418
1419 /* Copy guest data to an allocated buffer, so
1420 * services can use the data.
1421 */
1422
1423 if (size == 0)
1424 {
1425 pHostParm->u.pointer.addr = NULL;
1426 }
1427 else
1428 {
1429 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_TO_HOST)
1430 {
1431 /* Copy pages to the pcBuf[size]. */
1432 rc = vmmdevHGCMPageListRead(pVMMDevState->pDevIns, pu8Buf, size, pPageListInfo);
1433 }
1434 else
1435 rc = VINF_SUCCESS;
1436
1437 if (RT_SUCCESS(rc))
1438 {
1439 pHostParm->u.pointer.addr = pu8Buf;
1440 pu8Buf += size;
1441 }
1442 }
1443
1444 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
1445 break;
1446 }
1447
1448 /* just to shut up gcc */
1449 default:
1450 AssertFailed();
1451 break;
1452 }
1453 }
1454 }
1455 else
1456 {
1457#ifdef VBOX_WITH_64_BITS_GUESTS
1458 HGCMFunctionParameter32 *pGuestParm = VMMDEV_HGCM_CALL_PARMS32(pHGCMCall);
1459#else
1460 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
1461#endif /* VBOX_WITH_64_BITS_GUESTS */
1462
1463 for (iParm = 0; iParm < cParms && RT_SUCCESS(rc); iParm++, pGuestParm++, pHostParm++)
1464 {
1465 switch (pGuestParm->type)
1466 {
1467 case VMMDevHGCMParmType_32bit:
1468 {
1469 uint32_t u32 = pGuestParm->u.value32;
1470
1471 pHostParm->type = VBOX_HGCM_SVC_PARM_32BIT;
1472 pHostParm->u.uint32 = u32;
1473
1474 Log(("vmmdevHGCMCall: uint32 guest parameter %u\n", u32));
1475 break;
1476 }
1477
1478 case VMMDevHGCMParmType_64bit:
1479 {
1480 uint64_t u64 = pGuestParm->u.value64;
1481
1482 pHostParm->type = VBOX_HGCM_SVC_PARM_64BIT;
1483 pHostParm->u.uint64 = u64;
1484
1485 Log(("vmmdevHGCMCall: uint64 guest parameter %llu\n", u64));
1486 break;
1487 }
1488
1489 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
1490 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
1491 case VMMDevHGCMParmType_LinAddr: /* In & Out */
1492 {
1493 uint32_t size = pGuestParm->u.Pointer.size;
1494
1495 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
1496 pHostParm->u.pointer.size = size;
1497
1498 /* Copy guest data to an allocated buffer, so
1499 * services can use the data.
1500 */
1501
1502 if (size == 0)
1503 {
1504 pHostParm->u.pointer.addr = NULL;
1505 }
1506 else
1507 {
1508 /* The saved command already have the page list in pCmd->paLinPtrs.
1509 * Read data from guest pages.
1510 */
1511 /* Don't overdo it */
1512 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_Out)
1513 {
1514 if ( iLinPtr >= pCmd->cLinPtrs
1515 || pCmd->paLinPtrs[iLinPtr].iParm != iParm)
1516 {
1517 logRelLoadStatePointerIndexMismatch (iParm, pCmd->paLinPtrs[iLinPtr].iParm, iLinPtr, pCmd->cLinPtrs);
1518 rc = VERR_INVALID_PARAMETER;
1519 }
1520 else
1521 {
1522 VBOXHGCMLINPTR *pLinPtr = &pCmd->paLinPtrs[iLinPtr];
1523
1524 uint32_t iPage;
1525 uint32_t offPage = pLinPtr->offFirstPage;
1526 size_t cbRemaining = size;
1527 uint8_t *pu8Dst = pu8Buf;
1528 for (iPage = 0; iPage < pLinPtr->cPages; iPage++)
1529 {
1530 if (cbRemaining == 0)
1531 {
1532 logRelLoadStateBufferSizeMismatch (size, iPage, pLinPtr->cPages);
1533 break;
1534 }
1535
1536 size_t cbChunk = PAGE_SIZE - offPage;
1537
1538 if (cbChunk > cbRemaining)
1539 {
1540 cbChunk = cbRemaining;
1541 }
1542
1543 rc = PDMDevHlpPhysRead(pVMMDevState->pDevIns,
1544 pLinPtr->paPages[iPage] + offPage,
1545 pu8Dst, cbChunk);
1546
1547 AssertRCBreak(rc);
1548
1549 offPage = 0; /* A next page is read from 0 offset. */
1550 cbRemaining -= cbChunk;
1551 pu8Dst += cbChunk;
1552 }
1553 }
1554 }
1555 else
1556 rc = VINF_SUCCESS;
1557
1558 if (RT_SUCCESS(rc))
1559 {
1560 pHostParm->u.pointer.addr = pu8Buf;
1561 pu8Buf += size;
1562
1563 iLinPtr++;
1564 }
1565 }
1566
1567 Log(("vmmdevHGCMCall: LinAddr guest parameter %RGv, rc = %Rrc\n",
1568 pGuestParm->u.Pointer.u.linearAddr, rc));
1569 break;
1570 }
1571
1572 case VMMDevHGCMParmType_PageList:
1573 {
1574 uint32_t size = pGuestParm->u.PageList.size;
1575
1576 /* Check that the page list info is within the request. */
1577 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
1578 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
1579 {
1580 rc = VERR_INVALID_PARAMETER;
1581 break;
1582 }
1583
1584 /* At least the structure is within. */
1585 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
1586
1587 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
1588
1589 if ( pPageListInfo->cPages == 0
1590 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
1591 {
1592 rc = VERR_INVALID_PARAMETER;
1593 break;
1594 }
1595
1596 pHostParm->type = VBOX_HGCM_SVC_PARM_PTR;
1597 pHostParm->u.pointer.size = size;
1598
1599 /* Copy guest data to an allocated buffer, so
1600 * services can use the data.
1601 */
1602
1603 if (size == 0)
1604 {
1605 pHostParm->u.pointer.addr = NULL;
1606 }
1607 else
1608 {
1609 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_TO_HOST)
1610 {
1611 /* Copy pages to the pcBuf[size]. */
1612 rc = vmmdevHGCMPageListRead(pVMMDevState->pDevIns, pu8Buf, size, pPageListInfo);
1613 }
1614 else
1615 rc = VINF_SUCCESS;
1616
1617 if (RT_SUCCESS(rc))
1618 {
1619 pHostParm->u.pointer.addr = pu8Buf;
1620 pu8Buf += size;
1621 }
1622 }
1623
1624 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
1625 break;
1626 }
1627
1628 /* just to shut up gcc */
1629 default:
1630 AssertFailed();
1631 break;
1632 }
1633 }
1634 }
1635 }
1636
1637 if (RT_SUCCESS (rc))
1638 {
1639 /* Pass the function call to HGCM connector for actual processing */
1640 rc = pVMMDevState->pHGCMDrv->pfnCall (pVMMDevState->pHGCMDrv, pCmd, pHGCMCall->u32ClientID, pHGCMCall->u32Function, cParms, pCmd->paHostParms);
1641 if (RT_SUCCESS (rc))
1642 {
1643 *pfHGCMCalled = true;
1644 }
1645 }
1646
1647 return rc;
1648}
1649
1650/**
1651 * VMMDevReq_HGCMCancel worker.
1652 *
1653 * @thread EMT
1654 */
1655int vmmdevHGCMCancel (VMMDevState *pVMMDevState, VMMDevHGCMCancel *pHGCMCancel, RTGCPHYS GCPhys)
1656{
1657 NOREF(pHGCMCancel);
1658 int rc = vmmdevHGCMCancel2(pVMMDevState, GCPhys);
1659 return rc == VERR_NOT_FOUND ? VERR_INVALID_PARAMETER : rc;
1660}
1661
1662/**
1663 * VMMDevReq_HGCMCancel2 worker.
1664 *
1665 * @retval VINF_SUCCESS on success.
1666 * @retval VERR_NOT_FOUND if the request was not found.
1667 * @retval VERR_INVALID_PARAMETER if the request address is invalid.
1668 *
1669 * @param pThis The VMMDev instance data.
1670 * @param GCPhys The address of the request that should be cancelled.
1671 *
1672 * @thread EMT
1673 */
1674int vmmdevHGCMCancel2 (VMMDevState *pThis, RTGCPHYS GCPhys)
1675{
1676 if ( GCPhys == 0
1677 || GCPhys == NIL_RTGCPHYS
1678 || GCPhys == NIL_RTGCPHYS32)
1679 {
1680 Log(("vmmdevHGCMCancel2: GCPhys=%#x\n", GCPhys));
1681 return VERR_INVALID_PARAMETER;
1682 }
1683
1684 /*
1685 * Locate the command and cancel it while under the protection of
1686 * the lock. hgcmCompletedWorker makes assumptions about this.
1687 */
1688 int rc = vmmdevHGCMCmdListLock (pThis);
1689 AssertRCReturn(rc, rc);
1690
1691 PVBOXHGCMCMD pCmd = vmmdevHGCMFindCommandLocked (pThis, GCPhys);
1692 if (pCmd)
1693 {
1694 pCmd->fCancelled = true;
1695 Log(("vmmdevHGCMCancel2: Cancelled pCmd=%p / GCPhys=%#x\n", pCmd, GCPhys));
1696 }
1697 else
1698 rc = VERR_NOT_FOUND;
1699
1700 vmmdevHGCMCmdListUnlock (pThis);
1701 return rc;
1702}
1703
1704static int vmmdevHGCMCmdVerify (PVBOXHGCMCMD pCmd, VMMDevHGCMRequestHeader *pHeader)
1705{
1706 switch (pCmd->enmCmdType)
1707 {
1708 case VBOXHGCMCMDTYPE_CONNECT:
1709 if ( pHeader->header.requestType == VMMDevReq_HGCMConnect
1710 || pHeader->header.requestType == VMMDevReq_HGCMCancel) return VINF_SUCCESS;
1711 break;
1712
1713 case VBOXHGCMCMDTYPE_DISCONNECT:
1714 if ( pHeader->header.requestType == VMMDevReq_HGCMDisconnect
1715 || pHeader->header.requestType == VMMDevReq_HGCMCancel) return VINF_SUCCESS;
1716 break;
1717
1718 case VBOXHGCMCMDTYPE_CALL:
1719#ifdef VBOX_WITH_64_BITS_GUESTS
1720 if ( pHeader->header.requestType == VMMDevReq_HGCMCall32
1721 || pHeader->header.requestType == VMMDevReq_HGCMCall64
1722 || pHeader->header.requestType == VMMDevReq_HGCMCancel) return VINF_SUCCESS;
1723#else
1724 if ( pHeader->header.requestType == VMMDevReq_HGCMCall
1725 || pHeader->header.requestType == VMMDevReq_HGCMCancel) return VINF_SUCCESS;
1726#endif /* VBOX_WITH_64_BITS_GUESTS */
1727
1728 break;
1729
1730 default:
1731 AssertFailed ();
1732 }
1733
1734 LogRel(("VMMDEV: Invalid HGCM command: pCmd->enmCmdType = 0x%08X, pHeader->header.requestType = 0x%08X\n",
1735 pCmd->enmCmdType, pHeader->header.requestType));
1736 return VERR_INVALID_PARAMETER;
1737}
1738
1739#define PDMIHGCMPORT_2_VMMDEVSTATE(pInterface) ( (VMMDevState *) ((uintptr_t)pInterface - RT_OFFSETOF(VMMDevState, IHGCMPort)) )
1740
1741DECLCALLBACK(void) hgcmCompletedWorker (PPDMIHGCMPORT pInterface, int32_t result, PVBOXHGCMCMD pCmd)
1742{
1743 VMMDevState *pVMMDevState = PDMIHGCMPORT_2_VMMDEVSTATE(pInterface);
1744#ifdef VBOX_WITH_DTRACE
1745 uint32_t idFunction = 0;
1746 uint32_t idClient = 0;
1747#endif
1748
1749 int rc = VINF_SUCCESS;
1750
1751 if (result == VINF_HGCM_SAVE_STATE)
1752 {
1753 /* If the completion routine was called because HGCM saves its state,
1754 * then currently nothing to be done here. The pCmd stays in the list
1755 * and will be saved later when the VMMDev state will be saved.
1756 *
1757 * It it assumed that VMMDev saves state after the HGCM services,
1758 * and, therefore, VBOXHGCMCMD structures are not removed by
1759 * vmmdevHGCMSaveState from the list, while HGCM uses them.
1760 */
1761 LogFlowFunc(("VINF_HGCM_SAVE_STATE for command %p\n", pCmd));
1762 return;
1763 }
1764
1765 /*
1766 * The cancellation protocol requires us to remove the command here
1767 * and then check the flag. Cancelled commands must not be written
1768 * back to guest memory.
1769 */
1770 VBOXDD_HGCMCALL_COMPLETED_EMT(pCmd, result);
1771 vmmdevHGCMRemoveCommand (pVMMDevState, pCmd);
1772
1773 if (pCmd->fCancelled)
1774 {
1775 LogFlowFunc(("A cancelled command %p\n", pCmd));
1776 }
1777 else
1778 {
1779 /* Preallocated block for requests which have up to 8 parameters (most of requests). */
1780#ifdef VBOX_WITH_64_BITS_GUESTS
1781 uint8_t au8Prealloc[sizeof (VMMDevHGCMCall) + 8 * sizeof (HGCMFunctionParameter64)];
1782#else
1783 uint8_t au8Prealloc[sizeof (VMMDevHGCMCall) + 8 * sizeof (HGCMFunctionParameter)];
1784#endif /* VBOX_WITH_64_BITS_GUESTS */
1785
1786 VMMDevHGCMRequestHeader *pHeader;
1787
1788 if (pCmd->cbSize <= sizeof (au8Prealloc))
1789 {
1790 pHeader = (VMMDevHGCMRequestHeader *)&au8Prealloc[0];
1791 }
1792 else
1793 {
1794 pHeader = (VMMDevHGCMRequestHeader *)RTMemAlloc (pCmd->cbSize);
1795 if (pHeader == NULL)
1796 {
1797 LogRel(("VMMDev: Failed to allocate %u bytes for HGCM request completion!!!\n", pCmd->cbSize));
1798
1799 /* Free it. The command have to be excluded from list of active commands anyway. */
1800 RTMemFree (pCmd);
1801 return;
1802 }
1803 }
1804
1805 /*
1806 * Enter and leave the critical section here so we make sure
1807 * vmmdevRequestHandler has completed before we read & write
1808 * the request. (This isn't 100% optimal, but it solves the
1809 * 3.0 blocker.)
1810 */
1811 /** @todo s/pVMMDevState/pThis/g */
1812 /** @todo It would be faster if this interface would use MMIO2 memory and we
1813 * didn't have to mess around with PDMDevHlpPhysRead/Write. We're
1814 * reading the header 3 times now and writing the request back twice. */
1815
1816 PDMCritSectEnter(&pVMMDevState->CritSect, VERR_SEM_BUSY);
1817 PDMCritSectLeave(&pVMMDevState->CritSect);
1818
1819 PDMDevHlpPhysRead(pVMMDevState->pDevIns, pCmd->GCPhys, pHeader, pCmd->cbSize);
1820
1821 /* Setup return codes. */
1822 pHeader->result = result;
1823
1824 /* Verify the request type. */
1825 rc = vmmdevHGCMCmdVerify (pCmd, pHeader);
1826
1827 if (RT_SUCCESS (rc))
1828 {
1829 /* Update parameters and data buffers. */
1830
1831 switch (pHeader->header.requestType)
1832 {
1833#ifdef VBOX_WITH_64_BITS_GUESTS
1834 case VMMDevReq_HGCMCall64:
1835 {
1836 VMMDevHGCMCall *pHGCMCall = (VMMDevHGCMCall *)pHeader;
1837
1838 uint32_t cParms = pHGCMCall->cParms;
1839
1840 VBOXHGCMSVCPARM *pHostParm = pCmd->paHostParms;
1841
1842 uint32_t i;
1843 uint32_t iLinPtr = 0;
1844
1845 HGCMFunctionParameter64 *pGuestParm = VMMDEV_HGCM_CALL_PARMS64(pHGCMCall);
1846
1847 for (i = 0; i < cParms; i++, pGuestParm++, pHostParm++)
1848 {
1849 switch (pGuestParm->type)
1850 {
1851 case VMMDevHGCMParmType_32bit:
1852 {
1853 pGuestParm->u.value32 = pHostParm->u.uint32;
1854 } break;
1855
1856 case VMMDevHGCMParmType_64bit:
1857 {
1858 pGuestParm->u.value64 = pHostParm->u.uint64;
1859 } break;
1860
1861 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
1862 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
1863 case VMMDevHGCMParmType_LinAddr: /* In & Out */
1864 {
1865 /* Copy buffer back to guest memory. */
1866 uint32_t size = pGuestParm->u.Pointer.size;
1867
1868 if (size > 0)
1869 {
1870 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_In)
1871 {
1872 /* Use the saved page list to write data back to the guest RAM. */
1873 rc = vmmdevHGCMWriteLinPtr (pVMMDevState->pDevIns, i, pHostParm->u.pointer.addr,
1874 size, iLinPtr, pCmd->paLinPtrs);
1875 AssertReleaseRC(rc);
1876 }
1877
1878 /* All linptrs with size > 0 were saved. Advance the index to the next linptr. */
1879 iLinPtr++;
1880 }
1881 } break;
1882
1883 case VMMDevHGCMParmType_PageList:
1884 {
1885 uint32_t cbHGCMCall = pCmd->cbSize; /* Size of the request. */
1886
1887 uint32_t size = pGuestParm->u.PageList.size;
1888
1889 /* Check that the page list info is within the request. */
1890 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
1891 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
1892 {
1893 rc = VERR_INVALID_PARAMETER;
1894 break;
1895 }
1896
1897 /* At least the structure is within. */
1898 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
1899
1900 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
1901
1902 if ( pPageListInfo->cPages == 0
1903 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
1904 {
1905 rc = VERR_INVALID_PARAMETER;
1906 break;
1907 }
1908
1909 if (size > 0)
1910 {
1911 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_FROM_HOST)
1912 {
1913 /* Copy pHostParm->u.pointer.addr[pHostParm->u.pointer.size] to pages. */
1914 rc = vmmdevHGCMPageListWrite(pVMMDevState->pDevIns, pPageListInfo, pHostParm->u.pointer.addr, size);
1915 }
1916 else
1917 rc = VINF_SUCCESS;
1918 }
1919
1920 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
1921 } break;
1922
1923 default:
1924 {
1925 /* This indicates that the guest request memory was corrupted. */
1926 AssertReleaseMsgFailed(("hgcmCompleted: invalid parameter type %08X\n", pGuestParm->type));
1927 }
1928 }
1929 }
1930# ifdef VBOX_WITH_DTRACE
1931 idFunction = pHGCMCall->u32Function;
1932 idClient = pHGCMCall->u32ClientID;
1933# endif
1934 break;
1935 }
1936
1937 case VMMDevReq_HGCMCall32:
1938 {
1939 VMMDevHGCMCall *pHGCMCall = (VMMDevHGCMCall *)pHeader;
1940
1941 uint32_t cParms = pHGCMCall->cParms;
1942
1943 VBOXHGCMSVCPARM *pHostParm = pCmd->paHostParms;
1944
1945 uint32_t i;
1946 uint32_t iLinPtr = 0;
1947
1948 HGCMFunctionParameter32 *pGuestParm = VMMDEV_HGCM_CALL_PARMS32(pHGCMCall);
1949
1950 for (i = 0; i < cParms; i++, pGuestParm++, pHostParm++)
1951 {
1952 switch (pGuestParm->type)
1953 {
1954 case VMMDevHGCMParmType_32bit:
1955 {
1956 pGuestParm->u.value32 = pHostParm->u.uint32;
1957 } break;
1958
1959 case VMMDevHGCMParmType_64bit:
1960 {
1961 pGuestParm->u.value64 = pHostParm->u.uint64;
1962 } break;
1963
1964 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
1965 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
1966 case VMMDevHGCMParmType_LinAddr: /* In & Out */
1967 {
1968 /* Copy buffer back to guest memory. */
1969 uint32_t size = pGuestParm->u.Pointer.size;
1970
1971 if (size > 0)
1972 {
1973 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_In)
1974 {
1975 /* Use the saved page list to write data back to the guest RAM. */
1976 rc = vmmdevHGCMWriteLinPtr (pVMMDevState->pDevIns, i, pHostParm->u.pointer.addr, size, iLinPtr, pCmd->paLinPtrs);
1977 AssertReleaseRC(rc);
1978 }
1979
1980 /* All linptrs with size > 0 were saved. Advance the index to the next linptr. */
1981 iLinPtr++;
1982 }
1983 } break;
1984
1985 case VMMDevHGCMParmType_PageList:
1986 {
1987 uint32_t cbHGCMCall = pCmd->cbSize; /* Size of the request. */
1988
1989 uint32_t size = pGuestParm->u.PageList.size;
1990
1991 /* Check that the page list info is within the request. */
1992 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
1993 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
1994 {
1995 rc = VERR_INVALID_PARAMETER;
1996 break;
1997 }
1998
1999 /* At least the structure is within. */
2000 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
2001
2002 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
2003
2004 if ( pPageListInfo->cPages == 0
2005 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
2006 {
2007 rc = VERR_INVALID_PARAMETER;
2008 break;
2009 }
2010
2011 if (size > 0)
2012 {
2013 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_FROM_HOST)
2014 {
2015 /* Copy pHostParm->u.pointer.addr[pHostParm->u.pointer.size] to pages. */
2016 rc = vmmdevHGCMPageListWrite(pVMMDevState->pDevIns, pPageListInfo, pHostParm->u.pointer.addr, size);
2017 }
2018 else
2019 rc = VINF_SUCCESS;
2020 }
2021
2022 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
2023 } break;
2024
2025 default:
2026 {
2027 /* This indicates that the guest request memory was corrupted. */
2028 AssertReleaseMsgFailed(("hgcmCompleted: invalid parameter type %08X\n", pGuestParm->type));
2029 }
2030 }
2031 }
2032# ifdef VBOX_WITH_DTRACE
2033 idFunction = pHGCMCall->u32Function;
2034 idClient = pHGCMCall->u32ClientID;
2035# endif
2036 break;
2037 }
2038#else
2039 case VMMDevReq_HGCMCall:
2040 {
2041 VMMDevHGCMCall *pHGCMCall = (VMMDevHGCMCall *)pHeader;
2042
2043 uint32_t cParms = pHGCMCall->cParms;
2044
2045 VBOXHGCMSVCPARM *pHostParm = pCmd->paHostParms;
2046
2047 uint32_t i;
2048 uint32_t iLinPtr = 0;
2049
2050 HGCMFunctionParameter *pGuestParm = VMMDEV_HGCM_CALL_PARMS(pHGCMCall);
2051
2052 for (i = 0; i < cParms; i++, pGuestParm++, pHostParm++)
2053 {
2054 switch (pGuestParm->type)
2055 {
2056 case VMMDevHGCMParmType_32bit:
2057 {
2058 pGuestParm->u.value32 = pHostParm->u.uint32;
2059 } break;
2060
2061 case VMMDevHGCMParmType_64bit:
2062 {
2063 pGuestParm->u.value64 = pHostParm->u.uint64;
2064 } break;
2065
2066 case VMMDevHGCMParmType_LinAddr_In: /* In (read) */
2067 case VMMDevHGCMParmType_LinAddr_Out: /* Out (write) */
2068 case VMMDevHGCMParmType_LinAddr: /* In & Out */
2069 {
2070 /* Copy buffer back to guest memory. */
2071 uint32_t size = pGuestParm->u.Pointer.size;
2072
2073 if (size > 0)
2074 {
2075 if (pGuestParm->type != VMMDevHGCMParmType_LinAddr_In)
2076 {
2077 /* Use the saved page list to write data back to the guest RAM. */
2078 rc = vmmdevHGCMWriteLinPtr (pVMMDevState->pDevIns, i, pHostParm->u.pointer.addr, size, iLinPtr, pCmd->paLinPtrs);
2079 AssertReleaseRC(rc);
2080 }
2081
2082 /* All linptrs with size > 0 were saved. Advance the index to the next linptr. */
2083 iLinPtr++;
2084 }
2085 } break;
2086
2087 case VMMDevHGCMParmType_PageList:
2088 {
2089 uint32_t cbHGCMCall = pCmd->cbSize; /* Size of the request. */
2090
2091 uint32_t size = pGuestParm->u.PageList.size;
2092
2093 /* Check that the page list info is within the request. */
2094 if ( cbHGCMCall < sizeof (HGCMPageListInfo)
2095 || pGuestParm->u.PageList.offset > cbHGCMCall - sizeof (HGCMPageListInfo))
2096 {
2097 rc = VERR_INVALID_PARAMETER;
2098 break;
2099 }
2100
2101 /* At least the structure is within. */
2102 HGCMPageListInfo *pPageListInfo = (HGCMPageListInfo *)((uint8_t *)pHGCMCall + pGuestParm->u.PageList.offset);
2103
2104 uint32_t cbPageListInfo = sizeof (HGCMPageListInfo) + (pPageListInfo->cPages - 1) * sizeof (pPageListInfo->aPages[0]);
2105
2106 if ( pPageListInfo->cPages == 0
2107 || cbHGCMCall < pGuestParm->u.PageList.offset + cbPageListInfo)
2108 {
2109 rc = VERR_INVALID_PARAMETER;
2110 break;
2111 }
2112
2113 if (size > 0)
2114 {
2115 if (pPageListInfo->flags & VBOX_HGCM_F_PARM_DIRECTION_FROM_HOST)
2116 {
2117 /* Copy pHostParm->u.pointer.addr[pHostParm->u.pointer.size] to pages. */
2118 rc = vmmdevHGCMPageListWrite(pVMMDevState->pDevIns, pPageListInfo, pHostParm->u.pointer.addr, size);
2119 }
2120 else
2121 rc = VINF_SUCCESS;
2122 }
2123
2124 Log(("vmmdevHGCMCall: PageList guest parameter rc = %Rrc\n", rc));
2125 } break;
2126
2127 default:
2128 {
2129 /* This indicates that the guest request memory was corrupted. */
2130 AssertReleaseMsgFailed(("hgcmCompleted: invalid parameter type %08X\n", pGuestParm->type));
2131 }
2132 }
2133 }
2134# ifdef VBOX_WITH_DTRACE
2135 idFunction = pHGCMCall->u32Function;
2136 idClient = pHGCMCall->u32ClientID;
2137# endif
2138 break;
2139 }
2140#endif /* VBOX_WITH_64_BITS_GUESTS */
2141 case VMMDevReq_HGCMConnect:
2142 {
2143 VMMDevHGCMConnect *pHGCMConnectCopy = (VMMDevHGCMConnect *)(pCmd+1);
2144
2145 /* save the client id in the guest request packet */
2146 VMMDevHGCMConnect *pHGCMConnect = (VMMDevHGCMConnect *)pHeader;
2147 pHGCMConnect->u32ClientID = pHGCMConnectCopy->u32ClientID;
2148 break;
2149 }
2150
2151 default:
2152 /* make gcc happy */
2153 break;
2154 }
2155 }
2156 else
2157 {
2158 /* Command type is wrong. Return error to the guest. */
2159 pHeader->header.rc = rc;
2160 }
2161
2162 /* Mark request as processed. */
2163 pHeader->fu32Flags |= VBOX_HGCM_REQ_DONE;
2164
2165 /* Write back the request */
2166 PDMDevHlpPhysWrite(pVMMDevState->pDevIns, pCmd->GCPhys, pHeader, pCmd->cbSize);
2167
2168 /* Now, when the command was removed from the internal list, notify the guest. */
2169 VMMDevNotifyGuest (pVMMDevState, VMMDEV_EVENT_HGCM);
2170
2171 if ((uint8_t *)pHeader != &au8Prealloc[0])
2172 {
2173 /* Only if it was allocated from heap. */
2174 RTMemFree (pHeader);
2175 }
2176 }
2177
2178 /* Deallocate the command memory. */
2179 if (pCmd->paLinPtrs)
2180 {
2181 RTMemFree (pCmd->paLinPtrs);
2182 }
2183
2184 RTMemFree (pCmd);
2185
2186 VBOXDD_HGCMCALL_COMPLETED_DONE(pCmd, idFunction, idClient, result);
2187 return;
2188}
2189
2190DECLCALLBACK(void) hgcmCompleted (PPDMIHGCMPORT pInterface, int32_t result, PVBOXHGCMCMD pCmd)
2191{
2192 VMMDevState *pVMMDevState = PDMIHGCMPORT_2_VMMDEVSTATE(pInterface);
2193
2194 VBOXDD_HGCMCALL_COMPLETED_REQ(pCmd, result);
2195
2196/** @todo no longer necessary to forward to EMT, but it might be more
2197 * efficient...? */
2198 /* Not safe to execute asynchroneously; forward to EMT */
2199 int rc = VMR3ReqCallVoidNoWait(PDMDevHlpGetVM(pVMMDevState->pDevIns), VMCPUID_ANY,
2200 (PFNRT)hgcmCompletedWorker, 3, pInterface, result, pCmd);
2201 AssertRC(rc);
2202}
2203
2204/* @thread EMT */
2205int vmmdevHGCMSaveState(VMMDevState *pVMMDevState, PSSMHANDLE pSSM)
2206{
2207 /* Save information about pending requests.
2208 * Only GCPtrs are of interest.
2209 */
2210 int rc = VINF_SUCCESS;
2211
2212 LogFlowFunc(("\n"));
2213
2214 /* Compute how many commands are pending. */
2215 uint32_t cCmds = 0;
2216
2217 PVBOXHGCMCMD pIter = pVMMDevState->pHGCMCmdList;
2218
2219 while (pIter)
2220 {
2221 LogFlowFunc (("pIter %p\n", pIter));
2222 cCmds++;
2223 pIter = pIter->pNext;
2224 }
2225
2226 LogFlowFunc(("cCmds = %d\n", cCmds));
2227
2228 /* Save number of commands. */
2229 rc = SSMR3PutU32(pSSM, cCmds);
2230 AssertRCReturn(rc, rc);
2231
2232 if (cCmds > 0)
2233 {
2234 pIter = pVMMDevState->pHGCMCmdList;
2235
2236 while (pIter)
2237 {
2238 PVBOXHGCMCMD pNext = pIter->pNext;
2239
2240 LogFlowFunc (("Saving %RGp, size %d\n", pIter->GCPhys, pIter->cbSize));
2241
2242 /* GC physical address of the guest request. */
2243 rc = SSMR3PutGCPhys(pSSM, pIter->GCPhys);
2244 AssertRCReturn(rc, rc);
2245
2246 /* Request packet size */
2247 rc = SSMR3PutU32(pSSM, pIter->cbSize);
2248 AssertRCReturn(rc, rc);
2249
2250 /*
2251 * Version 9+: save complete information about commands.
2252 */
2253
2254 /* The type of the command. */
2255 rc = SSMR3PutU32(pSSM, (uint32_t)pIter->enmCmdType);
2256 AssertRCReturn(rc, rc);
2257
2258 /* Whether the command was cancelled by the guest. */
2259 rc = SSMR3PutBool(pSSM, pIter->fCancelled);
2260 AssertRCReturn(rc, rc);
2261
2262 /* Linear pointer parameters information. How many pointers. Always 0 if not a call command. */
2263 rc = SSMR3PutU32(pSSM, (uint32_t)pIter->cLinPtrs);
2264 AssertRCReturn(rc, rc);
2265
2266 if (pIter->cLinPtrs > 0)
2267 {
2268 /* How many pages for all linptrs in this command. */
2269 rc = SSMR3PutU32(pSSM, (uint32_t)pIter->cLinPtrPages);
2270 AssertRCReturn(rc, rc);
2271 }
2272
2273 int i;
2274 for (i = 0; i < pIter->cLinPtrs; i++)
2275 {
2276 /* Pointer to descriptions of linear pointers. */
2277 VBOXHGCMLINPTR *pLinPtr = &pIter->paLinPtrs[i];
2278
2279 /* Index of the parameter. */
2280 rc = SSMR3PutU32(pSSM, (uint32_t)pLinPtr->iParm);
2281 AssertRCReturn(rc, rc);
2282
2283 /* Offset in the first physical page of the region. */
2284 rc = SSMR3PutU32(pSSM, pLinPtr->offFirstPage);
2285 AssertRCReturn(rc, rc);
2286
2287 /* How many pages. */
2288 rc = SSMR3PutU32(pSSM, pLinPtr->cPages);
2289 AssertRCReturn(rc, rc);
2290
2291 uint32_t iPage;
2292 for (iPage = 0; iPage < pLinPtr->cPages; iPage++)
2293 {
2294 /* Array of the GC physical addresses for these pages.
2295 * It is assumed that the physical address of the locked resident
2296 * guest page does not change.
2297 */
2298 rc = SSMR3PutGCPhys(pSSM, pLinPtr->paPages[iPage]);
2299 AssertRCReturn(rc, rc);
2300 }
2301 }
2302
2303 /* A reserved field, will allow to extend saved data for a command. */
2304 rc = SSMR3PutU32(pSSM, 0);
2305 AssertRCReturn(rc, rc);
2306
2307 vmmdevHGCMRemoveCommand (pVMMDevState, pIter);
2308
2309 pIter = pNext;
2310 }
2311 }
2312
2313 /* A reserved field, will allow to extend saved data for VMMDevHGCM. */
2314 rc = SSMR3PutU32(pSSM, 0);
2315 AssertRCReturn(rc, rc);
2316
2317 return rc;
2318}
2319
2320/** @thread EMT(0) */
2321int vmmdevHGCMLoadState(VMMDevState *pVMMDevState, PSSMHANDLE pSSM, uint32_t uVersion)
2322{
2323 int rc = VINF_SUCCESS;
2324
2325 LogFlowFunc(("\n"));
2326
2327 /* Read how many commands were pending. */
2328 uint32_t cCmds = 0;
2329 rc = SSMR3GetU32(pSSM, &cCmds);
2330 AssertRCReturn(rc, rc);
2331
2332 LogFlowFunc(("cCmds = %d\n", cCmds));
2333
2334 if (uVersion < 9)
2335 {
2336 /* Only the guest physical address is saved. */
2337 while (cCmds--)
2338 {
2339 RTGCPHYS GCPhys;
2340 uint32_t cbSize;
2341
2342 rc = SSMR3GetGCPhys(pSSM, &GCPhys);
2343 AssertRCReturn(rc, rc);
2344
2345 rc = SSMR3GetU32(pSSM, &cbSize);
2346 AssertRCReturn(rc, rc);
2347
2348 LogFlowFunc (("Restoring %RGp size %x bytes\n", GCPhys, cbSize));
2349
2350 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (sizeof (struct VBOXHGCMCMD));
2351 AssertReturn(pCmd, VERR_NO_MEMORY);
2352
2353 pCmd->enmCmdType = VBOXHGCMCMDTYPE_LOADSTATE; /* This marks the "old" saved command. */
2354
2355 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, cbSize, VBOXHGCMCMDTYPE_LOADSTATE);
2356 }
2357 }
2358 else
2359 {
2360 /*
2361 * Version 9+: Load complete information about commands.
2362 */
2363 uint32_t u32;
2364 bool f;
2365
2366 while (cCmds--)
2367 {
2368 RTGCPHYS GCPhys;
2369 uint32_t cbSize;
2370
2371 /* GC physical address of the guest request. */
2372 rc = SSMR3GetGCPhys(pSSM, &GCPhys);
2373 AssertRCReturn(rc, rc);
2374
2375 /* The request packet size */
2376 rc = SSMR3GetU32(pSSM, &cbSize);
2377 AssertRCReturn(rc, rc);
2378
2379 LogFlowFunc (("Restoring %RGp size %x bytes\n", GCPhys, cbSize));
2380
2381 /* For uVersion <= 12, this was the size of entire command.
2382 * Now the size is recalculated in vmmdevHGCMLoadStateDone.
2383 */
2384 if (uVersion <= 12)
2385 {
2386 rc = SSMR3Skip(pSSM, sizeof (uint32_t));
2387 AssertRCReturn(rc, rc);
2388 }
2389
2390 /* Allocate only VBOXHGCMCMD structure. vmmdevHGCMLoadStateDone will rellocate the command
2391 * with aditional space for parameters and for pointer/pagelists buffer.
2392 */
2393 PVBOXHGCMCMD pCmd = (PVBOXHGCMCMD)RTMemAllocZ (sizeof (VBOXHGCMCMD));
2394 AssertReturn(pCmd, VERR_NO_MEMORY);
2395
2396 /* The type of the command. */
2397 rc = SSMR3GetU32(pSSM, &u32);
2398 AssertRCReturn(rc, rc);
2399 pCmd->enmCmdType = (VBOXHGCMCMDTYPE)u32;
2400
2401 /* Whether the command was cancelled by the guest. */
2402 rc = SSMR3GetBool(pSSM, &f);
2403 AssertRCReturn(rc, rc);
2404 pCmd->fCancelled = f;
2405
2406 /* Linear pointer parameters information. How many pointers. Always 0 if not a call command. */
2407 rc = SSMR3GetU32(pSSM, &u32);
2408 AssertRCReturn(rc, rc);
2409 pCmd->cLinPtrs = u32;
2410
2411 if (pCmd->cLinPtrs > 0)
2412 {
2413 /* How many pages for all linptrs in this command. */
2414 rc = SSMR3GetU32(pSSM, &u32);
2415 AssertRCReturn(rc, rc);
2416 pCmd->cLinPtrPages = u32;
2417
2418 pCmd->paLinPtrs = (VBOXHGCMLINPTR *)RTMemAllocZ ( sizeof (VBOXHGCMLINPTR) * pCmd->cLinPtrs
2419 + sizeof (RTGCPHYS) * pCmd->cLinPtrPages);
2420 AssertReturn(pCmd->paLinPtrs, VERR_NO_MEMORY);
2421
2422 RTGCPHYS *pPages = (RTGCPHYS *)((uint8_t *)pCmd->paLinPtrs + sizeof (VBOXHGCMLINPTR) * pCmd->cLinPtrs);
2423 int cPages = 0;
2424
2425 int i;
2426 for (i = 0; i < pCmd->cLinPtrs; i++)
2427 {
2428 /* Pointer to descriptions of linear pointers. */
2429 VBOXHGCMLINPTR *pLinPtr = &pCmd->paLinPtrs[i];
2430
2431 pLinPtr->paPages = pPages;
2432
2433 /* Index of the parameter. */
2434 rc = SSMR3GetU32(pSSM, &u32);
2435 AssertRCReturn(rc, rc);
2436 pLinPtr->iParm = u32;
2437
2438 /* Offset in the first physical page of the region. */
2439 rc = SSMR3GetU32(pSSM, &u32);
2440 AssertRCReturn(rc, rc);
2441 pLinPtr->offFirstPage = u32;
2442
2443 /* How many pages. */
2444 rc = SSMR3GetU32(pSSM, &u32);
2445 AssertRCReturn(rc, rc);
2446 pLinPtr->cPages = u32;
2447
2448 uint32_t iPage;
2449 for (iPage = 0; iPage < pLinPtr->cPages; iPage++)
2450 {
2451 /* Array of the GC physical addresses for these pages.
2452 * It is assumed that the physical address of the locked resident
2453 * guest page does not change.
2454 */
2455 RTGCPHYS GCPhysPage;
2456 rc = SSMR3GetGCPhys(pSSM, &GCPhysPage);
2457 AssertRCReturn(rc, rc);
2458
2459 /* Verify that the number of loaded pages is valid. */
2460 cPages++;
2461 if (cPages > pCmd->cLinPtrPages)
2462 {
2463 LogRel(("VMMDevHGCM load state failure: cPages %d, expected %d, ptr %d/%d\n",
2464 cPages, pCmd->cLinPtrPages, i, pCmd->cLinPtrs));
2465 return VERR_SSM_UNEXPECTED_DATA;
2466 }
2467
2468 *pPages++ = GCPhysPage;
2469 }
2470 }
2471 }
2472
2473 /* A reserved field, will allow to extend saved data for a command. */
2474 rc = SSMR3GetU32(pSSM, &u32);
2475 AssertRCReturn(rc, rc);
2476
2477 vmmdevHGCMAddCommand (pVMMDevState, pCmd, GCPhys, cbSize, VBOXHGCMCMDTYPE_LOADSTATE);
2478 }
2479
2480 /* A reserved field, will allow to extend saved data for VMMDevHGCM. */
2481 rc = SSMR3GetU32(pSSM, &u32);
2482 AssertRCReturn(rc, rc);
2483 }
2484
2485 return rc;
2486}
2487
2488/* @thread EMT */
2489int vmmdevHGCMLoadStateDone(VMMDevState *pVMMDevState, PSSMHANDLE pSSM)
2490{
2491 LogFlowFunc(("\n"));
2492
2493 /* Reissue pending requests. */
2494 PPDMDEVINS pDevIns = pVMMDevState->pDevIns;
2495
2496 int rc = vmmdevHGCMCmdListLock (pVMMDevState);
2497
2498 if (RT_SUCCESS (rc))
2499 {
2500 /* Start from the current list head and commands loaded from saved state.
2501 * New commands will be inserted at the list head, so they will not be seen by
2502 * this loop.
2503 *
2504 * Note: The list contains only VBOXHGCMCMD structures, place for HGCM parameters
2505 * and for data buffers has not been allocated.
2506 * Command handlers compute the command size and reallocate it before
2507 * resubmitting the command to HGCM services.
2508 * New commands will be inserted to the list.
2509 */
2510 PVBOXHGCMCMD pIter = pVMMDevState->pHGCMCmdList;
2511
2512 pVMMDevState->pHGCMCmdList = NULL; /* Reset the list. Saved commands will be processed and deallocated. */
2513
2514 while (pIter)
2515 {
2516 /* This will remove the command from the list if resubmitting fails. */
2517 bool fHGCMCalled = false;
2518
2519 LogFlowFunc (("pIter %p\n", pIter));
2520
2521 PVBOXHGCMCMD pNext = pIter->pNext;
2522
2523 PVBOXHGCMCMD pCmd = NULL; /* Resubmitted command. */
2524
2525 VMMDevHGCMRequestHeader *requestHeader = (VMMDevHGCMRequestHeader *)RTMemAllocZ (pIter->cbSize);
2526 Assert(requestHeader);
2527 if (requestHeader == NULL)
2528 return VERR_NO_MEMORY;
2529
2530 PDMDevHlpPhysRead(pDevIns, pIter->GCPhys, requestHeader, pIter->cbSize);
2531
2532 /* the structure size must be greater or equal to the header size */
2533 if (requestHeader->header.size < sizeof(VMMDevHGCMRequestHeader))
2534 {
2535 Log(("VMMDev request header size too small! size = %d\n", requestHeader->header.size));
2536 }
2537 else
2538 {
2539 /* check the version of the header structure */
2540 if (requestHeader->header.version != VMMDEV_REQUEST_HEADER_VERSION)
2541 {
2542 Log(("VMMDev: guest header version (0x%08X) differs from ours (0x%08X)\n", requestHeader->header.version, VMMDEV_REQUEST_HEADER_VERSION));
2543 }
2544 else
2545 {
2546 Log(("VMMDev request issued: %d, command type %d\n", requestHeader->header.requestType, pIter->enmCmdType));
2547
2548 /* Use the saved command type. Even if the guest has changed the memory already,
2549 * HGCM should see the same command as it was before saving state.
2550 */
2551 switch (pIter->enmCmdType)
2552 {
2553 case VBOXHGCMCMDTYPE_CONNECT:
2554 {
2555 if (requestHeader->header.size < sizeof(VMMDevHGCMConnect))
2556 {
2557 AssertMsgFailed(("VMMDevReq_HGCMConnect structure has invalid size!\n"));
2558 requestHeader->header.rc = VERR_INVALID_PARAMETER;
2559 }
2560 else if (!pVMMDevState->pHGCMDrv)
2561 {
2562 Log(("VMMDevReq_HGCMConnect HGCM Connector is NULL!\n"));
2563 requestHeader->header.rc = VERR_NOT_SUPPORTED;
2564 }
2565 else
2566 {
2567 VMMDevHGCMConnect *pHGCMConnect = (VMMDevHGCMConnect *)requestHeader;
2568
2569 Log(("VMMDevReq_HGCMConnect\n"));
2570
2571 requestHeader->header.rc = vmmdevHGCMConnectSaved (pVMMDevState, pHGCMConnect, pIter->GCPhys, &fHGCMCalled, pIter, &pCmd);
2572 }
2573 break;
2574 }
2575
2576 case VBOXHGCMCMDTYPE_DISCONNECT:
2577 {
2578 if (requestHeader->header.size < sizeof(VMMDevHGCMDisconnect))
2579 {
2580 AssertMsgFailed(("VMMDevReq_HGCMDisconnect structure has invalid size!\n"));
2581 requestHeader->header.rc = VERR_INVALID_PARAMETER;
2582 }
2583 else if (!pVMMDevState->pHGCMDrv)
2584 {
2585 Log(("VMMDevReq_HGCMDisconnect HGCM Connector is NULL!\n"));
2586 requestHeader->header.rc = VERR_NOT_SUPPORTED;
2587 }
2588 else
2589 {
2590 VMMDevHGCMDisconnect *pHGCMDisconnect = (VMMDevHGCMDisconnect *)requestHeader;
2591
2592 Log(("VMMDevReq_VMMDevHGCMDisconnect\n"));
2593 requestHeader->header.rc = vmmdevHGCMDisconnectSaved (pVMMDevState, pHGCMDisconnect, pIter->GCPhys, &fHGCMCalled, pIter, &pCmd);
2594 }
2595 break;
2596 }
2597
2598 case VBOXHGCMCMDTYPE_CALL:
2599 {
2600 if (requestHeader->header.size < sizeof(VMMDevHGCMCall))
2601 {
2602 AssertMsgFailed(("VMMDevReq_HGCMCall structure has invalid size!\n"));
2603 requestHeader->header.rc = VERR_INVALID_PARAMETER;
2604 }
2605 else if (!pVMMDevState->pHGCMDrv)
2606 {
2607 Log(("VMMDevReq_HGCMCall HGCM Connector is NULL!\n"));
2608 requestHeader->header.rc = VERR_NOT_SUPPORTED;
2609 }
2610 else
2611 {
2612 VMMDevHGCMCall *pHGCMCall = (VMMDevHGCMCall *)requestHeader;
2613
2614 Log(("VMMDevReq_HGCMCall: sizeof (VMMDevHGCMRequest) = %04X\n", sizeof (VMMDevHGCMCall)));
2615
2616 Log(("%.*Rhxd\n", requestHeader->header.size, requestHeader));
2617
2618#ifdef VBOX_WITH_64_BITS_GUESTS
2619 bool f64Bits = (requestHeader->header.requestType == VMMDevReq_HGCMCall64);
2620#else
2621 bool f64Bits = false;
2622#endif /* VBOX_WITH_64_BITS_GUESTS */
2623 requestHeader->header.rc = vmmdevHGCMCallSaved (pVMMDevState, pHGCMCall, pIter->GCPhys, requestHeader->header.size, f64Bits, &fHGCMCalled, pIter, &pCmd);
2624 }
2625 break;
2626 }
2627 case VBOXHGCMCMDTYPE_LOADSTATE:
2628 {
2629 /* Old saved state. */
2630 switch (requestHeader->header.requestType)
2631 {
2632 case VMMDevReq_HGCMConnect:
2633 {
2634 if (requestHeader->header.size < sizeof(VMMDevHGCMConnect))
2635 {
2636 AssertMsgFailed(("VMMDevReq_HGCMConnect structure has invalid size!\n"));
2637 requestHeader->header.rc = VERR_INVALID_PARAMETER;
2638 }
2639 else if (!pVMMDevState->pHGCMDrv)
2640 {
2641 Log(("VMMDevReq_HGCMConnect HGCM Connector is NULL!\n"));
2642 requestHeader->header.rc = VERR_NOT_SUPPORTED;
2643 }
2644 else
2645 {
2646 VMMDevHGCMConnect *pHGCMConnect = (VMMDevHGCMConnect *)requestHeader;
2647
2648 Log(("VMMDevReq_HGCMConnect\n"));
2649
2650 requestHeader->header.rc = vmmdevHGCMConnect (pVMMDevState, pHGCMConnect, pIter->GCPhys);
2651 }
2652 break;
2653 }
2654
2655 case VMMDevReq_HGCMDisconnect:
2656 {
2657 if (requestHeader->header.size < sizeof(VMMDevHGCMDisconnect))
2658 {
2659 AssertMsgFailed(("VMMDevReq_HGCMDisconnect structure has invalid size!\n"));
2660 requestHeader->header.rc = VERR_INVALID_PARAMETER;
2661 }
2662 else if (!pVMMDevState->pHGCMDrv)
2663 {
2664 Log(("VMMDevReq_HGCMDisconnect HGCM Connector is NULL!\n"));
2665 requestHeader->header.rc = VERR_NOT_SUPPORTED;
2666 }
2667 else
2668 {
2669 VMMDevHGCMDisconnect *pHGCMDisconnect = (VMMDevHGCMDisconnect *)requestHeader;
2670
2671 Log(("VMMDevReq_VMMDevHGCMDisconnect\n"));
2672 requestHeader->header.rc = vmmdevHGCMDisconnect (pVMMDevState, pHGCMDisconnect, pIter->GCPhys);
2673 }
2674 break;
2675 }
2676
2677#ifdef VBOX_WITH_64_BITS_GUESTS
2678 case VMMDevReq_HGCMCall64:
2679 case VMMDevReq_HGCMCall32:
2680#else
2681 case VMMDevReq_HGCMCall:
2682#endif /* VBOX_WITH_64_BITS_GUESTS */
2683 {
2684 if (requestHeader->header.size < sizeof(VMMDevHGCMCall))
2685 {
2686 AssertMsgFailed(("VMMDevReq_HGCMCall structure has invalid size!\n"));
2687 requestHeader->header.rc = VERR_INVALID_PARAMETER;
2688 }
2689 else if (!pVMMDevState->pHGCMDrv)
2690 {
2691 Log(("VMMDevReq_HGCMCall HGCM Connector is NULL!\n"));
2692 requestHeader->header.rc = VERR_NOT_SUPPORTED;
2693 }
2694 else
2695 {
2696 VMMDevHGCMCall *pHGCMCall = (VMMDevHGCMCall *)requestHeader;
2697
2698 Log(("VMMDevReq_HGCMCall: sizeof (VMMDevHGCMRequest) = %04X\n", sizeof (VMMDevHGCMCall)));
2699
2700 Log(("%.*Rhxd\n", requestHeader->header.size, requestHeader));
2701
2702#ifdef VBOX_WITH_64_BITS_GUESTS
2703 bool f64Bits = (requestHeader->header.requestType == VMMDevReq_HGCMCall64);
2704#else
2705 bool f64Bits = false;
2706#endif /* VBOX_WITH_64_BITS_GUESTS */
2707 requestHeader->header.rc = vmmdevHGCMCall (pVMMDevState, pHGCMCall, requestHeader->header.size, pIter->GCPhys, f64Bits);
2708 }
2709 break;
2710 }
2711 default:
2712 AssertMsgFailed(("Unknown request type %x during LoadState\n", requestHeader->header.requestType));
2713 LogRel(("VMMDEV: Ignoring unknown request type %x during LoadState\n", requestHeader->header.requestType));
2714 }
2715 } break;
2716
2717 default:
2718 AssertMsgFailed(("Unknown request type %x during LoadState\n", requestHeader->header.requestType));
2719 LogRel(("VMMDEV: Ignoring unknown request type %x during LoadState\n", requestHeader->header.requestType));
2720 }
2721 }
2722 }
2723
2724 if (pIter->enmCmdType == VBOXHGCMCMDTYPE_LOADSTATE)
2725 {
2726 /* Old saved state. */
2727
2728 /* Write back the request */
2729 PDMDevHlpPhysWrite(pDevIns, pIter->GCPhys, requestHeader, pIter->cbSize);
2730 RTMemFree(requestHeader);
2731 requestHeader = NULL;
2732 }
2733 else
2734 {
2735 if (!fHGCMCalled)
2736 {
2737 /* HGCM was not called. Return the error to the guest. Guest may try to repeat the call. */
2738 requestHeader->header.rc = VERR_TRY_AGAIN;
2739 requestHeader->fu32Flags |= VBOX_HGCM_REQ_DONE;
2740 }
2741
2742 /* Write back the request */
2743 PDMDevHlpPhysWrite(pDevIns, pIter->GCPhys, requestHeader, pIter->cbSize);
2744 RTMemFree(requestHeader);
2745 requestHeader = NULL;
2746
2747 if (!fHGCMCalled)
2748 {
2749 /* HGCM was not called. Deallocate the current command and then notify guest. */
2750 if (pCmd)
2751 {
2752 vmmdevHGCMRemoveCommand (pVMMDevState, pCmd);
2753
2754 if (pCmd->paLinPtrs != NULL)
2755 {
2756 RTMemFree(pCmd->paLinPtrs);
2757 }
2758
2759 RTMemFree(pCmd);
2760 pCmd = NULL;
2761 }
2762
2763 VMMDevNotifyGuest (pVMMDevState, VMMDEV_EVENT_HGCM);
2764 }
2765 }
2766
2767 /* Deallocate the saved command structure. */
2768 if (pIter->paLinPtrs != NULL)
2769 {
2770 RTMemFree(pIter->paLinPtrs);
2771 }
2772
2773 RTMemFree(pIter);
2774
2775 pIter = pNext;
2776 }
2777
2778 vmmdevHGCMCmdListUnlock (pVMMDevState);
2779 }
2780
2781 return rc;
2782}
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