diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-05-06 03:01:46 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-05-06 03:01:46 +0000 |
commit | f8fe689a81f906d1b91bb3220acde2a4ecb14c5b (patch) | |
tree | 26484e9d7e2c67806c2d1760196ff01aaa858e8c /src/VBox/Devices/VirtIO | |
parent | Initial commit. (diff) | |
download | virtualbox-upstream.tar.xz virtualbox-upstream.zip |
Adding upstream version 6.0.4-dfsg.upstream/6.0.4-dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'src/VBox/Devices/VirtIO')
-rw-r--r-- | src/VBox/Devices/VirtIO/Makefile.kup | 0 | ||||
-rw-r--r-- | src/VBox/Devices/VirtIO/Virtio.cpp | 1024 | ||||
-rw-r--r-- | src/VBox/Devices/VirtIO/Virtio.h | 334 |
3 files changed, 1358 insertions, 0 deletions
diff --git a/src/VBox/Devices/VirtIO/Makefile.kup b/src/VBox/Devices/VirtIO/Makefile.kup new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/src/VBox/Devices/VirtIO/Makefile.kup diff --git a/src/VBox/Devices/VirtIO/Virtio.cpp b/src/VBox/Devices/VirtIO/Virtio.cpp new file mode 100644 index 00000000..eac7276d --- /dev/null +++ b/src/VBox/Devices/VirtIO/Virtio.cpp @@ -0,0 +1,1024 @@ +/* $Id: Virtio.cpp $ */ +/** @file + * Virtio - Virtio Common Functions (VRing, VQueue, Virtio PCI) + */ + +/* + * Copyright (C) 2009-2019 Oracle Corporation + * + * This file is part of VirtualBox Open Source Edition (OSE), as + * available from http://www.virtualbox.org. This file is free software; + * you can redistribute it and/or modify it under the terms of the GNU + * General Public License (GPL) as published by the Free Software + * Foundation, in version 2 as it comes in the "COPYING" file of the + * VirtualBox OSE distribution. VirtualBox OSE is distributed in the + * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. + */ + + +/********************************************************************************************************************************* +* Header Files * +*********************************************************************************************************************************/ +#define LOG_GROUP LOG_GROUP_DEV_VIRTIO + +#include <iprt/param.h> +#include <iprt/uuid.h> +#include <VBox/vmm/pdmdev.h> +#include "Virtio.h" + +#define INSTANCE(pState) pState->szInstance +#define IFACE_TO_STATE(pIface, ifaceName) ((VPCISTATE *)((char*)(pIface) - RT_UOFFSETOF(VPCISTATE, ifaceName))) + +#ifdef LOG_ENABLED +# define QUEUENAME(s, q) (q->pcszName) +#endif + + + +#ifndef VBOX_DEVICE_STRUCT_TESTCASE + +//RT_C_DECLS_BEGIN +//RT_C_DECLS_END + + +static void vqueueReset(PVQUEUE pQueue) +{ + pQueue->VRing.addrDescriptors = 0; + pQueue->VRing.addrAvail = 0; + pQueue->VRing.addrUsed = 0; + pQueue->uNextAvailIndex = 0; + pQueue->uNextUsedIndex = 0; + pQueue->uPageNumber = 0; +} + +static void vqueueInit(PVQUEUE pQueue, uint32_t uPageNumber) +{ + pQueue->VRing.addrDescriptors = (uint64_t)uPageNumber << PAGE_SHIFT; + pQueue->VRing.addrAvail = pQueue->VRing.addrDescriptors + + sizeof(VRINGDESC) * pQueue->VRing.uSize; + pQueue->VRing.addrUsed = RT_ALIGN( + pQueue->VRing.addrAvail + RT_UOFFSETOF_DYN(VRINGAVAIL, auRing[pQueue->VRing.uSize]), + PAGE_SIZE); /* The used ring must start from the next page. */ + pQueue->uNextAvailIndex = 0; + pQueue->uNextUsedIndex = 0; +} + +// void vqueueElemFree(PVQUEUEELEM pElem) +// { +// } + +void vringReadDesc(PVPCISTATE pState, PVRING pVRing, uint32_t uIndex, PVRINGDESC pDesc) +{ + //Log(("%s vringReadDesc: ring=%p idx=%u\n", INSTANCE(pState), pVRing, uIndex)); + PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), + pVRing->addrDescriptors + sizeof(VRINGDESC) * (uIndex % pVRing->uSize), + pDesc, sizeof(VRINGDESC)); +} + +uint16_t vringReadAvail(PVPCISTATE pState, PVRING pVRing, uint32_t uIndex) +{ + uint16_t tmp; + + PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), + pVRing->addrAvail + RT_UOFFSETOF_DYN(VRINGAVAIL, auRing[uIndex % pVRing->uSize]), + &tmp, sizeof(tmp)); + return tmp; +} + +uint16_t vringReadAvailFlags(PVPCISTATE pState, PVRING pVRing) +{ + uint16_t tmp; + + PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), + pVRing->addrAvail + RT_UOFFSETOF(VRINGAVAIL, uFlags), + &tmp, sizeof(tmp)); + return tmp; +} + +void vringSetNotification(PVPCISTATE pState, PVRING pVRing, bool fEnabled) +{ + uint16_t tmp; + + PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), + pVRing->addrUsed + RT_UOFFSETOF(VRINGUSED, uFlags), + &tmp, sizeof(tmp)); + + if (fEnabled) + tmp &= ~ VRINGUSED_F_NO_NOTIFY; + else + tmp |= VRINGUSED_F_NO_NOTIFY; + + PDMDevHlpPCIPhysWrite(pState->CTX_SUFF(pDevIns), + pVRing->addrUsed + RT_UOFFSETOF(VRINGUSED, uFlags), + &tmp, sizeof(tmp)); +} + +bool vqueueSkip(PVPCISTATE pState, PVQUEUE pQueue) +{ + if (vqueueIsEmpty(pState, pQueue)) + return false; + + Log2(("%s vqueueSkip: %s avail_idx=%u\n", INSTANCE(pState), + QUEUENAME(pState, pQueue), pQueue->uNextAvailIndex)); + pQueue->uNextAvailIndex++; + return true; +} + +bool vqueueGet(PVPCISTATE pState, PVQUEUE pQueue, PVQUEUEELEM pElem, bool fRemove) +{ + if (vqueueIsEmpty(pState, pQueue)) + return false; + + pElem->nIn = pElem->nOut = 0; + + Log2(("%s vqueueGet: %s avail_idx=%u\n", INSTANCE(pState), + QUEUENAME(pState, pQueue), pQueue->uNextAvailIndex)); + + VRINGDESC desc; + uint16_t idx = vringReadAvail(pState, &pQueue->VRing, pQueue->uNextAvailIndex); + if (fRemove) + pQueue->uNextAvailIndex++; + pElem->uIndex = idx; + do + { + VQUEUESEG *pSeg; + + /* + * Malicious guests may try to trick us into writing beyond aSegsIn or + * aSegsOut boundaries by linking several descriptors into a loop. We + * cannot possibly get a sequence of linked descriptors exceeding the + * total number of descriptors in the ring (see @bugref{8620}). + */ + if (pElem->nIn + pElem->nOut >= VRING_MAX_SIZE) + { + static volatile uint32_t s_cMessages = 0; + static volatile uint32_t s_cThreshold = 1; + if (ASMAtomicIncU32(&s_cMessages) == ASMAtomicReadU32(&s_cThreshold)) + { + LogRel(("%s: too many linked descriptors; check if the guest arranges descriptors in a loop.\n", + INSTANCE(pState))); + if (ASMAtomicReadU32(&s_cMessages) != 1) + LogRel(("%s: (the above error has occured %u times so far)\n", + INSTANCE(pState), ASMAtomicReadU32(&s_cMessages))); + ASMAtomicWriteU32(&s_cThreshold, ASMAtomicReadU32(&s_cThreshold) * 10); + } + break; + } + RT_UNTRUSTED_VALIDATED_FENCE(); + + vringReadDesc(pState, &pQueue->VRing, idx, &desc); + if (desc.u16Flags & VRINGDESC_F_WRITE) + { + Log2(("%s vqueueGet: %s IN seg=%u desc_idx=%u addr=%p cb=%u\n", INSTANCE(pState), + QUEUENAME(pState, pQueue), pElem->nIn, idx, desc.u64Addr, desc.uLen)); + pSeg = &pElem->aSegsIn[pElem->nIn++]; + } + else + { + Log2(("%s vqueueGet: %s OUT seg=%u desc_idx=%u addr=%p cb=%u\n", INSTANCE(pState), + QUEUENAME(pState, pQueue), pElem->nOut, idx, desc.u64Addr, desc.uLen)); + pSeg = &pElem->aSegsOut[pElem->nOut++]; + } + + pSeg->addr = desc.u64Addr; + pSeg->cb = desc.uLen; + pSeg->pv = NULL; + + idx = desc.u16Next; + } while (desc.u16Flags & VRINGDESC_F_NEXT); + + Log2(("%s vqueueGet: %s head_desc_idx=%u nIn=%u nOut=%u\n", INSTANCE(pState), + QUEUENAME(pState, pQueue), pElem->uIndex, pElem->nIn, pElem->nOut)); + return true; +} + +uint16_t vringReadUsedIndex(PVPCISTATE pState, PVRING pVRing) +{ + uint16_t tmp; + PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), + pVRing->addrUsed + RT_UOFFSETOF(VRINGUSED, uIndex), + &tmp, sizeof(tmp)); + return tmp; +} + +void vringWriteUsedIndex(PVPCISTATE pState, PVRING pVRing, uint16_t u16Value) +{ + PDMDevHlpPCIPhysWrite(pState->CTX_SUFF(pDevIns), + pVRing->addrUsed + RT_UOFFSETOF(VRINGUSED, uIndex), + &u16Value, sizeof(u16Value)); +} + +void vringWriteUsedElem(PVPCISTATE pState, PVRING pVRing, uint32_t uIndex, uint32_t uId, uint32_t uLen) +{ + VRINGUSEDELEM elem; + + elem.uId = uId; + elem.uLen = uLen; + PDMDevHlpPCIPhysWrite(pState->CTX_SUFF(pDevIns), + pVRing->addrUsed + RT_UOFFSETOF_DYN(VRINGUSED, aRing[uIndex % pVRing->uSize]), + &elem, sizeof(elem)); +} + + +void vqueuePut(PVPCISTATE pState, PVQUEUE pQueue, + PVQUEUEELEM pElem, uint32_t uTotalLen, uint32_t uReserved) +{ + Log2(("%s vqueuePut: %s" + " desc_idx=%u acb=%u (%u)\n", + INSTANCE(pState), QUEUENAME(pState, pQueue), + pElem->uIndex, uTotalLen, uReserved)); + + Assert(uReserved < uTotalLen); + + uint32_t cbLen = uTotalLen - uReserved; + uint32_t cbSkip = uReserved; + + for (unsigned i = 0; i < pElem->nIn && cbLen > 0; ++i) + { + if (cbSkip >= pElem->aSegsIn[i].cb) /* segment completely skipped? */ + { + cbSkip -= pElem->aSegsIn[i].cb; + continue; + } + + uint32_t cbSegLen = pElem->aSegsIn[i].cb - cbSkip; + if (cbSegLen > cbLen) /* last segment only partially used? */ + cbSegLen = cbLen; + + /* + * XXX: We should assert pv != NULL, but we need to check and + * fix all callers first. + */ + if (pElem->aSegsIn[i].pv != NULL) + { + Log2(("%s vqueuePut: %s" + " used_idx=%u seg=%u addr=%p pv=%p cb=%u acb=%u\n", + INSTANCE(pState), QUEUENAME(pState, pQueue), + pQueue->uNextUsedIndex, i, + (void *)pElem->aSegsIn[i].addr, pElem->aSegsIn[i].pv, + pElem->aSegsIn[i].cb, cbSegLen)); + + PDMDevHlpPCIPhysWrite(pState->CTX_SUFF(pDevIns), + pElem->aSegsIn[i].addr + cbSkip, + pElem->aSegsIn[i].pv, + cbSegLen); + } + + cbSkip = 0; + cbLen -= cbSegLen; + } + + Log2(("%s vqueuePut: %s" + " used_idx=%u guest_used_idx=%u id=%u len=%u\n", + INSTANCE(pState), QUEUENAME(pState, pQueue), + pQueue->uNextUsedIndex, vringReadUsedIndex(pState, &pQueue->VRing), + pElem->uIndex, uTotalLen)); + + vringWriteUsedElem(pState, &pQueue->VRing, + pQueue->uNextUsedIndex++, + pElem->uIndex, uTotalLen); +} + + +void vqueueNotify(PVPCISTATE pState, PVQUEUE pQueue) +{ + LogFlow(("%s vqueueNotify: %s availFlags=%x guestFeatures=%x vqueue is %sempty\n", + INSTANCE(pState), QUEUENAME(pState, pQueue), + vringReadAvailFlags(pState, &pQueue->VRing), + pState->uGuestFeatures, vqueueIsEmpty(pState, pQueue)?"":"not ")); + if (!(vringReadAvailFlags(pState, &pQueue->VRing) & VRINGAVAIL_F_NO_INTERRUPT) + || ((pState->uGuestFeatures & VPCI_F_NOTIFY_ON_EMPTY) && vqueueIsEmpty(pState, pQueue))) + { + int rc = vpciRaiseInterrupt(pState, VERR_INTERNAL_ERROR, VPCI_ISR_QUEUE); + if (RT_FAILURE(rc)) + Log(("%s vqueueNotify: Failed to raise an interrupt (%Rrc).\n", INSTANCE(pState), rc)); + } + else + { + STAM_COUNTER_INC(&pState->StatIntsSkipped); + } + +} + +void vqueueSync(PVPCISTATE pState, PVQUEUE pQueue) +{ + Log2(("%s vqueueSync: %s old_used_idx=%u new_used_idx=%u\n", INSTANCE(pState), + QUEUENAME(pState, pQueue), vringReadUsedIndex(pState, &pQueue->VRing), pQueue->uNextUsedIndex)); + vringWriteUsedIndex(pState, &pQueue->VRing, pQueue->uNextUsedIndex); + vqueueNotify(pState, pQueue); +} + +void vpciReset(PVPCISTATE pState) +{ + pState->uGuestFeatures = 0; + pState->uQueueSelector = 0; + pState->uStatus = 0; + pState->uISR = 0; + + for (unsigned i = 0; i < pState->nQueues; i++) + vqueueReset(&pState->Queues[i]); +} + + +/** + * Raise interrupt. + * + * @param pState The device state structure. + * @param rcBusy Status code to return when the critical section is busy. + * @param u8IntCause Interrupt cause bit mask to set in PCI ISR port. + */ +int vpciRaiseInterrupt(VPCISTATE *pState, int rcBusy, uint8_t u8IntCause) +{ + RT_NOREF_PV(rcBusy); + // int rc = vpciCsEnter(pState, rcBusy); + // if (RT_UNLIKELY(rc != VINF_SUCCESS)) + // return rc; + + STAM_COUNTER_INC(&pState->StatIntsRaised); + LogFlow(("%s vpciRaiseInterrupt: u8IntCause=%x\n", + INSTANCE(pState), u8IntCause)); + + pState->uISR |= u8IntCause; + PDMDevHlpPCISetIrq(pState->CTX_SUFF(pDevIns), 0, 1); + // vpciCsLeave(pState); + return VINF_SUCCESS; +} + +/** + * Lower interrupt. + * + * @param pState The device state structure. + */ +static void vpciLowerInterrupt(VPCISTATE *pState) +{ + LogFlow(("%s vpciLowerInterrupt\n", INSTANCE(pState))); + PDMDevHlpPCISetIrq(pState->CTX_SUFF(pDevIns), 0, 0); +} + +DECLINLINE(uint32_t) vpciGetHostFeatures(PVPCISTATE pState, + PFNGETHOSTFEATURES pfnGetHostFeatures) +{ + return pfnGetHostFeatures(pState) + | VPCI_F_NOTIFY_ON_EMPTY; +} + +/** + * Port I/O Handler for IN operations. + * + * @returns VBox status code. + * + * @param pDevIns The device instance. + * @param pvUser Pointer to the device state structure. + * @param Port Port number used for the IN operation. + * @param pu32 Where to store the result. + * @param cb Number of bytes read. + * @param pCallbacks Pointer to the callbacks. + * @thread EMT + */ +int vpciIOPortIn(PPDMDEVINS pDevIns, + void *pvUser, + RTIOPORT Port, + uint32_t *pu32, + unsigned cb, + PCVPCIIOCALLBACKS pCallbacks) +{ + VPCISTATE *pState = PDMINS_2_DATA(pDevIns, VPCISTATE *); + int rc = VINF_SUCCESS; + STAM_PROFILE_ADV_START(&pState->CTX_SUFF(StatIORead), a); + RT_NOREF_PV(pvUser); + + /* + * We probably do not need to enter critical section when reading registers + * as the most of them are either constant or being changed during + * initialization only, the exception being ISR which can be raced by all + * threads but I see no big harm in it. It also happens to be the most read + * register as it gets read in interrupt handler. By dropping cs protection + * here we gain the ability to deliver RX packets to the guest while TX is + * holding cs transmitting queued packets. + * + rc = vpciCsEnter(pState, VINF_IOM_R3_IOPORT_READ); + if (RT_UNLIKELY(rc != VINF_SUCCESS)) + { + STAM_PROFILE_ADV_STOP(&pState->CTX_SUFF(StatIORead), a); + return rc; + }*/ + + Port -= pState->IOPortBase; + switch (Port) + { + case VPCI_HOST_FEATURES: + /* Tell the guest what features we support. */ + *pu32 = vpciGetHostFeatures(pState, pCallbacks->pfnGetHostFeatures) + | VPCI_F_BAD_FEATURE; + break; + + case VPCI_GUEST_FEATURES: + *pu32 = pState->uGuestFeatures; + break; + + case VPCI_QUEUE_PFN: + *pu32 = pState->Queues[pState->uQueueSelector].uPageNumber; + break; + + case VPCI_QUEUE_NUM: + Assert(cb == 2); + *(uint16_t*)pu32 = pState->Queues[pState->uQueueSelector].VRing.uSize; + break; + + case VPCI_QUEUE_SEL: + Assert(cb == 2); + *(uint16_t*)pu32 = pState->uQueueSelector; + break; + + case VPCI_STATUS: + Assert(cb == 1); + *(uint8_t*)pu32 = pState->uStatus; + break; + + case VPCI_ISR: + Assert(cb == 1); + *(uint8_t*)pu32 = pState->uISR; + pState->uISR = 0; /* read clears all interrupts */ + vpciLowerInterrupt(pState); + break; + + default: + if (Port >= VPCI_CONFIG) + rc = pCallbacks->pfnGetConfig(pState, Port - VPCI_CONFIG, cb, pu32); + else + { + *pu32 = 0xFFFFFFFF; + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "%s vpciIOPortIn: no valid port at offset port=%RTiop cb=%08x\n", + INSTANCE(pState), Port, cb); + } + break; + } + Log3(("%s vpciIOPortIn: At %RTiop in %0*x\n", INSTANCE(pState), Port, cb*2, *pu32)); + STAM_PROFILE_ADV_STOP(&pState->CTX_SUFF(StatIORead), a); + //vpciCsLeave(pState); + return rc; +} + + +/** + * Port I/O Handler for OUT operations. + * + * @returns VBox status code. + * + * @param pDevIns The device instance. + * @param pvUser User argument. + * @param Port Port number used for the IN operation. + * @param u32 The value to output. + * @param cb The value size in bytes. + * @param pCallbacks Pointer to the callbacks. + * @thread EMT + */ +int vpciIOPortOut(PPDMDEVINS pDevIns, + void *pvUser, + RTIOPORT Port, + uint32_t u32, + unsigned cb, + PCVPCIIOCALLBACKS pCallbacks) +{ + VPCISTATE *pState = PDMINS_2_DATA(pDevIns, VPCISTATE *); + int rc = VINF_SUCCESS; + bool fHasBecomeReady; + STAM_PROFILE_ADV_START(&pState->CTX_SUFF(StatIOWrite), a); + RT_NOREF_PV(pvUser); + + Port -= pState->IOPortBase; + Log3(("%s virtioIOPortOut: At %RTiop out %0*x\n", INSTANCE(pState), Port, cb*2, u32)); + + switch (Port) + { + case VPCI_GUEST_FEATURES: + { + const uint32_t uHostFeatures = vpciGetHostFeatures(pState, pCallbacks->pfnGetHostFeatures); + + if (RT_LIKELY((u32 & ~uHostFeatures) == 0)) + { + pState->uGuestFeatures = u32; + } + else + { + /* + * Guest requests features we don't advertise. Stick + * to the minimum if negotiation looks completely + * botched, otherwise restrict to advertised features. + */ + if (u32 & VPCI_F_BAD_FEATURE) + { + Log(("%s WARNING! Guest failed to negotiate properly (guest=%x)\n", + INSTANCE(pState), u32)); + pState->uGuestFeatures = pCallbacks->pfnGetHostMinimalFeatures(pState); + } + else + { + Log(("%s Guest asked for features host does not support! (host=%x guest=%x)\n", + INSTANCE(pState), uHostFeatures, u32)); + pState->uGuestFeatures = u32 & uHostFeatures; + } + } + pCallbacks->pfnSetHostFeatures(pState, pState->uGuestFeatures); + break; + } + + case VPCI_QUEUE_PFN: + /* + * The guest is responsible for allocating the pages for queues, + * here it provides us with the page number of descriptor table. + * Note that we provide the size of the queue to the guest via + * VIRTIO_PCI_QUEUE_NUM. + */ + pState->Queues[pState->uQueueSelector].uPageNumber = u32; + if (u32) + vqueueInit(&pState->Queues[pState->uQueueSelector], u32); + else + rc = pCallbacks->pfnReset(pState); + break; + + case VPCI_QUEUE_SEL: + Assert(cb == 2); + u32 &= 0xFFFF; + if (u32 < pState->nQueues) + pState->uQueueSelector = u32; + else + Log3(("%s vpciIOPortOut: Invalid queue selector %08x\n", INSTANCE(pState), u32)); + break; + + case VPCI_QUEUE_NOTIFY: +#ifdef IN_RING3 + Assert(cb == 2); + u32 &= 0xFFFF; + if (u32 < pState->nQueues) + { + RT_UNTRUSTED_VALIDATED_FENCE(); + if (pState->Queues[u32].VRing.addrDescriptors) + { + // rc = vpciCsEnter(pState, VERR_SEM_BUSY); + // if (RT_LIKELY(rc == VINF_SUCCESS)) + // { + pState->Queues[u32].pfnCallback(pState, &pState->Queues[u32]); + // vpciCsLeave(pState); + // } + } + else + Log(("%s The queue (#%d) being notified has not been initialized.\n", + INSTANCE(pState), u32)); + } + else + Log(("%s Invalid queue number (%d)\n", INSTANCE(pState), u32)); +#else + rc = VINF_IOM_R3_IOPORT_WRITE; +#endif + break; + + case VPCI_STATUS: + Assert(cb == 1); + u32 &= 0xFF; + fHasBecomeReady = !(pState->uStatus & VPCI_STATUS_DRV_OK) && (u32 & VPCI_STATUS_DRV_OK); + pState->uStatus = u32; + /* Writing 0 to the status port triggers device reset. */ + if (u32 == 0) + rc = pCallbacks->pfnReset(pState); + else if (fHasBecomeReady) + { + /* Older hypervisors were lax and did not enforce bus mastering. Older guests + * (Linux prior to 2.6.34, NetBSD 6.x) were lazy and did not enable bus mastering. + * We automagically enable bus mastering on driver initialization to make existing + * drivers work. + */ + PDMPciDevSetCommand(&pState->pciDevice, PDMPciDevGetCommand(&pState->pciDevice) | PCI_COMMAND_BUSMASTER); + + pCallbacks->pfnReady(pState); + } + break; + + default: + if (Port >= VPCI_CONFIG) + rc = pCallbacks->pfnSetConfig(pState, Port - VPCI_CONFIG, cb, &u32); + else + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "%s vpciIOPortOut: no valid port at offset Port=%RTiop cb=%08x\n", + INSTANCE(pState), Port, cb); + break; + } + + STAM_PROFILE_ADV_STOP(&pState->CTX_SUFF(StatIOWrite), a); + return rc; +} + +#ifdef IN_RING3 + +/** + * @interface_method_impl{PDMIBASE,pfnQueryInterface} + */ +void *vpciQueryInterface(struct PDMIBASE *pInterface, const char *pszIID) +{ + VPCISTATE *pThis = IFACE_TO_STATE(pInterface, IBase); + Assert(&pThis->IBase == pInterface); + + PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase); + PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->ILeds); + return NULL; +} + +/** + * Gets the pointer to the status LED of a unit. + * + * @returns VBox status code. + * @param pInterface Pointer to the interface structure. + * @param iLUN The unit which status LED we desire. + * @param ppLed Where to store the LED pointer. + * @thread EMT + */ +static DECLCALLBACK(int) vpciQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed) +{ + VPCISTATE *pState = IFACE_TO_STATE(pInterface, ILeds); + int rc = VERR_PDM_LUN_NOT_FOUND; + + if (iLUN == 0) + { + *ppLed = &pState->led; + rc = VINF_SUCCESS; + } + return rc; +} + +/** + * Turns on/off the write status LED. + * + * @returns VBox status code. + * @param pState Pointer to the device state structure. + * @param fOn New LED state. + */ +void vpciSetWriteLed(PVPCISTATE pState, bool fOn) +{ + LogFlow(("%s vpciSetWriteLed: %s\n", INSTANCE(pState), fOn?"on":"off")); + if (fOn) + pState->led.Asserted.s.fWriting = pState->led.Actual.s.fWriting = 1; + else + pState->led.Actual.s.fWriting = fOn; +} + +/** + * Turns on/off the read status LED. + * + * @returns VBox status code. + * @param pState Pointer to the device state structure. + * @param fOn New LED state. + */ +void vpciSetReadLed(PVPCISTATE pState, bool fOn) +{ + LogFlow(("%s vpciSetReadLed: %s\n", INSTANCE(pState), fOn?"on":"off")); + if (fOn) + pState->led.Asserted.s.fReading = pState->led.Actual.s.fReading = 1; + else + pState->led.Actual.s.fReading = fOn; +} + + +#if 0 /* unused */ +/** + * Sets 32-bit register in PCI configuration space. + * @param refPciDev The PCI device. + * @param uOffset The register offset. + * @param u32Value The value to store in the register. + * @thread EMT + */ +DECLINLINE(void) vpciCfgSetU32(PDMPCIDEV& refPciDev, uint32_t uOffset, uint32_t u32Value) +{ + Assert(uOffset+sizeof(u32Value) <= sizeof(refPciDev.config)); + *(uint32_t*)&refPciDev.config[uOffset] = u32Value; +} +#endif /* unused */ + + +#ifdef DEBUG +static void vpciDumpState(PVPCISTATE pState, const char *pcszCaller) +{ + Log2(("vpciDumpState: (called from %s)\n" + " uGuestFeatures = 0x%08x\n" + " uQueueSelector = 0x%04x\n" + " uStatus = 0x%02x\n" + " uISR = 0x%02x\n", + pcszCaller, + pState->uGuestFeatures, + pState->uQueueSelector, + pState->uStatus, + pState->uISR)); + + for (unsigned i = 0; i < pState->nQueues; i++) + Log2((" %s queue:\n" + " VRing.uSize = %u\n" + " VRing.addrDescriptors = %p\n" + " VRing.addrAvail = %p\n" + " VRing.addrUsed = %p\n" + " uNextAvailIndex = %u\n" + " uNextUsedIndex = %u\n" + " uPageNumber = %x\n", + pState->Queues[i].pcszName, + pState->Queues[i].VRing.uSize, + pState->Queues[i].VRing.addrDescriptors, + pState->Queues[i].VRing.addrAvail, + pState->Queues[i].VRing.addrUsed, + pState->Queues[i].uNextAvailIndex, + pState->Queues[i].uNextUsedIndex, + pState->Queues[i].uPageNumber)); +} +#else +# define vpciDumpState(x, s) do {} while (0) +#endif + +/** + * Saves the state of device. + * + * @returns VBox status code. + * @param pDevIns The device instance. + * @param pSSM The handle to the saved state. + */ +int vpciSaveExec(PVPCISTATE pState, PSSMHANDLE pSSM) +{ + int rc; + + vpciDumpState(pState, "vpciSaveExec"); + + rc = SSMR3PutU32(pSSM, pState->uGuestFeatures); + AssertRCReturn(rc, rc); + rc = SSMR3PutU16(pSSM, pState->uQueueSelector); + AssertRCReturn(rc, rc); + rc = SSMR3PutU8( pSSM, pState->uStatus); + AssertRCReturn(rc, rc); + rc = SSMR3PutU8( pSSM, pState->uISR); + AssertRCReturn(rc, rc); + + /* Save queue states */ + rc = SSMR3PutU32(pSSM, pState->nQueues); + AssertRCReturn(rc, rc); + for (unsigned i = 0; i < pState->nQueues; i++) + { + rc = SSMR3PutU16(pSSM, pState->Queues[i].VRing.uSize); + AssertRCReturn(rc, rc); + rc = SSMR3PutU32(pSSM, pState->Queues[i].uPageNumber); + AssertRCReturn(rc, rc); + rc = SSMR3PutU16(pSSM, pState->Queues[i].uNextAvailIndex); + AssertRCReturn(rc, rc); + rc = SSMR3PutU16(pSSM, pState->Queues[i].uNextUsedIndex); + AssertRCReturn(rc, rc); + } + + return VINF_SUCCESS; +} + +/** + * Loads a saved device state. + * + * @returns VBox status code. + * @param pDevIns The device instance. + * @param pSSM The handle to the saved state. + * @param uVersion The data unit version number. + * @param uPass The data pass. + */ +int vpciLoadExec(PVPCISTATE pState, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass, uint32_t nQueues) +{ + int rc; + + if (uPass == SSM_PASS_FINAL) + { + /* Restore state data */ + rc = SSMR3GetU32(pSSM, &pState->uGuestFeatures); + AssertRCReturn(rc, rc); + rc = SSMR3GetU16(pSSM, &pState->uQueueSelector); + AssertRCReturn(rc, rc); + rc = SSMR3GetU8( pSSM, &pState->uStatus); + AssertRCReturn(rc, rc); + rc = SSMR3GetU8( pSSM, &pState->uISR); + AssertRCReturn(rc, rc); + + /* Restore queues */ + if (uVersion > VIRTIO_SAVEDSTATE_VERSION_3_1_BETA1) + { + rc = SSMR3GetU32(pSSM, &pState->nQueues); + AssertRCReturn(rc, rc); + } + else + pState->nQueues = nQueues; + AssertLogRelMsgReturn(pState->nQueues <= VIRTIO_MAX_NQUEUES, ("%#x\n", pState->nQueues), VERR_SSM_LOAD_CONFIG_MISMATCH); + AssertLogRelMsgReturn(pState->uQueueSelector < pState->nQueues || (pState->nQueues == 0 && pState->uQueueSelector), + ("uQueueSelector=%u nQueues=%u\n", pState->uQueueSelector, pState->nQueues), + VERR_SSM_LOAD_CONFIG_MISMATCH); + + for (unsigned i = 0; i < pState->nQueues; i++) + { + rc = SSMR3GetU16(pSSM, &pState->Queues[i].VRing.uSize); + AssertRCReturn(rc, rc); + rc = SSMR3GetU32(pSSM, &pState->Queues[i].uPageNumber); + AssertRCReturn(rc, rc); + + if (pState->Queues[i].uPageNumber) + vqueueInit(&pState->Queues[i], pState->Queues[i].uPageNumber); + + rc = SSMR3GetU16(pSSM, &pState->Queues[i].uNextAvailIndex); + AssertRCReturn(rc, rc); + rc = SSMR3GetU16(pSSM, &pState->Queues[i].uNextUsedIndex); + AssertRCReturn(rc, rc); + } + } + + vpciDumpState(pState, "vpciLoadExec"); + + return VINF_SUCCESS; +} + +/** + * Set PCI configuration space registers. + * + * @param pci Reference to PCI device structure. + * @param uDeviceId VirtiO Device Id + * @param uClass Class of PCI device (network, etc) + * @thread EMT + */ +static DECLCALLBACK(void) vpciConfigure(PDMPCIDEV& pci, + uint16_t uDeviceId, + uint16_t uClass) +{ + /* Configure PCI Device, assume 32-bit mode ******************************/ + PCIDevSetVendorId(&pci, DEVICE_PCI_VENDOR_ID); + PCIDevSetDeviceId(&pci, DEVICE_PCI_BASE_ID + uDeviceId); + PDMPciDevSetWord(&pci, VBOX_PCI_SUBSYSTEM_VENDOR_ID, DEVICE_PCI_SUBSYSTEM_VENDOR_ID); + PDMPciDevSetWord(&pci, VBOX_PCI_SUBSYSTEM_ID, DEVICE_PCI_SUBSYSTEM_BASE_ID + uDeviceId); + + /* ABI version, must be equal 0 as of 2.6.30 kernel. */ + PDMPciDevSetByte(&pci, VBOX_PCI_REVISION_ID, 0x00); + /* Ethernet adapter */ + PDMPciDevSetByte(&pci, VBOX_PCI_CLASS_PROG, 0x00); + PDMPciDevSetWord(&pci, VBOX_PCI_CLASS_DEVICE, uClass); + /* Interrupt Pin: INTA# */ + PDMPciDevSetByte(&pci, VBOX_PCI_INTERRUPT_PIN, 0x01); + +#ifdef VBOX_WITH_MSI_DEVICES + PCIDevSetCapabilityList(&pci, 0x80); + PCIDevSetStatus( &pci, VBOX_PCI_STATUS_CAP_LIST); +#endif +} + +#ifdef VBOX_WITH_STATISTICS +/* WARNING! This function must never be used in multithreaded context! */ +static const char *vpciCounter(const char *pszDevFmt, + const char *pszCounter) +{ + static char s_szCounterName[80]; + + RTStrPrintf(s_szCounterName, sizeof(s_szCounterName), + "/Devices/%s/%s", pszDevFmt, pszCounter); + + return s_szCounterName; +} +#endif + +/// @todo header +int vpciConstruct(PPDMDEVINS pDevIns, VPCISTATE *pState, + int iInstance, const char *pcszNameFmt, + uint16_t uDeviceId, uint16_t uClass, + uint32_t nQueues) +{ + /* Init handles and log related stuff. */ + RTStrPrintf(pState->szInstance, sizeof(pState->szInstance), + pcszNameFmt, iInstance); + + pState->pDevInsR3 = pDevIns; + pState->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns); + pState->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns); + pState->led.u32Magic = PDMLED_MAGIC; + + pState->ILeds.pfnQueryStatusLed = vpciQueryStatusLed; + + /* Initialize critical section. */ + int rc = PDMDevHlpCritSectInit(pDevIns, &pState->cs, RT_SRC_POS, "%s", pState->szInstance); + if (RT_FAILURE(rc)) + return rc; + + /* Set PCI config registers */ + vpciConfigure(pState->pciDevice, uDeviceId, uClass); + /* Register PCI device */ + rc = PDMDevHlpPCIRegister(pDevIns, &pState->pciDevice); + if (RT_FAILURE(rc)) + return rc; + +#ifdef VBOX_WITH_MSI_DEVICES +#if 0 + { + PDMMSIREG aMsiReg; + + RT_ZERO(aMsiReg); + aMsiReg.cMsixVectors = 1; + aMsiReg.iMsixCapOffset = 0x80; + aMsiReg.iMsixNextOffset = 0x0; + aMsiReg.iMsixBar = 0; + rc = PDMDevHlpPCIRegisterMsi(pDevIns, &aMsiReg); + if (RT_FAILURE (rc)) + PCIDevSetCapabilityList(&pState->pciDevice, 0x0); + } +#endif +#endif + + /* Status driver */ + PPDMIBASE pBase; + rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pState->IBase, &pBase, "Status Port"); + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach the status LUN")); + pState->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS); + + pState->nQueues = nQueues; + +#if defined(VBOX_WITH_STATISTICS) + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIOReadR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in R3", vpciCounter(pcszNameFmt, "IO/ReadR3"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIOReadR0, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in R0", vpciCounter(pcszNameFmt, "IO/ReadR0"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIOReadRC, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in RC", vpciCounter(pcszNameFmt, "IO/ReadRC"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIOWriteR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in R3", vpciCounter(pcszNameFmt, "IO/WriteR3"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIOWriteR0, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in R0", vpciCounter(pcszNameFmt, "IO/WriteR0"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIOWriteRC, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in RC", vpciCounter(pcszNameFmt, "IO/WriteRC"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIntsRaised, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of raised interrupts", vpciCounter(pcszNameFmt, "Interrupts/Raised"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatIntsSkipped, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of skipped interrupts", vpciCounter(pcszNameFmt, "Interrupts/Skipped"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatCsR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling CS wait in R3", vpciCounter(pcszNameFmt, "Cs/CsR3"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatCsR0, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling CS wait in R0", vpciCounter(pcszNameFmt, "Cs/CsR0"), iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pState->StatCsRC, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling CS wait in RC", vpciCounter(pcszNameFmt, "Cs/CsRC"), iInstance); +#endif /* VBOX_WITH_STATISTICS */ + + return rc; +} + +/** + * Destruct PCI-related part of device. + * + * We need to free non-VM resources only. + * + * @returns VBox status code. + * @param pState The device state structure. + */ +int vpciDestruct(VPCISTATE* pState) +{ + Log(("%s Destroying PCI instance\n", INSTANCE(pState))); + + if (PDMCritSectIsInitialized(&pState->cs)) + PDMR3CritSectDelete(&pState->cs); + + return VINF_SUCCESS; +} + +/** + * Device relocation callback. + * + * When this callback is called the device instance data, and if the + * device have a GC component, is being relocated, or/and the selectors + * have been changed. The device must use the chance to perform the + * necessary pointer relocations and data updates. + * + * Before the GC code is executed the first time, this function will be + * called with a 0 delta so GC pointer calculations can be one in one place. + * + * @param pDevIns Pointer to the device instance. + * @param offDelta The relocation delta relative to the old location. + * + * @remark A relocation CANNOT fail. + */ +void vpciRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta) +{ + RT_NOREF(offDelta); + VPCISTATE *pState = PDMINS_2_DATA(pDevIns, VPCISTATE*); + pState->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns); + // TBD +} + +PVQUEUE vpciAddQueue(VPCISTATE* pState, unsigned uSize, PFNVPCIQUEUECALLBACK pfnCallback, const char *pcszName) +{ + PVQUEUE pQueue = NULL; + /* Find an empty queue slot */ + for (unsigned i = 0; i < pState->nQueues; i++) + { + if (pState->Queues[i].VRing.uSize == 0) + { + pQueue = &pState->Queues[i]; + break; + } + } + + if (!pQueue) + { + Log(("%s Too many queues being added, no empty slots available!\n", INSTANCE(pState))); + } + else + { + pQueue->VRing.uSize = uSize; + pQueue->VRing.addrDescriptors = 0; + pQueue->uPageNumber = 0; + pQueue->pfnCallback = pfnCallback; + pQueue->pcszName = pcszName; + } + + return pQueue; +} + +#endif /* IN_RING3 */ + +#endif /* VBOX_DEVICE_STRUCT_TESTCASE */ + diff --git a/src/VBox/Devices/VirtIO/Virtio.h b/src/VBox/Devices/VirtIO/Virtio.h new file mode 100644 index 00000000..6b569bb0 --- /dev/null +++ b/src/VBox/Devices/VirtIO/Virtio.h @@ -0,0 +1,334 @@ +/* $Id: Virtio.h $ */ +/** @file + * Virtio.h - Virtio Declarations + */ + +/* + * Copyright (C) 2009-2019 Oracle Corporation + * + * This file is part of VirtualBox Open Source Edition (OSE), as + * available from http://www.virtualbox.org. This file is free software; + * you can redistribute it and/or modify it under the terms of the GNU + * General Public License (GPL) as published by the Free Software + * Foundation, in version 2 as it comes in the "COPYING" file of the + * VirtualBox OSE distribution. VirtualBox OSE is distributed in the + * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. + */ + +#ifndef VBOX_INCLUDED_SRC_VirtIO_Virtio_h +#define VBOX_INCLUDED_SRC_VirtIO_Virtio_h +#ifndef RT_WITHOUT_PRAGMA_ONCE +# pragma once +#endif + +#include <iprt/ctype.h> + + +/** @name Saved state versions. + * The saved state version is changed if either common or any of specific + * parts are changed. That is, it is perfectly possible that the version + * of saved vnet state will increase as a result of change in vblk structure + * for example. + */ +#define VIRTIO_SAVEDSTATE_VERSION_3_1_BETA1 1 +#define VIRTIO_SAVEDSTATE_VERSION 2 +/** @} */ + +#define DEVICE_PCI_VENDOR_ID 0x1AF4 +#define DEVICE_PCI_BASE_ID 0x1000 +#define DEVICE_PCI_SUBSYSTEM_VENDOR_ID 0x1AF4 +#define DEVICE_PCI_SUBSYSTEM_BASE_ID 1 + +#define VIRTIO_MAX_NQUEUES 3 + +#define VPCI_HOST_FEATURES 0x0 +#define VPCI_GUEST_FEATURES 0x4 +#define VPCI_QUEUE_PFN 0x8 +#define VPCI_QUEUE_NUM 0xC +#define VPCI_QUEUE_SEL 0xE +#define VPCI_QUEUE_NOTIFY 0x10 +#define VPCI_STATUS 0x12 +#define VPCI_ISR 0x13 +#define VPCI_CONFIG 0x14 + +#define VPCI_ISR_QUEUE 0x1 +#define VPCI_ISR_CONFIG 0x3 + +#define VPCI_STATUS_ACK 0x01 +#define VPCI_STATUS_DRV 0x02 +#define VPCI_STATUS_DRV_OK 0x04 +#define VPCI_STATUS_FAILED 0x80 + +#define VPCI_F_NOTIFY_ON_EMPTY 0x01000000 +#define VPCI_F_ANY_LAYOUT 0x08000000 +#define VPCI_F_RING_INDIRECT_DESC 0x10000000 +#define VPCI_F_RING_EVENT_IDX 0x20000000 +#define VPCI_F_BAD_FEATURE 0x40000000 + +#define VRINGDESC_MAX_SIZE (2 * 1024 * 1024) +#define VRINGDESC_F_NEXT 0x01 +#define VRINGDESC_F_WRITE 0x02 +#define VRINGDESC_F_INDIRECT 0x04 + +typedef struct VRingDesc +{ + uint64_t u64Addr; + uint32_t uLen; + uint16_t u16Flags; + uint16_t u16Next; +} VRINGDESC; +typedef VRINGDESC *PVRINGDESC; + +#define VRINGAVAIL_F_NO_INTERRUPT 0x01 + +typedef struct VRingAvail +{ + uint16_t uFlags; + uint16_t uNextFreeIndex; + uint16_t auRing[1]; +} VRINGAVAIL; + +typedef struct VRingUsedElem +{ + uint32_t uId; + uint32_t uLen; +} VRINGUSEDELEM; + +#define VRINGUSED_F_NO_NOTIFY 0x01 + +typedef struct VRingUsed +{ + uint16_t uFlags; + uint16_t uIndex; + VRINGUSEDELEM aRing[1]; +} VRINGUSED; +typedef VRINGUSED *PVRINGUSED; + +#define VRING_MAX_SIZE 1024 + +typedef struct VRing +{ + uint16_t uSize; + uint16_t padding[3]; + RTGCPHYS addrDescriptors; + RTGCPHYS addrAvail; + RTGCPHYS addrUsed; +} VRING; +typedef VRING *PVRING; + +/** + * Queue callback (consumer?). + * + * @param pvState Pointer to the VirtIO PCI core state, VPCISTATE. + * @param pQueue Pointer to the queue structure. + */ +typedef DECLCALLBACK(void) FNVPCIQUEUECALLBACK(void *pvState, struct VQueue *pQueue); +/** Pointer to a VQUEUE callback function. */ +typedef FNVPCIQUEUECALLBACK *PFNVPCIQUEUECALLBACK; + +typedef struct VQueue +{ + VRING VRing; + uint16_t uNextAvailIndex; + uint16_t uNextUsedIndex; + uint32_t uPageNumber; + R3PTRTYPE(PFNVPCIQUEUECALLBACK) pfnCallback; + R3PTRTYPE(const char *) pcszName; +} VQUEUE; +typedef VQUEUE *PVQUEUE; + +typedef struct VQueueElemSeg +{ + RTGCPHYS addr; + void *pv; + uint32_t cb; +} VQUEUESEG; + +typedef struct VQueueElem +{ + uint32_t uIndex; + uint32_t nIn; + uint32_t nOut; + VQUEUESEG aSegsIn[VRING_MAX_SIZE]; + VQUEUESEG aSegsOut[VRING_MAX_SIZE]; +} VQUEUEELEM; +typedef VQUEUEELEM *PVQUEUEELEM; + + +enum VirtioDeviceType +{ + VIRTIO_NET_ID = 0, + VIRTIO_BLK_ID = 1, + VIRTIO_32BIT_HACK = 0x7fffffff +}; + + +/** + * The core (/common) state of the VirtIO PCI device + * + * @implements PDMILEDPORTS + */ +typedef struct VPCIState_st +{ + PDMCRITSECT cs; /**< Critical section - what is it protecting? */ + /* Read-only part, never changes after initialization. */ + char szInstance[8]; /**< Instance name, e.g. VNet#1. */ + +#if HC_ARCH_BITS != 64 + uint32_t padding1; +#endif + + /** Status LUN: Base interface. */ + PDMIBASE IBase; + /** Status LUN: LED port interface. */ + PDMILEDPORTS ILeds; + /** Status LUN: LED connector (peer). */ + R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector; + + PPDMDEVINSR3 pDevInsR3; /**< Device instance - R3. */ + PPDMDEVINSR0 pDevInsR0; /**< Device instance - R0. */ + PPDMDEVINSRC pDevInsRC; /**< Device instance - RC. */ + +#if HC_ARCH_BITS == 64 + uint32_t padding2; +#endif + + /** TODO */ + PDMPCIDEV pciDevice; + /** Base port of I/O space region. */ + RTIOPORT IOPortBase; + + /* Read/write part, protected with critical section. */ + /** Status LED. */ + PDMLED led; + + uint32_t uGuestFeatures; + uint16_t uQueueSelector; /**< An index in aQueues array. */ + uint8_t uStatus; /**< Device Status (bits are device-specific). */ + uint8_t uISR; /**< Interrupt Status Register. */ + +#if HC_ARCH_BITS != 64 + uint32_t padding3; +#endif + + uint32_t nQueues; /**< Actual number of queues used. */ + VQUEUE Queues[VIRTIO_MAX_NQUEUES]; + +#if defined(VBOX_WITH_STATISTICS) + STAMPROFILEADV StatIOReadR3; + STAMPROFILEADV StatIOReadR0; + STAMPROFILEADV StatIOReadRC; + STAMPROFILEADV StatIOWriteR3; + STAMPROFILEADV StatIOWriteR0; + STAMPROFILEADV StatIOWriteRC; + STAMCOUNTER StatIntsRaised; + STAMCOUNTER StatIntsSkipped; + STAMPROFILE StatCsR3; + STAMPROFILE StatCsR0; + STAMPROFILE StatCsRC; +#endif /* VBOX_WITH_STATISTICS */ +} VPCISTATE; +/** Pointer to the core (/common) state of a VirtIO PCI device. */ +typedef VPCISTATE *PVPCISTATE; + +typedef DECLCALLBACK(uint32_t) FNGETHOSTFEATURES(void *pvState); +typedef FNGETHOSTFEATURES *PFNGETHOSTFEATURES; + +/** @name VirtIO port I/O callbacks. + * @{ */ +typedef struct VPCIIOCALLBACKS +{ + DECLCALLBACKMEMBER(uint32_t, pfnGetHostFeatures)(void *pvState); + DECLCALLBACKMEMBER(uint32_t, pfnGetHostMinimalFeatures)(void *pvState); + DECLCALLBACKMEMBER(void, pfnSetHostFeatures)(void *pvState, uint32_t fFeatures); + DECLCALLBACKMEMBER(int, pfnGetConfig)(void *pvState, uint32_t offCfg, uint32_t cb, void *pvData); + DECLCALLBACKMEMBER(int, pfnSetConfig)(void *pvState, uint32_t offCfg, uint32_t cb, void *pvData); + DECLCALLBACKMEMBER(int, pfnReset)(void *pvState); + DECLCALLBACKMEMBER(void, pfnReady)(void *pvState); +} VPCIIOCALLBACKS; +/** Pointer to a const VirtIO port I/O callback structure. */ +typedef const VPCIIOCALLBACKS *PCVPCIIOCALLBACKS; +/** @} */ + +int vpciRaiseInterrupt(VPCISTATE *pState, int rcBusy, uint8_t u8IntCause); +int vpciIOPortIn(PPDMDEVINS pDevIns, + void *pvUser, + RTIOPORT port, + uint32_t *pu32, + unsigned cb, + PCVPCIIOCALLBACKS pCallbacks); + +int vpciIOPortOut(PPDMDEVINS pDevIns, + void *pvUser, + RTIOPORT port, + uint32_t u32, + unsigned cb, + PCVPCIIOCALLBACKS pCallbacks); + +void vpciSetWriteLed(PVPCISTATE pState, bool fOn); +void vpciSetReadLed(PVPCISTATE pState, bool fOn); +int vpciSaveExec(PVPCISTATE pState, PSSMHANDLE pSSM); +int vpciLoadExec(PVPCISTATE pState, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass, uint32_t nQueues); +int vpciConstruct(PPDMDEVINS pDevIns, VPCISTATE *pState, int iInstance, const char *pcszNameFmt, + uint16_t uDeviceId, uint16_t uClass, uint32_t nQueues); +int vpciDestruct(VPCISTATE* pState); +void vpciRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta); +void vpciReset(PVPCISTATE pState); +void *vpciQueryInterface(struct PDMIBASE *pInterface, const char *pszIID); +PVQUEUE vpciAddQueue(VPCISTATE* pState, unsigned uSize, PFNVPCIQUEUECALLBACK pfnCallback, const char *pcszName); + +#define VPCI_CS +DECLINLINE(int) vpciCsEnter(VPCISTATE *pState, int rcBusy) +{ +#ifdef VPCI_CS + STAM_PROFILE_START(&pState->CTX_SUFF(StatCs), a); + int rc = PDMCritSectEnter(&pState->cs, rcBusy); + STAM_PROFILE_STOP(&pState->CTX_SUFF(StatCs), a); + return rc; +#else + return VINF_SUCCESS; +#endif +} + +DECLINLINE(void) vpciCsLeave(VPCISTATE *pState) +{ +#ifdef VPCI_CS + PDMCritSectLeave(&pState->cs); +#endif +} + +void vringSetNotification(PVPCISTATE pState, PVRING pVRing, bool fEnabled); + +DECLINLINE(uint16_t) vringReadAvailIndex(PVPCISTATE pState, PVRING pVRing) +{ + uint16_t tmp; + + PDMDevHlpPhysRead(pState->CTX_SUFF(pDevIns), + pVRing->addrAvail + RT_UOFFSETOF(VRINGAVAIL, uNextFreeIndex), + &tmp, sizeof(tmp)); + return tmp; +} + +bool vqueueSkip(PVPCISTATE pState, PVQUEUE pQueue); +bool vqueueGet(PVPCISTATE pState, PVQUEUE pQueue, PVQUEUEELEM pElem, bool fRemove = true); +void vqueuePut(PVPCISTATE pState, PVQUEUE pQueue, PVQUEUEELEM pElem, uint32_t uLen, uint32_t uReserved = 0); +void vqueueNotify(PVPCISTATE pState, PVQUEUE pQueue); +void vqueueSync(PVPCISTATE pState, PVQUEUE pQueue); + +DECLINLINE(bool) vqueuePeek(PVPCISTATE pState, PVQUEUE pQueue, PVQUEUEELEM pElem) +{ + return vqueueGet(pState, pQueue, pElem, /* fRemove */ false); +} + +DECLINLINE(bool) vqueueIsReady(PVPCISTATE pState, PVQUEUE pQueue) +{ + NOREF(pState); + return !!pQueue->VRing.addrAvail; +} + +DECLINLINE(bool) vqueueIsEmpty(PVPCISTATE pState, PVQUEUE pQueue) +{ + return (vringReadAvailIndex(pState, &pQueue->VRing) == pQueue->uNextAvailIndex); +} + +#endif /* !VBOX_INCLUDED_SRC_VirtIO_Virtio_h */ |