Adding message API support for add/remove memory regions
for vhost-user protocol. A virtio front-end can connect to
vhost-user back-end and add/remove memory regions that the
latter maps into it's address space. Below are the changes
made:
* add/remove memory region function definitions
* corresponding definitions for vhost user message macros
* Tested with SPDK, libblkio and qemu

Signed-off-By: Pravin M Bathija <[email protected]>
---
 lib/vhost/vhost_user.c | 220 ++++++++++++++++++++++++++++++++++++++++-
 1 file changed, 218 insertions(+), 2 deletions(-)

diff --git a/lib/vhost/vhost_user.c b/lib/vhost/vhost_user.c
index 4bfb13fb98..ac9d24d067 100644
--- a/lib/vhost/vhost_user.c
+++ b/lib/vhost/vhost_user.c
@@ -71,6 +71,9 @@ VHOST_MESSAGE_HANDLER(VHOST_USER_SET_FEATURES, 
vhost_user_set_features, false, t
 VHOST_MESSAGE_HANDLER(VHOST_USER_SET_OWNER, vhost_user_set_owner, false, true) 
\
 VHOST_MESSAGE_HANDLER(VHOST_USER_RESET_OWNER, vhost_user_reset_owner, false, 
false) \
 VHOST_MESSAGE_HANDLER(VHOST_USER_SET_MEM_TABLE, vhost_user_set_mem_table, 
true, true) \
+VHOST_MESSAGE_HANDLER(VHOST_USER_GET_MAX_MEM_SLOTS, 
vhost_user_get_max_mem_slots, false, false) \
+VHOST_MESSAGE_HANDLER(VHOST_USER_ADD_MEM_REG, vhost_user_add_mem_reg, true, 
true) \
+VHOST_MESSAGE_HANDLER(VHOST_USER_REM_MEM_REG, vhost_user_rem_mem_reg, false, 
true) \
 VHOST_MESSAGE_HANDLER(VHOST_USER_SET_LOG_BASE, vhost_user_set_log_base, true, 
true) \
 VHOST_MESSAGE_HANDLER(VHOST_USER_SET_LOG_FD, vhost_user_set_log_fd, true, 
true) \
 VHOST_MESSAGE_HANDLER(VHOST_USER_SET_VRING_NUM, vhost_user_set_vring_num, 
false, true) \
@@ -1531,11 +1534,182 @@ vhost_user_set_mem_table(struct virtio_net **pdev,
        return RTE_VHOST_MSG_RESULT_OK;
 
 free_mem_table:
-       free_mem_region(dev);
+       free_all_mem_regions(dev);
        rte_free(dev->mem);
        dev->mem = NULL;
+       rte_free(dev->guest_pages);
+       dev->guest_pages = NULL;
+close_msg_fds:
+       close_msg_fds(ctx);
+       return RTE_VHOST_MSG_RESULT_ERR;
+}
+
+
+static int
+vhost_user_get_max_mem_slots(struct virtio_net **pdev __rte_unused,
+                       struct vhu_msg_context *ctx,
+                       int main_fd __rte_unused)
+{
+       uint32_t max_mem_slots = VHOST_MEMORY_MAX_NREGIONS;
+
+       ctx->msg.payload.u64 = (uint64_t)max_mem_slots;
+       ctx->msg.size = sizeof(ctx->msg.payload.u64);
+       ctx->fd_num = 0;
+
+       return RTE_VHOST_MSG_RESULT_REPLY;
+}
+
+static int
+vhost_user_add_mem_reg(struct virtio_net **pdev,
+                       struct vhu_msg_context *ctx,
+                       int main_fd __rte_unused)
+{
+       struct virtio_net *dev = *pdev;
+       struct VhostUserMemoryRegion *region = 
&ctx->msg.payload.memory_single.region;
+       uint32_t i;
+
+       /* make sure new region will fit */
+       if (dev->mem != NULL && dev->mem->nregions >= 
VHOST_MEMORY_MAX_NREGIONS) {
+               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                       "too many memory regions already (%u)",
+                       dev->mem->nregions);
+               goto close_msg_fds;
+       }
+
+       /* make sure supplied memory fd present */
+       if (ctx->fd_num != 1) {
+               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                       "fd count makes no sense (%u)",
+                       ctx->fd_num);
+               goto close_msg_fds;
+       }
+
+       /* Make sure no overlap in guest virtual address space */
+       if (dev->mem != NULL && dev->mem->nregions > 0) {
+               for (uint32_t i = 0; i < VHOST_MEMORY_MAX_NREGIONS; i++) {
+                       struct rte_vhost_mem_region *current_region = 
&dev->mem->regions[i];
+
+                       if (current_region->mmap_size == 0)
+                               continue;
+
+                       uint64_t current_region_guest_start = 
current_region->guest_user_addr;
+                       uint64_t current_region_guest_end = 
current_region_guest_start
+                                                               + 
current_region->mmap_size - 1;
+                       uint64_t proposed_region_guest_start = 
region->userspace_addr;
+                       uint64_t proposed_region_guest_end = 
proposed_region_guest_start
+                                                               + 
region->memory_size - 1;
+                       bool overlap = false;
+
+                       bool curent_region_guest_start_overlap =
+                               current_region_guest_start >= 
proposed_region_guest_start
+                               && current_region_guest_start <= 
proposed_region_guest_end;
+                       bool curent_region_guest_end_overlap =
+                               current_region_guest_end >= 
proposed_region_guest_start
+                               && current_region_guest_end <= 
proposed_region_guest_end;
+                       bool proposed_region_guest_start_overlap =
+                               proposed_region_guest_start >= 
current_region_guest_start
+                               && proposed_region_guest_start <= 
current_region_guest_end;
+                       bool proposed_region_guest_end_overlap =
+                               proposed_region_guest_end >= 
current_region_guest_start
+                               && proposed_region_guest_end <= 
current_region_guest_end;
+
+                       overlap = curent_region_guest_start_overlap
+                               || curent_region_guest_end_overlap
+                               || proposed_region_guest_start_overlap
+                               || proposed_region_guest_end_overlap;
+
+                       if (overlap) {
+                               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                                       "requested memory region overlaps with 
another region");
+                               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                                       "\tRequested region address:0x%" PRIx64,
+                                       region->userspace_addr);
+                               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                                       "\tRequested region size:0x%" PRIx64,
+                                       region->memory_size);
+                               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                                       "\tOverlapping region address:0x%" 
PRIx64,
+                                       current_region->guest_user_addr);
+                               VHOST_CONFIG_LOG(dev->ifname, ERR,
+                                       "\tOverlapping region size:0x%" PRIx64,
+                                       current_region->mmap_size);
+                               goto close_msg_fds;
+                       }
+
+               }
+       }
+
+       /* convert first region add to normal memory table set */
+       if (dev->mem == NULL) {
+               if (vhost_user_initialize_memory(pdev) < 0)
+                       goto close_msg_fds;
+       }
+
+       /* find a new region and set it like memory table set does */
+       struct rte_vhost_mem_region *reg = NULL;
+       uint64_t mmap_offset;
+
+       for (uint32_t i = 0; i < VHOST_MEMORY_MAX_NREGIONS; i++) {
+               if (dev->mem->regions[i].guest_user_addr == 0) {
+                       reg = &dev->mem->regions[i];
+                       break;
+               }
+       }
+       if (reg == NULL) {
+               VHOST_CONFIG_LOG(dev->ifname, ERR, "no free memory region");
+               goto close_msg_fds;
+       }
+
+       reg->guest_phys_addr = region->guest_phys_addr;
+       reg->guest_user_addr = region->userspace_addr;
+       reg->size            = region->memory_size;
+       reg->fd              = ctx->fds[0];
+
+       mmap_offset = region->mmap_offset;
 
