LCOV - code coverage report
Current view: top level - drivers/regex/mlx5 - mlx5_rxp.c (source / functions) Hit Total Coverage
Test: Code coverage Lines: 0 139 0.0 %
Date: 2025-01-02 22:41:34 Functions: 0 8 0.0 %
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 0 75 0.0 %

           Branch data     Line data    Source code
       1                 :            : /* SPDX-License-Identifier: BSD-3-Clause
       2                 :            :  * Copyright 2020 Mellanox Technologies, Ltd
       3                 :            :  */
       4                 :            : 
       5                 :            : #include <rte_log.h>
       6                 :            : #include <rte_errno.h>
       7                 :            : #include <rte_malloc.h>
       8                 :            : #include <rte_regexdev.h>
       9                 :            : #include <rte_regexdev_core.h>
      10                 :            : #include <rte_regexdev_driver.h>
      11                 :            : #include <sys/mman.h>
      12                 :            : 
      13                 :            : #include <mlx5_glue.h>
      14                 :            : #include <mlx5_devx_cmds.h>
      15                 :            : #include <mlx5_prm.h>
      16                 :            : #include <mlx5_common_os.h>
      17                 :            : 
      18                 :            : #include "mlx5_regex.h"
      19                 :            : #include "mlx5_regex_utils.h"
      20                 :            : #include "mlx5_rxp.h"
      21                 :            : 
      22                 :            : #define MLX5_REGEX_MAX_MATCHES MLX5_RXP_MAX_MATCHES
      23                 :            : #define MLX5_REGEX_MAX_PAYLOAD_SIZE MLX5_RXP_MAX_JOB_LENGTH
      24                 :            : #define MLX5_REGEX_MAX_RULES_PER_GROUP UINT32_MAX
      25                 :            : #define MLX5_REGEX_MAX_GROUPS MLX5_RXP_MAX_SUBSETS
      26                 :            : 
      27                 :            : #define MLX5_REGEX_RXP_ROF2_LINE_LEN 34
      28                 :            : 
      29                 :            : const uint64_t combined_rof_tag = 0xff52544424a52475;
      30                 :            : 
      31                 :            : /* Private Declarations */
      32                 :            : static int
      33                 :            : rxp_create_mkey(struct mlx5_regex_priv *priv, void *ptr, size_t size,
      34                 :            :         uint32_t access, struct mlx5_regex_mkey *mkey);
      35                 :            : static inline void
      36                 :            : rxp_destroy_mkey(struct mlx5_regex_mkey *mkey);
      37                 :            : 
      38                 :            : int
      39                 :          0 : mlx5_regex_info_get(struct rte_regexdev *dev __rte_unused,
      40                 :            :                     struct rte_regexdev_info *info)
      41                 :            : {
      42                 :          0 :         info->max_matches = MLX5_REGEX_MAX_MATCHES;
      43                 :          0 :         info->max_payload_size = MLX5_REGEX_MAX_PAYLOAD_SIZE;
      44                 :          0 :         info->max_rules_per_group = MLX5_REGEX_MAX_RULES_PER_GROUP;
      45                 :          0 :         info->max_groups = MLX5_REGEX_MAX_GROUPS;
      46                 :          0 :         info->regexdev_capa = RTE_REGEXDEV_SUPP_PCRE_GREEDY_F |
      47                 :            :                               RTE_REGEXDEV_CAPA_QUEUE_PAIR_OOS_F;
      48                 :          0 :         info->rule_flags = 0;
      49                 :          0 :         info->max_queue_pairs = UINT16_MAX;
      50                 :          0 :         info->max_segs = mlx5_regexdev_max_segs_get();
      51                 :          0 :         return 0;
      52                 :            : }
      53                 :            : 
      54                 :            : static int
      55                 :          0 : rxp_create_mkey(struct mlx5_regex_priv *priv, void *ptr, size_t size,
      56                 :            :         uint32_t access, struct mlx5_regex_mkey *mkey)
      57                 :            : {
      58                 :            :         struct mlx5_devx_mkey_attr mkey_attr;
      59                 :            : 
      60                 :            :         /* Register the memory. */
      61                 :          0 :         mkey->umem = mlx5_glue->devx_umem_reg(priv->cdev->ctx, ptr, size, access);
      62         [ #  # ]:          0 :         if (!mkey->umem) {
      63                 :          0 :                 DRV_LOG(ERR, "Failed to register memory!");
      64                 :          0 :                 return -ENODEV;
      65                 :            :         }
      66                 :            :         /* Create mkey */
      67                 :          0 :         mkey_attr = (struct mlx5_devx_mkey_attr) {
      68                 :          0 :                 .addr = (uintptr_t)ptr,
      69                 :            :                 .size = (uint32_t)size,
      70                 :            :                 .umem_id = mlx5_os_get_umem_id(mkey->umem),
      71                 :            :                 .pg_access = 1,
      72                 :            :                 .umr_en = 0,
      73                 :            :         };
      74                 :            : #ifdef HAVE_IBV_FLOW_DV_SUPPORT
      75                 :          0 :         mkey_attr.pd = priv->cdev->pdn;
      76                 :            : #endif
      77                 :          0 :         mkey->mkey = mlx5_devx_cmd_mkey_create(priv->cdev->ctx, &mkey_attr);
      78         [ #  # ]:          0 :         if (!mkey->mkey) {
      79                 :          0 :                 DRV_LOG(ERR, "Failed to create direct mkey!");
      80                 :          0 :                 return -ENODEV;
      81                 :            :         }
      82                 :            :         return 0;
      83                 :            : }
      84                 :            : 
      85                 :            : static inline void
      86                 :          0 : rxp_destroy_mkey(struct mlx5_regex_mkey *mkey)
      87                 :            : {
      88         [ #  # ]:          0 :         if (mkey->mkey)
      89                 :          0 :                 claim_zero(mlx5_devx_cmd_destroy(mkey->mkey));
      90         [ #  # ]:          0 :         if (mkey->umem)
      91                 :          0 :                 claim_zero(mlx5_glue->devx_umem_dereg(mkey->umem));
      92                 :          0 : }
      93                 :            : 
      94                 :            : int
      95                 :          0 : mlx5_regex_get_rxp_vers(uint32_t regexp_version, uint32_t *target_rxp_vers)
      96                 :            : {
      97                 :            :         int ret = 0;
      98      [ #  #  # ]:          0 :         switch (regexp_version) {
      99                 :          0 :         case MLX5_RXP_BF2_IDENTIFIER:
     100                 :          0 :                 *target_rxp_vers = MLX5_RXP_BF2_ROF_VERSION_STRING;
     101                 :          0 :                 break;
     102                 :          0 :         case MLX5_RXP_BF3_IDENTIFIER:
     103                 :          0 :                 *target_rxp_vers = MLX5_RXP_BF3_ROF_VERSION_STRING;
     104                 :          0 :                 break;
     105                 :          0 :         default:
     106                 :          0 :                 DRV_LOG(ERR, "Unsupported rxp version: %u", regexp_version);
     107                 :            :                 ret = -EINVAL;
     108                 :          0 :                 break;
     109                 :            :         }
     110                 :          0 :         return ret;
     111                 :            : }
     112                 :            : 
     113                 :            : int
     114                 :          0 : mlx5_regex_check_rof_version(uint32_t combined_rof_vers)
     115                 :            : {
     116                 :            :         int ret = 0;
     117                 :            :         /* Check if combined rof version is supported */
     118         [ #  # ]:          0 :         switch (combined_rof_vers) {
     119                 :            :         case 1:
     120                 :            :                 break;
     121                 :          0 :         default:
     122                 :          0 :                 DRV_LOG(ERR, "Unsupported combined rof version: %u",
     123                 :            :                 combined_rof_vers);
     124                 :            :                 ret = -EINVAL;
     125                 :          0 :                 break;
     126                 :            :         }
     127                 :          0 :         return ret;
     128                 :            : }
     129                 :            : 
     130                 :            : int
     131                 :          0 : mlx5_regex_parse_rules_db(struct mlx5_regex_priv *priv,
     132                 :            :                         const char **rules_db, uint32_t *rules_db_len)
     133                 :            : {
     134                 :            :         int i = 0;
     135                 :            :         uint32_t j = 0;
     136                 :            :         int ret = 0;
     137                 :            :         bool combined_rof = true;
     138                 :          0 :         const char *rof_ptr = *rules_db;
     139                 :            :         uint32_t combined_rof_vers = 0;
     140                 :            :         uint32_t num_rof_blocks = 0;
     141                 :            :         uint32_t rxpc_vers = 0;
     142                 :          0 :         uint32_t target_rxp_vers = 0;
     143                 :            :         uint32_t byte_count = 0;
     144                 :            :         uint32_t rof_bytes_read = 0;
     145                 :            :         bool rof_binary_found = false;
     146                 :          0 :         struct mlx5_hca_attr *attr = &priv->cdev->config.hca_attr;
     147                 :            : 
     148                 :            :         /* Need minimum of 8 bytes to process single or combined rof */
     149         [ #  # ]:          0 :         if (*rules_db_len < 8)
     150                 :            :                 return -EINVAL;
     151                 :            : 
     152         [ #  # ]:          0 :         for (i = 0; i < 8; i++) {
     153                 :          0 :                 if ((char) *rof_ptr !=
     154         [ #  # ]:          0 :                         (char)((combined_rof_tag >> (i * 8)) & 0xFF)) {
     155                 :            :                         combined_rof = false;
     156                 :            :                         break;
     157                 :            :                 }
     158                 :          0 :                 rof_ptr++;
     159                 :            :         }
     160                 :            :         rof_bytes_read += 8;
     161                 :            : 
     162         [ #  # ]:          0 :         if (combined_rof == true) {
     163                 :            :                 /* Need at least 24 bytes of header info: 16 byte combined */
     164                 :            :                 /* rof header and 8 byte binary rof blob header.           */
     165         [ #  # ]:          0 :                 if (*rules_db_len < 24)
     166                 :            :                         return -EINVAL;
     167                 :            : 
     168                 :            :                 /* Read the combined rof version and number of rof blocks */
     169         [ #  # ]:          0 :                 for (i = 0; i < 4; i++) {
     170                 :          0 :                         combined_rof_vers |= *rof_ptr << (i * 8);
     171                 :          0 :                         rof_ptr++;
     172                 :            :                 }
     173                 :            : 
     174                 :            :                 rof_bytes_read += 4;
     175                 :          0 :                 ret = mlx5_regex_check_rof_version(combined_rof_vers);
     176         [ #  # ]:          0 :                 if (ret < 0)
     177                 :            :                         return ret;
     178                 :            : 
     179         [ #  # ]:          0 :                 for (i = 0; i < 4; i++) {
     180                 :          0 :                         num_rof_blocks |= *rof_ptr << (i * 8);
     181                 :          0 :                         rof_ptr++;
     182                 :            :                 }
     183                 :            :                 rof_bytes_read += 4;
     184                 :            : 
     185         [ #  # ]:          0 :                 if (num_rof_blocks == 0)
     186                 :            :                         return -EINVAL;
     187                 :            : 
     188                 :            :                 /* Get the version of rxp we need the rof for */
     189                 :          0 :                 ret = mlx5_regex_get_rxp_vers(attr->regexp_version, &target_rxp_vers);
     190         [ #  # ]:          0 :                 if (ret < 0)
     191                 :            :                         return ret;
     192                 :            : 
     193                 :            :                 /* Try to find the rof binary blob for this version of rxp */
     194         [ #  # ]:          0 :                 for (j = 0; j < num_rof_blocks; j++) {
     195                 :            :                         rxpc_vers = 0;
     196                 :            :                         byte_count = 0;
     197         [ #  # ]:          0 :                         for (i = 0; i < 4; i++) {
     198                 :          0 :                                 rxpc_vers |= (*rof_ptr & 0xFF) << (i * 8);
     199                 :          0 :                                 rof_ptr++;
     200                 :            :                         }
     201         [ #  # ]:          0 :                         for (i = 0; i < 4; i++) {
     202                 :          0 :                                 byte_count |= (*rof_ptr & 0xFF) << (i * 8);
     203                 :          0 :                                 rof_ptr++;
     204                 :            :                         }
     205                 :          0 :                         rof_bytes_read += 8;
     206                 :            : 
     207         [ #  # ]:          0 :                         if (rxpc_vers == target_rxp_vers) {
     208                 :            :                                 /* Found corresponding binary rof entry */
     209         [ #  # ]:          0 :                                 if (rof_bytes_read + byte_count <= (*rules_db_len))
     210                 :            :                                         rof_binary_found = true;
     211                 :            :                                 else
     212                 :          0 :                                         DRV_LOG(ERR, "Compatible rof file found - invalid length!");
     213                 :            :                                 break;
     214                 :            :                         }
     215                 :            :                                 /* Move on to next rof blob */
     216         [ #  # ]:          0 :                         if (rof_bytes_read + byte_count + 8 < (*rules_db_len)) {
     217                 :          0 :                                 rof_ptr += byte_count;
     218                 :            :                                 rof_bytes_read += byte_count;
     219                 :            :                         } else {
     220                 :            :                                 /* Cannot parse any more of combined rof file */
     221                 :            :                                 break;
     222                 :            :                         }
     223                 :            :                 }
     224         [ #  # ]:          0 :                 if (rof_binary_found == true) {
     225                 :          0 :                         *rules_db = rof_ptr;
     226                 :          0 :                         *rules_db_len = byte_count;
     227                 :            :                 } else {
     228                 :          0 :                         DRV_LOG(ERR, "Compatible rof file not found!");
     229                 :          0 :                         return -EINVAL;
     230                 :            :                 }
     231                 :            :         }
     232                 :            :         return 0;
     233                 :            : }
     234                 :            : 
     235                 :            : int
     236                 :          0 : mlx5_regex_rules_db_import(struct rte_regexdev *dev,
     237                 :            :                      const char *rule_db, uint32_t rule_db_len)
     238                 :            : {
     239                 :          0 :         struct mlx5_regex_priv *priv = dev->data->dev_private;
     240                 :            :         struct mlx5_regex_mkey mkey;
     241                 :            :         uint32_t id;
     242                 :            :         int ret;
     243                 :            :         void *ptr;
     244                 :            : 
     245         [ #  # ]:          0 :         if (priv->prog_mode == MLX5_RXP_MODE_NOT_DEFINED) {
     246                 :          0 :                 DRV_LOG(ERR, "RXP programming mode not set!");
     247                 :          0 :                 return -1;
     248                 :            :         }
     249         [ #  # ]:          0 :         if (rule_db == NULL) {
     250                 :          0 :                 DRV_LOG(ERR, "Database empty!");
     251                 :          0 :                 return -ENODEV;
     252                 :            :         }
     253         [ #  # ]:          0 :         if (rule_db_len == 0)
     254                 :            :                 return -EINVAL;
     255                 :            : 
     256                 :          0 :         ret = mlx5_regex_parse_rules_db(priv, &rule_db, &rule_db_len);
     257         [ #  # ]:          0 :         if (ret < 0)
     258                 :            :                 return ret;
     259                 :            : 
     260                 :            :         /* copy rules - rules have to be 4KB aligned. */
     261                 :          0 :         ptr = rte_malloc("", rule_db_len, 1 << 12);
     262         [ #  # ]:          0 :         if (!ptr) {
     263                 :          0 :                 DRV_LOG(ERR, "Failed to allocate rules file memory.");
     264                 :          0 :                 return -ENOMEM;
     265                 :            :         }
     266         [ #  # ]:          0 :         rte_memcpy(ptr, rule_db, rule_db_len);
     267                 :            :         /* Register umem and create rof mkey. */
     268                 :          0 :         ret = rxp_create_mkey(priv, ptr, rule_db_len, /*access=*/7, &mkey);
     269         [ #  # ]:          0 :         if (ret < 0)
     270                 :            :                 return ret;
     271                 :            : 
     272         [ #  # ]:          0 :         for (id = 0; id < priv->nb_engines; id++) {
     273                 :          0 :                 ret = mlx5_devx_regex_rules_program(priv->cdev->ctx, id,
     274                 :          0 :                         mkey.mkey->id, rule_db_len, (uintptr_t)ptr);
     275         [ #  # ]:          0 :                 if (ret < 0) {
     276                 :          0 :                         DRV_LOG(ERR, "Failed to program rxp rules.");
     277                 :            :                         ret = -ENODEV;
     278                 :          0 :                         break;
     279                 :            :                 }
     280                 :            :                 ret = 0;
     281                 :            :         }
     282                 :          0 :         rxp_destroy_mkey(&mkey);
     283                 :          0 :         rte_free(ptr);
     284                 :          0 :         return ret;
     285                 :            : }
     286                 :            : 
     287                 :            : int
     288                 :          0 : mlx5_regex_configure(struct rte_regexdev *dev,
     289                 :            :                      const struct rte_regexdev_config *cfg)
     290                 :            : {
     291                 :          0 :         struct mlx5_regex_priv *priv = dev->data->dev_private;
     292                 :            :         int ret;
     293                 :            : 
     294         [ #  # ]:          0 :         if (priv->prog_mode == MLX5_RXP_MODE_NOT_DEFINED)
     295                 :            :                 return -1;
     296         [ #  # ]:          0 :         if (cfg->nb_max_matches != MLX5_REGEX_MAX_MATCHES) {
     297                 :          0 :                 DRV_LOG(ERR, "nb_max_matches is not configurable.");
     298                 :          0 :                 rte_errno = EINVAL;
     299                 :          0 :                 return -rte_errno;
     300                 :            :         }
     301                 :          0 :         priv->nb_queues = cfg->nb_queue_pairs;
     302                 :          0 :         dev->data->dev_conf.nb_queue_pairs = priv->nb_queues;
     303                 :          0 :         priv->qps = rte_zmalloc(NULL, sizeof(struct mlx5_regex_qp) *
     304                 :          0 :                                 priv->nb_queues, 0);
     305         [ #  # ]:          0 :         if (!priv->qps) {
     306                 :          0 :                 DRV_LOG(ERR, "can't allocate qps memory");
     307                 :          0 :                 rte_errno = ENOMEM;
     308                 :          0 :                 return -rte_errno;
     309                 :            :         }
     310                 :          0 :         priv->nb_max_matches = cfg->nb_max_matches;
     311         [ #  # ]:          0 :         if (cfg->rule_db != NULL) {
     312                 :          0 :                 ret = mlx5_regex_rules_db_import(dev, cfg->rule_db,
     313                 :          0 :                                                  cfg->rule_db_len);
     314         [ #  # ]:          0 :                 if (ret < 0) {
     315                 :          0 :                         DRV_LOG(ERR, "Failed to program rxp rules.");
     316                 :          0 :                         rte_errno = ENODEV;
     317                 :          0 :                         goto configure_error;
     318                 :            :                 }
     319                 :            :         } else
     320                 :          0 :                 DRV_LOG(DEBUG, "Regex config without rules programming!");
     321                 :            :         return 0;
     322                 :            : configure_error:
     323                 :          0 :         rte_free(priv->qps);
     324                 :          0 :         return -rte_errno;
     325                 :            : }

Generated by: LCOV version 1.14