net/virtio_user: fix wrong sequence of messages

When virtio_user is used with VPP's native vhost user, it cannot
send/receive any packets.

The root cause is that vpp-vhost-user translates the message
VHOST_USER_SET_FEATURES as puting this device into init state,
aka, zero all related structures. However, previous code
puts this message at last in the whole initialization process,
which leads to all previous information are zeroed.

To fix this issue, we rearrange the sequence of those messages.
  - step 0, send VHOST_USER_SET_VRING_CALL so that vhost allocates
    virtqueue structures;
  - step 1, send VHOST_USER_SET_FEATURES to confirm the features;
  - step 2, send VHOST_USER_SET_MEM_TABLE to share mem regions;
  - step 3, send VHOST_USER_SET_VRING_NUM, VHOST_USER_SET_VRING_BASE,
    VHOST_USER_SET_VRING_ADDR, VHOST_USER_SET_VRING_KICK for each
    queue;
  - ...

Fixes: 37a7eb2ae816 ("net/virtio-user: add device emulation layer")

Reported-by: Zhihong Wang <zhihong.wang@intel.com>
Signed-off-by: Jianfeng Tan <jianfeng.tan@intel.com>
Acked-by: Yuanhan Liu <yuanhan.liu@linux.intel.com>
This commit is contained in:
Jianfeng Tan 2016-09-27 19:11:05 +00:00 committed by Yuanhan Liu
parent 33a290899d
commit 57ae79a75b

View File

