diff options
Diffstat (limited to 'src/VBox/Devices/VirtIO/Virtio.h')
-rw-r--r-- | src/VBox/Devices/VirtIO/Virtio.h | 334 |
1 files changed, 334 insertions, 0 deletions
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 */ |