net/mlx4: separate Rx/Tx definitions
[dpdk.git] / drivers / net / mlx4 / mlx4_flow.c
index 8ade106..61455ce 100644 (file)
@@ -2,7 +2,7 @@
  *   BSD LICENSE
  *
  *   Copyright 2017 6WIND S.A.
- *   Copyright 2017 Mellanox.
+ *   Copyright 2017 Mellanox
  *
  *   Redistribution and use in source and binary forms, with or without
  *   modification, are permitted provided that the following conditions
 #include <rte_flow_driver.h>
 #include <rte_malloc.h>
 
-/* Generated configuration header. */
-#include "mlx4_autoconf.h"
-
 /* PMD headers. */
 #include "mlx4.h"
 #include "mlx4_flow.h"
+#include "mlx4_rxtx.h"
+#include "mlx4_utils.h"
 
 /** Static initializer for items. */
 #define ITEMS(...) \
@@ -112,7 +111,6 @@ struct rte_flow_drop {
 static const enum rte_flow_action_type valid_actions[] = {
        RTE_FLOW_ACTION_TYPE_DROP,
        RTE_FLOW_ACTION_TYPE_QUEUE,
-       RTE_FLOW_ACTION_TYPE_RSS,
        RTE_FLOW_ACTION_TYPE_END,
 };
 
@@ -673,76 +671,6 @@ priv_flow_validate(struct priv *priv,
                        if (!queue || (queue->index > (priv->rxqs_n - 1)))
                                goto exit_action_not_supported;
                        action.queue = 1;
-                       action.queues_n = 1;
-                       action.queues[0] = queue->index;
-               } else if (actions->type == RTE_FLOW_ACTION_TYPE_RSS) {
-                       int i;
-                       int ierr;
-                       const struct rte_flow_action_rss *rss =
-                               (const struct rte_flow_action_rss *)
-                               actions->conf;
-
-                       if (!priv->hw_rss) {
-                               rte_flow_error_set(error, ENOTSUP,
-                                          RTE_FLOW_ERROR_TYPE_ACTION,
-                                          actions,
-                                          "RSS cannot be used with "
-                                          "the current configuration");
-                               return -rte_errno;
-                       }
-                       if (!priv->isolated) {
-                               rte_flow_error_set(error, ENOTSUP,
-                                          RTE_FLOW_ERROR_TYPE_ACTION,
-                                          actions,
-                                          "RSS cannot be used without "
-                                          "isolated mode");
-                               return -rte_errno;
-                       }
-                       if (!rte_is_power_of_2(rss->num)) {
-                               rte_flow_error_set(error, ENOTSUP,
-                                          RTE_FLOW_ERROR_TYPE_ACTION,
-                                          actions,
-                                          "the number of queues "
-                                          "should be power of two");
-                               return -rte_errno;
-                       }
-                       if (priv->max_rss_tbl_sz < rss->num) {
-                               rte_flow_error_set(error, ENOTSUP,
-                                          RTE_FLOW_ERROR_TYPE_ACTION,
-                                          actions,
-                                          "the number of queues "
-                                          "is too large");
-                               return -rte_errno;
-                       }
-                       /* checking indexes array */
-                       ierr = 0;
-                       for (i = 0; i < rss->num; ++i) {
-                               int j;
-                               if (rss->queue[i] >= priv->rxqs_n)
-                                       ierr = 1;
-                               /*
-                                * Prevent the user from specifying
-                                * the same queue twice in the RSS array.
-                                */
-                               for (j = i + 1; j < rss->num && !ierr; ++j)
-                                       if (rss->queue[j] == rss->queue[i])
-                                               ierr = 1;
-                               if (ierr) {
-                                       rte_flow_error_set(
-                                               error,
-                                               ENOTSUP,
-                                               RTE_FLOW_ERROR_TYPE_HANDLE,
-                                               NULL,
-                                               "RSS action only supports "
-                                               "unique queue indices "
-                                               "in a list");
-                                       return -rte_errno;
-                               }
-                       }
-                       action.queue = 1;
-                       action.queues_n = rss->num;
-                       for (i = 0; i < rss->num; ++i)
-                               action.queues[i] = rss->queue[i];
                } else {
                        goto exit_action_not_supported;
                }
@@ -777,13 +705,9 @@ mlx4_flow_validate(struct rte_eth_dev *dev,
                   struct rte_flow_error *error)
 {
        struct priv *priv = dev->data->dev_private;
-       int ret;
        struct mlx4_flow flow = { .offset = sizeof(struct ibv_flow_attr) };
 
-       priv_lock(priv);
-       ret = priv_flow_validate(priv, attr, items, actions, error, &flow);
-       priv_unlock(priv);
-       return ret;
+       return priv_flow_validate(priv, attr, items, actions, error, &flow);
 }
 
 /**
@@ -826,29 +750,21 @@ mlx4_flow_create_drop_queue(struct priv *priv)
                ERROR("Cannot allocate memory for drop struct");
                goto err;
        }
-       cq = ibv_exp_create_cq(priv->ctx, 1, NULL, NULL, 0,
-                             &(struct ibv_exp_cq_init_attr){
-                                       .comp_mask = 0,
-                             });
+       cq = ibv_create_cq(priv->ctx, 1, NULL, NULL, 0);
        if (!cq) {
                ERROR("Cannot create drop CQ");
                goto err_create_cq;
        }
-       qp = ibv_exp_create_qp(priv->ctx,
-                             &(struct ibv_exp_qp_init_attr){
-                                       .send_cq = cq,
-                                       .recv_cq = cq,
-                                       .cap = {
-                                               .max_recv_wr = 1,
-                                               .max_recv_sge = 1,
-                                       },
-                                       .qp_type = IBV_QPT_RAW_PACKET,
-                                       .comp_mask =
-                                               IBV_EXP_QP_INIT_ATTR_PD |
-                                               IBV_EXP_QP_INIT_ATTR_PORT,
-                                       .pd = priv->pd,
-                                       .port_num = priv->port,
-                             });
+       qp = ibv_create_qp(priv->pd,
+                          &(struct ibv_qp_init_attr){
+                               .send_cq = cq,
+                               .recv_cq = cq,
+                               .cap = {
+                                       .max_recv_wr = 1,
+                                       .max_recv_sge = 1,
+                               },
+                               .qp_type = IBV_QPT_RAW_PACKET,
+                          });
        if (!qp) {
                ERROR("Cannot create drop QP");
                goto err_create_qp;
@@ -867,82 +783,6 @@ err:
        return -1;
 }
 
-/**
- * Get RSS parent rxq structure for given queues.
- *
- * Creates a new or returns an existed one.
- *
- * @param priv
- *   Pointer to private structure.
- * @param queues
- *   queues indices array, NULL in default RSS case.
- * @param children_n
- *   the size of queues array.
- *
- * @return
- *   Pointer to a parent rxq structure, NULL on failure.
- */
-static struct rxq *
-priv_parent_get(struct priv *priv,
-               uint16_t queues[],
-               uint16_t children_n,
-               struct rte_flow_error *error)
-{
-       unsigned int i;
-       struct rxq *parent;
-
-       for (parent = LIST_FIRST(&priv->parents);
-            parent;
-            parent = LIST_NEXT(parent, next)) {
-               unsigned int same = 0;
-               unsigned int overlap = 0;
-
-               /*
-                * Find out whether an appropriate parent queue already exists
-                * and can be reused, otherwise make sure there are no overlaps.
-                */
-               for (i = 0; i < children_n; ++i) {
-                       unsigned int j;
-
-                       for (j = 0; j < parent->rss.queues_n; ++j) {
-                               if (parent->rss.queues[j] != queues[i])
-                                       continue;
-                               ++overlap;
-                               if (i == j)
-                                       ++same;
-                       }
-               }
-               if (same == children_n &&
-                       children_n == parent->rss.queues_n)
-                       return parent;
-               else if (overlap)
-                       goto error;
-       }
-       /* Exclude the cases when some QPs were created without RSS */
-       for (i = 0; i < children_n; ++i) {
-               struct rxq *rxq = (*priv->rxqs)[queues[i]];
-               if (rxq->qp)
-                       goto error;
-       }
-       parent = priv_parent_create(priv, queues, children_n);
-       if (!parent) {
-               rte_flow_error_set(error,
-                                  ENOMEM, RTE_FLOW_ERROR_TYPE_UNSPECIFIED,
-                                  NULL, "flow rule creation failure");
-               return NULL;
-       }
-       return parent;
-
-error:
-       rte_flow_error_set(error,
-                          EEXIST,
-                          RTE_FLOW_ERROR_TYPE_UNSPECIFIED,
-                          NULL,
-                          "sharing a queue between several"
-                          " RSS groups is not supported");
-       return NULL;
-}
-
 /**
  * Complete flow rule creation.
  *
@@ -966,7 +806,6 @@ priv_flow_create_action_queue(struct priv *priv,
 {
        struct ibv_qp *qp;
        struct rte_flow *rte_flow;
-       struct rxq *rxq_parent = NULL;
 
        assert(priv->pd);
        assert(priv->ctx);
@@ -977,44 +816,16 @@ priv_flow_create_action_queue(struct priv *priv,
                return NULL;
        }
        if (action->drop) {
-               qp = priv->flow_drop_queue->qp;
+               qp = priv->flow_drop_queue ? priv->flow_drop_queue->qp : NULL;
        } else {
-               int ret;
-               unsigned int i;
-               struct rxq *rxq = NULL;
+               struct rxq *rxq = (*priv->rxqs)[action->queue_id];
 
-               if (action->queues_n > 1) {
-                       rxq_parent = priv_parent_get(priv, action->queues,
-                                                    action->queues_n, error);
-                       if (!rxq_parent)
-                               goto error;
-               }
-               for (i = 0; i < action->queues_n; ++i) {
-                       rxq = (*priv->rxqs)[action->queues[i]];
-                       /*
-                        * In case of isolated mode we postpone
-                        * ibv receive queue creation till the first
-                        * rte_flow rule will be applied on that queue.
-                        */
-                       if (!rxq->qp) {
-                               assert(priv->isolated);
-                               ret = rxq_create_qp(rxq, rxq->elts_n,
-                                                   0, 0, rxq_parent);
-                               if (ret) {
-                                       rte_flow_error_set(
-                                               error,
-                                               ENOMEM,
-                                               RTE_FLOW_ERROR_TYPE_HANDLE,
-                                               NULL,
-                                               "flow rule creation failure");
-                                       goto error;
-                               }
-                       }
-               }
-               qp = action->queues_n > 1 ? rxq_parent->qp : rxq->qp;
+               qp = rxq->qp;
                rte_flow->qp = qp;
        }
        rte_flow->ibv_attr = ibv_attr;
