Signed-off-by: Sasha Levin <levinsasha928@xxxxxxxxx> --- tools/kvm/virtio/blk.c | 263 ++++++++++++++++-------------------------------- 1 files changed, 88 insertions(+), 175 deletions(-) diff --git a/tools/kvm/virtio/blk.c b/tools/kvm/virtio/blk.c index 2ccf585..2e047d7 100644 --- a/tools/kvm/virtio/blk.c +++ b/tools/kvm/virtio/blk.c @@ -1,10 +1,8 @@ #include "kvm/virtio-blk.h" #include "kvm/virtio-pci-dev.h" -#include "kvm/irq.h" #include "kvm/disk-image.h" #include "kvm/virtio.h" -#include "kvm/ioport.h" #include "kvm/mutex.h" #include "kvm/util.h" #include "kvm/kvm.h" @@ -12,10 +10,12 @@ #include "kvm/threadpool.h" #include "kvm/ioeventfd.h" #include "kvm/guest_compat.h" +#include "kvm/virtio-pci.h" #include <linux/virtio_ring.h> #include <linux/virtio_blk.h> +#include <linux/kernel.h> #include <linux/list.h> #include <linux/types.h> #include <pthread.h> @@ -41,88 +41,19 @@ struct blk_dev { pthread_mutex_t mutex; struct list_head list; + struct virtio_pci vpci; struct virtio_blk_config blk_config; struct disk_image *disk; - u64 base_addr; - u32 host_features; - u32 guest_features; - u16 config_vector; - u8 status; - u8 isr; int compat_id; - - /* virtio queue */ - u16 queue_selector; + u32 features; struct virt_queue vqs[NUM_VIRT_QUEUES]; struct blk_dev_job jobs[VIRTIO_BLK_QUEUE_SIZE]; u16 job_idx; - struct pci_device_header pci_hdr; }; static LIST_HEAD(bdevs); -static bool virtio_blk_dev_in(struct blk_dev *bdev, void *data, unsigned long offset, int size) -{ - u8 *config_space = (u8 *) &bdev->blk_config; - - if (size != 1) - return false; - - ioport__write8(data, config_space[offset - VIRTIO_MSI_CONFIG_VECTOR]); - - return true; -} - -static bool virtio_blk_pci_io_in(struct ioport *ioport, struct kvm *kvm, u16 port, void *data, int size) -{ - struct blk_dev *bdev; - u16 offset; - bool ret = true; - - bdev = ioport->priv; - offset = port - bdev->base_addr; - - mutex_lock(&bdev->mutex); - - switch (offset) { - case VIRTIO_PCI_HOST_FEATURES: - ioport__write32(data, bdev->host_features); - break; - case VIRTIO_PCI_GUEST_FEATURES: - ret = false; - break; - case VIRTIO_PCI_QUEUE_PFN: - ioport__write32(data, bdev->vqs[bdev->queue_selector].pfn); - break; - case VIRTIO_PCI_QUEUE_NUM: - ioport__write16(data, VIRTIO_BLK_QUEUE_SIZE); - break; - case VIRTIO_PCI_QUEUE_SEL: - case VIRTIO_PCI_QUEUE_NOTIFY: - ret = false; - break; - case VIRTIO_PCI_STATUS: - ioport__write8(data, bdev->status); - break; - case VIRTIO_PCI_ISR: - ioport__write8(data, bdev->isr); - kvm__irq_line(kvm, bdev->pci_hdr.irq_line, VIRTIO_IRQ_LOW); - bdev->isr = VIRTIO_IRQ_LOW; - break; - case VIRTIO_MSI_CONFIG_VECTOR: - ioport__write16(data, bdev->config_vector); - break; - default: - ret = virtio_blk_dev_in(bdev, data, offset, size); - break; - }; - - mutex_unlock(&bdev->mutex); - - return ret; -} - static void virtio_blk_do_io_request(struct kvm *kvm, void *param) { struct virtio_blk_outhdr *req; @@ -172,7 +103,7 @@ static void virtio_blk_do_io_request(struct kvm *kvm, void *param) virt_queue__set_used_elem(queue, head, block_cnt); mutex_unlock(&bdev->mutex); - virt_queue__trigger_irq(queue, bdev->pci_hdr.irq_line, &bdev->isr, kvm); + virtio_pci__signal_vq(kvm, &bdev->vpci, queue - bdev->vqs); } static void virtio_blk_do_io(struct kvm *kvm, struct virt_queue *vq, struct blk_dev *bdev) @@ -191,82 +122,93 @@ static void virtio_blk_do_io(struct kvm *kvm, struct virt_queue *vq, struct blk_ } } -static bool virtio_blk_pci_io_out(struct ioport *ioport, struct kvm *kvm, u16 port, void *data, int size) +static void ioevent_callback(struct kvm *kvm, void *param) { - struct blk_dev *bdev; - u16 offset; - bool ret = true; + struct blk_dev *bdev = param; - bdev = ioport->priv; - offset = port - bdev->base_addr; + virtio_blk_do_io(kvm, &bdev->vqs[0], bdev); +} - mutex_lock(&bdev->mutex); +static void set_config(struct kvm *kvm, void *dev, u8 data, u32 offset) +{ + struct blk_dev *bdev = dev; - switch (offset) { - case VIRTIO_PCI_GUEST_FEATURES: - bdev->guest_features = ioport__read32(data); - break; - case VIRTIO_PCI_QUEUE_PFN: { - struct virt_queue *queue; - void *p; + ((u8 *)(&bdev->blk_config))[offset] = data; +} - compat__remove_message(bdev->compat_id); +static u8 get_config(struct kvm *kvm, void *dev, u32 offset) +{ + struct blk_dev *bdev = dev; - queue = &bdev->vqs[bdev->queue_selector]; - queue->pfn = ioport__read32(data); - p = guest_pfn_to_host(kvm, queue->pfn); + return ((u8 *)(&bdev->blk_config))[offset]; +} - vring_init(&queue->vring, VIRTIO_BLK_QUEUE_SIZE, p, VIRTIO_PCI_VRING_ALIGN); +static u32 get_host_features(struct kvm *kvm, void *dev) +{ + return 1UL << VIRTIO_BLK_F_SEG_MAX | 1UL << VIRTIO_BLK_F_FLUSH; +} - break; - } - case VIRTIO_PCI_QUEUE_SEL: - bdev->queue_selector = ioport__read16(data); - break; - case VIRTIO_PCI_QUEUE_NOTIFY: { - u16 queue_index; +static void set_guest_features(struct kvm *kvm, void *dev, u32 features) +{ + struct blk_dev *bdev = dev; - queue_index = ioport__read16(data); - virtio_blk_do_io(kvm, &bdev->vqs[queue_index], bdev); + bdev->features = features; +} - break; - } - case VIRTIO_PCI_STATUS: - bdev->status = ioport__read8(data); - break; - case VIRTIO_MSI_CONFIG_VECTOR: - bdev->config_vector = VIRTIO_MSI_NO_VECTOR; - break; - case VIRTIO_MSI_QUEUE_VECTOR: - break; - default: - ret = false; - break; +static int init_vq(struct kvm *kvm, void *dev, u32 vq, u32 pfn) +{ + struct blk_dev *bdev = dev; + struct virt_queue *queue; + void *p; + struct ioevent ioevent; + + compat__remove_message(bdev->compat_id); + + queue = &bdev->vqs[vq]; + queue->pfn = pfn; + p = guest_pfn_to_host(kvm, queue->pfn); + + vring_init(&queue->vring, VIRTIO_BLK_QUEUE_SIZE, p, VIRTIO_PCI_VRING_ALIGN); + + ioevent = (struct ioevent) { + .io_addr = bdev->vpci.base_addr + VIRTIO_PCI_QUEUE_NOTIFY, + .io_len = sizeof(u16), + .fn = ioevent_callback, + .fn_ptr = bdev, + .datamatch = vq, + .fn_kvm = kvm, + .fd = eventfd(0, 0), }; - mutex_unlock(&bdev->mutex); + ioeventfd__add_event(&ioevent); - return ret; + return 0; } -static struct ioport_operations virtio_blk_io_ops = { - .io_in = virtio_blk_pci_io_in, - .io_out = virtio_blk_pci_io_out, -}; +static int notify_vq(struct kvm *kvm, void *dev, u32 vq) +{ + struct blk_dev *bdev = dev; -static void ioevent_callback(struct kvm *kvm, void *param) + virtio_blk_do_io(kvm, &bdev->vqs[vq], bdev); + + return 0; +} + +static int get_pfn_vq(struct kvm *kvm, void *dev, u32 vq) { - struct blk_dev *bdev = param; + struct blk_dev *bdev = dev; - virtio_blk_do_io(kvm, &bdev->vqs[0], bdev); + return bdev->vqs[vq].pfn; +} + +static int get_size_vq(struct kvm *kvm, void *dev, u32 vq) +{ + return VIRTIO_BLK_QUEUE_SIZE; } void virtio_blk__init(struct kvm *kvm, struct disk_image *disk) { - u16 blk_dev_base_addr; - u8 dev, pin, line, i; struct blk_dev *bdev; - struct ioevent ioevent; if (!disk) return; @@ -275,57 +217,28 @@ void virtio_blk__init(struct kvm *kvm, struct disk_image *disk) if (bdev == NULL) die("Failed allocating bdev"); - blk_dev_base_addr = ioport__register(IOPORT_EMPTY, &virtio_blk_io_ops, IOPORT_SIZE, bdev); - - *bdev = (struct blk_dev) { - .mutex = PTHREAD_MUTEX_INITIALIZER, - .disk = disk, - .base_addr = blk_dev_base_addr, - .blk_config = (struct virtio_blk_config) { - .capacity = disk->size / SECTOR_SIZE, - .seg_max = DISK_SEG_MAX, - }, - .pci_hdr = (struct pci_device_header) { - .vendor_id = PCI_VENDOR_ID_REDHAT_QUMRANET, - .device_id = PCI_DEVICE_ID_VIRTIO_BLK, - .header_type = PCI_HEADER_TYPE_NORMAL, - .revision_id = 0, - .class = 0x010000, - .subsys_vendor_id = PCI_SUBSYSTEM_VENDOR_ID_REDHAT_QUMRANET, - .subsys_id = VIRTIO_ID_BLOCK, - .bar[0] = blk_dev_base_addr | PCI_BASE_ADDRESS_SPACE_IO, + *bdev = (struct blk_dev) { + .mutex = PTHREAD_MUTEX_INITIALIZER, + .disk = disk, + .blk_config = (struct virtio_blk_config) { + .capacity = disk->size / SECTOR_SIZE, + .seg_max = DISK_SEG_MAX, }, - /* - * Note we don't set VIRTIO_BLK_F_GEOMETRY here so the - * guest kernel will compute disk geometry by own, the - * same applies to VIRTIO_BLK_F_BLK_SIZE - */ - .host_features = (1UL << VIRTIO_BLK_F_SEG_MAX | 1UL << VIRTIO_BLK_F_FLUSH), }; - list_add_tail(&bdev->list, &bdevs); - - if (irq__register_device(VIRTIO_ID_BLOCK, &dev, &pin, &line) < 0) - return; - - bdev->pci_hdr.irq_pin = pin; - bdev->pci_hdr.irq_line = line; - - pci__register(&bdev->pci_hdr, dev); - - for (i = 0; i < NUM_VIRT_QUEUES; i++) { - ioevent = (struct ioevent) { - .io_addr = blk_dev_base_addr + VIRTIO_PCI_QUEUE_NOTIFY, - .io_len = sizeof(u16), - .fn = ioevent_callback, - .datamatch = i, - .fn_ptr = bdev, - .fn_kvm = kvm, - .fd = eventfd(0, 0), - }; + virtio_pci__init(kvm, &bdev->vpci, bdev, PCI_DEVICE_ID_VIRTIO_BLK, VIRTIO_ID_BLOCK); + bdev->vpci.ops = (struct virtio_pci_ops) { + .set_config = set_config, + .get_config = get_config, + .get_host_features = get_host_features, + .set_guest_features = set_guest_features, + .init_vq = init_vq, + .notify_vq = notify_vq, + .get_pfn_vq = get_pfn_vq, + .get_size_vq = get_size_vq, + }; - ioeventfd__add_event(&ioevent); - } + list_add_tail(&bdev->list, &bdevs); bdev->compat_id = compat__add_message("virtio-blk device was not detected", "While you have requested a virtio-blk device, " @@ -348,7 +261,7 @@ void virtio_blk__delete_all(struct kvm *kvm) struct blk_dev *bdev; bdev = list_first_entry(&bdevs, struct blk_dev, list); - ioeventfd__del_event(bdev->base_addr + VIRTIO_PCI_QUEUE_NOTIFY, 0); + ioeventfd__del_event(bdev->vpci.base_addr + VIRTIO_PCI_QUEUE_NOTIFY, 0); list_del(&bdev->list); free(bdev); } -- 1.7.6 -- To unsubscribe from this list: send the line "unsubscribe kvm" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html