@ -44,10 +44,36 @@
#include "virtio_user_dev.h" #include "virtio_user_dev.h"
#include "../virtio_ethdev.h" #include "../virtio_ethdev.h"
static int
virtio_user_create_queue(struct virtio_user_dev *dev, uint32_t queue_sel)
{
/* Of all per virtqueue MSGs, make sure VHOST_SET_VRING_CALL come
* firstly because vhost depends on this msg to allocate virtqueue
* pair.
*/
int callfd;
struct vhost_vring_file file;
/* May use invalid flag, but some backend leverages kickfd and callfd as
* criteria to judge if dev is alive. so finally we use real event_fd.
*/
callfd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
if (callfd < 0) {
PMD_DRV_LOG(ERR, "callfd error, %s\n", strerror(errno));
return -1;
}
file.index = queue_sel;
file.fd = callfd;
vhost_user_sock(dev->vhostfd, VHOST_USER_SET_VRING_CALL, &file);
dev->callfds[queue_sel] = callfd;
return 0;
}
static int static int
virtio_user_kick_queue(struct virtio_user_dev *dev, uint32_t queue_sel) virtio_user_kick_queue(struct virtio_user_dev *dev, uint32_t queue_sel)
{ {
int callfd, kickfd; int kickfd;
struct vhost_vring_file file; struct vhost_vring_file file;
struct vhost_vring_state state; struct vhost_vring_state state;
struct vring *vring = &dev->vrings[queue_sel]; struct vring *vring = &dev->vrings[queue_sel];
@ -60,30 +86,6 @@ virtio_user_kick_queue(struct virtio_user_dev *dev, uint32_t queue_sel)
.flags = 0, /* disable log */ .flags = 0, /* disable log */
}; };
/* May use invalid flag, but some backend leverages kickfd and callfd as
* criteria to judge if dev is alive. so finally we use real event_fd.
*/
callfd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
if (callfd < 0) {
PMD_DRV_LOG(ERR, "callfd error, %s\n", strerror(errno));
return -1;
}
kickfd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
if (kickfd < 0) {
close(callfd);
PMD_DRV_LOG(ERR, "kickfd error, %s\n", strerror(errno));
return -1;
}
/* Of all per virtqueue MSGs, make sure VHOST_SET_VRING_CALL come
* firstly because vhost depends on this msg to allocate virtqueue
* pair.
*/
file.index = queue_sel;
file.fd = callfd;
vhost_user_sock(dev->vhostfd, VHOST_USER_SET_VRING_CALL, &file);
dev->callfds[queue_sel] = callfd;
state.index = queue_sel; state.index = queue_sel;
state.num = vring->num; state.num = vring->num;
vhost_user_sock(dev->vhostfd, VHOST_USER_SET_VRING_NUM, &state); vhost_user_sock(dev->vhostfd, VHOST_USER_SET_VRING_NUM, &state);
@ -97,6 +99,12 @@ virtio_user_kick_queue(struct virtio_user_dev *dev, uint32_t queue_sel)
* lastly because vhost depends on this msg to judge if * lastly because vhost depends on this msg to judge if
* virtio is ready. * virtio is ready.
*/ */
kickfd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
if (kickfd < 0) {
PMD_DRV_LOG(ERR, "kickfd error, %s\n", strerror(errno));
return -1;
}
file.index = queue_sel;
file.fd = kickfd; file.fd = kickfd;
vhost_user_sock(dev->vhostfd, VHOST_USER_SET_VRING_KICK, &file); vhost_user_sock(dev->vhostfd, VHOST_USER_SET_VRING_KICK, &file);
dev->kickfds[queue_sel] = kickfd; dev->kickfds[queue_sel] = kickfd;
@ -104,40 +112,43 @@ virtio_user_kick_queue(struct virtio_user_dev *dev, uint32_t queue_sel)
return 0; return 0;
} }
static int
virtio_user_queue_setup(struct virtio_user_dev *dev,
int (*fn)(struct virtio_user_dev *, uint32_t))
{
uint32_t i, queue_sel;
for (i = 0; i < dev->max_queue_pairs; ++i) {
queue_sel = 2 * i + VTNET_SQ_RQ_QUEUE_IDX;
if (fn(dev, queue_sel) < 0) {
PMD_DRV_LOG(INFO, "setup rx vq fails: %u", i);
return -1;
}
}
for (i = 0; i < dev->max_queue_pairs; ++i) {
queue_sel = 2 * i + VTNET_SQ_TQ_QUEUE_IDX;
if (fn(dev, queue_sel) < 0) {
PMD_DRV_LOG(INFO, "setup tx vq fails: %u", i);
return -1;
}
}
return 0;
}
int int
virtio_user_start_device(struct virtio_user_dev *dev) virtio_user_start_device(struct virtio_user_dev *dev)
{ {
uint64_t features; uint64_t features;
uint32_t i, queue_sel;
int ret; int ret;
/* construct memory region inside each implementation */ /* Step 0: tell vhost to create queues */
ret = vhost_user_sock(dev->vhostfd, VHOST_USER_SET_MEM_TABLE, NULL); if (virtio_user_queue_setup(dev, virtio_user_create_queue) < 0)
if (ret < 0)
goto error; goto error;
for (i = 0; i < dev->max_queue_pairs; ++i) { /* Step 1: set features
queue_sel = 2 * i + VTNET_SQ_RQ_QUEUE_IDX; * Make sure VHOST_USER_F_PROTOCOL_FEATURES is added if mq is enabled,
if (virtio_user_kick_queue(dev, queue_sel) < 0) { * and VIRTIO_NET_F_MAC is stripped.
PMD_DRV_LOG(INFO, "kick rx vq fails: %u", i);
goto error;
}
}
for (i = 0; i < dev->max_queue_pairs; ++i) {
queue_sel = 2 * i + VTNET_SQ_TQ_QUEUE_IDX;
if (virtio_user_kick_queue(dev, queue_sel) < 0) {
PMD_DRV_LOG(INFO, "kick tx vq fails: %u", i);
goto error;
}
}
/* we enable the 1st queue pair by default. */
vhost_user_enable_queue_pair(dev->vhostfd, 0, 1);
/* After setup all virtqueues, we need to set_features so that these
* features can be set into each virtqueue in vhost side. And before
* that, make sure VHOST_USER_F_PROTOCOL_FEATURES is added if mq is
* enabled, and VIRTIO_NET_F_MAC is stripped.
*/ */
features = dev->features; features = dev->features;
if (dev->max_queue_pairs > 1) if (dev->max_queue_pairs > 1)
@ -148,6 +159,20 @@ virtio_user_start_device(struct virtio_user_dev *dev)
goto error; goto error;
PMD_DRV_LOG(INFO, "set features: %" PRIx64, features); PMD_DRV_LOG(INFO, "set features: %" PRIx64, features);
/* Step 2: share memory regions */
ret = vhost_user_sock(dev->vhostfd, VHOST_USER_SET_MEM_TABLE, NULL);
if (ret < 0)
goto error;
/* Step 3: kick queues */
if (virtio_user_queue_setup(dev, virtio_user_kick_queue) < 0)
goto error;
/* Step 4: enable queues
* we enable the 1st queue pair by default.
*/
vhost_user_enable_queue_pair(dev->vhostfd, 0, 1);
return 0; return 0;
error: error:
/* TODO: free resource here or caller to check */ /* TODO: free resource here or caller to check */