VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest.cpp@ 6118

Last change on this file since 6118 was 6033, checked in by vboxsync, 17 years ago

vboxadd changes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 43.6 KB
Line 
1/** $Id: */
2/** @file
3 * VBoxGuest - Guest Additions Driver.
4 */
5
6/*
7 * Copyright (C) 2007 innotek GmbH
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19
20/*******************************************************************************
21* Header Files *
22*******************************************************************************/
23#define LOG_GROUP LOG_GROUP_DEFAULT
24#include "VBoxGuestInternal.h"
25#include <VBox/VBoxDev.h> /* for VMMDEV_RAM_SIZE */
26#include <VBox/log.h>
27#include <iprt/mem.h>
28#include <iprt/time.h>
29#include <iprt/memobj.h>
30#include <iprt/asm.h>
31#include <iprt/string.h>
32#include <iprt/process.h>
33#include <iprt/assert.h>
34#include <iprt/param.h>
35#ifdef VBOX_HGCM
36# include <iprt/thread.h>
37#endif
38
39
40/*******************************************************************************
41* Internal Functions *
42*******************************************************************************/
43#ifdef VBOX_HGCM
44static DECLCALLBACK(void) VBoxGuestHGCMAsyncWaitCallback(VMMDevHGCMRequestHeader *pHdrNonVolatile, void *pvUser, uint32_t u32User);
45#endif
46
47
48
49/**
50 * Reserves memory in which the VMM can relocate any guest mappings
51 * that are floating around.
52 *
53 * This operation is a little bit tricky since the VMM might not accept
54 * just any address because of address clashes between the three contexts
55 * it operates in, so use a small stack to perform this operation.
56 *
57 * @returns VBox status code (ignored).
58 * @param pDevExt The device extension.
59 */
60static int vboxGuestInitFixateGuestMappings(PVBOXGUESTDEVEXT pDevExt)
61{
62 /** @todo implement this using RTR0MemObjReserveKernel() (it needs to be implemented everywhere too). */
63 return VINF_SUCCESS;
64}
65
66
67/**
68 * Initializes the interrupt filter mask.
69 *
70 * This will ASSUME that we're the ones in carge over the mask, so
71 * we'll simply clear all bits we don't set.
72 *
73 * @returns VBox status code (ignored).
74 * @param pDevExt The device extension.
75 * @param fMask The new mask.
76 */
77static int vboxGuestInitFilterMask(PVBOXGUESTDEVEXT pDevExt, uint32_t fMask)
78{
79 VMMDevCtlGuestFilterMask *pReq;
80 int rc = VbglGRAlloc((VMMDevRequestHeader **)&pReq, sizeof(*pReq), VMMDevReq_CtlGuestFilterMask);
81 if (RT_SUCCESS(rc))
82 {
83 pReq->u32OrMask = fMask;
84 pReq->u32NotMask = ~fMask; /* It's an AND mask. */
85 rc = VbglGRPerform(&pReq->header);
86 if ( RT_FAILURE(rc)
87 || RT_FAILURE(pReq->header.rc))
88 LogRel(("vboxGuestInitCtlFilterMask: failed with rc=%Rrc and VMMDev rc=%Rrc\n",
89 rc, pReq->header.rc));
90 VbglGRFree(&pReq->header);
91 }
92 return rc;
93}
94
95
96/**
97 * Report guest information to the VMMDev.
98 *
99 * @returns VBox status code.
100 * @param pDevExt The device extension.
101 * @param enmOSType The OS type to report.
102 */
103static int vboxGuestInitReportGuestInfo(PVBOXGUESTDEVEXT pDevExt, VBOXOSTYPE enmOSType)
104{
105 VMMDevReportGuestInfo *pReq;
106 int rc = VbglGRAlloc((VMMDevRequestHeader **)&pReq, sizeof(*pReq), VMMDevReq_ReportGuestInfo);
107 if (RT_SUCCESS(rc))
108 {
109 pReq->guestInfo.additionsVersion = VMMDEV_VERSION;
110 pReq->guestInfo.osType = enmOSType;
111 rc = VbglGRPerform(&pReq->header);
112 if ( RT_FAILURE(rc)
113 || RT_FAILURE(pReq->header.rc))
114 LogRel(("vboxGuestInitReportGuestInfo: failed with rc=%Rrc and VMMDev rc=%Rrc\n",
115 rc, pReq->header.rc));
116 VbglGRFree(&pReq->header);
117 }
118 return rc;
119}
120
121
122/**
123 * Initializes the VBoxGuest device extension when the
124 * device driver is loaded.
125 *
126 * The native code locates the VMMDev on the PCI bus and retrieve
127 * the MMIO and I/O port ranges, this function will take care of
128 * mapping the MMIO memory (if present). Upon successful return
129 * the native code should set up the interrupt handler.
130 *
131 * @returns VBox status code.
132 *
133 * @param pDevExt The device extension. Allocated by the native code.
134 * @param IOPortBase The base of the I/O port range.
135 * @param pvMMIOBase The base of the MMIO memory mapping.
136 * This is optional, pass NULL if not present.
137 * @param cbMMIO The size of the MMIO memory mapping.
138 * This is optional, pass 0 if not present.
139 * @param enmOSType The guest OS type to report to the VMMDev.
140 */
141int VBoxGuestInitDevExt(PVBOXGUESTDEVEXT pDevExt, uint16_t IOPortBase,
142 void *pvMMIOBase, uint32_t cbMMIO, VBOXOSTYPE enmOSType)
143{
144 int rc, rc2;
145
146 /*
147 * Initalize the data.
148 */
149 pDevExt->IOPortBase = IOPortBase;
150 pDevExt->pVMMDevMemory = NULL;
151 pDevExt->pIrqAckEvents = NULL;
152 pDevExt->WaitList.pHead = NULL;
153 pDevExt->WaitList.pTail = NULL;
154#ifdef VBOX_HGCM
155 pDevExt->HGCMWaitList.pHead = NULL;
156 pDevExt->HGCMWaitList.pTail = NULL;
157#endif
158 pDevExt->FreeList.pHead = NULL;
159 pDevExt->FreeList.pTail = NULL;
160 pDevExt->f32PendingEvents = 0;
161 pDevExt->u32ClipboardClientId = 0;
162
163 /*
164 * If there is an MMIO region validate the version and size.
165 */
166 if (pvMMIOBase)
167 {
168 Assert(cbMMIO);
169 VMMDevMemory *pVMMDev = (VMMDevMemory *)pvMMIOBase;
170 if ( pVMMDev->u32Version == VMMDEV_MEMORY_VERSION
171 && pVMMDev->u32Size >= 32
172 && pVMMDev->u32Size <= cbMMIO)
173 {
174 pDevExt->pVMMDevMemory = pVMMDev;
175 Log(("VBoxGuestInitDevExt: VMMDevMemory: mapping=%p size=%#RX32 (%#RX32) version=%#RX32\n",
176 pVMMDev, pVMMDev->u32Size, cbMMIO, pVMMDev->u32Version));
177 }
178 else /* try live without it. */
179 LogRel(("VBoxGuestInitDevExt: Bogus VMMDev memory; u32Version=%RX32 (expected %RX32) u32Size=%RX32 (expected <= %RX32)\n",
180 pVMMDev->u32Version, VMMDEV_MEMORY_VERSION, pVMMDev->u32Size, cbMMIO));
181 }
182
183 /*
184 * Create the wait and seesion spinlocks.
185 */
186 rc = RTSpinlockCreate(&pDevExt->WaitSpinlock);
187 if (RT_SUCCESS(rc))
188 rc = RTSpinlockCreate(&pDevExt->SessionSpinlock);
189 if (RT_FAILURE(rc))
190 {
191 Log(("VBoxGuestInitDevExt: failed to spinlock, rc=%d!\n", rc));
192 if (pDevExt->WaitSpinlock != NIL_RTSPINLOCK)
193 RTSpinlockDestroy(pDevExt->WaitSpinlock);
194 return rc;
195 }
196
197 /*
198 * Initialize the guest library and report the guest info back to VMMDev,
199 * set the interrupt control filter mask, and fixate the guest mappings
200 * made by the VMM.
201 */
202 rc = VbglInit(pDevExt->IOPortBase, (VMMDevMemory *)pDevExt->pVMMDevMemory);
203 if (RT_SUCCESS(rc))
204 {
205 rc = VbglGRAlloc((VMMDevRequestHeader **)&pDevExt->pIrqAckEvents, sizeof(VMMDevEvents), VMMDevReq_AcknowledgeEvents);
206 if (RT_SUCCESS(rc))
207 {
208 rc = vboxGuestInitReportGuestInfo(pDevExt, enmOSType);
209 if (RT_SUCCESS(rc))
210 {
211#ifdef VBOX_HGCM
212 rc = vboxGuestInitFilterMask(pDevExt, VMMDEV_EVENT_HGCM);
213#else
214 rc = vboxGuestInitFilterMask(pDevExt, 0);
215#endif
216 if (RT_SUCCESS(rc))
217 {
218 vboxGuestInitFixateGuestMappings(pDevExt);
219 Log(("VBoxGuestInitDevExt: returns success\n"));
220 return VINF_SUCCESS;
221 }
222 }
223
224 /* failure cleanup */
225 }
226 else
227 Log(("VBoxGuestInitDevExt: VBoxGRAlloc failed, rc=%Rrc\n", rc));
228
229 VbglTerminate();
230 }
231 else
232 Log(("VBoxGuestInitDevExt: VbglInit failed, rc=%Rrc\n", rc));
233
234 rc2 = RTSpinlockDestroy(pDevExt->WaitSpinlock); AssertRC(rc2);
235 rc2 = RTSpinlockDestroy(pDevExt->SessionSpinlock); AssertRC(rc2);
236 return rc; /* (failed) */
237}
238
239
240/**
241 * Deletes all the items in a wait chain.
242 * @param pWait The head of the chain.
243 */
244static void VBoxGuestDeleteWaitList(PVBOXGUESTWAITLIST pList)
245{
246 while (pList->pHead)
247 {
248 PVBOXGUESTWAIT pWait = pList->pHead;
249 pList->pHead = pWait->pNext;
250
251 pWait->pNext = NULL;
252 pWait->pPrev = NULL;
253 int rc2 = RTSemEventMultiDestroy(pWait->Event); AssertRC(rc2);
254 pWait->Event = NIL_RTSEMEVENTMULTI;
255 RTMemFree(pWait);
256 }
257 pList->pHead = NULL;
258 pList->pTail = NULL;
259}
260
261
262/**
263 * Destroys the VBoxGuest device extension.
264 *
265 * The native code should call this before the driver is loaded,
266 * but don't call this on shutdown.
267 *
268 * @param pDevExt The device extension.
269 */
270void VBoxGuestDeleteDevExt(PVBOXGUESTDEVEXT pDevExt)
271{
272 int rc2;
273 Log(("VBoxGuestDeleteDevExt:\n"));
274
275 rc2 = RTSpinlockDestroy(pDevExt->WaitSpinlock); AssertRC(rc2);
276
277 VBoxGuestDeleteWaitList(&pDevExt->WaitList);
278#ifdef VBOX_HGCM
279 VBoxGuestDeleteWaitList(&pDevExt->HGCMWaitList);
280#endif
281 VBoxGuestDeleteWaitList(&pDevExt->FreeList);
282
283 VbglTerminate();
284
285 pDevExt->pVMMDevMemory = NULL;
286
287 pDevExt->IOPortBase = 0;
288 pDevExt->pIrqAckEvents = NULL;
289}
290
291
292/**
293 * Creates a VBoxGuest user session.
294 *
295 * The native code calls this when a ring-3 client opens the device.
296 * Use VBoxGuestCreateKernelSession when a ring-0 client connects.
297 *
298 * @returns VBox status code.
299 * @param pDevExt The device extension.
300 * @param ppSession Where to store the session on success.
301 */
302int VBoxGuestCreateUserSession(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION *ppSession)
303{
304 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)RTMemAllocZ(sizeof(*pSession));
305 if (RT_UNLIKELY(!pSession))
306 {
307 LogRel(("VBoxGuestCreateUserSession: no memory!\n"));
308 return VERR_NO_MEMORY;
309 }
310
311 pSession->Process = RTProcSelf();
312 pSession->R0Process = RTR0ProcHandleSelf();
313 pSession->pDevExt = pDevExt;
314
315 *ppSession = pSession;
316 LogFlow(("VBoxGuestCreateUserSession: pSession=%p proc=%RTproc (%d) r0proc=%p\n",
317 pSession, pSession->Process, (int)pSession->Process, (uintptr_t)pSession->R0Process)); /** @todo %RTr0proc */
318 return VINF_SUCCESS;
319}
320
321
322/**
323 * Creates a VBoxGuest kernel session.
324 *
325 * The native code calls this when a ring-0 client connects to the device.
326 * Use VBoxGuestCreateUserSession when a ring-3 client opens the device.
327 *
328 * @returns VBox status code.
329 * @param pDevExt The device extension.
330 * @param ppSession Where to store the session on success.
331 */
332int VBoxGuestCreateKernelSession(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION *ppSession)
333{
334 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)RTMemAllocZ(sizeof(*pSession));
335 if (RT_UNLIKELY(!pSession))
336 {
337 LogRel(("VBoxGuestCreateKernelSession: no memory!\n"));
338 return VERR_NO_MEMORY;
339 }
340
341 pSession->Process = NIL_RTPROCESS;
342 pSession->R0Process = NIL_RTR0PROCESS;
343 pSession->pDevExt = pDevExt;
344
345 *ppSession = pSession;
346 LogFlow(("VBoxGuestCreateKernelSession: pSession=%p proc=%RTproc (%d) r0proc=%p\n",
347 pSession, pSession->Process, (int)pSession->Process, (uintptr_t)pSession->R0Process)); /** @todo %RTr0proc */
348 return VINF_SUCCESS;
349}
350
351
352
353/**
354 * Closes a VBoxGuest session.
355 *
356 * @param pDevExt The device extension.
357 * @param pSession The session to close (and free).
358 */
359void VBoxGuestCloseSession(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION pSession)
360{
361 Log(("VBoxGuestCloseSession: pSession=%p proc=%RTproc (%d) r0proc=%p\n",
362 pSession, pSession->Process, (int)pSession->Process, (uintptr_t)pSession->R0Process)); /** @todo %RTr0proc */
363
364#ifdef VBOX_HGCM
365 for (unsigned i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
366 if (pSession->aHGCMClientIds[i])
367 {
368 VBoxGuestHGCMDisconnectInfo Info;
369 Info.result = 0;
370 Info.u32ClientID = pSession->aHGCMClientIds[i];
371 pSession->aHGCMClientIds[i] = 0;
372 Log(("VBoxGuestCloseSession: disconnecting client id %#RX32\n", Info.u32ClientID));
373 VbglHGCMDisconnect(&Info, VBoxGuestHGCMAsyncWaitCallback, pDevExt, false /* uninterruptible */);
374 }
375#endif
376
377 pSession->pDevExt = NULL;
378 pSession->Process = NIL_RTPROCESS;
379 pSession->R0Process = NIL_RTR0PROCESS;
380 RTMemFree(pSession);
381}
382
383
384/**
385 * Links the wait-for-event entry into the tail of the given list.
386 *
387 * @param pList The list to link it into.
388 * @param pWait The wait for event entry to append.
389 */
390DECLINLINE(void) VBoxGuestWaitAppend(PVBOXGUESTWAITLIST pList, PVBOXGUESTWAIT pWait)
391{
392 const PVBOXGUESTWAIT pTail = pList->pTail;
393 pWait->pNext = NULL;
394 pWait->pPrev = pTail;
395 if (pTail)
396 pTail->pNext = pWait;
397 else
398 pList->pHead = pWait;
399 pList->pTail = pWait;
400}
401
402
403/**
404 * Unlinks the wait-for-event entry.
405 *
406 * @param pList The list to unlink it from.
407 * @param pWait The wait for event entry to unlink.
408 */
409DECLINLINE(void) VBoxGuestWaitUnlink(PVBOXGUESTWAITLIST pList, PVBOXGUESTWAIT pWait)
410{
411 const PVBOXGUESTWAIT pPrev = pWait->pPrev;
412 const PVBOXGUESTWAIT pNext = pWait->pNext;
413 if (pNext)
414 pNext->pPrev = pPrev;
415 else
416 pList->pTail = pPrev;
417 if (pPrev)
418 pPrev->pNext = pNext;
419 else
420 pList->pHead = pNext;
421}
422
423
424/**
425 * Allocates a wiat-for-event entry.
426 *
427 * @returns The wait-for-event entry.
428 * @param pDevExt The device extension.
429 */
430static PVBOXGUESTWAIT VBoxGuestWaitAlloc(PVBOXGUESTDEVEXT pDevExt)
431{
432 /*
433 * Allocate it one way or the other.
434 */
435 PVBOXGUESTWAIT pWait = pDevExt->FreeList.pTail;
436 if (pWait)
437 {
438 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
439 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
440
441 pWait = pDevExt->FreeList.pTail;
442 if (pWait)
443 VBoxGuestWaitUnlink(&pDevExt->FreeList, pWait);
444
445 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
446 }
447 if (!pWait)
448 {
449 static unsigned s_cErrors = 0;
450
451 pWait = (PVBOXGUESTWAIT)RTMemAlloc(sizeof(*pWait));
452 if (!pWait)
453 {
454 if (s_cErrors++ < 32)
455 LogRel(("VBoxGuestWaitAlloc: out-of-memory!\n"));
456 return NULL;
457 }
458
459 int rc = RTSemEventMultiCreate(&pWait->Event);
460 if (RT_FAILURE(rc))
461 {
462 if (s_cErrors++ < 32)
463 LogRel(("VBoxGuestCommonIOCtl: RTSemEventMultiCreate failed with rc=%Rrc!\n", rc));
464 RTMemFree(pWait);
465 return NULL;
466 }
467 }
468
469 /*
470 * Zero members just as an precaution.
471 */
472 pWait->pNext = NULL;
473 pWait->pPrev = NULL;
474 pWait->fReqEvents = 0;
475 pWait->fResEvents = 0;
476#ifdef VBOX_HGCM
477 pWait->pHGCMReq = NULL;
478#endif
479 RTSemEventMultiReset(pWait->Event);
480 return pWait;
481}
482
483
484/**
485 * Frees the wait-for-event entry.
486 * The caller must own the wait spinlock!
487 *
488 * @param pDevExt The device extension.
489 * @param pWait The wait-for-event entry to free.
490 */
491static void VBoxGuestWaitFreeLocked(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTWAIT pWait)
492{
493 pWait->fReqEvents = 0;
494 pWait->fResEvents = 0;
495#ifdef VBOX_HGCM
496 pWait->pHGCMReq = NULL;
497#endif
498 VBoxGuestWaitAppend(&pDevExt->FreeList, pWait);
499}
500
501
502/**
503 * Frees the wait-for-event entry.
504 *
505 * @param pDevExt The device extension.
506 * @param pWait The wait-for-event entry to free.
507 */
508static void VBoxGuestWaitFreeUnlocked(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTWAIT pWait)
509{
510 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
511 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
512 VBoxGuestWaitFreeLocked(pDevExt, pWait);
513 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
514}
515
516
517/**
518 * Implements the fast (no input or output) type of IOCtls.
519 *
520 * This is currently just a placeholder stub inherited from the support driver code.
521 *
522 * @returns VBox status code.
523 * @param iFunction The IOCtl function number.
524 * @param pDevExt The device extension.
525 * @param pSession The session.
526 */
527int VBoxGuestCommonIOCtlFast(unsigned iFunction, PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION pSession)
528{
529 Log(("VBoxGuestCommonIOCtlFast: iFunction=%#x pDevExt=%p pSession=%p\n", iFunction, pDevExt, pSession));
530
531 return VERR_NOT_SUPPORTED;
532}
533
534
535
536static int VBoxGuestCommonIOCtl_GetVMMDevPort(PVBOXGUESTDEVEXT pDevExt, VBoxGuestPortInfo *pInfo, size_t *pcbDataReturned)
537{
538 Log(("VBoxGuestCommonIOCtl: GETVMMDEVPORT\n"));
539 pInfo->portAddress = pDevExt->IOPortBase;
540 pInfo->pVMMDevMemory = (VMMDevMemory *)pDevExt->pVMMDevMemory;
541 if (pcbDataReturned)
542 *pcbDataReturned = sizeof(*pInfo);
543 return VINF_SUCCESS;
544}
545
546
547/**
548 * Worker VBoxGuestCommonIOCtl_WaitEvent.
549 * The caller enters the spinlock, we may or may not leave it.
550 *
551 * @returns VINF_SUCCESS if we've left the spinlock and can return immediately.
552 */
553DECLINLINE(int) WaitEventCheckCondition(PVBOXGUESTDEVEXT pDevExt, VBoxGuestWaitEventInfo *pInfo,
554 int iEvent, const uint32_t fReqEvents, PRTSPINLOCKTMP pTmp)
555{
556 uint32_t fMatches = pDevExt->f32PendingEvents & fReqEvents;
557 if (fMatches)
558 {
559 ASMAtomicAndU32(&pDevExt->f32PendingEvents, ~fMatches);
560 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, pTmp);
561
562 pInfo->u32EventFlagsOut = fMatches;
563 pInfo->u32Result = VBOXGUEST_WAITEVENT_OK;
564 if (fReqEvents & ~((uint32_t)1 << iEvent))
565 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns %#x\n", pInfo->u32EventFlagsOut));
566 else
567 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns %#x/%d\n", pInfo->u32EventFlagsOut, iEvent));
568 return VINF_SUCCESS;
569 }
570 return VERR_TIMEOUT;
571}
572
573
574static int VBoxGuestCommonIOCtl_WaitEvent(PVBOXGUESTDEVEXT pDevExt, VBoxGuestWaitEventInfo *pInfo, size_t *pcbDataReturned,
575 bool fInterruptible)
576{
577 pInfo->u32EventFlagsOut = 0;
578 pInfo->u32Result = VBOXGUEST_WAITEVENT_ERROR;
579 if (pcbDataReturned)
580 *pcbDataReturned = sizeof(*pInfo);
581
582 /*
583 * Copy and verify the input mask.
584 */
585 const uint32_t fReqEvents = pInfo->u32EventMaskIn;
586 int iEvent = ASMBitFirstSetU32(fReqEvents) - 1;
587 if (RT_UNLIKELY(iEvent < 0))
588 {
589 Log(("VBoxGuestCommonIOCtl: WAITEVENT: Invalid input mask %#x!!\n", fReqEvents));
590 return VERR_INVALID_PARAMETER;
591 }
592
593 /*
594 * Check the condition up front, before doing the wait-for-event allocations.
595 */
596 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
597 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
598 int rc = WaitEventCheckCondition(pDevExt, pInfo, iEvent, fReqEvents, &Tmp);
599 if (rc == VINF_SUCCESS)
600 return rc;
601 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
602
603 if (!pInfo->u32TimeoutIn)
604 {
605 pInfo->u32Result = VBOXGUEST_WAITEVENT_TIMEOUT;
606 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns VINF_TIMEOUT\n"));
607 return VERR_TIMEOUT;
608 }
609
610 PVBOXGUESTWAIT pWait = VBoxGuestWaitAlloc(pDevExt);
611 if (!pWait)
612 return VERR_NO_MEMORY;
613 pWait->fReqEvents = fReqEvents;
614
615 /*
616 * We've got the wait entry now, re-enter the spinlock and check for the condition.
617 * If the wait condition is met, return.
618 * Otherwise enter into the list and go to sleep waiting for the ISR to signal us.
619 */
620 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
621 rc = WaitEventCheckCondition(pDevExt, pInfo, iEvent, fReqEvents, &Tmp);
622 if (rc == VINF_SUCCESS)
623 {
624 VBoxGuestWaitFreeUnlocked(pDevExt, pWait);
625 return rc;
626 }
627 VBoxGuestWaitAppend(&pDevExt->WaitList, pWait);
628 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
629
630 if (fInterruptible)
631 rc = RTSemEventMultiWaitNoResume(pWait->Event,
632 pInfo->u32TimeoutIn == UINT32_MAX ? RT_INDEFINITE_WAIT : pInfo->u32TimeoutIn);
633 else
634 rc = RTSemEventMultiWait(pWait->Event,
635 pInfo->u32TimeoutIn == UINT32_MAX ? RT_INDEFINITE_WAIT : pInfo->u32TimeoutIn);
636
637 /*
638 * There is one special case here and that's when the semaphore is
639 * destroyed upon device driver unload. This shouldn't happen of course,
640 * but in case it does, just get out of here ASAP.
641 */
642 if (rc == VERR_SEM_DESTROYED)
643 return rc;
644
645 /*
646 * Unlink the wait item and dispose of it.
647 */
648 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
649 VBoxGuestWaitUnlink(&pDevExt->WaitList, pWait);
650 const uint32_t fResEvents = pWait->fResEvents;
651 VBoxGuestWaitFreeLocked(pDevExt, pWait);
652 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
653
654 /*
655 * Now deal with the return code.
656 */
657 if (fResEvents)
658 {
659 pInfo->u32EventFlagsOut = fResEvents;
660 pInfo->u32Result = VBOXGUEST_WAITEVENT_OK;
661 if (fReqEvents & ~((uint32_t)1 << iEvent))
662 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns %#x\n", pInfo->u32EventFlagsOut));
663 else
664 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns %#x/%d\n", pInfo->u32EventFlagsOut, iEvent));
665 rc = VINF_SUCCESS;
666 }
667 else if (rc == VERR_TIMEOUT)
668 {
669 pInfo->u32Result = VBOXGUEST_WAITEVENT_TIMEOUT;
670 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns VINF_TIMEOUT\n"));
671 }
672 else if (rc == VERR_INTERRUPTED)
673 {
674 pInfo->u32Result = VBOXGUEST_WAITEVENT_INTERRUPTED;
675 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns VERR_INTERRUPTED\n"));
676 }
677 else
678 {
679 if (RT_SUCCESS(rc))
680 {
681 static unsigned s_cErrors = 0;
682 if (s_cErrors++ < 32)
683 LogRel(("VBoxGuestCommonIOCtl: WAITEVENT: returns %Rrc but no events!\n", rc));
684 rc = VERR_INTERNAL_ERROR;
685 }
686 pInfo->u32Result = VBOXGUEST_WAITEVENT_ERROR;
687 Log(("VBoxGuestCommonIOCtl: WAITEVENT: returns %Rrc\n", rc));
688 }
689
690 return rc;
691}
692
693
694static int VBoxGuestCommonIOCtl_VMMRequest(PVBOXGUESTDEVEXT pDevExt, VMMDevRequestHeader *pReqHdr,
695 size_t cbData, size_t *pcbDataReturned)
696{
697 Log(("VBoxGuestCommonIOCtl: VMMREQUEST type %d\n", pReqHdr->requestType));
698
699 /*
700 * Validate the header and request size.
701 */
702 const uint32_t cbReq = pReqHdr->size;
703 const uint32_t cbMinSize = vmmdevGetRequestSize(pReqHdr->requestType);
704 if (cbReq < cbMinSize)
705 {
706 Log(("VBoxGuestCommonIOCtl: VMMREQUEST: invalid hdr size %#x, expected >= %#x; type=%#x!!\n",
707 cbReq, cbMinSize, pReqHdr->requestType));
708 return VERR_INVALID_PARAMETER;
709 }
710 if (cbReq > cbData)
711 {
712 Log(("VBoxGuestCommonIOCtl: VMMREQUEST: invalid size %#x, expected >= %#x (hdr); type=%#x!!\n",
713 cbData, cbReq, pReqHdr->requestType));
714 return VERR_INVALID_PARAMETER;
715 }
716
717 /*
718 * Make a copy of the request in the physical memory heap so
719 * the VBoxGuestLibrary can more easily deal with the request.
720 * (This is really a waste of time since the OS or the OS specific
721 * code has already buffered or locked the input/output buffer, but
722 * it does makes things a bit simpler wrt to phys address.)
723 */
724 VMMDevRequestHeader *pReqCopy;
725 int rc = VbglGRAlloc(&pReqCopy, cbReq, pReqHdr->requestType);
726 if (RT_FAILURE(rc))
727 {
728 Log(("VBoxGuestCommonIOCtl: VMMREQUEST: failed to allocate %u (%#x) bytes to cache the request. rc=%d!!\n",
729 cbReq, cbReq, rc));
730 return rc;
731 }
732
733 memcpy(pReqCopy, pReqHdr, cbReq);
734 rc = VbglGRPerform(pReqCopy);
735 if ( RT_SUCCESS(rc)
736 && RT_SUCCESS(pReqCopy->rc))
737 {
738 Assert(rc != VINF_HGCM_ASYNC_EXECUTE);
739 Assert(pReqCopy->rc != VINF_HGCM_ASYNC_EXECUTE);
740
741 memcpy(pReqHdr, pReqCopy, cbReq);
742 if (pcbDataReturned)
743 *pcbDataReturned = cbReq;
744 }
745 else if (RT_FAILURE(rc))
746 Log(("VBoxGuestCommonIOCtl: VMMREQUEST: VbglGRPerform - rc=%Rrc!\n", rc));
747 else
748 {
749 Log(("VBoxGuestCommonIOCtl: VMMREQUEST: request execution failed; VMMDev rc=%Rrc!\n", pReqCopy->rc));
750 rc = pReqCopy->rc;
751 }
752
753 VbglGRFree(pReqCopy);
754 return rc;
755}
756
757
758static int VBoxGuestCommonIOCtl_CtlFilterMask(PVBOXGUESTDEVEXT pDevExt, VBoxGuestFilterMaskInfo *pInfo)
759{
760 VMMDevCtlGuestFilterMask *pReq;
761 int rc = VbglGRAlloc((VMMDevRequestHeader **)&pReq, sizeof(*pReq), VMMDevReq_CtlGuestFilterMask);
762 if (RT_FAILURE(rc))
763 {
764 Log(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: failed to allocate %u (%#x) bytes to cache the request. rc=%d!!\n",
765 sizeof(*pReq), sizeof(*pReq), rc));
766 return rc;
767 }
768
769 pReq->u32OrMask = pInfo->u32OrMask;
770 pReq->u32NotMask = pInfo->u32NotMask;
771
772 rc = VbglGRPerform(&pReq->header);
773 if (RT_FAILURE(rc))
774 Log(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: VbglGRPerform failed, rc=%Rrc!\n", rc));
775 else if (RT_FAILURE(pReq->header.rc))
776 {
777 Log(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: The request failed; VMMDev rc=%Rrc!\n", pReq->header.rc));
778 rc = pReq->header.rc;
779 }
780
781 VbglGRFree(&pReq->header);
782 return rc;
783}
784
785
786#ifdef VBOX_HGCM
787
788/**
789 * This is a callback for dealing with async waits.
790 *
791 * It operates in a manner similar to VBoxGuestCommonIOCtl_WaitEvent.
792 */
793static DECLCALLBACK(void)
794VBoxGuestHGCMAsyncWaitCallback(VMMDevHGCMRequestHeader *pHdrNonVolatile, void *pvUser, uint32_t u32User)
795{
796 VMMDevHGCMRequestHeader volatile *pHdr = (VMMDevHGCMRequestHeader volatile *)pHdrNonVolatile;
797 const bool fInterruptible = (bool)u32User;
798 PVBOXGUESTDEVEXT pDevExt = (PVBOXGUESTDEVEXT)pvUser;
799 Log(("VBoxGuestHGCMAsyncWaitCallback: requestType=%d\n", pHdr->header.requestType));
800
801 /*
802 * Check to see if the condition was met by the time we got here.
803 *
804 * We create a simple poll loop here for dealing with out-of-memory
805 * conditions since the caller isn't necessarily able to deal with
806 * us returning too early.
807 */
808 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
809 PVBOXGUESTWAIT pWait;
810 for (;;)
811 {
812 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
813 if ((pHdr->fu32Flags & VBOX_HGCM_REQ_DONE) != 0)
814 {
815 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
816 return;
817 }
818 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
819
820 pWait = VBoxGuestWaitAlloc(pDevExt);
821 if (pWait)
822 break;
823 if (fInterruptible)
824 return;
825 RTThreadSleep(1);
826 }
827 pWait->fReqEvents = VMMDEV_EVENT_HGCM;
828 pWait->pHGCMReq = pHdr;
829
830 /*
831 * Re-enter the spinlock and re-check for the condition.
832 * If the condition is met, return.
833 * Otherwise link us into the HGCM wait list and go to sleep.
834 */
835 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
836 if ((pHdr->fu32Flags & VBOX_HGCM_REQ_DONE) != 0)
837 {
838 VBoxGuestWaitFreeLocked(pDevExt, pWait);
839 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
840 return;
841 }
842 VBoxGuestWaitAppend(&pDevExt->HGCMWaitList, pWait);
843 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
844
845 int rc;
846 if (fInterruptible)
847 rc = RTSemEventMultiWaitNoResume(pWait->Event, RT_INDEFINITE_WAIT);
848 else
849 rc = RTSemEventMultiWait(pWait->Event, RT_INDEFINITE_WAIT);
850
851 /*
852 * This shouldn't ever return failure...
853 * Unlink, free and return.
854 */
855 if (rc == VERR_SEM_DESTROYED)
856 return;
857 if (RT_FAILURE(rc))
858 LogRel(("VBoxGuestHGCMAsyncWaitCallback: wait failed! %Rrc\n", rc));
859
860 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
861 VBoxGuestWaitUnlink(&pDevExt->HGCMWaitList, pWait);
862 VBoxGuestWaitFreeLocked(pDevExt, pWait);
863 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
864}
865
866
867static int VBoxGuestCommonIOCtl_HGCMConnect(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION pSession, VBoxGuestHGCMConnectInfo *pInfo,
868 size_t *pcbDataReturned)
869{
870 /*
871 * The VbglHGCMConnect call will invoke the callback if the HGCM
872 * call is performed in an ASYNC fashion. The function is not able
873 * to deal with cancelled requests.
874 */
875 Log(("VBoxGuestCommonIOCtl: HGCM_CONNECT: %.128s\n",
876 pInfo->Loc.type == VMMDevHGCMLoc_LocalHost || pInfo->Loc.type == VMMDevHGCMLoc_LocalHost_Existing
877 ? pInfo->Loc.u.host.achName : "<not local host>"));
878
879 int rc = VbglHGCMConnect(pInfo, VBoxGuestHGCMAsyncWaitCallback, pDevExt, false /* uninterruptible */);
880 if (RT_SUCCESS(rc))
881 {
882 Log(("VBoxGuestCommonIOCtl: HGCM_CONNECT: u32Client=%RX32 result=%Rrc (rc=%Rrc)\n",
883 pInfo->u32ClientID, pInfo->result, rc));
884 if (RT_SUCCESS(pInfo->result))
885 {
886 /*
887 * Append the client id to the client id table.
888 * If the table has somehow become filled up, we'll disconnect the session.
889 */
890 unsigned i;
891 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
892 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
893 for (i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
894 if (!pSession->aHGCMClientIds[i])
895 {
896 pSession->aHGCMClientIds[i] = pInfo->u32ClientID;
897 break;
898 }
899 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
900 if (i >= RT_ELEMENTS(pSession->aHGCMClientIds))
901 {
902 static unsigned s_cErrors = 0;
903 if (s_cErrors++ < 32)
904 LogRel(("VBoxGuestCommonIOCtl: HGCM_CONNECT: too many HGCMConnect calls for one session!\n"));
905
906 VBoxGuestHGCMDisconnectInfo Info;
907 Info.result = 0;
908 Info.u32ClientID = pInfo->u32ClientID;
909 VbglHGCMDisconnect(&Info, VBoxGuestHGCMAsyncWaitCallback, pDevExt, false /* uninterruptible */);
910 return VERR_TOO_MANY_OPEN_FILES;
911 }
912 }
913 if (pcbDataReturned)
914 *pcbDataReturned = sizeof(*pInfo);
915 }
916 return rc;
917}
918
919
920static int VBoxGuestCommonIOCtl_HGCMDisconnect(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION pSession, VBoxGuestHGCMDisconnectInfo *pInfo,
921 size_t *pcbDataReturned)
922{
923 /*
924 * Validate the client id and invalidate its entry while we're in the call.
925 */
926 const uint32_t u32ClientId = pInfo->u32ClientID;
927 unsigned i;
928 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
929 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
930 for (i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
931 if (pSession->aHGCMClientIds[i] == u32ClientId)
932 {
933 pSession->aHGCMClientIds[i] = UINT32_MAX;
934 break;
935 }
936 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
937 if (i >= RT_ELEMENTS(pSession->aHGCMClientIds))
938 {
939 static unsigned s_cErrors = 0;
940 if (s_cErrors++ > 32)
941 LogRel(("VBoxGuestCommonIOCtl: HGCM_DISCONNECT: u32Client=%RX32\n", u32ClientId));
942 return VERR_INVALID_HANDLE;
943 }
944
945 /*
946 * The VbglHGCMConnect call will invoke the callback if the HGCM
947 * call is performed in an ASYNC fashion. The function is not able
948 * to deal with cancelled requests.
949 */
950 Log(("VBoxGuestCommonIOCtl: HGCM_DISCONNECT: u32Client=%RX32\n", pInfo->u32ClientID));
951 int rc = VbglHGCMDisconnect(pInfo, VBoxGuestHGCMAsyncWaitCallback, pDevExt, false /* uninterruptible */);
952 if (RT_SUCCESS(rc))
953 {
954 Log(("VBoxGuestCommonIOCtl: HGCM_DISCONNECT: result=%Rrc\n", pInfo->result));
955 if (pcbDataReturned)
956 *pcbDataReturned = sizeof(*pInfo);
957 }
958
959 /* Update the client id array according to the result. */
960 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
961 if (pSession->aHGCMClientIds[i] == UINT32_MAX)
962 pSession->aHGCMClientIds[i] = RT_SUCCESS(rc) && RT_SUCCESS(pInfo->result) ? 0 : u32ClientId;
963 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
964
965 return rc;
966}
967
968
969static int VBoxGuestCommonIOCtl_HGCMCall(PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION pSession, VBoxGuestHGCMCallInfo *pInfo,
970 size_t cbData, size_t *pcbDataReturned)
971{
972 /*
973 * Some more validations.
974 */
975 if (pInfo->cParms > 4096) /* (Just make sure it doesn't overflow the next check.) */
976 {
977 Log(("VBoxGuestCommonIOCtl: HGCM_CALL: cParm=%RX32 is not sane\n", pInfo->cParms));
978 return VERR_INVALID_PARAMETER;
979 }
980 const size_t cbActual = sizeof(*pInfo) + pInfo->cParms * sizeof(HGCMFunctionParameter);
981 if (cbData < cbActual)
982 {
983 Log(("VBoxGuestCommonIOCtl: HGCM_CALL: cbData=%#zx (%zu) required size is %#zx (%zu)\n",
984 cbData, cbActual));
985 return VERR_INVALID_PARAMETER;
986 }
987
988 /*
989 * Validate the client id.
990 */
991 const uint32_t u32ClientId = pInfo->u32ClientID;
992 unsigned i;
993 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
994 RTSpinlockAcquireNoInts(pDevExt->SessionSpinlock, &Tmp);
995 for (i = 0; i < RT_ELEMENTS(pSession->aHGCMClientIds); i++)
996 if (pSession->aHGCMClientIds[i] == u32ClientId)
997 break;
998 RTSpinlockReleaseNoInts(pDevExt->SessionSpinlock, &Tmp);
999 if (RT_UNLIKELY(i >= RT_ELEMENTS(pSession->aHGCMClientIds)))
1000 {
1001 static unsigned s_cErrors = 0;
1002 if (s_cErrors++ > 32)
1003 LogRel(("VBoxGuestCommonIOCtl: HGCM_CALL: u32Client=%RX32\n", u32ClientId));
1004 return VERR_INVALID_HANDLE;
1005 }
1006
1007 /*
1008 * The VbglHGCMCall call will invoke the callback if the HGCM
1009 * call is performed in an ASYNC fashion. This function can
1010 * deal with cancelled requests, so we let user more requests
1011 * be interruptible (should add a flag for this later I guess).
1012 */
1013 Log(("VBoxGuestCommonIOCtl: HGCM_CALL: u32Client=%RX32\n", pInfo->u32ClientID));
1014 int rc = VbglHGCMCall(pInfo, VBoxGuestHGCMAsyncWaitCallback, pDevExt, pSession->R0Process != NIL_RTR0PROCESS);
1015 if (RT_SUCCESS(rc))
1016 {
1017 Log(("VBoxGuestCommonIOCtl: HGCM_CALL: result=%Rrc\n", pInfo->result));
1018 if (pcbDataReturned)
1019 *pcbDataReturned = cbActual;
1020 }
1021 return rc;
1022}
1023
1024
1025/**
1026 * @returns VBox status code. Unlike the other HGCM IOCtls this will combine
1027 * the VbglHGCMConnect/Disconnect return code with the Info.result.
1028 */
1029static int VBoxGuestCommonIOCtl_HGCMClipboardReConnect(PVBOXGUESTDEVEXT pDevExt, uint32_t *pu32ClientId, size_t *pcbDataReturned)
1030{
1031 int rc;
1032 Log(("VBoxGuestCommonIOCtl: CLIPBOARD_CONNECT: Current u32ClientId=%RX32\n", pDevExt->u32ClipboardClientId));
1033
1034
1035 /*
1036 * If there is an old client, try disconnect it first.
1037 */
1038 if (pDevExt->u32ClipboardClientId != 0)
1039 {
1040 VBoxGuestHGCMDisconnectInfo Info;
1041 Info.result = (uint32_t)VERR_WRONG_ORDER; /** @todo Vitali, why is this member unsigned? */
1042 Info.u32ClientID = pDevExt->u32ClipboardClientId;
1043 rc = VbglHGCMDisconnect(&Info, VBoxGuestHGCMAsyncWaitCallback, pDevExt, 0);
1044 if (RT_SUCCESS(rc))
1045 {
1046 LogRel(("VBoxGuestCommonIOCtl: CLIPBOARD_CONNECT: failed to disconnect old client. VbglHGCMDisconnect -> rc=%Rrc\n", rc));
1047 return rc;
1048 }
1049 if (RT_FAILURE((int32_t)Info.result))
1050 {
1051 Log(("VBoxGuestCommonIOCtl: CLIPBOARD_CONNECT: failed to disconnect old client. Info.result=%Rrc\n", rc));
1052 return Info.result;
1053 }
1054 pDevExt->u32ClipboardClientId = 0;
1055 }
1056
1057 /*
1058 * Try connect.
1059 */
1060 VBoxGuestHGCMConnectInfo Info;
1061 Info.Loc.type = VMMDevHGCMLoc_LocalHost_Existing;
1062 strcpy(Info.Loc.u.host.achName, "VBoxSharedClipboard");
1063 Info.u32ClientID = 0;
1064 Info.result = (uint32_t)VERR_WRONG_ORDER;
1065
1066 rc = VbglHGCMConnect(&Info,VBoxGuestHGCMAsyncWaitCallback, pDevExt, 0);
1067 if (RT_FAILURE(rc))
1068 {
1069 LogRel(("VBoxGuestCommonIOCtl: CLIPBOARD_CONNECT: VbglHGCMConnected -> rc=%Rrc\n", rc));
1070 return rc;
1071 }
1072 if (RT_FAILURE((int32_t)Info.result))
1073 {
1074 LogRel(("VBoxGuestCommonIOCtl: CLIPBOARD_CONNECT: VbglHGCMConnected -> rc=%Rrc\n", rc));
1075 return rc;
1076 }
1077
1078 Log(("VBoxGuestCommonIOCtl: CLIPBOARD_CONNECT: connected successfully u32ClientId=%RX32\n", Info.u32ClientID));
1079
1080 pDevExt->u32ClipboardClientId = Info.u32ClientID;
1081 *pu32ClientId = Info.u32ClientID;
1082 if (pcbDataReturned)
1083 *pcbDataReturned = sizeof(uint32_t);
1084
1085 return VINF_SUCCESS;
1086}
1087
1088#endif /* VBOX_HGCM */
1089
1090
1091/**
1092 * Common IOCtl for user to kernel and kernel to kernel communcation.
1093 *
1094 * This function only does the basic validation and then invokes
1095 * worker functions that takes care of each specific function.
1096 *
1097 * @returns VBox status code.
1098 *
1099 * @param iFunction The requested function.
1100 * @param pDevExt The device extension.
1101 * @param pSession The client session.
1102 * @param pvData The input/output data buffer. Can be NULL depending on the function.
1103 * @param cbData The max size of the data buffer.
1104 * @param pcbDataReturned Where to store the amount of returned data. Can be NULL.
1105 */
1106int VBoxGuestCommonIOCtl(unsigned iFunction, PVBOXGUESTDEVEXT pDevExt, PVBOXGUESTSESSION pSession,
1107 void *pvData, size_t cbData, size_t *pcbDataReturned)
1108{
1109 Log(("VBoxGuestCommonIOCtl: iFunction=%#x pDevExt=%p pSession=%p pvData=%p cbData=%zu\n",
1110 iFunction, pDevExt, pSession, pvData, cbData));
1111
1112 /*
1113 * Define some helper macros to simplify validation.
1114 */
1115#define CHECKRET_RING0(mnemonic) \
1116 do { \
1117 if (pSession->R0Process != NIL_RTR0PROCESS) \
1118 { \
1119 Log(("VBoxGuestCommonIOCtl: " mnemonic ": Ring-0 only, caller is %RTproc/%p\n", \
1120 pSession->Process, (uintptr_t)pSession->R0Process)); \
1121 return VERR_PERMISSION_DENIED; \
1122 } \
1123 } while (0)
1124#define CHECKRET_MIN_SIZE(mnemonic, cbMin) \
1125 do { \
1126 if (cbData < (cbMin)) \
1127 { \
1128 Log(("VBoxGuestCommonIOCtl: " mnemonic ": cbData=%#zx (%zu) min is %#zx (%zu)\n", \
1129 cbData, cbData, (size_t)(cbMin), (size_t)(cbMin))); \
1130 return VERR_BUFFER_OVERFLOW; \
1131 } \
1132 if ((cbMin) != 0 && !VALID_PTR(pvData)) \
1133 { \
1134 Log(("VBoxGuestCommonIOCtl: " mnemonic ": Invalid pointer %p\n", pvData)); \
1135 return VERR_INVALID_POINTER; \
1136 } \
1137 } while (0)
1138
1139
1140 /*
1141 * Deal with variably sized requests first.
1142 */
1143 int rc = VINF_SUCCESS;
1144 if ( iFunction >= VBOXGUEST_IOCTL_VMMREQUEST(0)
1145 && iFunction <= VBOXGUEST_IOCTL_VMMREQUEST(0xfff)) /** @todo find a better way to do this*/
1146 {
1147 CHECKRET_MIN_SIZE("VMMREQUEST", sizeof(VMMDevRequestHeader));
1148 rc = VBoxGuestCommonIOCtl_VMMRequest(pDevExt, (VMMDevRequestHeader *)pvData, cbData, pcbDataReturned);
1149 }
1150#ifdef VBOX_HGCM
1151 /*
1152 * This one is tricky and can be done later.
1153 */
1154 else if ( iFunction >= VBOXGUEST_IOCTL_HGCM_CALL(0)
1155 && iFunction <= VBOXGUEST_IOCTL_HGCM_CALL(0xfff))
1156 {
1157 CHECKRET_MIN_SIZE("HGCM_CALL", sizeof(VBoxGuestHGCMCallInfo));
1158 rc = VBoxGuestCommonIOCtl_HGCMCall(pDevExt, pSession, (VBoxGuestHGCMCallInfo *)pvData, cbData, pcbDataReturned);
1159 }
1160#endif /* VBOX_HGCM */
1161 else
1162 {
1163 switch (iFunction)
1164 {
1165 case VBOXGUEST_IOCTL_GETVMMDEVPORT:
1166 CHECKRET_RING0("GETVMMDEVPORT");
1167 CHECKRET_MIN_SIZE("GETVMMDEVPORT", sizeof(VBoxGuestPortInfo));
1168 rc = VBoxGuestCommonIOCtl_GetVMMDevPort(pDevExt, (VBoxGuestPortInfo *)pvData, pcbDataReturned);
1169 break;
1170
1171 case VBOXGUEST_IOCTL_WAITEVENT:
1172 CHECKRET_MIN_SIZE("WAITEVENT", sizeof(VBoxGuestWaitEventInfo));
1173 rc = VBoxGuestCommonIOCtl_WaitEvent(pDevExt, (VBoxGuestWaitEventInfo *)pvData, pcbDataReturned,
1174 pSession->R0Process != NIL_RTR0PROCESS);
1175 break;
1176
1177 case VBOXGUEST_IOCTL_CTL_FILTER_MASK:
1178 CHECKRET_MIN_SIZE("CTL_FILTER_MASK", sizeof(VBoxGuestFilterMaskInfo));
1179 rc = VBoxGuestCommonIOCtl_CtlFilterMask(pDevExt, (VBoxGuestFilterMaskInfo *)pvData);
1180 break;
1181
1182#ifdef VBOX_HGCM
1183 case VBOXGUEST_IOCTL_HGCM_CONNECT:
1184 CHECKRET_MIN_SIZE("HGCM_CONNECT", sizeof(VBoxGuestHGCMConnectInfo));
1185 rc = VBoxGuestCommonIOCtl_HGCMConnect(pDevExt, pSession, (VBoxGuestHGCMConnectInfo *)pvData, pcbDataReturned);
1186 break;
1187
1188 case VBOXGUEST_IOCTL_HGCM_DISCONNECT:
1189 CHECKRET_MIN_SIZE("HGCM_DISCONNECT", sizeof(VBoxGuestHGCMDisconnectInfo));
1190 rc = VBoxGuestCommonIOCtl_HGCMDisconnect(pDevExt, pSession, (VBoxGuestHGCMDisconnectInfo *)pvData, pcbDataReturned);
1191 break;
1192
1193
1194 case VBOXGUEST_IOCTL_CLIPBOARD_CONNECT:
1195 CHECKRET_MIN_SIZE("CLIPBOARD_CONNECT", sizeof(uint32_t));
1196 rc = VBoxGuestCommonIOCtl_HGCMClipboardReConnect(pDevExt, (uint32_t *)pvData, pcbDataReturned);
1197 break;
1198#endif /* VBOX_HGCM */
1199
1200 default:
1201 {
1202 Log(("VBoxGuestCommonIOCtl: Unkown request %#x\n", iFunction));
1203 rc = VERR_NOT_SUPPORTED;
1204 break;
1205 }
1206 }
1207 }
1208
1209 Log(("VBoxGuestCommonIOCtl: returns %Rrc *pcbDataReturned=%zu\n", rc, pcbDataReturned ? *pcbDataReturned : 0));
1210 return rc;
1211}
1212
1213
1214
1215/**
1216 * Common interrupt service routine.
1217 *
1218 * This deals with events and with waking up thread waiting for those events.
1219 *
1220 * @returns true if it was our interrupt, false if it wasn't.
1221 * @param pDevExt The VBoxGuest device extension.
1222 */
1223bool VBoxGuestCommonISR(PVBOXGUESTDEVEXT pDevExt)
1224{
1225 /*
1226 * Now we have to find out whether it was our IRQ. Read the event mask
1227 * from our device to see if there are any pending events.
1228 */
1229 bool fOurIrq = pDevExt->pVMMDevMemory->V.V1_04.fHaveEvents;
1230 if (fOurIrq)
1231 {
1232 /* Acknowlegde events. */
1233 VMMDevEvents *pReq = pDevExt->pIrqAckEvents;
1234 int rc = VbglGRPerform(&pReq->header);
1235 if ( RT_SUCCESS(rc)
1236 && RT_SUCCESS(pReq->header.rc))
1237 {
1238 uint32_t fEvents = pReq->events;
1239 Log(("VBoxGuestCommonISR: acknowledge events succeeded %#RX32\n", fEvents));
1240
1241 /*
1242 * Enter the spinlock and examin the waiting threads.
1243 */
1244 int rc2 = 0;
1245 PVBOXGUESTWAIT pWait;
1246 RTSPINLOCKTMP Tmp = RTSPINLOCKTMP_INITIALIZER;
1247 RTSpinlockAcquireNoInts(pDevExt->WaitSpinlock, &Tmp);
1248
1249#ifdef VBOX_HGCM
1250 /* The HGCM event/list is kind of different in that we evaluate all entries. */
1251 if (fEvents & VMMDEV_EVENT_HGCM)
1252 for (pWait = pDevExt->HGCMWaitList.pHead; pWait; pWait = pWait->pNext)
1253 if ( !pWait->fResEvents
1254 && (pWait->pHGCMReq->fu32Flags & VBOX_HGCM_REQ_DONE))
1255 {
1256 pWait->fResEvents = VMMDEV_EVENT_HGCM;
1257 rc2 |= RTSemEventMultiSignal(pWait->Event);
1258 }
1259#endif
1260
1261 /* Normal FIFO evaluation. */
1262 fEvents |= pDevExt->f32PendingEvents;
1263 for (pWait = pDevExt->WaitList.pHead; pWait; pWait = pWait->pNext)
1264 if (!pWait->fResEvents)
1265 {
1266 pWait->fResEvents = pWait->fReqEvents & fEvents;
1267 fEvents &= ~pWait->fResEvents;
1268 rc2 |= RTSemEventMultiSignal(pWait->Event);
1269 if (!fEvents)
1270 break;
1271 }
1272
1273 ASMAtomicXchgU32(&pDevExt->f32PendingEvents, fEvents);
1274 RTSpinlockReleaseNoInts(pDevExt->WaitSpinlock, &Tmp);
1275 Assert(rc2 == 0);
1276 }
1277 else /* something is serious wrong... */
1278 Log(("VBoxGuestCommonISR: acknowledge events failed rc=%d, header rc=%d (events=%#x)!!\n",
1279 rc, pReq->header.rc, pReq->events));
1280 }
1281 else
1282 LogFlow(("VBoxGuestCommonISR: not ours\n"));
1283
1284 return fOurIrq;
1285}
1286
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