VirtualBox

source: vbox/trunk/src/VBox/Devices/Bus/SrvPciRawR0.cpp@ 38274

Last change on this file since 38274 was 37810, checked in by vboxsync, 14 years ago

more PCI passthrough fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.5 KB
Line 
1/* $Id: SrvPciRawR0.cpp 37810 2011-07-07 08:36:00Z vboxsync $ */
2/** @file
3 * PCI passthrough - The ring 0 service.
4 */
5
6/*
7 * Copyright (C) 2011 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
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_DEV_PCI_RAW
23#include <VBox/log.h>
24#include <VBox/sup.h>
25#include <VBox/rawpci.h>
26#include <VBox/vmm/pdmpci.h>
27#include <VBox/vmm/pdm.h>
28#include <VBox/vmm/gvm.h>
29#include <VBox/vmm/gvmm.h>
30#include <VBox/vmm/vm.h>
31
32#include <iprt/asm.h>
33#include <iprt/assert.h>
34#include <iprt/handletable.h>
35#include <iprt/mp.h>
36#include <iprt/mem.h>
37#include <iprt/semaphore.h>
38#include <iprt/spinlock.h>
39#include <iprt/string.h>
40#include <iprt/thread.h>
41#include <iprt/time.h>
42#include <iprt/asm-amd64-x86.h>
43
44typedef struct PCIRAWSRVSTATE
45{
46 /** Structure lock. */
47 RTSPINLOCK hSpinlock;
48
49 /** Handle table for devices. */
50 RTHANDLETABLE hHtDevs;
51
52} PCIRAWSRVSTATE;
53typedef PCIRAWSRVSTATE *PPCIRAWSRVSTATE;
54
55typedef struct PCIRAWDEV
56{
57 /* Port pointer. */
58 PRAWPCIDEVPORT pPort;
59
60 /* Handle used by everybody else. */
61 PCIRAWDEVHANDLE hHandle;
62
63 /** The session this device is associated with. */
64 PSUPDRVSESSION pSession;
65
66 /** Structure lock. */
67 RTSPINLOCK hSpinlock;
68
69 /** Event for IRQ updates. */
70 RTSEMEVENT hIrqEvent;
71
72 /** Current pending IRQ for the device. */
73 int32_t iPendingIrq;
74
75 /** ISR handle. */
76 PCIRAWISRHANDLE hIsr;
77
78 /* If object is being destroyed. */
79 bool fTerminate;
80
81 /** The SUPR0 object. */
82 void *pvObj;
83} PCIRAWDEV;
84typedef PCIRAWDEV *PPCIRAWDEV;
85
86static PCIRAWSRVSTATE g_State;
87
88/* Interrupt handler. Could be called in the interrupt context,
89 * depending on host OS implmenetation. */
90static DECLCALLBACK(bool) pcirawr0Isr(void* pContext, int32_t iHostIrq)
91{
92 RTSPINLOCKTMP aTmp;
93 PPCIRAWDEV pThis = (PPCIRAWDEV)pContext;
94
95#ifdef VBOX_WITH_SHARED_PCI_INTERRUPTS
96 uint16_t uStatus;
97 PCIRAWMEMLOC Loc;
98 int rc;
99
100 Loc.cb = 2;
101 rc = pThis->pPort->pfnPciCfgRead(pThis->pPort, VBOX_PCI_STATUS, &Loc);
102 /* Cannot read, assume non-shared. */
103 if (RT_FAILURE(rc))
104 return false;
105
106 /* Check interrupt status bit. */
107 if ((Loc.u.u16 & (1 << 3)) == 0)
108 return false;
109#endif
110
111 RTSpinlockAcquireNoInts(pThis->hSpinlock, &aTmp);
112 pThis->iPendingIrq = iHostIrq;
113 RTSpinlockReleaseNoInts(pThis->hSpinlock, &aTmp);
114
115 /**
116 * @todo: RTSemEventSignal() docs claims that it's platform-dependent
117 * if RTSemEventSignal() could be called from the ISR, but it seems IPRT
118 * doesn't provide primitives that guaranteed to work this way.
119 */
120 RTSemEventSignal(pThis->hIrqEvent);
121
122 return true;
123}
124
125static DECLCALLBACK(int) pcirawr0DevRetainHandle(RTHANDLETABLE hHandleTable, void *pvObj, void *pvCtx, void *pvUser)
126{
127 NOREF(pvUser);
128 NOREF(hHandleTable);
129 PPCIRAWDEV pDev = (PPCIRAWDEV)pvObj;
130 if (pDev->hHandle != 0)
131 return SUPR0ObjAddRefEx(pDev->pvObj, (PSUPDRVSESSION)pvCtx, true /* fNoBlocking */);
132
133 return VINF_SUCCESS;
134}
135
136
137/**
138 * Initializes the raw PCI ring-0 service.
139 *
140 * @returns VBox status code.
141 */
142PCIRAWR0DECL(int) PciRawR0Init(void)
143{
144 LogFlow(("PciRawR0Init:\n"));
145 int rc = VINF_SUCCESS;
146
147 rc = RTHandleTableCreateEx(&g_State.hHtDevs, RTHANDLETABLE_FLAGS_LOCKED | RTHANDLETABLE_FLAGS_CONTEXT,
148 UINT32_C(0xfefe0000), 4096, pcirawr0DevRetainHandle, NULL);
149
150 LogFlow(("PciRawR0Init: returns %Rrc\n", rc));
151 return rc;
152}
153
154/**
155 * Destroys raw PCI ring-0 service.
156 */
157PCIRAWR0DECL(void) PciRawR0Term(void)
158{
159 LogFlow(("PciRawR0Term:\n"));
160 RTHandleTableDestroy(g_State.hHtDevs, NULL, NULL);
161 g_State.hHtDevs = NIL_RTHANDLETABLE;
162}
163
164
165/**
166 * Per-VM R0 module init.
167 */
168PCIRAWR0DECL(int) PciRawR0InitVM(PVM pVM)
169{
170 PRAWPCIFACTORY pFactory = NULL;
171 int rc;
172
173 rc = SUPR0ComponentQueryFactory(pVM->pSession, "VBoxRawPci", RAWPCIFACTORY_UUID_STR, (void **)&pFactory);
174
175 if (RT_SUCCESS(rc))
176 {
177 if (pFactory)
178 {
179 PGVM pGVM = NULL;
180 rc = GVMMR0ByVM(pVM, &pGVM);
181 if (RT_SUCCESS(rc))
182 rc = pFactory->pfnInitVm(pFactory, pVM, &pGVM->rawpci.s);
183 pFactory->pfnRelease(pFactory);
184 }
185 }
186
187 return VINF_SUCCESS;
188}
189
190/**
191 * Per-VM R0 module termination routine.
192 */
193PCIRAWR0DECL(void) PciRawR0TermVM(PVM pVM)
194{
195 PRAWPCIFACTORY pFactory = NULL;
196 int rc;
197
198 rc = SUPR0ComponentQueryFactory(pVM->pSession, "VBoxRawPci", RAWPCIFACTORY_UUID_STR, (void **)&pFactory);
199
200 if (RT_SUCCESS(rc))
201 {
202 if (pFactory)
203 {
204 PGVM pGVM = NULL;
205 rc = GVMMR0ByVM(pVM, &pGVM);
206 if (RT_SUCCESS(rc))
207 pFactory->pfnDeinitVm(pFactory, pVM, &pGVM->rawpci.s);
208 pFactory->pfnRelease(pFactory);
209 }
210 }
211}
212
213static int pcirawr0DevTerm(PPCIRAWDEV pThis, int32_t fFlags)
214{
215 ASMAtomicWriteBool(&pThis->fTerminate, true);
216
217 if (pThis->hIrqEvent)
218 RTSemEventSignal(pThis->hIrqEvent);
219
220 /* Enable that, once figure our how to make sure
221 IRQ getter thread notified and woke up. */
222#if 0
223 if (pThis->hIrqEvent)
224 {
225 RTSemEventDestroy(pThis->hIrqEvent);
226 pThis->hIrqEvent = NIL_RTSEMEVENT;
227 }
228#endif
229
230 if (pThis->hSpinlock)
231 {
232 RTSpinlockDestroy(pThis->hSpinlock);
233 pThis->hSpinlock = NIL_RTSPINLOCK;
234 }
235
236 /* Forcefully deinit. */
237 return pThis->pPort->pfnDeinit(pThis->pPort, fFlags);
238}
239
240#define GET_PORT(hDev) \
241 PPCIRAWDEV pDev = (PPCIRAWDEV)RTHandleTableLookupWithCtx(g_State.hHtDevs, hDev, pSession); \
242 if (!pDev) \
243 return VERR_INVALID_HANDLE; \
244 PRAWPCIDEVPORT pDevPort = pDev->pPort; \
245 AssertReturn(pDevPort != NULL, VERR_INVALID_PARAMETER); \
246 AssertReturn(pDevPort->u32Version == RAWPCIDEVPORT_VERSION, VERR_INVALID_PARAMETER); \
247 AssertReturn(pDevPort->u32VersionEnd == RAWPCIDEVPORT_VERSION, VERR_INVALID_PARAMETER);
248
249#define PUT_PORT() if (pDev->pvObj) SUPR0ObjRelease(pDev->pvObj, pSession)
250
251#ifdef DEBUG_nike
252
253/* Code to perform debugging without host driver. */
254typedef struct DUMMYRAWPCIINS
255{
256 /* Host PCI address of this device. */
257 uint32_t HostPciAddress;
258 /* Padding */
259 uint32_t pad0;
260
261 uint8_t aPciCfg[256];
262
263 /** Port, given to the outside world. */
264 RAWPCIDEVPORT DevPort;
265} DUMMYRAWPCIINS;
266typedef struct DUMMYRAWPCIINS *PDUMMYRAWPCIINS;
267
268#define DEVPORT_2_DUMMYRAWPCIINS(pPort) \
269 ( (PDUMMYRAWPCIINS)((uint8_t *)pPort - RT_OFFSETOF(DUMMYRAWPCIINS, DevPort)) )
270
271static uint8_t dummyPciGetByte(PDUMMYRAWPCIINS pThis, uint32_t iRegister)
272{
273 return pThis->aPciCfg[iRegister];
274}
275
276static void dummyPciSetByte(PDUMMYRAWPCIINS pThis, uint32_t iRegister, uint8_t u8)
277{
278 pThis->aPciCfg[iRegister] = u8;
279}
280
281static uint16_t dummyPciGetWord(PDUMMYRAWPCIINS pThis, uint32_t iRegister)
282{
283 uint16_t u16Value = *(uint16_t*)&pThis->aPciCfg[iRegister];
284 return RT_H2LE_U16(u16Value);
285}
286
287static void dummyPciSetWord(PDUMMYRAWPCIINS pThis, uint32_t iRegister, uint16_t u16)
288{
289 *(uint16_t*)&pThis->aPciCfg[iRegister] = RT_H2LE_U16(u16);
290}
291
292static uint32_t dummyPciGetDWord(PDUMMYRAWPCIINS pThis, uint32_t iRegister)
293{
294 uint32_t u32Value = *(uint32_t*)&pThis->aPciCfg[iRegister];
295 return RT_H2LE_U32(u32Value);
296}
297
298static void dummyPciSetDWord(PDUMMYRAWPCIINS pThis, uint32_t iRegister, uint32_t u32)
299{
300 *(uint32_t*)&pThis->aPciCfg[iRegister] = RT_H2LE_U32(u32);
301}
302
303/**
304 * @copydoc RAWPCIDEVPORT:: pfnInit
305 */
306static DECLCALLBACK(int) dummyPciDevInit(PRAWPCIDEVPORT pPort, uint32_t fFlags)
307{
308 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
309
310 dummyPciSetWord(pThis, VBOX_PCI_VENDOR_ID, 0xccdd);
311 dummyPciSetWord(pThis, VBOX_PCI_DEVICE_ID, 0xeeff);
312 dummyPciSetWord(pThis, VBOX_PCI_COMMAND, PCI_COMMAND_IOACCESS | PCI_COMMAND_MEMACCESS | PCI_COMMAND_BUSMASTER);
313 dummyPciSetByte(pThis, VBOX_PCI_INTERRUPT_PIN, 1);
314
315 return VINF_SUCCESS;
316}
317
318/**
319 * @copydoc RAWPCIDEVPORT:: pfnDeinit
320 */
321static DECLCALLBACK(int) dummyPciDevDeinit(PRAWPCIDEVPORT pPort, uint32_t fFlags)
322{
323 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
324
325 return VINF_SUCCESS;
326}
327
328/**
329 * @copydoc RAWPCIDEVPORT:: pfnDestroy
330 */
331static DECLCALLBACK(int) dummyPciDevDestroy(PRAWPCIDEVPORT pPort)
332{
333 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
334
335 RTMemFree(pThis);
336
337 return VINF_SUCCESS;
338}
339
340
341/**
342 * @copydoc RAWPCIDEVPORT:: pfnGetRegionInfo
343 */
344static DECLCALLBACK(int) dummyPciDevGetRegionInfo(PRAWPCIDEVPORT pPort,
345 int32_t iRegion,
346 RTHCPHYS *pRegionStart,
347 uint64_t *pu64RegionSize,
348 bool *pfPresent,
349 uint32_t *pfFlags)
350{
351 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
352
353 if (iRegion == 0)
354 {
355 *pfPresent = true;
356 *pRegionStart = 0xfef0;
357 *pu64RegionSize = 0x10;
358 *pfFlags = PCIRAW_ADDRESS_SPACE_IO;
359 }
360 else if (iRegion == 2)
361 {
362 *pfPresent = true;
363 *pRegionStart = 0xffff0000;
364 *pu64RegionSize = 0x1000;
365 *pfFlags = PCIRAW_ADDRESS_SPACE_BAR64 | PCIRAW_ADDRESS_SPACE_MEM;
366 }
367 else
368 *pfPresent = false;
369
370 return VINF_SUCCESS;
371}
372
373/**
374 * @copydoc RAWPCIDEVPORT:: pfnMapRegion
375 */
376static DECLCALLBACK(int) dummyPciDevMapRegion(PRAWPCIDEVPORT pPort,
377 int32_t iRegion,
378 RTHCPHYS HCRegionStart,
379 uint64_t u64RegionSize,
380 int32_t fFlags,
381 RTR0PTR *pRegionBase)
382{
383 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
384 return VINF_SUCCESS;
385}
386
387/**
388 * @copydoc RAWPCIDEVPORT:: pfnUnapRegion
389 */
390static DECLCALLBACK(int) dummyPciDevUnmapRegion(PRAWPCIDEVPORT pPort,
391 int32_t iRegion,
392 RTHCPHYS HCRegionStart,
393 uint64_t u64RegionSize,
394 RTR0PTR RegionBase)
395{
396 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
397 return VINF_SUCCESS;
398}
399
400/**
401 * @copydoc RAWPCIDEVPORT:: pfnPciCfgRead
402 */
403static DECLCALLBACK(int) dummyPciDevPciCfgRead(PRAWPCIDEVPORT pPort,
404 uint32_t Register,
405 PCIRAWMEMLOC *pValue)
406{
407 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
408
409 switch (pValue->cb)
410 {
411 case 1:
412 pValue->u.u8 = dummyPciGetByte(pThis, Register);
413 break;
414 case 2:
415 pValue->u.u16 = dummyPciGetWord(pThis, Register);
416 break;
417 case 4:
418 pValue->u.u32 = dummyPciGetDWord(pThis, Register);
419 break;
420 }
421
422 return VINF_SUCCESS;
423}
424
425/**
426 * @copydoc RAWPCIDEVPORT:: pfnPciCfgWrite
427 */
428static DECLCALLBACK(int) dummyPciDevPciCfgWrite(PRAWPCIDEVPORT pPort,
429 uint32_t Register,
430 PCIRAWMEMLOC *pValue)
431{
432 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
433
434 switch (pValue->cb)
435 {
436 case 1:
437 dummyPciSetByte(pThis, Register, pValue->u.u8);
438 break;
439 case 2:
440 dummyPciSetWord(pThis, Register, pValue->u.u16);
441 break;
442 case 4:
443 dummyPciSetDWord(pThis, Register, pValue->u.u32);
444 break;
445 }
446
447 return VINF_SUCCESS;
448}
449
450static DECLCALLBACK(int) dummyPciDevRegisterIrqHandler(PRAWPCIDEVPORT pPort,
451 PFNRAWPCIISR pfnHandler,
452 void* pIrqContext,
453 PCIRAWISRHANDLE *phIsr)
454{
455 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
456 return VINF_SUCCESS;
457}
458
459static DECLCALLBACK(int) dummyPciDevUnregisterIrqHandler(PRAWPCIDEVPORT pPort,
460 PCIRAWISRHANDLE hIsr)
461{
462 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
463 return VINF_SUCCESS;
464}
465
466static DECLCALLBACK(int) dummyPciDevPowerStateChange(PRAWPCIDEVPORT pPort,
467 PCIRAWPOWERSTATE aState,
468 uint64_t *pu64Param)
469{
470 PDUMMYRAWPCIINS pThis = DEVPORT_2_DUMMYRAWPCIINS(pPort);
471 return VINF_SUCCESS;
472}
473
474static PRAWPCIDEVPORT pcirawr0CreateDummyDevice(uint32_t HostDevice, uint32_t fFlags)
475{
476 PDUMMYRAWPCIINS pNew = (PDUMMYRAWPCIINS)RTMemAllocZ(sizeof(*pNew));
477 if (!pNew)
478 return NULL;
479
480 pNew->HostPciAddress = HostDevice;
481
482 pNew->DevPort.u32Version = RAWPCIDEVPORT_VERSION;
483 pNew->DevPort.pfnInit = dummyPciDevInit;
484 pNew->DevPort.pfnDeinit = dummyPciDevDeinit;
485 pNew->DevPort.pfnDestroy = dummyPciDevDestroy;
486 pNew->DevPort.pfnGetRegionInfo = dummyPciDevGetRegionInfo;
487 pNew->DevPort.pfnMapRegion = dummyPciDevMapRegion;
488 pNew->DevPort.pfnUnmapRegion = dummyPciDevUnmapRegion;
489 pNew->DevPort.pfnPciCfgRead = dummyPciDevPciCfgRead;
490 pNew->DevPort.pfnPciCfgWrite = dummyPciDevPciCfgWrite;
491 pNew->DevPort.pfnRegisterIrqHandler = dummyPciDevRegisterIrqHandler;
492 pNew->DevPort.pfnUnregisterIrqHandler = dummyPciDevUnregisterIrqHandler;
493 pNew->DevPort.pfnPowerStateChange = dummyPciDevPowerStateChange;
494
495 pNew->DevPort.u32VersionEnd = RAWPCIDEVPORT_VERSION;
496
497 return &pNew->DevPort;
498}
499#endif
500
501static DECLCALLBACK(void) pcirawr0DevObjDestructor(void *pvObj, void *pvIns, void *pvUnused)
502{
503 PPCIRAWDEV pThis = (PPCIRAWDEV)pvIns;
504
505 /* Forcefully deinit. */
506 pcirawr0DevTerm(pThis, 0);
507
508 /* And destroy. */
509 pThis->pPort->pfnDestroy(pThis->pPort);
510
511 RTMemFree(pThis);
512}
513
514
515static int pcirawr0OpenDevice(PSUPDRVSESSION pSession,
516 PVM pVM,
517 uint32_t HostDevice,
518 uint32_t fFlags,
519 PCIRAWDEVHANDLE *pHandle,
520 uint32_t *pfDevFlags)
521{
522 /*
523 * Query the factory we want, then use it create and connect the host device.
524 */
525 PRAWPCIFACTORY pFactory = NULL;
526 PRAWPCIDEVPORT pDevPort = NULL;
527 int rc;
528 PPCIRAWDEV pNew;
529
530 pNew = (PPCIRAWDEV)RTMemAllocZ(sizeof(*pNew));
531 if (!pNew)
532 return VERR_NO_MEMORY;
533
534
535 rc = SUPR0ComponentQueryFactory(pSession, "VBoxRawPci", RAWPCIFACTORY_UUID_STR, (void **)&pFactory);
536 /* No host driver registered, provide some fake implementation
537 for debugging purposes. */
538#ifdef DEBUG_nike
539 if (rc == VERR_SUPDRV_COMPONENT_NOT_FOUND)
540 {
541 pDevPort = pcirawr0CreateDummyDevice(HostDevice, fFlags);
542 if (pDevPort)
543 {
544 pDevPort->pfnInit(pDevPort, fFlags);
545 rc = VINF_SUCCESS;
546 }
547 else
548 rc = VERR_NO_MEMORY;
549 }
550#endif
551
552 if (RT_SUCCESS(rc))
553 {
554 if (pFactory)
555 {
556 PGVM pGVM = NULL;
557 rc = GVMMR0ByVM(pVM, &pGVM);
558
559 if (RT_SUCCESS(rc))
560 rc = pFactory->pfnCreateAndConnect(pFactory,
561 HostDevice,
562 fFlags,
563 &pGVM->rawpci.s,
564 &pDevPort,
565 pfDevFlags);
566 pFactory->pfnRelease(pFactory);
567 }
568
569 if (RT_SUCCESS(rc))
570 {
571 rc = RTSpinlockCreate(&pNew->hSpinlock);
572 AssertRC(rc);
573 rc = RTSemEventCreate(&pNew->hIrqEvent);
574 AssertRC(rc);
575
576 pNew->pSession = pSession;
577 pNew->pPort = pDevPort;
578 pNew->pvObj = SUPR0ObjRegister(pSession, SUPDRVOBJTYPE_RAW_PCI_DEVICE,
579 pcirawr0DevObjDestructor, pNew, NULL);
580
581 uint32_t hHandle = 0;
582 rc = RTHandleTableAllocWithCtx(g_State.hHtDevs, pNew, pSession, &hHandle);
583 if (RT_SUCCESS(rc))
584 {
585 pNew->hHandle = (PCIRAWDEVHANDLE)hHandle;
586 *pHandle = pNew->hHandle;
587 }
588 else
589 {
590 SUPR0ObjRelease(pNew->pvObj, pSession);
591 RTSpinlockDestroy(pNew->hSpinlock);
592 RTSemEventDestroy(pNew->hIrqEvent);
593 }
594 }
595 }
596
597 if (RT_FAILURE(rc))
598 RTMemFree(pNew);
599
600 return rc;
601}
602
603static int pcirawr0CloseDevice(PSUPDRVSESSION pSession,
604 PCIRAWDEVHANDLE TargetDevice,
605 uint32_t fFlags)
606{
607 GET_PORT(TargetDevice);
608 int rc;
609
610 pDevPort->pfnUnregisterIrqHandler(pDevPort, pDev->hIsr);
611 pDev->hIsr = 0;
612
613 rc = pcirawr0DevTerm(pDev, fFlags);
614
615 RTHandleTableFreeWithCtx(g_State.hHtDevs, TargetDevice, pSession);
616
617 PUT_PORT();
618
619 return rc;
620}
621
622/* We may want to call many functions here directly, so no static */
623static int pcirawr0GetRegionInfo(PSUPDRVSESSION pSession,
624 PCIRAWDEVHANDLE TargetDevice,
625 int32_t iRegion,
626 RTHCPHYS *pRegionStart,
627 uint64_t *pu64RegionSize,
628 bool *pfPresent,
629 uint32_t *pfFlags)
630{
631 LogFlow(("pcirawr0GetRegionInfo: %d\n", iRegion));
632 GET_PORT(TargetDevice);
633
634 int rc = pDevPort->pfnGetRegionInfo(pDevPort, iRegion, pRegionStart, pu64RegionSize, pfPresent, pfFlags);
635
636 PUT_PORT();
637
638 return rc;
639}
640
641static int pcirawr0MapRegion(PSUPDRVSESSION pSession,
642 PCIRAWDEVHANDLE TargetDevice,
643 int32_t iRegion,
644 RTHCPHYS HCRegionStart,
645 uint64_t u64RegionSize,
646 uint32_t fFlags,
647 RTR3PTR *ppvAddressR3,
648 RTR0PTR *ppvAddressR0)
649{
650 LogFlow(("pcirawr0MapRegion\n"));
651 GET_PORT(TargetDevice);
652 int rc;
653
654 rc = pDevPort->pfnMapRegion(pDevPort, iRegion, HCRegionStart, u64RegionSize, fFlags, ppvAddressR0);
655 if (RT_SUCCESS(rc))
656 {
657 Assert(*ppvAddressR0 != NULL);
658
659 /* Do we need to do something to help with R3 mapping, if ((fFlags & PCIRAWRFLAG_ALLOW_R3MAP) != 0) */
660 }
661
662 *ppvAddressR3 = 0;
663
664 PUT_PORT();
665
666 return rc;
667}
668
669static int pcirawr0UnmapRegion(PSUPDRVSESSION pSession,
670 PCIRAWDEVHANDLE TargetDevice,
671 int32_t iRegion,
672 RTHCPHYS HCRegionStart,
673 uint64_t u64RegionSize,
674 RTR3PTR pvAddressR3,
675 RTR0PTR pvAddressR0)
676{
677 LogFlow(("pcirawr0UnmapRegion\n"));
678 int rc;
679
680 GET_PORT(TargetDevice);
681
682 rc = pDevPort->pfnUnmapRegion(pDevPort, iRegion, HCRegionStart, u64RegionSize, pvAddressR0);
683
684 PUT_PORT();
685
686 return rc;
687}
688
689static int pcirawr0PioWrite(PSUPDRVSESSION pSession,
690 PCIRAWDEVHANDLE TargetDevice,
691 uint16_t Port,
692 uint32_t u32,
693 unsigned cb)
694{
695 /// @todo: add check that port fits into device range
696 switch (cb)
697 {
698 case 1:
699 ASMOutU8 (Port, u32);
700 break;
701 case 2:
702 ASMOutU16(Port, u32);
703 break;
704 case 4:
705 ASMOutU32(Port, u32);
706 break;
707 default:
708 AssertMsgFailed(("Unhandled port write: %d\n", cb));
709 }
710
711 return VINF_SUCCESS;
712}
713
714
715static int pcirawr0PioRead(PSUPDRVSESSION pSession,
716 PCIRAWDEVHANDLE TargetDevice,
717 uint16_t Port,
718 uint32_t *pu32,
719 unsigned cb)
720{
721 /// @todo: add check that port fits into device range
722 switch (cb)
723 {
724 case 1:
725 *pu32 = ASMInU8 (Port);
726 break;
727 case 2:
728 *pu32 = ASMInU16(Port);
729 break;
730 case 4:
731 *pu32 = ASMInU32(Port);
732 break;
733 default:
734 AssertMsgFailed(("Unhandled port read: %d\n", cb));
735 }
736
737 return VINF_SUCCESS;
738}
739
740
741static int pcirawr0MmioRead(PSUPDRVSESSION pSession,
742 PCIRAWDEVHANDLE TargetDevice,
743 RTR0PTR Address,
744 PCIRAWMEMLOC *pValue)
745{
746 /// @todo: add check that address fits into device range
747#if 1
748 switch (pValue->cb)
749 {
750 case 1:
751 pValue->u.u8 = *(uint8_t*)Address;
752 break;
753 case 2:
754 pValue->u.u16 = *(uint16_t*)Address;
755 break;
756 case 4:
757 pValue->u.u32 = *(uint32_t*)Address;
758 break;
759 case 8:
760 pValue->u.u64 = *(uint64_t*)Address;
761 break;
762 }
763#else
764 memset(&pValue->u.u64, 0, 8);
765#endif
766 return VINF_SUCCESS;
767}
768
769static int pcirawr0MmioWrite(PSUPDRVSESSION pSession,
770 PCIRAWDEVHANDLE TargetDevice,
771 RTR0PTR Address,
772 PCIRAWMEMLOC *pValue)
773{
774 /// @todo: add check that address fits into device range
775#if 1
776 switch (pValue->cb)
777 {
778 case 1:
779 *(uint8_t*)Address = pValue->u.u8;
780 break;
781 case 2:
782 *(uint16_t*)Address = pValue->u.u16;
783 break;
784 case 4:
785 *(uint32_t*)Address = pValue->u.u32;
786 break;
787 case 8:
788 *(uint64_t*)Address = pValue->u.u64;
789 break;
790 }
791#endif
792 return VINF_SUCCESS;
793}
794
795static int pcirawr0PciCfgRead(PSUPDRVSESSION pSession,
796 PCIRAWDEVHANDLE TargetDevice,
797 uint32_t Register,
798 PCIRAWMEMLOC *pValue)
799{
800 GET_PORT(TargetDevice);
801
802 return pDevPort->pfnPciCfgRead(pDevPort, Register, pValue);
803}
804
805static int pcirawr0PciCfgWrite(PSUPDRVSESSION pSession,
806 PCIRAWDEVHANDLE TargetDevice,
807 uint32_t Register,
808 PCIRAWMEMLOC *pValue)
809{
810 int rc;
811
812 GET_PORT(TargetDevice);
813
814 rc = pDevPort->pfnPciCfgWrite(pDevPort, Register, pValue);
815
816 PUT_PORT();
817
818 return rc;
819}
820
821static int pcirawr0EnableIrq(PSUPDRVSESSION pSession,
822 PCIRAWDEVHANDLE TargetDevice)
823{
824 int rc = VINF_SUCCESS;
825 GET_PORT(TargetDevice);
826
827 rc = pDevPort->pfnRegisterIrqHandler(pDevPort, pcirawr0Isr, pDev,
828 &pDev->hIsr);
829
830 PUT_PORT();
831 return rc;
832}
833
834static int pcirawr0DisableIrq(PSUPDRVSESSION pSession,
835 PCIRAWDEVHANDLE TargetDevice)
836{
837 int rc = VINF_SUCCESS;
838 GET_PORT(TargetDevice);
839
840 rc = pDevPort->pfnUnregisterIrqHandler(pDevPort, pDev->hIsr);
841 pDev->hIsr = 0;
842
843 PUT_PORT();
844 return rc;
845}
846
847static int pcirawr0GetIrq(PSUPDRVSESSION pSession,
848 PCIRAWDEVHANDLE TargetDevice,
849 int64_t iTimeout,
850 int32_t *piIrq)
851{
852 int rc = VINF_SUCCESS;
853 RTSPINLOCKTMP aTmp;
854 bool fTerminate = false;
855 int32_t iPendingIrq = 0;
856
857 LogFlow(("pcirawr0GetIrq\n"));
858
859 GET_PORT(TargetDevice);
860
861 RTSpinlockAcquireNoInts(pDev->hSpinlock, &aTmp);
862 iPendingIrq = pDev->iPendingIrq;
863 pDev->iPendingIrq = 0;
864 fTerminate = pDev->fTerminate;
865 RTSpinlockReleaseNoInts(pDev->hSpinlock, &aTmp);
866
867 /* Block until new IRQs arrives */
868 if (!fTerminate)
869 {
870 if (iPendingIrq == 0)
871 {
872 rc = RTSemEventWaitNoResume(pDev->hIrqEvent, iTimeout);
873 if (RT_SUCCESS(rc))
874 {
875 /** @todo: racy */
876 if (!ASMAtomicReadBool(&pDev->fTerminate))
877 {
878 RTSpinlockAcquireNoInts(pDev->hSpinlock, &aTmp);
879 iPendingIrq = pDev->iPendingIrq;
880 pDev->iPendingIrq = 0;
881 RTSpinlockReleaseNoInts(pDev->hSpinlock, &aTmp);
882 }
883 else
884 rc = VERR_INTERRUPTED;
885 }
886 }
887
888 if (RT_SUCCESS(rc))
889 *piIrq = iPendingIrq;
890 }
891 else
892 rc = VERR_INTERRUPTED;
893
894 PUT_PORT();
895
896 return rc;
897}
898
899static int pcirawr0PowerStateChange(PSUPDRVSESSION pSession,
900 PCIRAWDEVHANDLE TargetDevice,
901 PCIRAWPOWERSTATE aState,
902 uint64_t *pu64Param)
903{
904 LogFlow(("pcirawr0PowerStateChange\n"));
905 GET_PORT(TargetDevice);
906
907 int rc = pDevPort->pfnPowerStateChange(pDevPort, aState, pu64Param);
908
909 PUT_PORT();
910
911 return rc;
912}
913
914/**
915 * Process PCI raw request
916 *
917 * @returns VBox status code.
918 */
919PCIRAWR0DECL(int) PciRawR0ProcessReq(PSUPDRVSESSION pSession, PVM pVM, PPCIRAWSENDREQ pReq)
920{
921 LogFlow(("PciRawR0ProcessReq: %d for %x\n", pReq->iRequest, pReq->TargetDevice));
922 int rc = VINF_SUCCESS;
923
924 /* Route request to the host driver */
925 switch (pReq->iRequest)
926 {
927 case PCIRAWR0_DO_OPEN_DEVICE:
928 rc = pcirawr0OpenDevice(pSession, pVM,
929 pReq->u.aOpenDevice.PciAddress,
930 pReq->u.aOpenDevice.fFlags,
931 &pReq->u.aOpenDevice.Device,
932 &pReq->u.aOpenDevice.fDevFlags);
933 break;
934 case PCIRAWR0_DO_CLOSE_DEVICE:
935 rc = pcirawr0CloseDevice(pSession,
936 pReq->TargetDevice,
937 pReq->u.aCloseDevice.fFlags);
938 break;
939 case PCIRAWR0_DO_GET_REGION_INFO:
940 rc = pcirawr0GetRegionInfo(pSession,
941 pReq->TargetDevice,
942 pReq->u.aGetRegionInfo.iRegion,
943 &pReq->u.aGetRegionInfo.RegionStart,
944 &pReq->u.aGetRegionInfo.u64RegionSize,
945 &pReq->u.aGetRegionInfo.fPresent,
946 &pReq->u.aGetRegionInfo.fFlags);
947 break;
948 case PCIRAWR0_DO_MAP_REGION:
949 rc = pcirawr0MapRegion(pSession,
950 pReq->TargetDevice,
951 pReq->u.aMapRegion.iRegion,
952 pReq->u.aMapRegion.StartAddress,
953 pReq->u.aMapRegion.iRegionSize,
954 pReq->u.aMapRegion.fFlags,
955 &pReq->u.aMapRegion.pvAddressR3,
956 &pReq->u.aMapRegion.pvAddressR0);
957 break;
958 case PCIRAWR0_DO_UNMAP_REGION:
959 rc = pcirawr0UnmapRegion(pSession,
960 pReq->TargetDevice,
961 pReq->u.aUnmapRegion.iRegion,
962 pReq->u.aUnmapRegion.StartAddress,
963 pReq->u.aUnmapRegion.iRegionSize,
964 pReq->u.aUnmapRegion.pvAddressR3,
965 pReq->u.aUnmapRegion.pvAddressR0);
966 break;
967 case PCIRAWR0_DO_PIO_WRITE:
968 rc = pcirawr0PioWrite(pSession,
969 pReq->TargetDevice,
970 pReq->u.aPioWrite.iPort,
971 pReq->u.aPioWrite.iValue,
972 pReq->u.aPioWrite.cb);
973 break;
974 case PCIRAWR0_DO_PIO_READ:
975 rc = pcirawr0PioRead(pSession,
976 pReq->TargetDevice,
977 pReq->u.aPioRead.iPort,
978 &pReq->u.aPioWrite.iValue,
979 pReq->u.aPioRead.cb);
980 break;
981 case PCIRAWR0_DO_MMIO_WRITE:
982 rc = pcirawr0MmioWrite(pSession,
983 pReq->TargetDevice,
984 pReq->u.aMmioWrite.Address,
985 &pReq->u.aMmioWrite.Value);
986 break;
987 case PCIRAWR0_DO_MMIO_READ:
988 rc = pcirawr0MmioRead(pSession,
989 pReq->TargetDevice,
990 pReq->u.aMmioRead.Address,
991 &pReq->u.aMmioRead.Value);
992 break;
993 case PCIRAWR0_DO_PCICFG_WRITE:
994 rc = pcirawr0PciCfgWrite(pSession,
995 pReq->TargetDevice,
996 pReq->u.aPciCfgWrite.iOffset,
997 &pReq->u.aPciCfgWrite.Value);
998 break;
999 case PCIRAWR0_DO_PCICFG_READ:
1000 rc = pcirawr0PciCfgRead(pSession,
1001 pReq->TargetDevice,
1002 pReq->u.aPciCfgRead.iOffset,
1003 &pReq->u.aPciCfgRead.Value);
1004 break;
1005 case PCIRAWR0_DO_ENABLE_IRQ:
1006 rc = pcirawr0EnableIrq(pSession,
1007 pReq->TargetDevice);
1008 break;
1009 case PCIRAWR0_DO_DISABLE_IRQ:
1010 rc = pcirawr0DisableIrq(pSession,
1011 pReq->TargetDevice);
1012 break;
1013 case PCIRAWR0_DO_GET_IRQ:
1014 rc = pcirawr0GetIrq(pSession,
1015 pReq->TargetDevice,
1016 pReq->u.aGetIrq.iTimeout,
1017 &pReq->u.aGetIrq.iIrq);
1018 break;
1019 case PCIRAWR0_DO_POWER_STATE_CHANGE:
1020 rc = pcirawr0PowerStateChange(pSession,
1021 pReq->TargetDevice,
1022 (PCIRAWPOWERSTATE)pReq->u.aPowerStateChange.iState,
1023 &pReq->u.aPowerStateChange.u64Param);
1024 break;
1025 default:
1026 rc = VERR_NOT_SUPPORTED;
1027 }
1028
1029 LogFlow(("PciRawR0ProcessReq: returns %Rrc\n", rc));
1030 return rc;
1031}
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