[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Qemu-devel] [RFC] Functions bus_foreach and device_find from libqos vir
From: |
Marc Marí |
Subject: |
[Qemu-devel] [RFC] Functions bus_foreach and device_find from libqos virtio API |
Date: |
Mon, 23 Jun 2014 16:55:22 +0200 |
Virtio header has been changed to compile and work with a real device.
Functions bus_foreach and device_find have been implemented for PCI.
Virtio-blk test case now opens a fake device.
Signed-off-by: Marc Marí <address@hidden>
---
tests/Makefile | 3 +-
tests/libqos/virtio-pci.c | 123 +++++++++++++++++++++++++++++++++
tests/libqos/virtio-pci.h | 40 +++++++++++
tests/libqos/virtio.h | 167 +++++++++++++++++++++++++++++++++++++++++++++
tests/virtio-blk-test.c | 38 +++++++++--
5 files changed, 364 insertions(+), 7 deletions(-)
create mode 100644 tests/libqos/virtio-pci.c
create mode 100644 tests/libqos/virtio-pci.h
create mode 100644 tests/libqos/virtio.h
diff --git a/tests/Makefile b/tests/Makefile
index 4caf7de..b85a036 100644
--- a/tests/Makefile
+++ b/tests/Makefile
@@ -282,6 +282,7 @@ libqos-obj-y += tests/libqos/i2c.o
libqos-pc-obj-y = $(libqos-obj-y) tests/libqos/pci-pc.o
libqos-pc-obj-y += tests/libqos/malloc-pc.o
libqos-omap-obj-y = $(libqos-obj-y) tests/libqos/i2c-omap.o
+libqos-virtio-obj-y = $(libqos-obj-y) $(libqos-pc-obj-y)
tests/libqos/virtio-pci.o
tests/rtc-test$(EXESUF): tests/rtc-test.o
tests/m48t59-test$(EXESUF): tests/m48t59-test.o
@@ -302,7 +303,7 @@ tests/eepro100-test$(EXESUF): tests/eepro100-test.o
tests/vmxnet3-test$(EXESUF): tests/vmxnet3-test.o
tests/ne2000-test$(EXESUF): tests/ne2000-test.o
tests/virtio-balloon-test$(EXESUF): tests/virtio-balloon-test.o
-tests/virtio-blk-test$(EXESUF): tests/virtio-blk-test.o
+tests/virtio-blk-test$(EXESUF): tests/virtio-blk-test.o $(libqos-virtio-obj-y)
tests/virtio-net-test$(EXESUF): tests/virtio-net-test.o
tests/virtio-rng-test$(EXESUF): tests/virtio-rng-test.o
tests/virtio-scsi-test$(EXESUF): tests/virtio-scsi-test.o
diff --git a/tests/libqos/virtio-pci.c b/tests/libqos/virtio-pci.c
new file mode 100644
index 0000000..62c238f
--- /dev/null
+++ b/tests/libqos/virtio-pci.c
@@ -0,0 +1,123 @@
+/*
+ * libqos virtio definitions
+ *
+ * Copyright (c) 2014 Marc Marí
+ *
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
+ * See the COPYING file in the top-level directory.
+ */
+
+#include <glib.h>
+#include <stdio.h>
+#include "libqtest.h"
+#include "libqos/virtio.h"
+#include "libqos/virtio-pci.h"
+#include "libqos/pci.h"
+#include "libqos/pci-pc.h"
+
+#include "hw/pci/pci_regs.h"
+
+static QVirtioDevice *qpcidevice_to_qvirtiodevice(QPCIDevice *dpci)
+{
+ QVirtioDevice *dvirtio;
+ dvirtio = g_malloc0(sizeof(*dvirtio));
+
+ if (dpci) {
+ dvirtio->device_id = qpci_config_readw(dpci, PCI_DEVICE_ID);
+ dvirtio->device_type = qpci_config_readw(dpci, PCI_SUBSYSTEM_ID);
+ dvirtio->location = dpci->devfn;
+ dvirtio->vq = g_malloc0(sizeof(QVirtQueue));
+ }
+
+ return dvirtio;
+}
+
+static void qvirtio_pci_assign_device(QVirtioDevice *d, void *data)
+{
+ QVirtioDevice *dev = data;
+ dev->device_id = d->device_id;
+ dev->location = d->location;
+ dev->device_type = d->device_type;
+ dev->vq = d->vq;
+}
+
+void qvirtio_pci_notify(QVirtioDevice *d, uint16_t vector)
+{
+
+}
+
+void qvirtio_pci_get_config(QVirtioDevice *d, void *config)
+{
+
+}
+
+void qvirtio_pci_set_config(QVirtioDevice *d, void *config)
+{
+
+}
+
+uint32_t qvirtio_pci_get_features(QVirtioDevice *d)
+{
+ return 0;
+}
+
+uint8_t qvirtio_pci_get_status(QVirtioDevice *d)
+{
+ return 0;
+}
+
+void qvirtio_pci_set_status(QVirtioDevice *d, uint8_t val)
+{
+
+}
+
+void qvirtio_pci_reset(QVirtioDevice *d)
+{
+
+}
+
+uint8_t qvirtio_pci_query_isr(QVirtioDevice *d)
+{
+ return 0;
+}
+
+void qvirtio_pci_foreach(uint16_t device_type,
+ void (*func)(QVirtioDevice *d, void *data), void *data)
+{
+ QPCIBus *bus;
+ QPCIDevice *dev;
+ int slot;
+ int fn;
+
+ bus = qpci_init_pc();
+
+ for (slot = 0; slot < 32; slot++) {
+ for (fn = 0; fn < 8; fn++) {
+ dev = g_malloc0(sizeof(*dev));
+ dev->bus = bus;
+ dev->devfn = QPCI_DEVFN(slot, fn);
+
+ if (qpci_config_readw(dev, PCI_VENDOR_ID) == 0xFFFF) {
+ g_free(dev);
+ continue;
+ }
+
+ if (device_type != (uint16_t)-1 &&
+ qpci_config_readw(dev, PCI_SUBSYSTEM_ID) != device_type) {
+ continue;
+ }
+
+ func(qpcidevice_to_qvirtiodevice(dev), data);
+ }
+ }
+}
+
+QVirtioDevice *qvirtio_pci_device_find(uint16_t device_type)
+{
+ QVirtioDevice *dev;
+
+ dev = g_malloc0(sizeof(*dev));
+ qvirtio_pci_foreach(device_type, qvirtio_pci_assign_device, dev);
+
+ return dev;
+}
diff --git a/tests/libqos/virtio-pci.h b/tests/libqos/virtio-pci.h
new file mode 100644
index 0000000..d92bcf2
--- /dev/null
+++ b/tests/libqos/virtio-pci.h
@@ -0,0 +1,40 @@
+/*
+ * libqos virtio PCI definitions
+ *
+ * Copyright (c) 2014 Marc Marí
+ *
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
+ * See the COPYING file in the top-level directory.
+ */
+
+#ifndef LIBQOS_VIRTIO_PCI_H
+#define LIBQOS_VIRTIO_PCI_H
+
+#include "libqos/virtio.h"
+
+void qvirtio_pci_notify(QVirtioDevice *d, uint16_t vector);
+void qvirtio_pci_get_config(QVirtioDevice *d, void *config);
+void qvirtio_pci_set_config(QVirtioDevice *d, void *config);
+uint32_t qvirtio_pci_get_features(QVirtioDevice *d);
+uint8_t qvirtio_pci_get_status(QVirtioDevice *d);
+void qvirtio_pci_set_status(QVirtioDevice *d, uint8_t val);
+void qvirtio_pci_reset(QVirtioDevice *d);
+uint8_t qvirtio_pci_query_isr(QVirtioDevice *d);
+void qvirtio_pci_foreach(uint16_t device_id,
+ void (*func)(QVirtioDevice *d, void *data), void *data);
+QVirtioDevice *qvirtio_pci_device_find(uint16_t device_id);
+
+const QVirtioBus qvirtio_pci = {
+ .notify = qvirtio_pci_notify,
+ .get_config = qvirtio_pci_get_config,
+ .set_config = qvirtio_pci_set_config,
+ .get_features = qvirtio_pci_get_features,
+ .get_status = qvirtio_pci_get_status,
+ .set_status = qvirtio_pci_set_status,
+ .reset = qvirtio_pci_reset,
+ .query_isr = qvirtio_pci_query_isr,
+ .bus_foreach = qvirtio_pci_foreach,
+ .device_find = qvirtio_pci_device_find,
+};
+
+#endif
diff --git a/tests/libqos/virtio.h b/tests/libqos/virtio.h
new file mode 100644
index 0000000..838aae4
--- /dev/null
+++ b/tests/libqos/virtio.h
@@ -0,0 +1,167 @@
+/*
+ * libqos virtio definitions
+ *
+ * Copyright (c) 2014 Marc Marí
+ *
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
+ * See the COPYING file in the top-level directory.
+ */
+
+#ifndef LIBQOS_VIRTIO_H
+#define LIBQOS_VIRTIO_H
+
+#define VIRTQUEUE_MAX_SIZE 1024
+#define VIRTIO_VENDOR_ID 0x1AF4
+
+#define VIRTIO_NET_DEVICE_ID 0x1
+#define VIRTIO_BLK_DEVICE_ID 0x2
+
+typedef struct QVirtioDevice QVirtioDevice;
+typedef struct QVirtQueue QVirtQueue;
+typedef struct QVRing QVRing;
+
+typedef struct QVirtioDevice {
+ /* Device id */
+ uint16_t device_id;
+
+ /* Device location (depends on transport) */
+ int location;
+
+ /* Device type */
+ uint16_t device_type;
+
+ /* VQs associated with the device */
+ QVirtQueue *vq;
+} QVirtioDevice;
+
+typedef struct QVirtioBus {
+ /* Send a notification IRQ to the device */
+ void (*notify)(QVirtioDevice *d, uint16_t vector);
+
+ /* Get configuration of the device */
+ void (*get_config)(QVirtioDevice *d, void *config);
+
+ /* Set configuration of the device */
+ void (*set_config)(QVirtioDevice *d, void *config);
+
+ /* Get features of the device */
+ uint32_t (*get_features)(QVirtioDevice *d);
+
+ /* Get status of the device */
+ uint8_t (*get_status)(QVirtioDevice *d);
+
+ /* Set status of the device */
+ void (*set_status)(QVirtioDevice *d, uint8_t val);
+
+ /* Reset the device */
+ void (*reset)(QVirtioDevice *d);
+
+ /* Check pending IRQs */
+ uint8_t (*query_isr)(QVirtioDevice *d);
+
+ /* Loop all devices, applying a function to all, or the one specified */
+ void (*bus_foreach)(uint16_t device_t,
+ void (*func)(QVirtioDevice *d, void *data), void *data);
+
+ /* Find and return a device */
+ QVirtioDevice *(*device_find)(uint16_t device_type);
+} QVirtioBus;
+
+/*QVirtioBus *qvirtio_pci_init();
+QVirtioBus *qvirtio_mmio_init();
+QVirtioBus *qvirtio_ccw_init();*/
+
+/*
+I'm not sure what implementation of VirtQueue is better, QEMU's or Linux's. I
+think QEMU implementation is better, because it will be easier to connect the
+QEMU Virtqueue with the Libaos VirtQueue.
+
+The functions to use the VirtQueue are the ones I think necessary in Libqos,
but
+probably there are some missing and some others that are not necessary.
+*/
+
+typedef struct QVRing {
+ unsigned int num;
+ unsigned int align;
+ uint64_t desc;
+ uint64_t avail;
+ uint64_t used;
+} QVRing;
+
+struct QVirtQueue {
+ QVRing vring;
+ uint64_t pa;
+ uint16_t last_avail_idx;
+ uint16_t signalled_used;
+ bool signalled_used_valid;
+ bool notification;
+ uint16_t queue_index;
+ int inuse;
+ uint16_t vector;
+ void (*handle_output)(QVirtioDevice *d, QVirtQueue *vq);
+ QVirtioDevice *dev;
+};
+
+typedef struct QVirtQueueElement {
+ unsigned int index;
+ unsigned int out_num;
+ unsigned int in_num;
+ uint64_t in_addr[VIRTQUEUE_MAX_SIZE];
+ uint64_t out_addr[VIRTQUEUE_MAX_SIZE];
+ /*struct iovec in_sg[VIRTQUEUE_MAX_SIZE];*/
+ /*struct iovec out_sg[VIRTQUEUE_MAX_SIZE];*/
+} QVirtQueueElement;
+
+typedef struct QVRingDesc {
+ uint64_t addr;
+ uint32_t len;
+ uint16_t flags;
+ uint16_t next;
+} QVRingDesc;
+
+typedef struct QVRingAvail {
+ uint16_t flags;
+ uint16_t idx;
+ uint16_t ring[0];
+} VRingAvail;
+
+typedef struct QVRingUsedElem {
+ uint32_t id;
+ uint32_t len;
+} QVRingUsedElem;
+
+typedef struct QVRingUsed {
+ uint16_t flags;
+ uint16_t idx;
+ QVRingUsedElem ring[0];
+} QVRingUsed;
+
+uint64_t qvring_desc_addr(uint64_t desc_pa, int i);
+uint32_t qvring_desc_len(uint64_t desc_pa, int i);
+uint16_t qvring_desc_flags(uint64_t desc_pa, int i);
+uint16_t qvring_desc_next(uint64_t desc_pa, int i);
+uint16_t qvring_avail_flags(QVirtQueue *vq);
+uint16_t qvring_avail_idx(QVirtQueue *vq);
+uint16_t qvring_avail_ring(QVirtQueue *vq, int i);
+uint16_t qvring_used_event(QVirtQueue *vq);
+void qvring_used_ring_id(QVirtQueue *vq, int i, uint32_t val);
+void qvring_used_ring_len(QVirtQueue *vq, int i, uint32_t val);
+uint16_t qvring_used_idx(QVirtQueue *vq);
+void qvring_used_idx_set(QVirtQueue *vq, uint16_t val);
+void qvring_used_flags_set_bit(QVirtQueue *vq, int mask);
+void qvring_used_flags_unset_bit(QVirtQueue *vq, int mask);
+void qvring_avail_event(QVirtQueue *vq, uint16_t val);
+
+void qvirtqueue_push(QVirtQueue *vq, const QVirtQueueElement *elem,
+ unsigned int len);
+void qvirtqueue_flush(QVirtQueue *vq, unsigned int count);
+void qvirtqueue_fill(QVirtQueue *vq, const QVirtQueueElement *elem,
+ unsigned int len, unsigned int idx);
+void qvirtqueue_pop(QVirtQueue *vq, QVirtQueueElement *elem);
+void qvirtqueue_map_sg(/*struct iovec *sg,*/ uint64_t *addr,
+ size_t num_sg, int is_write);
+void qvirtio_notify(QVirtioDevice *vdev, QVirtQueue *vq);
+int qvirtio_queue_ready(QVirtQueue *vq);
+int qvirtio_queue_empty(QVirtQueue *vq);
+
+#endif
diff --git a/tests/virtio-blk-test.c b/tests/virtio-blk-test.c
index d53f875..4f5b1e0 100644
--- a/tests/virtio-blk-test.c
+++ b/tests/virtio-blk-test.c
@@ -2,6 +2,7 @@
* QTest testcase for VirtIO Block Device
*
* Copyright (c) 2014 SUSE LINUX Products GmbH
+ * Copyright (c) 2014 Marc Marí
*
* This work is licensed under the terms of the GNU GPL, version 2 or later.
* See the COPYING file in the top-level directory.
@@ -9,26 +10,51 @@
#include <glib.h>
#include <string.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <stdio.h>
#include "libqtest.h"
-#include "qemu/osdep.h"
+#include "libqos/virtio.h"
+/*#include "qemu/osdep.h"*/
-/* Tests only initialization so far. TODO: Replace with functional tests */
-static void pci_nop(void)
+#define TEST_IMAGE_SIZE (64 * 1024 * 1024)
+
+static char tmp_path[] = "/tmp/qtest.XXXXXX";
+extern QVirtioBus qvirtio_pci;
+
+static void pci_basic(void)
{
+ QVirtioDevice *dev;
+ dev = qvirtio_pci.device_find(VIRTIO_BLK_DEVICE_ID);
+ fprintf(stderr, "Device: %x %x %x\n",
+ dev->device_id, dev->location, dev->device_type);
}
int main(int argc, char **argv)
{
int ret;
+ int fd;
+ char test_start[100];
g_test_init(&argc, &argv, NULL);
- qtest_add_func("/virtio/blk/pci/nop", pci_nop);
+ qtest_add_func("/virtio/blk/pci/basic", pci_basic);
- qtest_start("-drive id=drv0,if=none,file=/dev/null "
- "-device virtio-blk-pci,drive=drv0");
+ /* Create a temporary raw image */
+ fd = mkstemp(tmp_path);
+ g_assert_cmpint(fd, >=, 0);
+ ret = ftruncate(fd, TEST_IMAGE_SIZE);
+ g_assert_cmpint(ret, ==, 0);
+ close(fd);
+
+ sprintf(test_start, "-drive if=none,id=drive0,file=%s "
+ "-device virtio-blk-pci,drive=drive0", tmp_path);
+ qtest_start(test_start);
ret = g_test_run();
qtest_end();
+ /* Cleanup */
+ unlink(tmp_path);
+
return ret;
}
--
1.7.10.4
- [Qemu-devel] [RFC] Functions bus_foreach and device_find from libqos virtio API,
Marc Marí <=