-free_guest_pages:
+       if (vhost_user_mmap_region(dev, reg, mmap_offset) < 0) {
+               VHOST_CONFIG_LOG(dev->ifname, ERR, "failed to mmap region");
+               goto close_msg_fds;
+       }
+
+       dev->mem->nregions++;
+
+       if (dev->async_copy && rte_vfio_is_enabled("vfio"))
+               async_dma_map(dev, true);
+
+       if (vhost_user_postcopy_register(dev, main_fd, ctx) < 0)
+               goto free_mem_table;
+
+       for (i = 0; i < dev->nr_vring; i++) {
+               struct vhost_virtqueue *vq = dev->virtqueue[i];
+
+               if (!vq)
+                       continue;
+
+               if (vq->desc || vq->avail || vq->used) {
+                       /* vhost_user_lock_all_queue_pairs locked all qps */
+                       VHOST_USER_ASSERT_LOCK(dev, vq, VHOST_USER_ADD_MEM_REG);
+
+                       /*
+                        * If the memory table got updated, the ring addresses
+                        * need to be translated again as virtual addresses have
+                        * changed.
+                        */
+                       vring_invalidate(dev, vq);
+
+                       translate_ring_addresses(&dev, &vq);
+                       *pdev = dev;
+               }
+       }
+
+       dump_guest_pages(dev);
+
+       return RTE_VHOST_MSG_RESULT_OK;
+
+free_mem_table:
+       free_all_mem_regions(dev);
+       rte_free(dev->mem);
+       dev->mem = NULL;
        rte_free(dev->guest_pages);
        dev->guest_pages = NULL;
 close_msg_fds:
