examples/l3fwd: share queue size variables
[dpdk.git] / drivers / net / mlx5 / mlx5_rxq.c
index e7284f9..ff293d9 100644 (file)
@@ -30,6 +30,7 @@
 #include "mlx5_utils.h"
 #include "mlx5_autoconf.h"
 #include "mlx5_devx.h"
+#include "rte_pmd_mlx5.h"
 
 
 /* Default RSS hash key also used for ConnectX-3. */
@@ -1391,8 +1392,7 @@ mlx5_mprq_alloc_mp(struct rte_eth_dev *dev)
                struct mlx5_rxq_ctrl *rxq_ctrl = mlx5_rxq_ctrl_get(dev, i);
                struct mlx5_rxq_data *rxq;
 
-               if (rxq_ctrl == NULL ||
-                   rxq_ctrl->type != MLX5_RXQ_TYPE_STANDARD)
+               if (rxq_ctrl == NULL || rxq_ctrl->is_hairpin)
                        continue;
                rxq = &rxq_ctrl->rxq;
                n_ibv++;
@@ -1480,8 +1480,7 @@ exit:
        for (i = 0; i != priv->rxqs_n; ++i) {
                struct mlx5_rxq_ctrl *rxq_ctrl = mlx5_rxq_ctrl_get(dev, i);
 
-               if (rxq_ctrl == NULL ||
-                   rxq_ctrl->type != MLX5_RXQ_TYPE_STANDARD)
+               if (rxq_ctrl == NULL || rxq_ctrl->is_hairpin)
                        continue;
                rxq_ctrl->rxq.mprq_mp = mp;
        }
@@ -1798,7 +1797,7 @@ mlx5_rxq_new(struct rte_eth_dev *dev, uint16_t idx, uint16_t desc,
                rte_errno = ENOSPC;
                goto error;
        }
