Acquiring the access lock is not enough to ensure virtqueue's metadata such as vring pointers are valid.
The access status must also be checked. Fixes: be75dc99ea1f ("vhost: support per-virtqueue statistics") Cc: sta...@dpdk.org Signed-off-by: Maxime Coquelin <maxime.coque...@redhat.com> --- lib/vhost/vhost.c | 21 +++++++++++++++++++-- 1 file changed, 19 insertions(+), 2 deletions(-) diff --git a/lib/vhost/vhost.c b/lib/vhost/vhost.c index d8d74623d4..b03e3b391d 100644 --- a/lib/vhost/vhost.c +++ b/lib/vhost/vhost.c @@ -2199,6 +2199,7 @@ rte_vhost_vring_stats_get(int vid, uint16_t queue_id, struct virtio_net *dev = get_device(vid); struct vhost_virtqueue *vq; unsigned int i; + int ret = VHOST_NB_VQ_STATS; if (dev == NULL) return -1; @@ -2215,6 +2216,12 @@ rte_vhost_vring_stats_get(int vid, uint16_t queue_id, vq = dev->virtqueue[queue_id]; rte_rwlock_write_lock(&vq->access_lock); + + if (unlikely(!vq->access_ok)) { + ret = -1; + goto out_unlock; + } + for (i = 0; i < VHOST_NB_VQ_STATS; i++) { /* * No need to the read atomic counters as such, due to the @@ -2224,15 +2231,18 @@ rte_vhost_vring_stats_get(int vid, uint16_t queue_id, *(uint64_t *)(((char *)vq) + vhost_vq_stat_strings[i].offset); stats[i].id = i; } + +out_unlock: rte_rwlock_write_unlock(&vq->access_lock); - return VHOST_NB_VQ_STATS; + return ret; } int rte_vhost_vring_stats_reset(int vid, uint16_t queue_id) { struct virtio_net *dev = get_device(vid); struct vhost_virtqueue *vq; + int ret = 0; if (dev == NULL) return -1; @@ -2246,14 +2256,21 @@ int rte_vhost_vring_stats_reset(int vid, uint16_t queue_id) vq = dev->virtqueue[queue_id]; rte_rwlock_write_lock(&vq->access_lock); + + if (unlikely(!vq->access_ok)) { + ret = -1; + goto out_unlock; + } /* * No need to the reset atomic counters as such, due to the * above write access_lock preventing them to be updated. */ memset(&vq->stats, 0, sizeof(vq->stats)); + +out_unlock: rte_rwlock_write_unlock(&vq->access_lock); - return 0; + return ret; } int -- 2.41.0