]> git.karo-electronics.de Git - karo-tx-linux.git/commitdiff
kvm tools: Use threadpool for virtio-blk
authorSasha Levin <levinsasha928@gmail.com>
Thu, 28 Apr 2011 13:40:43 +0000 (16:40 +0300)
committerPekka Enberg <penberg@kernel.org>
Thu, 28 Apr 2011 17:52:53 +0000 (20:52 +0300)
virtio-blk has been converted to use the threadpool. All the threading code has
been removed, which left only simple callback handling code.

New threadpool job types are created within VIRTIO_PCI_QUEUE_PFN for every
queue (just one in the case of virtio-blk).  The module signals for work after
receiving VIRTIO_PCI_QUEUE_NOTIFY and expects the threadpool to call
virtio_blk_do_io to handle the I/O.  It is possible that the module will signal
work several times while virtio_blk_do_io is already working, but there is no
need to handle multithreading there since the threadpool will call each job in
linear and not in parallel.

Signed-off-by: Sasha Levin <levinsasha928@gmail.com>
Signed-off-by: Pekka Enberg <penberg@kernel.org>
tools/kvm/virtio-blk.c

index 3516b1c9712e9192f4f130b18c39b00cd965b02f..3feabd060e157e996ca2eec25a519cebf1cc88d2 100644 (file)
@@ -9,6 +9,7 @@
 #include "kvm/util.h"
 #include "kvm/kvm.h"
 #include "kvm/pci.h"
+#include "kvm/threadpool.h"
 
 #include <linux/virtio_ring.h>
 #include <linux/virtio_blk.h>
@@ -31,15 +32,13 @@ struct blk_device {
        uint32_t                        guest_features;
        uint16_t                        config_vector;
        uint8_t                         status;
-       pthread_t                       io_thread;
-       pthread_mutex_t         io_mutex;
-       pthread_cond_t          io_cond;
 
        /* virtio queue */
        uint16_t                        queue_selector;
-       uint64_t                        virtio_blk_queue_set_flags;
 
        struct virt_queue               vqs[NUM_VIRT_QUEUES];
+
+       void                    *jobs[NUM_VIRT_QUEUES];
 };
 
 #define DISK_SEG_MAX   126
@@ -57,9 +56,6 @@ static struct blk_device blk_device = {
         * same applies to VIRTIO_BLK_F_BLK_SIZE
         */
        .host_features          = (1UL << VIRTIO_BLK_F_SEG_MAX),
-
-       .io_mutex                       = PTHREAD_MUTEX_INITIALIZER,
-       .io_cond                        = PTHREAD_COND_INITIALIZER
 };
 
 static bool virtio_blk_pci_io_device_specific_in(void *data, unsigned long offset, int size, uint32_t count)
@@ -156,73 +152,14 @@ static bool virtio_blk_do_io_request(struct kvm *self, struct virt_queue *queue)
        return true;
 }
 
-
-
-static int virtio_blk_get_selected_queue(struct blk_device *dev)
-{
-       int i;
-
-       for (i = 0 ; i < NUM_VIRT_QUEUES ; i++) {
-               if (dev->virtio_blk_queue_set_flags & (1 << i)) {
-                       dev->virtio_blk_queue_set_flags &= ~(1 << i);
-                       return i;
-               }
-       }
-
-       return -1;
-}
-
-static void virtio_blk_do_io(struct kvm *kvm, struct blk_device *dev)
+static void virtio_blk_do_io(struct kvm *kvm, void *param)
 {
-       for (;;) {
-               struct virt_queue *vq;
-               int queue_index;
-
-               mutex_lock(&dev->io_mutex);
-               queue_index = virtio_blk_get_selected_queue(dev);
-               mutex_unlock(&dev->io_mutex);
-
-               if (queue_index < 0)
-                       break;
+       struct virt_queue *vq = param;
 
-               vq = &dev->vqs[queue_index];
+       while (virt_queue__available(vq))
+               virtio_blk_do_io_request(kvm, vq);
 
-               while (virt_queue__available(vq))
-                       virtio_blk_do_io_request(kvm, vq);
-
-               kvm__irq_line(kvm, VIRTIO_BLK_IRQ, 1);
-       }
-}
-
-static void *virtio_blk_io_thread(void *ptr)
-{
-       struct kvm *self = ptr;
-
-       for (;;) {
-               int ret;
-
-               mutex_lock(&blk_device.io_mutex);
-               ret = pthread_cond_wait(&blk_device.io_cond, &blk_device.io_mutex);
-               mutex_unlock(&blk_device.io_mutex);
-
-               if (ret != 0)
-                       break;
-
-               virtio_blk_do_io(self, &blk_device);
-       }
-
-       return NULL;
-}
-
-static void virtio_blk_handle_callback(struct blk_device *dev, uint16_t queue_index)
-{
-       mutex_lock(&dev->io_mutex);
-
-       dev->virtio_blk_queue_set_flags |= (1 << queue_index);
-
-       mutex_unlock(&dev->io_mutex);
-
-       pthread_cond_signal(&dev->io_cond);
+       kvm__irq_line(kvm, VIRTIO_BLK_IRQ, 1);
 }
 
 static bool virtio_blk_pci_io_out(struct kvm *self, uint16_t port, void *data, int size, uint32_t count)
@@ -250,6 +187,9 @@ static bool virtio_blk_pci_io_out(struct kvm *self, uint16_t port, void *data, i
 
                vring_init(&queue->vring, VIRTIO_BLK_QUEUE_SIZE, p, 4096);
 
+               blk_device.jobs[blk_device.queue_selector] =
+                       thread_pool__add_jobtype(self, virtio_blk_do_io, queue);
+
                break;
        }
        case VIRTIO_PCI_QUEUE_SEL:
@@ -258,7 +198,7 @@ static bool virtio_blk_pci_io_out(struct kvm *self, uint16_t port, void *data, i
        case VIRTIO_PCI_QUEUE_NOTIFY: {
                uint16_t queue_index;
                queue_index             = ioport__read16(data);
-               virtio_blk_handle_callback(&blk_device, queue_index);
+               thread_pool__signal_work(blk_device.jobs[queue_index]);
                break;
        }
        case VIRTIO_PCI_STATUS:
@@ -308,8 +248,6 @@ void virtio_blk__init(struct kvm *self)
        if (!self->disk_image)
                return;
 
-       pthread_create(&blk_device.io_thread, NULL, virtio_blk_io_thread, self);
-
        blk_device.blk_config.capacity = self->disk_image->size / SECTOR_SIZE;
 
        pci__register(&virtio_blk_pci_device, PCI_VIRTIO_BLK_DEVNUM);