Commit c502eb85 authored by Jiri Pirko's avatar Jiri Pirko Committed by Michael S. Tsirkin

virtio: introduce virtio_queue_info struct and find_vqs_info() config op

Introduce a structure virtio_queue_info to carry name, callback and ctx
together. In order to allow config implementations to accept config op
with array of virtio_queue_info structures, introduce a new
find_vqs_info() op. Do the needed conversion in virtio_find_vqs_ctx().
Note that whole virtio_find_vqs_ctx() is going to be eventually removed
at the and of this patchset.
Suggested-by: default avatarXuan Zhuo <xuanzhuo@linux.alibaba.com>
Signed-off-by: default avatarJiri Pirko <jiri@nvidia.com>
Message-Id: <20240708074814.1739223-5-jiri@resnulli.us>
Signed-off-by: default avatarMichael S. Tsirkin <mst@redhat.com>
parent 959538c1
...@@ -18,6 +18,20 @@ struct virtio_shm_region { ...@@ -18,6 +18,20 @@ struct virtio_shm_region {
typedef void vq_callback_t(struct virtqueue *); typedef void vq_callback_t(struct virtqueue *);
/**
* struct virtqueue_info - Info for a virtqueue passed to find_vqs().
* @name: virtqueue description. Used mainly for debugging, NULL for
* a virtqueue unused by the driver.
* @callback: A callback to invoke on a used buffer notification.
* NULL for a virtqueue that does not need a callback.
* @ctx: A flag to indicate to maintain an extra context per virtqueue.
*/
struct virtqueue_info {
const char *name;
vq_callback_t *callback;
bool ctx;
};
/** /**
* struct virtio_config_ops - operations for configuring a virtio device * struct virtio_config_ops - operations for configuring a virtio device
* Note: Do not assume that a transport implements all of the operations * Note: Do not assume that a transport implements all of the operations
...@@ -58,6 +72,12 @@ typedef void vq_callback_t(struct virtqueue *); ...@@ -58,6 +72,12 @@ typedef void vq_callback_t(struct virtqueue *);
* names: array of virtqueue names (mainly for debugging) * names: array of virtqueue names (mainly for debugging)
* include a NULL entry for vqs unused by driver * include a NULL entry for vqs unused by driver
* Returns 0 on success or error status * Returns 0 on success or error status
* @find_vqs_info: find virtqueues and instantiate them.
* vdev: the virtio_device
* nvqs: the number of virtqueues to find
* vqs: on success, includes new virtqueues
* vqs_info: array of virtqueue info structures
* Returns 0 on success or error status
* @del_vqs: free virtqueues found by find_vqs(). * @del_vqs: free virtqueues found by find_vqs().
* @synchronize_cbs: synchronize with the virtqueue callbacks (optional) * @synchronize_cbs: synchronize with the virtqueue callbacks (optional)
* The function guarantees that all memory operations on the * The function guarantees that all memory operations on the
...@@ -109,6 +129,10 @@ struct virtio_config_ops { ...@@ -109,6 +129,10 @@ struct virtio_config_ops {
struct virtqueue *vqs[], vq_callback_t *callbacks[], struct virtqueue *vqs[], vq_callback_t *callbacks[],
const char * const names[], const bool *ctx, const char * const names[], const bool *ctx,
struct irq_affinity *desc); struct irq_affinity *desc);
int (*find_vqs_info)(struct virtio_device *vdev, unsigned int nvqs,
struct virtqueue *vqs[],
struct virtqueue_info vqs_info[],
struct irq_affinity *desc);
void (*del_vqs)(struct virtio_device *); void (*del_vqs)(struct virtio_device *);
void (*synchronize_cbs)(struct virtio_device *); void (*synchronize_cbs)(struct virtio_device *);
u64 (*get_features)(struct virtio_device *vdev); u64 (*get_features)(struct virtio_device *vdev);
...@@ -210,14 +234,39 @@ static inline bool virtio_has_dma_quirk(const struct virtio_device *vdev) ...@@ -210,14 +234,39 @@ static inline bool virtio_has_dma_quirk(const struct virtio_device *vdev)
return !virtio_has_feature(vdev, VIRTIO_F_ACCESS_PLATFORM); return !virtio_has_feature(vdev, VIRTIO_F_ACCESS_PLATFORM);
} }
static inline
int virtio_find_vqs_info(struct virtio_device *vdev, unsigned int nvqs,
struct virtqueue *vqs[],
struct virtqueue_info vqs_info[],
struct irq_affinity *desc)
{
return vdev->config->find_vqs_info(vdev, nvqs, vqs, vqs_info, desc);
}
static inline static inline
int virtio_find_vqs_ctx(struct virtio_device *vdev, unsigned nvqs, int virtio_find_vqs_ctx(struct virtio_device *vdev, unsigned nvqs,
struct virtqueue *vqs[], vq_callback_t *callbacks[], struct virtqueue *vqs[], vq_callback_t *callbacks[],
const char * const names[], const bool *ctx, const char * const names[], const bool *ctx,
struct irq_affinity *desc) struct irq_affinity *desc)
{ {
return vdev->config->find_vqs(vdev, nvqs, vqs, callbacks, names, ctx, struct virtqueue_info *vqs_info;
desc); int err, i;
if (!vdev->config->find_vqs_info)
return vdev->config->find_vqs(vdev, nvqs, vqs, callbacks,
names, ctx, desc);
vqs_info = kmalloc_array(nvqs, sizeof(*vqs_info), GFP_KERNEL);
if (!vqs_info)
return -ENOMEM;
for (i = 0; i < nvqs; i++) {
vqs_info[i].name = names[i];
vqs_info[i].callback = callbacks[i];
vqs_info[i].ctx = ctx ? ctx[i] : false;
}
err = virtio_find_vqs_info(vdev, nvqs, vqs, vqs_info, desc);
kfree(vqs_info);
return err;
} }
static inline static inline
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment