xref: /dpdk/drivers/net/mlx5/mlx5.c (revision db209cc32ab244c4bc7efa29a03ba6242266544c)
18fd92a66SOlivier Matz /* SPDX-License-Identifier: BSD-3-Clause
2771fa900SAdrien Mazarguil  * Copyright 2015 6WIND S.A.
35feecc57SShahaf Shuler  * Copyright 2015 Mellanox Technologies, Ltd
4771fa900SAdrien Mazarguil  */
5771fa900SAdrien Mazarguil 
6771fa900SAdrien Mazarguil #include <stddef.h>
7771fa900SAdrien Mazarguil #include <unistd.h>
8771fa900SAdrien Mazarguil #include <string.h>
9771fa900SAdrien Mazarguil #include <assert.h>
1059b91becSAdrien Mazarguil #include <dlfcn.h>
11771fa900SAdrien Mazarguil #include <stdint.h>
12771fa900SAdrien Mazarguil #include <stdlib.h>
13e72dd09bSNélio Laranjeiro #include <errno.h>
14771fa900SAdrien Mazarguil #include <net/if.h>
154a984153SXueming Li #include <sys/mman.h>
16ccdcba53SNélio Laranjeiro #include <linux/rtnetlink.h>
17771fa900SAdrien Mazarguil 
18771fa900SAdrien Mazarguil /* Verbs header. */
19771fa900SAdrien Mazarguil /* ISO C doesn't support unnamed structs/unions, disabling -pedantic. */
20771fa900SAdrien Mazarguil #ifdef PEDANTIC
21fc5b160fSBruce Richardson #pragma GCC diagnostic ignored "-Wpedantic"
22771fa900SAdrien Mazarguil #endif
23771fa900SAdrien Mazarguil #include <infiniband/verbs.h>
24771fa900SAdrien Mazarguil #ifdef PEDANTIC
25fc5b160fSBruce Richardson #pragma GCC diagnostic error "-Wpedantic"
26771fa900SAdrien Mazarguil #endif
27771fa900SAdrien Mazarguil 
28771fa900SAdrien Mazarguil #include <rte_malloc.h>
29ffc905f3SFerruh Yigit #include <rte_ethdev_driver.h>
30fdf91e0fSJan Blunck #include <rte_ethdev_pci.h>
31771fa900SAdrien Mazarguil #include <rte_pci.h>
32c752998bSGaetan Rivet #include <rte_bus_pci.h>
33771fa900SAdrien Mazarguil #include <rte_common.h>
3459b91becSAdrien Mazarguil #include <rte_config.h>
354a984153SXueming Li #include <rte_eal_memconfig.h>
36e72dd09bSNélio Laranjeiro #include <rte_kvargs.h>
37771fa900SAdrien Mazarguil 
38771fa900SAdrien Mazarguil #include "mlx5.h"
39771fa900SAdrien Mazarguil #include "mlx5_utils.h"
402e22920bSAdrien Mazarguil #include "mlx5_rxtx.h"
41771fa900SAdrien Mazarguil #include "mlx5_autoconf.h"
4213d57bd5SAdrien Mazarguil #include "mlx5_defs.h"
430e83b8e5SNelio Laranjeiro #include "mlx5_glue.h"
44771fa900SAdrien Mazarguil 
4599c12dccSNélio Laranjeiro /* Device parameter to enable RX completion queue compression. */
4699c12dccSNélio Laranjeiro #define MLX5_RXQ_CQE_COMP_EN "rxq_cqe_comp_en"
4799c12dccSNélio Laranjeiro 
482a66cf37SYaacov Hazan /* Device parameter to configure inline send. */
492a66cf37SYaacov Hazan #define MLX5_TXQ_INLINE "txq_inline"
502a66cf37SYaacov Hazan 
512a66cf37SYaacov Hazan /*
522a66cf37SYaacov Hazan  * Device parameter to configure the number of TX queues threshold for
532a66cf37SYaacov Hazan  * enabling inline send.
542a66cf37SYaacov Hazan  */
552a66cf37SYaacov Hazan #define MLX5_TXQS_MIN_INLINE "txqs_min_inline"
562a66cf37SYaacov Hazan 
57230189d9SNélio Laranjeiro /* Device parameter to enable multi-packet send WQEs. */
58230189d9SNélio Laranjeiro #define MLX5_TXQ_MPW_EN "txq_mpw_en"
59230189d9SNélio Laranjeiro 
606ce84bd8SYongseok Koh /* Device parameter to include 2 dsegs in the title WQEBB. */
616ce84bd8SYongseok Koh #define MLX5_TXQ_MPW_HDR_DSEG_EN "txq_mpw_hdr_dseg_en"
626ce84bd8SYongseok Koh 
636ce84bd8SYongseok Koh /* Device parameter to limit the size of inlining packet. */
646ce84bd8SYongseok Koh #define MLX5_TXQ_MAX_INLINE_LEN "txq_max_inline_len"
656ce84bd8SYongseok Koh 
665644d5b9SNelio Laranjeiro /* Device parameter to enable hardware Tx vector. */
675644d5b9SNelio Laranjeiro #define MLX5_TX_VEC_EN "tx_vec_en"
685644d5b9SNelio Laranjeiro 
695644d5b9SNelio Laranjeiro /* Device parameter to enable hardware Rx vector. */
705644d5b9SNelio Laranjeiro #define MLX5_RX_VEC_EN "rx_vec_en"
715644d5b9SNelio Laranjeiro 
72*db209cc3SNélio Laranjeiro /* Activate Netlink support in VF mode. */
73*db209cc3SNélio Laranjeiro #define MLX5_VF_NL_EN "vf_nl_en"
74*db209cc3SNélio Laranjeiro 
7543e9d979SShachar Beiser #ifndef HAVE_IBV_MLX5_MOD_MPW
7643e9d979SShachar Beiser #define MLX5DV_CONTEXT_FLAGS_MPW_ALLOWED (1 << 2)
7743e9d979SShachar Beiser #define MLX5DV_CONTEXT_FLAGS_ENHANCED_MPW (1 << 3)
7843e9d979SShachar Beiser #endif
7943e9d979SShachar Beiser 
80523f5a74SYongseok Koh #ifndef HAVE_IBV_MLX5_MOD_CQE_128B_COMP
81523f5a74SYongseok Koh #define MLX5DV_CONTEXT_FLAGS_CQE_128B_COMP (1 << 4)
82523f5a74SYongseok Koh #endif
83523f5a74SYongseok Koh 
84a170a30dSNélio Laranjeiro /** Driver-specific log messages type. */
85a170a30dSNélio Laranjeiro int mlx5_logtype;
86a170a30dSNélio Laranjeiro 
87771fa900SAdrien Mazarguil /**
884d803a72SOlga Shern  * Retrieve integer value from environment variable.
894d803a72SOlga Shern  *
904d803a72SOlga Shern  * @param[in] name
914d803a72SOlga Shern  *   Environment variable name.
924d803a72SOlga Shern  *
934d803a72SOlga Shern  * @return
944d803a72SOlga Shern  *   Integer value, 0 if the variable is not set.
954d803a72SOlga Shern  */
964d803a72SOlga Shern int
974d803a72SOlga Shern mlx5_getenv_int(const char *name)
984d803a72SOlga Shern {
994d803a72SOlga Shern 	const char *val = getenv(name);
1004d803a72SOlga Shern 
1014d803a72SOlga Shern 	if (val == NULL)
1024d803a72SOlga Shern 		return 0;
1034d803a72SOlga Shern 	return atoi(val);
1044d803a72SOlga Shern }
1054d803a72SOlga Shern 
1064d803a72SOlga Shern /**
1071e3a39f7SXueming Li  * Verbs callback to allocate a memory. This function should allocate the space
1081e3a39f7SXueming Li  * according to the size provided residing inside a huge page.
1091e3a39f7SXueming Li  * Please note that all allocation must respect the alignment from libmlx5
1101e3a39f7SXueming Li  * (i.e. currently sysconf(_SC_PAGESIZE)).
1111e3a39f7SXueming Li  *
1121e3a39f7SXueming Li  * @param[in] size
1131e3a39f7SXueming Li  *   The size in bytes of the memory to allocate.
1141e3a39f7SXueming Li  * @param[in] data
1151e3a39f7SXueming Li  *   A pointer to the callback data.
1161e3a39f7SXueming Li  *
1171e3a39f7SXueming Li  * @return
118a6d83b6aSNélio Laranjeiro  *   Allocated buffer, NULL otherwise and rte_errno is set.
1191e3a39f7SXueming Li  */
1201e3a39f7SXueming Li static void *
1211e3a39f7SXueming Li mlx5_alloc_verbs_buf(size_t size, void *data)
1221e3a39f7SXueming Li {
1231e3a39f7SXueming Li 	struct priv *priv = data;
1241e3a39f7SXueming Li 	void *ret;
1251e3a39f7SXueming Li 	size_t alignment = sysconf(_SC_PAGESIZE);
126d10b09dbSOlivier Matz 	unsigned int socket = SOCKET_ID_ANY;
1271e3a39f7SXueming Li 
128d10b09dbSOlivier Matz 	if (priv->verbs_alloc_ctx.type == MLX5_VERBS_ALLOC_TYPE_TX_QUEUE) {
129d10b09dbSOlivier Matz 		const struct mlx5_txq_ctrl *ctrl = priv->verbs_alloc_ctx.obj;
130d10b09dbSOlivier Matz 
131d10b09dbSOlivier Matz 		socket = ctrl->socket;
132d10b09dbSOlivier Matz 	} else if (priv->verbs_alloc_ctx.type ==
133d10b09dbSOlivier Matz 		   MLX5_VERBS_ALLOC_TYPE_RX_QUEUE) {
134d10b09dbSOlivier Matz 		const struct mlx5_rxq_ctrl *ctrl = priv->verbs_alloc_ctx.obj;
135d10b09dbSOlivier Matz 
136d10b09dbSOlivier Matz 		socket = ctrl->socket;
137d10b09dbSOlivier Matz 	}
1381e3a39f7SXueming Li 	assert(data != NULL);
139d10b09dbSOlivier Matz 	ret = rte_malloc_socket(__func__, size, alignment, socket);
140a6d83b6aSNélio Laranjeiro 	if (!ret && size)
141a6d83b6aSNélio Laranjeiro 		rte_errno = ENOMEM;
1421e3a39f7SXueming Li 	return ret;
1431e3a39f7SXueming Li }
1441e3a39f7SXueming Li 
1451e3a39f7SXueming Li /**
1461e3a39f7SXueming Li  * Verbs callback to free a memory.
1471e3a39f7SXueming Li  *
1481e3a39f7SXueming Li  * @param[in] ptr
1491e3a39f7SXueming Li  *   A pointer to the memory to free.
1501e3a39f7SXueming Li  * @param[in] data
1511e3a39f7SXueming Li  *   A pointer to the callback data.
1521e3a39f7SXueming Li  */
1531e3a39f7SXueming Li static void
1541e3a39f7SXueming Li mlx5_free_verbs_buf(void *ptr, void *data __rte_unused)
1551e3a39f7SXueming Li {
1561e3a39f7SXueming Li 	assert(data != NULL);
1571e3a39f7SXueming Li 	rte_free(ptr);
1581e3a39f7SXueming Li }
1591e3a39f7SXueming Li 
1601e3a39f7SXueming Li /**
161771fa900SAdrien Mazarguil  * DPDK callback to close the device.
162771fa900SAdrien Mazarguil  *
163771fa900SAdrien Mazarguil  * Destroy all queues and objects, free memory.
164771fa900SAdrien Mazarguil  *
165771fa900SAdrien Mazarguil  * @param dev
166771fa900SAdrien Mazarguil  *   Pointer to Ethernet device structure.
167771fa900SAdrien Mazarguil  */
168771fa900SAdrien Mazarguil static void
169771fa900SAdrien Mazarguil mlx5_dev_close(struct rte_eth_dev *dev)
170771fa900SAdrien Mazarguil {
17101d79216SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
1722e22920bSAdrien Mazarguil 	unsigned int i;
1736af6b973SNélio Laranjeiro 	int ret;
174771fa900SAdrien Mazarguil 
175a170a30dSNélio Laranjeiro 	DRV_LOG(DEBUG, "port %u closing device \"%s\"",
1760f99970bSNélio Laranjeiro 		dev->data->port_id,
177771fa900SAdrien Mazarguil 		((priv->ctx != NULL) ? priv->ctx->device->name : ""));
178ecc1c29dSAdrien Mazarguil 	/* In case mlx5_dev_stop() has not been called. */
179af4f09f2SNélio Laranjeiro 	mlx5_dev_interrupt_handler_uninstall(dev);
180af4f09f2SNélio Laranjeiro 	mlx5_traffic_disable(dev);
1812e22920bSAdrien Mazarguil 	/* Prevent crashes when queues are still in use. */
1822e22920bSAdrien Mazarguil 	dev->rx_pkt_burst = removed_rx_burst;
1832e22920bSAdrien Mazarguil 	dev->tx_pkt_burst = removed_tx_burst;
1842e22920bSAdrien Mazarguil 	if (priv->rxqs != NULL) {
1852e22920bSAdrien Mazarguil 		/* XXX race condition if mlx5_rx_burst() is still running. */
1862e22920bSAdrien Mazarguil 		usleep(1000);
187a1366b1aSNélio Laranjeiro 		for (i = 0; (i != priv->rxqs_n); ++i)
188af4f09f2SNélio Laranjeiro 			mlx5_rxq_release(dev, i);
1892e22920bSAdrien Mazarguil 		priv->rxqs_n = 0;
1902e22920bSAdrien Mazarguil 		priv->rxqs = NULL;
1912e22920bSAdrien Mazarguil 	}
1922e22920bSAdrien Mazarguil 	if (priv->txqs != NULL) {
1932e22920bSAdrien Mazarguil 		/* XXX race condition if mlx5_tx_burst() is still running. */
1942e22920bSAdrien Mazarguil 		usleep(1000);
1956e78005aSNélio Laranjeiro 		for (i = 0; (i != priv->txqs_n); ++i)
196af4f09f2SNélio Laranjeiro 			mlx5_txq_release(dev, i);
1972e22920bSAdrien Mazarguil 		priv->txqs_n = 0;
1982e22920bSAdrien Mazarguil 		priv->txqs = NULL;
1992e22920bSAdrien Mazarguil 	}
200771fa900SAdrien Mazarguil 	if (priv->pd != NULL) {
201771fa900SAdrien Mazarguil 		assert(priv->ctx != NULL);
2020e83b8e5SNelio Laranjeiro 		claim_zero(mlx5_glue->dealloc_pd(priv->pd));
2030e83b8e5SNelio Laranjeiro 		claim_zero(mlx5_glue->close_device(priv->ctx));
204771fa900SAdrien Mazarguil 	} else
205771fa900SAdrien Mazarguil 		assert(priv->ctx == NULL);
20629c1d8bbSNélio Laranjeiro 	if (priv->rss_conf.rss_key != NULL)
20729c1d8bbSNélio Laranjeiro 		rte_free(priv->rss_conf.rss_key);
208634efbc2SNelio Laranjeiro 	if (priv->reta_idx != NULL)
209634efbc2SNelio Laranjeiro 		rte_free(priv->reta_idx);
2108c5bca92SXueming Li 	if (priv->primary_socket)
211af4f09f2SNélio Laranjeiro 		mlx5_socket_uninit(dev);
212ccdcba53SNélio Laranjeiro 	if (priv->config.vf)
213ccdcba53SNélio Laranjeiro 		mlx5_nl_mac_addr_flush(dev);
214ccdcba53SNélio Laranjeiro 	if (priv->nl_socket >= 0)
215ccdcba53SNélio Laranjeiro 		close(priv->nl_socket);
216af4f09f2SNélio Laranjeiro 	ret = mlx5_hrxq_ibv_verify(dev);
217f5479b68SNélio Laranjeiro 	if (ret)
218a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some hash Rx queue still remain",
2190f99970bSNélio Laranjeiro 			dev->data->port_id);
220af4f09f2SNélio Laranjeiro 	ret = mlx5_ind_table_ibv_verify(dev);
2214c7a0f5fSNélio Laranjeiro 	if (ret)
222a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some indirection table still remain",
2230f99970bSNélio Laranjeiro 			dev->data->port_id);
224af4f09f2SNélio Laranjeiro 	ret = mlx5_rxq_ibv_verify(dev);
22509cb5b58SNélio Laranjeiro 	if (ret)
226a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some Verbs Rx queue still remain",
2270f99970bSNélio Laranjeiro 			dev->data->port_id);
228af4f09f2SNélio Laranjeiro 	ret = mlx5_rxq_verify(dev);
229a1366b1aSNélio Laranjeiro 	if (ret)
230a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some Rx queues still remain",
2310f99970bSNélio Laranjeiro 			dev->data->port_id);
232af4f09f2SNélio Laranjeiro 	ret = mlx5_txq_ibv_verify(dev);
233faf2667fSNélio Laranjeiro 	if (ret)
234a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some Verbs Tx queue still remain",
2350f99970bSNélio Laranjeiro 			dev->data->port_id);
236af4f09f2SNélio Laranjeiro 	ret = mlx5_txq_verify(dev);
2376e78005aSNélio Laranjeiro 	if (ret)
238a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some Tx queues still remain",
2390f99970bSNélio Laranjeiro 			dev->data->port_id);
240af4f09f2SNélio Laranjeiro 	ret = mlx5_flow_verify(dev);
2416af6b973SNélio Laranjeiro 	if (ret)
242a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some flows still remain",
243a170a30dSNélio Laranjeiro 			dev->data->port_id);
244af4f09f2SNélio Laranjeiro 	ret = mlx5_mr_verify(dev);
245f8fb87d5SNélio Laranjeiro 	if (ret)
246a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u some memory region still remain",
2470f99970bSNélio Laranjeiro 			dev->data->port_id);
248771fa900SAdrien Mazarguil 	memset(priv, 0, sizeof(*priv));
249771fa900SAdrien Mazarguil }
250771fa900SAdrien Mazarguil 
2510887aa7fSNélio Laranjeiro const struct eth_dev_ops mlx5_dev_ops = {
252e60fbd5bSAdrien Mazarguil 	.dev_configure = mlx5_dev_configure,
253e60fbd5bSAdrien Mazarguil 	.dev_start = mlx5_dev_start,
254e60fbd5bSAdrien Mazarguil 	.dev_stop = mlx5_dev_stop,
25562072098SOr Ami 	.dev_set_link_down = mlx5_set_link_down,
25662072098SOr Ami 	.dev_set_link_up = mlx5_set_link_up,
257771fa900SAdrien Mazarguil 	.dev_close = mlx5_dev_close,
2581bdbe1afSAdrien Mazarguil 	.promiscuous_enable = mlx5_promiscuous_enable,
2591bdbe1afSAdrien Mazarguil 	.promiscuous_disable = mlx5_promiscuous_disable,
2601bdbe1afSAdrien Mazarguil 	.allmulticast_enable = mlx5_allmulticast_enable,
2611bdbe1afSAdrien Mazarguil 	.allmulticast_disable = mlx5_allmulticast_disable,
262cb8faed7SAdrien Mazarguil 	.link_update = mlx5_link_update,
26387011737SAdrien Mazarguil 	.stats_get = mlx5_stats_get,
26487011737SAdrien Mazarguil 	.stats_reset = mlx5_stats_reset,
265a4193ae3SShahaf Shuler 	.xstats_get = mlx5_xstats_get,
266a4193ae3SShahaf Shuler 	.xstats_reset = mlx5_xstats_reset,
267a4193ae3SShahaf Shuler 	.xstats_get_names = mlx5_xstats_get_names,
268e60fbd5bSAdrien Mazarguil 	.dev_infos_get = mlx5_dev_infos_get,
26978a38edfSJianfeng Tan 	.dev_supported_ptypes_get = mlx5_dev_supported_ptypes_get,
270e9086978SAdrien Mazarguil 	.vlan_filter_set = mlx5_vlan_filter_set,
2712e22920bSAdrien Mazarguil 	.rx_queue_setup = mlx5_rx_queue_setup,
2722e22920bSAdrien Mazarguil 	.tx_queue_setup = mlx5_tx_queue_setup,
2732e22920bSAdrien Mazarguil 	.rx_queue_release = mlx5_rx_queue_release,
2742e22920bSAdrien Mazarguil 	.tx_queue_release = mlx5_tx_queue_release,
27502d75430SAdrien Mazarguil 	.flow_ctrl_get = mlx5_dev_get_flow_ctrl,
27602d75430SAdrien Mazarguil 	.flow_ctrl_set = mlx5_dev_set_flow_ctrl,
2773318aef7SAdrien Mazarguil 	.mac_addr_remove = mlx5_mac_addr_remove,
2783318aef7SAdrien Mazarguil 	.mac_addr_add = mlx5_mac_addr_add,
27986977fccSDavid Marchand 	.mac_addr_set = mlx5_mac_addr_set,
280cf37ca95SAdrien Mazarguil 	.mtu_set = mlx5_dev_set_mtu,
281f3db9489SYaacov Hazan 	.vlan_strip_queue_set = mlx5_vlan_strip_queue_set,
282f3db9489SYaacov Hazan 	.vlan_offload_set = mlx5_vlan_offload_set,
283634efbc2SNelio Laranjeiro 	.reta_update = mlx5_dev_rss_reta_update,
284634efbc2SNelio Laranjeiro 	.reta_query = mlx5_dev_rss_reta_query,
2852f97422eSNelio Laranjeiro 	.rss_hash_update = mlx5_rss_hash_update,
2862f97422eSNelio Laranjeiro 	.rss_hash_conf_get = mlx5_rss_hash_conf_get,
28776f5c99eSYaacov Hazan 	.filter_ctrl = mlx5_dev_filter_ctrl,
2888788fec1SOlivier Matz 	.rx_descriptor_status = mlx5_rx_descriptor_status,
2898788fec1SOlivier Matz 	.tx_descriptor_status = mlx5_tx_descriptor_status,
2903c7d44afSShahaf Shuler 	.rx_queue_intr_enable = mlx5_rx_intr_enable,
2913c7d44afSShahaf Shuler 	.rx_queue_intr_disable = mlx5_rx_intr_disable,
292d3e0f392SMatan Azrad 	.is_removed = mlx5_is_removed,
293771fa900SAdrien Mazarguil };
294771fa900SAdrien Mazarguil 
29587ec44ceSXueming Li static const struct eth_dev_ops mlx5_dev_sec_ops = {
29687ec44ceSXueming Li 	.stats_get = mlx5_stats_get,
29787ec44ceSXueming Li 	.stats_reset = mlx5_stats_reset,
29887ec44ceSXueming Li 	.xstats_get = mlx5_xstats_get,
29987ec44ceSXueming Li 	.xstats_reset = mlx5_xstats_reset,
30087ec44ceSXueming Li 	.xstats_get_names = mlx5_xstats_get_names,
30187ec44ceSXueming Li 	.dev_infos_get = mlx5_dev_infos_get,
30287ec44ceSXueming Li 	.rx_descriptor_status = mlx5_rx_descriptor_status,
30387ec44ceSXueming Li 	.tx_descriptor_status = mlx5_tx_descriptor_status,
30487ec44ceSXueming Li };
30587ec44ceSXueming Li 
3060887aa7fSNélio Laranjeiro /* Available operators in flow isolated mode. */
3070887aa7fSNélio Laranjeiro const struct eth_dev_ops mlx5_dev_ops_isolate = {
3080887aa7fSNélio Laranjeiro 	.dev_configure = mlx5_dev_configure,
3090887aa7fSNélio Laranjeiro 	.dev_start = mlx5_dev_start,
3100887aa7fSNélio Laranjeiro 	.dev_stop = mlx5_dev_stop,
3110887aa7fSNélio Laranjeiro 	.dev_set_link_down = mlx5_set_link_down,
3120887aa7fSNélio Laranjeiro 	.dev_set_link_up = mlx5_set_link_up,
3130887aa7fSNélio Laranjeiro 	.dev_close = mlx5_dev_close,
3140887aa7fSNélio Laranjeiro 	.link_update = mlx5_link_update,
3150887aa7fSNélio Laranjeiro 	.stats_get = mlx5_stats_get,
3160887aa7fSNélio Laranjeiro 	.stats_reset = mlx5_stats_reset,
3170887aa7fSNélio Laranjeiro 	.xstats_get = mlx5_xstats_get,
3180887aa7fSNélio Laranjeiro 	.xstats_reset = mlx5_xstats_reset,
3190887aa7fSNélio Laranjeiro 	.xstats_get_names = mlx5_xstats_get_names,
3200887aa7fSNélio Laranjeiro 	.dev_infos_get = mlx5_dev_infos_get,
3210887aa7fSNélio Laranjeiro 	.dev_supported_ptypes_get = mlx5_dev_supported_ptypes_get,
3220887aa7fSNélio Laranjeiro 	.vlan_filter_set = mlx5_vlan_filter_set,
3230887aa7fSNélio Laranjeiro 	.rx_queue_setup = mlx5_rx_queue_setup,
3240887aa7fSNélio Laranjeiro 	.tx_queue_setup = mlx5_tx_queue_setup,
3250887aa7fSNélio Laranjeiro 	.rx_queue_release = mlx5_rx_queue_release,
3260887aa7fSNélio Laranjeiro 	.tx_queue_release = mlx5_tx_queue_release,
3270887aa7fSNélio Laranjeiro 	.flow_ctrl_get = mlx5_dev_get_flow_ctrl,
3280887aa7fSNélio Laranjeiro 	.flow_ctrl_set = mlx5_dev_set_flow_ctrl,
3290887aa7fSNélio Laranjeiro 	.mac_addr_remove = mlx5_mac_addr_remove,
3300887aa7fSNélio Laranjeiro 	.mac_addr_add = mlx5_mac_addr_add,
3310887aa7fSNélio Laranjeiro 	.mac_addr_set = mlx5_mac_addr_set,
3320887aa7fSNélio Laranjeiro 	.mtu_set = mlx5_dev_set_mtu,
3330887aa7fSNélio Laranjeiro 	.vlan_strip_queue_set = mlx5_vlan_strip_queue_set,
3340887aa7fSNélio Laranjeiro 	.vlan_offload_set = mlx5_vlan_offload_set,
3350887aa7fSNélio Laranjeiro 	.filter_ctrl = mlx5_dev_filter_ctrl,
3360887aa7fSNélio Laranjeiro 	.rx_descriptor_status = mlx5_rx_descriptor_status,
3370887aa7fSNélio Laranjeiro 	.tx_descriptor_status = mlx5_tx_descriptor_status,
3380887aa7fSNélio Laranjeiro 	.rx_queue_intr_enable = mlx5_rx_intr_enable,
3390887aa7fSNélio Laranjeiro 	.rx_queue_intr_disable = mlx5_rx_intr_disable,
340d3e0f392SMatan Azrad 	.is_removed = mlx5_is_removed,
3410887aa7fSNélio Laranjeiro };
3420887aa7fSNélio Laranjeiro 
343771fa900SAdrien Mazarguil static struct {
344771fa900SAdrien Mazarguil 	struct rte_pci_addr pci_addr; /* associated PCI address */
345771fa900SAdrien Mazarguil 	uint32_t ports; /* physical ports bitfield. */
346771fa900SAdrien Mazarguil } mlx5_dev[32];
347771fa900SAdrien Mazarguil 
348771fa900SAdrien Mazarguil /**
349771fa900SAdrien Mazarguil  * Get device index in mlx5_dev[] from PCI bus address.
350771fa900SAdrien Mazarguil  *
351771fa900SAdrien Mazarguil  * @param[in] pci_addr
352771fa900SAdrien Mazarguil  *   PCI bus address to look for.
353771fa900SAdrien Mazarguil  *
354771fa900SAdrien Mazarguil  * @return
355771fa900SAdrien Mazarguil  *   mlx5_dev[] index on success, -1 on failure.
356771fa900SAdrien Mazarguil  */
357771fa900SAdrien Mazarguil static int
358771fa900SAdrien Mazarguil mlx5_dev_idx(struct rte_pci_addr *pci_addr)
359771fa900SAdrien Mazarguil {
360771fa900SAdrien Mazarguil 	unsigned int i;
361771fa900SAdrien Mazarguil 	int ret = -1;
362771fa900SAdrien Mazarguil 
363771fa900SAdrien Mazarguil 	assert(pci_addr != NULL);
364771fa900SAdrien Mazarguil 	for (i = 0; (i != RTE_DIM(mlx5_dev)); ++i) {
365771fa900SAdrien Mazarguil 		if ((mlx5_dev[i].pci_addr.domain == pci_addr->domain) &&
366771fa900SAdrien Mazarguil 		    (mlx5_dev[i].pci_addr.bus == pci_addr->bus) &&
367771fa900SAdrien Mazarguil 		    (mlx5_dev[i].pci_addr.devid == pci_addr->devid) &&
368771fa900SAdrien Mazarguil 		    (mlx5_dev[i].pci_addr.function == pci_addr->function))
369771fa900SAdrien Mazarguil 			return i;
370771fa900SAdrien Mazarguil 		if ((mlx5_dev[i].ports == 0) && (ret == -1))
371771fa900SAdrien Mazarguil 			ret = i;
372771fa900SAdrien Mazarguil 	}
373771fa900SAdrien Mazarguil 	return ret;
374771fa900SAdrien Mazarguil }
375771fa900SAdrien Mazarguil 
376e72dd09bSNélio Laranjeiro /**
377e72dd09bSNélio Laranjeiro  * Verify and store value for device argument.
378e72dd09bSNélio Laranjeiro  *
379e72dd09bSNélio Laranjeiro  * @param[in] key
380e72dd09bSNélio Laranjeiro  *   Key argument to verify.
381e72dd09bSNélio Laranjeiro  * @param[in] val
382e72dd09bSNélio Laranjeiro  *   Value associated with key.
383e72dd09bSNélio Laranjeiro  * @param opaque
384e72dd09bSNélio Laranjeiro  *   User data.
385e72dd09bSNélio Laranjeiro  *
386e72dd09bSNélio Laranjeiro  * @return
387a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
388e72dd09bSNélio Laranjeiro  */
389e72dd09bSNélio Laranjeiro static int
390e72dd09bSNélio Laranjeiro mlx5_args_check(const char *key, const char *val, void *opaque)
391e72dd09bSNélio Laranjeiro {
3927fe24446SShahaf Shuler 	struct mlx5_dev_config *config = opaque;
39399c12dccSNélio Laranjeiro 	unsigned long tmp;
394e72dd09bSNélio Laranjeiro 
39599c12dccSNélio Laranjeiro 	errno = 0;
39699c12dccSNélio Laranjeiro 	tmp = strtoul(val, NULL, 0);
39799c12dccSNélio Laranjeiro 	if (errno) {
398a6d83b6aSNélio Laranjeiro 		rte_errno = errno;
399a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "%s: \"%s\" is not a valid integer", key, val);
400a6d83b6aSNélio Laranjeiro 		return -rte_errno;
40199c12dccSNélio Laranjeiro 	}
40299c12dccSNélio Laranjeiro 	if (strcmp(MLX5_RXQ_CQE_COMP_EN, key) == 0) {
4037fe24446SShahaf Shuler 		config->cqe_comp = !!tmp;
4042a66cf37SYaacov Hazan 	} else if (strcmp(MLX5_TXQ_INLINE, key) == 0) {
4057fe24446SShahaf Shuler 		config->txq_inline = tmp;
4062a66cf37SYaacov Hazan 	} else if (strcmp(MLX5_TXQS_MIN_INLINE, key) == 0) {
4077fe24446SShahaf Shuler 		config->txqs_inline = tmp;
408230189d9SNélio Laranjeiro 	} else if (strcmp(MLX5_TXQ_MPW_EN, key) == 0) {
4097fe24446SShahaf Shuler 		config->mps = !!tmp ? config->mps : 0;
4106ce84bd8SYongseok Koh 	} else if (strcmp(MLX5_TXQ_MPW_HDR_DSEG_EN, key) == 0) {
4117fe24446SShahaf Shuler 		config->mpw_hdr_dseg = !!tmp;
4126ce84bd8SYongseok Koh 	} else if (strcmp(MLX5_TXQ_MAX_INLINE_LEN, key) == 0) {
4137fe24446SShahaf Shuler 		config->inline_max_packet_sz = tmp;
4145644d5b9SNelio Laranjeiro 	} else if (strcmp(MLX5_TX_VEC_EN, key) == 0) {
4157fe24446SShahaf Shuler 		config->tx_vec_en = !!tmp;
4165644d5b9SNelio Laranjeiro 	} else if (strcmp(MLX5_RX_VEC_EN, key) == 0) {
4177fe24446SShahaf Shuler 		config->rx_vec_en = !!tmp;
418*db209cc3SNélio Laranjeiro 	} else if (strcmp(MLX5_VF_NL_EN, key) == 0) {
419*db209cc3SNélio Laranjeiro 		config->vf_nl_en = !!tmp;
42099c12dccSNélio Laranjeiro 	} else {
421a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "%s: unknown parameter", key);
422a6d83b6aSNélio Laranjeiro 		rte_errno = EINVAL;
423a6d83b6aSNélio Laranjeiro 		return -rte_errno;
424e72dd09bSNélio Laranjeiro 	}
42599c12dccSNélio Laranjeiro 	return 0;
42699c12dccSNélio Laranjeiro }
427e72dd09bSNélio Laranjeiro 
428e72dd09bSNélio Laranjeiro /**
429e72dd09bSNélio Laranjeiro  * Parse device parameters.
430e72dd09bSNélio Laranjeiro  *
4317fe24446SShahaf Shuler  * @param config
4327fe24446SShahaf Shuler  *   Pointer to device configuration structure.
433e72dd09bSNélio Laranjeiro  * @param devargs
434e72dd09bSNélio Laranjeiro  *   Device arguments structure.
435e72dd09bSNélio Laranjeiro  *
436e72dd09bSNélio Laranjeiro  * @return
437a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
438e72dd09bSNélio Laranjeiro  */
439e72dd09bSNélio Laranjeiro static int
4407fe24446SShahaf Shuler mlx5_args(struct mlx5_dev_config *config, struct rte_devargs *devargs)
441e72dd09bSNélio Laranjeiro {
442e72dd09bSNélio Laranjeiro 	const char **params = (const char *[]){
44399c12dccSNélio Laranjeiro 		MLX5_RXQ_CQE_COMP_EN,
4442a66cf37SYaacov Hazan 		MLX5_TXQ_INLINE,
4452a66cf37SYaacov Hazan 		MLX5_TXQS_MIN_INLINE,
446230189d9SNélio Laranjeiro 		MLX5_TXQ_MPW_EN,
4476ce84bd8SYongseok Koh 		MLX5_TXQ_MPW_HDR_DSEG_EN,
4486ce84bd8SYongseok Koh 		MLX5_TXQ_MAX_INLINE_LEN,
4495644d5b9SNelio Laranjeiro 		MLX5_TX_VEC_EN,
4505644d5b9SNelio Laranjeiro 		MLX5_RX_VEC_EN,
451*db209cc3SNélio Laranjeiro 		MLX5_VF_NL_EN,
452e72dd09bSNélio Laranjeiro 		NULL,
453e72dd09bSNélio Laranjeiro 	};
454e72dd09bSNélio Laranjeiro 	struct rte_kvargs *kvlist;
455e72dd09bSNélio Laranjeiro 	int ret = 0;
456e72dd09bSNélio Laranjeiro 	int i;
457e72dd09bSNélio Laranjeiro 
458e72dd09bSNélio Laranjeiro 	if (devargs == NULL)
459e72dd09bSNélio Laranjeiro 		return 0;
460e72dd09bSNélio Laranjeiro 	/* Following UGLY cast is done to pass checkpatch. */
461e72dd09bSNélio Laranjeiro 	kvlist = rte_kvargs_parse(devargs->args, params);
462e72dd09bSNélio Laranjeiro 	if (kvlist == NULL)
463e72dd09bSNélio Laranjeiro 		return 0;
464e72dd09bSNélio Laranjeiro 	/* Process parameters. */
465e72dd09bSNélio Laranjeiro 	for (i = 0; (params[i] != NULL); ++i) {
466e72dd09bSNélio Laranjeiro 		if (rte_kvargs_count(kvlist, params[i])) {
467e72dd09bSNélio Laranjeiro 			ret = rte_kvargs_process(kvlist, params[i],
4687fe24446SShahaf Shuler 						 mlx5_args_check, config);
469a6d83b6aSNélio Laranjeiro 			if (ret) {
470a6d83b6aSNélio Laranjeiro 				rte_errno = EINVAL;
471a67323e4SShahaf Shuler 				rte_kvargs_free(kvlist);
472a6d83b6aSNélio Laranjeiro 				return -rte_errno;
473e72dd09bSNélio Laranjeiro 			}
474e72dd09bSNélio Laranjeiro 		}
475a67323e4SShahaf Shuler 	}
476e72dd09bSNélio Laranjeiro 	rte_kvargs_free(kvlist);
477e72dd09bSNélio Laranjeiro 	return 0;
478e72dd09bSNélio Laranjeiro }
479e72dd09bSNélio Laranjeiro 
480fdf91e0fSJan Blunck static struct rte_pci_driver mlx5_driver;
481771fa900SAdrien Mazarguil 
4824a984153SXueming Li /*
4834a984153SXueming Li  * Reserved UAR address space for TXQ UAR(hw doorbell) mapping, process
4844a984153SXueming Li  * local resource used by both primary and secondary to avoid duplicate
4854a984153SXueming Li  * reservation.
4864a984153SXueming Li  * The space has to be available on both primary and secondary process,
4874a984153SXueming Li  * TXQ UAR maps to this area using fixed mmap w/o double check.
4884a984153SXueming Li  */
4894a984153SXueming Li static void *uar_base;
4904a984153SXueming Li 
4918594a202SAnatoly Burakov static int
49266cc45e2SAnatoly Burakov find_lower_va_bound(const struct rte_memseg_list *msl __rte_unused,
49366cc45e2SAnatoly Burakov 		const struct rte_memseg *ms, void *arg)
4948594a202SAnatoly Burakov {
4958594a202SAnatoly Burakov 	void **addr = arg;
4968594a202SAnatoly Burakov 
4978594a202SAnatoly Burakov 	if (*addr == NULL)
4988594a202SAnatoly Burakov 		*addr = ms->addr;
4998594a202SAnatoly Burakov 	else
5008594a202SAnatoly Burakov 		*addr = RTE_MIN(*addr, ms->addr);
5018594a202SAnatoly Burakov 
5028594a202SAnatoly Burakov 	return 0;
5038594a202SAnatoly Burakov }
5048594a202SAnatoly Burakov 
5054a984153SXueming Li /**
5064a984153SXueming Li  * Reserve UAR address space for primary process.
5074a984153SXueming Li  *
508af4f09f2SNélio Laranjeiro  * @param[in] dev
509af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
5104a984153SXueming Li  *
5114a984153SXueming Li  * @return
512a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
5134a984153SXueming Li  */
5144a984153SXueming Li static int
515af4f09f2SNélio Laranjeiro mlx5_uar_init_primary(struct rte_eth_dev *dev)
5164a984153SXueming Li {
517af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
5184a984153SXueming Li 	void *addr = (void *)0;
5194a984153SXueming Li 
5204a984153SXueming Li 	if (uar_base) { /* UAR address space mapped. */
5214a984153SXueming Li 		priv->uar_base = uar_base;
5224a984153SXueming Li 		return 0;
5234a984153SXueming Li 	}
5244a984153SXueming Li 	/* find out lower bound of hugepage segments */
5258594a202SAnatoly Burakov 	rte_memseg_walk(find_lower_va_bound, &addr);
5268594a202SAnatoly Burakov 
5274a984153SXueming Li 	/* keep distance to hugepages to minimize potential conflicts. */
5284a984153SXueming Li 	addr = RTE_PTR_SUB(addr, MLX5_UAR_OFFSET + MLX5_UAR_SIZE);
5294a984153SXueming Li 	/* anonymous mmap, no real memory consumption. */
5304a984153SXueming Li 	addr = mmap(addr, MLX5_UAR_SIZE,
5314a984153SXueming Li 		    PROT_NONE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
5324a984153SXueming Li 	if (addr == MAP_FAILED) {
533a170a30dSNélio Laranjeiro 		DRV_LOG(ERR,
534a170a30dSNélio Laranjeiro 			"port %u failed to reserve UAR address space, please"
5350f99970bSNélio Laranjeiro 			" adjust MLX5_UAR_SIZE or try --base-virtaddr",
5360f99970bSNélio Laranjeiro 			dev->data->port_id);
537a6d83b6aSNélio Laranjeiro 		rte_errno = ENOMEM;
538a6d83b6aSNélio Laranjeiro 		return -rte_errno;
5394a984153SXueming Li 	}
5404a984153SXueming Li 	/* Accept either same addr or a new addr returned from mmap if target
5414a984153SXueming Li 	 * range occupied.
5424a984153SXueming Li 	 */
543a170a30dSNélio Laranjeiro 	DRV_LOG(INFO, "port %u reserved UAR address space: %p",
544a170a30dSNélio Laranjeiro 		dev->data->port_id, addr);
5454a984153SXueming Li 	priv->uar_base = addr; /* for primary and secondary UAR re-mmap. */
5464a984153SXueming Li 	uar_base = addr; /* process local, don't reserve again. */
5474a984153SXueming Li 	return 0;
5484a984153SXueming Li }
5494a984153SXueming Li 
5504a984153SXueming Li /**
5514a984153SXueming Li  * Reserve UAR address space for secondary process, align with
5524a984153SXueming Li  * primary process.
5534a984153SXueming Li  *
554af4f09f2SNélio Laranjeiro  * @param[in] dev
555af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
5564a984153SXueming Li  *
5574a984153SXueming Li  * @return
558a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
5594a984153SXueming Li  */
5604a984153SXueming Li static int
561af4f09f2SNélio Laranjeiro mlx5_uar_init_secondary(struct rte_eth_dev *dev)
5624a984153SXueming Li {
563af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
5644a984153SXueming Li 	void *addr;
5654a984153SXueming Li 
5664a984153SXueming Li 	assert(priv->uar_base);
5674a984153SXueming Li 	if (uar_base) { /* already reserved. */
5684a984153SXueming Li 		assert(uar_base == priv->uar_base);
5694a984153SXueming Li 		return 0;
5704a984153SXueming Li 	}
5714a984153SXueming Li 	/* anonymous mmap, no real memory consumption. */
5724a984153SXueming Li 	addr = mmap(priv->uar_base, MLX5_UAR_SIZE,
5734a984153SXueming Li 		    PROT_NONE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
5744a984153SXueming Li 	if (addr == MAP_FAILED) {
575a170a30dSNélio Laranjeiro 		DRV_LOG(ERR, "port %u UAR mmap failed: %p size: %llu",
5760f99970bSNélio Laranjeiro 			dev->data->port_id, priv->uar_base, MLX5_UAR_SIZE);
577a6d83b6aSNélio Laranjeiro 		rte_errno = ENXIO;
578a6d83b6aSNélio Laranjeiro 		return -rte_errno;
5794a984153SXueming Li 	}
5804a984153SXueming Li 	if (priv->uar_base != addr) {
581a170a30dSNélio Laranjeiro 		DRV_LOG(ERR,
582a170a30dSNélio Laranjeiro 			"port %u UAR address %p size %llu occupied, please"
583a170a30dSNélio Laranjeiro 			" adjust MLX5_UAR_OFFSET or try EAL parameter"
584a170a30dSNélio Laranjeiro 			" --base-virtaddr",
5850f99970bSNélio Laranjeiro 			dev->data->port_id, priv->uar_base, MLX5_UAR_SIZE);
586a6d83b6aSNélio Laranjeiro 		rte_errno = ENXIO;
587a6d83b6aSNélio Laranjeiro 		return -rte_errno;
5884a984153SXueming Li 	}
5894a984153SXueming Li 	uar_base = addr; /* process local, don't reserve again */
590a170a30dSNélio Laranjeiro 	DRV_LOG(INFO, "port %u reserved UAR address space: %p",
591a170a30dSNélio Laranjeiro 		dev->data->port_id, addr);
5924a984153SXueming Li 	return 0;
5934a984153SXueming Li }
5944a984153SXueming Li 
595771fa900SAdrien Mazarguil /**
596771fa900SAdrien Mazarguil  * DPDK callback to register a PCI device.
597771fa900SAdrien Mazarguil  *
598771fa900SAdrien Mazarguil  * This function creates an Ethernet device for each port of a given
599771fa900SAdrien Mazarguil  * PCI device.
600771fa900SAdrien Mazarguil  *
601771fa900SAdrien Mazarguil  * @param[in] pci_drv
602771fa900SAdrien Mazarguil  *   PCI driver structure (mlx5_driver).
603771fa900SAdrien Mazarguil  * @param[in] pci_dev
604771fa900SAdrien Mazarguil  *   PCI device information.
605771fa900SAdrien Mazarguil  *
606771fa900SAdrien Mazarguil  * @return
607a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
608771fa900SAdrien Mazarguil  */
609771fa900SAdrien Mazarguil static int
61056f08e16SNélio Laranjeiro mlx5_pci_probe(struct rte_pci_driver *pci_drv __rte_unused,
61156f08e16SNélio Laranjeiro 	       struct rte_pci_device *pci_dev)
612771fa900SAdrien Mazarguil {
613a6d83b6aSNélio Laranjeiro 	struct ibv_device **list = NULL;
614771fa900SAdrien Mazarguil 	struct ibv_device *ibv_dev;
615771fa900SAdrien Mazarguil 	int err = 0;
616771fa900SAdrien Mazarguil 	struct ibv_context *attr_ctx = NULL;
61743e9d979SShachar Beiser 	struct ibv_device_attr_ex device_attr;
618ccdcba53SNélio Laranjeiro 	unsigned int vf;
619e192ef80SYaacov Hazan 	unsigned int mps;
620523f5a74SYongseok Koh 	unsigned int cqe_comp;
621772d3435SXueming Li 	unsigned int tunnel_en = 0;
622771fa900SAdrien Mazarguil 	int idx;
623771fa900SAdrien Mazarguil 	int i;
624038e7251SShahaf Shuler 	struct mlx5dv_context attrs_out = {0};
6259a761de8SOri Kam #ifdef HAVE_IBV_DEVICE_COUNTERS_SET_SUPPORT
6269a761de8SOri Kam 	struct ibv_counter_set_description cs_desc;
6279a761de8SOri Kam #endif
628771fa900SAdrien Mazarguil 
629fdf91e0fSJan Blunck 	assert(pci_drv == &mlx5_driver);
630771fa900SAdrien Mazarguil 	/* Get mlx5_dev[] index. */
631771fa900SAdrien Mazarguil 	idx = mlx5_dev_idx(&pci_dev->addr);
632771fa900SAdrien Mazarguil 	if (idx == -1) {
633a170a30dSNélio Laranjeiro 		DRV_LOG(ERR, "this driver cannot support any more adapters");
634a6d83b6aSNélio Laranjeiro 		err = ENOMEM;
635a6d83b6aSNélio Laranjeiro 		goto error;
636771fa900SAdrien Mazarguil 	}
637a170a30dSNélio Laranjeiro 	DRV_LOG(DEBUG, "using driver device index %d", idx);
638771fa900SAdrien Mazarguil 	/* Save PCI address. */
639771fa900SAdrien Mazarguil 	mlx5_dev[idx].pci_addr = pci_dev->addr;
6400e83b8e5SNelio Laranjeiro 	list = mlx5_glue->get_device_list(&i);
641771fa900SAdrien Mazarguil 	if (list == NULL) {
642771fa900SAdrien Mazarguil 		assert(errno);
643a6d83b6aSNélio Laranjeiro 		err = errno;
6445525aa8fSGaetan Rivet 		if (errno == ENOSYS)
645a170a30dSNélio Laranjeiro 			DRV_LOG(ERR,
646a170a30dSNélio Laranjeiro 				"cannot list devices, is ib_uverbs loaded?");
647a6d83b6aSNélio Laranjeiro 		goto error;
648771fa900SAdrien Mazarguil 	}
649771fa900SAdrien Mazarguil 	assert(i >= 0);
650771fa900SAdrien Mazarguil 	/*
651771fa900SAdrien Mazarguil 	 * For each listed device, check related sysfs entry against
652771fa900SAdrien Mazarguil 	 * the provided PCI ID.
653771fa900SAdrien Mazarguil 	 */
654771fa900SAdrien Mazarguil 	while (i != 0) {
655771fa900SAdrien Mazarguil 		struct rte_pci_addr pci_addr;
656771fa900SAdrien Mazarguil 
657771fa900SAdrien Mazarguil 		--i;
658a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "checking device \"%s\"", list[i]->name);
659771fa900SAdrien Mazarguil 		if (mlx5_ibv_device_to_pci_addr(list[i], &pci_addr))
660771fa900SAdrien Mazarguil 			continue;
661771fa900SAdrien Mazarguil 		if ((pci_dev->addr.domain != pci_addr.domain) ||
662771fa900SAdrien Mazarguil 		    (pci_dev->addr.bus != pci_addr.bus) ||
663771fa900SAdrien Mazarguil 		    (pci_dev->addr.devid != pci_addr.devid) ||
664771fa900SAdrien Mazarguil 		    (pci_dev->addr.function != pci_addr.function))
665771fa900SAdrien Mazarguil 			continue;
666a170a30dSNélio Laranjeiro 		DRV_LOG(INFO, "PCI information matches, using device \"%s\"",
667a61888c8SNélio Laranjeiro 			list[i]->name);
668ccdcba53SNélio Laranjeiro 		vf = ((pci_dev->id.device_id ==
669ccdcba53SNélio Laranjeiro 		       PCI_DEVICE_ID_MELLANOX_CONNECTX4VF) ||
670ccdcba53SNélio Laranjeiro 		      (pci_dev->id.device_id ==
671ccdcba53SNélio Laranjeiro 		       PCI_DEVICE_ID_MELLANOX_CONNECTX4LXVF) ||
672ccdcba53SNélio Laranjeiro 		      (pci_dev->id.device_id ==
673ccdcba53SNélio Laranjeiro 		       PCI_DEVICE_ID_MELLANOX_CONNECTX5VF) ||
674ccdcba53SNélio Laranjeiro 		      (pci_dev->id.device_id ==
675ccdcba53SNélio Laranjeiro 		       PCI_DEVICE_ID_MELLANOX_CONNECTX5EXVF));
6760e83b8e5SNelio Laranjeiro 		attr_ctx = mlx5_glue->open_device(list[i]);
677a6d83b6aSNélio Laranjeiro 		rte_errno = errno;
678a6d83b6aSNélio Laranjeiro 		err = rte_errno;
679771fa900SAdrien Mazarguil 		break;
680771fa900SAdrien Mazarguil 	}
681771fa900SAdrien Mazarguil 	if (attr_ctx == NULL) {
6820e83b8e5SNelio Laranjeiro 		mlx5_glue->free_device_list(list);
683771fa900SAdrien Mazarguil 		switch (err) {
684771fa900SAdrien Mazarguil 		case 0:
685a170a30dSNélio Laranjeiro 			DRV_LOG(ERR,
686a170a30dSNélio Laranjeiro 				"cannot access device, is mlx5_ib loaded?");
687a6d83b6aSNélio Laranjeiro 			err = ENODEV;
688a6d83b6aSNélio Laranjeiro 			goto error;
689771fa900SAdrien Mazarguil 		case EINVAL:
690a170a30dSNélio Laranjeiro 			DRV_LOG(ERR,
691a170a30dSNélio Laranjeiro 				"cannot use device, are drivers up to date?");
692a6d83b6aSNélio Laranjeiro 			goto error;
693771fa900SAdrien Mazarguil 		}
694771fa900SAdrien Mazarguil 	}
695771fa900SAdrien Mazarguil 	ibv_dev = list[i];
696a170a30dSNélio Laranjeiro 	DRV_LOG(DEBUG, "device opened");
69743e9d979SShachar Beiser 	/*
69843e9d979SShachar Beiser 	 * Multi-packet send is supported by ConnectX-4 Lx PF as well
69943e9d979SShachar Beiser 	 * as all ConnectX-5 devices.
70043e9d979SShachar Beiser 	 */
701038e7251SShahaf Shuler #ifdef HAVE_IBV_DEVICE_TUNNEL_SUPPORT
702038e7251SShahaf Shuler 	attrs_out.comp_mask |= MLX5DV_CONTEXT_MASK_TUNNEL_OFFLOADS;
703038e7251SShahaf Shuler #endif
7040e83b8e5SNelio Laranjeiro 	mlx5_glue->dv_query_device(attr_ctx, &attrs_out);
705e589960cSYongseok Koh 	if (attrs_out.flags & MLX5DV_CONTEXT_FLAGS_MPW_ALLOWED) {
706e589960cSYongseok Koh 		if (attrs_out.flags & MLX5DV_CONTEXT_FLAGS_ENHANCED_MPW) {
707a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG, "enhanced MPW is supported");
70843e9d979SShachar Beiser 			mps = MLX5_MPW_ENHANCED;
70943e9d979SShachar Beiser 		} else {
710a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG, "MPW is supported");
711e589960cSYongseok Koh 			mps = MLX5_MPW;
712e589960cSYongseok Koh 		}
713e589960cSYongseok Koh 	} else {
714a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "MPW isn't supported");
71543e9d979SShachar Beiser 		mps = MLX5_MPW_DISABLED;
71643e9d979SShachar Beiser 	}
717523f5a74SYongseok Koh 	if (RTE_CACHE_LINE_SIZE == 128 &&
718523f5a74SYongseok Koh 	    !(attrs_out.flags & MLX5DV_CONTEXT_FLAGS_CQE_128B_COMP))
719523f5a74SYongseok Koh 		cqe_comp = 0;
720523f5a74SYongseok Koh 	else
721523f5a74SYongseok Koh 		cqe_comp = 1;
722038e7251SShahaf Shuler #ifdef HAVE_IBV_DEVICE_TUNNEL_SUPPORT
723038e7251SShahaf Shuler 	if (attrs_out.comp_mask & MLX5DV_CONTEXT_MASK_TUNNEL_OFFLOADS) {
724038e7251SShahaf Shuler 		tunnel_en = ((attrs_out.tunnel_offloads_caps &
725038e7251SShahaf Shuler 			      MLX5DV_RAW_PACKET_CAP_TUNNELED_OFFLOAD_VXLAN) &&
726038e7251SShahaf Shuler 			     (attrs_out.tunnel_offloads_caps &
727038e7251SShahaf Shuler 			      MLX5DV_RAW_PACKET_CAP_TUNNELED_OFFLOAD_GRE));
728038e7251SShahaf Shuler 	}
729a170a30dSNélio Laranjeiro 	DRV_LOG(DEBUG, "tunnel offloading is %ssupported",
730a170a30dSNélio Laranjeiro 		tunnel_en ? "" : "not ");
731038e7251SShahaf Shuler #else
732a170a30dSNélio Laranjeiro 	DRV_LOG(WARNING,
733a170a30dSNélio Laranjeiro 		"tunnel offloading disabled due to old OFED/rdma-core version");
734038e7251SShahaf Shuler #endif
735a6d83b6aSNélio Laranjeiro 	if (mlx5_glue->query_device_ex(attr_ctx, NULL, &device_attr)) {
736a6d83b6aSNélio Laranjeiro 		err = errno;
737771fa900SAdrien Mazarguil 		goto error;
738a6d83b6aSNélio Laranjeiro 	}
739a170a30dSNélio Laranjeiro 	DRV_LOG(INFO, "%u port(s) detected",
740a170a30dSNélio Laranjeiro 		device_attr.orig_attr.phys_port_cnt);
74143e9d979SShachar Beiser 	for (i = 0; i < device_attr.orig_attr.phys_port_cnt; i++) {
742ad831a11SYuanhan Liu 		char name[RTE_ETH_NAME_MAX_LEN];
743ad831a11SYuanhan Liu 		int len;
744771fa900SAdrien Mazarguil 		uint32_t port = i + 1; /* ports are indexed from one */
745771fa900SAdrien Mazarguil 		uint32_t test = (1 << i);
746771fa900SAdrien Mazarguil 		struct ibv_context *ctx = NULL;
747771fa900SAdrien Mazarguil 		struct ibv_port_attr port_attr;
748771fa900SAdrien Mazarguil 		struct ibv_pd *pd = NULL;
749771fa900SAdrien Mazarguil 		struct priv *priv = NULL;
750af4f09f2SNélio Laranjeiro 		struct rte_eth_dev *eth_dev = NULL;
75143e9d979SShachar Beiser 		struct ibv_device_attr_ex device_attr_ex;
752771fa900SAdrien Mazarguil 		struct ether_addr mac;
7537fe24446SShahaf Shuler 		struct mlx5_dev_config config = {
7547fe24446SShahaf Shuler 			.cqe_comp = cqe_comp,
7557fe24446SShahaf Shuler 			.mps = mps,
7567fe24446SShahaf Shuler 			.tunnel_en = tunnel_en,
7577fe24446SShahaf Shuler 			.tx_vec_en = 1,
7587fe24446SShahaf Shuler 			.rx_vec_en = 1,
7597fe24446SShahaf Shuler 			.mpw_hdr_dseg = 0,
76050b244a1SShahaf Shuler 			.txq_inline = MLX5_ARG_UNSET,
76150b244a1SShahaf Shuler 			.txqs_inline = MLX5_ARG_UNSET,
76250b244a1SShahaf Shuler 			.inline_max_packet_sz = MLX5_ARG_UNSET,
763*db209cc3SNélio Laranjeiro 			.vf_nl_en = 1,
76450b244a1SShahaf Shuler 		};
765771fa900SAdrien Mazarguil 
766ad831a11SYuanhan Liu 		len = snprintf(name, sizeof(name), PCI_PRI_FMT,
767ad831a11SYuanhan Liu 			 pci_dev->addr.domain, pci_dev->addr.bus,
768ad831a11SYuanhan Liu 			 pci_dev->addr.devid, pci_dev->addr.function);
769ad831a11SYuanhan Liu 		if (device_attr.orig_attr.phys_port_cnt > 1)
770ad831a11SYuanhan Liu 			snprintf(name + len, sizeof(name), " port %u", i);
771f8b9a3baSXueming Li 		mlx5_dev[idx].ports |= test;
77251e7fa8dSNélio Laranjeiro 		if (rte_eal_process_type() == RTE_PROC_SECONDARY) {
773f8b9a3baSXueming Li 			eth_dev = rte_eth_dev_attach_secondary(name);
774f8b9a3baSXueming Li 			if (eth_dev == NULL) {
775a170a30dSNélio Laranjeiro 				DRV_LOG(ERR, "can not attach rte ethdev");
776a6d83b6aSNélio Laranjeiro 				rte_errno = ENOMEM;
777a6d83b6aSNélio Laranjeiro 				err = rte_errno;
778f8b9a3baSXueming Li 				goto error;
779f8b9a3baSXueming Li 			}
780f8b9a3baSXueming Li 			eth_dev->device = &pci_dev->device;
78187ec44ceSXueming Li 			eth_dev->dev_ops = &mlx5_dev_sec_ops;
782af4f09f2SNélio Laranjeiro 			err = mlx5_uar_init_secondary(eth_dev);
783a6d83b6aSNélio Laranjeiro 			if (err)
7844a984153SXueming Li 				goto error;
785f8b9a3baSXueming Li 			/* Receive command fd from primary process */
786af4f09f2SNélio Laranjeiro 			err = mlx5_socket_connect(eth_dev);
787a6d83b6aSNélio Laranjeiro 			if (err)
788f8b9a3baSXueming Li 				goto error;
789f8b9a3baSXueming Li 			/* Remap UAR for Tx queues. */
790af4f09f2SNélio Laranjeiro 			err = mlx5_tx_uar_remap(eth_dev, err);
7914a984153SXueming Li 			if (err)
792f8b9a3baSXueming Li 				goto error;
7931cfa649bSShahaf Shuler 			/*
7941cfa649bSShahaf Shuler 			 * Ethdev pointer is still required as input since
7951cfa649bSShahaf Shuler 			 * the primary device is not accessible from the
7961cfa649bSShahaf Shuler 			 * secondary process.
7971cfa649bSShahaf Shuler 			 */
7981cfa649bSShahaf Shuler 			eth_dev->rx_pkt_burst =
799af4f09f2SNélio Laranjeiro 				mlx5_select_rx_function(eth_dev);
8001cfa649bSShahaf Shuler 			eth_dev->tx_pkt_burst =
801af4f09f2SNélio Laranjeiro 				mlx5_select_tx_function(eth_dev);
802f8b9a3baSXueming Li 			continue;
803f8b9a3baSXueming Li 		}
804a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "using port %u (%08" PRIx32 ")", port, test);
8050e83b8e5SNelio Laranjeiro 		ctx = mlx5_glue->open_device(ibv_dev);
806e1c3e305SMatan Azrad 		if (ctx == NULL) {
807e1c3e305SMatan Azrad 			err = ENODEV;
808771fa900SAdrien Mazarguil 			goto port_error;
809e1c3e305SMatan Azrad 		}
810771fa900SAdrien Mazarguil 		/* Check port status. */
8110e83b8e5SNelio Laranjeiro 		err = mlx5_glue->query_port(ctx, port, &port_attr);
812771fa900SAdrien Mazarguil 		if (err) {
813a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "port query failed: %s", strerror(err));
814771fa900SAdrien Mazarguil 			goto port_error;
815771fa900SAdrien Mazarguil 		}
8161371f4dfSOr Ami 		if (port_attr.link_layer != IBV_LINK_LAYER_ETHERNET) {
817a170a30dSNélio Laranjeiro 			DRV_LOG(ERR,
818a170a30dSNélio Laranjeiro 				"port %d is not configured in Ethernet mode",
8191371f4dfSOr Ami 				port);
820e1c3e305SMatan Azrad 			err = EINVAL;
8211371f4dfSOr Ami 			goto port_error;
8221371f4dfSOr Ami 		}
823771fa900SAdrien Mazarguil 		if (port_attr.state != IBV_PORT_ACTIVE)
824a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG, "port %d is not active: \"%s\" (%d)",
825a170a30dSNélio Laranjeiro 				port,
826a170a30dSNélio Laranjeiro 				mlx5_glue->port_state_str(port_attr.state),
827771fa900SAdrien Mazarguil 				port_attr.state);
828771fa900SAdrien Mazarguil 		/* Allocate protection domain. */
8290e83b8e5SNelio Laranjeiro 		pd = mlx5_glue->alloc_pd(ctx);
830771fa900SAdrien Mazarguil 		if (pd == NULL) {
831a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "PD allocation failure");
832771fa900SAdrien Mazarguil 			err = ENOMEM;
833771fa900SAdrien Mazarguil 			goto port_error;
834771fa900SAdrien Mazarguil 		}
835771fa900SAdrien Mazarguil 		mlx5_dev[idx].ports |= test;
836771fa900SAdrien Mazarguil 		/* from rte_ethdev.c */
837771fa900SAdrien Mazarguil 		priv = rte_zmalloc("ethdev private structure",
838771fa900SAdrien Mazarguil 				   sizeof(*priv),
839771fa900SAdrien Mazarguil 				   RTE_CACHE_LINE_SIZE);
840771fa900SAdrien Mazarguil 		if (priv == NULL) {
841a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "priv allocation failure");
842771fa900SAdrien Mazarguil 			err = ENOMEM;
843771fa900SAdrien Mazarguil 			goto port_error;
844771fa900SAdrien Mazarguil 		}
845771fa900SAdrien Mazarguil 		priv->ctx = ctx;
84687ec44ceSXueming Li 		strncpy(priv->ibdev_path, priv->ctx->device->ibdev_path,
84787ec44ceSXueming Li 			sizeof(priv->ibdev_path));
848771fa900SAdrien Mazarguil 		priv->device_attr = device_attr;
849771fa900SAdrien Mazarguil 		priv->port = port;
850771fa900SAdrien Mazarguil 		priv->pd = pd;
851771fa900SAdrien Mazarguil 		priv->mtu = ETHER_MTU;
8527fe24446SShahaf Shuler 		err = mlx5_args(&config, pci_dev->device.devargs);
853e72dd09bSNélio Laranjeiro 		if (err) {
854a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "failed to process device arguments: %s",
855e72dd09bSNélio Laranjeiro 				strerror(err));
856e72dd09bSNélio Laranjeiro 			goto port_error;
857e72dd09bSNélio Laranjeiro 		}
8580e83b8e5SNelio Laranjeiro 		if (mlx5_glue->query_device_ex(ctx, NULL, &device_attr_ex)) {
859a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "ibv_query_device_ex() failed");
860a6d83b6aSNélio Laranjeiro 			err = errno;
861771fa900SAdrien Mazarguil 			goto port_error;
862771fa900SAdrien Mazarguil 		}
8637fe24446SShahaf Shuler 		config.hw_csum = !!(device_attr_ex.device_cap_flags_ex &
86443e9d979SShachar Beiser 				    IBV_DEVICE_RAW_IP_CSUM);
865a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "checksum offloading is %ssupported",
8667fe24446SShahaf Shuler 			(config.hw_csum ? "" : "not "));
8679a761de8SOri Kam #ifdef HAVE_IBV_DEVICE_COUNTERS_SET_SUPPORT
86873b620f2SNelio Laranjeiro 		config.flow_counter_en = !!(device_attr.max_counter_sets);
8690e83b8e5SNelio Laranjeiro 		mlx5_glue->describe_counter_set(ctx, 0, &cs_desc);
870a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG,
871a170a30dSNélio Laranjeiro 			"counter type = %d, num of cs = %ld, attributes = %d",
8729a761de8SOri Kam 			cs_desc.counter_type, cs_desc.num_of_cs,
8739a761de8SOri Kam 			cs_desc.attributes);
8749a761de8SOri Kam #endif
8757fe24446SShahaf Shuler 		config.ind_table_max_size =
87643e9d979SShachar Beiser 			device_attr_ex.rss_caps.max_rwq_indirection_table_size;
87713d57bd5SAdrien Mazarguil 		/* Remove this check once DPDK supports larger/variable
87813d57bd5SAdrien Mazarguil 		 * indirection tables. */
8797fe24446SShahaf Shuler 		if (config.ind_table_max_size >
880ec1fed22SYongseok Koh 				(unsigned int)ETH_RSS_RETA_SIZE_512)
8817fe24446SShahaf Shuler 			config.ind_table_max_size = ETH_RSS_RETA_SIZE_512;
882a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "maximum Rx indirection table size is %u",
8837fe24446SShahaf Shuler 			config.ind_table_max_size);
8847fe24446SShahaf Shuler 		config.hw_vlan_strip = !!(device_attr_ex.raw_packet_caps &
88543e9d979SShachar Beiser 					 IBV_RAW_PACKET_CAP_CVLAN_STRIPPING);
886a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "VLAN stripping is %ssupported",
8877fe24446SShahaf Shuler 			(config.hw_vlan_strip ? "" : "not "));
88895e16ef3SNelio Laranjeiro 
889cd230a3eSShahaf Shuler 		config.hw_fcs_strip = !!(device_attr_ex.raw_packet_caps &
890cd230a3eSShahaf Shuler 					 IBV_RAW_PACKET_CAP_SCATTER_FCS);
891a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "FCS stripping configuration is %ssupported",
8927fe24446SShahaf Shuler 			(config.hw_fcs_strip ? "" : "not "));
8934d326709SOlga Shern 
89443e9d979SShachar Beiser #ifdef HAVE_IBV_WQ_FLAG_RX_END_PADDING
8957fe24446SShahaf Shuler 		config.hw_padding = !!device_attr_ex.rx_pad_end_addr_align;
89643e9d979SShachar Beiser #endif
897a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG,
898a170a30dSNélio Laranjeiro 			"hardware Rx end alignment padding is %ssupported",
8997fe24446SShahaf Shuler 			(config.hw_padding ? "" : "not "));
900ccdcba53SNélio Laranjeiro 		config.vf = vf;
9017fe24446SShahaf Shuler 		config.tso = ((device_attr_ex.tso_caps.max_tso > 0) &&
90243e9d979SShachar Beiser 			      (device_attr_ex.tso_caps.supported_qpts &
90343e9d979SShachar Beiser 			      (1 << IBV_QPT_RAW_PACKET)));
9047fe24446SShahaf Shuler 		if (config.tso)
9057fe24446SShahaf Shuler 			config.tso_max_payload_sz =
90643e9d979SShachar Beiser 					device_attr_ex.tso_caps.max_tso;
9077fe24446SShahaf Shuler 		if (config.mps && !mps) {
908a170a30dSNélio Laranjeiro 			DRV_LOG(ERR,
909a170a30dSNélio Laranjeiro 				"multi-packet send not supported on this device"
910230189d9SNélio Laranjeiro 				" (" MLX5_TXQ_MPW_EN ")");
911230189d9SNélio Laranjeiro 			err = ENOTSUP;
912230189d9SNélio Laranjeiro 			goto port_error;
913230189d9SNélio Laranjeiro 		}
914a170a30dSNélio Laranjeiro 		DRV_LOG(INFO, "%s MPS is %s",
9150f99970bSNélio Laranjeiro 			config.mps == MLX5_MPW_ENHANCED ? "enhanced " : "",
916a170a30dSNélio Laranjeiro 			config.mps != MLX5_MPW_DISABLED ? "enabled" :
917a170a30dSNélio Laranjeiro 			"disabled");
9187fe24446SShahaf Shuler 		if (config.cqe_comp && !cqe_comp) {
919a170a30dSNélio Laranjeiro 			DRV_LOG(WARNING, "Rx CQE compression isn't supported");
9207fe24446SShahaf Shuler 			config.cqe_comp = 0;
921523f5a74SYongseok Koh 		}
922af4f09f2SNélio Laranjeiro 		eth_dev = rte_eth_dev_allocate(name);
923af4f09f2SNélio Laranjeiro 		if (eth_dev == NULL) {
924a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "can not allocate rte ethdev");
925af4f09f2SNélio Laranjeiro 			err = ENOMEM;
926af4f09f2SNélio Laranjeiro 			goto port_error;
927af4f09f2SNélio Laranjeiro 		}
928af4f09f2SNélio Laranjeiro 		eth_dev->data->dev_private = priv;
929af4f09f2SNélio Laranjeiro 		priv->dev = eth_dev;
930af4f09f2SNélio Laranjeiro 		eth_dev->data->mac_addrs = priv->mac;
931af4f09f2SNélio Laranjeiro 		eth_dev->device = &pci_dev->device;
932af4f09f2SNélio Laranjeiro 		rte_eth_copy_pci_info(eth_dev, pci_dev);
933af4f09f2SNélio Laranjeiro 		eth_dev->device->driver = &mlx5_driver.driver;
934af4f09f2SNélio Laranjeiro 		err = mlx5_uar_init_primary(eth_dev);
9354a984153SXueming Li 		if (err)
9364a984153SXueming Li 			goto port_error;
937771fa900SAdrien Mazarguil 		/* Configure the first MAC address by default. */
938af4f09f2SNélio Laranjeiro 		if (mlx5_get_mac(eth_dev, &mac.addr_bytes)) {
939a170a30dSNélio Laranjeiro 			DRV_LOG(ERR,
940a170a30dSNélio Laranjeiro 				"port %u cannot get MAC address, is mlx5_en"
941a170a30dSNélio Laranjeiro 				" loaded? (errno: %s)",
942a170a30dSNélio Laranjeiro 				eth_dev->data->port_id, strerror(errno));
943e1c3e305SMatan Azrad 			err = ENODEV;
944771fa900SAdrien Mazarguil 			goto port_error;
945771fa900SAdrien Mazarguil 		}
946a170a30dSNélio Laranjeiro 		DRV_LOG(INFO,
947a170a30dSNélio Laranjeiro 			"port %u MAC address is %02x:%02x:%02x:%02x:%02x:%02x",
9480f99970bSNélio Laranjeiro 			eth_dev->data->port_id,
949771fa900SAdrien Mazarguil 			mac.addr_bytes[0], mac.addr_bytes[1],
950771fa900SAdrien Mazarguil 			mac.addr_bytes[2], mac.addr_bytes[3],
951771fa900SAdrien Mazarguil 			mac.addr_bytes[4], mac.addr_bytes[5]);
952771fa900SAdrien Mazarguil #ifndef NDEBUG
953771fa900SAdrien Mazarguil 		{
954771fa900SAdrien Mazarguil 			char ifname[IF_NAMESIZE];
955771fa900SAdrien Mazarguil 
956af4f09f2SNélio Laranjeiro 			if (mlx5_get_ifname(eth_dev, &ifname) == 0)
957a170a30dSNélio Laranjeiro 				DRV_LOG(DEBUG, "port %u ifname is \"%s\"",
9580f99970bSNélio Laranjeiro 					eth_dev->data->port_id, ifname);
959771fa900SAdrien Mazarguil 			else
960a170a30dSNélio Laranjeiro 				DRV_LOG(DEBUG, "port %u ifname is unknown",
9610f99970bSNélio Laranjeiro 					eth_dev->data->port_id);
962771fa900SAdrien Mazarguil 		}
963771fa900SAdrien Mazarguil #endif
964771fa900SAdrien Mazarguil 		/* Get actual MTU if possible. */
965a6d83b6aSNélio Laranjeiro 		err = mlx5_get_mtu(eth_dev, &priv->mtu);
966a6d83b6aSNélio Laranjeiro 		if (err)
967a6d83b6aSNélio Laranjeiro 			goto port_error;
968a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "port %u MTU is %u", eth_dev->data->port_id,
969a170a30dSNélio Laranjeiro 			priv->mtu);
970e313ef4cSShahaf Shuler 		/*
971e313ef4cSShahaf Shuler 		 * Initialize burst functions to prevent crashes before link-up.
972e313ef4cSShahaf Shuler 		 */
973e313ef4cSShahaf Shuler 		eth_dev->rx_pkt_burst = removed_rx_burst;
974e313ef4cSShahaf Shuler 		eth_dev->tx_pkt_burst = removed_tx_burst;
975771fa900SAdrien Mazarguil 		eth_dev->dev_ops = &mlx5_dev_ops;
976272733b5SNélio Laranjeiro 		/* Register MAC address. */
977272733b5SNélio Laranjeiro 		claim_zero(mlx5_mac_addr_add(eth_dev, &mac, 0, 0));
978ccdcba53SNélio Laranjeiro 		priv->nl_socket = -1;
979ccdcba53SNélio Laranjeiro 		priv->nl_sn = 0;
980*db209cc3SNélio Laranjeiro 		if (vf && config.vf_nl_en) {
981ccdcba53SNélio Laranjeiro 			priv->nl_socket = mlx5_nl_init(RTMGRP_LINK);
982ccdcba53SNélio Laranjeiro 			if (priv->nl_socket < 0)
983ccdcba53SNélio Laranjeiro 				priv->nl_socket = -1;
984ccdcba53SNélio Laranjeiro 			mlx5_nl_mac_addr_sync(eth_dev);
985ccdcba53SNélio Laranjeiro 		}
986c8ffb8a9SNélio Laranjeiro 		TAILQ_INIT(&priv->flows);
9871b37f5d8SNélio Laranjeiro 		TAILQ_INIT(&priv->ctrl_flows);
9881e3a39f7SXueming Li 		/* Hint libmlx5 to use PMD allocator for data plane resources */
9891e3a39f7SXueming Li 		struct mlx5dv_ctx_allocators alctr = {
9901e3a39f7SXueming Li 			.alloc = &mlx5_alloc_verbs_buf,
9911e3a39f7SXueming Li 			.free = &mlx5_free_verbs_buf,
9921e3a39f7SXueming Li 			.data = priv,
9931e3a39f7SXueming Li 		};
9940e83b8e5SNelio Laranjeiro 		mlx5_glue->dv_set_context_attr(ctx,
9950e83b8e5SNelio Laranjeiro 					       MLX5DV_CTX_ATTR_BUF_ALLOCATORS,
9961e3a39f7SXueming Li 					       (void *)((uintptr_t)&alctr));
997771fa900SAdrien Mazarguil 		/* Bring Ethernet device up. */
998a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "port %u forcing Ethernet interface up",
9990f99970bSNélio Laranjeiro 			eth_dev->data->port_id);
10007ba5320bSNélio Laranjeiro 		mlx5_set_link_up(eth_dev);
10017fe24446SShahaf Shuler 		/* Store device configuration on private structure. */
10027fe24446SShahaf Shuler 		priv->config = config;
1003771fa900SAdrien Mazarguil 		continue;
1004771fa900SAdrien Mazarguil port_error:
100529c1d8bbSNélio Laranjeiro 		if (priv)
1006771fa900SAdrien Mazarguil 			rte_free(priv);
1007771fa900SAdrien Mazarguil 		if (pd)
10080e83b8e5SNelio Laranjeiro 			claim_zero(mlx5_glue->dealloc_pd(pd));
1009771fa900SAdrien Mazarguil 		if (ctx)
10100e83b8e5SNelio Laranjeiro 			claim_zero(mlx5_glue->close_device(ctx));
1011771fa900SAdrien Mazarguil 		break;
1012771fa900SAdrien Mazarguil 	}
1013771fa900SAdrien Mazarguil 	/*
1014771fa900SAdrien Mazarguil 	 * XXX if something went wrong in the loop above, there is a resource
1015771fa900SAdrien Mazarguil 	 * leak (ctx, pd, priv, dpdk ethdev) but we can do nothing about it as
1016771fa900SAdrien Mazarguil 	 * long as the dpdk does not provide a way to deallocate a ethdev and a
1017771fa900SAdrien Mazarguil 	 * way to enumerate the registered ethdevs to free the previous ones.
1018771fa900SAdrien Mazarguil 	 */
1019771fa900SAdrien Mazarguil 	/* no port found, complain */
1020771fa900SAdrien Mazarguil 	if (!mlx5_dev[idx].ports) {
1021a6d83b6aSNélio Laranjeiro 		rte_errno = ENODEV;
1022a6d83b6aSNélio Laranjeiro 		err = rte_errno;
1023771fa900SAdrien Mazarguil 	}
1024771fa900SAdrien Mazarguil error:
1025771fa900SAdrien Mazarguil 	if (attr_ctx)
10260e83b8e5SNelio Laranjeiro 		claim_zero(mlx5_glue->close_device(attr_ctx));
1027771fa900SAdrien Mazarguil 	if (list)
10280e83b8e5SNelio Laranjeiro 		mlx5_glue->free_device_list(list);
1029a6d83b6aSNélio Laranjeiro 	if (err) {
1030a6d83b6aSNélio Laranjeiro 		rte_errno = err;
1031a6d83b6aSNélio Laranjeiro 		return -rte_errno;
1032a6d83b6aSNélio Laranjeiro 	}
1033a6d83b6aSNélio Laranjeiro 	return 0;
1034771fa900SAdrien Mazarguil }
1035771fa900SAdrien Mazarguil 
1036771fa900SAdrien Mazarguil static const struct rte_pci_id mlx5_pci_id_map[] = {
1037771fa900SAdrien Mazarguil 	{
10381d1bc870SNélio Laranjeiro 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
10391d1bc870SNélio Laranjeiro 			       PCI_DEVICE_ID_MELLANOX_CONNECTX4)
1040771fa900SAdrien Mazarguil 	},
1041771fa900SAdrien Mazarguil 	{
10421d1bc870SNélio Laranjeiro 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
10431d1bc870SNélio Laranjeiro 			       PCI_DEVICE_ID_MELLANOX_CONNECTX4VF)
1044771fa900SAdrien Mazarguil 	},
1045771fa900SAdrien Mazarguil 	{
10461d1bc870SNélio Laranjeiro 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
10471d1bc870SNélio Laranjeiro 			       PCI_DEVICE_ID_MELLANOX_CONNECTX4LX)
1048771fa900SAdrien Mazarguil 	},
1049771fa900SAdrien Mazarguil 	{
10501d1bc870SNélio Laranjeiro 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
10511d1bc870SNélio Laranjeiro 			       PCI_DEVICE_ID_MELLANOX_CONNECTX4LXVF)
1052771fa900SAdrien Mazarguil 	},
1053771fa900SAdrien Mazarguil 	{
1054528a9fbeSYongseok Koh 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
1055528a9fbeSYongseok Koh 			       PCI_DEVICE_ID_MELLANOX_CONNECTX5)
1056528a9fbeSYongseok Koh 	},
1057528a9fbeSYongseok Koh 	{
1058528a9fbeSYongseok Koh 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
1059528a9fbeSYongseok Koh 			       PCI_DEVICE_ID_MELLANOX_CONNECTX5VF)
1060528a9fbeSYongseok Koh 	},
1061528a9fbeSYongseok Koh 	{
1062528a9fbeSYongseok Koh 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
1063528a9fbeSYongseok Koh 			       PCI_DEVICE_ID_MELLANOX_CONNECTX5EX)
1064528a9fbeSYongseok Koh 	},
1065528a9fbeSYongseok Koh 	{
1066528a9fbeSYongseok Koh 		RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
1067528a9fbeSYongseok Koh 			       PCI_DEVICE_ID_MELLANOX_CONNECTX5EXVF)
1068528a9fbeSYongseok Koh 	},
1069528a9fbeSYongseok Koh 	{
1070771fa900SAdrien Mazarguil 		.vendor_id = 0
1071771fa900SAdrien Mazarguil 	}
1072771fa900SAdrien Mazarguil };
1073771fa900SAdrien Mazarguil 
1074fdf91e0fSJan Blunck static struct rte_pci_driver mlx5_driver = {
10752f3193cfSJan Viktorin 	.driver = {
10762f3193cfSJan Viktorin 		.name = MLX5_DRIVER_NAME
10772f3193cfSJan Viktorin 	},
1078771fa900SAdrien Mazarguil 	.id_table = mlx5_pci_id_map,
1079af424af8SShreyansh Jain 	.probe = mlx5_pci_probe,
10807d7d7ad1SMatan Azrad 	.drv_flags = RTE_PCI_DRV_INTR_LSC | RTE_PCI_DRV_INTR_RMV,
1081771fa900SAdrien Mazarguil };
1082771fa900SAdrien Mazarguil 
108359b91becSAdrien Mazarguil #ifdef RTE_LIBRTE_MLX5_DLOPEN_DEPS
108459b91becSAdrien Mazarguil 
108559b91becSAdrien Mazarguil /**
108608c028d0SAdrien Mazarguil  * Suffix RTE_EAL_PMD_PATH with "-glue".
108708c028d0SAdrien Mazarguil  *
108808c028d0SAdrien Mazarguil  * This function performs a sanity check on RTE_EAL_PMD_PATH before
108908c028d0SAdrien Mazarguil  * suffixing its last component.
109008c028d0SAdrien Mazarguil  *
109108c028d0SAdrien Mazarguil  * @param buf[out]
109208c028d0SAdrien Mazarguil  *   Output buffer, should be large enough otherwise NULL is returned.
109308c028d0SAdrien Mazarguil  * @param size
109408c028d0SAdrien Mazarguil  *   Size of @p out.
109508c028d0SAdrien Mazarguil  *
109608c028d0SAdrien Mazarguil  * @return
109708c028d0SAdrien Mazarguil  *   Pointer to @p buf or @p NULL in case suffix cannot be appended.
109808c028d0SAdrien Mazarguil  */
109908c028d0SAdrien Mazarguil static char *
110008c028d0SAdrien Mazarguil mlx5_glue_path(char *buf, size_t size)
110108c028d0SAdrien Mazarguil {
110208c028d0SAdrien Mazarguil 	static const char *const bad[] = { "/", ".", "..", NULL };
110308c028d0SAdrien Mazarguil 	const char *path = RTE_EAL_PMD_PATH;
110408c028d0SAdrien Mazarguil 	size_t len = strlen(path);
110508c028d0SAdrien Mazarguil 	size_t off;
110608c028d0SAdrien Mazarguil 	int i;
110708c028d0SAdrien Mazarguil 
110808c028d0SAdrien Mazarguil 	while (len && path[len - 1] == '/')
110908c028d0SAdrien Mazarguil 		--len;
111008c028d0SAdrien Mazarguil 	for (off = len; off && path[off - 1] != '/'; --off)
111108c028d0SAdrien Mazarguil 		;
111208c028d0SAdrien Mazarguil 	for (i = 0; bad[i]; ++i)
111308c028d0SAdrien Mazarguil 		if (!strncmp(path + off, bad[i], (int)(len - off)))
111408c028d0SAdrien Mazarguil 			goto error;
111508c028d0SAdrien Mazarguil 	i = snprintf(buf, size, "%.*s-glue", (int)len, path);
111608c028d0SAdrien Mazarguil 	if (i == -1 || (size_t)i >= size)
111708c028d0SAdrien Mazarguil 		goto error;
111808c028d0SAdrien Mazarguil 	return buf;
111908c028d0SAdrien Mazarguil error:
1120a170a30dSNélio Laranjeiro 	DRV_LOG(ERR,
1121a170a30dSNélio Laranjeiro 		"unable to append \"-glue\" to last component of"
112208c028d0SAdrien Mazarguil 		" RTE_EAL_PMD_PATH (\"" RTE_EAL_PMD_PATH "\"),"
112308c028d0SAdrien Mazarguil 		" please re-configure DPDK");
112408c028d0SAdrien Mazarguil 	return NULL;
112508c028d0SAdrien Mazarguil }
112608c028d0SAdrien Mazarguil 
112708c028d0SAdrien Mazarguil /**
112859b91becSAdrien Mazarguil  * Initialization routine for run-time dependency on rdma-core.
112959b91becSAdrien Mazarguil  */
113059b91becSAdrien Mazarguil static int
113159b91becSAdrien Mazarguil mlx5_glue_init(void)
113259b91becSAdrien Mazarguil {
113308c028d0SAdrien Mazarguil 	char glue_path[sizeof(RTE_EAL_PMD_PATH) - 1 + sizeof("-glue")];
1134f6242d06SAdrien Mazarguil 	const char *path[] = {
1135f6242d06SAdrien Mazarguil 		/*
1136f6242d06SAdrien Mazarguil 		 * A basic security check is necessary before trusting
1137f6242d06SAdrien Mazarguil 		 * MLX5_GLUE_PATH, which may override RTE_EAL_PMD_PATH.
1138f6242d06SAdrien Mazarguil 		 */
1139f6242d06SAdrien Mazarguil 		(geteuid() == getuid() && getegid() == getgid() ?
1140f6242d06SAdrien Mazarguil 		 getenv("MLX5_GLUE_PATH") : NULL),
114108c028d0SAdrien Mazarguil 		/*
114208c028d0SAdrien Mazarguil 		 * When RTE_EAL_PMD_PATH is set, use its glue-suffixed
114308c028d0SAdrien Mazarguil 		 * variant, otherwise let dlopen() look up libraries on its
114408c028d0SAdrien Mazarguil 		 * own.
114508c028d0SAdrien Mazarguil 		 */
114608c028d0SAdrien Mazarguil 		(*RTE_EAL_PMD_PATH ?
114708c028d0SAdrien Mazarguil 		 mlx5_glue_path(glue_path, sizeof(glue_path)) : ""),
1148f6242d06SAdrien Mazarguil 	};
1149f6242d06SAdrien Mazarguil 	unsigned int i = 0;
115059b91becSAdrien Mazarguil 	void *handle = NULL;
115159b91becSAdrien Mazarguil 	void **sym;
115259b91becSAdrien Mazarguil 	const char *dlmsg;
115359b91becSAdrien Mazarguil 
1154f6242d06SAdrien Mazarguil 	while (!handle && i != RTE_DIM(path)) {
1155f6242d06SAdrien Mazarguil 		const char *end;
1156f6242d06SAdrien Mazarguil 		size_t len;
1157f6242d06SAdrien Mazarguil 		int ret;
1158f6242d06SAdrien Mazarguil 
1159f6242d06SAdrien Mazarguil 		if (!path[i]) {
1160f6242d06SAdrien Mazarguil 			++i;
1161f6242d06SAdrien Mazarguil 			continue;
1162f6242d06SAdrien Mazarguil 		}
1163f6242d06SAdrien Mazarguil 		end = strpbrk(path[i], ":;");
1164f6242d06SAdrien Mazarguil 		if (!end)
1165f6242d06SAdrien Mazarguil 			end = path[i] + strlen(path[i]);
1166f6242d06SAdrien Mazarguil 		len = end - path[i];
1167f6242d06SAdrien Mazarguil 		ret = 0;
1168f6242d06SAdrien Mazarguil 		do {
1169f6242d06SAdrien Mazarguil 			char name[ret + 1];
1170f6242d06SAdrien Mazarguil 
1171f6242d06SAdrien Mazarguil 			ret = snprintf(name, sizeof(name), "%.*s%s" MLX5_GLUE,
1172f6242d06SAdrien Mazarguil 				       (int)len, path[i],
1173f6242d06SAdrien Mazarguil 				       (!len || *(end - 1) == '/') ? "" : "/");
1174f6242d06SAdrien Mazarguil 			if (ret == -1)
1175f6242d06SAdrien Mazarguil 				break;
1176f6242d06SAdrien Mazarguil 			if (sizeof(name) != (size_t)ret + 1)
1177f6242d06SAdrien Mazarguil 				continue;
1178a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG, "looking for rdma-core glue as \"%s\"",
1179a170a30dSNélio Laranjeiro 				name);
1180f6242d06SAdrien Mazarguil 			handle = dlopen(name, RTLD_LAZY);
1181f6242d06SAdrien Mazarguil 			break;
1182f6242d06SAdrien Mazarguil 		} while (1);
1183f6242d06SAdrien Mazarguil 		path[i] = end + 1;
1184f6242d06SAdrien Mazarguil 		if (!*end)
1185f6242d06SAdrien Mazarguil 			++i;
1186f6242d06SAdrien Mazarguil 	}
118759b91becSAdrien Mazarguil 	if (!handle) {
118859b91becSAdrien Mazarguil 		rte_errno = EINVAL;
118959b91becSAdrien Mazarguil 		dlmsg = dlerror();
119059b91becSAdrien Mazarguil 		if (dlmsg)
1191a170a30dSNélio Laranjeiro 			DRV_LOG(WARNING, "cannot load glue library: %s", dlmsg);
119259b91becSAdrien Mazarguil 		goto glue_error;
119359b91becSAdrien Mazarguil 	}
119459b91becSAdrien Mazarguil 	sym = dlsym(handle, "mlx5_glue");
119559b91becSAdrien Mazarguil 	if (!sym || !*sym) {
119659b91becSAdrien Mazarguil 		rte_errno = EINVAL;
119759b91becSAdrien Mazarguil 		dlmsg = dlerror();
119859b91becSAdrien Mazarguil 		if (dlmsg)
1199a170a30dSNélio Laranjeiro 			DRV_LOG(ERR, "cannot resolve glue symbol: %s", dlmsg);
120059b91becSAdrien Mazarguil 		goto glue_error;
120159b91becSAdrien Mazarguil 	}
120259b91becSAdrien Mazarguil 	mlx5_glue = *sym;
120359b91becSAdrien Mazarguil 	return 0;
120459b91becSAdrien Mazarguil glue_error:
120559b91becSAdrien Mazarguil 	if (handle)
120659b91becSAdrien Mazarguil 		dlclose(handle);
1207a170a30dSNélio Laranjeiro 	DRV_LOG(WARNING,
1208a170a30dSNélio Laranjeiro 		"cannot initialize PMD due to missing run-time dependency on"
1209a170a30dSNélio Laranjeiro 		" rdma-core libraries (libibverbs, libmlx5)");
121059b91becSAdrien Mazarguil 	return -rte_errno;
121159b91becSAdrien Mazarguil }
121259b91becSAdrien Mazarguil 
121359b91becSAdrien Mazarguil #endif
121459b91becSAdrien Mazarguil 
1215771fa900SAdrien Mazarguil /**
1216771fa900SAdrien Mazarguil  * Driver initialization routine.
1217771fa900SAdrien Mazarguil  */
1218c830cb29SDavid Marchand RTE_INIT(rte_mlx5_pmd_init);
1219c830cb29SDavid Marchand static void
1220c830cb29SDavid Marchand rte_mlx5_pmd_init(void)
1221771fa900SAdrien Mazarguil {
1222ea16068cSYongseok Koh 	/* Build the static table for ptype conversion. */
1223ea16068cSYongseok Koh 	mlx5_set_ptype_table();
1224771fa900SAdrien Mazarguil 	/*
1225771fa900SAdrien Mazarguil 	 * RDMAV_HUGEPAGES_SAFE tells ibv_fork_init() we intend to use
1226771fa900SAdrien Mazarguil 	 * huge pages. Calling ibv_fork_init() during init allows
1227771fa900SAdrien Mazarguil 	 * applications to use fork() safely for purposes other than
1228771fa900SAdrien Mazarguil 	 * using this PMD, which is not supported in forked processes.
1229771fa900SAdrien Mazarguil 	 */
1230771fa900SAdrien Mazarguil 	setenv("RDMAV_HUGEPAGES_SAFE", "1", 1);
1231161b93e5SYongseok Koh 	/* Match the size of Rx completion entry to the size of a cacheline. */
1232161b93e5SYongseok Koh 	if (RTE_CACHE_LINE_SIZE == 128)
1233161b93e5SYongseok Koh 		setenv("MLX5_CQE_SIZE", "128", 0);
123459b91becSAdrien Mazarguil #ifdef RTE_LIBRTE_MLX5_DLOPEN_DEPS
123559b91becSAdrien Mazarguil 	if (mlx5_glue_init())
123659b91becSAdrien Mazarguil 		return;
123759b91becSAdrien Mazarguil 	assert(mlx5_glue);
123859b91becSAdrien Mazarguil #endif
12392a3b0097SAdrien Mazarguil #ifndef NDEBUG
12402a3b0097SAdrien Mazarguil 	/* Glue structure must not contain any NULL pointers. */
12412a3b0097SAdrien Mazarguil 	{
12422a3b0097SAdrien Mazarguil 		unsigned int i;
12432a3b0097SAdrien Mazarguil 
12442a3b0097SAdrien Mazarguil 		for (i = 0; i != sizeof(*mlx5_glue) / sizeof(void *); ++i)
12452a3b0097SAdrien Mazarguil 			assert(((const void *const *)mlx5_glue)[i]);
12462a3b0097SAdrien Mazarguil 	}
12472a3b0097SAdrien Mazarguil #endif
12486d5df2eaSAdrien Mazarguil 	if (strcmp(mlx5_glue->version, MLX5_GLUE_VERSION)) {
1249a170a30dSNélio Laranjeiro 		DRV_LOG(ERR,
1250a170a30dSNélio Laranjeiro 			"rdma-core glue \"%s\" mismatch: \"%s\" is required",
12516d5df2eaSAdrien Mazarguil 			mlx5_glue->version, MLX5_GLUE_VERSION);
12526d5df2eaSAdrien Mazarguil 		return;
12536d5df2eaSAdrien Mazarguil 	}
12540e83b8e5SNelio Laranjeiro 	mlx5_glue->fork_init();
12553dcfe039SThomas Monjalon 	rte_pci_register(&mlx5_driver);
1256771fa900SAdrien Mazarguil }
1257771fa900SAdrien Mazarguil 
125801f19227SShreyansh Jain RTE_PMD_EXPORT_NAME(net_mlx5, __COUNTER__);
125901f19227SShreyansh Jain RTE_PMD_REGISTER_PCI_TABLE(net_mlx5, mlx5_pci_id_map);
12600880c401SOlivier Matz RTE_PMD_REGISTER_KMOD_DEP(net_mlx5, "* ib_uverbs & mlx5_core & mlx5_ib");
1261a170a30dSNélio Laranjeiro 
1262a170a30dSNélio Laranjeiro /** Initialize driver log type. */
1263a170a30dSNélio Laranjeiro RTE_INIT(vdev_netvsc_init_log)
1264a170a30dSNélio Laranjeiro {
1265a170a30dSNélio Laranjeiro 	mlx5_logtype = rte_log_register("pmd.net.mlx5");
1266a170a30dSNélio Laranjeiro 	if (mlx5_logtype >= 0)
1267a170a30dSNélio Laranjeiro 		rte_log_set_level(mlx5_logtype, RTE_LOG_NOTICE);
1268a170a30dSNélio Laranjeiro }
1269