[dpdk-dev] [PATCH v3 00/10] Rework vdev probing to use rte_bus infrastructure

2017-02-25 Thread Jan Blunck
With the rte_bus infrastructure present in 17.02 it is possible to refactor
the virtual device probing into a bus. This series also introduces the
rte_vdev_device to better keep track of devices.

This patchset depends on:
http://dpdk.org/dev/patchwork/patch/20416/
http://dpdk.org/dev/patchwork/patch/20417/

Changes since version 2:
 * implicit bus registration through rte_eal_vdrv_register()
 * explicit delay probing of virtual bus in rte_bus_probe()
 * addition of rte_vdev_device_args() helper
 * make virtual driver probe and remove take rte_vdev_device

Changes since version 1:
 * addition of rte_vdev_device_name() helper
 * removed rte_eal_dev_init() from *.map files
 * use SOCKET_ID_ANY

Jan Blunck (10):
  eal: probe legacy PCI devices before other bus devices
  eal: probe new virtual bus after other bus devices
  eal: move virtual device probing into a bus
  eal: remove unused rte_eal_dev_init()
  eal: Refactor vdev driver probe/remove
  eal: add struct rte_vdev_device
  eal: add virtual device name helper function
  eal: add virtual device arguments helper function
  eal: make virtual bus use rte_vdev_device
  eal: make virtual driver probe and remove take rte_vdev_device

 drivers/crypto/null/null_crypto_pmd.c   |  18 +-
 drivers/net/af_packet/rte_eth_af_packet.c   |  11 +-
 drivers/net/bonding/rte_eth_bond_pmd.c  |  13 +-
 drivers/net/mpipe/mpipe_tilegx.c|  10 +-
 drivers/net/null/rte_eth_null.c |  13 +-
 drivers/net/pcap/rte_eth_pcap.c |  12 +-
 drivers/net/ring/rte_eth_ring.c |   9 +-
 drivers/net/tap/rte_eth_tap.c   |  10 +-
 drivers/net/vhost/rte_eth_vhost.c   |  10 +-
 drivers/net/virtio/virtio_user_ethdev.c |  18 +-
 drivers/net/xenvirt/rte_eth_xenvirt.c   |   9 +-
 lib/librte_eal/bsdapp/eal/eal.c |   9 +-
 lib/librte_eal/bsdapp/eal/rte_eal_version.map   |   1 -
 lib/librte_eal/common/eal_common_bus.c  |  16 +-
 lib/librte_eal/common/eal_common_dev.c  |  28 ---
 lib/librte_eal/common/eal_common_vdev.c | 259 +---
 lib/librte_eal/common/include/rte_dev.h |   5 -
 lib/librte_eal/common/include/rte_vdev.h|  26 ++-
 lib/librte_eal/linuxapp/eal/eal.c   |   9 +-
 lib/librte_eal/linuxapp/eal/rte_eal_version.map |   1 -
 20 files changed, 355 insertions(+), 132 deletions(-)

-- 
2.7.4



[dpdk-dev] [PATCH v3 03/10] eal: move virtual device probing into a bus

2017-02-25 Thread Jan Blunck
This is a refactoring of the virtual device probing which moves into into
a proper bus structure.

Signed-off-by: Jan Blunck 
Tested-by: Ferruh Yigit 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/common/eal_common_dev.c  | 22 -
 lib/librte_eal/common/eal_common_vdev.c | 58 +
 2 files changed, 58 insertions(+), 22 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_dev.c 
b/lib/librte_eal/common/eal_common_dev.c
index 4f3b493..1ce90f6 100644
--- a/lib/librte_eal/common/eal_common_dev.c
+++ b/lib/librte_eal/common/eal_common_dev.c
@@ -79,28 +79,6 @@ void rte_eal_device_remove(struct rte_device *dev)
 int
 rte_eal_dev_init(void)
 {
-   struct rte_devargs *devargs;
-
-   /*
-* Note that the dev_driver_list is populated here
-* from calls made to rte_eal_driver_register from constructor functions
-* embedded into PMD modules via the RTE_PMD_REGISTER_VDEV macro
-*/
-
-   /* call the init function for each virtual device */
-   TAILQ_FOREACH(devargs, &devargs_list, next) {
-
-   if (devargs->type != RTE_DEVTYPE_VIRTUAL)
-   continue;
-
-   if (rte_eal_vdev_init(devargs->virt.drv_name,
-   devargs->args)) {
-   RTE_LOG(ERR, EAL, "failed to initialize %s device\n",
-   devargs->virt.drv_name);
-   return -1;
-   }
-   }
-
return 0;
 }
 
diff --git a/lib/librte_eal/common/eal_common_vdev.c 
b/lib/librte_eal/common/eal_common_vdev.c
index 7d6e54f..0fcfdd7 100644
--- a/lib/librte_eal/common/eal_common_vdev.c
+++ b/lib/librte_eal/common/eal_common_vdev.c
@@ -37,16 +37,22 @@
 #include 
 #include 
 
+#include 
 #include 
 #include 
+#include 
 
 struct vdev_driver_list vdev_driver_list =
TAILQ_HEAD_INITIALIZER(vdev_driver_list);
 
+static void rte_vdev_bus_register(void);
+
 /* register a driver */
 void
 rte_eal_vdrv_register(struct rte_vdev_driver *driver)
 {
+   rte_vdev_bus_register();
+
TAILQ_INSERT_TAIL(&vdev_driver_list, driver, next);
rte_eal_driver_register(&driver->driver);
 }
@@ -122,3 +128,55 @@ rte_eal_vdev_uninit(const char *name)
RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
return -EINVAL;
 }
+
+static int
+vdev_scan(void)
+{
+   /* for virtual devices we don't need to scan anything */
+   return 0;
+}
+
+static int
+vdev_probe(void)
+{
+   struct rte_devargs *devargs;
+
+   /*
+* Note that the dev_driver_list is populated here
+* from calls made to rte_eal_driver_register from constructor functions
+* embedded into PMD modules via the RTE_PMD_REGISTER_VDEV macro
+*/
+
+   /* call the init function for each virtual device */
+   TAILQ_FOREACH(devargs, &devargs_list, next) {
+
+   if (devargs->type != RTE_DEVTYPE_VIRTUAL)
+   continue;
+
+   if (rte_eal_vdev_init(devargs->virt.drv_name,
+ devargs->args)) {
+   RTE_LOG(ERR, EAL, "failed to initialize %s device\n",
+   devargs->virt.drv_name);
+   return -1;
+   }
+   }
+
+   return 0;
+}
+
+static struct rte_bus rte_vdev_bus = {
+   .scan = vdev_scan,
+   .probe = vdev_probe,
+};
+
+static void rte_vdev_bus_register(void)
+{
+   static int registered = 0;
+
+   if (registered)
+   return;
+
+   registered = 1;
+   rte_vdev_bus.name = RTE_STR(virtual);
+   rte_bus_register(&rte_vdev_bus);
+}
-- 
2.7.4



[dpdk-dev] [PATCH v3 04/10] eal: remove unused rte_eal_dev_init()

2017-02-25 Thread Jan Blunck
Signed-off-by: Jan Blunck 
Tested-by: Ferruh Yigit 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/bsdapp/eal/eal.c | 3 ---
 lib/librte_eal/bsdapp/eal/rte_eal_version.map   | 1 -
 lib/librte_eal/common/eal_common_dev.c  | 6 --
 lib/librte_eal/common/include/rte_dev.h | 5 -
 lib/librte_eal/linuxapp/eal/eal.c   | 3 ---
 lib/librte_eal/linuxapp/eal/rte_eal_version.map | 1 -
 6 files changed, 19 deletions(-)

diff --git a/lib/librte_eal/bsdapp/eal/eal.c b/lib/librte_eal/bsdapp/eal/eal.c
index a584447..3d29fcb 100644
--- a/lib/librte_eal/bsdapp/eal/eal.c
+++ b/lib/librte_eal/bsdapp/eal/eal.c
@@ -621,9 +621,6 @@ rte_eal_init(int argc, char **argv)
if (rte_bus_probe())
rte_panic("Cannot probe devices\n");
 
-   if (rte_eal_dev_init() < 0)
-   rte_panic("Cannot init pmd devices\n");
-
rte_eal_mcfg_complete();
 
return fctret;
diff --git a/lib/librte_eal/bsdapp/eal/rte_eal_version.map 
b/lib/librte_eal/bsdapp/eal/rte_eal_version.map
index 67f2ffb..b1996e0 100644
--- a/lib/librte_eal/bsdapp/eal/rte_eal_version.map
+++ b/lib/librte_eal/bsdapp/eal/rte_eal_version.map
@@ -22,7 +22,6 @@ DPDK_2.0 {
rte_dump_tailq;
rte_eal_alarm_cancel;
rte_eal_alarm_set;
-   rte_eal_dev_init;
rte_eal_devargs_add;
rte_eal_devargs_dump;
rte_eal_devargs_type_count;
diff --git a/lib/librte_eal/common/eal_common_dev.c 
b/lib/librte_eal/common/eal_common_dev.c
index 1ce90f6..4bde430 100644
--- a/lib/librte_eal/common/eal_common_dev.c
+++ b/lib/librte_eal/common/eal_common_dev.c
@@ -76,12 +76,6 @@ void rte_eal_device_remove(struct rte_device *dev)
TAILQ_REMOVE(&dev_device_list, dev, next);
 }
 
-int
-rte_eal_dev_init(void)
-{
-   return 0;
-}
-
 int rte_eal_dev_attach(const char *name, const char *devargs)
 {
struct rte_pci_addr addr;
diff --git a/lib/librte_eal/common/include/rte_dev.h 
b/lib/librte_eal/common/include/rte_dev.h
index b17791f..4251099 100644
--- a/lib/librte_eal/common/include/rte_dev.h
+++ b/lib/librte_eal/common/include/rte_dev.h
@@ -171,11 +171,6 @@ void rte_eal_driver_register(struct rte_driver *driver);
 void rte_eal_driver_unregister(struct rte_driver *driver);
 
 /**
- * Initalize all the registered drivers in this process
- */
-int rte_eal_dev_init(void);
-
-/**
  * Initialize a driver specified by name.
  *
  * @param name
diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index f77ff5c..88479de 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -892,9 +892,6 @@ rte_eal_init(int argc, char **argv)
if (rte_bus_probe())
rte_panic("Cannot probe devices\n");
 
-   if (rte_eal_dev_init() < 0)
-   rte_panic("Cannot init pmd devices\n");
-
rte_eal_mcfg_complete();
 
return fctret;
diff --git a/lib/librte_eal/linuxapp/eal/rte_eal_version.map 
b/lib/librte_eal/linuxapp/eal/rte_eal_version.map
index 9c134b4..2ada2f0 100644
--- a/lib/librte_eal/linuxapp/eal/rte_eal_version.map
+++ b/lib/librte_eal/linuxapp/eal/rte_eal_version.map
@@ -22,7 +22,6 @@ DPDK_2.0 {
rte_dump_tailq;
rte_eal_alarm_cancel;
rte_eal_alarm_set;
-   rte_eal_dev_init;
rte_eal_devargs_add;
rte_eal_devargs_dump;
rte_eal_devargs_type_count;
-- 
2.7.4



[dpdk-dev] [PATCH v3 02/10] eal: probe new virtual bus after other bus devices

2017-02-25 Thread Jan Blunck
Also see commit f4ce209a ("eal: postpone vdev initialization").

Signed-off-by: Jan Blunck 
---
 lib/librte_eal/common/eal_common_bus.c | 16 +++-
 1 file changed, 15 insertions(+), 1 deletion(-)

diff --git a/lib/librte_eal/common/eal_common_bus.c 
b/lib/librte_eal/common/eal_common_bus.c
index 4638e78..8f9baf8 100644
--- a/lib/librte_eal/common/eal_common_bus.c
+++ b/lib/librte_eal/common/eal_common_bus.c
@@ -86,9 +86,14 @@ int
 rte_bus_probe(void)
 {
int ret;
-   struct rte_bus *bus;
+   struct rte_bus *bus, *vbus = NULL;
 
TAILQ_FOREACH(bus, &rte_bus_list, next) {
+   if (!strcmp(bus->name, "virtual")) {
+   vbus = bus;
+   continue;
+   }
+
ret = bus->probe();
if (ret) {
RTE_LOG(ERR, EAL, "Bus (%s) probe failed.\n",
@@ -97,6 +102,15 @@ rte_bus_probe(void)
}
}
 
+   if (vbus) {
+   ret = vbus->probe();
+   if (ret) {
+   RTE_LOG(ERR, EAL, "Bus (%s) probe failed.\n",
+   vbus->name);
+   return ret;
+   }
+   }
+
return 0;
 }
 
-- 
2.7.4



[dpdk-dev] [PATCH v3 01/10] eal: probe legacy PCI devices before other bus devices

2017-02-25 Thread Jan Blunck
Make sure that the PCI devices are probed before the virtual devices after
the legacy virtual device probing has been moved to a bus.

Signed-off-by: Jan Blunck 
Tested-by: Ferruh Yigit 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/bsdapp/eal/eal.c   | 8 
 lib/librte_eal/linuxapp/eal/eal.c | 8 
 2 files changed, 8 insertions(+), 8 deletions(-)

diff --git a/lib/librte_eal/bsdapp/eal/eal.c b/lib/librte_eal/bsdapp/eal/eal.c
index ee7c9de..a584447 100644
--- a/lib/librte_eal/bsdapp/eal/eal.c
+++ b/lib/librte_eal/bsdapp/eal/eal.c
@@ -613,14 +613,14 @@ rte_eal_init(int argc, char **argv)
rte_eal_mp_remote_launch(sync_func, NULL, SKIP_MASTER);
rte_eal_mp_wait_lcore();
 
-   /* Probe all the buses and devices/drivers on them */
-   if (rte_bus_probe())
-   rte_panic("Cannot probe devices\n");
-
/* Probe & Initialize PCI devices */
if (rte_eal_pci_probe())
rte_panic("Cannot probe PCI\n");
 
+   /* Probe all the buses and devices/drivers on them */
+   if (rte_bus_probe())
+   rte_panic("Cannot probe devices\n");
+
if (rte_eal_dev_init() < 0)
rte_panic("Cannot init pmd devices\n");
 
diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index bf6b818..f77ff5c 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -884,14 +884,14 @@ rte_eal_init(int argc, char **argv)
rte_eal_mp_remote_launch(sync_func, NULL, SKIP_MASTER);
rte_eal_mp_wait_lcore();
 
-   /* Probe all the buses and devices/drivers on them */
-   if (rte_bus_probe())
-   rte_panic("Cannot probe devices\n");
-
/* Probe & Initialize PCI devices */
if (rte_eal_pci_probe())
rte_panic("Cannot probe PCI\n");
 
+   /* Probe all the buses and devices/drivers on them */
+   if (rte_bus_probe())
+   rte_panic("Cannot probe devices\n");
+
if (rte_eal_dev_init() < 0)
rte_panic("Cannot init pmd devices\n");
 
-- 
2.7.4



[dpdk-dev] [PATCH v3 05/10] eal: Refactor vdev driver probe/remove

2017-02-25 Thread Jan Blunck
This is a preparation for the introduction of the struct rte_vdev_device.

Signed-off-by: Jan Blunck 
Tested-by: Ferruh Yigit 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/common/eal_common_vdev.c | 44 -
 1 file changed, 33 insertions(+), 11 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_vdev.c 
b/lib/librte_eal/common/eal_common_vdev.c
index 0fcfdd7..5a267ee 100644
--- a/lib/librte_eal/common/eal_common_vdev.c
+++ b/lib/librte_eal/common/eal_common_vdev.c
@@ -65,14 +65,11 @@ rte_eal_vdrv_unregister(struct rte_vdev_driver *driver)
TAILQ_REMOVE(&vdev_driver_list, driver, next);
 }
 
