[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
Re: [Qemu-devel] [PATCH 6/8] virtio-blk: do not use vring in dataplane
From: |
Fam Zheng |
Subject: |
Re: [Qemu-devel] [PATCH 6/8] virtio-blk: do not use vring in dataplane |
Date: |
Tue, 16 Feb 2016 16:15:14 +0800 |
User-agent: |
Mutt/1.5.21 (2010-09-15) |
On Sun, 02/14 18:17, Paolo Bonzini wrote:
> Signed-off-by: Paolo Bonzini <address@hidden>
> ---
> hw/block/dataplane/virtio-blk.c | 112
> +++++-----------------------------------
> hw/block/dataplane/virtio-blk.h | 1 +
> hw/block/virtio-blk.c | 49 +++---------------
> include/hw/virtio/virtio-blk.h | 3 --
> 4 files changed, 19 insertions(+), 146 deletions(-)
>
> diff --git a/hw/block/dataplane/virtio-blk.c b/hw/block/dataplane/virtio-blk.c
> index cc521c1..36f3d2b 100644
> --- a/hw/block/dataplane/virtio-blk.c
> +++ b/hw/block/dataplane/virtio-blk.c
> @@ -18,8 +18,6 @@
> #include "qemu/thread.h"
> #include "qemu/error-report.h"
> #include "hw/virtio/virtio-access.h"
> -#include "hw/virtio/dataplane/vring.h"
> -#include "hw/virtio/dataplane/vring-accessors.h"
> #include "sysemu/block-backend.h"
> #include "hw/virtio/virtio-blk.h"
> #include "virtio-blk.h"
> @@ -35,7 +33,7 @@ struct VirtIOBlockDataPlane {
> VirtIOBlkConf *conf;
>
> VirtIODevice *vdev;
> - Vring vring; /* virtqueue vring */
> + VirtQueue *vq; /* virtqueue vring */
> EventNotifier *guest_notifier; /* irq */
> QEMUBH *bh; /* bh for guest notification */
>
> @@ -48,94 +46,26 @@ struct VirtIOBlockDataPlane {
> */
> IOThread *iothread;
> AioContext *ctx;
> - EventNotifier host_notifier; /* doorbell */
>
> /* Operation blocker on BDS */
> Error *blocker;
> - void (*saved_complete_request)(struct VirtIOBlockReq *req,
> - unsigned char status);
> };
>
> /* Raise an interrupt to signal guest, if necessary */
> -static void notify_guest(VirtIOBlockDataPlane *s)
> +void virtio_blk_data_plane_notify(VirtIOBlockDataPlane *s)
> {
> - if (!vring_should_notify(s->vdev, &s->vring)) {
> - return;
> - }
> -
> - event_notifier_set(s->guest_notifier);
> + qemu_bh_schedule(s->bh);
> }
>
> static void notify_guest_bh(void *opaque)
> {
> VirtIOBlockDataPlane *s = opaque;
>
> - notify_guest(s);
> -}
> -
> -static void complete_request_vring(VirtIOBlockReq *req, unsigned char status)
> -{
> - VirtIOBlockDataPlane *s = req->dev->dataplane;
> - stb_p(&req->in->status, status);
> -
> - vring_push(s->vdev, &req->dev->dataplane->vring, &req->elem,
> req->in_len);
> -
> - /* Suppress notification to guest by BH and its scheduled
> - * flag because requests are completed as a batch after io
> - * plug & unplug is introduced, and the BH can still be
> - * executed in dataplane aio context even after it is
> - * stopped, so needn't worry about notification loss with BH.
> - */
> - qemu_bh_schedule(s->bh);
> -}
> -
> -static void handle_notify(EventNotifier *e)
> -{
> - VirtIOBlockDataPlane *s = container_of(e, VirtIOBlockDataPlane,
> - host_notifier);
> - VirtIOBlock *vblk = VIRTIO_BLK(s->vdev);
> -
> - event_notifier_test_and_clear(&s->host_notifier);
> - blk_io_plug(s->conf->conf.blk);
> - for (;;) {
> - MultiReqBuffer mrb = {};
> -
> - /* Disable guest->host notifies to avoid unnecessary vmexits */
> - vring_disable_notification(s->vdev, &s->vring);
> -
> - for (;;) {
> - VirtIOBlockReq *req = vring_pop(s->vdev, &s->vring,
> - sizeof(VirtIOBlockReq));
> -
> - if (req == NULL) {
> - break; /* no more requests */
> - }
> -
> - virtio_blk_init_request(vblk, req);
> - trace_virtio_blk_data_plane_process_request(s, req->elem.out_num,
> - req->elem.in_num,
> - req->elem.index);
> -
> - virtio_blk_handle_request(req, &mrb);
> - }
> -
> - if (mrb.num_reqs) {
> - virtio_blk_submit_multireq(s->conf->conf.blk, &mrb);
> - }
> -
> - if (likely(!vring_more_avail(s->vdev, &s->vring))) { /* vring
> emptied */
> - /* Re-enable guest->host notifies and stop processing the vring.
> - * But if the guest has snuck in more descriptors, keep
> processing.
> - */
> - vring_enable_notification(s->vdev, &s->vring);
> - if (!vring_more_avail(s->vdev, &s->vring)) {
> - break;
> - }
> - } else { /* fatal error */
> - break;
> - }
> + if (!virtio_should_notify(s->vdev, s->vq)) {
> + return;
> }
> - blk_io_unplug(s->conf->conf.blk);
> +
> + event_notifier_set(s->guest_notifier);
> }
>
> static void data_plane_set_up_op_blockers(VirtIOBlockDataPlane *s)
> @@ -260,7 +190,6 @@ void virtio_blk_data_plane_start(VirtIOBlockDataPlane *s)
> BusState *qbus = BUS(qdev_get_parent_bus(DEVICE(s->vdev)));
> VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
> VirtIOBlock *vblk = VIRTIO_BLK(s->vdev);
> - VirtQueue *vq;
> int r;
>
> if (vblk->dataplane_started || s->starting) {
> @@ -268,11 +197,7 @@ void virtio_blk_data_plane_start(VirtIOBlockDataPlane *s)
> }
>
> s->starting = true;
> -
> - vq = virtio_get_queue(s->vdev, 0);
> - if (!vring_setup(&s->vring, s->vdev, 0)) {
> - goto fail_vring;
> - }
> + s->vq = virtio_get_queue(s->vdev, 0);
>
> /* Set up guest notifier (irq) */
> r = k->set_guest_notifiers(qbus->parent, 1, true);
> @@ -281,7 +206,7 @@ void virtio_blk_data_plane_start(VirtIOBlockDataPlane *s)
> "ensure -enable-kvm is set\n", r);
> goto fail_guest_notifiers;
> }
> - s->guest_notifier = virtio_queue_get_guest_notifier(vq);
> + s->guest_notifier = virtio_queue_get_guest_notifier(s->vq);
>
> /* Set up virtqueue notify */
> r = k->set_host_notifier(qbus->parent, 0, true);
> @@ -289,10 +214,6 @@ void virtio_blk_data_plane_start(VirtIOBlockDataPlane *s)
> fprintf(stderr, "virtio-blk failed to set host notifier (%d)\n", r);
> goto fail_host_notifier;
> }
> - s->host_notifier = *virtio_queue_get_host_notifier(vq);
> -
> - s->saved_complete_request = vblk->complete_request;
> - vblk->complete_request = complete_request_vring;
>
> s->starting = false;
> vblk->dataplane_started = true;
> @@ -301,20 +222,17 @@ void virtio_blk_data_plane_start(VirtIOBlockDataPlane
> *s)
> blk_set_aio_context(s->conf->conf.blk, s->ctx);
>
> /* Kick right away to begin processing requests already in vring */
> - event_notifier_set(virtio_queue_get_host_notifier(vq));
> + event_notifier_set(virtio_queue_get_host_notifier(s->vq));
>
> /* Get this show started by hooking up our callbacks */
> aio_context_acquire(s->ctx);
> - aio_set_event_notifier(s->ctx, &s->host_notifier, true,
> - handle_notify);
> + virtio_queue_aio_set_host_notifier_handler(s->vq, s->ctx, true, true);
> aio_context_release(s->ctx);
> return;
>
> fail_host_notifier:
> k->set_guest_notifiers(qbus->parent, 1, false);
> fail_guest_notifiers:
> - vring_teardown(&s->vring, s->vdev, 0);
> - fail_vring:
> s->disabled = true;
> s->starting = false;
> vblk->dataplane_started = true;
> @@ -338,24 +256,18 @@ void virtio_blk_data_plane_stop(VirtIOBlockDataPlane *s)
> return;
> }
> s->stopping = true;
> - vblk->complete_request = s->saved_complete_request;
> trace_virtio_blk_data_plane_stop(s);
>
> aio_context_acquire(s->ctx);
>
> /* Stop notifications for new requests from guest */
> - aio_set_event_notifier(s->ctx, &s->host_notifier, true, NULL);
> + virtio_queue_aio_set_host_notifier_handler(s->vq, s->ctx, false, false);
>
> /* Drain and switch bs back to the QEMU main loop */
> blk_set_aio_context(s->conf->conf.blk, qemu_get_aio_context());
>
> aio_context_release(s->ctx);
>
> - /* Sync vring state back to virtqueue so that non-dataplane request
> - * processing can continue when we disable the host notifier below.
> - */
> - vring_teardown(&s->vring, s->vdev, 0);
> -
> k->set_host_notifier(qbus->parent, 0, false);
>
> /* Clean up guest notifier (irq) */
> diff --git a/hw/block/dataplane/virtio-blk.h b/hw/block/dataplane/virtio-blk.h
> index c88d40e..0714c11 100644
> --- a/hw/block/dataplane/virtio-blk.h
> +++ b/hw/block/dataplane/virtio-blk.h
> @@ -26,5 +26,6 @@ void virtio_blk_data_plane_destroy(VirtIOBlockDataPlane *s);
> void virtio_blk_data_plane_start(VirtIOBlockDataPlane *s);
> void virtio_blk_data_plane_stop(VirtIOBlockDataPlane *s);
> void virtio_blk_data_plane_drain(VirtIOBlockDataPlane *s);
> +void virtio_blk_data_plane_notify(VirtIOBlockDataPlane *s);
>
> #endif /* HW_DATAPLANE_VIRTIO_BLK_H */
> diff --git a/hw/block/virtio-blk.c b/hw/block/virtio-blk.c
> index e04c8f5..cb710f1 100644
> --- a/hw/block/virtio-blk.c
> +++ b/hw/block/virtio-blk.c
> @@ -21,7 +21,6 @@
> #include "sysemu/blockdev.h"
> #include "hw/virtio/virtio-blk.h"
> #include "dataplane/virtio-blk.h"
> -#include "migration/migration.h"
> #include "block/scsi.h"
> #ifdef __linux__
> # include <scsi/sg.h>
> @@ -45,8 +44,7 @@ void virtio_blk_free_request(VirtIOBlockReq *req)
> }
> }
>
> -static void virtio_blk_complete_request(VirtIOBlockReq *req,
> - unsigned char status)
> +static void virtio_blk_req_complete(VirtIOBlockReq *req, unsigned char
> status)
> {
> VirtIOBlock *s = req->dev;
> VirtIODevice *vdev = VIRTIO_DEVICE(s);
> @@ -55,12 +53,11 @@ static void virtio_blk_complete_request(VirtIOBlockReq
> *req,
>
> stb_p(&req->in->status, status);
> virtqueue_push(s->vq, &req->elem, req->in_len);
> - virtio_notify(vdev, s->vq);
> -}
> -
> -static void virtio_blk_req_complete(VirtIOBlockReq *req, unsigned char
> status)
> -{
> - req->dev->complete_request(req, status);
> + if (s->dataplane) {
> + virtio_blk_data_plane_notify(s->dataplane);
> + } else {
> + virtio_notify(vdev, s->vq);
> + }
> }
>
> static int virtio_blk_handle_rw_error(VirtIOBlockReq *req, int error,
> @@ -852,36 +849,6 @@ static const BlockDevOps virtio_block_ops = {
> .resize_cb = virtio_blk_resize,
> };
>
> -/* Disable dataplane thread during live migration since it does not
> - * update the dirty memory bitmap yet.
> - */
> -static void virtio_blk_migration_state_changed(Notifier *notifier, void
> *data)
> -{
> - VirtIOBlock *s = container_of(notifier, VirtIOBlock,
> - migration_state_notifier);
> - MigrationState *mig = data;
> - Error *err = NULL;
> -
> - if (migration_in_setup(mig)) {
> - if (!s->dataplane) {
> - return;
> - }
> - virtio_blk_data_plane_destroy(s->dataplane);
> - s->dataplane = NULL;
> - } else if (migration_has_finished(mig) ||
> - migration_has_failed(mig)) {
> - if (s->dataplane) {
> - return;
> - }
> - blk_drain_all(); /* complete in-flight non-dataplane requests */
> - virtio_blk_data_plane_create(VIRTIO_DEVICE(s), &s->conf,
> - &s->dataplane, &err);
> - if (err != NULL) {
> - error_report_err(err);
> - }
> - }
> -}
> -
> static void virtio_blk_device_realize(DeviceState *dev, Error **errp)
> {
> VirtIODevice *vdev = VIRTIO_DEVICE(dev);
> @@ -916,15 +883,12 @@ static void virtio_blk_device_realize(DeviceState *dev,
> Error **errp)
> s->sector_mask = (s->conf.conf.logical_block_size / BDRV_SECTOR_SIZE) -
> 1;
>
> s->vq = virtio_add_queue(vdev, 128, virtio_blk_handle_output);
> - s->complete_request = virtio_blk_complete_request;
> virtio_blk_data_plane_create(vdev, conf, &s->dataplane, &err);
> if (err != NULL) {
> error_propagate(errp, err);
> virtio_cleanup(vdev);
> return;
> }
> - s->migration_state_notifier.notify = virtio_blk_migration_state_changed;
> - add_migration_state_change_notifier(&s->migration_state_notifier);
>
> s->change = qemu_add_vm_change_state_handler(virtio_blk_dma_restart_cb,
> s);
> register_savevm(dev, "virtio-blk", virtio_blk_id++, 2,
> @@ -940,7 +904,6 @@ static void virtio_blk_device_unrealize(DeviceState *dev,
> Error **errp)
> VirtIODevice *vdev = VIRTIO_DEVICE(dev);
> VirtIOBlock *s = VIRTIO_BLK(dev);
>
> - remove_migration_state_change_notifier(&s->migration_state_notifier);
> virtio_blk_data_plane_destroy(s->dataplane);
> s->dataplane = NULL;
> qemu_del_vm_change_state_handler(s->change);
> diff --git a/include/hw/virtio/virtio-blk.h b/include/hw/virtio/virtio-blk.h
> index 781969d..ae84d92 100644
> --- a/include/hw/virtio/virtio-blk.h
> +++ b/include/hw/virtio/virtio-blk.h
> @@ -53,9 +53,6 @@ typedef struct VirtIOBlock {
> unsigned short sector_mask;
> bool original_wce;
> VMChangeStateEntry *change;
> - /* Function to push to vq and notify guest */
> - void (*complete_request)(struct VirtIOBlockReq *req, unsigned char
> status);
> - Notifier migration_state_notifier;
> bool dataplane_started;
> struct VirtIOBlockDataPlane *dataplane;
> } VirtIOBlock;
> --
> 1.8.3.1
>
>
>
Reviewed-by: Fam Zheng <address@hidden>
- [Qemu-devel] [PATCH 2/8] vring: make vring_enable_notification return void, (continued)
- [Qemu-devel] [PATCH 2/8] vring: make vring_enable_notification return void, Paolo Bonzini, 2016/02/14
- [Qemu-devel] [PATCH 3/8] virtio: add AioContext-specific function for host notifiers, Paolo Bonzini, 2016/02/14
- [Qemu-devel] [PATCH 4/8] virtio: export vring_notify as virtio_should_notify, Paolo Bonzini, 2016/02/14
- [Qemu-devel] [PATCH 6/8] virtio-blk: do not use vring in dataplane, Paolo Bonzini, 2016/02/14
- Re: [Qemu-devel] [PATCH 6/8] virtio-blk: do not use vring in dataplane,
Fam Zheng <=
- [Qemu-devel] [PATCH 7/8] virtio-scsi: do not use vring in dataplane, Paolo Bonzini, 2016/02/14
- [Qemu-devel] [PATCH 8/8] vring: remove, Paolo Bonzini, 2016/02/14
- Re: [Qemu-devel] [PATCH 0/8] virtio: allow migration with dataplane, Christian Borntraeger, 2016/02/16
- Re: [Qemu-devel] [PATCH 0/8] virtio: allow migration with dataplane, Michael S. Tsirkin, 2016/02/19
- Re: [Qemu-devel] [PATCH 0/8] virtio: allow migration with dataplane, Stefan Hajnoczi, 2016/02/22