In virtio-1.0, it is not enough to pass the base address of the virtio
queue to the hypervisor (as a frame number); instead it will want the
addresses of the descriptor table, the available ring, and the used ring
separately. Pass the VRING object to the SetQueueAddress() member
function; this will enable a virtio-1.0 implementation. Convert the
current producers and consumers to this prototype.
Cc: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Cc: Jordan Justen <jordan.l.justen@intel.com>
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Laszlo Ersek <lersek@redhat.com>
Tested-by: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Reviewed-by: Jordan Justen <jordan.l.justen@intel.com>
#ifndef __VIRTIO_DEVICE_H__\r
#define __VIRTIO_DEVICE_H__\r
\r
#ifndef __VIRTIO_DEVICE_H__\r
#define __VIRTIO_DEVICE_H__\r
\r
+#include <IndustryStandard/Virtio.h>\r
+\r
// VirtIo Specification Revision: Major[31:24].Minor[23:16].Revision[15:0\r
#define VIRTIO_SPEC_REVISION(major,minor,revision) \\r
((((major) & 0xFF) << 24) | (((minor) & 0xFF) << 16) | ((revision) & 0xFFFF))\r
// VirtIo Specification Revision: Major[31:24].Minor[23:16].Revision[15:0\r
#define VIRTIO_SPEC_REVISION(major,minor,revision) \\r
((((major) & 0xFF) << 24) | (((minor) & 0xFF) << 16) | ((revision) & 0xFFFF))\r
- Write the queue address field in the Virtio Header.\r
-\r
- The parameter Address must be the base address of the virtqueue divided\r
- by 4096.\r
+ Write the queue address field(s) in the Virtio Header.\r
\r
@param[in] This This instance of VIRTIO_DEVICE_PROTOCOL\r
\r
\r
@param[in] This This instance of VIRTIO_DEVICE_PROTOCOL\r
\r
- @param[in] Address The 32-bit Queue Address field\r
+ @param[in] Ring The initialized VRING object to take the\r
+ addresses from.\r
\r
@retval EFI_SUCCESS The data was written successfully.\r
@retval EFI_UNSUPPORTED The underlying IO device doesn't support the\r
\r
@retval EFI_SUCCESS The data was written successfully.\r
@retval EFI_UNSUPPORTED The underlying IO device doesn't support the\r
EFI_STATUS\r
(EFIAPI *VIRTIO_SET_QUEUE_ADDRESS) (\r
IN VIRTIO_DEVICE_PROTOCOL *This,\r
EFI_STATUS\r
(EFIAPI *VIRTIO_SET_QUEUE_ADDRESS) (\r
IN VIRTIO_DEVICE_PROTOCOL *This,\r
\r
EFI_STATUS\r
VirtioMmioSetQueueAddress (\r
\r
EFI_STATUS\r
VirtioMmioSetQueueAddress (\r
- VIRTIO_DEVICE_PROTOCOL *This,\r
- UINT32 Address\r
+ IN VIRTIO_DEVICE_PROTOCOL *This,\r
+ IN VRING *Ring\r
\r
EFI_STATUS\r
VirtioMmioSetQueueAddress (\r
\r
EFI_STATUS\r
VirtioMmioSetQueueAddress (\r
- VIRTIO_DEVICE_PROTOCOL *This,\r
- UINT32 Address\r
+ IN VIRTIO_DEVICE_PROTOCOL *This,\r
+ IN VRING *Ring\r
)\r
{\r
VIRTIO_MMIO_DEVICE *Device;\r
\r
Device = VIRTIO_MMIO_DEVICE_FROM_VIRTIO_DEVICE (This);\r
\r
)\r
{\r
VIRTIO_MMIO_DEVICE *Device;\r
\r
Device = VIRTIO_MMIO_DEVICE_FROM_VIRTIO_DEVICE (This);\r
\r
- VIRTIO_CFG_WRITE (Device, VIRTIO_MMIO_OFFSET_QUEUE_PFN, Address);\r
+ VIRTIO_CFG_WRITE (Device, VIRTIO_MMIO_OFFSET_QUEUE_PFN,\r
+ (UINT32)((UINTN)Ring->Base >> EFI_PAGE_SHIFT));\r
\r
return EFI_SUCCESS;\r
}\r
\r
return EFI_SUCCESS;\r
}\r
//\r
// step 4c -- Report GPFN (guest-physical frame number) of queue.\r
//\r
//\r
// step 4c -- Report GPFN (guest-physical frame number) of queue.\r
//\r
- Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo,\r
- (UINT32) ((UINTN) Dev->Ring.Base >> EFI_PAGE_SHIFT));\r
+ Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo, &Dev->Ring);\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
//\r
// step 4c -- report GPFN (guest-physical frame number) of queue\r
//\r
//\r
// step 4c -- report GPFN (guest-physical frame number) of queue\r
//\r
- Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo,\r
- (UINT32) ((UINTN) Ring->Base >> EFI_PAGE_SHIFT));\r
+ Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo, Ring);\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
EFI_STATUS\r
EFIAPI\r
VirtioPciSetQueueAddress (\r
EFI_STATUS\r
EFIAPI\r
VirtioPciSetQueueAddress (\r
- VIRTIO_DEVICE_PROTOCOL *This,\r
- UINT32 Address\r
+ IN VIRTIO_DEVICE_PROTOCOL *This,\r
+ IN VRING *Ring\r
EFI_STATUS\r
EFIAPI\r
VirtioPciSetQueueAddress (\r
EFI_STATUS\r
EFIAPI\r
VirtioPciSetQueueAddress (\r
- VIRTIO_DEVICE_PROTOCOL *This,\r
- UINT32 Address\r
+ IN VIRTIO_DEVICE_PROTOCOL *This,\r
+ IN VRING *Ring\r
)\r
{\r
VIRTIO_PCI_DEVICE *Dev;\r
)\r
{\r
VIRTIO_PCI_DEVICE *Dev;\r
Dev = VIRTIO_PCI_DEVICE_FROM_VIRTIO_DEVICE (This);\r
\r
return VirtioPciIoWrite (Dev, VIRTIO_PCI_OFFSET_QUEUE_ADDRESS, sizeof (UINT32),\r
Dev = VIRTIO_PCI_DEVICE_FROM_VIRTIO_DEVICE (This);\r
\r
return VirtioPciIoWrite (Dev, VIRTIO_PCI_OFFSET_QUEUE_ADDRESS, sizeof (UINT32),\r
+ (UINT32)((UINTN)Ring->Base >> EFI_PAGE_SHIFT));\r
//\r
// step 4c -- Report GPFN (guest-physical frame number) of queue.\r
//\r
//\r
// step 4c -- Report GPFN (guest-physical frame number) of queue.\r
//\r
- Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo,\r
- (UINT32) ((UINTN) Dev->Ring.Base >> EFI_PAGE_SHIFT));\r
+ Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo, &Dev->Ring);\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
//\r
// step 4c -- Report GPFN (guest-physical frame number) of queue.\r
//\r
//\r
// step 4c -- Report GPFN (guest-physical frame number) of queue.\r
//\r
- Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo,\r
- (UINT32) ((UINTN) Dev->Ring.Base >> EFI_PAGE_SHIFT));\r
+ Status = Dev->VirtIo->SetQueueAddress (Dev->VirtIo, &Dev->Ring);\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r
if (EFI_ERROR (Status)) {\r
goto ReleaseQueue;\r
}\r