xref: /dpdk/drivers/net/mlx5/mlx5_ethdev.c (revision 79d0989213cd39decfe802cc42e334db4f0978f0)
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 
6f8b9a3baSXueming Li #define _GNU_SOURCE
7f8b9a3baSXueming Li 
8771fa900SAdrien Mazarguil #include <stddef.h>
9e60fbd5bSAdrien Mazarguil #include <assert.h>
10fc40db99SAdrien Mazarguil #include <inttypes.h>
11771fa900SAdrien Mazarguil #include <unistd.h>
12771fa900SAdrien Mazarguil #include <stdint.h>
13771fa900SAdrien Mazarguil #include <stdio.h>
14771fa900SAdrien Mazarguil #include <string.h>
15771fa900SAdrien Mazarguil #include <stdlib.h>
16771fa900SAdrien Mazarguil #include <errno.h>
17771fa900SAdrien Mazarguil #include <dirent.h>
18771fa900SAdrien Mazarguil #include <net/if.h>
19771fa900SAdrien Mazarguil #include <sys/ioctl.h>
20771fa900SAdrien Mazarguil #include <sys/socket.h>
21771fa900SAdrien Mazarguil #include <netinet/in.h>
22cb8faed7SAdrien Mazarguil #include <linux/ethtool.h>
23cb8faed7SAdrien Mazarguil #include <linux/sockios.h>
24198a3c33SNelio Laranjeiro #include <fcntl.h>
2590260d57SNélio Laranjeiro #include <stdalign.h>
26f8b9a3baSXueming Li #include <sys/un.h>
27cfee9475SNélio Laranjeiro #include <time.h>
28771fa900SAdrien Mazarguil 
29771fa900SAdrien Mazarguil #include <rte_atomic.h>
30ffc905f3SFerruh Yigit #include <rte_ethdev_driver.h>
31c752998bSGaetan Rivet #include <rte_bus_pci.h>
32771fa900SAdrien Mazarguil #include <rte_mbuf.h>
33771fa900SAdrien Mazarguil #include <rte_common.h>
34198a3c33SNelio Laranjeiro #include <rte_interrupts.h>
35a48deadaSOr Ami #include <rte_malloc.h>
36c022cb40SBruce Richardson #include <rte_string_fns.h>
37974f1e7eSYongseok Koh #include <rte_rwlock.h>
38771fa900SAdrien Mazarguil 
39771fa900SAdrien Mazarguil #include "mlx5.h"
400e83b8e5SNelio Laranjeiro #include "mlx5_glue.h"
41e60fbd5bSAdrien Mazarguil #include "mlx5_rxtx.h"
42771fa900SAdrien Mazarguil #include "mlx5_utils.h"
43771fa900SAdrien Mazarguil 
44*79d09892SMoti Haimovsky /* Supported speed values found in /usr/include/linux/ethtool.h */
45*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_40000baseKR4_Full
46*79d09892SMoti Haimovsky #define SUPPORTED_40000baseKR4_Full (1 << 23)
47*79d09892SMoti Haimovsky #endif
48*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_40000baseCR4_Full
49*79d09892SMoti Haimovsky #define SUPPORTED_40000baseCR4_Full (1 << 24)
50*79d09892SMoti Haimovsky #endif
51*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_40000baseSR4_Full
52*79d09892SMoti Haimovsky #define SUPPORTED_40000baseSR4_Full (1 << 25)
53*79d09892SMoti Haimovsky #endif
54*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_40000baseLR4_Full
55*79d09892SMoti Haimovsky #define SUPPORTED_40000baseLR4_Full (1 << 26)
56*79d09892SMoti Haimovsky #endif
57*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_56000baseKR4_Full
58*79d09892SMoti Haimovsky #define SUPPORTED_56000baseKR4_Full (1 << 27)
59*79d09892SMoti Haimovsky #endif
60*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_56000baseCR4_Full
61*79d09892SMoti Haimovsky #define SUPPORTED_56000baseCR4_Full (1 << 28)
62*79d09892SMoti Haimovsky #endif
63*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_56000baseSR4_Full
64*79d09892SMoti Haimovsky #define SUPPORTED_56000baseSR4_Full (1 << 29)
65*79d09892SMoti Haimovsky #endif
66*79d09892SMoti Haimovsky #ifndef HAVE_SUPPORTED_56000baseLR4_Full
67*79d09892SMoti Haimovsky #define SUPPORTED_56000baseLR4_Full (1 << 30)
68*79d09892SMoti Haimovsky #endif
69*79d09892SMoti Haimovsky 
703a49ffe3SShahaf Shuler /* Add defines in case the running kernel is not the same as user headers. */
713a49ffe3SShahaf Shuler #ifndef ETHTOOL_GLINKSETTINGS
723a49ffe3SShahaf Shuler struct ethtool_link_settings {
733a49ffe3SShahaf Shuler 	uint32_t cmd;
743a49ffe3SShahaf Shuler 	uint32_t speed;
753a49ffe3SShahaf Shuler 	uint8_t duplex;
763a49ffe3SShahaf Shuler 	uint8_t port;
773a49ffe3SShahaf Shuler 	uint8_t phy_address;
783a49ffe3SShahaf Shuler 	uint8_t autoneg;
793a49ffe3SShahaf Shuler 	uint8_t mdio_support;
803a49ffe3SShahaf Shuler 	uint8_t eth_to_mdix;
813a49ffe3SShahaf Shuler 	uint8_t eth_tp_mdix_ctrl;
823a49ffe3SShahaf Shuler 	int8_t link_mode_masks_nwords;
833a49ffe3SShahaf Shuler 	uint32_t reserved[8];
843a49ffe3SShahaf Shuler 	uint32_t link_mode_masks[];
853a49ffe3SShahaf Shuler };
863a49ffe3SShahaf Shuler 
873a49ffe3SShahaf Shuler #define ETHTOOL_GLINKSETTINGS 0x0000004c
883a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_1000baseT_Full_BIT 5
893a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_Autoneg_BIT 6
903a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_1000baseKX_Full_BIT 17
913a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_10000baseKX4_Full_BIT 18
923a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_10000baseKR_Full_BIT 19
933a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_10000baseR_FEC_BIT 20
943a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_20000baseMLD2_Full_BIT 21
953a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_20000baseKR2_Full_BIT 22
963a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT 23
973a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT 24
983a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_40000baseSR4_Full_BIT 25
993a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_40000baseLR4_Full_BIT 26
1003a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_56000baseKR4_Full_BIT 27
1013a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_56000baseCR4_Full_BIT 28
1023a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_56000baseSR4_Full_BIT 29
1033a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_56000baseLR4_Full_BIT 30
1043a49ffe3SShahaf Shuler #endif
1053a49ffe3SShahaf Shuler #ifndef HAVE_ETHTOOL_LINK_MODE_25G
1063a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_25000baseCR_Full_BIT 31
1073a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_25000baseKR_Full_BIT 32
1083a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_25000baseSR_Full_BIT 33
1093a49ffe3SShahaf Shuler #endif
1103a49ffe3SShahaf Shuler #ifndef HAVE_ETHTOOL_LINK_MODE_50G
1113a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_50000baseCR2_Full_BIT 34
1123a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_50000baseKR2_Full_BIT 35
1133a49ffe3SShahaf Shuler #endif
1143a49ffe3SShahaf Shuler #ifndef HAVE_ETHTOOL_LINK_MODE_100G
1153a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT 36
1163a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_100000baseSR4_Full_BIT 37
1173a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT 38
1183a49ffe3SShahaf Shuler #define ETHTOOL_LINK_MODE_100000baseLR4_ER4_Full_BIT 39
1193a49ffe3SShahaf Shuler #endif
1203a49ffe3SShahaf Shuler 
121771fa900SAdrien Mazarguil /**
1222b730263SAdrien Mazarguil  * Get master interface name from private structure.
123771fa900SAdrien Mazarguil  *
124af4f09f2SNélio Laranjeiro  * @param[in] dev
125af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
126771fa900SAdrien Mazarguil  * @param[out] ifname
127771fa900SAdrien Mazarguil  *   Interface name output buffer.
128771fa900SAdrien Mazarguil  *
129771fa900SAdrien Mazarguil  * @return
130a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
131771fa900SAdrien Mazarguil  */
132771fa900SAdrien Mazarguil int
1332b730263SAdrien Mazarguil mlx5_get_master_ifname(const struct rte_eth_dev *dev,
1342b730263SAdrien Mazarguil 		       char (*ifname)[IF_NAMESIZE])
135771fa900SAdrien Mazarguil {
136af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
137771fa900SAdrien Mazarguil 	DIR *dir;
138771fa900SAdrien Mazarguil 	struct dirent *dent;
139771fa900SAdrien Mazarguil 	unsigned int dev_type = 0;
140771fa900SAdrien Mazarguil 	unsigned int dev_port_prev = ~0u;
141771fa900SAdrien Mazarguil 	char match[IF_NAMESIZE] = "";
142771fa900SAdrien Mazarguil 
143771fa900SAdrien Mazarguil 	{
14487ec44ceSXueming Li 		MKSTR(path, "%s/device/net", priv->ibdev_path);
145771fa900SAdrien Mazarguil 
146771fa900SAdrien Mazarguil 		dir = opendir(path);
147a6d83b6aSNélio Laranjeiro 		if (dir == NULL) {
148a6d83b6aSNélio Laranjeiro 			rte_errno = errno;
149a6d83b6aSNélio Laranjeiro 			return -rte_errno;
150a6d83b6aSNélio Laranjeiro 		}
151771fa900SAdrien Mazarguil 	}
152771fa900SAdrien Mazarguil 	while ((dent = readdir(dir)) != NULL) {
153771fa900SAdrien Mazarguil 		char *name = dent->d_name;
154771fa900SAdrien Mazarguil 		FILE *file;
155771fa900SAdrien Mazarguil 		unsigned int dev_port;
156771fa900SAdrien Mazarguil 		int r;
157771fa900SAdrien Mazarguil 
158771fa900SAdrien Mazarguil 		if ((name[0] == '.') &&
159771fa900SAdrien Mazarguil 		    ((name[1] == '\0') ||
160771fa900SAdrien Mazarguil 		     ((name[1] == '.') && (name[2] == '\0'))))
161771fa900SAdrien Mazarguil 			continue;
162771fa900SAdrien Mazarguil 
163771fa900SAdrien Mazarguil 		MKSTR(path, "%s/device/net/%s/%s",
16487ec44ceSXueming Li 		      priv->ibdev_path, name,
165771fa900SAdrien Mazarguil 		      (dev_type ? "dev_id" : "dev_port"));
166771fa900SAdrien Mazarguil 
167771fa900SAdrien Mazarguil 		file = fopen(path, "rb");
168771fa900SAdrien Mazarguil 		if (file == NULL) {
169771fa900SAdrien Mazarguil 			if (errno != ENOENT)
170771fa900SAdrien Mazarguil 				continue;
171771fa900SAdrien Mazarguil 			/*
172771fa900SAdrien Mazarguil 			 * Switch to dev_id when dev_port does not exist as
173771fa900SAdrien Mazarguil 			 * is the case with Linux kernel versions < 3.15.
174771fa900SAdrien Mazarguil 			 */
175771fa900SAdrien Mazarguil try_dev_id:
176771fa900SAdrien Mazarguil 			match[0] = '\0';
177771fa900SAdrien Mazarguil 			if (dev_type)
178771fa900SAdrien Mazarguil 				break;
179771fa900SAdrien Mazarguil 			dev_type = 1;
180771fa900SAdrien Mazarguil 			dev_port_prev = ~0u;
181771fa900SAdrien Mazarguil 			rewinddir(dir);
182771fa900SAdrien Mazarguil 			continue;
183771fa900SAdrien Mazarguil 		}
184771fa900SAdrien Mazarguil 		r = fscanf(file, (dev_type ? "%x" : "%u"), &dev_port);
185771fa900SAdrien Mazarguil 		fclose(file);
186771fa900SAdrien Mazarguil 		if (r != 1)
187771fa900SAdrien Mazarguil 			continue;
188771fa900SAdrien Mazarguil 		/*
189771fa900SAdrien Mazarguil 		 * Switch to dev_id when dev_port returns the same value for
190771fa900SAdrien Mazarguil 		 * all ports. May happen when using a MOFED release older than
191771fa900SAdrien Mazarguil 		 * 3.0 with a Linux kernel >= 3.15.
192771fa900SAdrien Mazarguil 		 */
193771fa900SAdrien Mazarguil 		if (dev_port == dev_port_prev)
194771fa900SAdrien Mazarguil 			goto try_dev_id;
195771fa900SAdrien Mazarguil 		dev_port_prev = dev_port;
1969083982cSAdrien Mazarguil 		if (dev_port == 0)
197c022cb40SBruce Richardson 			strlcpy(match, name, sizeof(match));
198771fa900SAdrien Mazarguil 	}
199771fa900SAdrien Mazarguil 	closedir(dir);
200a6d83b6aSNélio Laranjeiro 	if (match[0] == '\0') {
201a6d83b6aSNélio Laranjeiro 		rte_errno = ENOENT;
202a6d83b6aSNélio Laranjeiro 		return -rte_errno;
203a6d83b6aSNélio Laranjeiro 	}
204771fa900SAdrien Mazarguil 	strncpy(*ifname, match, sizeof(*ifname));
205771fa900SAdrien Mazarguil 	return 0;
206771fa900SAdrien Mazarguil }
207771fa900SAdrien Mazarguil 
208771fa900SAdrien Mazarguil /**
2092b730263SAdrien Mazarguil  * Get interface name from private structure.
2102b730263SAdrien Mazarguil  *
2112b730263SAdrien Mazarguil  * This is a port representor-aware version of mlx5_get_master_ifname().
2122b730263SAdrien Mazarguil  *
2132b730263SAdrien Mazarguil  * @param[in] dev
2142b730263SAdrien Mazarguil  *   Pointer to Ethernet device.
2152b730263SAdrien Mazarguil  * @param[out] ifname
2162b730263SAdrien Mazarguil  *   Interface name output buffer.
2172b730263SAdrien Mazarguil  *
2182b730263SAdrien Mazarguil  * @return
2192b730263SAdrien Mazarguil  *   0 on success, a negative errno value otherwise and rte_errno is set.
2202b730263SAdrien Mazarguil  */
2212b730263SAdrien Mazarguil int
2222b730263SAdrien Mazarguil mlx5_get_ifname(const struct rte_eth_dev *dev, char (*ifname)[IF_NAMESIZE])
2232b730263SAdrien Mazarguil {
2242b730263SAdrien Mazarguil 	struct priv *priv = dev->data->dev_private;
2252b730263SAdrien Mazarguil 	unsigned int ifindex =
2262b730263SAdrien Mazarguil 		priv->nl_socket_rdma >= 0 ?
2272b730263SAdrien Mazarguil 		mlx5_nl_ifindex(priv->nl_socket_rdma, priv->ibdev_name) : 0;
2282b730263SAdrien Mazarguil 
2292b730263SAdrien Mazarguil 	if (!ifindex) {
2302b730263SAdrien Mazarguil 		if (!priv->representor)
2312b730263SAdrien Mazarguil 			return mlx5_get_master_ifname(dev, ifname);
2322b730263SAdrien Mazarguil 		rte_errno = ENXIO;
2332b730263SAdrien Mazarguil 		return -rte_errno;
2342b730263SAdrien Mazarguil 	}
2352b730263SAdrien Mazarguil 	if (if_indextoname(ifindex, &(*ifname)[0]))
2362b730263SAdrien Mazarguil 		return 0;
2372b730263SAdrien Mazarguil 	rte_errno = errno;
2382b730263SAdrien Mazarguil 	return -rte_errno;
2392b730263SAdrien Mazarguil }
2402b730263SAdrien Mazarguil 
2412b730263SAdrien Mazarguil /**
242ccdcba53SNélio Laranjeiro  * Get the interface index from device name.
243ccdcba53SNélio Laranjeiro  *
244ccdcba53SNélio Laranjeiro  * @param[in] dev
245ccdcba53SNélio Laranjeiro  *   Pointer to Ethernet device.
246ccdcba53SNélio Laranjeiro  *
247ccdcba53SNélio Laranjeiro  * @return
248ccdcba53SNélio Laranjeiro  *   Interface index on success, a negative errno value otherwise and
249ccdcba53SNélio Laranjeiro  *   rte_errno is set.
250ccdcba53SNélio Laranjeiro  */
251ccdcba53SNélio Laranjeiro int
252ccdcba53SNélio Laranjeiro mlx5_ifindex(const struct rte_eth_dev *dev)
253ccdcba53SNélio Laranjeiro {
254ccdcba53SNélio Laranjeiro 	char ifname[IF_NAMESIZE];
255f264c798SAdrien Mazarguil 	unsigned int ret;
256ccdcba53SNélio Laranjeiro 
257ccdcba53SNélio Laranjeiro 	ret = mlx5_get_ifname(dev, &ifname);
258ccdcba53SNélio Laranjeiro 	if (ret)
259ccdcba53SNélio Laranjeiro 		return ret;
260ccdcba53SNélio Laranjeiro 	ret = if_nametoindex(ifname);
261f264c798SAdrien Mazarguil 	if (ret == 0) {
262ccdcba53SNélio Laranjeiro 		rte_errno = errno;
263ccdcba53SNélio Laranjeiro 		return -rte_errno;
264ccdcba53SNélio Laranjeiro 	}
265ccdcba53SNélio Laranjeiro 	return ret;
266ccdcba53SNélio Laranjeiro }
267ccdcba53SNélio Laranjeiro 
268ccdcba53SNélio Laranjeiro /**
269771fa900SAdrien Mazarguil  * Perform ifreq ioctl() on associated Ethernet device.
270771fa900SAdrien Mazarguil  *
271af4f09f2SNélio Laranjeiro  * @param[in] dev
272af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
273771fa900SAdrien Mazarguil  * @param req
274771fa900SAdrien Mazarguil  *   Request number to pass to ioctl().
275771fa900SAdrien Mazarguil  * @param[out] ifr
276771fa900SAdrien Mazarguil  *   Interface request structure output buffer.
2772b730263SAdrien Mazarguil  * @param master
2782b730263SAdrien Mazarguil  *   When device is a port representor, perform request on master device
2792b730263SAdrien Mazarguil  *   instead.
280771fa900SAdrien Mazarguil  *
281771fa900SAdrien Mazarguil  * @return
282a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
283771fa900SAdrien Mazarguil  */
284771fa900SAdrien Mazarguil int
2852b730263SAdrien Mazarguil mlx5_ifreq(const struct rte_eth_dev *dev, int req, struct ifreq *ifr,
2862b730263SAdrien Mazarguil 	   int master)
287771fa900SAdrien Mazarguil {
288771fa900SAdrien Mazarguil 	int sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP);
289a6d83b6aSNélio Laranjeiro 	int ret = 0;
290771fa900SAdrien Mazarguil 
291a6d83b6aSNélio Laranjeiro 	if (sock == -1) {
292a6d83b6aSNélio Laranjeiro 		rte_errno = errno;
293a6d83b6aSNélio Laranjeiro 		return -rte_errno;
294a6d83b6aSNélio Laranjeiro 	}
2952b730263SAdrien Mazarguil 	if (master)
2962b730263SAdrien Mazarguil 		ret = mlx5_get_master_ifname(dev, &ifr->ifr_name);
2972b730263SAdrien Mazarguil 	else
298a6d83b6aSNélio Laranjeiro 		ret = mlx5_get_ifname(dev, &ifr->ifr_name);
299a6d83b6aSNélio Laranjeiro 	if (ret)
300a6d83b6aSNélio Laranjeiro 		goto error;
301771fa900SAdrien Mazarguil 	ret = ioctl(sock, req, ifr);
302a6d83b6aSNélio Laranjeiro 	if (ret == -1) {
303a6d83b6aSNélio Laranjeiro 		rte_errno = errno;
304a6d83b6aSNélio Laranjeiro 		goto error;
305a6d83b6aSNélio Laranjeiro 	}
306771fa900SAdrien Mazarguil 	close(sock);
307a6d83b6aSNélio Laranjeiro 	return 0;
308a6d83b6aSNélio Laranjeiro error:
309a6d83b6aSNélio Laranjeiro 	close(sock);
310a6d83b6aSNélio Laranjeiro 	return -rte_errno;
311771fa900SAdrien Mazarguil }
312771fa900SAdrien Mazarguil 
313771fa900SAdrien Mazarguil /**
314771fa900SAdrien Mazarguil  * Get device MTU.
315771fa900SAdrien Mazarguil  *
316af4f09f2SNélio Laranjeiro  * @param dev
317af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
318771fa900SAdrien Mazarguil  * @param[out] mtu
319771fa900SAdrien Mazarguil  *   MTU value output buffer.
320771fa900SAdrien Mazarguil  *
321771fa900SAdrien Mazarguil  * @return
322a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
323771fa900SAdrien Mazarguil  */
324771fa900SAdrien Mazarguil int
325af4f09f2SNélio Laranjeiro mlx5_get_mtu(struct rte_eth_dev *dev, uint16_t *mtu)
326771fa900SAdrien Mazarguil {
327fc40db99SAdrien Mazarguil 	struct ifreq request;
3282b730263SAdrien Mazarguil 	int ret = mlx5_ifreq(dev, SIOCGIFMTU, &request, 0);
329771fa900SAdrien Mazarguil 
330fc40db99SAdrien Mazarguil 	if (ret)
331fc40db99SAdrien Mazarguil 		return ret;
332fc40db99SAdrien Mazarguil 	*mtu = request.ifr_mtu;
333859081d3SShahaf Shuler 	return 0;
334859081d3SShahaf Shuler }
335859081d3SShahaf Shuler 
336859081d3SShahaf Shuler /**
337cf37ca95SAdrien Mazarguil  * Set device MTU.
338cf37ca95SAdrien Mazarguil  *
339af4f09f2SNélio Laranjeiro  * @param dev
340af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
341cf37ca95SAdrien Mazarguil  * @param mtu
342cf37ca95SAdrien Mazarguil  *   MTU value to set.
343cf37ca95SAdrien Mazarguil  *
344cf37ca95SAdrien Mazarguil  * @return
345a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
346cf37ca95SAdrien Mazarguil  */
347cf37ca95SAdrien Mazarguil static int
348af4f09f2SNélio Laranjeiro mlx5_set_mtu(struct rte_eth_dev *dev, uint16_t mtu)
349cf37ca95SAdrien Mazarguil {
350fc40db99SAdrien Mazarguil 	struct ifreq request = { .ifr_mtu = mtu, };
351f3b492d7SAdrien Mazarguil 
3522b730263SAdrien Mazarguil 	return mlx5_ifreq(dev, SIOCSIFMTU, &request, 0);
353cf37ca95SAdrien Mazarguil }
354cf37ca95SAdrien Mazarguil 
355cf37ca95SAdrien Mazarguil /**
356771fa900SAdrien Mazarguil  * Set device flags.
357771fa900SAdrien Mazarguil  *
358af4f09f2SNélio Laranjeiro  * @param dev
359af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
360771fa900SAdrien Mazarguil  * @param keep
361771fa900SAdrien Mazarguil  *   Bitmask for flags that must remain untouched.
362771fa900SAdrien Mazarguil  * @param flags
363771fa900SAdrien Mazarguil  *   Bitmask for flags to modify.
364771fa900SAdrien Mazarguil  *
365771fa900SAdrien Mazarguil  * @return
366a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
367771fa900SAdrien Mazarguil  */
368771fa900SAdrien Mazarguil int
369af4f09f2SNélio Laranjeiro mlx5_set_flags(struct rte_eth_dev *dev, unsigned int keep, unsigned int flags)
370771fa900SAdrien Mazarguil {
371fc40db99SAdrien Mazarguil 	struct ifreq request;
3722b730263SAdrien Mazarguil 	int ret = mlx5_ifreq(dev, SIOCGIFFLAGS, &request, 0);
373771fa900SAdrien Mazarguil 
374fc40db99SAdrien Mazarguil 	if (ret)
375fc40db99SAdrien Mazarguil 		return ret;
376fc40db99SAdrien Mazarguil 	request.ifr_flags &= keep;
377fc40db99SAdrien Mazarguil 	request.ifr_flags |= flags & ~keep;
3782b730263SAdrien Mazarguil 	return mlx5_ifreq(dev, SIOCSIFFLAGS, &request, 0);
379771fa900SAdrien Mazarguil }
380771fa900SAdrien Mazarguil 
381771fa900SAdrien Mazarguil /**
3827b2423cdSNélio Laranjeiro  * DPDK callback for Ethernet device configuration.
383e60fbd5bSAdrien Mazarguil  *
384e60fbd5bSAdrien Mazarguil  * @param dev
385e60fbd5bSAdrien Mazarguil  *   Pointer to Ethernet device structure.
386e60fbd5bSAdrien Mazarguil  *
387e60fbd5bSAdrien Mazarguil  * @return
388a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
389e60fbd5bSAdrien Mazarguil  */
3907b2423cdSNélio Laranjeiro int
3917b2423cdSNélio Laranjeiro mlx5_dev_configure(struct rte_eth_dev *dev)
392e60fbd5bSAdrien Mazarguil {
393e60fbd5bSAdrien Mazarguil 	struct priv *priv = dev->data->dev_private;
394e60fbd5bSAdrien Mazarguil 	unsigned int rxqs_n = dev->data->nb_rx_queues;
395e60fbd5bSAdrien Mazarguil 	unsigned int txqs_n = dev->data->nb_tx_queues;
396634efbc2SNelio Laranjeiro 	unsigned int i;
397634efbc2SNelio Laranjeiro 	unsigned int j;
398634efbc2SNelio Laranjeiro 	unsigned int reta_idx_n;
39929c1d8bbSNélio Laranjeiro 	const uint8_t use_app_rss_key =
40021e3a974SShahaf Shuler 		!!dev->data->dev_conf.rx_adv_conf.rss_conf.rss_key;
401a6d83b6aSNélio Laranjeiro 	int ret = 0;
402e60fbd5bSAdrien Mazarguil 
40329c1d8bbSNélio Laranjeiro 	if (use_app_rss_key &&
40429c1d8bbSNélio Laranjeiro 	    (dev->data->dev_conf.rx_adv_conf.rss_conf.rss_key_len !=
405c388a2f6SNelio Laranjeiro 	     MLX5_RSS_HASH_KEY_LEN)) {
406c388a2f6SNelio Laranjeiro 		DRV_LOG(ERR, "port %u RSS key len must be %s Bytes long",
407c388a2f6SNelio Laranjeiro 			dev->data->port_id, RTE_STR(MLX5_RSS_HASH_KEY_LEN));
408a6d83b6aSNélio Laranjeiro 		rte_errno = EINVAL;
409a6d83b6aSNélio Laranjeiro 		return -rte_errno;
41029c1d8bbSNélio Laranjeiro 	}
41129c1d8bbSNélio Laranjeiro 	priv->rss_conf.rss_key =
41229c1d8bbSNélio Laranjeiro 		rte_realloc(priv->rss_conf.rss_key,
413c388a2f6SNelio Laranjeiro 			    MLX5_RSS_HASH_KEY_LEN, 0);
41429c1d8bbSNélio Laranjeiro 	if (!priv->rss_conf.rss_key) {
415a170a30dSNélio Laranjeiro 		DRV_LOG(ERR, "port %u cannot allocate RSS hash key memory (%u)",
4160f99970bSNélio Laranjeiro 			dev->data->port_id, rxqs_n);
417a6d83b6aSNélio Laranjeiro 		rte_errno = ENOMEM;
418a6d83b6aSNélio Laranjeiro 		return -rte_errno;
41929c1d8bbSNélio Laranjeiro 	}
42029c1d8bbSNélio Laranjeiro 	memcpy(priv->rss_conf.rss_key,
42129c1d8bbSNélio Laranjeiro 	       use_app_rss_key ?
42229c1d8bbSNélio Laranjeiro 	       dev->data->dev_conf.rx_adv_conf.rss_conf.rss_key :
42329c1d8bbSNélio Laranjeiro 	       rss_hash_default_key,
424c388a2f6SNelio Laranjeiro 	       MLX5_RSS_HASH_KEY_LEN);
425c388a2f6SNelio Laranjeiro 	priv->rss_conf.rss_key_len = MLX5_RSS_HASH_KEY_LEN;
42629c1d8bbSNélio Laranjeiro 	priv->rss_conf.rss_hf = dev->data->dev_conf.rx_adv_conf.rss_conf.rss_hf;
427e60fbd5bSAdrien Mazarguil 	priv->rxqs = (void *)dev->data->rx_queues;
428e60fbd5bSAdrien Mazarguil 	priv->txqs = (void *)dev->data->tx_queues;
429e60fbd5bSAdrien Mazarguil 	if (txqs_n != priv->txqs_n) {
430a170a30dSNélio Laranjeiro 		DRV_LOG(INFO, "port %u Tx queues number update: %u -> %u",
4310f99970bSNélio Laranjeiro 			dev->data->port_id, priv->txqs_n, txqs_n);
432e60fbd5bSAdrien Mazarguil 		priv->txqs_n = txqs_n;
433e60fbd5bSAdrien Mazarguil 	}
4347fe24446SShahaf Shuler 	if (rxqs_n > priv->config.ind_table_max_size) {
435a170a30dSNélio Laranjeiro 		DRV_LOG(ERR, "port %u cannot handle this many Rx queues (%u)",
4360f99970bSNélio Laranjeiro 			dev->data->port_id, rxqs_n);
437a6d83b6aSNélio Laranjeiro 		rte_errno = EINVAL;
438a6d83b6aSNélio Laranjeiro 		return -rte_errno;
439634efbc2SNelio Laranjeiro 	}
440e60fbd5bSAdrien Mazarguil 	if (rxqs_n == priv->rxqs_n)
441e60fbd5bSAdrien Mazarguil 		return 0;
442a170a30dSNélio Laranjeiro 	DRV_LOG(INFO, "port %u Rx queues number update: %u -> %u",
4430f99970bSNélio Laranjeiro 		dev->data->port_id, priv->rxqs_n, rxqs_n);
444e60fbd5bSAdrien Mazarguil 	priv->rxqs_n = rxqs_n;
445634efbc2SNelio Laranjeiro 	/* If the requested number of RX queues is not a power of two, use the
446634efbc2SNelio Laranjeiro 	 * maximum indirection table size for better balancing.
447634efbc2SNelio Laranjeiro 	 * The result is always rounded to the next power of two. */
448634efbc2SNelio Laranjeiro 	reta_idx_n = (1 << log2above((rxqs_n & (rxqs_n - 1)) ?
4497fe24446SShahaf Shuler 				     priv->config.ind_table_max_size :
450634efbc2SNelio Laranjeiro 				     rxqs_n));
451a6d83b6aSNélio Laranjeiro 	ret = mlx5_rss_reta_index_resize(dev, reta_idx_n);
452a6d83b6aSNélio Laranjeiro 	if (ret)
453a6d83b6aSNélio Laranjeiro 		return ret;
454634efbc2SNelio Laranjeiro 	/* When the number of RX queues is not a power of two, the remaining
455634efbc2SNelio Laranjeiro 	 * table entries are padded with reused WQs and hashes are not spread
456634efbc2SNelio Laranjeiro 	 * uniformly. */
457634efbc2SNelio Laranjeiro 	for (i = 0, j = 0; (i != reta_idx_n); ++i) {
458634efbc2SNelio Laranjeiro 		(*priv->reta_idx)[i] = j;
459634efbc2SNelio Laranjeiro 		if (++j == rxqs_n)
460634efbc2SNelio Laranjeiro 			j = 0;
461634efbc2SNelio Laranjeiro 	}
462e60fbd5bSAdrien Mazarguil 	return 0;
463e60fbd5bSAdrien Mazarguil }
464e60fbd5bSAdrien Mazarguil 
465e60fbd5bSAdrien Mazarguil /**
466d11d651fSShahaf Shuler  * Sets default tuning parameters.
467d11d651fSShahaf Shuler  *
468d11d651fSShahaf Shuler  * @param dev
469d11d651fSShahaf Shuler  *   Pointer to Ethernet device.
470d11d651fSShahaf Shuler  * @param[out] info
471d11d651fSShahaf Shuler  *   Info structure output buffer.
472d11d651fSShahaf Shuler  */
473d11d651fSShahaf Shuler static void
474d11d651fSShahaf Shuler mlx5_set_default_params(struct rte_eth_dev *dev, struct rte_eth_dev_info *info)
475d11d651fSShahaf Shuler {
476d11d651fSShahaf Shuler 	struct priv *priv = dev->data->dev_private;
477d11d651fSShahaf Shuler 
478d11d651fSShahaf Shuler 	/* Minimum CPU utilization. */
479d11d651fSShahaf Shuler 	info->default_rxportconf.ring_size = 256;
480d11d651fSShahaf Shuler 	info->default_txportconf.ring_size = 256;
481d11d651fSShahaf Shuler 	info->default_rxportconf.burst_size = 64;
482d11d651fSShahaf Shuler 	info->default_txportconf.burst_size = 64;
483d11d651fSShahaf Shuler 	if (priv->link_speed_capa & ETH_LINK_SPEED_100G) {
484d11d651fSShahaf Shuler 		info->default_rxportconf.nb_queues = 16;
485d11d651fSShahaf Shuler 		info->default_txportconf.nb_queues = 16;
486d11d651fSShahaf Shuler 		if (dev->data->nb_rx_queues > 2 ||
487d11d651fSShahaf Shuler 		    dev->data->nb_tx_queues > 2) {
488d11d651fSShahaf Shuler 			/* Max Throughput. */
489d11d651fSShahaf Shuler 			info->default_rxportconf.ring_size = 2048;
490d11d651fSShahaf Shuler 			info->default_txportconf.ring_size = 2048;
491d11d651fSShahaf Shuler 		}
492d11d651fSShahaf Shuler 	} else {
493d11d651fSShahaf Shuler 		info->default_rxportconf.nb_queues = 8;
494d11d651fSShahaf Shuler 		info->default_txportconf.nb_queues = 8;
495d11d651fSShahaf Shuler 		if (dev->data->nb_rx_queues > 2 ||
496d11d651fSShahaf Shuler 		    dev->data->nb_tx_queues > 2) {
497d11d651fSShahaf Shuler 			/* Max Throughput. */
498d11d651fSShahaf Shuler 			info->default_rxportconf.ring_size = 4096;
499d11d651fSShahaf Shuler 			info->default_txportconf.ring_size = 4096;
500d11d651fSShahaf Shuler 		}
501d11d651fSShahaf Shuler 	}
502d11d651fSShahaf Shuler }
503d11d651fSShahaf Shuler 
504d11d651fSShahaf Shuler /**
505e60fbd5bSAdrien Mazarguil  * DPDK callback to get information about the device.
506e60fbd5bSAdrien Mazarguil  *
507e60fbd5bSAdrien Mazarguil  * @param dev
508e60fbd5bSAdrien Mazarguil  *   Pointer to Ethernet device structure.
509e60fbd5bSAdrien Mazarguil  * @param[out] info
510e60fbd5bSAdrien Mazarguil  *   Info structure output buffer.
511e60fbd5bSAdrien Mazarguil  */
512e60fbd5bSAdrien Mazarguil void
513e60fbd5bSAdrien Mazarguil mlx5_dev_infos_get(struct rte_eth_dev *dev, struct rte_eth_dev_info *info)
514e60fbd5bSAdrien Mazarguil {
51501d79216SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
5167fe24446SShahaf Shuler 	struct mlx5_dev_config *config = &priv->config;
517e60fbd5bSAdrien Mazarguil 	unsigned int max;
518e60fbd5bSAdrien Mazarguil 	char ifname[IF_NAMESIZE];
519e60fbd5bSAdrien Mazarguil 
520e60fbd5bSAdrien Mazarguil 	/* FIXME: we should ask the device for these values. */
521e60fbd5bSAdrien Mazarguil 	info->min_rx_bufsize = 32;
522e60fbd5bSAdrien Mazarguil 	info->max_rx_pktlen = 65536;
523e60fbd5bSAdrien Mazarguil 	/*
524e60fbd5bSAdrien Mazarguil 	 * Since we need one CQ per QP, the limit is the minimum number
525e60fbd5bSAdrien Mazarguil 	 * between the two values.
526e60fbd5bSAdrien Mazarguil 	 */
52743e9d979SShachar Beiser 	max = RTE_MIN(priv->device_attr.orig_attr.max_cq,
52843e9d979SShachar Beiser 		      priv->device_attr.orig_attr.max_qp);
529e60fbd5bSAdrien Mazarguil 	/* If max >= 65535 then max = 0, max_rx_queues is uint16_t. */
530e60fbd5bSAdrien Mazarguil 	if (max >= 65535)
531e60fbd5bSAdrien Mazarguil 		max = 65535;
532e60fbd5bSAdrien Mazarguil 	info->max_rx_queues = max;
533e60fbd5bSAdrien Mazarguil 	info->max_tx_queues = max;
53418c01b98SNélio Laranjeiro 	info->max_mac_addrs = MLX5_MAX_UC_MAC_ADDRESSES;
535af4f09f2SNélio Laranjeiro 	info->rx_queue_offload_capa = mlx5_get_rx_queue_offloads(dev);
536af4f09f2SNélio Laranjeiro 	info->rx_offload_capa = (mlx5_get_rx_port_offloads() |
53717b843ebSShahaf Shuler 				 info->rx_queue_offload_capa);
538af4f09f2SNélio Laranjeiro 	info->tx_offload_capa = mlx5_get_tx_port_offloads(dev);
539af4f09f2SNélio Laranjeiro 	if (mlx5_get_ifname(dev, &ifname) == 0)
540e60fbd5bSAdrien Mazarguil 		info->if_index = if_nametoindex(ifname);
541d365210eSYongseok Koh 	info->reta_size = priv->reta_idx_n ?
5427fe24446SShahaf Shuler 		priv->reta_idx_n : config->ind_table_max_size;
543c388a2f6SNelio Laranjeiro 	info->hash_key_size = MLX5_RSS_HASH_KEY_LEN;
54475ef62a9SNélio Laranjeiro 	info->speed_capa = priv->link_speed_capa;
545b233b027SShahaf Shuler 	info->flow_type_rss_offloads = ~MLX5_RSS_HF_MASK;
546d11d651fSShahaf Shuler 	mlx5_set_default_params(dev, info);
5472b730263SAdrien Mazarguil 	info->switch_info.name = dev->data->name;
5482b730263SAdrien Mazarguil 	info->switch_info.domain_id = priv->domain_id;
5492b730263SAdrien Mazarguil 	info->switch_info.port_id = priv->representor_id;
5502b730263SAdrien Mazarguil 	if (priv->representor) {
5512b730263SAdrien Mazarguil 		unsigned int i = mlx5_dev_to_port_id(dev->device, NULL, 0);
5522b730263SAdrien Mazarguil 		uint16_t port_id[i];
5532b730263SAdrien Mazarguil 
5542b730263SAdrien Mazarguil 		i = RTE_MIN(mlx5_dev_to_port_id(dev->device, port_id, i), i);
5552b730263SAdrien Mazarguil 		while (i--) {
5562b730263SAdrien Mazarguil 			struct priv *opriv =
5572b730263SAdrien Mazarguil 				rte_eth_devices[port_id[i]].data->dev_private;
5582b730263SAdrien Mazarguil 
5592b730263SAdrien Mazarguil 			if (!opriv ||
5602b730263SAdrien Mazarguil 			    opriv->representor ||
5612b730263SAdrien Mazarguil 			    opriv->domain_id != priv->domain_id)
5622b730263SAdrien Mazarguil 				continue;
5632b730263SAdrien Mazarguil 			/*
5642b730263SAdrien Mazarguil 			 * Override switch name with that of the master
5652b730263SAdrien Mazarguil 			 * device.
5662b730263SAdrien Mazarguil 			 */
5672b730263SAdrien Mazarguil 			info->switch_info.name = opriv->dev_data->name;
5682b730263SAdrien Mazarguil 			break;
5692b730263SAdrien Mazarguil 		}
5702b730263SAdrien Mazarguil 	}
571e60fbd5bSAdrien Mazarguil }
572e60fbd5bSAdrien Mazarguil 
573fb732b0aSNélio Laranjeiro /**
574fb732b0aSNélio Laranjeiro  * Get supported packet types.
575fb732b0aSNélio Laranjeiro  *
576fb732b0aSNélio Laranjeiro  * @param dev
577fb732b0aSNélio Laranjeiro  *   Pointer to Ethernet device structure.
578fb732b0aSNélio Laranjeiro  *
579fb732b0aSNélio Laranjeiro  * @return
580fb732b0aSNélio Laranjeiro  *   A pointer to the supported Packet types array.
581fb732b0aSNélio Laranjeiro  */
58278a38edfSJianfeng Tan const uint32_t *
58378a38edfSJianfeng Tan mlx5_dev_supported_ptypes_get(struct rte_eth_dev *dev)
58478a38edfSJianfeng Tan {
58578a38edfSJianfeng Tan 	static const uint32_t ptypes[] = {
58678a38edfSJianfeng Tan 		/* refers to rxq_cq_to_pkt_type() */
587ea16068cSYongseok Koh 		RTE_PTYPE_L2_ETHER,
588c4ba5434SNélio Laranjeiro 		RTE_PTYPE_L3_IPV4_EXT_UNKNOWN,
589c4ba5434SNélio Laranjeiro 		RTE_PTYPE_L3_IPV6_EXT_UNKNOWN,
590ea16068cSYongseok Koh 		RTE_PTYPE_L4_NONFRAG,
591ea16068cSYongseok Koh 		RTE_PTYPE_L4_FRAG,
592ea16068cSYongseok Koh 		RTE_PTYPE_L4_TCP,
593ea16068cSYongseok Koh 		RTE_PTYPE_L4_UDP,
594c4ba5434SNélio Laranjeiro 		RTE_PTYPE_INNER_L3_IPV4_EXT_UNKNOWN,
595c4ba5434SNélio Laranjeiro 		RTE_PTYPE_INNER_L3_IPV6_EXT_UNKNOWN,
596ea16068cSYongseok Koh 		RTE_PTYPE_INNER_L4_NONFRAG,
597ea16068cSYongseok Koh 		RTE_PTYPE_INNER_L4_FRAG,
598ea16068cSYongseok Koh 		RTE_PTYPE_INNER_L4_TCP,
599ea16068cSYongseok Koh 		RTE_PTYPE_INNER_L4_UDP,
60078a38edfSJianfeng Tan 		RTE_PTYPE_UNKNOWN
60178a38edfSJianfeng Tan 	};
60278a38edfSJianfeng Tan 
6036cb559d6SYongseok Koh 	if (dev->rx_pkt_burst == mlx5_rx_burst ||
6047d6bf6b8SYongseok Koh 	    dev->rx_pkt_burst == mlx5_rx_burst_mprq ||
6056cb559d6SYongseok Koh 	    dev->rx_pkt_burst == mlx5_rx_burst_vec)
60678a38edfSJianfeng Tan 		return ptypes;
60778a38edfSJianfeng Tan 	return NULL;
60878a38edfSJianfeng Tan }
60978a38edfSJianfeng Tan 
610e60fbd5bSAdrien Mazarguil /**
6112c960a51SMatthieu Ternisien d'Ouville  * DPDK callback to retrieve physical link information.
612cb8faed7SAdrien Mazarguil  *
613cb8faed7SAdrien Mazarguil  * @param dev
614cb8faed7SAdrien Mazarguil  *   Pointer to Ethernet device structure.
615cfee9475SNélio Laranjeiro  * @param[out] link
616cfee9475SNélio Laranjeiro  *   Storage for current link status.
617fb732b0aSNélio Laranjeiro  *
618fb732b0aSNélio Laranjeiro  * @return
619a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
620cb8faed7SAdrien Mazarguil  */
62118840871SNélio Laranjeiro static int
622cfee9475SNélio Laranjeiro mlx5_link_update_unlocked_gset(struct rte_eth_dev *dev,
623cfee9475SNélio Laranjeiro 			       struct rte_eth_link *link)
624cb8faed7SAdrien Mazarguil {
62501d79216SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
626cb8faed7SAdrien Mazarguil 	struct ethtool_cmd edata = {
62775ef62a9SNélio Laranjeiro 		.cmd = ETHTOOL_GSET /* Deprecated since Linux v4.5. */
628cb8faed7SAdrien Mazarguil 	};
629cb8faed7SAdrien Mazarguil 	struct ifreq ifr;
630cb8faed7SAdrien Mazarguil 	struct rte_eth_link dev_link;
631cb8faed7SAdrien Mazarguil 	int link_speed = 0;
632a6d83b6aSNélio Laranjeiro 	int ret;
633cb8faed7SAdrien Mazarguil 
6342b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCGIFFLAGS, &ifr, 1);
635a6d83b6aSNélio Laranjeiro 	if (ret) {
636a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u ioctl(SIOCGIFFLAGS) failed: %s",
6370f99970bSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
638a6d83b6aSNélio Laranjeiro 		return ret;
639cb8faed7SAdrien Mazarguil 	}
640cb8faed7SAdrien Mazarguil 	memset(&dev_link, 0, sizeof(dev_link));
641cb8faed7SAdrien Mazarguil 	dev_link.link_status = ((ifr.ifr_flags & IFF_UP) &&
642cb8faed7SAdrien Mazarguil 				(ifr.ifr_flags & IFF_RUNNING));
643d06c608cSAdrien Mazarguil 	ifr.ifr_data = (void *)&edata;
6442b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCETHTOOL, &ifr, 1);
645a6d83b6aSNélio Laranjeiro 	if (ret) {
646a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING,
647a170a30dSNélio Laranjeiro 			"port %u ioctl(SIOCETHTOOL, ETHTOOL_GSET) failed: %s",
6480f99970bSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
649a6d83b6aSNélio Laranjeiro 		return ret;
650cb8faed7SAdrien Mazarguil 	}
651cb8faed7SAdrien Mazarguil 	link_speed = ethtool_cmd_speed(&edata);
652cb8faed7SAdrien Mazarguil 	if (link_speed == -1)
6533fef0822SFerruh Yigit 		dev_link.link_speed = ETH_SPEED_NUM_NONE;
654cb8faed7SAdrien Mazarguil 	else
655cb8faed7SAdrien Mazarguil 		dev_link.link_speed = link_speed;
65675ef62a9SNélio Laranjeiro 	priv->link_speed_capa = 0;
65775ef62a9SNélio Laranjeiro 	if (edata.supported & SUPPORTED_Autoneg)
65875ef62a9SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_AUTONEG;
65975ef62a9SNélio Laranjeiro 	if (edata.supported & (SUPPORTED_1000baseT_Full |
66075ef62a9SNélio Laranjeiro 			       SUPPORTED_1000baseKX_Full))
66175ef62a9SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_1G;
66275ef62a9SNélio Laranjeiro 	if (edata.supported & SUPPORTED_10000baseKR_Full)
66375ef62a9SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_10G;
66475ef62a9SNélio Laranjeiro 	if (edata.supported & (SUPPORTED_40000baseKR4_Full |
66575ef62a9SNélio Laranjeiro 			       SUPPORTED_40000baseCR4_Full |
66675ef62a9SNélio Laranjeiro 			       SUPPORTED_40000baseSR4_Full |
66775ef62a9SNélio Laranjeiro 			       SUPPORTED_40000baseLR4_Full))
66875ef62a9SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_40G;
669cb8faed7SAdrien Mazarguil 	dev_link.link_duplex = ((edata.duplex == DUPLEX_HALF) ?
670cb8faed7SAdrien Mazarguil 				ETH_LINK_HALF_DUPLEX : ETH_LINK_FULL_DUPLEX);
67182113036SMarc Sune 	dev_link.link_autoneg = !(dev->data->dev_conf.link_speeds &
67282113036SMarc Sune 			ETH_LINK_SPEED_FIXED);
673cfee9475SNélio Laranjeiro 	if ((dev_link.link_speed && !dev_link.link_status) ||
674cfee9475SNélio Laranjeiro 	    (!dev_link.link_speed && dev_link.link_status)) {
675a6d83b6aSNélio Laranjeiro 		rte_errno = EAGAIN;
676a6d83b6aSNélio Laranjeiro 		return -rte_errno;
677cb8faed7SAdrien Mazarguil 	}
678cfee9475SNélio Laranjeiro 	*link = dev_link;
679cfee9475SNélio Laranjeiro 	return 0;
680cfee9475SNélio Laranjeiro }
681cb8faed7SAdrien Mazarguil 
682cb8faed7SAdrien Mazarguil /**
6833a49ffe3SShahaf Shuler  * Retrieve physical link information (unlocked version using new ioctl).
68418840871SNélio Laranjeiro  *
68518840871SNélio Laranjeiro  * @param dev
68618840871SNélio Laranjeiro  *   Pointer to Ethernet device structure.
687cfee9475SNélio Laranjeiro  * @param[out] link
688cfee9475SNélio Laranjeiro  *   Storage for current link status.
689fb732b0aSNélio Laranjeiro  *
690fb732b0aSNélio Laranjeiro  * @return
691a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
69218840871SNélio Laranjeiro  */
69318840871SNélio Laranjeiro static int
694cfee9475SNélio Laranjeiro mlx5_link_update_unlocked_gs(struct rte_eth_dev *dev,
695cfee9475SNélio Laranjeiro 			     struct rte_eth_link *link)
696cfee9475SNélio Laranjeiro 
69718840871SNélio Laranjeiro {
69801d79216SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
69937a4406cSNélio Laranjeiro 	struct ethtool_link_settings gcmd = { .cmd = ETHTOOL_GLINKSETTINGS };
70018840871SNélio Laranjeiro 	struct ifreq ifr;
70118840871SNélio Laranjeiro 	struct rte_eth_link dev_link;
70218840871SNélio Laranjeiro 	uint64_t sc;
703a6d83b6aSNélio Laranjeiro 	int ret;
70418840871SNélio Laranjeiro 
7052b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCGIFFLAGS, &ifr, 1);
706a6d83b6aSNélio Laranjeiro 	if (ret) {
707a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING, "port %u ioctl(SIOCGIFFLAGS) failed: %s",
7080f99970bSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
709a6d83b6aSNélio Laranjeiro 		return ret;
71018840871SNélio Laranjeiro 	}
71118840871SNélio Laranjeiro 	memset(&dev_link, 0, sizeof(dev_link));
71218840871SNélio Laranjeiro 	dev_link.link_status = ((ifr.ifr_flags & IFF_UP) &&
71318840871SNélio Laranjeiro 				(ifr.ifr_flags & IFF_RUNNING));
71437a4406cSNélio Laranjeiro 	ifr.ifr_data = (void *)&gcmd;
7152b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCETHTOOL, &ifr, 1);
716a6d83b6aSNélio Laranjeiro 	if (ret) {
717a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG,
718a170a30dSNélio Laranjeiro 			"port %u ioctl(SIOCETHTOOL, ETHTOOL_GLINKSETTINGS)"
719a170a30dSNélio Laranjeiro 			" failed: %s",
720a170a30dSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
721a6d83b6aSNélio Laranjeiro 		return ret;
72218840871SNélio Laranjeiro 	}
72337a4406cSNélio Laranjeiro 	gcmd.link_mode_masks_nwords = -gcmd.link_mode_masks_nwords;
72437a4406cSNélio Laranjeiro 
72537a4406cSNélio Laranjeiro 	alignas(struct ethtool_link_settings)
72637a4406cSNélio Laranjeiro 	uint8_t data[offsetof(struct ethtool_link_settings, link_mode_masks) +
72737a4406cSNélio Laranjeiro 		     sizeof(uint32_t) * gcmd.link_mode_masks_nwords * 3];
72837a4406cSNélio Laranjeiro 	struct ethtool_link_settings *ecmd = (void *)data;
72937a4406cSNélio Laranjeiro 
73037a4406cSNélio Laranjeiro 	*ecmd = gcmd;
73137a4406cSNélio Laranjeiro 	ifr.ifr_data = (void *)ecmd;
7322b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCETHTOOL, &ifr, 1);
733a6d83b6aSNélio Laranjeiro 	if (ret) {
734a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG,
735a170a30dSNélio Laranjeiro 			"port %u ioctl(SIOCETHTOOL, ETHTOOL_GLINKSETTINGS)"
736a170a30dSNélio Laranjeiro 			" failed: %s",
737a170a30dSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
738a6d83b6aSNélio Laranjeiro 		return ret;
739ef09a7fcSGowrishankar Muthukrishnan 	}
74090260d57SNélio Laranjeiro 	dev_link.link_speed = ecmd->speed;
74190260d57SNélio Laranjeiro 	sc = ecmd->link_mode_masks[0] |
74290260d57SNélio Laranjeiro 		((uint64_t)ecmd->link_mode_masks[1] << 32);
74318840871SNélio Laranjeiro 	priv->link_speed_capa = 0;
744b113cb5eSEdward Makarov 	if (sc & MLX5_BITSHIFT(ETHTOOL_LINK_MODE_Autoneg_BIT))
74518840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_AUTONEG;
746b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_1000baseT_Full_BIT) |
747b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_1000baseKX_Full_BIT)))
74818840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_1G;
749b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_10000baseKX4_Full_BIT) |
750b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_10000baseKR_Full_BIT) |
751b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_10000baseR_FEC_BIT)))
75218840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_10G;
753b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_20000baseMLD2_Full_BIT) |
754b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_20000baseKR2_Full_BIT)))
75518840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_20G;
756b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT) |
757b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT) |
758b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_40000baseSR4_Full_BIT) |
759b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_40000baseLR4_Full_BIT)))
76018840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_40G;
761b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_56000baseKR4_Full_BIT) |
762b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_56000baseCR4_Full_BIT) |
763b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_56000baseSR4_Full_BIT) |
764b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_56000baseLR4_Full_BIT)))
76518840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_56G;
766b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_25000baseCR_Full_BIT) |
767b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_25000baseKR_Full_BIT) |
768b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_25000baseSR_Full_BIT)))
76918840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_25G;
770b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_50000baseCR2_Full_BIT) |
771b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_50000baseKR2_Full_BIT)))
77218840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_50G;
773b113cb5eSEdward Makarov 	if (sc & (MLX5_BITSHIFT(ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT) |
774b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_100000baseSR4_Full_BIT) |
775b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT) |
776b113cb5eSEdward Makarov 		  MLX5_BITSHIFT(ETHTOOL_LINK_MODE_100000baseLR4_ER4_Full_BIT)))
77718840871SNélio Laranjeiro 		priv->link_speed_capa |= ETH_LINK_SPEED_100G;
77890260d57SNélio Laranjeiro 	dev_link.link_duplex = ((ecmd->duplex == DUPLEX_HALF) ?
77918840871SNélio Laranjeiro 				ETH_LINK_HALF_DUPLEX : ETH_LINK_FULL_DUPLEX);
78018840871SNélio Laranjeiro 	dev_link.link_autoneg = !(dev->data->dev_conf.link_speeds &
78118840871SNélio Laranjeiro 				  ETH_LINK_SPEED_FIXED);
782cfee9475SNélio Laranjeiro 	if ((dev_link.link_speed && !dev_link.link_status) ||
783cfee9475SNélio Laranjeiro 	    (!dev_link.link_speed && dev_link.link_status)) {
784a6d83b6aSNélio Laranjeiro 		rte_errno = EAGAIN;
785a6d83b6aSNélio Laranjeiro 		return -rte_errno;
78618840871SNélio Laranjeiro 	}
787cfee9475SNélio Laranjeiro 	*link = dev_link;
788cfee9475SNélio Laranjeiro 	return 0;
789cfee9475SNélio Laranjeiro }
79018840871SNélio Laranjeiro 
79118840871SNélio Laranjeiro /**
792cb8faed7SAdrien Mazarguil  * DPDK callback to retrieve physical link information.
793cb8faed7SAdrien Mazarguil  *
794cb8faed7SAdrien Mazarguil  * @param dev
795cb8faed7SAdrien Mazarguil  *   Pointer to Ethernet device structure.
796cb8faed7SAdrien Mazarguil  * @param wait_to_complete
797cfee9475SNélio Laranjeiro  *   Wait for request completion.
798fb732b0aSNélio Laranjeiro  *
799fb732b0aSNélio Laranjeiro  * @return
800cfee9475SNélio Laranjeiro  *   0 if link status was not updated, positive if it was, a negative errno
801cfee9475SNélio Laranjeiro  *   value otherwise and rte_errno is set.
802cb8faed7SAdrien Mazarguil  */
803cb8faed7SAdrien Mazarguil int
804cfee9475SNélio Laranjeiro mlx5_link_update(struct rte_eth_dev *dev, int wait_to_complete)
805cb8faed7SAdrien Mazarguil {
806c7bf6225SYongseok Koh 	int ret;
807cfee9475SNélio Laranjeiro 	struct rte_eth_link dev_link;
808cfee9475SNélio Laranjeiro 	time_t start_time = time(NULL);
809cb8faed7SAdrien Mazarguil 
810cfee9475SNélio Laranjeiro 	do {
811cfee9475SNélio Laranjeiro 		ret = mlx5_link_update_unlocked_gs(dev, &dev_link);
8127d2e32f7SShahaf Shuler 		if (ret)
8137d2e32f7SShahaf Shuler 			ret = mlx5_link_update_unlocked_gset(dev, &dev_link);
814cfee9475SNélio Laranjeiro 		if (ret == 0)
815cfee9475SNélio Laranjeiro 			break;
816cfee9475SNélio Laranjeiro 		/* Handle wait to complete situation. */
817cfee9475SNélio Laranjeiro 		if (wait_to_complete && ret == -EAGAIN) {
818cfee9475SNélio Laranjeiro 			if (abs((int)difftime(time(NULL), start_time)) <
819cfee9475SNélio Laranjeiro 			    MLX5_LINK_STATUS_TIMEOUT) {
820cfee9475SNélio Laranjeiro 				usleep(0);
821cfee9475SNélio Laranjeiro 				continue;
822cfee9475SNélio Laranjeiro 			} else {
823cfee9475SNélio Laranjeiro 				rte_errno = EBUSY;
824cfee9475SNélio Laranjeiro 				return -rte_errno;
825cfee9475SNélio Laranjeiro 			}
826cfee9475SNélio Laranjeiro 		} else if (ret < 0) {
827cfee9475SNélio Laranjeiro 			return ret;
828cfee9475SNélio Laranjeiro 		}
829cfee9475SNélio Laranjeiro 	} while (wait_to_complete);
830cfee9475SNélio Laranjeiro 	ret = !!memcmp(&dev->data->dev_link, &dev_link,
831cfee9475SNélio Laranjeiro 		       sizeof(struct rte_eth_link));
832cfee9475SNélio Laranjeiro 	dev->data->dev_link = dev_link;
833cfee9475SNélio Laranjeiro 	return ret;
834cb8faed7SAdrien Mazarguil }
835cb8faed7SAdrien Mazarguil 
836cb8faed7SAdrien Mazarguil /**
837cf37ca95SAdrien Mazarguil  * DPDK callback to change the MTU.
838cf37ca95SAdrien Mazarguil  *
839cf37ca95SAdrien Mazarguil  * @param dev
840cf37ca95SAdrien Mazarguil  *   Pointer to Ethernet device structure.
841cf37ca95SAdrien Mazarguil  * @param in_mtu
842cf37ca95SAdrien Mazarguil  *   New MTU.
843cf37ca95SAdrien Mazarguil  *
844cf37ca95SAdrien Mazarguil  * @return
845a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
846cf37ca95SAdrien Mazarguil  */
847cf37ca95SAdrien Mazarguil int
848cf37ca95SAdrien Mazarguil mlx5_dev_set_mtu(struct rte_eth_dev *dev, uint16_t mtu)
849cf37ca95SAdrien Mazarguil {
850cf37ca95SAdrien Mazarguil 	struct priv *priv = dev->data->dev_private;
851a6d83b6aSNélio Laranjeiro 	uint16_t kern_mtu = 0;
852a6d83b6aSNélio Laranjeiro 	int ret;
853cf37ca95SAdrien Mazarguil 
854af4f09f2SNélio Laranjeiro 	ret = mlx5_get_mtu(dev, &kern_mtu);
855a0edafe4SNelio Laranjeiro 	if (ret)
856a6d83b6aSNélio Laranjeiro 		return ret;
857cf37ca95SAdrien Mazarguil 	/* Set kernel interface MTU first. */
858af4f09f2SNélio Laranjeiro 	ret = mlx5_set_mtu(dev, mtu);
859a0edafe4SNelio Laranjeiro 	if (ret)
860a6d83b6aSNélio Laranjeiro 		return ret;
861af4f09f2SNélio Laranjeiro 	ret = mlx5_get_mtu(dev, &kern_mtu);
862a0edafe4SNelio Laranjeiro 	if (ret)
863a6d83b6aSNélio Laranjeiro 		return ret;
864a0edafe4SNelio Laranjeiro 	if (kern_mtu == mtu) {
865a0edafe4SNelio Laranjeiro 		priv->mtu = mtu;
866a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "port %u adapter MTU set to %u",
867a170a30dSNélio Laranjeiro 			dev->data->port_id, mtu);
868a0edafe4SNelio Laranjeiro 		return 0;
869a6d83b6aSNélio Laranjeiro 	}
870a6d83b6aSNélio Laranjeiro 	rte_errno = EAGAIN;
871a6d83b6aSNélio Laranjeiro 	return -rte_errno;
872cf37ca95SAdrien Mazarguil }
873cf37ca95SAdrien Mazarguil 
874cf37ca95SAdrien Mazarguil /**
87502d75430SAdrien Mazarguil  * DPDK callback to get flow control status.
87602d75430SAdrien Mazarguil  *
87702d75430SAdrien Mazarguil  * @param dev
87802d75430SAdrien Mazarguil  *   Pointer to Ethernet device structure.
87902d75430SAdrien Mazarguil  * @param[out] fc_conf
88002d75430SAdrien Mazarguil  *   Flow control output buffer.
88102d75430SAdrien Mazarguil  *
88202d75430SAdrien Mazarguil  * @return
883a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
88402d75430SAdrien Mazarguil  */
88502d75430SAdrien Mazarguil int
88602d75430SAdrien Mazarguil mlx5_dev_get_flow_ctrl(struct rte_eth_dev *dev, struct rte_eth_fc_conf *fc_conf)
88702d75430SAdrien Mazarguil {
88802d75430SAdrien Mazarguil 	struct ifreq ifr;
88902d75430SAdrien Mazarguil 	struct ethtool_pauseparam ethpause = {
89002d75430SAdrien Mazarguil 		.cmd = ETHTOOL_GPAUSEPARAM
89102d75430SAdrien Mazarguil 	};
89202d75430SAdrien Mazarguil 	int ret;
89302d75430SAdrien Mazarguil 
894d06c608cSAdrien Mazarguil 	ifr.ifr_data = (void *)&ethpause;
8952b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCETHTOOL, &ifr, 1);
896a6d83b6aSNélio Laranjeiro 	if (ret) {
897a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING,
898a170a30dSNélio Laranjeiro 			"port %u ioctl(SIOCETHTOOL, ETHTOOL_GPAUSEPARAM) failed:"
899a170a30dSNélio Laranjeiro 			" %s",
900a170a30dSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
901a6d83b6aSNélio Laranjeiro 		return ret;
90202d75430SAdrien Mazarguil 	}
90302d75430SAdrien Mazarguil 	fc_conf->autoneg = ethpause.autoneg;
90402d75430SAdrien Mazarguil 	if (ethpause.rx_pause && ethpause.tx_pause)
90502d75430SAdrien Mazarguil 		fc_conf->mode = RTE_FC_FULL;
90602d75430SAdrien Mazarguil 	else if (ethpause.rx_pause)
90702d75430SAdrien Mazarguil 		fc_conf->mode = RTE_FC_RX_PAUSE;
90802d75430SAdrien Mazarguil 	else if (ethpause.tx_pause)
90902d75430SAdrien Mazarguil 		fc_conf->mode = RTE_FC_TX_PAUSE;
91002d75430SAdrien Mazarguil 	else
91102d75430SAdrien Mazarguil 		fc_conf->mode = RTE_FC_NONE;
912a6d83b6aSNélio Laranjeiro 	return 0;
91302d75430SAdrien Mazarguil }
91402d75430SAdrien Mazarguil 
91502d75430SAdrien Mazarguil /**
91602d75430SAdrien Mazarguil  * DPDK callback to modify flow control parameters.
91702d75430SAdrien Mazarguil  *
91802d75430SAdrien Mazarguil  * @param dev
91902d75430SAdrien Mazarguil  *   Pointer to Ethernet device structure.
92002d75430SAdrien Mazarguil  * @param[in] fc_conf
92102d75430SAdrien Mazarguil  *   Flow control parameters.
92202d75430SAdrien Mazarguil  *
92302d75430SAdrien Mazarguil  * @return
924a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
92502d75430SAdrien Mazarguil  */
92602d75430SAdrien Mazarguil int
92702d75430SAdrien Mazarguil mlx5_dev_set_flow_ctrl(struct rte_eth_dev *dev, struct rte_eth_fc_conf *fc_conf)
92802d75430SAdrien Mazarguil {
92902d75430SAdrien Mazarguil 	struct ifreq ifr;
93002d75430SAdrien Mazarguil 	struct ethtool_pauseparam ethpause = {
93102d75430SAdrien Mazarguil 		.cmd = ETHTOOL_SPAUSEPARAM
93202d75430SAdrien Mazarguil 	};
93302d75430SAdrien Mazarguil 	int ret;
93402d75430SAdrien Mazarguil 
935d06c608cSAdrien Mazarguil 	ifr.ifr_data = (void *)&ethpause;
93602d75430SAdrien Mazarguil 	ethpause.autoneg = fc_conf->autoneg;
93702d75430SAdrien Mazarguil 	if (((fc_conf->mode & RTE_FC_FULL) == RTE_FC_FULL) ||
93802d75430SAdrien Mazarguil 	    (fc_conf->mode & RTE_FC_RX_PAUSE))
93902d75430SAdrien Mazarguil 		ethpause.rx_pause = 1;
94002d75430SAdrien Mazarguil 	else
94102d75430SAdrien Mazarguil 		ethpause.rx_pause = 0;
94202d75430SAdrien Mazarguil 
94302d75430SAdrien Mazarguil 	if (((fc_conf->mode & RTE_FC_FULL) == RTE_FC_FULL) ||
94402d75430SAdrien Mazarguil 	    (fc_conf->mode & RTE_FC_TX_PAUSE))
94502d75430SAdrien Mazarguil 		ethpause.tx_pause = 1;
94602d75430SAdrien Mazarguil 	else
94702d75430SAdrien Mazarguil 		ethpause.tx_pause = 0;
9482b730263SAdrien Mazarguil 	ret = mlx5_ifreq(dev, SIOCETHTOOL, &ifr, 0);
949a6d83b6aSNélio Laranjeiro 	if (ret) {
950a170a30dSNélio Laranjeiro 		DRV_LOG(WARNING,
951a170a30dSNélio Laranjeiro 			"port %u ioctl(SIOCETHTOOL, ETHTOOL_SPAUSEPARAM)"
952a170a30dSNélio Laranjeiro 			" failed: %s",
953a170a30dSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
954a6d83b6aSNélio Laranjeiro 		return ret;
95502d75430SAdrien Mazarguil 	}
956a6d83b6aSNélio Laranjeiro 	return 0;
95702d75430SAdrien Mazarguil }
95802d75430SAdrien Mazarguil 
95902d75430SAdrien Mazarguil /**
960771fa900SAdrien Mazarguil  * Get PCI information from struct ibv_device.
961771fa900SAdrien Mazarguil  *
962771fa900SAdrien Mazarguil  * @param device
963771fa900SAdrien Mazarguil  *   Pointer to Ethernet device structure.
964771fa900SAdrien Mazarguil  * @param[out] pci_addr
965771fa900SAdrien Mazarguil  *   PCI bus address output buffer.
966771fa900SAdrien Mazarguil  *
967771fa900SAdrien Mazarguil  * @return
968a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
969771fa900SAdrien Mazarguil  */
970771fa900SAdrien Mazarguil int
971771fa900SAdrien Mazarguil mlx5_ibv_device_to_pci_addr(const struct ibv_device *device,
972771fa900SAdrien Mazarguil 			    struct rte_pci_addr *pci_addr)
973771fa900SAdrien Mazarguil {
974771fa900SAdrien Mazarguil 	FILE *file;
975771fa900SAdrien Mazarguil 	char line[32];
976771fa900SAdrien Mazarguil 	MKSTR(path, "%s/device/uevent", device->ibdev_path);
977771fa900SAdrien Mazarguil 
978771fa900SAdrien Mazarguil 	file = fopen(path, "rb");
979a6d83b6aSNélio Laranjeiro 	if (file == NULL) {
980a6d83b6aSNélio Laranjeiro 		rte_errno = errno;
981a6d83b6aSNélio Laranjeiro 		return -rte_errno;
982a6d83b6aSNélio Laranjeiro 	}
983771fa900SAdrien Mazarguil 	while (fgets(line, sizeof(line), file) == line) {
984771fa900SAdrien Mazarguil 		size_t len = strlen(line);
985771fa900SAdrien Mazarguil 		int ret;
986771fa900SAdrien Mazarguil 
987771fa900SAdrien Mazarguil 		/* Truncate long lines. */
988771fa900SAdrien Mazarguil 		if (len == (sizeof(line) - 1))
989771fa900SAdrien Mazarguil 			while (line[(len - 1)] != '\n') {
990771fa900SAdrien Mazarguil 				ret = fgetc(file);
991771fa900SAdrien Mazarguil 				if (ret == EOF)
992771fa900SAdrien Mazarguil 					break;
993771fa900SAdrien Mazarguil 				line[(len - 1)] = ret;
994771fa900SAdrien Mazarguil 			}
995771fa900SAdrien Mazarguil 		/* Extract information. */
996771fa900SAdrien Mazarguil 		if (sscanf(line,
997771fa900SAdrien Mazarguil 			   "PCI_SLOT_NAME="
998463ced95SStephen Hemminger 			   "%" SCNx32 ":%" SCNx8 ":%" SCNx8 ".%" SCNx8 "\n",
999771fa900SAdrien Mazarguil 			   &pci_addr->domain,
1000771fa900SAdrien Mazarguil 			   &pci_addr->bus,
1001771fa900SAdrien Mazarguil 			   &pci_addr->devid,
1002771fa900SAdrien Mazarguil 			   &pci_addr->function) == 4) {
1003771fa900SAdrien Mazarguil 			ret = 0;
1004771fa900SAdrien Mazarguil 			break;
1005771fa900SAdrien Mazarguil 		}
1006771fa900SAdrien Mazarguil 	}
1007771fa900SAdrien Mazarguil 	fclose(file);
1008771fa900SAdrien Mazarguil 	return 0;
1009771fa900SAdrien Mazarguil }
1010198a3c33SNelio Laranjeiro 
1011198a3c33SNelio Laranjeiro /**
1012c26ae069SMatan Azrad  * Device status handler.
1013c26ae069SMatan Azrad  *
1014af4f09f2SNélio Laranjeiro  * @param dev
1015af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
1016c26ae069SMatan Azrad  * @param events
1017c26ae069SMatan Azrad  *   Pointer to event flags holder.
1018c26ae069SMatan Azrad  *
1019c26ae069SMatan Azrad  * @return
1020c26ae069SMatan Azrad  *   Events bitmap of callback process which can be called immediately.
1021c26ae069SMatan Azrad  */
1022c26ae069SMatan Azrad static uint32_t
1023af4f09f2SNélio Laranjeiro mlx5_dev_status_handler(struct rte_eth_dev *dev)
1024198a3c33SNelio Laranjeiro {
1025af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
1026198a3c33SNelio Laranjeiro 	struct ibv_async_event event;
1027c26ae069SMatan Azrad 	uint32_t ret = 0;
1028198a3c33SNelio Laranjeiro 
1029cfee9475SNélio Laranjeiro 	if (mlx5_link_update(dev, 0) == -EAGAIN) {
1030cfee9475SNélio Laranjeiro 		usleep(0);
1031cfee9475SNélio Laranjeiro 		return 0;
1032cfee9475SNélio Laranjeiro 	}
1033198a3c33SNelio Laranjeiro 	/* Read all message and acknowledge them. */
1034198a3c33SNelio Laranjeiro 	for (;;) {
10350e83b8e5SNelio Laranjeiro 		if (mlx5_glue->get_async_event(priv->ctx, &event))
1036198a3c33SNelio Laranjeiro 			break;
1037c26ae069SMatan Azrad 		if ((event.event_type == IBV_EVENT_PORT_ACTIVE ||
1038c26ae069SMatan Azrad 			event.event_type == IBV_EVENT_PORT_ERR) &&
1039af4f09f2SNélio Laranjeiro 			(dev->data->dev_conf.intr_conf.lsc == 1))
1040c26ae069SMatan Azrad 			ret |= (1 << RTE_ETH_EVENT_INTR_LSC);
10417d7d7ad1SMatan Azrad 		else if (event.event_type == IBV_EVENT_DEVICE_FATAL &&
1042af4f09f2SNélio Laranjeiro 			dev->data->dev_conf.intr_conf.rmv == 1)
10437d7d7ad1SMatan Azrad 			ret |= (1 << RTE_ETH_EVENT_INTR_RMV);
1044c26ae069SMatan Azrad 		else
1045a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG,
1046a170a30dSNélio Laranjeiro 				"port %u event type %d on not handled",
10470f99970bSNélio Laranjeiro 				dev->data->port_id, event.event_type);
10480e83b8e5SNelio Laranjeiro 		mlx5_glue->ack_async_event(&event);
1049198a3c33SNelio Laranjeiro 	}
1050198a3c33SNelio Laranjeiro 	return ret;
1051198a3c33SNelio Laranjeiro }
1052198a3c33SNelio Laranjeiro 
1053198a3c33SNelio Laranjeiro /**
1054198a3c33SNelio Laranjeiro  * Handle interrupts from the NIC.
1055198a3c33SNelio Laranjeiro  *
1056198a3c33SNelio Laranjeiro  * @param[in] intr_handle
1057198a3c33SNelio Laranjeiro  *   Interrupt handler.
1058198a3c33SNelio Laranjeiro  * @param cb_arg
1059198a3c33SNelio Laranjeiro  *   Callback argument.
1060198a3c33SNelio Laranjeiro  */
1061198a3c33SNelio Laranjeiro void
1062c23a1a30SQi Zhang mlx5_dev_interrupt_handler(void *cb_arg)
1063198a3c33SNelio Laranjeiro {
1064198a3c33SNelio Laranjeiro 	struct rte_eth_dev *dev = cb_arg;
1065c26ae069SMatan Azrad 	uint32_t events;
1066198a3c33SNelio Laranjeiro 
1067af4f09f2SNélio Laranjeiro 	events = mlx5_dev_status_handler(dev);
1068c26ae069SMatan Azrad 	if (events & (1 << RTE_ETH_EVENT_INTR_LSC))
1069cebe3d7bSThomas Monjalon 		_rte_eth_dev_callback_process(dev, RTE_ETH_EVENT_INTR_LSC, NULL);
10707d7d7ad1SMatan Azrad 	if (events & (1 << RTE_ETH_EVENT_INTR_RMV))
1071cebe3d7bSThomas Monjalon 		_rte_eth_dev_callback_process(dev, RTE_ETH_EVENT_INTR_RMV, NULL);
1072198a3c33SNelio Laranjeiro }
1073198a3c33SNelio Laranjeiro 
1074198a3c33SNelio Laranjeiro /**
1075f8b9a3baSXueming Li  * Handle interrupts from the socket.
1076f8b9a3baSXueming Li  *
1077f8b9a3baSXueming Li  * @param cb_arg
1078f8b9a3baSXueming Li  *   Callback argument.
1079f8b9a3baSXueming Li  */
1080f8b9a3baSXueming Li static void
1081f8b9a3baSXueming Li mlx5_dev_handler_socket(void *cb_arg)
1082f8b9a3baSXueming Li {
1083f8b9a3baSXueming Li 	struct rte_eth_dev *dev = cb_arg;
1084f8b9a3baSXueming Li 
1085af4f09f2SNélio Laranjeiro 	mlx5_socket_handle(dev);
1086f8b9a3baSXueming Li }
1087f8b9a3baSXueming Li 
1088f8b9a3baSXueming Li /**
1089198a3c33SNelio Laranjeiro  * Uninstall interrupt handler.
1090198a3c33SNelio Laranjeiro  *
1091198a3c33SNelio Laranjeiro  * @param dev
1092af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
1093198a3c33SNelio Laranjeiro  */
1094198a3c33SNelio Laranjeiro void
1095af4f09f2SNélio Laranjeiro mlx5_dev_interrupt_handler_uninstall(struct rte_eth_dev *dev)
1096198a3c33SNelio Laranjeiro {
1097af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
1098af4f09f2SNélio Laranjeiro 
1099f8b9a3baSXueming Li 	if (dev->data->dev_conf.intr_conf.lsc ||
1100f8b9a3baSXueming Li 	    dev->data->dev_conf.intr_conf.rmv)
1101198a3c33SNelio Laranjeiro 		rte_intr_callback_unregister(&priv->intr_handle,
1102f8b9a3baSXueming Li 					     mlx5_dev_interrupt_handler, dev);
1103f8b9a3baSXueming Li 	if (priv->primary_socket)
1104f8b9a3baSXueming Li 		rte_intr_callback_unregister(&priv->intr_handle_socket,
1105f8b9a3baSXueming Li 					     mlx5_dev_handler_socket, dev);
1106198a3c33SNelio Laranjeiro 	priv->intr_handle.fd = 0;
110736351ea3SFerruh Yigit 	priv->intr_handle.type = RTE_INTR_HANDLE_UNKNOWN;
1108f8b9a3baSXueming Li 	priv->intr_handle_socket.fd = 0;
1109f8b9a3baSXueming Li 	priv->intr_handle_socket.type = RTE_INTR_HANDLE_UNKNOWN;
1110198a3c33SNelio Laranjeiro }
1111198a3c33SNelio Laranjeiro 
1112198a3c33SNelio Laranjeiro /**
1113198a3c33SNelio Laranjeiro  * Install interrupt handler.
1114198a3c33SNelio Laranjeiro  *
1115198a3c33SNelio Laranjeiro  * @param dev
1116af4f09f2SNélio Laranjeiro  *   Pointer to Ethernet device.
1117198a3c33SNelio Laranjeiro  */
1118198a3c33SNelio Laranjeiro void
1119af4f09f2SNélio Laranjeiro mlx5_dev_interrupt_handler_install(struct rte_eth_dev *dev)
1120198a3c33SNelio Laranjeiro {
1121af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
1122a6d83b6aSNélio Laranjeiro 	int ret;
1123a6d83b6aSNélio Laranjeiro 	int flags;
1124198a3c33SNelio Laranjeiro 
1125198a3c33SNelio Laranjeiro 	assert(priv->ctx->async_fd > 0);
1126198a3c33SNelio Laranjeiro 	flags = fcntl(priv->ctx->async_fd, F_GETFL);
1127a6d83b6aSNélio Laranjeiro 	ret = fcntl(priv->ctx->async_fd, F_SETFL, flags | O_NONBLOCK);
1128a6d83b6aSNélio Laranjeiro 	if (ret) {
1129a170a30dSNélio Laranjeiro 		DRV_LOG(INFO,
1130a170a30dSNélio Laranjeiro 			"port %u failed to change file descriptor async event"
1131a170a30dSNélio Laranjeiro 			" queue",
1132a170a30dSNélio Laranjeiro 			dev->data->port_id);
1133198a3c33SNelio Laranjeiro 		dev->data->dev_conf.intr_conf.lsc = 0;
11347d7d7ad1SMatan Azrad 		dev->data->dev_conf.intr_conf.rmv = 0;
1135f8b9a3baSXueming Li 	}
1136f8b9a3baSXueming Li 	if (dev->data->dev_conf.intr_conf.lsc ||
1137f8b9a3baSXueming Li 	    dev->data->dev_conf.intr_conf.rmv) {
1138198a3c33SNelio Laranjeiro 		priv->intr_handle.fd = priv->ctx->async_fd;
1139198a3c33SNelio Laranjeiro 		priv->intr_handle.type = RTE_INTR_HANDLE_EXT;
1140198a3c33SNelio Laranjeiro 		rte_intr_callback_register(&priv->intr_handle,
1141f8b9a3baSXueming Li 					   mlx5_dev_interrupt_handler, dev);
1142f8b9a3baSXueming Li 	}
1143a6d83b6aSNélio Laranjeiro 	ret = mlx5_socket_init(dev);
1144a6d83b6aSNélio Laranjeiro 	if (ret)
1145a170a30dSNélio Laranjeiro 		DRV_LOG(ERR, "port %u cannot initialise socket: %s",
11460f99970bSNélio Laranjeiro 			dev->data->port_id, strerror(rte_errno));
1147a6d83b6aSNélio Laranjeiro 	else if (priv->primary_socket) {
1148f8b9a3baSXueming Li 		priv->intr_handle_socket.fd = priv->primary_socket;
1149f8b9a3baSXueming Li 		priv->intr_handle_socket.type = RTE_INTR_HANDLE_EXT;
1150f8b9a3baSXueming Li 		rte_intr_callback_register(&priv->intr_handle_socket,
1151f8b9a3baSXueming Li 					   mlx5_dev_handler_socket, dev);
1152198a3c33SNelio Laranjeiro 	}
1153198a3c33SNelio Laranjeiro }
115462072098SOr Ami 
115562072098SOr Ami /**
115662072098SOr Ami  * DPDK callback to bring the link DOWN.
115762072098SOr Ami  *
115862072098SOr Ami  * @param dev
115962072098SOr Ami  *   Pointer to Ethernet device structure.
116062072098SOr Ami  *
116162072098SOr Ami  * @return
1162a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
116362072098SOr Ami  */
116462072098SOr Ami int
116562072098SOr Ami mlx5_set_link_down(struct rte_eth_dev *dev)
116662072098SOr Ami {
1167af4f09f2SNélio Laranjeiro 	return mlx5_set_flags(dev, ~IFF_UP, ~IFF_UP);
116862072098SOr Ami }
116962072098SOr Ami 
117062072098SOr Ami /**
117162072098SOr Ami  * DPDK callback to bring the link UP.
117262072098SOr Ami  *
117362072098SOr Ami  * @param dev
117462072098SOr Ami  *   Pointer to Ethernet device structure.
117562072098SOr Ami  *
117662072098SOr Ami  * @return
1177a6d83b6aSNélio Laranjeiro  *   0 on success, a negative errno value otherwise and rte_errno is set.
117862072098SOr Ami  */
117962072098SOr Ami int
118062072098SOr Ami mlx5_set_link_up(struct rte_eth_dev *dev)
118162072098SOr Ami {
1182af4f09f2SNélio Laranjeiro 	return mlx5_set_flags(dev, ~IFF_UP, IFF_UP);
118362072098SOr Ami }
1184a48deadaSOr Ami 
1185a48deadaSOr Ami /**
1186cdab90cbSNélio Laranjeiro  * Configure the TX function to use.
1187cdab90cbSNélio Laranjeiro  *
1188aee1b165SXueming Li  * @param dev
1189af4f09f2SNélio Laranjeiro  *   Pointer to private data structure.
11901cfa649bSShahaf Shuler  *
11911cfa649bSShahaf Shuler  * @return
11921cfa649bSShahaf Shuler  *   Pointer to selected Tx burst function.
1193cdab90cbSNélio Laranjeiro  */
11941cfa649bSShahaf Shuler eth_tx_burst_t
1195af4f09f2SNélio Laranjeiro mlx5_select_tx_function(struct rte_eth_dev *dev)
1196cdab90cbSNélio Laranjeiro {
1197af4f09f2SNélio Laranjeiro 	struct priv *priv = dev->data->dev_private;
11981cfa649bSShahaf Shuler 	eth_tx_burst_t tx_pkt_burst = mlx5_tx_burst;
11997fe24446SShahaf Shuler 	struct mlx5_dev_config *config = &priv->config;
1200dbccb4cdSShahaf Shuler 	uint64_t tx_offloads = dev->data->dev_conf.txmode.offloads;
1201dbccb4cdSShahaf Shuler 	int tso = !!(tx_offloads & (DEV_TX_OFFLOAD_TCP_TSO |
1202dbccb4cdSShahaf Shuler 				    DEV_TX_OFFLOAD_VXLAN_TNL_TSO |
120332d4246cSXueming Li 				    DEV_TX_OFFLOAD_GRE_TNL_TSO |
120432d4246cSXueming Li 				    DEV_TX_OFFLOAD_IP_TNL_TSO |
120532d4246cSXueming Li 				    DEV_TX_OFFLOAD_UDP_TNL_TSO));
12065f8ba81cSXueming Li 	int swp = !!(tx_offloads & (DEV_TX_OFFLOAD_IP_TNL_TSO |
12075f8ba81cSXueming Li 				    DEV_TX_OFFLOAD_UDP_TNL_TSO |
12085f8ba81cSXueming Li 				    DEV_TX_OFFLOAD_OUTER_IPV4_CKSUM));
1209dbccb4cdSShahaf Shuler 	int vlan_insert = !!(tx_offloads & DEV_TX_OFFLOAD_VLAN_INSERT);
12101cfa649bSShahaf Shuler 
1211aee1b165SXueming Li 	assert(priv != NULL);
1212230189d9SNélio Laranjeiro 	/* Select appropriate TX function. */
12135f8ba81cSXueming Li 	if (vlan_insert || tso || swp)
1214dbccb4cdSShahaf Shuler 		return tx_pkt_burst;
12157fe24446SShahaf Shuler 	if (config->mps == MLX5_MPW_ENHANCED) {
1216af4f09f2SNélio Laranjeiro 		if (mlx5_check_vec_tx_support(dev) > 0) {
1217af4f09f2SNélio Laranjeiro 			if (mlx5_check_raw_vec_tx_support(dev) > 0)
12181cfa649bSShahaf Shuler 				tx_pkt_burst = mlx5_tx_burst_raw_vec;
12196cb559d6SYongseok Koh 			else
12201cfa649bSShahaf Shuler 				tx_pkt_burst = mlx5_tx_burst_vec;
1221a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG,
1222a170a30dSNélio Laranjeiro 				"port %u selected enhanced MPW Tx vectorized"
1223a170a30dSNélio Laranjeiro 				" function",
1224a170a30dSNélio Laranjeiro 				dev->data->port_id);
12256cb559d6SYongseok Koh 		} else {
12261cfa649bSShahaf Shuler 			tx_pkt_burst = mlx5_tx_burst_empw;
1227a170a30dSNélio Laranjeiro 			DRV_LOG(DEBUG,
1228a170a30dSNélio Laranjeiro 				"port %u selected enhanced MPW Tx function",
12290f99970bSNélio Laranjeiro 				dev->data->port_id);
12306cb559d6SYongseok Koh 		}
12317fe24446SShahaf Shuler 	} else if (config->mps && (config->txq_inline > 0)) {
12321cfa649bSShahaf Shuler 		tx_pkt_burst = mlx5_tx_burst_mpw_inline;
1233a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "port %u selected MPW inline Tx function",
12340f99970bSNélio Laranjeiro 			dev->data->port_id);
12357fe24446SShahaf Shuler 	} else if (config->mps) {
12361cfa649bSShahaf Shuler 		tx_pkt_burst = mlx5_tx_burst_mpw;
1237a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "port %u selected MPW Tx function",
1238a170a30dSNélio Laranjeiro 			dev->data->port_id);
12392a66cf37SYaacov Hazan 	}
12401cfa649bSShahaf Shuler 	return tx_pkt_burst;
1241cdab90cbSNélio Laranjeiro }
1242cdab90cbSNélio Laranjeiro 
1243cdab90cbSNélio Laranjeiro /**
1244cdab90cbSNélio Laranjeiro  * Configure the RX function to use.
1245cdab90cbSNélio Laranjeiro  *
1246aee1b165SXueming Li  * @param dev
1247af4f09f2SNélio Laranjeiro  *   Pointer to private data structure.
12481cfa649bSShahaf Shuler  *
12491cfa649bSShahaf Shuler  * @return
12501cfa649bSShahaf Shuler  *   Pointer to selected Rx burst function.
1251cdab90cbSNélio Laranjeiro  */
12521cfa649bSShahaf Shuler eth_rx_burst_t
1253af4f09f2SNélio Laranjeiro mlx5_select_rx_function(struct rte_eth_dev *dev)
1254cdab90cbSNélio Laranjeiro {
12551cfa649bSShahaf Shuler 	eth_rx_burst_t rx_pkt_burst = mlx5_rx_burst;
12561cfa649bSShahaf Shuler 
1257af4f09f2SNélio Laranjeiro 	assert(dev != NULL);
1258af4f09f2SNélio Laranjeiro 	if (mlx5_check_vec_rx_support(dev) > 0) {
12591cfa649bSShahaf Shuler 		rx_pkt_burst = mlx5_rx_burst_vec;
1260a170a30dSNélio Laranjeiro 		DRV_LOG(DEBUG, "port %u selected Rx vectorized function",
12610f99970bSNélio Laranjeiro 			dev->data->port_id);
12627d6bf6b8SYongseok Koh 	} else if (mlx5_mprq_enabled(dev)) {
12637d6bf6b8SYongseok Koh 		rx_pkt_burst = mlx5_rx_burst_mprq;
1264cdab90cbSNélio Laranjeiro 	}
12651cfa649bSShahaf Shuler 	return rx_pkt_burst;
12666cb559d6SYongseok Koh }
1267d3e0f392SMatan Azrad 
1268d3e0f392SMatan Azrad /**
1269d3e0f392SMatan Azrad  * Check if mlx5 device was removed.
1270d3e0f392SMatan Azrad  *
1271d3e0f392SMatan Azrad  * @param dev
1272d3e0f392SMatan Azrad  *   Pointer to Ethernet device structure.
1273d3e0f392SMatan Azrad  *
1274d3e0f392SMatan Azrad  * @return
1275d3e0f392SMatan Azrad  *   1 when device is removed, otherwise 0.
1276d3e0f392SMatan Azrad  */
1277d3e0f392SMatan Azrad int
1278d3e0f392SMatan Azrad mlx5_is_removed(struct rte_eth_dev *dev)
1279d3e0f392SMatan Azrad {
1280d3e0f392SMatan Azrad 	struct ibv_device_attr device_attr;
1281d3e0f392SMatan Azrad 	struct priv *priv = dev->data->dev_private;
1282d3e0f392SMatan Azrad 
12830e83b8e5SNelio Laranjeiro 	if (mlx5_glue->query_device(priv->ctx, &device_attr) == EIO)
1284d3e0f392SMatan Azrad 		return 1;
1285d3e0f392SMatan Azrad 	return 0;
1286d3e0f392SMatan Azrad }
12872b730263SAdrien Mazarguil 
12882b730263SAdrien Mazarguil /**
12892b730263SAdrien Mazarguil  * Get port ID list of mlx5 instances sharing a common device.
12902b730263SAdrien Mazarguil  *
12912b730263SAdrien Mazarguil  * @param[in] dev
12922b730263SAdrien Mazarguil  *   Device to look for.
12932b730263SAdrien Mazarguil  * @param[out] port_list
12942b730263SAdrien Mazarguil  *   Result buffer for collected port IDs.
12952b730263SAdrien Mazarguil  * @param port_list_n
12962b730263SAdrien Mazarguil  *   Maximum number of entries in result buffer. If 0, @p port_list can be
12972b730263SAdrien Mazarguil  *   NULL.
12982b730263SAdrien Mazarguil  *
12992b730263SAdrien Mazarguil  * @return
13002b730263SAdrien Mazarguil  *   Number of matching instances regardless of the @p port_list_n
13012b730263SAdrien Mazarguil  *   parameter, 0 if none were found.
13022b730263SAdrien Mazarguil  */
13032b730263SAdrien Mazarguil unsigned int
13042b730263SAdrien Mazarguil mlx5_dev_to_port_id(const struct rte_device *dev, uint16_t *port_list,
13052b730263SAdrien Mazarguil 		    unsigned int port_list_n)
13062b730263SAdrien Mazarguil {
13072b730263SAdrien Mazarguil 	uint16_t id;
13082b730263SAdrien Mazarguil 	unsigned int n = 0;
13092b730263SAdrien Mazarguil 
13102b730263SAdrien Mazarguil 	RTE_ETH_FOREACH_DEV(id) {
13112b730263SAdrien Mazarguil 		struct rte_eth_dev *ldev = &rte_eth_devices[id];
13122b730263SAdrien Mazarguil 
13132b730263SAdrien Mazarguil 		if (!ldev->device ||
13142b730263SAdrien Mazarguil 		    !ldev->device->driver ||
13152b730263SAdrien Mazarguil 		    strcmp(ldev->device->driver->name, MLX5_DRIVER_NAME) ||
13162b730263SAdrien Mazarguil 		    ldev->device != dev)
13172b730263SAdrien Mazarguil 			continue;
13182b730263SAdrien Mazarguil 		if (n < port_list_n)
13192b730263SAdrien Mazarguil 			port_list[n] = id;
13202b730263SAdrien Mazarguil 		n++;
13212b730263SAdrien Mazarguil 	}
13222b730263SAdrien Mazarguil 	return n;
13232b730263SAdrien Mazarguil }
1324