-       tmpl->type = MLX5_RXQ_TYPE_STANDARD;
+       tmpl->is_hairpin = false;
        if (mlx5_mr_ctrl_init(&tmpl->rxq.mr_ctrl,
                              &priv->sh->cdev->mr_scache.dev_gen, socket)) {
                /* rte_errno is already set. */
@@ -1969,7 +1968,7 @@ mlx5_rxq_hairpin_new(struct rte_eth_dev *dev, struct mlx5_rxq_priv *rxq,
        LIST_INIT(&tmpl->owners);
        rxq->ctrl = tmpl;
        LIST_INSERT_HEAD(&tmpl->owners, rxq, owner_entry);
-       tmpl->type = MLX5_RXQ_TYPE_HAIRPIN;
+       tmpl->is_hairpin = true;
        tmpl->socket = SOCKET_ID_ANY;
        tmpl->rxq.rss_hash = 0;
        tmpl->rxq.port_id = dev->data->port_id;
@@ -2085,6 +2084,65 @@ mlx5_rxq_data_get(struct rte_eth_dev *dev, uint16_t idx)
        return rxq == NULL ? NULL : &rxq->ctrl->rxq;
 }
 
+/**
+ * Increase an external Rx queue reference count.
+ *
+ * @param dev
+ *   Pointer to Ethernet device.
+ * @param idx
+ *   External RX queue index.
+ *
+ * @return
+ *   A pointer to the queue if it exists, NULL otherwise.
+ */
+struct mlx5_external_rxq *
+mlx5_ext_rxq_ref(struct rte_eth_dev *dev, uint16_t idx)
+{
+       struct mlx5_external_rxq *rxq = mlx5_ext_rxq_get(dev, idx);
+
+       __atomic_fetch_add(&rxq->refcnt, 1, __ATOMIC_RELAXED);
+       return rxq;
+}
+
+/**
+ * Decrease an external Rx queue reference count.
+ *
+ * @param dev
+ *   Pointer to Ethernet device.
+ * @param idx
+ *   External RX queue index.
+ *
+ * @return
+ *   Updated reference count.
+ */
+uint32_t
+mlx5_ext_rxq_deref(struct rte_eth_dev *dev, uint16_t idx)
+{
+       struct mlx5_external_rxq *rxq = mlx5_ext_rxq_get(dev, idx);
+
+       return __atomic_sub_fetch(&rxq->refcnt, 1, __ATOMIC_RELAXED);
+}
+
+/**
+ * Get an external Rx queue.
+ *
+ * @param dev
+ *   Pointer to Ethernet device.
+ * @param idx
+ *   External Rx queue index.
+ *
+ * @return
+ *   A pointer to the queue if it exists, NULL otherwise.
+ */
+struct mlx5_external_rxq *
+mlx5_ext_rxq_get(struct rte_eth_dev *dev, uint16_t idx)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+
+       MLX5_ASSERT(mlx5_is_external_rxq(dev, idx));
+       return &priv->ext_rxqs[idx - MLX5_EXTERNAL_RX_QUEUE_ID_MIN];
+}
+
 /**
  * Release a Rx queue.
  *
@@ -2120,7 +2178,7 @@ mlx5_rxq_release(struct rte_eth_dev *dev, uint16_t idx)
                        mlx5_free(rxq_ctrl->obj);
                        rxq_ctrl->obj = NULL;
                }
-               if (rxq_ctrl->type == MLX5_RXQ_TYPE_STANDARD) {
+               if (!rxq_ctrl->is_hairpin) {
                        if (!rxq_ctrl->started)
                                rxq_free_elts(rxq_ctrl);
                        dev->data->rx_queue_state[idx] =
@@ -2129,7 +2187,7 @@ mlx5_rxq_release(struct rte_eth_dev *dev, uint16_t idx)
        } else { /* Refcnt zero, closing device. */
                LIST_REMOVE(rxq, owner_entry);
                if (LIST_EMPTY(&rxq_ctrl->owners)) {
-                       if (rxq_ctrl->type == MLX5_RXQ_TYPE_STANDARD)
+                       if (!rxq_ctrl->is_hairpin)
                                mlx5_mr_btree_free
                                        (&rxq_ctrl->rxq.mr_ctrl.cache_bh);
                        if (rxq_ctrl->rxq.shared)
@@ -2169,7 +2227,38 @@ mlx5_rxq_verify(struct rte_eth_dev *dev)
 }
 
 /**
- * Get a Rx queue type.
+ * Verify the external Rx Queue list is empty.
+ *
+ * @param dev
+ *   Pointer to Ethernet device.
+ *
+ * @return
+ *   The number of object not released.
+ */
+int
+mlx5_ext_rxq_verify(struct rte_eth_dev *dev)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_external_rxq *rxq;
+       uint32_t i;
+       int ret = 0;
+
+       if (priv->ext_rxqs == NULL)
+               return 0;
+
+       for (i = MLX5_EXTERNAL_RX_QUEUE_ID_MIN; i <= UINT16_MAX ; ++i) {
+               rxq = mlx5_ext_rxq_get(dev, i);
+               if (rxq->refcnt < 2)
+                       continue;
+               DRV_LOG(DEBUG, "Port %u external RxQ %u still referenced.",
+                       dev->data->port_id, i);
+               ++ret;
+       }
+       return ret;
+}
+
+/**
+ * Check whether RxQ type is Hairpin.
  *
  * @param dev
  *   Pointer to Ethernet device.
@@ -2177,17 +2266,18 @@ mlx5_rxq_verify(struct rte_eth_dev *dev)
  *   Rx queue index.
  *
  * @return
- *   The Rx queue type.
+ *   True if Rx queue type is Hairpin, otherwise False.
  */