-int
-rte_eal_vdev_init(const char *name, const char *args)
+static int
+vdev_probe_all_drivers(const char *name, const char *args)
 {
struct rte_vdev_driver *driver;
 
-   if (name == NULL)
-   return -EINVAL;
-
TAILQ_FOREACH(driver, &vdev_driver_list, next) {
/*
 * search a driver prefix in virtual device name.
@@ -93,18 +90,29 @@ rte_eal_vdev_init(const char *name, const char *args)
return driver->probe(name, args);
}
 
-   RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
-   return -EINVAL;
+   return 1;
 }
 
 int
-rte_eal_vdev_uninit(const char *name)
+rte_eal_vdev_init(const char *name, const char *args)
 {
-   struct rte_vdev_driver *driver;
+   int ret;
 
if (name == NULL)
return -EINVAL;
 
+   ret = vdev_probe_all_drivers(name, args);
+   if (ret  > 0)
+   RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
+
+   return ret;
+}
+
+static int
+vdev_remove_driver(const char *name)
+{
+   struct rte_vdev_driver *driver;
+
TAILQ_FOREACH(driver, &vdev_driver_list, next) {
/*
 * search a driver prefix in virtual device name.
@@ -125,8 +133,22 @@ rte_eal_vdev_uninit(const char *name)
return driver->remove(name);
}
 
-   RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
-   return -EINVAL;
+   return 1;
+}
+
+int
+rte_eal_vdev_uninit(const char *name)
+{
+   int ret;
+
+   if (name == NULL)
+   return -EINVAL;
+
+   ret = vdev_remove_driver(name);
+   if (ret > 0)
+   RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
+
+   return ret;
 }
 
 static int
-- 
2.7.4



[dpdk-dev] [PATCH v3 06/10] eal: add struct rte_vdev_device

2017-02-25 Thread Jan Blunck
This adds the rte_vdev_device structure which embeds a generic rte_device.

Signed-off-by: Jan Blunck 
Tested-by: Ferruh Yigit 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/common/eal_common_vdev.c  | 5 +
 lib/librte_eal/common/include/rte_vdev.h | 5 +
 2 files changed, 10 insertions(+)

diff --git a/lib/librte_eal/common/eal_common_vdev.c 
b/lib/librte_eal/common/eal_common_vdev.c
index 5a267ee..c88cae9 100644
--- a/lib/librte_eal/common/eal_common_vdev.c
+++ b/lib/librte_eal/common/eal_common_vdev.c
@@ -42,6 +42,11 @@
 #include 
 #include 
 
+/** Double linked list of virtual device drivers. */
+TAILQ_HEAD(vdev_device_list, rte_vdev_device);
+
+static struct vdev_device_list vdev_device_list =
+   TAILQ_HEAD_INITIALIZER(vdev_device_list);
 struct vdev_driver_list vdev_driver_list =
TAILQ_HEAD_INITIALIZER(vdev_driver_list);
 
diff --git a/lib/librte_eal/common/include/rte_vdev.h 
b/lib/librte_eal/common/include/rte_vdev.h
index 784e837..8f98372 100644
--- a/lib/librte_eal/common/include/rte_vdev.h
+++ b/lib/librte_eal/common/include/rte_vdev.h
@@ -40,6 +40,11 @@ extern "C" {
 #include 
 #include 
 
+struct rte_vdev_device {
+   TAILQ_ENTRY(rte_vdev_device) next;  /**< Next attached vdev */
+   struct rte_device device;   /**< Inherit core device */
+};
+
 /** Double linked list of virtual device drivers. */
 TAILQ_HEAD(vdev_driver_list, rte_vdev_driver);
 
-- 
2.7.4



[dpdk-dev] [PATCH v3 09/10] eal: make virtual bus use rte_vdev_device

2017-02-25 Thread Jan Blunck
This allows the virtual bus to be rescanned and probed by tracking the
creation of rte_vdev_device.

Signed-off-by: Jan Blunck 
Tested-by: Ferruh Yigit 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/common/eal_common_vdev.c | 195 +---
 1 file changed, 155 insertions(+), 40 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_vdev.c 
b/lib/librte_eal/common/eal_common_vdev.c
index c88cae9..7215b50 100644
--- a/lib/librte_eal/common/eal_common_vdev.c
+++ b/lib/librte_eal/common/eal_common_vdev.c
@@ -41,6 +41,7 @@
 #include 
 #include 
 #include 
+#include 
 
 /** Double linked list of virtual device drivers. */
 TAILQ_HEAD(vdev_device_list, rte_vdev_device);
@@ -71,9 +72,12 @@ rte_eal_vdrv_unregister(struct rte_vdev_driver *driver)
 }
 
 static int
-vdev_probe_all_drivers(const char *name, const char *args)
+vdev_probe_all_drivers(struct rte_vdev_device *dev)
 {
+   const char *name = rte_vdev_device_name(dev);
+   const char *args = rte_vdev_device_args(dev);
struct rte_vdev_driver *driver;
+   int ret;
 
TAILQ_FOREACH(driver, &vdev_driver_list, next) {
/*
@@ -83,90 +87,202 @@ vdev_probe_all_drivers(const char *name, const char *args)
 * So use strncmp to compare.
 */
if (!strncmp(driver->driver.name, name,
-   strlen(driver->driver.name)))
-   return driver->probe(name, args);
+   strlen(driver->driver.name))) {
+   dev->device.driver = &driver->driver;
+   ret = driver->probe(name, args);
+   if (ret)
+   dev->device.driver = NULL;
+   return ret;
+   }
}
 
/* Give new names precedence over aliases. */
TAILQ_FOREACH(driver, &vdev_driver_list, next) {
if (driver->driver.alias &&
!strncmp(driver->driver.alias, name,
-   strlen(driver->driver.alias)))
-   return driver->probe(name, args);
+   strlen(driver->driver.alias))) {
+   dev->device.driver = &driver->driver;
+   ret = driver->probe(name, args);
+   if (ret)
+   dev->device.driver = NULL;
+   return ret;
+   }
}
 
return 1;
 }
 
+static struct rte_vdev_device *
+find_vdev(const char *name)
+{
+   struct rte_vdev_device *dev;
+
+   if (!name)
+   return NULL;
+
+   TAILQ_FOREACH(dev, &vdev_device_list, next) {
+   const char *devname = rte_vdev_device_name(dev);
+   if (!strncmp(devname, name, strlen(name)))
+   return dev;
+   }
+
+   return NULL;
+}
+
+static struct rte_devargs *
+alloc_devargs(const char *name, const char *args)
+{
+   struct rte_devargs *devargs;
+   int ret;
+
+   devargs = calloc(1, sizeof(*devargs));
+   if (!devargs)
+   return NULL;
+
+   devargs->type = RTE_DEVTYPE_VIRTUAL;
+   if (args)
+   devargs->args = strdup(args);
+
+   ret = snprintf(devargs->virt.drv_name,
+  sizeof(devargs->virt.drv_name), "%s", name);
+   if (ret < 0 || ret >= (int)sizeof(devargs->virt.drv_name)) {
+   free(devargs->args);
+   free(devargs);
+   return NULL;
+   }
+
+   return devargs;
+}
+
 int
 rte_eal_vdev_init(const char *name, const char *args)
 {
+   struct rte_vdev_device *dev;
+   struct rte_devargs *devargs;
int ret;
 
if (name == NULL)
return -EINVAL;
 
-   ret = vdev_probe_all_drivers(name, args);
-   if (ret  > 0)
-   RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
+   dev = find_vdev(name);
+   if (dev)
+   return -EEXIST;
+
+   devargs = alloc_devargs(name, args);
+   if (!devargs)
+   return -ENOMEM;
+
+   dev = calloc(1, sizeof(*dev));
+   if (!dev) {
+   ret = -ENOMEM;
+   goto fail;
+   }
+
+   dev->device.devargs = devargs;
+   dev->device.numa_node = SOCKET_ID_ANY;
+
+   ret = vdev_probe_all_drivers(dev);
+   if (ret) {
+   if (ret > 0)
+   RTE_LOG(ERR, EAL, "no driver found for %s\n", name);
+   goto fail;
+   }
+
+   TAILQ_INSERT_TAIL(&devargs_list, devargs, next);
+
+   rte_eal_device_insert(&dev->device);
+   TAILQ_INSERT_TAIL(&vdev_device_list, dev, next);
+   return 0;
 
+fail:
+   free(devargs->args);
+   free(devargs);
+   free(dev);
return ret;
 }
 
 static int
