examples/l3fwd: share queue size variables
[dpdk.git] / drivers / net / mlx5 / mlx5_rxq.c
index e96584d..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. */
@@ -2083,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.
  *
@@ -2166,6 +2226,37 @@ mlx5_rxq_verify(struct rte_eth_dev *dev)
        return ret;
 }
 
+/**
+ * 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.
  *
@@ -2181,8 +2272,11 @@ 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 (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);
 }
 
@@ -2357,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);
@@ -2370,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.",
@@ -3008,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;
+}