vhost: replace device ID in applications
[dpdk.git] / lib / librte_vhost / vdpa.c
index 321e11f..cececc3 100644 (file)
 
 #include <stdbool.h>
 
+#include <rte_class.h>
 #include <rte_malloc.h>
 #include "rte_vdpa.h"
 #include "vhost.h"
 
-static struct rte_vdpa_device *vdpa_devices[MAX_VHOST_DEVICE];
+static struct rte_vdpa_device vdpa_devices[MAX_VHOST_DEVICE];
 static uint32_t vdpa_device_num;
 
-static bool
-is_same_vdpa_device(struct rte_vdpa_dev_addr *a,
-               struct rte_vdpa_dev_addr *b)
+
+int
+rte_vdpa_find_device_id(struct rte_vdpa_device *dev)
 {
-       bool ret = true;
-
-       if (a->type != b->type)
-               return false;
-
-       switch (a->type) {
-       case PCI_ADDR:
-               if (a->pci_addr.domain != b->pci_addr.domain ||
-                               a->pci_addr.bus != b->pci_addr.bus ||
-                               a->pci_addr.devid != b->pci_addr.devid ||
-                               a->pci_addr.function != b->pci_addr.function)
-                       ret = false;
-               break;
-       default:
-               break;
+       struct rte_vdpa_device *tmp_dev;
+       int i;
+
+       if (dev == NULL)
+               return -1;
+
+       for (i = 0; i < MAX_VHOST_DEVICE; ++i) {
+               tmp_dev = &vdpa_devices[i];
+               if (tmp_dev->ops == NULL)
+                       continue;
+
+               if (tmp_dev == dev)
+                       return i;
        }
 
-       return ret;
+       return -1;
 }
 
-int
-rte_vdpa_register_device(struct rte_vdpa_dev_addr *addr,
-               struct rte_vdpa_dev_ops *ops)
+struct rte_vdpa_device *
+rte_vdpa_find_device_by_name(const char *name)
 {
        struct rte_vdpa_device *dev;
-       char device_name[MAX_VDPA_NAME_LEN];
        int i;
 
-       if (vdpa_device_num >= MAX_VHOST_DEVICE)
-               return -1;
+       if (name == NULL)
+               return NULL;
 
-       for (i = 0; i < MAX_VHOST_DEVICE; i++) {
-               dev = vdpa_devices[i];
-               if (dev && is_same_vdpa_device(&dev->addr, addr))
-                       return -1;
-       }
+       for (i = 0; i < MAX_VHOST_DEVICE; ++i) {
+               dev = &vdpa_devices[i];
+               if (dev->ops == NULL)
+                       continue;
 
-       for (i = 0; i < MAX_VHOST_DEVICE; i++) {
-               if (vdpa_devices[i] == NULL)
-                       break;
+               if (strncmp(dev->device->name, name, RTE_DEV_NAME_MAX_LEN) == 0)
+                       return dev;
        }
 
-       if (i == MAX_VHOST_DEVICE)
-               return -1;
-
-       snprintf(device_name, sizeof(device_name), "vdpa-dev-%d", i);
-       dev = rte_zmalloc(device_name, sizeof(struct rte_vdpa_device),
-                       RTE_CACHE_LINE_SIZE);
-       if (!dev)
-               return -1;
+       return NULL;
+}
 
-       memcpy(&dev->addr, addr, sizeof(struct rte_vdpa_dev_addr));
-       dev->ops = ops;
-       vdpa_devices[i] = dev;
-       vdpa_device_num++;
+struct rte_device *
+rte_vdpa_get_rte_device(struct rte_vdpa_device *vdpa_dev)
+{
+       if (vdpa_dev == NULL)
+               return NULL;
 
-       return i;
+       return vdpa_dev->device;
 }
 
-int
-rte_vdpa_unregister_device(int did)
+struct rte_vdpa_device *
+rte_vdpa_get_device(int did)
 {
-       if (did < 0 || did >= MAX_VHOST_DEVICE || vdpa_devices[did] == NULL)
-               return -1;
-
-       rte_free(vdpa_devices[did]);
-       vdpa_devices[did] = NULL;
-       vdpa_device_num--;
+       if (did < 0 || did >= MAX_VHOST_DEVICE)
+               return NULL;
 
-       return did;
+       return &vdpa_devices[did];
 }
 
-int
-rte_vdpa_find_device_id(struct rte_vdpa_dev_addr *addr)
+struct rte_vdpa_device *
+rte_vdpa_register_device(struct rte_device *rte_dev,
+               struct rte_vdpa_dev_ops *ops)
 {
        struct rte_vdpa_device *dev;
        int i;
 
-       for (i = 0; i < MAX_VHOST_DEVICE; ++i) {
-               dev = vdpa_devices[i];
-               if (dev && is_same_vdpa_device(&dev->addr, addr))
-                       return i;
+       if (vdpa_device_num >= MAX_VHOST_DEVICE || ops == NULL)
+               return NULL;
+
+       for (i = 0; i < MAX_VHOST_DEVICE; i++) {
+               dev = &vdpa_devices[i];
+               if (dev->ops == NULL)
+                       continue;
+
+               if (dev->device == rte_dev)
+                       return NULL;
        }
 
-       return -1;
+       for (i = 0; i < MAX_VHOST_DEVICE; i++) {
+               if (vdpa_devices[i].ops == NULL)
+                       break;
+       }
+
+       if (i == MAX_VHOST_DEVICE)
+               return NULL;
+
+       dev = &vdpa_devices[i];
+       dev->device = rte_dev;
+       dev->ops = ops;
+       vdpa_device_num++;
+
+       return dev;
 }
 
-struct rte_vdpa_device *
-rte_vdpa_get_device(int did)
+int
+rte_vdpa_unregister_device(struct rte_vdpa_device *vdev)
 {
-       if (did < 0 || did >= MAX_VHOST_DEVICE)
-               return NULL;
+       int did = rte_vdpa_find_device_id(vdev);
 
-       return vdpa_devices[did];
+       if (did < 0 || vdpa_devices[did].ops == NULL)
+               return -1;
+
+       memset(&vdpa_devices[did], 0, sizeof(struct rte_vdpa_device));
+       vdpa_device_num--;
+
+       return 0;
 }
 
 int
@@ -123,7 +134,7 @@ rte_vdpa_get_device_num(void)
        return vdpa_device_num;
 }
 
-int __rte_experimental
+int
 rte_vdpa_relay_vring_used(int vid, uint16_t qid, void *vring_m)
 {
        struct virtio_net *dev = get_device(vid);
@@ -178,7 +189,7 @@ rte_vdpa_relay_vring_used(int vid, uint16_t qid, void *vring_m)
                                return -1;
 
                        if (unlikely(dlen < vq->desc[desc_id].len)) {
-                               idesc = alloc_copy_ind_table(dev, vq,
+                               idesc = vhost_alloc_copy_ind_table(dev, vq,
                                                vq->desc[desc_id].addr,
                                                vq->desc[desc_id].len);
                                if (unlikely(!idesc))
@@ -198,7 +209,8 @@ rte_vdpa_relay_vring_used(int vid, uint16_t qid, void *vring_m)
                                goto fail;
                        desc = desc_ring[desc_id];
                        if (desc.flags & VRING_DESC_F_WRITE)
-                               vhost_log_write(dev, desc.addr, desc.len);
+                               vhost_log_write_iova(dev, vq, desc.addr,
+                                                    desc.len);
                        desc_id = desc.next;
                } while (desc.flags & VRING_DESC_F_NEXT);
 
@@ -223,3 +235,111 @@ fail:
                free_ind_table(idesc);
        return -1;
 }
+
+int
+rte_vdpa_get_stats_names(struct rte_vdpa_device *dev,
+               struct rte_vdpa_stat_name *stats_names,
+               unsigned int size)
+{
+       if (!dev)
+               return -EINVAL;
+
+       RTE_FUNC_PTR_OR_ERR_RET(dev->ops->get_stats_names, -ENOTSUP);
+
+       return dev->ops->get_stats_names(dev, stats_names, size);
+}
+
+int
+rte_vdpa_get_stats(struct rte_vdpa_device *dev, uint16_t qid,
+               struct rte_vdpa_stat *stats, unsigned int n)
+{
+       if (!dev || !stats || !n)
+               return -EINVAL;
+
+       RTE_FUNC_PTR_OR_ERR_RET(dev->ops->get_stats, -ENOTSUP);
+
+       return dev->ops->get_stats(dev, qid, stats, n);
+}
+
+int
+rte_vdpa_reset_stats(struct rte_vdpa_device *dev, uint16_t qid)
+{
+       if (!dev)
+               return -EINVAL;
+
+       RTE_FUNC_PTR_OR_ERR_RET(dev->ops->reset_stats, -ENOTSUP);
+
+       return dev->ops->reset_stats(dev, qid);
+}
+
+static uint16_t
+vdpa_dev_to_id(const struct rte_vdpa_device *dev)
+{
+       if (dev == NULL)
+               return MAX_VHOST_DEVICE;
+
+       if (dev < &vdpa_devices[0] ||
+                       dev >= &vdpa_devices[MAX_VHOST_DEVICE])
+               return MAX_VHOST_DEVICE;
+
+       return (uint16_t)(dev - vdpa_devices);
+}
+
+static int
+vdpa_dev_match(struct rte_vdpa_device *dev,
+             const struct rte_device *rte_dev)
+{
+       if (dev->device == rte_dev)
+               return 0;
+
+       return -1;
+}
+
+/* Generic rte_vdpa_dev comparison function. */
+typedef int (*rte_vdpa_cmp_t)(struct rte_vdpa_device *,
+               const struct rte_device *rte_dev);
+
+static struct rte_vdpa_device *
+vdpa_find_device(const struct rte_vdpa_device *start, rte_vdpa_cmp_t cmp,
+               struct rte_device *rte_dev)
+{
+       struct rte_vdpa_device *dev;
+       uint16_t idx;
+
+       if (start != NULL)
+               idx = vdpa_dev_to_id(start) + 1;
+       else
+               idx = 0;
+       for (; idx < MAX_VHOST_DEVICE; idx++) {
+               dev = &vdpa_devices[idx];
+               /*
+                * ToDo: Certainly better to introduce a state field,
+                * but rely on ops being set for now.
+                */
+               if (dev->ops == NULL)
+                       continue;
+               if (cmp(dev, rte_dev) == 0)
+                       return dev;
+       }
+       return NULL;
+}
+
+static void *
+vdpa_dev_iterate(const void *start,
+               const char *str,
+               const struct rte_dev_iterator *it)
+{
+       struct rte_vdpa_device *vdpa_dev = NULL;
+
+       RTE_SET_USED(str);
+
+       vdpa_dev = vdpa_find_device(start, vdpa_dev_match, it->device);
+
+       return vdpa_dev;
+}
+
+static struct rte_class rte_class_vdpa = {
+       .dev_iterate = vdpa_dev_iterate,
+};
+
+RTE_REGISTER_CLASS(vdpa, rte_class_vdpa);