-vdev_remove_driver(const char *name)
+vdev_remove_driver(struct rte_vdev_device *dev)
 {
-   struct rte_vdev_driver *driver;
+   const char *name = 

[dpdk-dev] [PATCH v3 10/10] eal: make virtual driver probe and remove take rte_vdev_device

2017-02-25 Thread Jan Blunck
This is a preparation to embed the generic rte_device into the rte_eth_dev
also for virtual devices.

Signed-off-by: Jan Blunck 
---
 drivers/crypto/null/null_crypto_pmd.c | 18 --
 drivers/net/af_packet/rte_eth_af_packet.c | 11 ++-
 drivers/net/bonding/rte_eth_bond_pmd.c| 13 +
 drivers/net/mpipe/mpipe_tilegx.c  | 10 ++
 drivers/net/null/rte_eth_null.c   | 13 -
 drivers/net/pcap/rte_eth_pcap.c   | 12 +++-
 drivers/net/ring/rte_eth_ring.c   |  9 +++--
 drivers/net/tap/rte_eth_tap.c | 10 +++---
 drivers/net/vhost/rte_eth_vhost.c | 10 +++---
 drivers/net/virtio/virtio_user_ethdev.c   | 18 +++---
 drivers/net/xenvirt/rte_eth_xenvirt.c |  9 +
 lib/librte_eal/common/eal_common_vdev.c   |  7 +++
 lib/librte_eal/common/include/rte_vdev.h  |  4 ++--
 13 files changed, 86 insertions(+), 58 deletions(-)

diff --git a/drivers/crypto/null/null_crypto_pmd.c 
b/drivers/crypto/null/null_crypto_pmd.c
index ed5a9fc..51d6de0 100644
--- a/drivers/crypto/null/null_crypto_pmd.c
+++ b/drivers/crypto/null/null_crypto_pmd.c
@@ -218,8 +218,7 @@ cryptodev_null_create(struct rte_crypto_vdev_init_params 
*init_params)
 
 /** Initialise null crypto device */
 static int
-cryptodev_null_probe(const char *name,
-   const char *input_args)
+cryptodev_null_probe(struct rte_vdev_device *dev)
 {
struct rte_crypto_vdev_init_params init_params = {
RTE_CRYPTODEV_VDEV_DEFAULT_MAX_NB_QUEUE_PAIRS,
@@ -228,10 +227,11 @@ cryptodev_null_probe(const char *name,
{0}
};
 
-   rte_cryptodev_parse_vdev_init_params(&init_params, input_args);
+   rte_cryptodev_parse_vdev_init_params(&init_params,
+   rte_vdev_device_args(dev));
 
-   RTE_LOG(INFO, PMD, "Initialising %s on NUMA node %d\n", name,
-   init_params.socket_id);
+   RTE_LOG(INFO, PMD, "Initialising %s on NUMA node %d\n",
+   rte_vdev_device_name(dev), init_params.socket_id);
if (init_params.name[0] != '\0')
RTE_LOG(INFO, PMD, "  User defined name = %s\n",
init_params.name);
@@ -256,9 +256,15 @@ cryptodev_null_remove(const char *name)
return 0;
 }
 
+static int
+cryptodev_null_remove_dev(struct rte_vdev_device *dev)
+{
+   return cryptodev_null_remove(rte_vdev_device_name(dev));
+}
+
 static struct rte_vdev_driver cryptodev_null_pmd_drv = {
.probe = cryptodev_null_probe,
-   .remove = cryptodev_null_remove
+   .remove = cryptodev_null_remove_dev,
 };
 
 RTE_PMD_REGISTER_VDEV(CRYPTODEV_NAME_NULL_PMD, cryptodev_null_pmd_drv);
diff --git a/drivers/net/af_packet/rte_eth_af_packet.c 
b/drivers/net/af_packet/rte_eth_af_packet.c
index 2f87553..77536e8 100644
--- a/drivers/net/af_packet/rte_eth_af_packet.c
+++ b/drivers/net/af_packet/rte_eth_af_packet.c
@@ -923,8 +923,9 @@ rte_eth_from_packet(const char *name,
 }
 
 static int
-rte_pmd_af_packet_probe(const char *name, const char *params)
+rte_pmd_af_packet_probe(struct rte_vdev_device *dev)
 {
+   const char *name = rte_vdev_device_name(dev);
unsigned numa_node;
int ret = 0;
struct rte_kvargs *kvlist;
@@ -934,7 +935,7 @@ rte_pmd_af_packet_probe(const char *name, const char 
*params)
 
numa_node = rte_socket_id();
 
-   kvlist = rte_kvargs_parse(params, valid_arguments);
+   kvlist = rte_kvargs_parse(rte_vdev_device_args(dev), valid_arguments);
if (kvlist == NULL) {
ret = -1;
goto exit;
@@ -961,7 +962,7 @@ rte_pmd_af_packet_probe(const char *name, const char 
*params)
 }
 
 static int
-rte_pmd_af_packet_remove(const char *name)
+rte_pmd_af_packet_remove(struct rte_vdev_device *dev)
 {
struct rte_eth_dev *eth_dev = NULL;
struct pmd_internals *internals;
@@ -970,11 +971,11 @@ rte_pmd_af_packet_remove(const char *name)
RTE_LOG(INFO, PMD, "Closing AF_PACKET ethdev on numa socket %u\n",
rte_socket_id());
 
-   if (name == NULL)
+   if (dev == NULL)
return -1;
 
/* find the ethdev entry */
-   eth_dev = rte_eth_dev_allocated(name);
+   eth_dev = rte_eth_dev_allocated(rte_vdev_device_name(dev));
if (eth_dev == NULL)
return -1;
 
diff --git a/drivers/net/bonding/rte_eth_bond_pmd.c 
b/drivers/net/bonding/rte_eth_bond_pmd.c
index f3ac9e2..6c03920 100644
--- a/drivers/net/bonding/rte_eth_bond_pmd.c
+++ b/drivers/net/bonding/rte_eth_bond_pmd.c
@@ -2230,16 +2230,19 @@ const struct eth_dev_ops default_dev_ops = {
 };
 
 static int
-bond_probe(const char *name, const char *params)
+bond_probe(struct rte_vdev_device *dev)
 {
+   const char *name;
struct bond_dev_private *internals;
struct rte_kvargs *kvlist;
uint8_t bonding_mode, socket_id;
int  arg_count, port_id;
 
+   name = rte_vdev_d

[dpdk-dev] [PATCH v3 07/10] eal: add virtual device name helper function

2017-02-25 Thread Jan Blunck
This adds the rte_vdev_device_name() helper function to retrieve the
rte_vdev_device name which makes moving the name of the low-level
device into struct rte_device easier in the future.

Signed-off-by: Jan Blunck 
Acked-by: Shreyansh Jain 
---
 lib/librte_eal/common/include/rte_vdev.h | 9 +
 1 file changed, 9 insertions(+)

diff --git a/lib/librte_eal/common/include/rte_vdev.h 
b/lib/librte_eal/common/include/rte_vdev.h
index 8f98372..abdefab 100644
--- a/lib/librte_eal/common/include/rte_vdev.h
+++ b/lib/librte_eal/common/include/rte_vdev.h
@@ -39,12 +39,21 @@ extern "C" {
 
 #include 
 #include 
+#include 
 
 struct rte_vdev_device {
TAILQ_ENTRY(rte_vdev_device) next;  /**< Next attached vdev */
struct rte_device device;   /**< Inherit core device */
 };
 
+static inline const char *
+rte_vdev_device_name(const struct rte_vdev_device *dev)
+{
+   if (dev && dev->device.devargs)
+   return dev->device.devargs->virt.drv_name;
+   return NULL;
+}
+
 /** Double linked list of virtual device drivers. */
 TAILQ_HEAD(vdev_driver_list, rte_vdev_driver);
 
-- 
2.7.4



[dpdk-dev] [PATCH v3 08/10] eal: add virtual device arguments helper function

2017-02-25 Thread Jan Blunck
This adds the rte_vdev_device_args() helper function to prepare for
changing the virtual drivers probe() functions take a rte_vdev_device
pointer instead of the name+args strings.

Signed-off-by: Jan Blunck 
---
 lib/librte_eal/common/include/rte_vdev.h | 8 
 1 file changed, 8 insertions(+)

diff --git a/lib/librte_eal/common/include/rte_vdev.h 
b/lib/librte_eal/common/include/rte_vdev.h
index abdefab..81f6beb 100644
--- a/lib/librte_eal/common/include/rte_vdev.h
+++ b/lib/librte_eal/common/include/rte_vdev.h
@@ -54,6 +54,14 @@ rte_vdev_device_name(const struct rte_vdev_device *dev)
return NULL;
 }
 
+static inline const char *
+rte_vdev_device_args(const struct rte_vdev_device *dev)
+{
+   if (dev && dev->device.devargs)
+   return dev->device.devargs->args;
+   return "";
+}
+
 /** Double linked list of virtual device drivers. */
 TAILQ_HEAD(vdev_driver_list, rte_vdev_driver);
 
-- 
2.7.4



Re: [dpdk-dev] [PATCH 0/4] New crypto algorithm string parser API

2017-02-25 Thread Hemant Agrawal

On 2/24/2017 9:11 PM, Trahe, Fiona wrote:




-Original Message-
From: dev [mailto:dev-boun...@dpdk.org] On Behalf Of Pablo de Lara
Sent: Thursday, February 23, 2017 12:34 PM
To: Doherty, Declan 
Cc: dev@dpdk.org; De Lara Guarch, Pablo 
Subject: [dpdk-dev] [PATCH 0/4] New crypto algorithm string parser API

Last release, an array with strings for the supported algorithms
by the cryptodev library was added and used in the crypto-perf app.

This patchset creates a new API to parse strings from the user,
to select the desired algorithm (using the array above),
which can be used by any application, making it consistent across
all the applications (now, L2fwd-crypto and crypto-perf apps are
using different strings).

Pablo de Lara (4):
  cryptodev: add missing algorithm strings
  cryptodev: add algorithm string parsers
  app/crypto-perf: use cryptodev algorithm parser
  examples/l2fwd-crypto: use cryptodev algorithm parser

 app/test-crypto-perf/cperf_options_parsing.c   | 206 ++---
 examples/l2fwd-crypto/main.c   |  85 ++
 lib/librte_cryptodev/rte_cryptodev.c   |  38 +
 lib/librte_cryptodev/rte_cryptodev.h   |  30 
 lib/librte_cryptodev/rte_cryptodev_version.map |   8 +
 5 files changed, 100 insertions(+), 267 deletions(-)

--
2.7.4

Acked-by: Fiona Trahe 


Acked-by: Hemant Agrawal 



[dpdk-dev] checkpatch.pl inconsistent results

2017-02-25 Thread Legacy, Allain
Hi,
I sent a patchset to the to the mailing list last night for which I received 
several coding style warnings.   Having discovered that I was using an older 
version of checkpatch.pl I downloaded the latest and set out to fix the 
warnings.  The tool is flagging the usage of PRIx64 and PRIu64 in debug logs as 
camelcase warnings.  I am unsure how to get around this.  Looking at other 
recent patches in patchwork I see that other patches use these macros without 
being flagged as errors.  

I thought perhaps that my version of checkpath.pl was newer because I just 
downloaded it so I ran it on one of the other patchwork patches to validate my 
results.   The results that I get are a bit confusing.  Running checkpatches.sh 
on this patch (http://dpdk.org/dev/patchwork/patch/19766/) reports no errors, 
warnings, or checks while this one (http://dpdk.org/dev/patchwork/patch/20742/) 
flags two different kinds of errors related to the usage of PRIx64.  It 
complains about the camelcase aspect of it, and it also complains about the 
lack of space between the PRIx64 and the concatenated strings at either side.  

Can anyone shed some light on why this is happening?

Regards,
Allain


Allain Legacy, Software Developer
direct 613.270.2279  fax 613.492.7870 skype allain.legacy
 





[dpdk-dev] [PATCH v4 00/26] linux/eal: Remove most causes of panic on init

2017-02-25 Thread Aaron Conole
In many cases, it's enough to simply let the application know that the
call to initialize DPDK has failed.  A complete halt can then be
decided by the application based on error returned (and the app could
even attempt a possible re-attempt after some corrective action by the
user or application).

Changes ->v2:
- Audited all "RTE_LOG (" calls that were introduced, and converted
  to "RTE_LOG("
- Added some fprintf(stderr, "") lines to indicate errors before logging
  is initialized
- Removed assignments to errno.
- Changed patch 14/25 to reflect EFAULT, and document in 25/25

Changes ->v3:
- Checkpatch issues in patches 3 (spelling mistake), 9 (issue with leading
  spaces), and 19 (braces around single line statement if-condition)

Changes ->v4:
- Error text cleanup.
- Add a new check around rte_bus_scan(), added during the development of
  this series.

I kept the rte_errno reflection, since this is control-path code and the
init function returns a sentinel value of -1.

Aaron Conole (26):
  eal: CPU init will no longer panic
  eal: return error instead of panic for cpu init
  eal: No panic on hugepages info init
  eal: do not panic on failed hugepage query
  eal: failure to parse args returns error
  eal-common: introduce a way to query cpu support
  eal: Signal error when CPU isn't supported
  eal: do not panic on memzone initialization fails
  eal: set errno when exiting for already called
  eal: Do not panic on log failures
  eal: Do not panic on pci-probe
  eal: do not panic on vfio failure
  eal: do not panic on memory init
  eal: do not panic on tailq init
  eal: do not panic on alarm init
  eal: convert timer_init not to call panic
  eal: change the private pipe call to reflect errno
  eal: Do not panic on interrupt thread init
  eal: do not error if plugins fail to init
  eal_pci: Continue probing even on failures
  eal: do not panic on failed PCI probe
  eal_common_dev: continue initializing vdevs
  eal: do not panic (or abort) if vdev init fails
  eal: do not panic when bus probe fails
  eal: do not panic on failed bus scan
  rte_eal_init: add info about rte_errno codes

 lib/librte_eal/common/eal_common_cpuflags.c|  13 +-
 lib/librte_eal/common/eal_common_dev.c |   5 +-
 lib/librte_eal/common/eal_common_lcore.c   |   7 +-
 lib/librte_eal/common/eal_common_pci.c |  15 ++-
 lib/librte_eal/common/eal_common_tailqs.c  |   3 +-
 .../common/include/generic/rte_cpuflags.h  |   9 ++
 lib/librte_eal/common/include/rte_eal.h|  27 -
 lib/librte_eal/linuxapp/eal/eal.c  | 131 +++--
 lib/librte_eal/linuxapp/eal/eal_hugepage_info.c|   6 +-
 lib/librte_eal/linuxapp/eal/eal_interrupts.c   |   5 +-
 10 files changed, 169 insertions(+), 52 deletions(-)

-- 
2.9.3



[dpdk-dev] [PATCH v4 03/26] eal: No panic on hugepages info init

2017-02-25 Thread Aaron Conole
When attempting to scan hugepages, signal to the eal.c that an error has
occurred, rather than performing a panic.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal_hugepage_info.c | 6 --
 1 file changed, 4 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal_hugepage_info.c 
b/lib/librte_eal/linuxapp/eal/eal_hugepage_info.c
index 18858e2..4d47eaf 100644
--- a/lib/librte_eal/linuxapp/eal/eal_hugepage_info.c
+++ b/lib/librte_eal/linuxapp/eal/eal_hugepage_info.c
@@ -283,9 +283,11 @@ eal_hugepage_info_init(void)
struct dirent *dirent;
 
dir = opendir(sys_dir_path);
-   if (dir == NULL)
-   rte_panic("Cannot open directory %s to read system hugepage "
+   if (dir == NULL) {
+   RTE_LOG(ERR, EAL, "Cannot open directory %s to read system 
hugepage "
  "info\n", sys_dir_path);
+   return -1;
+   }
 
for (dirent = readdir(dir); dirent != NULL; dirent = readdir(dir)) {
struct hugepage_info *hpi;
-- 
2.9.3



[dpdk-dev] [PATCH v4 01/26] eal: CPU init will no longer panic

2017-02-25 Thread Aaron Conole
After this change, the EAL CPU NUMA node resolution step can no longer
emit an rte_panic.  This aligns with the code in rte_eal_init, which
expects failures to return an error code.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/common/eal_common_lcore.c | 7 ---
 1 file changed, 4 insertions(+), 3 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_lcore.c 
b/lib/librte_eal/common/eal_common_lcore.c
index 2cd4132..84fa0cb 100644
--- a/lib/librte_eal/common/eal_common_lcore.c
+++ b/lib/librte_eal/common/eal_common_lcore.c
@@ -83,16 +83,17 @@ rte_eal_cpu_init(void)
config->lcore_role[lcore_id] = ROLE_RTE;
lcore_config[lcore_id].core_id = eal_cpu_core_id(lcore_id);
lcore_config[lcore_id].socket_id = eal_cpu_socket_id(lcore_id);
-   if (lcore_config[lcore_id].socket_id >= RTE_MAX_NUMA_NODES)
+   if (lcore_config[lcore_id].socket_id >= RTE_MAX_NUMA_NODES) {
 #ifdef RTE_EAL_ALLOW_INV_SOCKET_ID
lcore_config[lcore_id].socket_id = 0;
 #else
-   rte_panic("Socket ID (%u) is greater than "
+   RTE_LOG(ERR, EAL, "Socket ID (%u) is greater than "
"RTE_MAX_NUMA_NODES (%d)\n",
lcore_config[lcore_id].socket_id,
RTE_MAX_NUMA_NODES);
+   return -1;
 #endif
-
+   }
RTE_LOG(DEBUG, EAL, "Detected lcore %u as "
"core %u on socket %u\n",
lcore_id, lcore_config[lcore_id].core_id,
-- 
2.9.3



[dpdk-dev] [PATCH v4 02/26] eal: return error instead of panic for cpu init

2017-02-25 Thread Aaron Conole
There may be no way to gracefully recover, but the application
should be notified that a failure happened, rather than completely
aborting.  This allows the user to proceed with a "slow-path" type
solution.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 13 +++--
 1 file changed, 11 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index bf6b818..5023d0d 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -740,6 +740,12 @@ static int rte_eal_vfio_setup(void)
 }
 #endif
 
+static void rte_eal_init_alert(const char *msg)
+{
+fprintf(stderr, "EAL: FATAL: %s\n", msg);
+RTE_LOG(ERR, EAL, "%s\n", msg);
+}
+
 /* Launch threads, called at application init(). */
 int
 rte_eal_init(int argc, char **argv)
@@ -767,8 +773,11 @@ rte_eal_init(int argc, char **argv)
/* set log level as early as possible */
rte_set_log_level(internal_config.log_level);
 
-   if (rte_eal_cpu_init() < 0)
-   rte_panic("Cannot detect lcores\n");
+   if (rte_eal_cpu_init() < 0) {
+   rte_eal_init_alert("Cannot detect lcores.");
+   rte_errno = ENOTSUP;
+   return -1;
+   }
 
fctret = eal_parse_args(argc, argv);
if (fctret < 0)
-- 
2.9.3



[dpdk-dev] [PATCH v4 04/26] eal: do not panic on failed hugepage query

2017-02-25 Thread Aaron Conole
If we fail to acquire hugepage information, simply signal an error to
the application.  This clears the run_once counter, allowing the user or
application to take a corrective action and retry.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 8 ++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 5023d0d..c76ed94 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -786,8 +786,12 @@ rte_eal_init(int argc, char **argv)
if (internal_config.no_hugetlbfs == 0 &&
internal_config.process_type != RTE_PROC_SECONDARY &&
internal_config.xen_dom0_support == 0 &&
-   eal_hugepage_info_init() < 0)
-   rte_panic("Cannot get hugepage information\n");
+   eal_hugepage_info_init() < 0) {
+   rte_eal_init_alert("Cannot get hugepage information.");
+   rte_errno = EACCES;
+   rte_atomic32_clear(&run_once);
+   return -1;
+   }
 
if (internal_config.memory == 0 && internal_config.force_sockets == 0) {
if (internal_config.no_hugetlbfs)
-- 
2.9.3



[dpdk-dev] [PATCH v4 05/26] eal: failure to parse args returns error

2017-02-25 Thread Aaron Conole
It's possible that the application could take a corrective action here,
and either prompt the user for different arguments, or at least perform
a better logging.  Exiting this early prevents any useful information
gathering from the application layer.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 8 ++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index c76ed94..0f682e1 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -780,8 +780,12 @@ rte_eal_init(int argc, char **argv)
}
 
fctret = eal_parse_args(argc, argv);
-   if (fctret < 0)
-   exit(1);
+   if (fctret < 0) {
+   rte_eal_init_alert("Invalid 'command line' arguments.");
+   rte_errno = EINVAL;
+   rte_atomic32_clear(&run_once);
+   return -1;
+   }
 
if (internal_config.no_hugetlbfs == 0 &&
internal_config.process_type != RTE_PROC_SECONDARY &&
-- 
2.9.3



[dpdk-dev] [PATCH v4 09/26] eal: set errno when exiting for already called

2017-02-25 Thread Aaron Conole
Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 5 -
 1 file changed, 4 insertions(+), 1 deletion(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 0005ebc..beb786e 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -764,8 +764,11 @@ rte_eal_init(int argc, char **argv)
return -1;
}
 
-   if (!rte_atomic32_test_and_set(&run_once))
+   if (!rte_atomic32_test_and_set(&run_once)) {
+   rte_eal_init_alert("already called initialization.");
+   rte_errno = EALREADY;
return -1;
+   }
 
logid = strrchr(argv[0], '/');
logid = strdup(logid ? logid + 1: argv[0]);
-- 
2.9.3



[dpdk-dev] [PATCH v4 07/26] eal: Signal error when CPU isn't supported

2017-02-25 Thread Aaron Conole
It's now possible to gracefully exit the application, or for
applications which support non-dpdk datapaths working in concert with
DPDK datapaths, there no longer is the possibility of exiting for
unsupported CPUs.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 6 +-
 1 file changed, 5 insertions(+), 1 deletion(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 0f682e1..0adaaa2 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -758,7 +758,11 @@ rte_eal_init(int argc, char **argv)
char thread_name[RTE_MAX_THREAD_NAME_LEN];
 
/* checks if the machine is adequate */
-   rte_cpu_check_supported();
+   if (!rte_cpu_is_supported()) {
+   rte_eal_init_alert("unsupported cpu type.");
+   rte_errno = ENOTSUP;
+   return -1;
+   }
 
if (!rte_atomic32_test_and_set(&run_once))
return -1;
-- 
2.9.3



[dpdk-dev] [PATCH v4 08/26] eal: do not panic on memzone initialization fails

2017-02-25 Thread Aaron Conole
When memzone initialization fails, report the error to the calling
application rather than panic().  Without a good way of detaching /
releasing hugepages, at this point the application will have to restart.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 0adaaa2..0005ebc 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -838,8 +838,11 @@ rte_eal_init(int argc, char **argv)
/* the directories are locked during eal_hugepage_info_init */
eal_hugedirs_unlock();
 
-   if (rte_eal_memzone_init() < 0)
-   rte_panic("Cannot init memzone\n");
+   if (rte_eal_memzone_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init memzone\n");
+   rte_errno = ENODEV;
+   return -1;
+   }
 
if (rte_eal_tailqs_init() < 0)
rte_panic("Cannot init tail queues for objects\n");
-- 
2.9.3



[dpdk-dev] [PATCH v4 06/26] eal-common: introduce a way to query cpu support

2017-02-25 Thread Aaron Conole
This adds a new API to check for the eal cpu versions.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/common/eal_common_cpuflags.c  | 13 +++--
 lib/librte_eal/common/include/generic/rte_cpuflags.h |  9 +
 2 files changed, 20 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_cpuflags.c 
b/lib/librte_eal/common/eal_common_cpuflags.c
index b5f76f7..2c2127b 100644
--- a/lib/librte_eal/common/eal_common_cpuflags.c
+++ b/lib/librte_eal/common/eal_common_cpuflags.c
@@ -43,6 +43,13 @@
 void
 rte_cpu_check_supported(void)
 {
+   if (!rte_cpu_is_supported())
+   exit(1);
+}
+
+bool
+rte_cpu_is_supported(void)
+{
/* This is generated at compile-time by the build system */
static const enum rte_cpu_flag_t compile_time_flags[] = {
RTE_COMPILE_TIME_CPUFLAGS
@@ -57,14 +64,16 @@ rte_cpu_check_supported(void)
fprintf(stderr,
"ERROR: CPU feature flag lookup failed with 
error %d\n",
ret);
-   exit(1);
+   return false;
}
if (!ret) {
fprintf(stderr,
"ERROR: This system does not support \"%s\".\n"
"Please check that RTE_MACHINE is set 
correctly.\n",
rte_cpu_get_flag_name(compile_time_flags[i]));
-   exit(1);
+   return false;
}
}
+
+   return true;
 }
diff --git a/lib/librte_eal/common/include/generic/rte_cpuflags.h 
b/lib/librte_eal/common/include/generic/rte_cpuflags.h
index 71321f3..e4342ad 100644
--- a/lib/librte_eal/common/include/generic/rte_cpuflags.h
+++ b/lib/librte_eal/common/include/generic/rte_cpuflags.h
@@ -40,6 +40,7 @@
  */
 
 #include 
+#include 
 
 /**
  * Enumeration of all CPU features supported
@@ -82,4 +83,12 @@ rte_cpu_get_flag_enabled(enum rte_cpu_flag_t feature);
 void
 rte_cpu_check_supported(void);
 
+/**
+ * This function checks that the currently used CPU supports the CPU features
+ * that were specified at compile time. It is called automatically within the
+ * EAL, so does not need to be used by applications.  This version returns a
+ * result so that decisions may be made (for instance, graceful shutdowns).
+ */
+bool
+rte_cpu_is_supported(void);
 #endif /* _RTE_CPUFLAGS_H_ */
-- 
2.9.3



[dpdk-dev] [PATCH v4 10/26] eal: Do not panic on log failures

2017-02-25 Thread Aaron Conole
When log initialization fails, it's generally because the fopencookie
failed.  While this is rare in practice, it could happen, and it is
likely because of memory pressure.  So, flag the error, and allow the
user to retry.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 8 ++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index beb786e..25f8ae8 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -824,8 +824,12 @@ rte_eal_init(int argc, char **argv)
 
rte_config_init();
 
-   if (rte_eal_log_init(logid, internal_config.syslog_facility) < 0)
-   rte_panic("Cannot init logs\n");
+   if (rte_eal_log_init(logid, internal_config.syslog_facility) < 0) {
+   rte_eal_init_alert("Cannot init logging.");
+   rte_errno = ENOMEM;
+   rte_atomic32_clear(&run_once);
+   return -1;
+   }
 
if (rte_eal_pci_init() < 0)
rte_panic("Cannot init PCI\n");
-- 
2.9.3



[dpdk-dev] [PATCH v4 13/26] eal: do not panic on memory init

2017-02-25 Thread Aaron Conole
This can only happen when access to hugepages (either as primary or
secondary process) fails (and that is usually permissions).  Since the
manner of failure is not reversible, we cannot allow retry.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 0e7e8c8..30d6a7d 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -847,8 +847,11 @@ rte_eal_init(int argc, char **argv)
}
 #endif
 
-   if (rte_eal_memory_init() < 0)
-   rte_panic("Cannot init memory\n");
+   if (rte_eal_memory_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init memory\n");
+   rte_errno = ENOMEM;
+   return -1;
+   }
 
/* the directories are locked during eal_hugepage_info_init */
eal_hugedirs_unlock();
-- 
2.9.3



[dpdk-dev] [PATCH v4 12/26] eal: do not panic on vfio failure

2017-02-25 Thread Aaron Conole
Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 8 ++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 5534b4b..0e7e8c8 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -839,8 +839,12 @@ rte_eal_init(int argc, char **argv)
}
 
 #ifdef VFIO_PRESENT
-   if (rte_eal_vfio_setup() < 0)
-   rte_panic("Cannot init VFIO\n");
+   if (rte_eal_vfio_setup() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init VFIO\n");
+   rte_errno = EAGAIN;
+   rte_atomic32_clear(&run_once);
+   return -1;
+   }
 #endif
 
if (rte_eal_memory_init() < 0)
-- 
2.9.3



[dpdk-dev] [PATCH v4 15/26] eal: do not panic on alarm init

2017-02-25 Thread Aaron Conole
rte_eal_alarm_init() call uses the linux timerfd framework to create a
poll()-able timer using standard posix file operations.  This could fail
for a few reasons given in the man-pages, but many could be
corrected by the user application.  No need to panic.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 8 ++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 008e804..aa64f69 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -61,6 +61,7 @@
 #include 
 #include 
 #include 
+#include 
 #include 
 #include 
 #include 
@@ -868,8 +869,11 @@ rte_eal_init(int argc, char **argv)
return -1;
}
 
-   if (rte_eal_alarm_init() < 0)
-   rte_panic("Cannot init interrupt-handling thread\n");
+   if (rte_eal_alarm_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init interrupt-handling thread\n");
+   /* rte_eal_alarm_init sets rte_errno on failure. */
+   return -1;
+   }
 
if (rte_eal_timer_init() < 0)
rte_panic("Cannot init HPET or TSC timers\n");
-- 
2.9.3



[dpdk-dev] [PATCH v4 14/26] eal: do not panic on tailq init

2017-02-25 Thread Aaron Conole
There are some theoretical racy conditions in the system that _could_
cause early tailq init to fail;  however, no need to panic the
application.  While it can't continue using DPDK, it could make better
alerts to the user.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/common/eal_common_tailqs.c | 3 +--
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 2 files changed, 6 insertions(+), 4 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_tailqs.c 
b/lib/librte_eal/common/eal_common_tailqs.c
index bb08ec8..4f69828 100644
--- a/lib/librte_eal/common/eal_common_tailqs.c
+++ b/lib/librte_eal/common/eal_common_tailqs.c
@@ -188,8 +188,7 @@ rte_eal_tailqs_init(void)
if (t->head == NULL) {
RTE_LOG(ERR, EAL,
"Cannot initialize tailq: %s\n", t->name);
-   /* no need to TAILQ_REMOVE, we are going to panic in
-* rte_eal_init() */
+   /* TAILQ_REMOVE not needed, error is already fatal */
goto fail;
}
}
diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 30d6a7d..008e804 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -862,8 +862,11 @@ rte_eal_init(int argc, char **argv)
return -1;
}
 
-   if (rte_eal_tailqs_init() < 0)
-   rte_panic("Cannot init tail queues for objects\n");
+   if (rte_eal_tailqs_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init tail queues for objects\n");
+   rte_errno = EFAULT;
+   return -1;
+   }
 
if (rte_eal_alarm_init() < 0)
rte_panic("Cannot init interrupt-handling thread\n");
-- 
2.9.3



[dpdk-dev] [PATCH v4 16/26] eal: convert timer_init not to call panic

2017-02-25 Thread Aaron Conole
After code inspection, there is no way for eal_timer_init() to fail.  It
simply returns 0 in all cases.  As such, this test could either go-away
or stay here as 'future-proofing'.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index aa64f69..bf20245 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -875,8 +875,11 @@ rte_eal_init(int argc, char **argv)
return -1;
}
 
-   if (rte_eal_timer_init() < 0)
-   rte_panic("Cannot init HPET or TSC timers\n");
+   if (rte_eal_timer_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init HPET or TSC timers\n");
+   rte_errno = ENOTSUP;
+   return -1;
+   }
 
eal_check_mem_on_local_socket();
 
-- 
2.9.3



[dpdk-dev] [PATCH v4 11/26] eal: Do not panic on pci-probe

2017-02-25 Thread Aaron Conole
This will usually be an issue because of permissions.  However, it could
also be caused by OOM.  In either case, errno will contain the
underlying cause.  It is safe to re-init the system here, so allow the
application to take corrective action and reinit.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 8 ++--
 1 file changed, 6 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 25f8ae8..5534b4b 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -831,8 +831,12 @@ rte_eal_init(int argc, char **argv)
return -1;
}
 
-   if (rte_eal_pci_init() < 0)
-   rte_panic("Cannot init PCI\n");
+   if (rte_eal_pci_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init PCI\n");
+   rte_errno = EUNATCH;
+   rte_atomic32_clear(&run_once);
+   return -1;
+   }
 
 #ifdef VFIO_PRESENT
if (rte_eal_vfio_setup() < 0)
-- 
2.9.3



[dpdk-dev] [PATCH v4 18/26] eal: Do not panic on interrupt thread init

2017-02-25 Thread Aaron Conole
When initializing the interrupt thread, there are a number of possible
reasons for failure - some of which are correctable by the application.
Do not panic() needlessly, and give the application a change to reflect
this information to the user.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 6 --
 1 file changed, 4 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index bf20245..02b075c 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -894,8 +894,10 @@ rte_eal_init(int argc, char **argv)
rte_config.master_lcore, (int)thread_id, cpuset,
ret == 0 ? "" : "...");
 
-   if (rte_eal_intr_init() < 0)
-   rte_panic("Cannot init interrupt-handling thread\n");
+   if (rte_eal_intr_init() < 0) {
+   RTE_LOG(ERR, EAL, "Cannot init interrupt-handling thread\n");
+   return -1;
+   }
 
if (rte_bus_scan())
rte_panic("Cannot scan the buses for devices\n");
-- 
2.9.3



[dpdk-dev] [PATCH v4 19/26] eal: do not error if plugins fail to init

2017-02-25 Thread Aaron Conole
Plugins are useful and important.  However, it seems crazy to abort
everything just because they don't initialize properly.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 02b075c..2069b6f 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -884,7 +884,7 @@ rte_eal_init(int argc, char **argv)
eal_check_mem_on_local_socket();
 
if (eal_plugins_init() < 0)
-   rte_panic("Cannot init plugins\n");
+   RTE_LOG(ERR, EAL, "Cannot init plugins\n");
 
eal_thread_init_master(rte_config.master_lcore);
 
-- 
2.9.3



[dpdk-dev] [PATCH v4 17/26] eal: change the private pipe call to reflect errno

2017-02-25 Thread Aaron Conole
There could be some confusion as to why the call failed - this change
will always reflect the value of the error in rte_error.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal_interrupts.c | 5 -
 1 file changed, 4 insertions(+), 1 deletion(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal_interrupts.c 
b/lib/librte_eal/linuxapp/eal/eal_interrupts.c
index 92a19cb..5bb833e 100644
--- a/lib/librte_eal/linuxapp/eal/eal_interrupts.c
+++ b/lib/librte_eal/linuxapp/eal/eal_interrupts.c
@@ -898,13 +898,16 @@ rte_eal_intr_init(void)
 * create a pipe which will be waited by epoll and notified to
 * rebuild the wait list of epoll.
 */
-   if (pipe(intr_pipe.pipefd) < 0)
+   if (pipe(intr_pipe.pipefd) < 0) {
+   rte_errno = errno;
return -1;
+   }
 
/* create the host thread to wait/handle the interrupt */
ret = pthread_create(&intr_thread, NULL,
eal_intr_thread_main, NULL);
if (ret != 0) {
+   rte_errno = ret;
RTE_LOG(ERR, EAL,
"Failed to create thread for interrupt handling\n");
} else {
-- 
2.9.3



[dpdk-dev] [PATCH v4 20/26] eal_pci: Continue probing even on failures

2017-02-25 Thread Aaron Conole
Some devices may be inaccessible for a variety of reasons, or the
PCI-bus may be unavailable causing the whole thing to fail.  Still,
better to continue attempts at probes.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/common/eal_common_pci.c | 15 ++-
 1 file changed, 10 insertions(+), 5 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_pci.c 
b/lib/librte_eal/common/eal_common_pci.c
index 72547bd..9416190 100644
--- a/lib/librte_eal/common/eal_common_pci.c
+++ b/lib/librte_eal/common/eal_common_pci.c
@@ -69,6 +69,7 @@
 #include 
 #include 
 
+#include 
 #include 
 #include 
 #include 
@@ -416,6 +417,7 @@ rte_eal_pci_probe(void)
struct rte_pci_device *dev = NULL;
struct rte_devargs *devargs;
int probe_all = 0;
+   int ret_1 = 0;
int ret = 0;
 
if (rte_eal_devargs_type_count(RTE_DEVTYPE_WHITELISTED_PCI) == 0)
@@ -430,17 +432,20 @@ rte_eal_pci_probe(void)
 
/* probe all or only whitelisted devices */
if (probe_all)
-   ret = pci_probe_all_drivers(dev);
+   ret_1 = pci_probe_all_drivers(dev);
else if (devargs != NULL &&
devargs->type == RTE_DEVTYPE_WHITELISTED_PCI)
-   ret = pci_probe_all_drivers(dev);
-   if (ret < 0)
-   rte_exit(EXIT_FAILURE, "Requested device " PCI_PRI_FMT
+   ret_1 = pci_probe_all_drivers(dev);
+   if (ret_1 < 0) {
+   RTE_LOG(ERR, EAL, "Requested device " PCI_PRI_FMT
 " cannot be used\n", dev->addr.domain, 
dev->addr.bus,
 dev->addr.devid, dev->addr.function);
+   rte_errno = errno;
+   ret = 1;
+   }
}
 
-   return 0;
+   return -ret;
 }
 
 /* dump one device */
-- 
2.9.3



[dpdk-dev] [PATCH v4 21/26] eal: do not panic on failed PCI probe

2017-02-25 Thread Aaron Conole
It may even be possible to simply log the error and continue on letting
the user check the logs and restart the application when things are failed.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 2069b6f..481b53b 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -943,8 +943,11 @@ rte_eal_init(int argc, char **argv)
rte_panic("Cannot probe devices\n");
 
/* Probe & Initialize PCI devices */
-   if (rte_eal_pci_probe())
-   rte_panic("Cannot probe PCI\n");
+   if (rte_eal_pci_probe()) {
+   RTE_LOG(ERR, EAL, "Cannot probe PCI\n");
+   rte_errno = ENOTSUP;
+   return -1;
+   }
 
if (rte_eal_dev_init() < 0)
rte_panic("Cannot init pmd devices\n");
-- 
2.9.3



[dpdk-dev] [PATCH v4 23/26] eal: do not panic (or abort) if vdev init fails

2017-02-25 Thread Aaron Conole
Seems like it's possible to continue.  At least, the error is reflected
properly in the logs.  A user could then go and correct or investigate
the situation.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 481b53b..803cdd2 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -950,7 +950,7 @@ rte_eal_init(int argc, char **argv)
}
 
if (rte_eal_dev_init() < 0)
-   rte_panic("Cannot init pmd devices\n");
+   RTE_LOG(ERR, EAL, "Cannot init pmd devices\n");
 
rte_eal_mcfg_complete();
 
-- 
2.9.3



[dpdk-dev] [PATCH v4 22/26] eal_common_dev: continue initializing vdevs

2017-02-25 Thread Aaron Conole
Even if one vdev should fail, there's no need to prevent further
processing.  Log the error, and reflect it to the higher levels to
decide.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/common/eal_common_dev.c | 5 +++--
 1 file changed, 3 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/common/eal_common_dev.c 
b/lib/librte_eal/common/eal_common_dev.c
index 4f3b493..9889997 100644
--- a/lib/librte_eal/common/eal_common_dev.c
+++ b/lib/librte_eal/common/eal_common_dev.c
@@ -80,6 +80,7 @@ int
 rte_eal_dev_init(void)
 {
struct rte_devargs *devargs;
+   int ret = 0;
 
/*
 * Note that the dev_driver_list is populated here
@@ -97,11 +98,11 @@ rte_eal_dev_init(void)
devargs->args)) {
RTE_LOG(ERR, EAL, "failed to initialize %s device\n",
devargs->virt.drv_name);
-   return -1;
+   ret = -1;
}
}
 
-   return 0;
+   return ret;
 }
 
 int rte_eal_dev_attach(const char *name, const char *devargs)
-- 
2.9.3



[dpdk-dev] [PATCH v4 25/26] eal: do not panic on failed bus scan

2017-02-25 Thread Aaron Conole
For now, do an abort.  It's likely that even aborting the initialization
is premature in this case, as it may be possible to proceed even if one
bus or another is not available.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index c94ac9b..41050a0 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -899,8 +899,11 @@ rte_eal_init(int argc, char **argv)
return -1;
}
 
-   if (rte_bus_scan())
-   rte_panic("Cannot scan the buses for devices\n");
+   if (rte_bus_scan()) {
+   RTE_LOG(ERR, EAL, "Cannot scan the buses for devices\n");
+   rte_errno = ENODEV;
+   return -1;
+   }
 
RTE_LCORE_FOREACH_SLAVE(i) {
 
-- 
2.9.3



[dpdk-dev] [PATCH v4 24/26] eal: do not panic when bus probe fails

2017-02-25 Thread Aaron Conole
Signed-off-by: Aaron Conole 
---
 lib/librte_eal/linuxapp/eal/eal.c | 7 +--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/librte_eal/linuxapp/eal/eal.c 
b/lib/librte_eal/linuxapp/eal/eal.c
index 803cdd2..c94ac9b 100644
--- a/lib/librte_eal/linuxapp/eal/eal.c
+++ b/lib/librte_eal/linuxapp/eal/eal.c
@@ -939,8 +939,11 @@ rte_eal_init(int argc, char **argv)
rte_eal_mp_wait_lcore();
 
/* Probe all the buses and devices/drivers on them */
-   if (rte_bus_probe())
-   rte_panic("Cannot probe devices\n");
+   if (rte_bus_probe()) {
+   RTE_LOG(ERR, EAL, "Cannot probe devices\n");
+   rte_errno = ENOTSUP;
+   return -1;
+   }
 
/* Probe & Initialize PCI devices */
if (rte_eal_pci_probe()) {
-- 
2.9.3



[dpdk-dev] [PATCH v4 26/26] rte_eal_init: add info about rte_errno codes

2017-02-25 Thread Aaron Conole
The rte_eal_init function will now pass failure reason hints to the
application.  To help app developers deciper this, add some brief
information about what the codes are indicating.

Signed-off-by: Aaron Conole 
---
 lib/librte_eal/common/include/rte_eal.h | 27 ++-
 1 file changed, 26 insertions(+), 1 deletion(-)

diff --git a/lib/librte_eal/common/include/rte_eal.h 
b/lib/librte_eal/common/include/rte_eal.h
index 03fee50..9251244 100644
--- a/lib/librte_eal/common/include/rte_eal.h
+++ b/lib/librte_eal/common/include/rte_eal.h
@@ -159,7 +159,32 @@ int rte_eal_iopl_init(void);
  * function call and should not be further interpreted by the
  * application.  The EAL does not take any ownership of the memory used
  * for either the argv array, or its members.
- *   - On failure, a negative error value.
+ *   - On failure, -1 and rte_errno is set to a value indicating the cause
+ * for failure.  In some instances, the application will need to be
+ * restarted as part of clearing the issue.
+ *
+ *   Error codes returned via rte_errno:
+ * EACCES indicates a permissions issue.
+ *
+ * EAGAIN indicates either a bus or system resource was not available,
+ *setup may be attempted again.
+ *
+ * EALREADY indicates that the rte_eal_init function has already been
+ *  called, and cannot be called again.
+ *
+ * EFAULT indicates the tailq configuration name was not found in
+ *memory configuration.
+ *
+ * EINVAL indicates invalid parameters were passed as argv/argc.
+ *
+ * ENOMEM indicates failure likely caused by an out-of-memory condition.
+ *
+ * ENODEV indicates memory setup issues.
+ *
+ * ENOTSUP indicates that the EAL cannot initialize on this system.
+ *
+ * EUNATCH indicates that the PCI bus is either not present, or is not
+ * readable by the eal.
  */
 int rte_eal_init(int argc, char **argv);
 
-- 
2.9.3



Re: [dpdk-dev] [PATCH 50/50] doc: added documents

2017-02-25 Thread Shijith Thotton


On Thursday 23 February 2017 08:05 PM, Ferruh Yigit wrote:
> On 2/21/2017 9:27 AM, Shijith Thotton wrote:
>> Added doc/guides/nics/liquidio.rst and
>> doc/guides/nics/features/liquidio.ini. Updated release notes.
>>
>> Signed-off-by: Shijith Thotton 
>> Signed-off-by: Jerin Jacob 
>> Signed-off-by: Derek Chickles 
>> ---
>>  doc/guides/nics/features/liquidio.ini  |  29 
>>  doc/guides/nics/index.rst  |   1 +
>>  doc/guides/nics/liquidio.rst   | 269 
>> +
>>  doc/guides/rel_notes/release_17_05.rst |   3 +
> 
> The web page that list the supported NICs [1] also needs to be updated,
> in a separate patch.
> 
> [1]
> http://dpdk.org/doc/nics
> 
>>  4 files changed, 302 insertions(+)
>>  create mode 100644 doc/guides/nics/features/liquidio.ini
>>  create mode 100644 doc/guides/nics/liquidio.rst
>>
>> diff --git a/doc/guides/nics/features/liquidio.ini 
>> b/doc/guides/nics/features/liquidio.ini
>> new file mode 100644
>> index 000..eac32ba
>> --- /dev/null
>> +++ b/doc/guides/nics/features/liquidio.ini
>> @@ -0,0 +1,29 @@
>> +;
>> +; Supported features of the 'LiquidIO' network poll mode driver.
>> +;
>> +; Refer to default.ini for the full list of available PMD features.
>> +;
>> +[Features]
>> +Link status  = Y
>> +Link status event= Y
>> +MTU update   = Y
>> +Jumbo frame  = Y
>> +Scattered Rx = Y
>> +Allmulticast mode= Y
>> +RSS hash = Y
>> +RSS key update   = Y
>> +RSS reta update  = Y
>> +SR-IOV   = Y
>> +VLAN filter  = Y
>> +CRC offload  = Y
>> +VLAN offload = P
>> +L3 checksum offload  = Y
>> +L4 checksum offload  = Y
>> +Inner L3 checksum= Y
>> +Inner L4 checksum= Y
>> +Basic stats  = Y
>> +Extended stats   = Y
>> +Linux UIO= Y
>> +Linux VFIO   = Y
>> +x86-64   = Y
> 
> No arm support?
> 
>> +Usage doc= Y
>> diff --git a/doc/guides/nics/index.rst b/doc/guides/nics/index.rst
>> index 5248625..37e6416 100644
>> --- a/doc/guides/nics/index.rst
>> +++ b/doc/guides/nics/index.rst
>> @@ -47,6 +47,7 @@ Network Interface Controller Drivers
>>  ixgbe
>>  intel_vf
>>  kni
>> +liquidio
>>  mlx4
>>  mlx5
>>  nfp
>> diff --git a/doc/guides/nics/liquidio.rst b/doc/guides/nics/liquidio.rst
>> new file mode 100644
>> index 000..4bf586b
>> --- /dev/null
>> +++ b/doc/guides/nics/liquidio.rst
>> @@ -0,0 +1,269 @@
> <...>
>> +LiquidIO VF Poll Mode Driver
>> +
>> +
>> +The LiquidIO VF PMD library(librte_pmd_lio) provides poll mode driver 
>> support for
>> +Cavium LiquidIO® II server adapter VFs. PF management and VF creation can be
>> +done using kernel driver.
> 
> Is it possible to provide a link here to point NIC documentation.
> 

Hi Ferruh,

Thanks a lot for the reviews. We are yet to completely verify ARM
support. Will make changes as suggested.

Shijith


[dpdk-dev] [PATCH 6/7] vmxnet3: introduce command to register memory region

2017-02-25 Thread Shrikrishna Khare
In vmxnet3 version 3, the emulation added support for the vmxnet3 driver
to communicate information about the memory regions the driver will use
for rx/tx buffers. The driver can also indicate which rx/tx queue the
memory region is applicable for. If this information is communicated
to the emulation, the emulation will always keep these memory regions
mapped, thereby avoiding the mapping/unmapping overhead for every packet.

Signed-off-by: Shrikrishna Khare 
Signed-off-by: Guolin Yang 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/base/vmxnet3_defs.h |  25 
 drivers/net/vmxnet3/vmxnet3_ethdev.c| 102 
 drivers/net/vmxnet3/vmxnet3_ethdev.h|   2 +
 3 files changed, 129 insertions(+)

diff --git a/drivers/net/vmxnet3/base/vmxnet3_defs.h 
b/drivers/net/vmxnet3/base/vmxnet3_defs.h
index c708498..bfa9622 100644
--- a/drivers/net/vmxnet3/base/vmxnet3_defs.h
+++ b/drivers/net/vmxnet3/base/vmxnet3_defs.h
@@ -111,6 +111,7 @@ typedef enum {
VMXNET3_CMD_ACTIVATE_VF,
VMXNET3_CMD_RESERVED3,
VMXNET3_CMD_RESERVED4,
+   VMXNET3_CMD_REGISTER_MEMREGS,
 
VMXNET3_CMD_FIRST_GET = 0xF00D,
VMXNET3_CMD_GET_QUEUE_STATUS = VMXNET3_CMD_FIRST_GET,
@@ -722,6 +723,30 @@ struct Vmxnet3_SetPolling {
 #include "vmware_pack_end.h"
 Vmxnet3_SetPolling;
 
+typedef
+#include "vmware_pack_begin.h"
+struct Vmxnet3_MemoryRegion {
+   __le64startPA;
+   __le32length;
+   __le16txQueueBits; /* bit n corresponding to tx queue n */
+   __le16rxQueueBits; /* bit n corresponding to rx queue n */
+}
+#include "vmware_pack_end.h"
+Vmxnet3_MemoryRegion;
+
+#define MAX_MEMORY_REGION_PER_QUEUE 16
+#define MAX_MEMORY_REGION_PER_DEVICE 256
+
+typedef
+#include "vmware_pack_begin.h"
+struct Vmxnet3_MemRegs {
+   __le16   numRegs;
+   __le16   pad[3];
+   Vmxnet3_MemoryRegion memRegs[1];
+}
+#include "vmware_pack_end.h"
+Vmxnet3_MemRegs;
+
 /*
  * If the command data <= 16 bytes, use the shared memory direcly.
  * Otherwise, use the variable length configuration descriptor.
diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.c 
b/drivers/net/vmxnet3/vmxnet3_ethdev.c
index 8591ce1..a4fc14d 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.c
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.c
@@ -489,6 +489,92 @@ vmxnet3_write_mac(struct vmxnet3_hw *hw, const uint8_t 
*addr)
 }
 
 static int
+vmxnet3_dev_setup_memreg(struct rte_eth_dev *dev)
+{
+   struct vmxnet3_hw *hw = dev->data->dev_private;
+   Vmxnet3_DriverShared *shared = hw->shared;
+   Vmxnet3_CmdInfo *cmdInfo;
+   struct rte_mempool *mp[VMXNET3_MAX_RX_QUEUES];
+   uint8_t index[VMXNET3_MAX_RX_QUEUES + VMXNET3_MAX_TX_QUEUES];
+   uint32_t num, i, j, size;
+
+   if (hw->memRegsPA == 0) {
+   const struct rte_memzone *mz;
+
+   size = sizeof(Vmxnet3_MemRegs) +
+   (VMXNET3_MAX_RX_QUEUES + VMXNET3_MAX_TX_QUEUES) *
+   sizeof(Vmxnet3_MemoryRegion);
+
+   mz = gpa_zone_reserve(dev, size, "memRegs", rte_socket_id(), 8,
+ 1);
+   if (mz == NULL) {
+   PMD_INIT_LOG(ERR, "ERROR: Creating memRegs zone");
+   return -ENOMEM;
+   }
+   memset(mz->addr, 0, mz->len);
+   hw->memRegs = mz->addr;
+   hw->memRegsPA = mz->phys_addr;
+   }
+
+   num = hw->num_rx_queues;
+
+   for (i = 0; i < num; i++) {
+   vmxnet3_rx_queue_t *rxq = dev->data->rx_queues[i];
+
+   mp[i] = rxq->mp;
+   index[i] = 1 << i;
+   }
+
+   /*
+* The same mempool could be used by multiple queues. In such a case,
+* remove duplicate mempool entries. Only one entry is kept with
+* bitmask indicating queues that are using this mempool.
+*/
+   for (i = 1; i < num; i++) {
+   for (j = 0; j < i; j++) {
+   if (mp[i] == mp[j]) {
+   mp[i] = NULL;
+   index[j] |= 1 << i;
+   break;
+   }
+   }
+   }
+
+   j = 0;
+   for (i = 0; i < num; i++) {
+   if (mp[i] == NULL) {
+   continue;
+   }
+
+   Vmxnet3_MemoryRegion *mr = &hw->memRegs->memRegs[j];
+
+   mr->startPA =
+   (uintptr_t)STAILQ_FIRST(&mp[i]->mem_list)->phys_addr;
+   mr->length = STAILQ_FIRST(&mp[i]->mem_list)->len <= INT32_MAX ?
+   STAILQ_FIRST(&mp[i]->mem_list)->len : INT32_MAX;
+   mr->txQueueBits = index[i];
+   mr->rxQueueBits = index[i];
+
+   PMD_INIT_LOG(INFO,
+"index: %u startPA: %lu  length: %u, rxBits: %x",
+j, mr->startPA, mr->le

[dpdk-dev] [PATCH 4/7] vmxnet3: add receive data ring support

2017-02-25 Thread Shrikrishna Khare
vmxnet3 driver preallocates buffers for receiving packets and posts the
buffers to the emulation. In order to deliver a received packet to the
guest, the emulation must map buffer(s) and copy the packet into it.

To avoid this memory mapping overhead, this patch introduces the receive
data ring - a set of small sized buffers that are always mapped by
the emulation. If a packet fits into the receive data ring buffer, the
emulation delivers the packet via the receive data ring (which must be
copied by the guest driver), or else the usual receive path is used.

Receive Data Ring buffer length is configurable via ethtool -G ethX rx-mini

Signed-off-by: Shrikrishna Khare 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/base/vmxnet3_defs.h | 12 ++--
 drivers/net/vmxnet3/vmxnet3_ethdev.c|  9 +
 drivers/net/vmxnet3/vmxnet3_ethdev.h|  1 +
 drivers/net/vmxnet3/vmxnet3_ring.h  | 20 
 drivers/net/vmxnet3/vmxnet3_rxtx.c  | 31 ++-
 5 files changed, 70 insertions(+), 3 deletions(-)

diff --git a/drivers/net/vmxnet3/base/vmxnet3_defs.h 
b/drivers/net/vmxnet3/base/vmxnet3_defs.h
index e201808..746d709 100644
--- a/drivers/net/vmxnet3/base/vmxnet3_defs.h
+++ b/drivers/net/vmxnet3/base/vmxnet3_defs.h
@@ -408,6 +408,10 @@ typedef union Vmxnet3_GenericDesc {
 #define VMXNET3_TXDATA_DESC_SIZE_ALIGN 64
 #define VMXNET3_TXDATA_DESC_SIZE_MASK  (VMXNET3_TXDATA_DESC_SIZE_ALIGN - 1)
 
+/* Rx Data Ring buffer size must be a multiple of 64 */
+#define VMXNET3_RXDATA_DESC_SIZE_ALIGN 64
+#define VMXNET3_RXDATA_DESC_SIZE_MASK  (VMXNET3_RXDATA_DESC_SIZE_ALIGN - 1)
+
 /* Max ring size */
 #define VMXNET3_TX_RING_MAX_SIZE   4096
 #define VMXNET3_TC_RING_MAX_SIZE   4096
@@ -417,6 +421,8 @@ typedef union Vmxnet3_GenericDesc {
 #define VMXNET3_TXDATA_DESC_MIN_SIZE 128
 #define VMXNET3_TXDATA_DESC_MAX_SIZE 2048
 
+#define VMXNET3_RXDATA_DESC_MAX_SIZE 2048
+
 /* a list of reasons for queue stop */
 
 #define VMXNET3_ERR_NOEOP0x8000  /* cannot find the EOP desc of a 
pkt */
@@ -529,12 +535,14 @@ struct Vmxnet3_RxQueueConf {
__le64rxRingBasePA[2];
__le64compRingBasePA;
__le64ddPA;/* driver data */
-   __le64reserved;
+   __le64rxDataRingBasePA;
__le32rxRingSize[2];   /* # of rx desc */
__le32compRingSize;/* # of rx comp desc */
__le32ddLen;   /* size of driver data */
uint8 intrIdx;
-   uint8 _pad[7];
+   uint8 _pad1[1];
+   __le16rxDataRingDescSize;  /* size of rx data ring buffer */
+   uint8 _pad2[4];
 }
 #include "vmware_pack_end.h"
 Vmxnet3_RxQueueConf;
diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.c 
b/drivers/net/vmxnet3/vmxnet3_ethdev.c
index 05b0864..8591ce1 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.c
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.c
@@ -341,6 +341,11 @@ eth_vmxnet3_dev_init(struct rte_eth_dev *eth_dev)
hw->txdata_desc_size = VMXNET3_VERSION_GE_3(hw) ?
eth_vmxnet3_txdata_get(hw) : sizeof(struct Vmxnet3_TxDataDesc);
 
+   hw->rxdata_desc_size = VMXNET3_VERSION_GE_3(hw) ?
+   VMXNET3_DEF_RXDATA_DESC_SIZE : 0;
+   RTE_ASSERT((hw->rx_data_desc_size & ~VMXNET3_RXDATA_DESC_SIZE_MASK) ==
+  hw->rx_data_desc_size);
+
return 0;
 }
 
@@ -551,6 +556,10 @@ vmxnet3_setup_driver_shared(struct rte_eth_dev *dev)
rqd->conf.rxRingSize[1]   = rxq->cmd_ring[1].size;
rqd->conf.compRingSize= rxq->comp_ring.size;
rqd->conf.intrIdx = rxq->comp_ring.intr_idx;
+   if (VMXNET3_VERSION_GE_3(hw)) {
+   rqd->conf.rxDataRingBasePA = rxq->data_ring.basePA;
+   rqd->conf.rxDataRingDescSize = rxq->data_desc_size;
+   }
rqd->status.stopped   = TRUE;
rqd->status.error = 0;
memset(&rqd->stats, 0, sizeof(rqd->stats));
diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.h 
b/drivers/net/vmxnet3/vmxnet3_ethdev.h
index f8e670d..1c1afc6 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.h
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.h
@@ -104,6 +104,7 @@ struct vmxnet3_hw {
uint8_t version;
 
uint16_t txdata_desc_size; /* tx data ring buffer size */
+   uint16_t rxdata_desc_size; /* rx data ring buffer size */
 
Vmxnet3_TxQueueDesc   *tqd_start;   /* start address of all tx 
queue desc */
Vmxnet3_RxQueueDesc   *rqd_start;   /* start address of all rx 
queue desc */
diff --git a/drivers/net/vmxnet3/vmxnet3_ring.h 
b/drivers/net/vmxnet3/vmxnet3_ring.h
index b27290f..8bc8239 100644
--- a/drivers/net/vmxnet3/vmxnet3_ring.h
+++ b/drivers/net/vmxnet3/vmxnet3_ring.h
@@ -42,6 +42,9 @@
 #define VMXNET3_DEF_TX_RING_SIZE 512
 #define VMXNET3_DEF_RX_RING_SIZE 128
 
+/* Default rx data ring desc size */
+#define VMXNET3_DEF_RXDATA_DESC_SIZE 128
+
 #define VMXNET3_SUCCESS 0
 #define VMXNE

[dpdk-dev] [PATCH 3/7] vmxnet3: allow variable length transmit data ring buffer

2017-02-25 Thread Shrikrishna Khare
vmxnet3 driver supports transmit data ring viz. a set of fixed size
buffers used by the driver to copy packet headers. Small packets that
fit these buffers are copied into these buffers entirely.

Currently this buffer size of fixed at 128 bytes. This patch extends
transmit data ring implementation to allow variable length transmit
data ring buffers. The length of the buffer is read from the emulation
during initialization.

Signed-off-by: Shrikrishna Khare 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/base/vmxnet3_defs.h | 14 --
 drivers/net/vmxnet3/vmxnet3_ethdev.c| 22 ++
 drivers/net/vmxnet3/vmxnet3_ethdev.h|  2 ++
 drivers/net/vmxnet3/vmxnet3_ring.h  |  1 +
 drivers/net/vmxnet3/vmxnet3_rxtx.c  | 14 +-
 5 files changed, 46 insertions(+), 7 deletions(-)

diff --git a/drivers/net/vmxnet3/base/vmxnet3_defs.h 
b/drivers/net/vmxnet3/base/vmxnet3_defs.h
index 7484b84..e201808 100644
--- a/drivers/net/vmxnet3/base/vmxnet3_defs.h
+++ b/drivers/net/vmxnet3/base/vmxnet3_defs.h
@@ -121,7 +121,8 @@ typedef enum {
VMXNET3_CMD_GET_DID_HI,
VMXNET3_CMD_GET_DEV_EXTRA_INFO,
VMXNET3_CMD_GET_CONF_INTR,
-   VMXNET3_CMD_GET_ADAPTIVE_RING_INFO
+   VMXNET3_CMD_GET_ADAPTIVE_RING_INFO,
+   VMXNET3_CMD_GET_TXDATA_DESC_SIZE
 } Vmxnet3_Cmd;
 
 /* Adaptive Ring Info Flags */
@@ -403,12 +404,19 @@ typedef union Vmxnet3_GenericDesc {
 #define VMXNET3_RING_SIZE_ALIGN 32
 #define VMXNET3_RING_SIZE_MASK  (VMXNET3_RING_SIZE_ALIGN - 1)
 
+/* Tx Data Ring buffer size must be a multiple of 64 */
+#define VMXNET3_TXDATA_DESC_SIZE_ALIGN 64
+#define VMXNET3_TXDATA_DESC_SIZE_MASK  (VMXNET3_TXDATA_DESC_SIZE_ALIGN - 1)
+
 /* Max ring size */
 #define VMXNET3_TX_RING_MAX_SIZE   4096
 #define VMXNET3_TC_RING_MAX_SIZE   4096
 #define VMXNET3_RX_RING_MAX_SIZE   4096
 #define VMXNET3_RC_RING_MAX_SIZE   8192
 
+#define VMXNET3_TXDATA_DESC_MIN_SIZE 128
+#define VMXNET3_TXDATA_DESC_MAX_SIZE 2048
+
 /* a list of reasons for queue stop */
 
 #define VMXNET3_ERR_NOEOP0x8000  /* cannot find the EOP desc of a 
pkt */
@@ -508,7 +516,9 @@ struct Vmxnet3_TxQueueConf {
__le32compRingSize; /* # of comp desc */
__le32ddLen;/* size of driver data */
uint8 intrIdx;
-   uint8 _pad[7];
+   uint8 _pad[1];
+   __le16txDataRingDescSize;
+   uint8 _pad2[4];
 }
 #include "vmware_pack_end.h"
 Vmxnet3_TxQueueConf;
diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.c 
b/drivers/net/vmxnet3/vmxnet3_ethdev.c
index 09f2085..05b0864 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.c
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.c
@@ -225,6 +225,24 @@ vmxnet3_disable_intr(struct vmxnet3_hw *hw)
 }
 
 /*
+ * Gets tx data ring descriptor size.
+ */
+static uint16_t
+eth_vmxnet3_txdata_get(struct vmxnet3_hw *hw)
+{
+   uint16 txdata_desc_size;
+
+   VMXNET3_WRITE_BAR1_REG(hw, VMXNET3_REG_CMD,
+  VMXNET3_CMD_GET_TXDATA_DESC_SIZE);
+   txdata_desc_size = VMXNET3_READ_BAR1_REG(hw, VMXNET3_REG_CMD);
+
+   return (txdata_desc_size < VMXNET3_TXDATA_DESC_MIN_SIZE ||
+   txdata_desc_size > VMXNET3_TXDATA_DESC_MAX_SIZE ||
+   txdata_desc_size & VMXNET3_TXDATA_DESC_SIZE_MASK) ?
+   sizeof(struct Vmxnet3_TxDataDesc) : txdata_desc_size;
+}
+
+/*
  * It returns 0 on success.
  */
 static int
@@ -320,6 +338,9 @@ eth_vmxnet3_dev_init(struct rte_eth_dev *eth_dev)
/* allow untagged pkts */
VMXNET3_SET_VFTABLE_ENTRY(hw->shadow_vfta, 0);
 
+   hw->txdata_desc_size = VMXNET3_VERSION_GE_3(hw) ?
+   eth_vmxnet3_txdata_get(hw) : sizeof(struct Vmxnet3_TxDataDesc);
+
return 0;
 }
 
@@ -511,6 +532,7 @@ vmxnet3_setup_driver_shared(struct rte_eth_dev *dev)
tqd->conf.txRingSize   = txq->cmd_ring.size;
tqd->conf.compRingSize = txq->comp_ring.size;
tqd->conf.dataRingSize = txq->data_ring.size;
+   tqd->conf.txDataRingDescSize = txq->txdata_desc_size;
tqd->conf.intrIdx  = txq->comp_ring.intr_idx;
tqd->status.stopped= TRUE;
tqd->status.error  = 0;
diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.h 
b/drivers/net/vmxnet3/vmxnet3_ethdev.h
index 516c407..f8e670d 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.h
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.h
@@ -103,6 +103,8 @@ struct vmxnet3_hw {
 
uint8_t version;
 
+   uint16_t txdata_desc_size; /* tx data ring buffer size */
+
Vmxnet3_TxQueueDesc   *tqd_start;   /* start address of all tx 
queue desc */
Vmxnet3_RxQueueDesc   *rqd_start;   /* start address of all rx 
queue desc */
 
diff --git a/drivers/net/vmxnet3/vmxnet3_ring.h 
b/drivers/net/vmxnet3/vmxnet3_ring.h
index b50d2b0..b27290f 100644
--- a/drivers/net/vmxnet3/vmxnet3_ring.h
+++ b/drivers/net/vmxnet3/vmxnet3_ring.h
@@ -141,6 +141,7 @@ typedef struct vmxnet3_tx_queue {
bool 

[dpdk-dev] [PATCH 2/7] vmxnet3: introduce generalized command interface to configure the device

2017-02-25 Thread Shrikrishna Khare
Shared memory is used to exchange information between the vmxnet3 driver
and the emulation. In order to request emulation to perform a task, the
driver first populates specific fields in this shared memory and then
issues corresponding command by writing to the command register(CMD). The
layout of the shared memory was defined by vmxnet3 version 1 and cannot
be extended for every new command without breaking backward compatibility.

To address this problem, in vmxnet3 version 3, the emulation repurposed
a reserved field in the shared memory to represent command information
instead. For new commands, the driver first populates the command
information field in the shared memory and then issues the command. The
emulation interprets the data written to the command information depending
on the type of the command. This patch exposes this capability to the driver.

Signed-off-by: Shrikrishna Khare 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/base/vmxnet3_defs.h | 32 +++-
 1 file changed, 31 insertions(+), 1 deletion(-)

diff --git a/drivers/net/vmxnet3/base/vmxnet3_defs.h 
b/drivers/net/vmxnet3/base/vmxnet3_defs.h
index 68ae8b6..7484b84 100644
--- a/drivers/net/vmxnet3/base/vmxnet3_defs.h
+++ b/drivers/net/vmxnet3/base/vmxnet3_defs.h
@@ -109,6 +109,7 @@ typedef enum {
VMXNET3_CMD_STOP_EMULATION,
VMXNET3_CMD_LOAD_PLUGIN,
VMXNET3_CMD_ACTIVATE_VF,
+   VMXNET3_CMD_RESERVED3,
 
VMXNET3_CMD_FIRST_GET = 0xF00D,
VMXNET3_CMD_GET_QUEUE_STATUS = VMXNET3_CMD_FIRST_GET,
@@ -695,12 +696,41 @@ Vmxnet3_RxQueueDesc;
 
 typedef
 #include "vmware_pack_begin.h"
+struct Vmxnet3_SetPolling {
+   uint8 enablePolling;
+}
+#include "vmware_pack_end.h"
+Vmxnet3_SetPolling;
+
+/*
+ * If the command data <= 16 bytes, use the shared memory direcly.
+ * Otherwise, use the variable length configuration descriptor.
+ */
+typedef
+#include "vmware_pack_begin.h"
+union Vmxnet3_CmdInfo {
+   Vmxnet3_VariableLenConfDesc varConf;
+   Vmxnet3_SetPolling  setPolling;
+   __le64  data[2];
+}
+#include "vmware_pack_end.h"
+Vmxnet3_CmdInfo;
+
+typedef
+#include "vmware_pack_begin.h"
 struct Vmxnet3_DriverShared {
__le32   magic;
__le32   pad; /* make devRead start at 64-bit boundaries */
Vmxnet3_DSDevReaddevRead;
__le32   ecr;
-   __le32   reserved[5];
+   __le32   reserved;
+
+   union {
+  __le32reserved1[4];
+  Vmxnet3_CmdInfo   cmdInfo; /* only valid in the context of executing the
+ * relevant command
+ */
+   } cu;
 }
 #include "vmware_pack_end.h"
 Vmxnet3_DriverShared;
-- 
2.6.2



[dpdk-dev] [PATCH 5/7] vmxnet3: add reserved version 3 command

2017-02-25 Thread Shrikrishna Khare
This command is reserved.

Signed-off-by: Shrikrishna Khare 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/base/vmxnet3_defs.h | 4 +++-
 1 file changed, 3 insertions(+), 1 deletion(-)

diff --git a/drivers/net/vmxnet3/base/vmxnet3_defs.h 
b/drivers/net/vmxnet3/base/vmxnet3_defs.h
index 746d709..c708498 100644
--- a/drivers/net/vmxnet3/base/vmxnet3_defs.h
+++ b/drivers/net/vmxnet3/base/vmxnet3_defs.h
@@ -110,6 +110,7 @@ typedef enum {
VMXNET3_CMD_LOAD_PLUGIN,
VMXNET3_CMD_ACTIVATE_VF,
VMXNET3_CMD_RESERVED3,
+   VMXNET3_CMD_RESERVED4,
 
VMXNET3_CMD_FIRST_GET = 0xF00D,
VMXNET3_CMD_GET_QUEUE_STATUS = VMXNET3_CMD_FIRST_GET,
@@ -122,7 +123,8 @@ typedef enum {
VMXNET3_CMD_GET_DEV_EXTRA_INFO,
VMXNET3_CMD_GET_CONF_INTR,
VMXNET3_CMD_GET_ADAPTIVE_RING_INFO,
-   VMXNET3_CMD_GET_TXDATA_DESC_SIZE
+   VMXNET3_CMD_GET_TXDATA_DESC_SIZE,
+   VMXNET3_CMD_RESERVED5,
 } Vmxnet3_Cmd;
 
 /* Adaptive Ring Info Flags */
-- 
2.6.2



[dpdk-dev] [PATCH 1/7] vmxnet3: prepare for version 3 changes

2017-02-25 Thread Shrikrishna Khare
Cleanup some code in preparation of vmxnet3 version 3 changes.

Signed-off-by: Shrikrishna Khare 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/vmxnet3_ethdev.c | 17 +
 drivers/net/vmxnet3/vmxnet3_ethdev.h |  9 +
 2 files changed, 22 insertions(+), 4 deletions(-)

diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.c 
b/drivers/net/vmxnet3/vmxnet3_ethdev.c
index b7b5377..09f2085 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.c
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.c
@@ -265,13 +265,22 @@ eth_vmxnet3_dev_init(struct rte_eth_dev *eth_dev)
/* Check h/w version compatibility with driver. */
ver = VMXNET3_READ_BAR1_REG(hw, VMXNET3_REG_VRRS);
PMD_INIT_LOG(DEBUG, "Hardware version : %d", ver);
-   if (ver & 0x1)
-   VMXNET3_WRITE_BAR1_REG(hw, VMXNET3_REG_VRRS, 1);
-   else {
-   PMD_INIT_LOG(ERR, "Incompatible h/w version, should be 0x1");
+
+   if (ver & (1 << VMXNET3_REV_2)) {
+   VMXNET3_WRITE_BAR1_REG(hw, VMXNET3_REG_VRRS,
+  1 << VMXNET3_REV_2);
+   hw->version = VMXNET3_REV_2 + 1;
+   } else if (ver & (1 << VMXNET3_REV_1)) {
+   VMXNET3_WRITE_BAR1_REG(hw, VMXNET3_REG_VRRS,
+  1 << VMXNET3_REV_1);
+   hw->version = VMXNET3_REV_1 + 1;
+   } else {
+   PMD_INIT_LOG(ERR, "Incompatible hardware version: %d", ver);
return -EIO;
}
 
+   PMD_INIT_LOG(DEBUG, "Using device version %d\n", hw->version);
+
/* Check UPT version compatibility with driver. */
ver = VMXNET3_READ_BAR1_REG(hw, VMXNET3_REG_UVRS);
PMD_INIT_LOG(DEBUG, "UPT hardware version : %d", ver);
diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.h 
b/drivers/net/vmxnet3/vmxnet3_ethdev.h
index 348c840..516c407 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.h
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.h
@@ -101,6 +101,8 @@ struct vmxnet3_hw {
uint8_t num_rx_queues;
uint8_t bufs_per_pkt;
 
+   uint8_t version;
+
Vmxnet3_TxQueueDesc   *tqd_start;   /* start address of all tx 
queue desc */
Vmxnet3_RxQueueDesc   *rqd_start;   /* start address of all rx 
queue desc */
 
@@ -117,6 +119,13 @@ struct vmxnet3_hw {
 #define VMXNET3_VFT_TABLE_SIZE (VMXNET3_VFT_SIZE * sizeof(uint32_t))
 };
 
+#define VMXNET3_REV_3  2   /* Vmxnet3 Rev. 3 */
+#define VMXNET3_REV_2  1   /* Vmxnet3 Rev. 2 */
+#define VMXNET3_REV_1  0   /* Vmxnet3 Rev. 1 */
+
+#define VMXNET3_VERSION_GE_3(hw) (hw->version >= VMXNET3_REV_3 + 1)
+#define VMXNET3_VERSION_GE_2(hw) (hw->version >= VMXNET3_REV_2 + 1)
+
 #define VMXNET3_GET_ADDR_LO(reg)   ((uint32_t)(reg))
 #define VMXNET3_GET_ADDR_HI(reg)   ((uint32_t)(((uint64_t)(reg)) >> 32))
 
-- 
2.6.2



[dpdk-dev] [PATCH 0/7] vmxnet3: upgrade to version 3

2017-02-25 Thread Shrikrishna Khare
vmxnet3 emulation has recently added several new features which includes
support for new commands the driver can issue to emulation, change in
descriptor fields etc. This patch series extends the vmxnet3 driver to
leverage these new features.

Compatibility is maintained using existing vmxnet3 versioning mechanism as
follows:
 - new features added to vmxnet3 emulation are associated with new vmxnet3
   version viz. vmxnet3 version 3.
 - emulation advertises all the versions it supports to the driver.
 - during initialization, vmxnet3 driver picks the highest version number
 supported by both the emulation and the driver and configures emulation
 to run at that version.

In particular, following changes are introduced:

Patch 1:
  Trivial cleanup in preparation of version 3 changes.

Patch 2:
  This patch introduces generalized command interface which allows
  for easily adding new commands that vmxnet3 driver can issue to the
  emulation. Further patches in this series make use of this facility.

Patch 3:
  Transmit data ring buffer is used to copy packet headers or small
  packets. It is a fixed size buffer. This patch extends the driver to
  allow variable sized transmit data ring buffer.

Patch 4:
  This patch introduces receive data ring buffer - a set of small sized
  buffers that are always mapped by the emulation. This avoids memory
  mapping/unmapping overhead for small packets.

Patch 5:
  This patch adds reserved commands.

Patch 6:
  In vmxnet3 version 3, the emulation added support for the vmxnet3 driver
  to communicate information about the memory regions the driver will use
  for rx/tx buffers. This patch exposes related commands to the driver. The
  driver is also extended to make use of this feaeture.

Patch 7:
  With all vmxnet3 version 3 changes incorporated in the vmxnet3 driver,
  with this patch, the driver can configure emulation to run at vmxnet3
  version 3.

Shrikrishna Khare (7):
  vmxnet3: prepare for version 3 changes
  vmxnet3: introduce generalized command interface to configure the
device
  vmxnet3: allow variable length transmit data ring buffer
  vmxnet3: add receive data ring support
  vmxnet3: add reserved version 3 command
  vmxnet3: introduce command to register memory region
  vmxnet3: update to version 3

 drivers/net/vmxnet3/base/vmxnet3_defs.h |  85 --
 drivers/net/vmxnet3/vmxnet3_ethdev.c| 154 +++-
 drivers/net/vmxnet3/vmxnet3_ethdev.h|  14 +++
 drivers/net/vmxnet3/vmxnet3_ring.h  |  21 +
 drivers/net/vmxnet3/vmxnet3_rxtx.c  |  45 --
 5 files changed, 304 insertions(+), 15 deletions(-)

-- 
2.6.2



[dpdk-dev] [PATCH 7/7] vmxnet3: update to version 3

2017-02-25 Thread Shrikrishna Khare
With all vmxnet3 version 3 changes incorporated in the vmxnet3 driver,
the driver can configure emulation to run at vmxnet3 version 3, provided
the emulation advertises support for version 3.

Signed-off-by: Shrikrishna Khare 
Acked-by: Yong Wang 
Acked-by: Jin Heo 
---
 drivers/net/vmxnet3/vmxnet3_ethdev.c | 6 +-
 1 file changed, 5 insertions(+), 1 deletion(-)

diff --git a/drivers/net/vmxnet3/vmxnet3_ethdev.c 
b/drivers/net/vmxnet3/vmxnet3_ethdev.c
index a4fc14d..21a78f6 100644
--- a/drivers/net/vmxnet3/vmxnet3_ethdev.c
+++ b/drivers/net/vmxnet3/vmxnet3_ethdev.c
@@ -284,7 +284,11 @@ eth_vmxnet3_dev_init(struct rte_eth_dev *eth_dev)
ver = VMXNET3_READ_BAR1_REG(hw, VMXNET3_REG_VRRS);
PMD_INIT_LOG(DEBUG, "Hardware version : %d", ver);
 
-   if (ver & (1 << VMXNET3_REV_2)) {
+   if (ver & (1 << VMXNET3_REV_3)) {
+   VMXNET3_WRITE_BAR1_REG(hw, VMXNET3_REG_VRRS,
+  1 << VMXNET3_REV_3);
+   hw->version = VMXNET3_REV_3 + 1;
+   } else if (ver & (1 << VMXNET3_REV_2)) {
VMXNET3_WRITE_BAR1_REG(hw, VMXNET3_REG_VRRS,
   1 << VMXNET3_REV_2);
hw->version = VMXNET3_REV_2 + 1;
-- 
2.6.2