goto free_ste_icm_pool;
        }
 
+       ret = mlx5dr_send_info_pool_create(dmn);
+       if (ret) {
+               mlx5dr_err(dmn, "Couldn't create send info pool\n");
+               goto free_action_icm_pool;
+       }
+
        return 0;
 
+free_action_icm_pool:
+       mlx5dr_icm_pool_destroy(dmn->action_icm_pool);
 free_ste_icm_pool:
        mlx5dr_icm_pool_destroy(dmn->ste_icm_pool);
        return ret;
 
 static void dr_domain_uninit_mem_resources(struct mlx5dr_domain *dmn)
 {
+       mlx5dr_send_info_pool_destroy(dmn);
        mlx5dr_icm_pool_destroy(dmn->action_icm_pool);
        mlx5dr_icm_pool_destroy(dmn->ste_icm_pool);
 }
 
 #define DR_RULE_MAX_STES_OPTIMIZED 5
 #define DR_RULE_MAX_STE_CHAIN_OPTIMIZED (DR_RULE_MAX_STES_OPTIMIZED + DR_ACTION_MAX_STES)
 
-static int dr_rule_append_to_miss_list(struct mlx5dr_ste_ctx *ste_ctx,
+static int dr_rule_append_to_miss_list(struct mlx5dr_domain *dmn,
+                                      enum mlx5dr_domain_nic_type nic_type,
                                       struct mlx5dr_ste *new_last_ste,
                                       struct list_head *miss_list,
                                       struct list_head *send_list)
 {
+       struct mlx5dr_ste_ctx *ste_ctx = dmn->ste_ctx;
        struct mlx5dr_ste_send_info *ste_info_last;
        struct mlx5dr_ste *last_ste;
 
        last_ste = list_last_entry(miss_list, struct mlx5dr_ste, miss_list_node);
        WARN_ON(!last_ste);
 
-       ste_info_last = kzalloc(sizeof(*ste_info_last), GFP_KERNEL);
+       ste_info_last = mlx5dr_send_info_alloc(dmn, nic_type);
        if (!ste_info_last)
                return -ENOMEM;
 
                goto out;
 
 out:
-       kfree(ste_info);
+       mlx5dr_send_info_free(ste_info);
        return ret;
 }
 
        new_ste->htbl->chunk->miss_list = mlx5dr_ste_get_miss_list(col_ste);
 
        /* Update the previous from the list */
-       ret = dr_rule_append_to_miss_list(dmn->ste_ctx, new_ste,
-                                         mlx5dr_ste_get_miss_list(col_ste),
+       ret = dr_rule_append_to_miss_list(dmn, nic_matcher->nic_tbl->nic_dmn->type,
+                                         new_ste, mlx5dr_ste_get_miss_list(col_ste),
                                          update_list);
        if (ret) {
                mlx5dr_dbg(dmn, "Failed update dup entry\n");
        new_htbl->ctrl.num_of_valid_entries++;
 
        if (use_update_list) {
-               ste_info = kzalloc(sizeof(*ste_info), GFP_KERNEL);
+               ste_info = mlx5dr_send_info_alloc(dmn,
+                                                 nic_matcher->nic_tbl->nic_dmn->type);
                if (!ste_info)
                        goto err_exit;
 
        nic_matcher = nic_rule->nic_matcher;
        nic_dmn = nic_matcher->nic_tbl->nic_dmn;
 
-       ste_info = kzalloc(sizeof(*ste_info), GFP_KERNEL);
+       ste_info = mlx5dr_send_info_alloc(dmn,
+                                         nic_matcher->nic_tbl->nic_dmn->type);
        if (!ste_info)
                return NULL;
 
        list_for_each_entry_safe(del_ste_info, tmp_ste_info,
                                 &rehash_table_send_list, send_list) {
                list_del(&del_ste_info->send_list);
-               kfree(del_ste_info);
+               mlx5dr_send_info_free(del_ste_info);
        }
 
 free_new_htbl:
        mlx5dr_ste_htbl_free(new_htbl);
 free_ste_info:
-       kfree(ste_info);
+       mlx5dr_send_info_free(ste_info);
        mlx5dr_info(dmn, "Failed creating rehash table\n");
        return NULL;
 }
                         struct list_head *send_list)
 {
        struct mlx5dr_domain *dmn = matcher->tbl->dmn;
-       struct mlx5dr_ste_ctx *ste_ctx = dmn->ste_ctx;
        struct mlx5dr_ste_send_info *ste_info;
        struct mlx5dr_ste *new_ste;
 
-       ste_info = kzalloc(sizeof(*ste_info), GFP_KERNEL);
+       ste_info = mlx5dr_send_info_alloc(dmn,
+                                         nic_matcher->nic_tbl->nic_dmn->type);
        if (!ste_info)
                return NULL;
 
        if (!new_ste)
                goto free_send_info;
 
-       if (dr_rule_append_to_miss_list(ste_ctx, new_ste,
-                                       miss_list, send_list)) {
+       if (dr_rule_append_to_miss_list(dmn, nic_matcher->nic_tbl->nic_dmn->type,
+                                       new_ste, miss_list, send_list)) {
                mlx5dr_dbg(dmn, "Failed to update prev miss_list\n");
                goto err_exit;
        }
 err_exit:
        mlx5dr_ste_free(new_ste, matcher, nic_matcher);
 free_send_info:
-       kfree(ste_info);
+       mlx5dr_send_info_free(ste_info);
        return NULL;
 }
 
                list_add_tail(&action_ste->miss_list_node,
                              mlx5dr_ste_get_miss_list(action_ste));
 
-               ste_info_arr[k] = kzalloc(sizeof(*ste_info_arr[k]),
-                                         GFP_KERNEL);
+               ste_info_arr[k] = mlx5dr_send_info_alloc(dmn,
+                                                        nic_matcher->nic_tbl->nic_dmn->type);
                if (!ste_info_arr[k])
                        goto err_exit;
 
 
        ste->ste_chain_location = ste_location;
 
-       ste_info = kzalloc(sizeof(*ste_info), GFP_KERNEL);
+       ste_info = mlx5dr_send_info_alloc(dmn,
+                                         nic_matcher->nic_tbl->nic_dmn->type);
        if (!ste_info)
                goto clean_ste_setting;
 
        return 0;
 
 clean_ste_info:
-       kfree(ste_info);
+       mlx5dr_send_info_free(ste_info);
 clean_ste_setting:
        list_del_init(&ste->miss_list_node);
        mlx5dr_htbl_put(cur_htbl);
        /* Clean all ste_info's */
        list_for_each_entry_safe(ste_info, tmp_ste_info, &send_ste_list, send_list) {
                list_del(&ste_info->send_list);
-               kfree(ste_info);
+               mlx5dr_send_info_free(ste_info);
        }
 
 remove_from_nic_tbl:
 
 #define QUEUE_SIZE 128
 #define SIGNAL_PER_DIV_QUEUE 16
 #define TH_NUMS_TO_DRAIN 2
+#define DR_SEND_INFO_POOL_SIZE 1000
 
 enum { CQ_OK = 0, CQ_EMPTY = -1, CQ_POLL_ERR = -2 };
 
        u8 isolate_vl_tc:1;
 };
 
+struct mlx5dr_send_info_pool_obj {
+       struct mlx5dr_ste_send_info ste_send_info;
+       struct mlx5dr_send_info_pool *pool;
+       struct list_head list_node;
+};
+
+struct mlx5dr_send_info_pool {
+       struct list_head free_list;
+};
+
+static int dr_send_info_pool_fill(struct mlx5dr_send_info_pool *pool)
+{
+       struct mlx5dr_send_info_pool_obj *pool_obj, *tmp_pool_obj;
+       int i;
+
+       for (i = 0; i < DR_SEND_INFO_POOL_SIZE; i++) {
+               pool_obj = kzalloc(sizeof(*pool_obj), GFP_KERNEL);
+               if (!pool_obj)
+                       goto clean_pool;
+
+               pool_obj->pool = pool;
+               list_add_tail(&pool_obj->list_node, &pool->free_list);
+       }
+
+       return 0;
+
+clean_pool:
+       list_for_each_entry_safe(pool_obj, tmp_pool_obj, &pool->free_list, list_node) {
+               list_del(&pool_obj->list_node);
+               kfree(pool_obj);
+       }
+
+       return -ENOMEM;
+}
+
+static void dr_send_info_pool_destroy(struct mlx5dr_send_info_pool *pool)
+{
+       struct mlx5dr_send_info_pool_obj *pool_obj, *tmp_pool_obj;
+
+       list_for_each_entry_safe(pool_obj, tmp_pool_obj, &pool->free_list, list_node) {
+               list_del(&pool_obj->list_node);
+               kfree(pool_obj);
+       }
+
+       kfree(pool);
+}
+
+void mlx5dr_send_info_pool_destroy(struct mlx5dr_domain *dmn)
+{
+       dr_send_info_pool_destroy(dmn->send_info_pool_tx);
+       dr_send_info_pool_destroy(dmn->send_info_pool_rx);
+}
+
+static struct mlx5dr_send_info_pool *dr_send_info_pool_create(void)
+{
+       struct mlx5dr_send_info_pool *pool;
+       int ret;
+
+       pool = kzalloc(sizeof(*pool), GFP_KERNEL);
+       if (!pool)
+               return NULL;
+
+       INIT_LIST_HEAD(&pool->free_list);
+
+       ret = dr_send_info_pool_fill(pool);
+       if (ret) {
+               kfree(pool);
+               return NULL;
+       }
+
+       return pool;
+}
+
+int mlx5dr_send_info_pool_create(struct mlx5dr_domain *dmn)
+{
+       dmn->send_info_pool_rx = dr_send_info_pool_create();
+       if (!dmn->send_info_pool_rx)
+               return -ENOMEM;
+
+       dmn->send_info_pool_tx = dr_send_info_pool_create();
+       if (!dmn->send_info_pool_tx) {
+               dr_send_info_pool_destroy(dmn->send_info_pool_rx);
+               return -ENOMEM;
+       }
+
+       return 0;
+}
+
+struct mlx5dr_ste_send_info
+*mlx5dr_send_info_alloc(struct mlx5dr_domain *dmn,
+                       enum mlx5dr_domain_nic_type nic_type)
+{
+       struct mlx5dr_send_info_pool_obj *pool_obj;
+       struct mlx5dr_send_info_pool *pool;
+       int ret;
+
+       pool = nic_type == DR_DOMAIN_NIC_TYPE_RX ? dmn->send_info_pool_rx :
+                                                  dmn->send_info_pool_tx;
+
+       if (unlikely(list_empty(&pool->free_list))) {
+               ret = dr_send_info_pool_fill(pool);
+               if (ret)
+                       return NULL;
+       }
+
+       pool_obj = list_first_entry_or_null(&pool->free_list,
+                                           struct mlx5dr_send_info_pool_obj,
+                                           list_node);
+
+       if (likely(pool_obj)) {
+               list_del_init(&pool_obj->list_node);
+       } else {
+               WARN_ONCE(!pool_obj, "Failed getting ste send info obj from pool");
+               return NULL;
+       }
+
+       return &pool_obj->ste_send_info;
+}
+
+void mlx5dr_send_info_free(struct mlx5dr_ste_send_info *ste_send_info)
+{
+       struct mlx5dr_send_info_pool_obj *pool_obj;
+
+       pool_obj = container_of(ste_send_info,
+                               struct mlx5dr_send_info_pool_obj,
+                               ste_send_info);
+
+       list_add(&pool_obj->list_node, &pool_obj->pool->free_list);
+}
+
 static int dr_parse_cqe(struct mlx5dr_cq *dr_cq, struct mlx5_cqe64 *cqe64)
 {
        unsigned int idx;
 
 struct mlx5dr_rule_rx_tx;
 struct mlx5dr_matcher_rx_tx;
 struct mlx5dr_ste_ctx;
+struct mlx5dr_send_info_pool;
 
 struct mlx5dr_ste {
        /* refcount: indicates the num of rules that using this ste */
        refcount_t refcount;
        struct mlx5dr_icm_pool *ste_icm_pool;
        struct mlx5dr_icm_pool *action_icm_pool;
+       struct mlx5dr_send_info_pool *send_info_pool_rx;
+       struct mlx5dr_send_info_pool *send_info_pool_tx;
        struct mlx5dr_send_ring *send_ring;
        struct mlx5dr_domain_info info;
        struct xarray csum_fts_xa;
 int mlx5dr_send_postsend_action(struct mlx5dr_domain *dmn,
                                struct mlx5dr_action *action);
 
+int mlx5dr_send_info_pool_create(struct mlx5dr_domain *dmn);
+void mlx5dr_send_info_pool_destroy(struct mlx5dr_domain *dmn);
+struct mlx5dr_ste_send_info *mlx5dr_send_info_alloc(struct mlx5dr_domain *dmn,
+                                                   enum mlx5dr_domain_nic_type nic_type);
+void mlx5dr_send_info_free(struct mlx5dr_ste_send_info *ste_send_info);
+
 struct mlx5dr_cmd_ft_info {
        u32 id;
        u16 vport;