@@ -1543,6 +1717,48 @@ vhost_user_set_mem_table(struct virtio_net **pdev,
        return RTE_VHOST_MSG_RESULT_ERR;
 }
 
+static int
+vhost_user_rem_mem_reg(struct virtio_net **pdev __rte_unused,
+                       struct vhu_msg_context *ctx __rte_unused,
+                       int main_fd __rte_unused)
+{
+       struct virtio_net *dev = *pdev;
+       struct VhostUserMemoryRegion *region = 
&ctx->msg.payload.memory_single.region;
+
+       if ((dev->mem) && (dev->flags & VIRTIO_DEV_VDPA_CONFIGURED)) {
+               struct rte_vdpa_device *vdpa_dev = dev->vdpa_dev;
+
+               if (vdpa_dev && vdpa_dev->ops->dev_close)
+                       vdpa_dev->ops->dev_close(dev->vid);
+               dev->flags &= ~VIRTIO_DEV_VDPA_CONFIGURED;
+       }
+
+       if (dev->mem != NULL && dev->mem->nregions > 0) {
+               for (uint32_t i = 0; i < VHOST_MEMORY_MAX_NREGIONS; i++) {
+                       struct rte_vhost_mem_region *current_region = 
&dev->mem->regions[i];
+
+                       if (current_region->guest_user_addr == 0)
+                               continue;
+
+                       /*
+                        * According to the vhost-user specification:
+                        * The memory region to be removed is identified by its 
guest address,
+                        * user address and size. The mmap offset is ignored.
+                        */
+                       if (region->userspace_addr == 
current_region->guest_user_addr
+                               && region->guest_phys_addr == 
current_region->guest_phys_addr
+                               && region->memory_size == current_region->size) 
{
+                               free_mem_region(current_region);
+                               dev->mem->nregions--;
+                               return RTE_VHOST_MSG_RESULT_OK;
+                       }
+               }
+       }
+
+       VHOST_CONFIG_LOG(dev->ifname, ERR, "failed to find region");
+       return RTE_VHOST_MSG_RESULT_ERR;
+}
+
 static bool
 vq_is_ready(struct virtio_net *dev, struct vhost_virtqueue *vq)
 {
-- 
2.43.0

Reply via email to