vhost: introduce vhost_vring_call
This commit introduces struct vhost_vring_call which replaced raw struct eventfd_ctx *call_ctx in struct vhost_virtqueue. Besides eventfd_ctx, it contains a spin lock and an irq_bypass_producer in its structure. Signed-off-by: Zhu Lingshan <lingshan.zhu@intel.com> Suggested-by: Jason Wang <jasowang@redhat.com> Link: https://lore.kernel.org/r/20200731065533.4144-2-lingshan.zhu@intel.com Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
This commit is contained in:
Родитель
bf11d71a0a
Коммит
265a0ad873
|
@ -63,7 +63,7 @@ static void handle_vq_kick(struct vhost_work *work)
|
||||||
static irqreturn_t vhost_vdpa_virtqueue_cb(void *private)
|
static irqreturn_t vhost_vdpa_virtqueue_cb(void *private)
|
||||||
{
|
{
|
||||||
struct vhost_virtqueue *vq = private;
|
struct vhost_virtqueue *vq = private;
|
||||||
struct eventfd_ctx *call_ctx = vq->call_ctx;
|
struct eventfd_ctx *call_ctx = vq->call_ctx.ctx;
|
||||||
|
|
||||||
if (call_ctx)
|
if (call_ctx)
|
||||||
eventfd_signal(call_ctx, 1);
|
eventfd_signal(call_ctx, 1);
|
||||||
|
@ -343,7 +343,7 @@ static long vhost_vdpa_vring_ioctl(struct vhost_vdpa *v, unsigned int cmd,
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case VHOST_SET_VRING_CALL:
|
case VHOST_SET_VRING_CALL:
|
||||||
if (vq->call_ctx) {
|
if (vq->call_ctx.ctx) {
|
||||||
cb.callback = vhost_vdpa_virtqueue_cb;
|
cb.callback = vhost_vdpa_virtqueue_cb;
|
||||||
cb.private = vq;
|
cb.private = vq;
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -298,6 +298,13 @@ static void vhost_vq_meta_reset(struct vhost_dev *d)
|
||||||
__vhost_vq_meta_reset(d->vqs[i]);
|
__vhost_vq_meta_reset(d->vqs[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void vhost_vring_call_reset(struct vhost_vring_call *call_ctx)
|
||||||
|
{
|
||||||
|
call_ctx->ctx = NULL;
|
||||||
|
memset(&call_ctx->producer, 0x0, sizeof(struct irq_bypass_producer));
|
||||||
|
spin_lock_init(&call_ctx->ctx_lock);
|
||||||
|
}
|
||||||
|
|
||||||
static void vhost_vq_reset(struct vhost_dev *dev,
|
static void vhost_vq_reset(struct vhost_dev *dev,
|
||||||
struct vhost_virtqueue *vq)
|
struct vhost_virtqueue *vq)
|
||||||
{
|
{
|
||||||
|
@ -319,13 +326,13 @@ static void vhost_vq_reset(struct vhost_dev *dev,
|
||||||
vq->log_base = NULL;
|
vq->log_base = NULL;
|
||||||
vq->error_ctx = NULL;
|
vq->error_ctx = NULL;
|
||||||
vq->kick = NULL;
|
vq->kick = NULL;
|
||||||
vq->call_ctx = NULL;
|
|
||||||
vq->log_ctx = NULL;
|
vq->log_ctx = NULL;
|
||||||
vhost_reset_is_le(vq);
|
vhost_reset_is_le(vq);
|
||||||
vhost_disable_cross_endian(vq);
|
vhost_disable_cross_endian(vq);
|
||||||
vq->busyloop_timeout = 0;
|
vq->busyloop_timeout = 0;
|
||||||
vq->umem = NULL;
|
vq->umem = NULL;
|
||||||
vq->iotlb = NULL;
|
vq->iotlb = NULL;
|
||||||
|
vhost_vring_call_reset(&vq->call_ctx);
|
||||||
__vhost_vq_meta_reset(vq);
|
__vhost_vq_meta_reset(vq);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -685,8 +692,8 @@ void vhost_dev_cleanup(struct vhost_dev *dev)
|
||||||
eventfd_ctx_put(dev->vqs[i]->error_ctx);
|
eventfd_ctx_put(dev->vqs[i]->error_ctx);
|
||||||
if (dev->vqs[i]->kick)
|
if (dev->vqs[i]->kick)
|
||||||
fput(dev->vqs[i]->kick);
|
fput(dev->vqs[i]->kick);
|
||||||
if (dev->vqs[i]->call_ctx)
|
if (dev->vqs[i]->call_ctx.ctx)
|
||||||
eventfd_ctx_put(dev->vqs[i]->call_ctx);
|
eventfd_ctx_put(dev->vqs[i]->call_ctx.ctx);
|
||||||
vhost_vq_reset(dev, dev->vqs[i]);
|
vhost_vq_reset(dev, dev->vqs[i]);
|
||||||
}
|
}
|
||||||
vhost_dev_free_iovecs(dev);
|
vhost_dev_free_iovecs(dev);
|
||||||
|
@ -1629,7 +1636,10 @@ long vhost_vring_ioctl(struct vhost_dev *d, unsigned int ioctl, void __user *arg
|
||||||
r = PTR_ERR(ctx);
|
r = PTR_ERR(ctx);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
swap(ctx, vq->call_ctx);
|
|
||||||
|
spin_lock(&vq->call_ctx.ctx_lock);
|
||||||
|
swap(ctx, vq->call_ctx.ctx);
|
||||||
|
spin_unlock(&vq->call_ctx.ctx_lock);
|
||||||
break;
|
break;
|
||||||
case VHOST_SET_VRING_ERR:
|
case VHOST_SET_VRING_ERR:
|
||||||
if (copy_from_user(&f, argp, sizeof f)) {
|
if (copy_from_user(&f, argp, sizeof f)) {
|
||||||
|
@ -2440,8 +2450,8 @@ static bool vhost_notify(struct vhost_dev *dev, struct vhost_virtqueue *vq)
|
||||||
void vhost_signal(struct vhost_dev *dev, struct vhost_virtqueue *vq)
|
void vhost_signal(struct vhost_dev *dev, struct vhost_virtqueue *vq)
|
||||||
{
|
{
|
||||||
/* Signal the Guest tell them we used something up. */
|
/* Signal the Guest tell them we used something up. */
|
||||||
if (vq->call_ctx && vhost_notify(dev, vq))
|
if (vq->call_ctx.ctx && vhost_notify(dev, vq))
|
||||||
eventfd_signal(vq->call_ctx, 1);
|
eventfd_signal(vq->call_ctx.ctx, 1);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(vhost_signal);
|
EXPORT_SYMBOL_GPL(vhost_signal);
|
||||||
|
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
#include <linux/virtio_ring.h>
|
#include <linux/virtio_ring.h>
|
||||||
#include <linux/atomic.h>
|
#include <linux/atomic.h>
|
||||||
#include <linux/vhost_iotlb.h>
|
#include <linux/vhost_iotlb.h>
|
||||||
|
#include <linux/irqbypass.h>
|
||||||
|
|
||||||
struct vhost_work;
|
struct vhost_work;
|
||||||
typedef void (*vhost_work_fn_t)(struct vhost_work *work);
|
typedef void (*vhost_work_fn_t)(struct vhost_work *work);
|
||||||
|
@ -60,6 +61,12 @@ enum vhost_uaddr_type {
|
||||||
VHOST_NUM_ADDRS = 3,
|
VHOST_NUM_ADDRS = 3,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct vhost_vring_call {
|
||||||
|
struct eventfd_ctx *ctx;
|
||||||
|
struct irq_bypass_producer producer;
|
||||||
|
spinlock_t ctx_lock;
|
||||||
|
};
|
||||||
|
|
||||||
/* The virtqueue structure describes a queue attached to a device. */
|
/* The virtqueue structure describes a queue attached to a device. */
|
||||||
struct vhost_virtqueue {
|
struct vhost_virtqueue {
|
||||||
struct vhost_dev *dev;
|
struct vhost_dev *dev;
|
||||||
|
@ -72,7 +79,7 @@ struct vhost_virtqueue {
|
||||||
vring_used_t __user *used;
|
vring_used_t __user *used;
|
||||||
const struct vhost_iotlb_map *meta_iotlb[VHOST_NUM_ADDRS];
|
const struct vhost_iotlb_map *meta_iotlb[VHOST_NUM_ADDRS];
|
||||||
struct file *kick;
|
struct file *kick;
|
||||||
struct eventfd_ctx *call_ctx;
|
struct vhost_vring_call call_ctx;
|
||||||
struct eventfd_ctx *error_ctx;
|
struct eventfd_ctx *error_ctx;
|
||||||
struct eventfd_ctx *log_ctx;
|
struct eventfd_ctx *log_ctx;
|
||||||
|
|
||||||
|
|
Загрузка…
Ссылка в новой задаче