From: Yevgeny Kliteynik <klit...@nvidia.com>

Current BWC implementation doesn't support a case when a rule is added
with AT that requires more action STEs than any other AT that already
exists on this matcher.
This patch adds this support: if such AT is required, we will add the
new AT and do full rehash to a new matcher that is created with all the
action templates, including the new larger AT.

Signed-off-by: Yevgeny Kliteynik <klit...@nvidia.com>
---
 drivers/net/mlx5/hws/mlx5dr_bwc.c     | 68 ++++++++++++++++++++-------
 drivers/net/mlx5/hws/mlx5dr_matcher.c |  4 +-
 2 files changed, 52 insertions(+), 20 deletions(-)

diff --git a/drivers/net/mlx5/hws/mlx5dr_bwc.c 
b/drivers/net/mlx5/hws/mlx5dr_bwc.c
index bfc7dbf100..8a8a143f17 100644
--- a/drivers/net/mlx5/hws/mlx5dr_bwc.c
+++ b/drivers/net/mlx5/hws/mlx5dr_bwc.c
@@ -712,7 +712,7 @@ mlx5dr_bwc_matcher_move(struct mlx5dr_bwc_matcher 
*bwc_matcher)
 }
 
 static int
-mlx5dr_bwc_matcher_rehash(struct mlx5dr_bwc_matcher *bwc_matcher, bool 
rehash_size)
+mlx5dr_bwc_matcher_rehash_size(struct mlx5dr_bwc_matcher *bwc_matcher)
 {
        uint32_t num_of_rules;
        int ret;
@@ -730,8 +730,7 @@ mlx5dr_bwc_matcher_rehash(struct mlx5dr_bwc_matcher 
*bwc_matcher, bool rehash_si
         */
        num_of_rules = rte_atomic_load_explicit(&bwc_matcher->num_of_rules,
                                                rte_memory_order_relaxed);
-       if (rehash_size &&
-           !mlx5dr_bwc_matcher_rehash_size_needed(bwc_matcher, num_of_rules))
+       if (!mlx5dr_bwc_matcher_rehash_size_needed(bwc_matcher, num_of_rules))
                return 0;
 
        /* Now we're done all the checking - do the rehash:
@@ -748,6 +747,19 @@ mlx5dr_bwc_matcher_rehash(struct mlx5dr_bwc_matcher 
*bwc_matcher, bool rehash_si
        return mlx5dr_bwc_matcher_move(bwc_matcher);
 }
 
+static int
+mlx5dr_bwc_matcher_rehash_at(struct mlx5dr_bwc_matcher *bwc_matcher)
+{
+       /* Rehash by action template doesn't require any additional checking.
+        * The bwc_matcher already contains the new action template.
+        * Just do the usual rehash:
+        *  - create new matcher
+        *  - move all the rules to the new matcher
+        *  - destroy the old matcher
+        */
+       return mlx5dr_bwc_matcher_move(bwc_matcher);
+}
+
 static struct mlx5dr_bwc_rule *
 mlx5dr_bwc_rule_create_root(struct mlx5dr_bwc_matcher *bwc_matcher,
                            const struct rte_flow_item flow_items[],
@@ -775,7 +787,6 @@ mlx5dr_bwc_rule_create_hws(struct mlx5dr_bwc_matcher 
*bwc_matcher,
        struct mlx5dr_bwc_rule *bwc_rule = NULL;
        struct mlx5dr_rule_attr rule_attr;
        rte_spinlock_t *queue_lock;
-       bool rehash_size = false;
        uint32_t num_of_rules;
        uint16_t idx;
        int at_idx;
@@ -791,7 +802,7 @@ mlx5dr_bwc_rule_create_hws(struct mlx5dr_bwc_matcher 
*bwc_matcher,
 
        /* check if rehash needed due to missing action template */
        at_idx = mlx5dr_bwc_matcher_find_at(bwc_matcher, rule_actions);
-       if (at_idx < 0) {
+       if (unlikely(at_idx < 0)) {
                /* we need to extend BWC matcher action templates array */
                rte_spinlock_unlock(queue_lock);
                mlx5dr_bwc_lock_all_queues(ctx);
@@ -810,13 +821,22 @@ mlx5dr_bwc_rule_create_hws(struct mlx5dr_bwc_matcher 
*bwc_matcher,
                ret = mlx5dr_matcher_attach_at(bwc_matcher->matcher,
                                               bwc_matcher->at[at_idx]);
                if (unlikely(ret)) {
-                       mlx5dr_action_template_destroy(bwc_matcher->at[at_idx]);
-                       bwc_matcher->at[at_idx] = NULL;
-                       bwc_matcher->num_of_at--;
-
-                       mlx5dr_bwc_unlock_all_queues(ctx);
-                       DR_LOG(ERR, "BWC rule: failed attaching action template 
- %d", ret);
-                       return NULL;
+                       /* Action template attach failed, possibly due to
+                        * requiring more action STEs.
+                        * Need to attempt creating new matcher with all
+                        * the action templates, including the new one.
+                        */
+                       ret = mlx5dr_bwc_matcher_rehash_at(bwc_matcher);
+                       if (unlikely(ret)) {
+                               
mlx5dr_action_template_destroy(bwc_matcher->at[at_idx]);
+                               bwc_matcher->at[at_idx] = NULL;
+                               bwc_matcher->num_of_at--;
+
+                               mlx5dr_bwc_unlock_all_queues(ctx);
+
+                               DR_LOG(ERR, "BWC rule insertion: rehash AT 
failed - %d", ret);
+                               return NULL;
+                       }
                }
 
                mlx5dr_bwc_unlock_all_queues(ctx);
@@ -826,9 +846,22 @@ mlx5dr_bwc_rule_create_hws(struct mlx5dr_bwc_matcher 
*bwc_matcher,
        /* check if number of rules require rehash */
        num_of_rules = rte_atomic_load_explicit(&bwc_matcher->num_of_rules,
                                                rte_memory_order_relaxed);
-       if (mlx5dr_bwc_matcher_rehash_size_needed(bwc_matcher, num_of_rules)) {
-               rehash_size = true;
-               goto rehash;
+       if (unlikely(mlx5dr_bwc_matcher_rehash_size_needed(bwc_matcher, 
num_of_rules))) {
+               rte_spinlock_unlock(queue_lock);
+
+               mlx5dr_bwc_lock_all_queues(ctx);
+               ret = mlx5dr_bwc_matcher_rehash_size(bwc_matcher);
+               mlx5dr_bwc_unlock_all_queues(ctx);
+
+               if (ret) {
+                       DR_LOG(ERR, "BWC rule insertion: rehash size [%d -> %d] 
failed - %d",
+                              bwc_matcher->size_log - 
MLX5DR_BWC_MATCHER_SIZE_LOG_STEP,
+                              bwc_matcher->size_log,
+                              ret);
+                       return NULL;
+               }
+
+               rte_spinlock_lock(queue_lock);
        }
 
        bwc_rule = mlx5dr_bwc_rule_create_hws_sync(bwc_matcher,
@@ -847,14 +880,13 @@ mlx5dr_bwc_rule_create_hws(struct mlx5dr_bwc_matcher 
*bwc_matcher,
         * It could be because there was collision, or some other problem.
         * If we don't dive deeper than API, the only thing we know is that
         * the status of completion is RTE_FLOW_OP_ERROR.
-        * Try rehash and insert rule again - last chance.
+        * Try rehash by size and insert rule again - last chance.
         */
 
-rehash:
        rte_spinlock_unlock(queue_lock);
 
        mlx5dr_bwc_lock_all_queues(ctx);
-       ret = mlx5dr_bwc_matcher_rehash(bwc_matcher, rehash_size);
+       ret = mlx5dr_bwc_matcher_rehash_size(bwc_matcher);
        mlx5dr_bwc_unlock_all_queues(ctx);
 
        if (ret) {
diff --git a/drivers/net/mlx5/hws/mlx5dr_matcher.c 
b/drivers/net/mlx5/hws/mlx5dr_matcher.c
index 2a84145566..aa5ab96d23 100644
--- a/drivers/net/mlx5/hws/mlx5dr_matcher.c
+++ b/drivers/net/mlx5/hws/mlx5dr_matcher.c
@@ -1347,7 +1347,7 @@ int mlx5dr_matcher_attach_at(struct mlx5dr_matcher 
*matcher,
        int ret;
 
        if (!matcher->attr.max_num_of_at_attach) {
-               DR_LOG(ERR, "Num of current at (%d) exceed allowed value",
+               DR_LOG(DEBUG, "Num of current at (%d) exceed allowed value",
                       matcher->num_of_at);
                rte_errno = ENOTSUP;
                return -rte_errno;
@@ -1359,7 +1359,7 @@ int mlx5dr_matcher_attach_at(struct mlx5dr_matcher 
*matcher,
 
        required_stes = at->num_of_action_stes - (!is_jumbo || at->only_term);
        if (matcher->action_ste.max_stes < required_stes) {
-               DR_LOG(ERR, "Required STEs [%d] exceeds initial action template 
STE [%d]",
+               DR_LOG(DEBUG, "Required STEs [%d] exceeds initial action 
template STE [%d]",
                       required_stes, matcher->action_ste.max_stes);
                rte_errno = ENOMEM;
                return -rte_errno;
-- 
2.25.1

Reply via email to