-enum mlx5_rxq_type
-mlx5_rxq_get_type(struct rte_eth_dev *dev, uint16_t idx)
+bool
+mlx5_rxq_is_hairpin(struct rte_eth_dev *dev, uint16_t idx)
 {
        struct mlx5_priv *priv = dev->data->dev_private;
-       struct mlx5_rxq_ctrl *rxq_ctrl = mlx5_rxq_ctrl_get(dev, idx);
+       struct mlx5_rxq_ctrl *rxq_ctrl;
 
-       if (idx < priv->rxqs_n && rxq_ctrl != NULL)
-               return rxq_ctrl->type;
-       return MLX5_RXQ_TYPE_UNDEFINED;
+       if (mlx5_is_external_rxq(dev, idx))
+               return false;
+       rxq_ctrl = mlx5_rxq_ctrl_get(dev, idx);
+       return (idx < priv->rxqs_n && rxq_ctrl != NULL && rxq_ctrl->is_hairpin);
 }
 
 /*
@@ -2204,14 +2294,9 @@ mlx5_rxq_get_type(struct rte_eth_dev *dev, uint16_t idx)
 const struct rte_eth_hairpin_conf *
 mlx5_rxq_get_hairpin_conf(struct rte_eth_dev *dev, uint16_t idx)
 {
-       struct mlx5_priv *priv = dev->data->dev_private;
        struct mlx5_rxq_priv *rxq = mlx5_rxq_get(dev, idx);
 
-       if (idx < priv->rxqs_n && rxq != NULL) {
-               if (rxq->ctrl->type == MLX5_RXQ_TYPE_HAIRPIN)
-                       return &rxq->hairpin_conf;
-       }
-       return NULL;
+       return mlx5_rxq_is_hairpin(dev, idx) ? &rxq->hairpin_conf : NULL;
 }
 
 /**
@@ -2366,9 +2451,16 @@ mlx5_ind_table_obj_setup(struct rte_eth_dev *dev,
 
        if (ref_qs)
                for (i = 0; i != queues_n; ++i) {
-                       if (mlx5_rxq_ref(dev, queues[i]) == NULL) {
-                               ret = -rte_errno;
-                               goto error;
+                       if (mlx5_is_external_rxq(dev, queues[i])) {
+                               if (mlx5_ext_rxq_ref(dev, queues[i]) == NULL) {
+                                       ret = -rte_errno;
+                                       goto error;
+                               }
+                       } else {
+                               if (mlx5_rxq_ref(dev, queues[i]) == NULL) {
+                                       ret = -rte_errno;
+                                       goto error;
+                               }
                        }
                }
        ret = priv->obj_ops.ind_table_new(dev, n, ind_tbl);
@@ -2379,8 +2471,12 @@ mlx5_ind_table_obj_setup(struct rte_eth_dev *dev,
 error:
        if (ref_qs) {
                err = rte_errno;
-               for (j = 0; j < i; j++)
-                       mlx5_rxq_deref(dev, queues[j]);
+               for (j = 0; j < i; j++) {
+                       if (mlx5_is_external_rxq(dev, queues[j]))
+                               mlx5_ext_rxq_deref(dev, queues[j]);
+                       else
+                               mlx5_rxq_deref(dev, queues[j]);
+               }
                rte_errno = err;
        }
        DRV_LOG(DEBUG, "Port %u cannot setup indirection table.",
@@ -3017,3 +3113,119 @@ mlx5_rxq_timestamp_set(struct rte_eth_dev *dev)
                data->rt_timestamp = sh->dev_cap.rt_timestamp;
        }
 }
+
+/**
+ * Validate given external RxQ rte_plow index, and get pointer to concurrent
+ * external RxQ object to map/unmap.
+ *
+ * @param[in] port_id
+ *   The port identifier of the Ethernet device.
+ * @param[in] dpdk_idx
+ *   Queue index in rte_flow.
+ *
+ * @return
+ *   Pointer to concurrent external RxQ on success,
+ *   NULL otherwise and rte_errno is set.
+ */
+static struct mlx5_external_rxq *
+mlx5_external_rx_queue_get_validate(uint16_t port_id, uint16_t dpdk_idx)
+{
+       struct rte_eth_dev *dev;
+       struct mlx5_priv *priv;
+
+       if (dpdk_idx < MLX5_EXTERNAL_RX_QUEUE_ID_MIN) {
+               DRV_LOG(ERR, "Queue index %u should be in range: [%u, %u].",
+                       dpdk_idx, MLX5_EXTERNAL_RX_QUEUE_ID_MIN, UINT16_MAX);
+               rte_errno = EINVAL;
+               return NULL;
+       }
+       if (rte_eth_dev_is_valid_port(port_id) < 0) {
+               DRV_LOG(ERR, "There is no Ethernet device for port %u.",
+                       port_id);
+               rte_errno = ENODEV;
+               return NULL;
+       }
+       dev = &rte_eth_devices[port_id];
+       priv = dev->data->dev_private;
+       if (!mlx5_imported_pd_and_ctx(priv->sh->cdev)) {
+               DRV_LOG(ERR, "Port %u "
+                       "external RxQ isn't supported on local PD and CTX.",
+                       port_id);
+               rte_errno = ENOTSUP;
+               return NULL;
+       }
+       if (!mlx5_devx_obj_ops_en(priv->sh)) {
+               DRV_LOG(ERR,
+                       "Port %u external RxQ isn't supported by Verbs API.",
+                       port_id);
+               rte_errno = ENOTSUP;
+               return NULL;
+       }
+       /*
+        * When user configures remote PD and CTX and device creates RxQ by
+        * DevX, external RxQs array is allocated.
+        */
+       MLX5_ASSERT(priv->ext_rxqs != NULL);
+       return &priv->ext_rxqs[dpdk_idx - MLX5_EXTERNAL_RX_QUEUE_ID_MIN];
+}
+
+int
+rte_pmd_mlx5_external_rx_queue_id_map(uint16_t port_id, uint16_t dpdk_idx,
+                                     uint32_t hw_idx)
+{
+       struct mlx5_external_rxq *ext_rxq;
+       uint32_t unmapped = 0;
+
+       ext_rxq = mlx5_external_rx_queue_get_validate(port_id, dpdk_idx);
+       if (ext_rxq == NULL)
+               return -rte_errno;
+       if (!__atomic_compare_exchange_n(&ext_rxq->refcnt, &unmapped, 1, false,
+                                        __ATOMIC_RELAXED, __ATOMIC_RELAXED)) {
+               if (ext_rxq->hw_id != hw_idx) {
+                       DRV_LOG(ERR, "Port %u external RxQ index %u "
+                               "is already mapped to HW index (requesting is "
+                               "%u, existing is %u).",
+                               port_id, dpdk_idx, hw_idx, ext_rxq->hw_id);
+                       rte_errno = EEXIST;
+                       return -rte_errno;
+               }
+               DRV_LOG(WARNING, "Port %u external RxQ index %u "
+                       "is already mapped to the requested HW index (%u)",
+                       port_id, dpdk_idx, hw_idx);
+
+       } else {
+               ext_rxq->hw_id = hw_idx;
+               DRV_LOG(DEBUG, "Port %u external RxQ index %u "
+                       "is successfully mapped to the requested HW index (%u)",
+                       port_id, dpdk_idx, hw_idx);
+       }
+       return 0;
+}
+
+int
+rte_pmd_mlx5_external_rx_queue_id_unmap(uint16_t port_id, uint16_t dpdk_idx)
+{
+       struct mlx5_external_rxq *ext_rxq;
+       uint32_t mapped = 1;
+
+       ext_rxq = mlx5_external_rx_queue_get_validate(port_id, dpdk_idx);
+       if (ext_rxq == NULL)
+               return -rte_errno;
+       if (ext_rxq->refcnt > 1) {
+               DRV_LOG(ERR, "Port %u external RxQ index %u still referenced.",
+                       port_id, dpdk_idx);
+               rte_errno = EINVAL;
+               return -rte_errno;
+       }
+       if (!__atomic_compare_exchange_n(&ext_rxq->refcnt, &mapped, 0, false,
+                                        __ATOMIC_RELAXED, __ATOMIC_RELAXED)) {
+               DRV_LOG(ERR, "Port %u external RxQ index %u doesn't exist.",
+                       port_id, dpdk_idx);
+               rte_errno = EINVAL;
+               return -rte_errno;
+       }
+       DRV_LOG(DEBUG,
+               "Port %u external RxQ index %u is successfully unmapped.",
+               port_id, dpdk_idx);
+       return 0;
+}