vdpa/mlx5: Add support for control VQ and MAC setting
Add support to handle control virtqueue configurations per virtio specification. The control virtqueue is implemented in software and no hardware offloading is involved. Control VQ configuration need task context, therefore all configurations are handled in a workqueue created for the purpose. Modifications are made to the memory registration code to allow for saving a copy of itolb to be used by the control VQ to access the vring. The max number of data virtqueus supported by the driver has been updated to 2 since multiqueue is not supported at this stage and we need to ensure consistency of VQ indices mapping to either data or control VQ. Signed-off-by: Eli Cohen <elic@nvidia.com> Link: https://lore.kernel.org/r/20210823052123.14909-6-elic@nvidia.com Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
This commit is contained in:
parent
e4fc66508c
commit
5262912ef3
@ -53,6 +53,7 @@ config MLX5_VDPA
|
|||||||
config MLX5_VDPA_NET
|
config MLX5_VDPA_NET
|
||||||
tristate "vDPA driver for ConnectX devices"
|
tristate "vDPA driver for ConnectX devices"
|
||||||
select MLX5_VDPA
|
select MLX5_VDPA
|
||||||
|
select VHOST_RING
|
||||||
depends on MLX5_CORE
|
depends on MLX5_CORE
|
||||||
help
|
help
|
||||||
VDPA network driver for ConnectX6 and newer. Provides offloading
|
VDPA network driver for ConnectX6 and newer. Provides offloading
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
#define __MLX5_VDPA_H__
|
#define __MLX5_VDPA_H__
|
||||||
|
|
||||||
#include <linux/etherdevice.h>
|
#include <linux/etherdevice.h>
|
||||||
|
#include <linux/vringh.h>
|
||||||
#include <linux/vdpa.h>
|
#include <linux/vdpa.h>
|
||||||
#include <linux/mlx5/driver.h>
|
#include <linux/mlx5/driver.h>
|
||||||
|
|
||||||
@ -47,6 +48,26 @@ struct mlx5_vdpa_resources {
|
|||||||
bool valid;
|
bool valid;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct mlx5_control_vq {
|
||||||
|
struct vhost_iotlb *iotlb;
|
||||||
|
/* spinlock to synchronize iommu table */
|
||||||
|
spinlock_t iommu_lock;
|
||||||
|
struct vringh vring;
|
||||||
|
bool ready;
|
||||||
|
u64 desc_addr;
|
||||||
|
u64 device_addr;
|
||||||
|
u64 driver_addr;
|
||||||
|
struct vdpa_callback event_cb;
|
||||||
|
struct vringh_kiov riov;
|
||||||
|
struct vringh_kiov wiov;
|
||||||
|
unsigned short head;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mlx5_ctrl_wq_ent {
|
||||||
|
struct work_struct work;
|
||||||
|
struct mlx5_vdpa_dev *mvdev;
|
||||||
|
};
|
||||||
|
|
||||||
struct mlx5_vdpa_dev {
|
struct mlx5_vdpa_dev {
|
||||||
struct vdpa_device vdev;
|
struct vdpa_device vdev;
|
||||||
struct mlx5_core_dev *mdev;
|
struct mlx5_core_dev *mdev;
|
||||||
@ -60,6 +81,8 @@ struct mlx5_vdpa_dev {
|
|||||||
u32 generation;
|
u32 generation;
|
||||||
|
|
||||||
struct mlx5_vdpa_mr mr;
|
struct mlx5_vdpa_mr mr;
|
||||||
|
struct mlx5_control_vq cvq;
|
||||||
|
struct workqueue_struct *wq;
|
||||||
};
|
};
|
||||||
|
|
||||||
int mlx5_vdpa_alloc_pd(struct mlx5_vdpa_dev *dev, u32 *pdn, u16 uid);
|
int mlx5_vdpa_alloc_pd(struct mlx5_vdpa_dev *dev, u32 *pdn, u16 uid);
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
|
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
|
||||||
/* Copyright (c) 2020 Mellanox Technologies Ltd. */
|
/* Copyright (c) 2020 Mellanox Technologies Ltd. */
|
||||||
|
|
||||||
|
#include <linux/vhost_types.h>
|
||||||
#include <linux/vdpa.h>
|
#include <linux/vdpa.h>
|
||||||
#include <linux/gcd.h>
|
#include <linux/gcd.h>
|
||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
@ -451,33 +452,30 @@ static void destroy_dma_mr(struct mlx5_vdpa_dev *mvdev, struct mlx5_vdpa_mr *mr)
|
|||||||
mlx5_vdpa_destroy_mkey(mvdev, &mr->mkey);
|
mlx5_vdpa_destroy_mkey(mvdev, &mr->mkey);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _mlx5_vdpa_create_mr(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *iotlb)
|
static int dup_iotlb(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *src)
|
||||||
{
|
{
|
||||||
struct mlx5_vdpa_mr *mr = &mvdev->mr;
|
struct vhost_iotlb_map *map;
|
||||||
|
u64 start = 0, last = ULLONG_MAX;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (mr->initialized)
|
if (!src) {
|
||||||
return 0;
|
err = vhost_iotlb_add_range(mvdev->cvq.iotlb, start, last, start, VHOST_ACCESS_RW);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
if (iotlb)
|
for (map = vhost_iotlb_itree_first(src, start, last); map;
|
||||||
err = create_user_mr(mvdev, iotlb);
|
map = vhost_iotlb_itree_next(map, start, last)) {
|
||||||
else
|
err = vhost_iotlb_add_range(mvdev->cvq.iotlb, map->start, map->last,
|
||||||
err = create_dma_mr(mvdev, mr);
|
map->addr, map->perm);
|
||||||
|
if (err)
|
||||||
if (!err)
|
return err;
|
||||||
mr->initialized = true;
|
}
|
||||||
|
return 0;
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int mlx5_vdpa_create_mr(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *iotlb)
|
static void prune_iotlb(struct mlx5_vdpa_dev *mvdev)
|
||||||
{
|
{
|
||||||
int err;
|
vhost_iotlb_del_range(mvdev->cvq.iotlb, 0, ULLONG_MAX);
|
||||||
|
|
||||||
mutex_lock(&mvdev->mr.mkey_mtx);
|
|
||||||
err = _mlx5_vdpa_create_mr(mvdev, iotlb);
|
|
||||||
mutex_unlock(&mvdev->mr.mkey_mtx);
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void destroy_user_mr(struct mlx5_vdpa_dev *mvdev, struct mlx5_vdpa_mr *mr)
|
static void destroy_user_mr(struct mlx5_vdpa_dev *mvdev, struct mlx5_vdpa_mr *mr)
|
||||||
@ -501,6 +499,7 @@ void mlx5_vdpa_destroy_mr(struct mlx5_vdpa_dev *mvdev)
|
|||||||
if (!mr->initialized)
|
if (!mr->initialized)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
prune_iotlb(mvdev);
|
||||||
if (mr->user_mr)
|
if (mr->user_mr)
|
||||||
destroy_user_mr(mvdev, mr);
|
destroy_user_mr(mvdev, mr);
|
||||||
else
|
else
|
||||||
@ -512,6 +511,48 @@ out:
|
|||||||
mutex_unlock(&mr->mkey_mtx);
|
mutex_unlock(&mr->mkey_mtx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int _mlx5_vdpa_create_mr(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *iotlb)
|
||||||
|
{
|
||||||
|
struct mlx5_vdpa_mr *mr = &mvdev->mr;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (mr->initialized)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (iotlb)
|
||||||
|
err = create_user_mr(mvdev, iotlb);
|
||||||
|
else
|
||||||
|
err = create_dma_mr(mvdev, mr);
|
||||||
|
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
|
err = dup_iotlb(mvdev, iotlb);
|
||||||
|
if (err)
|
||||||
|
goto out_err;
|
||||||
|
|
||||||
|
mr->initialized = true;
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
out_err:
|
||||||
|
if (iotlb)
|
||||||
|
destroy_user_mr(mvdev, mr);
|
||||||
|
else
|
||||||
|
destroy_dma_mr(mvdev, mr);
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mlx5_vdpa_create_mr(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *iotlb)
|
||||||
|
{
|
||||||
|
int err;
|
||||||
|
|
||||||
|
mutex_lock(&mvdev->mr.mkey_mtx);
|
||||||
|
err = _mlx5_vdpa_create_mr(mvdev, iotlb);
|
||||||
|
mutex_unlock(&mvdev->mr.mkey_mtx);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
int mlx5_vdpa_handle_set_map(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *iotlb,
|
int mlx5_vdpa_handle_set_map(struct mlx5_vdpa_dev *mvdev, struct vhost_iotlb *iotlb,
|
||||||
bool *change_map)
|
bool *change_map)
|
||||||
{
|
{
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
|
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
|
||||||
/* Copyright (c) 2020 Mellanox Technologies Ltd. */
|
/* Copyright (c) 2020 Mellanox Technologies Ltd. */
|
||||||
|
|
||||||
|
#include <linux/iova.h>
|
||||||
#include <linux/mlx5/driver.h>
|
#include <linux/mlx5/driver.h>
|
||||||
#include "mlx5_vdpa.h"
|
#include "mlx5_vdpa.h"
|
||||||
|
|
||||||
@ -221,6 +222,22 @@ int mlx5_vdpa_destroy_mkey(struct mlx5_vdpa_dev *mvdev, struct mlx5_core_mkey *m
|
|||||||
return mlx5_cmd_exec_in(mvdev->mdev, destroy_mkey, in);
|
return mlx5_cmd_exec_in(mvdev->mdev, destroy_mkey, in);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int init_ctrl_vq(struct mlx5_vdpa_dev *mvdev)
|
||||||
|
{
|
||||||
|
mvdev->cvq.iotlb = vhost_iotlb_alloc(0, 0);
|
||||||
|
if (!mvdev->cvq.iotlb)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
vringh_set_iotlb(&mvdev->cvq.vring, mvdev->cvq.iotlb, &mvdev->cvq.iommu_lock);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void cleanup_ctrl_vq(struct mlx5_vdpa_dev *mvdev)
|
||||||
|
{
|
||||||
|
vhost_iotlb_free(mvdev->cvq.iotlb);
|
||||||
|
}
|
||||||
|
|
||||||
int mlx5_vdpa_alloc_resources(struct mlx5_vdpa_dev *mvdev)
|
int mlx5_vdpa_alloc_resources(struct mlx5_vdpa_dev *mvdev)
|
||||||
{
|
{
|
||||||
u64 offset = MLX5_CAP64_DEV_VDPA_EMULATION(mvdev->mdev, doorbell_bar_offset);
|
u64 offset = MLX5_CAP64_DEV_VDPA_EMULATION(mvdev->mdev, doorbell_bar_offset);
|
||||||
@ -260,10 +277,17 @@ int mlx5_vdpa_alloc_resources(struct mlx5_vdpa_dev *mvdev)
|
|||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto err_key;
|
goto err_key;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = init_ctrl_vq(mvdev);
|
||||||
|
if (err)
|
||||||
|
goto err_ctrl;
|
||||||
|
|
||||||
res->valid = true;
|
res->valid = true;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
err_ctrl:
|
||||||
|
iounmap(res->kick_addr);
|
||||||
err_key:
|
err_key:
|
||||||
dealloc_pd(mvdev, res->pdn, res->uid);
|
dealloc_pd(mvdev, res->pdn, res->uid);
|
||||||
err_pd:
|
err_pd:
|
||||||
@ -282,6 +306,7 @@ void mlx5_vdpa_free_resources(struct mlx5_vdpa_dev *mvdev)
|
|||||||
if (!res->valid)
|
if (!res->valid)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
cleanup_ctrl_vq(mvdev);
|
||||||
iounmap(res->kick_addr);
|
iounmap(res->kick_addr);
|
||||||
res->kick_addr = NULL;
|
res->kick_addr = NULL;
|
||||||
dealloc_pd(mvdev, res->pdn, res->uid);
|
dealloc_pd(mvdev, res->pdn, res->uid);
|
||||||
|
@ -133,7 +133,7 @@ struct mlx5_vdpa_virtqueue {
|
|||||||
/* We will remove this limitation once mlx5_vdpa_alloc_resources()
|
/* We will remove this limitation once mlx5_vdpa_alloc_resources()
|
||||||
* provides for driver space allocation
|
* provides for driver space allocation
|
||||||
*/
|
*/
|
||||||
#define MLX5_MAX_SUPPORTED_VQS 16
|
#define MLX5_MAX_SUPPORTED_VQS 2
|
||||||
|
|
||||||
static bool is_index_valid(struct mlx5_vdpa_dev *mvdev, u16 idx)
|
static bool is_index_valid(struct mlx5_vdpa_dev *mvdev, u16 idx)
|
||||||
{
|
{
|
||||||
@ -160,6 +160,7 @@ struct mlx5_vdpa_net {
|
|||||||
struct mlx5_flow_handle *rx_rule;
|
struct mlx5_flow_handle *rx_rule;
|
||||||
bool setup;
|
bool setup;
|
||||||
u16 mtu;
|
u16 mtu;
|
||||||
|
u32 cur_num_vqs;
|
||||||
};
|
};
|
||||||
|
|
||||||
static void free_resources(struct mlx5_vdpa_net *ndev);
|
static void free_resources(struct mlx5_vdpa_net *ndev);
|
||||||
@ -169,6 +170,8 @@ static void teardown_driver(struct mlx5_vdpa_net *ndev);
|
|||||||
|
|
||||||
static bool mlx5_vdpa_debug;
|
static bool mlx5_vdpa_debug;
|
||||||
|
|
||||||
|
#define MLX5_CVQ_MAX_ENT 16
|
||||||
|
|
||||||
#define MLX5_LOG_VIO_FLAG(_feature) \
|
#define MLX5_LOG_VIO_FLAG(_feature) \
|
||||||
do { \
|
do { \
|
||||||
if (features & BIT_ULL(_feature)) \
|
if (features & BIT_ULL(_feature)) \
|
||||||
@ -186,6 +189,16 @@ static inline u32 mlx5_vdpa_max_qps(int max_vqs)
|
|||||||
return max_vqs / 2;
|
return max_vqs / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u16 ctrl_vq_idx(struct mlx5_vdpa_dev *mvdev)
|
||||||
|
{
|
||||||
|
return 2 * mlx5_vdpa_max_qps(mvdev->max_vqs);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool is_ctrl_vq_idx(struct mlx5_vdpa_dev *mvdev, u16 idx)
|
||||||
|
{
|
||||||
|
return idx == ctrl_vq_idx(mvdev);
|
||||||
|
}
|
||||||
|
|
||||||
static void print_status(struct mlx5_vdpa_dev *mvdev, u8 status, bool set)
|
static void print_status(struct mlx5_vdpa_dev *mvdev, u8 status, bool set)
|
||||||
{
|
{
|
||||||
if (status & ~VALID_STATUS_MASK)
|
if (status & ~VALID_STATUS_MASK)
|
||||||
@ -1358,15 +1371,132 @@ static void remove_fwd_to_tir(struct mlx5_vdpa_net *ndev)
|
|||||||
ndev->rx_rule = NULL;
|
ndev->rx_rule = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static virtio_net_ctrl_ack handle_ctrl_mac(struct mlx5_vdpa_dev *mvdev, u8 cmd)
|
||||||
|
{
|
||||||
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
|
struct mlx5_control_vq *cvq = &mvdev->cvq;
|
||||||
|
virtio_net_ctrl_ack status = VIRTIO_NET_ERR;
|
||||||
|
struct mlx5_core_dev *pfmdev;
|
||||||
|
size_t read;
|
||||||
|
u8 mac[ETH_ALEN];
|
||||||
|
|
||||||
|
pfmdev = pci_get_drvdata(pci_physfn(mvdev->mdev->pdev));
|
||||||
|
switch (cmd) {
|
||||||
|
case VIRTIO_NET_CTRL_MAC_ADDR_SET:
|
||||||
|
read = vringh_iov_pull_iotlb(&cvq->vring, &cvq->riov, (void *)mac, ETH_ALEN);
|
||||||
|
if (read != ETH_ALEN)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (!memcmp(ndev->config.mac, mac, 6)) {
|
||||||
|
status = VIRTIO_NET_OK;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_zero_ether_addr(ndev->config.mac)) {
|
||||||
|
if (mlx5_mpfs_del_mac(pfmdev, ndev->config.mac)) {
|
||||||
|
mlx5_vdpa_warn(mvdev, "failed to delete old MAC %pM from MPFS table\n",
|
||||||
|
ndev->config.mac);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mlx5_mpfs_add_mac(pfmdev, mac)) {
|
||||||
|
mlx5_vdpa_warn(mvdev, "failed to insert new MAC %pM into MPFS table\n",
|
||||||
|
mac);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(ndev->config.mac, mac, ETH_ALEN);
|
||||||
|
status = VIRTIO_NET_OK;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mlx5_cvq_kick_handler(struct work_struct *work)
|
||||||
|
{
|
||||||
|
virtio_net_ctrl_ack status = VIRTIO_NET_ERR;
|
||||||
|
struct virtio_net_ctrl_hdr ctrl;
|
||||||
|
struct mlx5_ctrl_wq_ent *wqent;
|
||||||
|
struct mlx5_vdpa_dev *mvdev;
|
||||||
|
struct mlx5_control_vq *cvq;
|
||||||
|
struct mlx5_vdpa_net *ndev;
|
||||||
|
size_t read, write;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
wqent = container_of(work, struct mlx5_ctrl_wq_ent, work);
|
||||||
|
mvdev = wqent->mvdev;
|
||||||
|
ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
|
cvq = &mvdev->cvq;
|
||||||
|
if (!(ndev->mvdev.actual_features & BIT_ULL(VIRTIO_NET_F_CTRL_VQ)))
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
if (!cvq->ready)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
err = vringh_getdesc_iotlb(&cvq->vring, &cvq->riov, &cvq->wiov, &cvq->head,
|
||||||
|
GFP_ATOMIC);
|
||||||
|
if (err <= 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
read = vringh_iov_pull_iotlb(&cvq->vring, &cvq->riov, &ctrl, sizeof(ctrl));
|
||||||
|
if (read != sizeof(ctrl))
|
||||||
|
break;
|
||||||
|
|
||||||
|
switch (ctrl.class) {
|
||||||
|
case VIRTIO_NET_CTRL_MAC:
|
||||||
|
status = handle_ctrl_mac(mvdev, ctrl.cmd);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make sure data is written before advancing index */
|
||||||
|
smp_wmb();
|
||||||
|
|
||||||
|
write = vringh_iov_push_iotlb(&cvq->vring, &cvq->wiov, &status, sizeof(status));
|
||||||
|
vringh_complete_iotlb(&cvq->vring, cvq->head, write);
|
||||||
|
vringh_kiov_cleanup(&cvq->riov);
|
||||||
|
vringh_kiov_cleanup(&cvq->wiov);
|
||||||
|
|
||||||
|
if (vringh_need_notify_iotlb(&cvq->vring))
|
||||||
|
vringh_notify(&cvq->vring);
|
||||||
|
}
|
||||||
|
out:
|
||||||
|
kfree(wqent);
|
||||||
|
}
|
||||||
|
|
||||||
static void mlx5_vdpa_kick_vq(struct vdpa_device *vdev, u16 idx)
|
static void mlx5_vdpa_kick_vq(struct vdpa_device *vdev, u16 idx)
|
||||||
{
|
{
|
||||||
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq = &ndev->vqs[idx];
|
struct mlx5_vdpa_virtqueue *mvq;
|
||||||
|
struct mlx5_ctrl_wq_ent *wqent;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (unlikely(is_ctrl_vq_idx(mvdev, idx))) {
|
||||||
|
if (!mvdev->cvq.ready)
|
||||||
|
return;
|
||||||
|
|
||||||
|
wqent = kzalloc(sizeof(*wqent), GFP_ATOMIC);
|
||||||
|
if (!wqent)
|
||||||
|
return;
|
||||||
|
|
||||||
|
wqent->mvdev = mvdev;
|
||||||
|
INIT_WORK(&wqent->work, mlx5_cvq_kick_handler);
|
||||||
|
queue_work(mvdev->wq, &wqent->work);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mvq = &ndev->vqs[idx];
|
||||||
if (unlikely(!mvq->ready))
|
if (unlikely(!mvq->ready))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -1378,11 +1508,19 @@ static int mlx5_vdpa_set_vq_address(struct vdpa_device *vdev, u16 idx, u64 desc_
|
|||||||
{
|
{
|
||||||
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq = &ndev->vqs[idx];
|
struct mlx5_vdpa_virtqueue *mvq;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (is_ctrl_vq_idx(mvdev, idx)) {
|
||||||
|
mvdev->cvq.desc_addr = desc_area;
|
||||||
|
mvdev->cvq.device_addr = device_area;
|
||||||
|
mvdev->cvq.driver_addr = driver_area;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
mvq = &ndev->vqs[idx];
|
||||||
mvq->desc_addr = desc_area;
|
mvq->desc_addr = desc_area;
|
||||||
mvq->device_addr = device_area;
|
mvq->device_addr = device_area;
|
||||||
mvq->driver_addr = driver_area;
|
mvq->driver_addr = driver_area;
|
||||||
@ -1395,7 +1533,7 @@ static void mlx5_vdpa_set_vq_num(struct vdpa_device *vdev, u16 idx, u32 num)
|
|||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq;
|
struct mlx5_vdpa_virtqueue *mvq;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx) || is_ctrl_vq_idx(mvdev, idx))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
mvq = &ndev->vqs[idx];
|
mvq = &ndev->vqs[idx];
|
||||||
@ -1410,15 +1548,42 @@ static void mlx5_vdpa_set_vq_cb(struct vdpa_device *vdev, u16 idx, struct vdpa_c
|
|||||||
ndev->event_cbs[idx] = *cb;
|
ndev->event_cbs[idx] = *cb;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void mlx5_cvq_notify(struct vringh *vring)
|
||||||
|
{
|
||||||
|
struct mlx5_control_vq *cvq = container_of(vring, struct mlx5_control_vq, vring);
|
||||||
|
|
||||||
|
if (!cvq->event_cb.callback)
|
||||||
|
return;
|
||||||
|
|
||||||
|
cvq->event_cb.callback(cvq->event_cb.private);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void set_cvq_ready(struct mlx5_vdpa_dev *mvdev, bool ready)
|
||||||
|
{
|
||||||
|
struct mlx5_control_vq *cvq = &mvdev->cvq;
|
||||||
|
|
||||||
|
cvq->ready = ready;
|
||||||
|
if (!ready)
|
||||||
|
return;
|
||||||
|
|
||||||
|
cvq->vring.notify = mlx5_cvq_notify;
|
||||||
|
}
|
||||||
|
|
||||||
static void mlx5_vdpa_set_vq_ready(struct vdpa_device *vdev, u16 idx, bool ready)
|
static void mlx5_vdpa_set_vq_ready(struct vdpa_device *vdev, u16 idx, bool ready)
|
||||||
{
|
{
|
||||||
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq = &ndev->vqs[idx];
|
struct mlx5_vdpa_virtqueue *mvq;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (is_ctrl_vq_idx(mvdev, idx)) {
|
||||||
|
set_cvq_ready(mvdev, ready);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mvq = &ndev->vqs[idx];
|
||||||
if (!ready)
|
if (!ready)
|
||||||
suspend_vq(ndev, mvq);
|
suspend_vq(ndev, mvq);
|
||||||
|
|
||||||
@ -1429,12 +1594,14 @@ static bool mlx5_vdpa_get_vq_ready(struct vdpa_device *vdev, u16 idx)
|
|||||||
{
|
{
|
||||||
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq = &ndev->vqs[idx];
|
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
return mvq->ready;
|
if (is_ctrl_vq_idx(mvdev, idx))
|
||||||
|
return mvdev->cvq.ready;
|
||||||
|
|
||||||
|
return ndev->vqs[idx].ready;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlx5_vdpa_set_vq_state(struct vdpa_device *vdev, u16 idx,
|
static int mlx5_vdpa_set_vq_state(struct vdpa_device *vdev, u16 idx,
|
||||||
@ -1442,11 +1609,17 @@ static int mlx5_vdpa_set_vq_state(struct vdpa_device *vdev, u16 idx,
|
|||||||
{
|
{
|
||||||
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq = &ndev->vqs[idx];
|
struct mlx5_vdpa_virtqueue *mvq;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (is_ctrl_vq_idx(mvdev, idx)) {
|
||||||
|
mvdev->cvq.vring.last_avail_idx = state->split.avail_index;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
mvq = &ndev->vqs[idx];
|
||||||
if (mvq->fw_state == MLX5_VIRTIO_NET_Q_OBJECT_STATE_RDY) {
|
if (mvq->fw_state == MLX5_VIRTIO_NET_Q_OBJECT_STATE_RDY) {
|
||||||
mlx5_vdpa_warn(mvdev, "can't modify available index\n");
|
mlx5_vdpa_warn(mvdev, "can't modify available index\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@ -1461,13 +1634,19 @@ static int mlx5_vdpa_get_vq_state(struct vdpa_device *vdev, u16 idx, struct vdpa
|
|||||||
{
|
{
|
||||||
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(vdev);
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
struct mlx5_vdpa_virtqueue *mvq = &ndev->vqs[idx];
|
struct mlx5_vdpa_virtqueue *mvq;
|
||||||
struct mlx5_virtq_attr attr;
|
struct mlx5_virtq_attr attr;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (is_ctrl_vq_idx(mvdev, idx)) {
|
||||||
|
state->split.avail_index = mvdev->cvq.vring.last_avail_idx;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
mvq = &ndev->vqs[idx];
|
||||||
/* If the virtq object was destroyed, use the value saved at
|
/* If the virtq object was destroyed, use the value saved at
|
||||||
* the last minute of suspend_vq. This caters for userspace
|
* the last minute of suspend_vq. This caters for userspace
|
||||||
* that cares about emulating the index after vq is stopped.
|
* that cares about emulating the index after vq is stopped.
|
||||||
@ -1524,10 +1703,13 @@ static u64 mlx5_vdpa_get_features(struct vdpa_device *vdev)
|
|||||||
u16 dev_features;
|
u16 dev_features;
|
||||||
|
|
||||||
dev_features = MLX5_CAP_DEV_VDPA_EMULATION(mvdev->mdev, device_features_bits_mask);
|
dev_features = MLX5_CAP_DEV_VDPA_EMULATION(mvdev->mdev, device_features_bits_mask);
|
||||||
ndev->mvdev.mlx_features = mlx_to_vritio_features(dev_features);
|
ndev->mvdev.mlx_features |= mlx_to_vritio_features(dev_features);
|
||||||
if (MLX5_CAP_DEV_VDPA_EMULATION(mvdev->mdev, virtio_version_1_0))
|
if (MLX5_CAP_DEV_VDPA_EMULATION(mvdev->mdev, virtio_version_1_0))
|
||||||
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_F_VERSION_1);
|
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_F_VERSION_1);
|
||||||
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_F_ACCESS_PLATFORM);
|
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_F_ACCESS_PLATFORM);
|
||||||
|
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_NET_F_CTRL_VQ);
|
||||||
|
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_NET_F_CTRL_MAC_ADDR);
|
||||||
|
|
||||||
print_features(mvdev, ndev->mvdev.mlx_features, false);
|
print_features(mvdev, ndev->mvdev.mlx_features, false);
|
||||||
return ndev->mvdev.mlx_features;
|
return ndev->mvdev.mlx_features;
|
||||||
}
|
}
|
||||||
@ -1543,6 +1725,7 @@ static int verify_min_features(struct mlx5_vdpa_dev *mvdev, u64 features)
|
|||||||
static int setup_virtqueues(struct mlx5_vdpa_dev *mvdev)
|
static int setup_virtqueues(struct mlx5_vdpa_dev *mvdev)
|
||||||
{
|
{
|
||||||
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
|
||||||
|
struct mlx5_control_vq *cvq = &mvdev->cvq;
|
||||||
int err;
|
int err;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
@ -1552,6 +1735,16 @@ static int setup_virtqueues(struct mlx5_vdpa_dev *mvdev)
|
|||||||
goto err_vq;
|
goto err_vq;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mvdev->actual_features & BIT_ULL(VIRTIO_NET_F_CTRL_VQ)) {
|
||||||
|
err = vringh_init_iotlb(&cvq->vring, mvdev->actual_features,
|
||||||
|
MLX5_CVQ_MAX_ENT, false,
|
||||||
|
(struct vring_desc *)(uintptr_t)cvq->desc_addr,
|
||||||
|
(struct vring_avail *)(uintptr_t)cvq->driver_addr,
|
||||||
|
(struct vring_used *)(uintptr_t)cvq->device_addr);
|
||||||
|
if (err)
|
||||||
|
goto err_vq;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_vq:
|
err_vq:
|
||||||
@ -1943,7 +2136,7 @@ static struct vdpa_notification_area mlx5_get_vq_notification(struct vdpa_device
|
|||||||
struct mlx5_vdpa_net *ndev;
|
struct mlx5_vdpa_net *ndev;
|
||||||
phys_addr_t addr;
|
phys_addr_t addr;
|
||||||
|
|
||||||
if (!is_index_valid(mvdev, idx))
|
if (!is_index_valid(mvdev, idx) || is_ctrl_vq_idx(mvdev, idx))
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
/* If SF BAR size is smaller than PAGE_SIZE, do not use direct
|
/* If SF BAR size is smaller than PAGE_SIZE, do not use direct
|
||||||
@ -2120,8 +2313,11 @@ static int mlx5_vdpa_dev_add(struct vdpa_mgmt_dev *v_mdev, const char *name)
|
|||||||
err = mlx5_mpfs_add_mac(pfmdev, config->mac);
|
err = mlx5_mpfs_add_mac(pfmdev, config->mac);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_mtu;
|
goto err_mtu;
|
||||||
|
|
||||||
|
ndev->mvdev.mlx_features |= BIT_ULL(VIRTIO_NET_F_MAC);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
config->max_virtqueue_pairs = cpu_to_mlx5vdpa16(mvdev, mlx5_vdpa_max_qps(max_vqs));
|
||||||
mvdev->vdev.dma_dev = &mdev->pdev->dev;
|
mvdev->vdev.dma_dev = &mdev->pdev->dev;
|
||||||
err = mlx5_vdpa_alloc_resources(&ndev->mvdev);
|
err = mlx5_vdpa_alloc_resources(&ndev->mvdev);
|
||||||
if (err)
|
if (err)
|
||||||
@ -2137,8 +2333,15 @@ static int mlx5_vdpa_dev_add(struct vdpa_mgmt_dev *v_mdev, const char *name)
|
|||||||
if (err)
|
if (err)
|
||||||
goto err_mr;
|
goto err_mr;
|
||||||
|
|
||||||
|
mvdev->wq = create_singlethread_workqueue("mlx5_vdpa_ctrl_wq");
|
||||||
|
if (!mvdev->wq) {
|
||||||
|
err = -ENOMEM;
|
||||||
|
goto err_res2;
|
||||||
|
}
|
||||||
|
|
||||||
|
ndev->cur_num_vqs = 2 * mlx5_vdpa_max_qps(max_vqs);
|
||||||
mvdev->vdev.mdev = &mgtdev->mgtdev;
|
mvdev->vdev.mdev = &mgtdev->mgtdev;
|
||||||
err = _vdpa_register_device(&mvdev->vdev, 2 * mlx5_vdpa_max_qps(max_vqs));
|
err = _vdpa_register_device(&mvdev->vdev, ndev->cur_num_vqs + 1);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_reg;
|
goto err_reg;
|
||||||
|
|
||||||
@ -2146,6 +2349,8 @@ static int mlx5_vdpa_dev_add(struct vdpa_mgmt_dev *v_mdev, const char *name)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_reg:
|
err_reg:
|
||||||
|
destroy_workqueue(mvdev->wq);
|
||||||
|
err_res2:
|
||||||
free_resources(ndev);
|
free_resources(ndev);
|
||||||
err_mr:
|
err_mr:
|
||||||
mlx5_vdpa_destroy_mr(mvdev);
|
mlx5_vdpa_destroy_mr(mvdev);
|
||||||
@ -2163,7 +2368,9 @@ err_mtu:
|
|||||||
static void mlx5_vdpa_dev_del(struct vdpa_mgmt_dev *v_mdev, struct vdpa_device *dev)
|
static void mlx5_vdpa_dev_del(struct vdpa_mgmt_dev *v_mdev, struct vdpa_device *dev)
|
||||||
{
|
{
|
||||||
struct mlx5_vdpa_mgmtdev *mgtdev = container_of(v_mdev, struct mlx5_vdpa_mgmtdev, mgtdev);
|
struct mlx5_vdpa_mgmtdev *mgtdev = container_of(v_mdev, struct mlx5_vdpa_mgmtdev, mgtdev);
|
||||||
|
struct mlx5_vdpa_dev *mvdev = to_mvdev(dev);
|
||||||
|
|
||||||
|
destroy_workqueue(mvdev->wq);
|
||||||
_vdpa_unregister_device(dev);
|
_vdpa_unregister_device(dev);
|
||||||
mgtdev->ndev = NULL;
|
mgtdev->ndev = NULL;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user