Add structure describing physical device, and manage physical device global list.
Signed-off-by: Michael Baum <michae...@nvidia.com> Acked-by: Suanming Mou <suanmi...@nvidia.com> --- drivers/net/mlx5/mlx5.c | 77 ++++++++++++++++++++++++++++++++++++----- drivers/net/mlx5/mlx5.h | 13 +++++++ 2 files changed, 82 insertions(+), 8 deletions(-) diff --git a/drivers/net/mlx5/mlx5.c b/drivers/net/mlx5/mlx5.c index 3a182de248..f9fc652136 100644 --- a/drivers/net/mlx5/mlx5.c +++ b/drivers/net/mlx5/mlx5.c @@ -190,9 +190,10 @@ struct mlx5_shared_data *mlx5_shared_data; /** Driver-specific log messages type. */ int mlx5_logtype; -static LIST_HEAD(, mlx5_dev_ctx_shared) mlx5_dev_ctx_list = - LIST_HEAD_INITIALIZER(); +static LIST_HEAD(mlx5_dev_ctx_list, mlx5_dev_ctx_shared) dev_ctx_list = LIST_HEAD_INITIALIZER(); +static LIST_HEAD(mlx5_phdev_list, mlx5_physical_device) phdev_list = LIST_HEAD_INITIALIZER(); static pthread_mutex_t mlx5_dev_ctx_list_mutex; + static const struct mlx5_indexed_pool_config mlx5_ipool_cfg[] = { #if defined(HAVE_IBV_FLOW_DV_SUPPORT) || !defined(HAVE_INFINIBAND_VERBS_H) [MLX5_IPOOL_DECAP_ENCAP] = { @@ -1692,6 +1693,60 @@ mlx5_init_shared_dev_registers(struct mlx5_dev_ctx_shared *sh) mlx5_init_hws_flow_tags_registers(sh); } +static struct mlx5_physical_device * +mlx5_get_physical_device(struct mlx5_common_device *cdev) +{ + struct mlx5_physical_device *phdev; + struct mlx5_hca_attr *attr = &cdev->config.hca_attr; + + /* Search for physical device by system_image_guid. */ + LIST_FOREACH(phdev, &phdev_list, next) { + if (phdev->guid == attr->system_image_guid) { + phdev->refcnt++; + return phdev; + } + } + phdev = mlx5_malloc(MLX5_MEM_ZERO | MLX5_MEM_RTE, + sizeof(struct mlx5_physical_device), + RTE_CACHE_LINE_SIZE, SOCKET_ID_ANY); + if (!phdev) { + DRV_LOG(ERR, "Physical device allocation failure."); + rte_errno = ENOMEM; + return NULL; + } + phdev->guid = attr->system_image_guid; + phdev->refcnt = 1; + LIST_INSERT_HEAD(&phdev_list, phdev, next); + DRV_LOG(DEBUG, "Physical device is created, guid=%" PRIu64 ".", + phdev->guid); + return phdev; +} + +static void +mlx5_physical_device_destroy(struct mlx5_physical_device *phdev) +{ +#ifdef RTE_LIBRTE_MLX5_DEBUG + /* Check the object presence in the list. */ + struct mlx5_physical_device *lphdev; + + LIST_FOREACH(lphdev, &phdev_list, next) + if (lphdev == phdev) + break; + MLX5_ASSERT(lphdev); + if (lphdev != phdev) { + DRV_LOG(ERR, "Freeing non-existing physical device"); + return; + } +#endif + MLX5_ASSERT(phdev); + MLX5_ASSERT(phdev->refcnt); + if (--phdev->refcnt) + return; + /* Remove physical device from the global device list. */ + LIST_REMOVE(phdev, next); + mlx5_free(phdev); +} + /** * Allocate shared device context. If there is multiport device the * master and representors will share this context, if there is single @@ -1725,7 +1780,7 @@ mlx5_alloc_shared_dev_ctx(const struct mlx5_dev_spawn_data *spawn, MLX5_ASSERT(rte_eal_process_type() == RTE_PROC_PRIMARY); pthread_mutex_lock(&mlx5_dev_ctx_list_mutex); /* Search for IB context by device name. */ - LIST_FOREACH(sh, &mlx5_dev_ctx_list, next) { + LIST_FOREACH(sh, &dev_ctx_list, next) { if (!strcmp(sh->ibdev_name, spawn->phys_dev_name)) { sh->refcnt++; goto exit; @@ -1765,6 +1820,9 @@ mlx5_alloc_shared_dev_ctx(const struct mlx5_dev_spawn_data *spawn, sizeof(sh->ibdev_name) - 1); strncpy(sh->ibdev_path, mlx5_os_get_ctx_device_path(sh->cdev->ctx), sizeof(sh->ibdev_path) - 1); + sh->phdev = mlx5_get_physical_device(sh->cdev); + if (!sh->phdev) + goto error; /* * Setting port_id to max unallowed value means there is no interrupt * subhandler installed for the given port index i. @@ -1798,7 +1856,7 @@ mlx5_alloc_shared_dev_ctx(const struct mlx5_dev_spawn_data *spawn, #endif } mlx5_os_dev_shared_handler_install(sh); - if (LIST_EMPTY(&mlx5_dev_ctx_list)) { + if (LIST_EMPTY(&dev_ctx_list)) { err = mlx5_flow_os_init_workspace_once(); if (err) goto error; @@ -1811,7 +1869,7 @@ mlx5_alloc_shared_dev_ctx(const struct mlx5_dev_spawn_data *spawn, mlx5_flow_aging_init(sh); mlx5_flow_ipool_create(sh); /* Add context to the global device list. */ - LIST_INSERT_HEAD(&mlx5_dev_ctx_list, sh, next); + LIST_INSERT_HEAD(&dev_ctx_list, sh, next); rte_spinlock_init(&sh->geneve_tlv_opt_sl); mlx5_init_shared_dev_registers(sh); /* Init counter pool list header and lock. */ @@ -1833,6 +1891,8 @@ mlx5_alloc_shared_dev_ctx(const struct mlx5_dev_spawn_data *spawn, } while (++i <= (uint32_t)sh->bond.n_port); if (sh->td) claim_zero(mlx5_devx_cmd_destroy(sh->td)); + if (sh->phdev) + mlx5_physical_device_destroy(sh->phdev); mlx5_free(sh); rte_errno = err; return NULL; @@ -1919,7 +1979,7 @@ mlx5_free_shared_dev_ctx(struct mlx5_dev_ctx_shared *sh) /* Check the object presence in the list. */ struct mlx5_dev_ctx_shared *lctx; - LIST_FOREACH(lctx, &mlx5_dev_ctx_list, next) + LIST_FOREACH(lctx, &dev_ctx_list, next) if (lctx == sh) break; MLX5_ASSERT(lctx); @@ -1945,7 +2005,7 @@ mlx5_free_shared_dev_ctx(struct mlx5_dev_ctx_shared *sh) /* Remove context from the global device list. */ LIST_REMOVE(sh, next); /* Release resources on the last device removal. */ - if (LIST_EMPTY(&mlx5_dev_ctx_list)) { + if (LIST_EMPTY(&dev_ctx_list)) { mlx5_os_net_cleanup(); mlx5_flow_os_release_workspace(); } @@ -1985,6 +2045,7 @@ mlx5_free_shared_dev_ctx(struct mlx5_dev_ctx_shared *sh) MLX5_ASSERT(sh->geneve_tlv_option_resource == NULL); pthread_mutex_destroy(&sh->txpp.mutex); mlx5_lwm_unset(sh); + mlx5_physical_device_destroy(sh->phdev); mlx5_free(sh); return; exit: @@ -2929,7 +2990,7 @@ mlx5_probe_again_args_validate(struct mlx5_common_device *cdev, return 0; pthread_mutex_lock(&mlx5_dev_ctx_list_mutex); /* Search for IB context by common device pointer. */ - LIST_FOREACH(sh, &mlx5_dev_ctx_list, next) + LIST_FOREACH(sh, &dev_ctx_list, next) if (sh->cdev == cdev) break; pthread_mutex_unlock(&mlx5_dev_ctx_list_mutex); diff --git a/drivers/net/mlx5/mlx5.h b/drivers/net/mlx5/mlx5.h index 779805bcd8..8bf7f86416 100644 --- a/drivers/net/mlx5/mlx5.h +++ b/drivers/net/mlx5/mlx5.h @@ -1419,6 +1419,18 @@ struct mlx5_dev_registers { #define HAVE_MLX5_DR_CREATE_ACTION_ASO_EXT #endif +/** + * Physical device structure. + * This device is created once per NIC to manage recourses shared by all ports + * under same physical device. + */ +struct mlx5_physical_device { + LIST_ENTRY(mlx5_physical_device) next; + struct mlx5_dev_ctx_shared *sh; /* Created on sherd context. */ + uint64_t guid; /* System image guid, the uniq ID of physical device. */ + uint32_t refcnt; +}; + /* * Shared Infiniband device context for Master/Representors * which belong to same IB device with multiple IB ports. @@ -1450,6 +1462,7 @@ struct mlx5_dev_ctx_shared { uint32_t max_port; /* Maximal IB device port index. */ struct mlx5_bond_info bond; /* Bonding information. */ struct mlx5_common_device *cdev; /* Backend mlx5 device. */ + struct mlx5_physical_device *phdev; /* Backend physical device. */ uint32_t tdn; /* Transport Domain number. */ char ibdev_name[MLX5_FS_NAME_MAX]; /* SYSFS dev name. */ char ibdev_path[MLX5_FS_PATH_MAX]; /* SYSFS dev path for secondary */ -- 2.25.1