+       if (!priv->started)
+               return rte_flow;
        rte_flow->ibv_flow = ibv_create_flow(qp, rte_flow->ibv_attr);
        if (!rte_flow->ibv_flow) {
                rte_flow_error_set(error, ENOMEM, RTE_FLOW_ERROR_TYPE_HANDLE,
@@ -1022,38 +833,25 @@ priv_flow_create_action_queue(struct priv *priv,
                goto error;
        }
        return rte_flow;
-
 error:
-       if (rxq_parent)
-               rxq_parent_cleanup(rxq_parent);
        rte_free(rte_flow);
        return NULL;
 }
 
 /**
- * Convert a flow.
- *
- * @param priv
- *   Pointer to private structure.
- * @param[in] attr
- *   Flow rule attributes.
- * @param[in] items
- *   Pattern specification (list terminated by the END pattern item).
- * @param[in] actions
- *   Associated actions (list terminated by the END action).
- * @param[out] error
- *   Perform verbose error reporting if not NULL.
+ * Create a flow.
  *
- * @return
- *   A flow on success, NULL otherwise.
+ * @see rte_flow_create()
+ * @see rte_flow_ops
  */
-static struct rte_flow *
-priv_flow_create(struct priv *priv,
+struct rte_flow *
+mlx4_flow_create(struct rte_eth_dev *dev,
                 const struct rte_flow_attr *attr,
                 const struct rte_flow_item items[],
                 const struct rte_flow_action actions[],
                 struct rte_flow_error *error)
 {
+       struct priv *priv = dev->data->dev_private;
        struct rte_flow *rte_flow;
        struct mlx4_flow_action action;
        struct mlx4_flow flow = { .offset = sizeof(struct ibv_flow_attr), };
@@ -1089,22 +887,11 @@ priv_flow_create(struct priv *priv,
                        continue;
                } else if (actions->type == RTE_FLOW_ACTION_TYPE_QUEUE) {
                        action.queue = 1;
-                       action.queues_n = 1;
-                       action.queues[0] =
+                       action.queue_id =
                                ((const struct rte_flow_action_queue *)
                                 actions->conf)->index;
                } else if (actions->type == RTE_FLOW_ACTION_TYPE_DROP) {
                        action.drop = 1;
-               } else if (actions->type == RTE_FLOW_ACTION_TYPE_RSS) {
-                       unsigned int i;
-                       const struct rte_flow_action_rss *rss =
-                               (const struct rte_flow_action_rss *)
-                                actions->conf;
-
-                       action.queue = 1;
-                       action.queues_n = rss->num;
-                       for (i = 0; i < rss->num; ++i)
-                               action.queues[i] = rss->queue[i];
                } else {
                        rte_flow_error_set(error, ENOTSUP,
                                           RTE_FLOW_ERROR_TYPE_ACTION,
@@ -1114,39 +901,16 @@ priv_flow_create(struct priv *priv,
        }
        rte_flow = priv_flow_create_action_queue(priv, flow.ibv_attr,
                                                 &action, error);
-       if (rte_flow)
+       if (rte_flow) {
+               LIST_INSERT_HEAD(&priv->flows, rte_flow, next);
+               DEBUG("Flow created %p", (void *)rte_flow);
                return rte_flow;
+       }
 exit:
        rte_free(flow.ibv_attr);
        return NULL;
 }
 
-/**
- * Create a flow.
- *
- * @see rte_flow_create()
- * @see rte_flow_ops
- */
-struct rte_flow *
-mlx4_flow_create(struct rte_eth_dev *dev,
-                const struct rte_flow_attr *attr,
-                const struct rte_flow_item items[],
-                const struct rte_flow_action actions[],
-                struct rte_flow_error *error)
-{
-       struct priv *priv = dev->data->dev_private;
-       struct rte_flow *flow;
-
-       priv_lock(priv);
-       flow = priv_flow_create(priv, attr, items, actions, error);
-       if (flow) {
-               LIST_INSERT_HEAD(&priv->flows, flow, next);
-               DEBUG("Flow created %p", (void *)flow);
-       }
-       priv_unlock(priv);
-       return flow;
-}
-
 /**
  * @see rte_flow_isolate()
  *
@@ -1170,40 +934,17 @@ mlx4_flow_isolate(struct rte_eth_dev *dev,
 {
        struct priv *priv = dev->data->dev_private;
 
-       priv_lock(priv);
        if (priv->rxqs) {
                rte_flow_error_set(error, ENOTSUP,
                                   RTE_FLOW_ERROR_TYPE_UNSPECIFIED,
                                   NULL, "isolated mode must be set"
                                   " before configuring the device");
-               priv_unlock(priv);
                return -rte_errno;
        }
        priv->isolated = !!enable;
-       priv_unlock(priv);
        return 0;
 }
 
-/**
- * Destroy a flow.
- *
- * @param priv
- *   Pointer to private structure.
- * @param[in] flow
- *   Flow to destroy.
- */
-static void
-priv_flow_destroy(struct priv *priv, struct rte_flow *flow)
-{
-       (void)priv;
-       LIST_REMOVE(flow, next);
-       if (flow->ibv_flow)
-               claim_zero(ibv_destroy_flow(flow->ibv_flow));
-       rte_free(flow->ibv_attr);
-       DEBUG("Flow destroyed %p", (void *)flow);
-       rte_free(flow);
-}
-
 /**
  * Destroy a flow.
  *
@@ -1215,32 +956,17 @@ mlx4_flow_destroy(struct rte_eth_dev *dev,
                  struct rte_flow *flow,
                  struct rte_flow_error *error)
 {
-       struct priv *priv = dev->data->dev_private;
-
+       (void)dev;
        (void)error;
-       priv_lock(priv);
-       priv_flow_destroy(priv, flow);
-       priv_unlock(priv);
+       LIST_REMOVE(flow, next);
+       if (flow->ibv_flow)
+               claim_zero(ibv_destroy_flow(flow->ibv_flow));
+       rte_free(flow->ibv_attr);
+       DEBUG("Flow destroyed %p", (void *)flow);
+       rte_free(flow);
        return 0;
 }
 
-/**
- * Destroy all flows.
- *
- * @param priv
- *   Pointer to private structure.
- */
-static void
-priv_flow_flush(struct priv *priv)
-{
-       while (!LIST_EMPTY(&priv->flows)) {
-               struct rte_flow *flow;
-
-               flow = LIST_FIRST(&priv->flows);
-               priv_flow_destroy(priv, flow);
-       }
-}
-
 /**
  * Destroy all flows.
  *
@@ -1253,10 +979,12 @@ mlx4_flow_flush(struct rte_eth_dev *dev,
 {
        struct priv *priv = dev->data->dev_private;
 
-       (void)error;
-       priv_lock(priv);
-       priv_flow_flush(priv);
-       priv_unlock(priv);
+       while (!LIST_EMPTY(&priv->flows)) {
+               struct rte_flow *flow;
+
+               flow = LIST_FIRST(&priv->flows);
+               mlx4_flow_destroy(dev, flow, error);
+       }
        return 0;
 }