virtio-scsi: suppress virtqueue kick during processing

The guest does not need to kick the virtqueue while we are processing
it.  This reduces the number of vmexits during periods of heavy I/O.

Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Message-id: 20161201192652.9509-9-stefanha@redhat.com
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
This commit is contained in:
Stefan Hajnoczi 2016-12-01 19:26:47 +00:00
parent 9ef9d40261
commit 23425cc2b7

View File

@ -592,26 +592,32 @@ static void virtio_scsi_handle_cmd_req_submit(VirtIOSCSI *s, VirtIOSCSIReq *req)
void virtio_scsi_handle_cmd_vq(VirtIOSCSI *s, VirtQueue *vq) void virtio_scsi_handle_cmd_vq(VirtIOSCSI *s, VirtQueue *vq)
{ {
VirtIOSCSIReq *req, *next; VirtIOSCSIReq *req, *next;
int ret; int ret = 0;
QTAILQ_HEAD(, VirtIOSCSIReq) reqs = QTAILQ_HEAD_INITIALIZER(reqs); QTAILQ_HEAD(, VirtIOSCSIReq) reqs = QTAILQ_HEAD_INITIALIZER(reqs);
while ((req = virtio_scsi_pop_req(s, vq))) { do {
ret = virtio_scsi_handle_cmd_req_prepare(s, req); virtio_queue_set_notification(vq, 0);
if (!ret) {
QTAILQ_INSERT_TAIL(&reqs, req, next); while ((req = virtio_scsi_pop_req(s, vq))) {
} else if (ret == -EINVAL) { ret = virtio_scsi_handle_cmd_req_prepare(s, req);
/* The device is broken and shouldn't process any request */ if (!ret) {
while (!QTAILQ_EMPTY(&reqs)) { QTAILQ_INSERT_TAIL(&reqs, req, next);
req = QTAILQ_FIRST(&reqs); } else if (ret == -EINVAL) {
QTAILQ_REMOVE(&reqs, req, next); /* The device is broken and shouldn't process any request */
blk_io_unplug(req->sreq->dev->conf.blk); while (!QTAILQ_EMPTY(&reqs)) {
scsi_req_unref(req->sreq); req = QTAILQ_FIRST(&reqs);
virtqueue_detach_element(req->vq, &req->elem, 0); QTAILQ_REMOVE(&reqs, req, next);
virtio_scsi_free_req(req); blk_io_unplug(req->sreq->dev->conf.blk);
scsi_req_unref(req->sreq);
virtqueue_detach_element(req->vq, &req->elem, 0);
virtio_scsi_free_req(req);
}
} }
} }
}
virtio_queue_set_notification(vq, 1);
} while (ret != -EINVAL && !virtio_queue_empty(vq));
QTAILQ_FOREACH_SAFE(req, &reqs, next, next) { QTAILQ_FOREACH_SAFE(req, &reqs, next, next) {
virtio_scsi_handle_cmd_req_submit(s, req); virtio_scsi_handle_cmd_req_submit(s, req);