VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/VBoxPci/VBoxPci.c@ 71874

Last change on this file since 71874 was 69500, checked in by vboxsync, 7 years ago

*: scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 22.3 KB
Line 
1/* $Id: VBoxPci.c 69500 2017-10-28 15:14:05Z vboxsync $ */
2/** @file
3 * VBoxPci - PCI card passthrough support (Host), Common Code.
4 */
5
6/*
7 * Copyright (C) 2011-2017 Oracle Corporation
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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27/** @page pg_rawpci VBoxPci - host PCI support
28 *
29 * This is a kernel module that works as host proxy between guest and
30 * PCI hardware.
31 *
32 */
33
34#define LOG_GROUP LOG_GROUP_DEV_PCI_RAW
35#include <VBox/log.h>
36#include <VBox/err.h>
37#include <VBox/sup.h>
38#include <VBox/version.h>
39
40#include <iprt/string.h>
41#include <iprt/assert.h>
42#include <iprt/spinlock.h>
43#include <iprt/uuid.h>
44#include <iprt/asm.h>
45#include <iprt/mem.h>
46
47#include "VBoxPciInternal.h"
48
49
50#define DEVPORT_2_VBOXRAWPCIINS(pPort) \
51 ( (PVBOXRAWPCIINS)((uint8_t *)pPort - RT_OFFSETOF(VBOXRAWPCIINS, DevPort)) )
52
53
54/**
55 * Implements the SUPDRV component factor interface query method.
56 *
57 * @returns Pointer to an interface. NULL if not supported.
58 *
59 * @param pSupDrvFactory Pointer to the component factory registration structure.
60 * @param pSession The session - unused.
61 * @param pszInterfaceUuid The factory interface id.
62 */
63static DECLCALLBACK(void *) vboxPciQueryFactoryInterface(PCSUPDRVFACTORY pSupDrvFactory, PSUPDRVSESSION pSession, const char *pszInterfaceUuid)
64{
65 PVBOXRAWPCIGLOBALS pGlobals = (PVBOXRAWPCIGLOBALS)((uint8_t *)pSupDrvFactory - RT_OFFSETOF(VBOXRAWPCIGLOBALS, SupDrvFactory));
66
67 /*
68 * Convert the UUID strings and compare them.
69 */
70 RTUUID UuidReq;
71 int rc = RTUuidFromStr(&UuidReq, pszInterfaceUuid);
72 if (RT_SUCCESS(rc))
73 {
74 if (!RTUuidCompareStr(&UuidReq, RAWPCIFACTORY_UUID_STR))
75 {
76 ASMAtomicIncS32(&pGlobals->cFactoryRefs);
77 return &pGlobals->RawPciFactory;
78 }
79 }
80 else
81 Log(("VBoxRawPci: rc=%Rrc, uuid=%s\n", rc, pszInterfaceUuid));
82
83 return NULL;
84}
85DECLINLINE(int) vboxPciDevLock(PVBOXRAWPCIINS pThis)
86{
87#ifdef VBOX_WITH_SHARED_PCI_INTERRUPTS
88 RTSpinlockAcquire(pThis->hSpinlock);
89 return VINF_SUCCESS;
90#else
91 int rc = RTSemFastMutexRequest(pThis->hFastMtx);
92
93 AssertRC(rc);
94 return rc;
95#endif
96}
97
98DECLINLINE(void) vboxPciDevUnlock(PVBOXRAWPCIINS pThis)
99{
100#ifdef VBOX_WITH_SHARED_PCI_INTERRUPTS
101 RTSpinlockRelease(pThis->hSpinlock);
102#else
103 RTSemFastMutexRelease(pThis->hFastMtx);
104#endif
105}
106
107DECLINLINE(int) vboxPciVmLock(PVBOXRAWPCIDRVVM pThis)
108{
109 int rc = RTSemFastMutexRequest(pThis->hFastMtx);
110 AssertRC(rc);
111 return rc;
112}
113
114DECLINLINE(void) vboxPciVmUnlock(PVBOXRAWPCIDRVVM pThis)
115{
116 RTSemFastMutexRelease(pThis->hFastMtx);
117}
118
119DECLINLINE(int) vboxPciGlobalsLock(PVBOXRAWPCIGLOBALS pGlobals)
120{
121 int rc = RTSemFastMutexRequest(pGlobals->hFastMtx);
122 AssertRC(rc);
123 return rc;
124}
125
126DECLINLINE(void) vboxPciGlobalsUnlock(PVBOXRAWPCIGLOBALS pGlobals)
127{
128 RTSemFastMutexRelease(pGlobals->hFastMtx);
129}
130
131static PVBOXRAWPCIINS vboxPciFindInstanceLocked(PVBOXRAWPCIGLOBALS pGlobals, uint32_t iHostAddress)
132{
133 PVBOXRAWPCIINS pCur;
134 for (pCur = pGlobals->pInstanceHead; pCur != NULL; pCur = pCur->pNext)
135 {
136 if (iHostAddress == pCur->HostPciAddress)
137 return pCur;
138 }
139 return NULL;
140}
141
142static void vboxPciUnlinkInstanceLocked(PVBOXRAWPCIGLOBALS pGlobals, PVBOXRAWPCIINS pToUnlink)
143{
144 if (pGlobals->pInstanceHead == pToUnlink)
145 pGlobals->pInstanceHead = pToUnlink->pNext;
146 else
147 {
148 PVBOXRAWPCIINS pCur;
149 for (pCur = pGlobals->pInstanceHead; pCur != NULL; pCur = pCur->pNext)
150 {
151 if (pCur->pNext == pToUnlink)
152 {
153 pCur->pNext = pToUnlink->pNext;
154 break;
155 }
156 }
157 }
158 pToUnlink->pNext = NULL;
159}
160
161
162DECLHIDDEN(void) vboxPciDevCleanup(PVBOXRAWPCIINS pThis)
163{
164 pThis->DevPort.pfnDeinit(&pThis->DevPort, 0);
165
166 if (pThis->hFastMtx)
167 {
168 RTSemFastMutexDestroy(pThis->hFastMtx);
169 pThis->hFastMtx = NIL_RTSEMFASTMUTEX;
170 }
171
172 if (pThis->hSpinlock)
173 {
174 RTSpinlockDestroy(pThis->hSpinlock);
175 pThis->hSpinlock = NIL_RTSPINLOCK;
176 }
177
178 vboxPciGlobalsLock(pThis->pGlobals);
179 vboxPciUnlinkInstanceLocked(pThis->pGlobals, pThis);
180 vboxPciGlobalsUnlock(pThis->pGlobals);
181}
182
183
184/**
185 * @interface_method_impl{RAWPCIDEVPORT,pfnInit}
186 */
187static DECLCALLBACK(int) vboxPciDevInit(PRAWPCIDEVPORT pPort, uint32_t fFlags)
188{
189 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
190 int rc;
191
192 vboxPciDevLock(pThis);
193
194 rc = vboxPciOsDevInit(pThis, fFlags);
195
196 vboxPciDevUnlock(pThis);
197
198 return rc;
199}
200
201/**
202 * @interface_method_impl{RAWPCIDEVPORT,pfnDeinit}
203 */
204static DECLCALLBACK(int) vboxPciDevDeinit(PRAWPCIDEVPORT pPort, uint32_t fFlags)
205{
206 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
207 int rc;
208
209 vboxPciDevLock(pThis);
210
211 if (pThis->IrqHandler.pfnIrqHandler)
212 {
213 vboxPciOsDevUnregisterIrqHandler(pThis, pThis->IrqHandler.iHostIrq);
214 pThis->IrqHandler.iHostIrq = 0;
215 pThis->IrqHandler.pfnIrqHandler = NULL;
216 }
217
218 rc = vboxPciOsDevDeinit(pThis, fFlags);
219
220 vboxPciDevUnlock(pThis);
221
222 return rc;
223}
224
225
226/**
227 * @interface_method_impl{RAWPCIDEVPORT,pfnDestroy}
228 */
229static DECLCALLBACK(int) vboxPciDevDestroy(PRAWPCIDEVPORT pPort)
230{
231 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
232 int rc;
233
234 rc = vboxPciOsDevDestroy(pThis);
235 if (rc == VINF_SUCCESS)
236 {
237 if (pThis->hFastMtx)
238 {
239 RTSemFastMutexDestroy(pThis->hFastMtx);
240 pThis->hFastMtx = NIL_RTSEMFASTMUTEX;
241 }
242
243 if (pThis->hSpinlock)
244 {
245 RTSpinlockDestroy(pThis->hSpinlock);
246 pThis->hSpinlock = NIL_RTSPINLOCK;
247 }
248
249 vboxPciGlobalsLock(pThis->pGlobals);
250 vboxPciUnlinkInstanceLocked(pThis->pGlobals, pThis);
251 vboxPciGlobalsUnlock(pThis->pGlobals);
252
253 RTMemFree(pThis);
254 }
255
256 return rc;
257}
258/**
259 * @interface_method_impl{RAWPCIDEVPORT,pfnGetRegionInfo}
260 */
261static DECLCALLBACK(int) vboxPciDevGetRegionInfo(PRAWPCIDEVPORT pPort,
262 int32_t iRegion,
263 RTHCPHYS *pRegionStart,
264 uint64_t *pu64RegionSize,
265 bool *pfPresent,
266 uint32_t *pfFlags)
267{
268 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
269 int rc;
270
271 vboxPciDevLock(pThis);
272
273 rc = vboxPciOsDevGetRegionInfo(pThis, iRegion,
274 pRegionStart, pu64RegionSize,
275 pfPresent, pfFlags);
276 vboxPciDevUnlock(pThis);
277
278 return rc;
279}
280
281/**
282 * @interface_method_impl{RAWPCIDEVPORT,pfnMapRegion}
283 */
284static DECLCALLBACK(int) vboxPciDevMapRegion(PRAWPCIDEVPORT pPort,
285 int32_t iRegion,
286 RTHCPHYS RegionStart,
287 uint64_t u64RegionSize,
288 int32_t fFlags,
289 RTR0PTR *pRegionBaseR0)
290{
291 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
292 int rc;
293
294 vboxPciDevLock(pThis);
295
296 rc = vboxPciOsDevMapRegion(pThis, iRegion, RegionStart, u64RegionSize, fFlags, pRegionBaseR0);
297
298 vboxPciDevUnlock(pThis);
299
300 return rc;
301}
302
303/**
304 * @interface_method_impl{RAWPCIDEVPORT,pfnUnmapRegion}
305 */
306static DECLCALLBACK(int) vboxPciDevUnmapRegion(PRAWPCIDEVPORT pPort,
307 int32_t iRegion,
308 RTHCPHYS RegionStart,
309 uint64_t u64RegionSize,
310 RTR0PTR RegionBase)
311{
312 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
313 int rc;
314
315 vboxPciDevLock(pThis);
316
317 rc = vboxPciOsDevUnmapRegion(pThis, iRegion, RegionStart, u64RegionSize, RegionBase);
318
319 vboxPciDevUnlock(pThis);
320
321 return rc;
322}
323
324/**
325 * @interface_method_impl{RAWPCIDEVPORT,pfnPciCfgRead}
326 */
327static DECLCALLBACK(int) vboxPciDevPciCfgRead(PRAWPCIDEVPORT pPort,
328 uint32_t Register,
329 PCIRAWMEMLOC *pValue)
330{
331 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
332 int rc;
333
334 vboxPciDevLock(pThis);
335
336 rc = vboxPciOsDevPciCfgRead(pThis, Register, pValue);
337
338 vboxPciDevUnlock(pThis);
339
340 return rc;
341}
342
343/**
344 * @interface_method_impl{RAWPCIDEVPORT,pfnPciCfgWrite}
345 */
346static DECLCALLBACK(int) vboxPciDevPciCfgWrite(PRAWPCIDEVPORT pPort,
347 uint32_t Register,
348 PCIRAWMEMLOC *pValue)
349{
350 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
351 int rc;
352
353 vboxPciDevLock(pThis);
354
355 rc = vboxPciOsDevPciCfgWrite(pThis, Register, pValue);
356
357 vboxPciDevUnlock(pThis);
358
359 return rc;
360}
361
362static DECLCALLBACK(int) vboxPciDevRegisterIrqHandler(PRAWPCIDEVPORT pPort,
363 PFNRAWPCIISR pfnHandler,
364 void* pIrqContext,
365 PCIRAWISRHANDLE *phIsr)
366{
367 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
368 int rc;
369 int32_t iHostIrq = 0;
370
371 if (pfnHandler == NULL)
372 return VERR_INVALID_PARAMETER;
373
374 vboxPciDevLock(pThis);
375
376 if (pThis->IrqHandler.pfnIrqHandler)
377 {
378 rc = VERR_ALREADY_EXISTS;
379 }
380 else
381 {
382 rc = vboxPciOsDevRegisterIrqHandler(pThis, pfnHandler, pIrqContext, &iHostIrq);
383 if (RT_SUCCESS(rc))
384 {
385 *phIsr = 0xcafe0000;
386 pThis->IrqHandler.iHostIrq = iHostIrq;
387 pThis->IrqHandler.pfnIrqHandler = pfnHandler;
388 pThis->IrqHandler.pIrqContext = pIrqContext;
389 }
390 }
391
392 vboxPciDevUnlock(pThis);
393
394 return rc;
395}
396
397static DECLCALLBACK(int) vboxPciDevUnregisterIrqHandler(PRAWPCIDEVPORT pPort,
398 PCIRAWISRHANDLE hIsr)
399{
400 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
401 int rc;
402
403 if (hIsr != 0xcafe0000)
404 return VERR_INVALID_PARAMETER;
405
406 vboxPciDevLock(pThis);
407
408 rc = vboxPciOsDevUnregisterIrqHandler(pThis, pThis->IrqHandler.iHostIrq);
409 if (RT_SUCCESS(rc))
410 {
411 pThis->IrqHandler.pfnIrqHandler = NULL;
412 pThis->IrqHandler.pIrqContext = NULL;
413 pThis->IrqHandler.iHostIrq = 0;
414 }
415 vboxPciDevUnlock(pThis);
416
417 return rc;
418}
419
420static DECLCALLBACK(int) vboxPciDevPowerStateChange(PRAWPCIDEVPORT pPort,
421 PCIRAWPOWERSTATE aState,
422 uint64_t *pu64Param)
423{
424 PVBOXRAWPCIINS pThis = DEVPORT_2_VBOXRAWPCIINS(pPort);
425 int rc;
426
427 vboxPciDevLock(pThis);
428
429 rc = vboxPciOsDevPowerStateChange(pThis, aState);
430
431 switch (aState)
432 {
433 case PCIRAW_POWER_ON:
434 /*
435 * Let virtual device know about VM caps.
436 */
437 *pu64Param = VBOX_DRV_VMDATA(pThis)->pPerVmData->fVmCaps;
438 break;
439 default:
440 pu64Param = 0;
441 break;
442 }
443
444
445 vboxPciDevUnlock(pThis);
446
447 return rc;
448}
449
450/**
451 * Creates a new instance.
452 *
453 * @returns VBox status code.
454 * @param pGlobals The globals.
455 * @param u32HostAddress Host address.
456 * @param fFlags Flags.
457 * @param pVmCtx VM context.
458 * @param ppDevPort Where to store the pointer to our port interface.
459 * @param pfDevFlags The device flags.
460 */
461static int vboxPciNewInstance(PVBOXRAWPCIGLOBALS pGlobals,
462 uint32_t u32HostAddress,
463 uint32_t fFlags,
464 PRAWPCIPERVM pVmCtx,
465 PRAWPCIDEVPORT *ppDevPort,
466 uint32_t *pfDevFlags)
467{
468 int rc;
469 PVBOXRAWPCIINS pNew = (PVBOXRAWPCIINS)RTMemAllocZ(sizeof(*pNew));
470 if (!pNew)
471 return VERR_NO_MEMORY;
472
473 pNew->pGlobals = pGlobals;
474 pNew->hSpinlock = NIL_RTSPINLOCK;
475 pNew->cRefs = 1;
476 pNew->pNext = NULL;
477 pNew->HostPciAddress = u32HostAddress;
478 pNew->pVmCtx = pVmCtx;
479
480 pNew->DevPort.u32Version = RAWPCIDEVPORT_VERSION;
481
482 pNew->DevPort.pfnInit = vboxPciDevInit;
483 pNew->DevPort.pfnDeinit = vboxPciDevDeinit;
484 pNew->DevPort.pfnDestroy = vboxPciDevDestroy;
485 pNew->DevPort.pfnGetRegionInfo = vboxPciDevGetRegionInfo;
486 pNew->DevPort.pfnMapRegion = vboxPciDevMapRegion;
487 pNew->DevPort.pfnUnmapRegion = vboxPciDevUnmapRegion;
488 pNew->DevPort.pfnPciCfgRead = vboxPciDevPciCfgRead;
489 pNew->DevPort.pfnPciCfgWrite = vboxPciDevPciCfgWrite;
490 pNew->DevPort.pfnPciCfgRead = vboxPciDevPciCfgRead;
491 pNew->DevPort.pfnPciCfgWrite = vboxPciDevPciCfgWrite;
492 pNew->DevPort.pfnRegisterIrqHandler = vboxPciDevRegisterIrqHandler;
493 pNew->DevPort.pfnUnregisterIrqHandler = vboxPciDevUnregisterIrqHandler;
494 pNew->DevPort.pfnPowerStateChange = vboxPciDevPowerStateChange;
495 pNew->DevPort.u32VersionEnd = RAWPCIDEVPORT_VERSION;
496
497 rc = RTSpinlockCreate(&pNew->hSpinlock, RTSPINLOCK_FLAGS_INTERRUPT_SAFE, "VBoxPCI");
498 if (RT_SUCCESS(rc))
499 {
500 rc = RTSemFastMutexCreate(&pNew->hFastMtx);
501 if (RT_SUCCESS(rc))
502 {
503 rc = pNew->DevPort.pfnInit(&pNew->DevPort, fFlags);
504 if (RT_SUCCESS(rc))
505 {
506 *ppDevPort = &pNew->DevPort;
507
508 pNew->pNext = pGlobals->pInstanceHead;
509 pGlobals->pInstanceHead = pNew;
510 }
511 else
512 {
513 RTSemFastMutexDestroy(pNew->hFastMtx);
514 RTSpinlockDestroy(pNew->hSpinlock);
515 RTMemFree(pNew);
516 }
517 }
518 }
519
520 return rc;
521}
522
523/**
524 * @interface_method_impl{RAWPCIFACTORY,pfnCreateAndConnect}
525 */
526static DECLCALLBACK(int) vboxPciFactoryCreateAndConnect(PRAWPCIFACTORY pFactory,
527 uint32_t u32HostAddress,
528 uint32_t fFlags,
529 PRAWPCIPERVM pVmCtx,
530 PRAWPCIDEVPORT *ppDevPort,
531 uint32_t *pfDevFlags)
532{
533 PVBOXRAWPCIGLOBALS pGlobals = (PVBOXRAWPCIGLOBALS)((uint8_t *)pFactory - RT_OFFSETOF(VBOXRAWPCIGLOBALS, RawPciFactory));
534 int rc;
535
536 LogFlow(("vboxPciFactoryCreateAndConnect: PCI=%x fFlags=%#x\n", u32HostAddress, fFlags));
537 Assert(pGlobals->cFactoryRefs > 0);
538 rc = vboxPciGlobalsLock(pGlobals);
539 AssertRCReturn(rc, rc);
540
541 /* First search if there's no existing instance with same host device
542 * address - if so - we cannot continue.
543 */
544 if (vboxPciFindInstanceLocked(pGlobals, u32HostAddress) != NULL)
545 {
546 rc = VERR_RESOURCE_BUSY;
547 goto unlock;
548 }
549
550 rc = vboxPciNewInstance(pGlobals, u32HostAddress, fFlags, pVmCtx, ppDevPort, pfDevFlags);
551
552unlock:
553 vboxPciGlobalsUnlock(pGlobals);
554
555 return rc;
556}
557
558/**
559 * @interface_method_impl{RAWPCIFACTORY,pfnRelease}
560 */
561static DECLCALLBACK(void) vboxPciFactoryRelease(PRAWPCIFACTORY pFactory)
562{
563 PVBOXRAWPCIGLOBALS pGlobals = (PVBOXRAWPCIGLOBALS)((uint8_t *)pFactory - RT_OFFSETOF(VBOXRAWPCIGLOBALS, RawPciFactory));
564
565 int32_t cRefs = ASMAtomicDecS32(&pGlobals->cFactoryRefs);
566 Assert(cRefs >= 0); NOREF(cRefs);
567 LogFlow(("vboxPciFactoryRelease: cRefs=%d (new)\n", cRefs));
568}
569
570/**
571 * @interface_method_impl{RAWPCIFACTORY,pfnInitVm}
572 */
573static DECLCALLBACK(int) vboxPciFactoryInitVm(PRAWPCIFACTORY pFactory,
574 PVM pVM,
575 PRAWPCIPERVM pVmData)
576{
577 PVBOXRAWPCIDRVVM pThis = (PVBOXRAWPCIDRVVM)RTMemAllocZ(sizeof(VBOXRAWPCIDRVVM));
578 int rc;
579
580 if (!pThis)
581 return VERR_NO_MEMORY;
582
583 rc = RTSemFastMutexCreate(&pThis->hFastMtx);
584 if (RT_SUCCESS(rc))
585 {
586 rc = vboxPciOsInitVm(pThis, pVM, pVmData);
587
588 if (RT_SUCCESS(rc))
589 {
590#ifdef VBOX_WITH_IOMMU
591 /* If IOMMU notification routine in pVmData->pfnContigMemInfo
592 is set - we have functional IOMMU hardware. */
593 if (pVmData->pfnContigMemInfo)
594 pVmData->fVmCaps |= PCIRAW_VMFLAGS_HAS_IOMMU;
595#endif
596 pThis->pPerVmData = pVmData;
597 pVmData->pDriverData = pThis;
598 return VINF_SUCCESS;
599 }
600
601 RTSemFastMutexDestroy(pThis->hFastMtx);
602 pThis->hFastMtx = NIL_RTSEMFASTMUTEX;
603 RTMemFree(pThis);
604 }
605
606 return rc;
607}
608
609/**
610 * @interface_method_impl{RAWPCIFACTORY,pfnDeinitVm}
611 */
612static DECLCALLBACK(void) vboxPciFactoryDeinitVm(PRAWPCIFACTORY pFactory,
613 PVM pVM,
614 PRAWPCIPERVM pVmData)
615{
616 if (pVmData->pDriverData)
617 {
618 PVBOXRAWPCIDRVVM pThis = (PVBOXRAWPCIDRVVM)pVmData->pDriverData;
619
620#ifdef VBOX_WITH_IOMMU
621 /* If we have IOMMU, need to unmap all guest's physical pages from IOMMU on VM termination. */
622#endif
623
624 vboxPciOsDeinitVm(pThis, pVM);
625
626 if (pThis->hFastMtx)
627 {
628 RTSemFastMutexDestroy(pThis->hFastMtx);
629 pThis->hFastMtx = NIL_RTSEMFASTMUTEX;
630 }
631
632 RTMemFree(pThis);
633 pVmData->pDriverData = NULL;
634 }
635}
636
637
638static bool vboxPciCanUnload(PVBOXRAWPCIGLOBALS pGlobals)
639{
640 int rc = vboxPciGlobalsLock(pGlobals);
641 bool fRc = !pGlobals->pInstanceHead
642 && pGlobals->cFactoryRefs <= 0;
643 vboxPciGlobalsUnlock(pGlobals);
644 AssertRC(rc);
645 return fRc;
646}
647
648
649static int vboxPciInitIdc(PVBOXRAWPCIGLOBALS pGlobals)
650{
651 int rc;
652 Assert(!pGlobals->fIDCOpen);
653
654 /*
655 * Establish a connection to SUPDRV and register our component factory.
656 */
657 rc = SUPR0IdcOpen(&pGlobals->SupDrvIDC, 0 /* iReqVersion = default */, 0 /* iMinVersion = default */, NULL, NULL, NULL);
658 if (RT_SUCCESS(rc))
659 {
660 rc = SUPR0IdcComponentRegisterFactory(&pGlobals->SupDrvIDC, &pGlobals->SupDrvFactory);
661 if (RT_SUCCESS(rc))
662 {
663 pGlobals->fIDCOpen = true;
664 Log(("VBoxRawPci: pSession=%p\n", SUPR0IdcGetSession(&pGlobals->SupDrvIDC)));
665 return rc;
666 }
667
668 /* bail out. */
669 LogRel(("VBoxRawPci: Failed to register component factory, rc=%Rrc\n", rc));
670 SUPR0IdcClose(&pGlobals->SupDrvIDC);
671 }
672
673 return rc;
674}
675
676/**
677 * Try to close the IDC connection to SUPDRV if established.
678 *
679 * @returns VBox status code.
680 * @retval VINF_SUCCESS on success.
681 * @retval VERR_WRONG_ORDER if we're busy.
682 *
683 * @param pGlobals Pointer to the globals.
684 */
685DECLHIDDEN(int) vboxPciDeleteIdc(PVBOXRAWPCIGLOBALS pGlobals)
686{
687 int rc;
688
689 Assert(pGlobals->hFastMtx != NIL_RTSEMFASTMUTEX);
690
691 /*
692 * Check before trying to deregister the factory.
693 */
694 if (!vboxPciCanUnload(pGlobals))
695 return VERR_WRONG_ORDER;
696
697 if (!pGlobals->fIDCOpen)
698 rc = VINF_SUCCESS;
699 else
700 {
701 /*
702 * Disconnect from SUPDRV.
703 */
704 rc = SUPR0IdcComponentDeregisterFactory(&pGlobals->SupDrvIDC, &pGlobals->SupDrvFactory);
705 AssertRC(rc);
706 SUPR0IdcClose(&pGlobals->SupDrvIDC);
707 pGlobals->fIDCOpen = false;
708 }
709
710 return rc;
711}
712
713
714/**
715 * Initializes the globals.
716 *
717 * @returns VBox status code.
718 * @param pGlobals Pointer to the globals.
719 */
720DECLHIDDEN(int) vboxPciInitGlobals(PVBOXRAWPCIGLOBALS pGlobals)
721{
722 /*
723 * Initialize the common portions of the structure.
724 */
725 int rc = RTSemFastMutexCreate(&pGlobals->hFastMtx);
726 if (RT_SUCCESS(rc))
727 {
728 pGlobals->pInstanceHead = NULL;
729 pGlobals->RawPciFactory.pfnRelease = vboxPciFactoryRelease;
730 pGlobals->RawPciFactory.pfnCreateAndConnect = vboxPciFactoryCreateAndConnect;
731 pGlobals->RawPciFactory.pfnInitVm = vboxPciFactoryInitVm;
732 pGlobals->RawPciFactory.pfnDeinitVm = vboxPciFactoryDeinitVm;
733 memcpy(pGlobals->SupDrvFactory.szName, "VBoxRawPci", sizeof("VBoxRawPci"));
734 pGlobals->SupDrvFactory.pfnQueryFactoryInterface = vboxPciQueryFactoryInterface;
735 pGlobals->fIDCOpen = false;
736 }
737 return rc;
738}
739
740
741/**
742 * Deletes the globals.
743 *
744 *
745 * @param pGlobals Pointer to the globals.
746 */
747DECLHIDDEN(void) vboxPciDeleteGlobals(PVBOXRAWPCIGLOBALS pGlobals)
748{
749 Assert(!pGlobals->fIDCOpen);
750
751 /*
752 * Release resources.
753 */
754 if (pGlobals->hFastMtx)
755 {
756 RTSemFastMutexDestroy(pGlobals->hFastMtx);
757 pGlobals->hFastMtx = NIL_RTSEMFASTMUTEX;
758 }
759}
760
761
762int vboxPciInit(PVBOXRAWPCIGLOBALS pGlobals)
763{
764
765 /*
766 * Initialize the common portions of the structure.
767 */
768 int rc = vboxPciInitGlobals(pGlobals);
769 if (RT_SUCCESS(rc))
770 {
771 rc = vboxPciInitIdc(pGlobals);
772 if (RT_SUCCESS(rc))
773 return rc;
774
775 /* bail out. */
776 vboxPciDeleteGlobals(pGlobals);
777 }
778
779 return rc;
780}
781
782void vboxPciShutdown(PVBOXRAWPCIGLOBALS pGlobals)
783{
784 int rc = vboxPciDeleteIdc(pGlobals);
785
786 if (RT_SUCCESS(rc))
787 vboxPciDeleteGlobals(pGlobals);
788}
789
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