struct mlx5hws_cmd_rtc_create_attr rtc_attr = {0};
        struct mlx5hws_match_template *mt = matcher->mt;
        struct mlx5hws_context *ctx = matcher->tbl->ctx;
+       union mlx5hws_matcher_size *size_rx, *size_tx;
        struct mlx5hws_table *tbl = matcher->tbl;
        u32 obj_id;
        int ret;
 
-       rtc_attr.log_size = attr->table.sz_row_log;
-       rtc_attr.log_depth = attr->table.sz_col_log;
+       size_rx = &attr->size[MLX5HWS_MATCHER_SIZE_TYPE_RX];
+       size_tx = &attr->size[MLX5HWS_MATCHER_SIZE_TYPE_TX];
+
+       rtc_attr.log_size = size_rx->table.sz_row_log;
+       rtc_attr.log_depth = size_rx->table.sz_col_log;
        rtc_attr.is_frst_jumbo = mlx5hws_matcher_mt_is_jumbo(mt);
        rtc_attr.is_scnd_range = 0;
        rtc_attr.miss_ft_id = matcher->end_ft_id;
        }
 
        if (tbl->type == MLX5HWS_TABLE_TYPE_FDB) {
+               rtc_attr.log_size = size_tx->table.sz_row_log;
+               rtc_attr.log_depth = size_tx->table.sz_col_log;
                rtc_attr.ste_base = matcher->match_ste.ste_1_base;
                rtc_attr.table_type = mlx5hws_table_get_res_fw_ft_type(tbl->type, true);
 
                          struct mlx5hws_matcher *matcher)
 {
        struct mlx5hws_matcher_attr *attr = &matcher->attr;
+       struct mlx5hws_context *ctx = matcher->tbl->ctx;
+       union mlx5hws_matcher_size *size;
+       int i;
 
-       if (attr->table.sz_col_log > caps->rtc_log_depth_max) {
-               mlx5hws_err(matcher->tbl->ctx, "Matcher depth exceeds limit %d\n",
-                           caps->rtc_log_depth_max);
-               return -EOPNOTSUPP;
-       }
+       for (i = 0; i < 2; i++) {
+               size = &attr->size[i];
 
-       if (attr->table.sz_col_log + attr->table.sz_row_log > caps->ste_alloc_log_max) {
-               mlx5hws_err(matcher->tbl->ctx, "Total matcher size exceeds limit %d\n",
-                           caps->ste_alloc_log_max);
-               return -EOPNOTSUPP;
-       }
+               if (size->table.sz_col_log > caps->rtc_log_depth_max) {
+                       mlx5hws_err(ctx, "Matcher depth exceeds limit %d\n",
+                                   caps->rtc_log_depth_max);
+                       return -EOPNOTSUPP;
+               }
 
-       if (attr->table.sz_col_log + attr->table.sz_row_log < caps->ste_alloc_log_gran) {
-               mlx5hws_err(matcher->tbl->ctx, "Total matcher size below limit %d\n",
-                           caps->ste_alloc_log_gran);
-               return -EOPNOTSUPP;
+               if (size->table.sz_col_log + size->table.sz_row_log >
+                   caps->ste_alloc_log_max) {
+                       mlx5hws_err(ctx,
+                                   "Total matcher size exceeds limit %d\n",
+                                   caps->ste_alloc_log_max);
+                       return -EOPNOTSUPP;
+               }
+
+               if (size->table.sz_col_log + size->table.sz_row_log <
+                   caps->ste_alloc_log_gran) {
+                       mlx5hws_err(ctx, "Total matcher size below limit %d\n",
+                                   caps->ste_alloc_log_gran);
+                       return -EOPNOTSUPP;
+               }
        }
 
        return 0;
 {
        struct mlx5hws_cmd_ste_create_attr ste_attr = {};
        struct mlx5hws_context *ctx = matcher->tbl->ctx;
+       union mlx5hws_matcher_size *size;
        int ret;
 
        /* Calculate match, range and hash definers */
 
        /* Create an STE range each for RX and TX. */
        ste_attr.table_type = FS_FT_FDB_RX;
+       size = &matcher->attr.size[MLX5HWS_MATCHER_SIZE_TYPE_RX];
        ste_attr.log_obj_range =
                matcher->attr.optimize_flow_src ==
-                               MLX5HWS_MATCHER_FLOW_SRC_VPORT ?
-                               0 : matcher->attr.table.sz_col_log +
-                                   matcher->attr.table.sz_row_log;
+                       MLX5HWS_MATCHER_FLOW_SRC_VPORT ?
+                       0 : size->table.sz_col_log + size->table.sz_row_log;
 
        ret = mlx5hws_cmd_ste_create(ctx->mdev, &ste_attr,
                                     &matcher->match_ste.ste_0_base);
        }
 
        ste_attr.table_type = FS_FT_FDB_TX;
+       size = &matcher->attr.size[MLX5HWS_MATCHER_SIZE_TYPE_TX];
        ste_attr.log_obj_range =
                matcher->attr.optimize_flow_src ==
-                               MLX5HWS_MATCHER_FLOW_SRC_WIRE ?
-                               0 : matcher->attr.table.sz_col_log +
-                                   matcher->attr.table.sz_row_log;
+                       MLX5HWS_MATCHER_FLOW_SRC_WIRE ?
+                       0 : size->table.sz_col_log + size->table.sz_row_log;
 
        ret = mlx5hws_cmd_ste_create(ctx->mdev, &ste_attr,
                                     &matcher->match_ste.ste_1_base);
 {
        struct mlx5hws_matcher_attr *attr = &matcher->attr;
        struct mlx5hws_context *ctx = matcher->tbl->ctx;
+       union mlx5hws_matcher_size *size_rx, *size_tx;
+
+       size_rx = &matcher->attr.size[MLX5HWS_MATCHER_SIZE_TYPE_RX];
+       size_tx = &matcher->attr.size[MLX5HWS_MATCHER_SIZE_TYPE_TX];
 
        switch (attr->insert_mode) {
        case MLX5HWS_MATCHER_INSERT_BY_HASH:
                break;
 
        case MLX5HWS_MATCHER_INSERT_BY_INDEX:
-               if (attr->table.sz_col_log) {
+               if (size_rx->table.sz_col_log || size_tx->table.sz_col_log) {
                        mlx5hws_err(ctx, "Matcher with INSERT_BY_INDEX supports only Nx1 table size\n");
                        return -EOPNOTSUPP;
                }
                                return -EOPNOTSUPP;
                        }
 
-                       if (attr->table.sz_row_log > MLX5_IFC_RTC_LINEAR_LOOKUP_TBL_LOG_MAX) {
+                       if (size_rx->table.sz_row_log >
+                               MLX5_IFC_RTC_LINEAR_LOOKUP_TBL_LOG_MAX ||
+                           size_tx->table.sz_row_log >
+                               MLX5_IFC_RTC_LINEAR_LOOKUP_TBL_LOG_MAX) {
                                mlx5hws_err(ctx, "Matcher with linear distribute: rows exceed limit %d",
                                            MLX5_IFC_RTC_LINEAR_LOOKUP_TBL_LOG_MAX);
                                return -EOPNOTSUPP;
                         struct mlx5hws_matcher *matcher)
 {
        struct mlx5hws_matcher_attr *attr = &matcher->attr;
+       union mlx5hws_matcher_size *size_rx, *size_tx;
+
+       size_rx = &attr->size[MLX5HWS_MATCHER_SIZE_TYPE_RX];
+       size_tx = &attr->size[MLX5HWS_MATCHER_SIZE_TYPE_TX];
 
        if (hws_matcher_validate_insert_mode(caps, matcher))
                return -EOPNOTSUPP;
 
        /* Convert number of rules to the required depth */
        if (attr->mode == MLX5HWS_MATCHER_RESOURCE_MODE_RULE &&
-           attr->insert_mode == MLX5HWS_MATCHER_INSERT_BY_HASH)
-               attr->table.sz_col_log = hws_matcher_rules_to_tbl_depth(attr->rule.num_log);
+           attr->insert_mode == MLX5HWS_MATCHER_INSERT_BY_HASH) {
+               size_rx->table.sz_col_log =
+                       hws_matcher_rules_to_tbl_depth(size_rx->rule.num_log);
+               size_tx->table.sz_col_log =
+                       hws_matcher_rules_to_tbl_depth(size_tx->rule.num_log);
+       }
 
        matcher->flags |= attr->resizable ? MLX5HWS_MATCHER_FLAGS_RESIZABLE : 0;
        matcher->flags |= attr->isolated_matcher_end_ft_id ?
 hws_matcher_create_col_matcher(struct mlx5hws_matcher *matcher)
 {
        struct mlx5hws_context *ctx = matcher->tbl->ctx;
+       union mlx5hws_matcher_size *size_rx, *size_tx;
        struct mlx5hws_matcher *col_matcher;
-       int ret;
+       int i, ret;
+
+       size_rx = &matcher->attr.size[MLX5HWS_MATCHER_SIZE_TYPE_RX];
+       size_tx = &matcher->attr.size[MLX5HWS_MATCHER_SIZE_TYPE_TX];
 
        if (matcher->attr.mode != MLX5HWS_MATCHER_RESOURCE_MODE_RULE ||
            matcher->attr.insert_mode == MLX5HWS_MATCHER_INSERT_BY_INDEX)
                return 0;
 
-       if (!hws_matcher_requires_col_tbl(matcher->attr.rule.num_log))
+       if (!hws_matcher_requires_col_tbl(size_rx->rule.num_log) &&
+           !hws_matcher_requires_col_tbl(size_tx->rule.num_log))
                return 0;
 
        col_matcher = kzalloc(sizeof(*matcher), GFP_KERNEL);
        col_matcher->flags |= MLX5HWS_MATCHER_FLAGS_COLLISION;
        col_matcher->attr.mode = MLX5HWS_MATCHER_RESOURCE_MODE_HTABLE;
        col_matcher->attr.optimize_flow_src = matcher->attr.optimize_flow_src;
-       col_matcher->attr.table.sz_row_log = matcher->attr.rule.num_log;
-       col_matcher->attr.table.sz_col_log = MLX5HWS_MATCHER_ASSURED_COL_TBL_DEPTH;
-       if (col_matcher->attr.table.sz_row_log > MLX5HWS_MATCHER_ASSURED_ROW_RATIO)
-               col_matcher->attr.table.sz_row_log -= MLX5HWS_MATCHER_ASSURED_ROW_RATIO;
+       for (i = 0; i < 2; i++) {
+               union mlx5hws_matcher_size *dst = &col_matcher->attr.size[i];
+               union mlx5hws_matcher_size *src = &matcher->attr.size[i];
+
+               dst->table.sz_row_log = src->rule.num_log;
+               dst->table.sz_col_log = MLX5HWS_MATCHER_ASSURED_COL_TBL_DEPTH;
+               if (dst->table.sz_row_log > MLX5HWS_MATCHER_ASSURED_ROW_RATIO)
+                       dst->table.sz_row_log -=
+                               MLX5HWS_MATCHER_ASSURED_ROW_RATIO;
+       }
 
        col_matcher->attr.max_num_of_at_attach = matcher->attr.max_num_of_at_attach;
        col_matcher->attr.isolated_matcher_end_ft_id =