xref: /dpdk/drivers/net/mvneta/mvneta_ethdev.c (revision c705c67d304b9450824a169b652520c2358c6aee)
14ccc8d77SZyta Szpak /* SPDX-License-Identifier: BSD-3-Clause
24ccc8d77SZyta Szpak  * Copyright(c) 2018 Marvell International Ltd.
34ccc8d77SZyta Szpak  * Copyright(c) 2018 Semihalf.
44ccc8d77SZyta Szpak  * All rights reserved.
54ccc8d77SZyta Szpak  */
64ccc8d77SZyta Szpak 
76723c0fcSBruce Richardson #include <rte_string_fns.h>
8df96fd0dSBruce Richardson #include <ethdev_driver.h>
94ccc8d77SZyta Szpak #include <rte_kvargs.h>
104851ef2bSDavid Marchand #include <bus_vdev_driver.h>
114ccc8d77SZyta Szpak 
124ccc8d77SZyta Szpak #include <stdio.h>
134ccc8d77SZyta Szpak #include <fcntl.h>
144ccc8d77SZyta Szpak #include <linux/ethtool.h>
154ccc8d77SZyta Szpak #include <linux/sockios.h>
164ccc8d77SZyta Szpak #include <net/if.h>
174ccc8d77SZyta Szpak #include <net/if_arp.h>
184ccc8d77SZyta Szpak #include <sys/ioctl.h>
194ccc8d77SZyta Szpak #include <sys/socket.h>
204ccc8d77SZyta Szpak #include <sys/stat.h>
214ccc8d77SZyta Szpak #include <sys/types.h>
224ccc8d77SZyta Szpak 
234ccc8d77SZyta Szpak #include <rte_mvep_common.h>
244ccc8d77SZyta Szpak 
25ce7ea764SZyta Szpak #include "mvneta_rxtx.h"
264ccc8d77SZyta Szpak 
274ccc8d77SZyta Szpak 
284ccc8d77SZyta Szpak #define MVNETA_IFACE_NAME_ARG "iface"
294ccc8d77SZyta Szpak 
304ccc8d77SZyta Szpak #define MVNETA_PKT_SIZE_MAX (16382 - MV_MH_SIZE) /* 9700B */
314ccc8d77SZyta Szpak #define MVNETA_DEFAULT_MTU 1500
324ccc8d77SZyta Szpak 
334ccc8d77SZyta Szpak #define MVNETA_MAC_ADDRS_MAX 256 /*16 UC, 256 IP, 256 MC/BC */
344ccc8d77SZyta Szpak /** Maximum length of a match string */
354ccc8d77SZyta Szpak #define MVNETA_MATCH_LEN 16
364ccc8d77SZyta Szpak 
374ccc8d77SZyta Szpak static const char * const valid_args[] = {
384ccc8d77SZyta Szpak 	MVNETA_IFACE_NAME_ARG,
394ccc8d77SZyta Szpak 	NULL
404ccc8d77SZyta Szpak };
414ccc8d77SZyta Szpak 
424ccc8d77SZyta Szpak struct mvneta_ifnames {
434ccc8d77SZyta Szpak 	const char *names[NETA_NUM_ETH_PPIO];
444ccc8d77SZyta Szpak 	int idx;
454ccc8d77SZyta Szpak };
464ccc8d77SZyta Szpak 
474ccc8d77SZyta Szpak static int mvneta_dev_num;
484ccc8d77SZyta Szpak 
499970a9adSIgor Romanov static int mvneta_stats_reset(struct rte_eth_dev *dev);
504b6d2743SYuri Chipchev static int rte_pmd_mvneta_remove(struct rte_vdev_device *vdev);
514b6d2743SYuri Chipchev 
5215ae783aSYuri Chipchev 
534ccc8d77SZyta Szpak /**
544ccc8d77SZyta Szpak  * Deinitialize packet processor.
554ccc8d77SZyta Szpak  */
564ccc8d77SZyta Szpak static void
574ccc8d77SZyta Szpak mvneta_neta_deinit(void)
584ccc8d77SZyta Szpak {
594ccc8d77SZyta Szpak 	neta_deinit();
604ccc8d77SZyta Szpak }
614ccc8d77SZyta Szpak 
624ccc8d77SZyta Szpak /**
634ccc8d77SZyta Szpak  * Initialize packet processor.
644ccc8d77SZyta Szpak  *
654ccc8d77SZyta Szpak  * @return
664ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
674ccc8d77SZyta Szpak  */
684ccc8d77SZyta Szpak static int
694ccc8d77SZyta Szpak mvneta_neta_init(void)
704ccc8d77SZyta Szpak {
714ccc8d77SZyta Szpak 	return neta_init();
724ccc8d77SZyta Szpak }
734ccc8d77SZyta Szpak 
744ccc8d77SZyta Szpak /**
754ccc8d77SZyta Szpak  * Callback used by rte_kvargs_process() during argument parsing.
764ccc8d77SZyta Szpak  *
774ccc8d77SZyta Szpak  * @param key
784ccc8d77SZyta Szpak  *   Pointer to the parsed key (unused).
794ccc8d77SZyta Szpak  * @param value
804ccc8d77SZyta Szpak  *   Pointer to the parsed value.
814ccc8d77SZyta Szpak  * @param extra_args
824ccc8d77SZyta Szpak  *   Pointer to the extra arguments which contains address of the
834ccc8d77SZyta Szpak  *   table of pointers to parsed interface names.
844ccc8d77SZyta Szpak  *
854ccc8d77SZyta Szpak  * @return
864ccc8d77SZyta Szpak  *   Always 0.
874ccc8d77SZyta Szpak  */
884ccc8d77SZyta Szpak static int
894ccc8d77SZyta Szpak mvneta_ifnames_get(const char *key __rte_unused, const char *value,
904ccc8d77SZyta Szpak 		 void *extra_args)
914ccc8d77SZyta Szpak {
924ccc8d77SZyta Szpak 	struct mvneta_ifnames *ifnames = extra_args;
934ccc8d77SZyta Szpak 
94*c705c67dSChengwen Feng 	if (ifnames->idx >= NETA_NUM_ETH_PPIO) {
95*c705c67dSChengwen Feng 		MVNETA_LOG(ERR, "Too many ifnames specified (max %u)",
96*c705c67dSChengwen Feng 			   NETA_NUM_ETH_PPIO);
97*c705c67dSChengwen Feng 		return -EINVAL;
98*c705c67dSChengwen Feng 	}
99*c705c67dSChengwen Feng 
1004ccc8d77SZyta Szpak 	ifnames->names[ifnames->idx++] = value;
1014ccc8d77SZyta Szpak 
1024ccc8d77SZyta Szpak 	return 0;
1034ccc8d77SZyta Szpak }
1044ccc8d77SZyta Szpak 
1054ccc8d77SZyta Szpak /**
1064ccc8d77SZyta Szpak  * Ethernet device configuration.
1074ccc8d77SZyta Szpak  *
1084ccc8d77SZyta Szpak  * Prepare the driver for a given number of TX and RX queues and
1094ccc8d77SZyta Szpak  * configure RSS if supported.
1104ccc8d77SZyta Szpak  *
1114ccc8d77SZyta Szpak  * @param dev
1124ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
1134ccc8d77SZyta Szpak  *
1144ccc8d77SZyta Szpak  * @return
1154ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
1164ccc8d77SZyta Szpak  */
1174ccc8d77SZyta Szpak static int
1184ccc8d77SZyta Szpak mvneta_dev_configure(struct rte_eth_dev *dev)
1194ccc8d77SZyta Szpak {
1204ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
1214ccc8d77SZyta Szpak 	struct neta_ppio_params *ppio_params;
1224ccc8d77SZyta Szpak 
123295968d1SFerruh Yigit 	if (dev->data->dev_conf.rxmode.mq_mode != RTE_ETH_MQ_RX_NONE) {
1244ccc8d77SZyta Szpak 		MVNETA_LOG(INFO, "Unsupported RSS and rx multi queue mode %d",
1254ccc8d77SZyta Szpak 			dev->data->dev_conf.rxmode.mq_mode);
1264ccc8d77SZyta Szpak 		if (dev->data->nb_rx_queues > 1)
1274ccc8d77SZyta Szpak 			return -EINVAL;
1284ccc8d77SZyta Szpak 	}
1294ccc8d77SZyta Szpak 
130295968d1SFerruh Yigit 	if (dev->data->dev_conf.txmode.offloads & RTE_ETH_TX_OFFLOAD_MULTI_SEGS)
1314ccc8d77SZyta Szpak 		priv->multiseg = 1;
1324ccc8d77SZyta Szpak 
1334ccc8d77SZyta Szpak 	ppio_params = &priv->ppio_params;
1344ccc8d77SZyta Szpak 	ppio_params->outqs_params.num_outqs = dev->data->nb_tx_queues;
1354ccc8d77SZyta Szpak 	/* Default: 1 TC, no QoS supported. */
1364ccc8d77SZyta Szpak 	ppio_params->inqs_params.num_tcs = 1;
1374ccc8d77SZyta Szpak 	ppio_params->inqs_params.tcs_params[0].pkt_offset = MRVL_NETA_PKT_OFFS;
1384ccc8d77SZyta Szpak 	priv->ppio_id = dev->data->port_id;
1394ccc8d77SZyta Szpak 
1404ccc8d77SZyta Szpak 	return 0;
1414ccc8d77SZyta Szpak }
1424ccc8d77SZyta Szpak 
1434ccc8d77SZyta Szpak /**
1444ccc8d77SZyta Szpak  * DPDK callback to get information about the device.
1454ccc8d77SZyta Szpak  *
1464ccc8d77SZyta Szpak  * @param dev
1474ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure (unused).
1484ccc8d77SZyta Szpak  * @param info
1494ccc8d77SZyta Szpak  *   Info structure output buffer.
1504ccc8d77SZyta Szpak  */
151bdad90d1SIvan Ilchenko static int
1524ccc8d77SZyta Szpak mvneta_dev_infos_get(struct rte_eth_dev *dev __rte_unused,
1534ccc8d77SZyta Szpak 		   struct rte_eth_dev_info *info)
1544ccc8d77SZyta Szpak {
155295968d1SFerruh Yigit 	info->speed_capa = RTE_ETH_LINK_SPEED_10M |
156295968d1SFerruh Yigit 			   RTE_ETH_LINK_SPEED_100M |
157295968d1SFerruh Yigit 			   RTE_ETH_LINK_SPEED_1G |
158295968d1SFerruh Yigit 			   RTE_ETH_LINK_SPEED_2_5G;
1594ccc8d77SZyta Szpak 
1604ccc8d77SZyta Szpak 	info->max_rx_queues = MRVL_NETA_RXQ_MAX;
1614ccc8d77SZyta Szpak 	info->max_tx_queues = MRVL_NETA_TXQ_MAX;
1624ccc8d77SZyta Szpak 	info->max_mac_addrs = MVNETA_MAC_ADDRS_MAX;
1634ccc8d77SZyta Szpak 
1644ccc8d77SZyta Szpak 	info->rx_desc_lim.nb_max = MRVL_NETA_RXD_MAX;
1654ccc8d77SZyta Szpak 	info->rx_desc_lim.nb_min = MRVL_NETA_RXD_MIN;
1664ccc8d77SZyta Szpak 	info->rx_desc_lim.nb_align = MRVL_NETA_RXD_ALIGN;
1674ccc8d77SZyta Szpak 
1684ccc8d77SZyta Szpak 	info->tx_desc_lim.nb_max = MRVL_NETA_TXD_MAX;
1694ccc8d77SZyta Szpak 	info->tx_desc_lim.nb_min = MRVL_NETA_TXD_MIN;
1704ccc8d77SZyta Szpak 	info->tx_desc_lim.nb_align = MRVL_NETA_TXD_ALIGN;
1714ccc8d77SZyta Szpak 
1724ccc8d77SZyta Szpak 	info->rx_offload_capa = MVNETA_RX_OFFLOADS;
1734ccc8d77SZyta Szpak 	info->rx_queue_offload_capa = MVNETA_RX_OFFLOADS;
1744ccc8d77SZyta Szpak 
1754ccc8d77SZyta Szpak 	info->tx_offload_capa =  MVNETA_TX_OFFLOADS;
1764ccc8d77SZyta Szpak 	info->tx_queue_offload_capa =  MVNETA_TX_OFFLOADS;
1774ccc8d77SZyta Szpak 
1784ccc8d77SZyta Szpak 	/* By default packets are dropped if no descriptors are available */
1794ccc8d77SZyta Szpak 	info->default_rxconf.rx_drop_en = 1;
1804ccc8d77SZyta Szpak 	/* Deferred tx queue start is not supported */
1814ccc8d77SZyta Szpak 	info->default_txconf.tx_deferred_start = 0;
1824ccc8d77SZyta Szpak 	info->default_txconf.offloads = 0;
1834ccc8d77SZyta Szpak 
1844ccc8d77SZyta Szpak 	info->max_rx_pktlen = MVNETA_PKT_SIZE_MAX;
185bdad90d1SIvan Ilchenko 
186bdad90d1SIvan Ilchenko 	return 0;
1874ccc8d77SZyta Szpak }
1884ccc8d77SZyta Szpak 
1894ccc8d77SZyta Szpak /**
1904ccc8d77SZyta Szpak  * Return supported packet types.
1914ccc8d77SZyta Szpak  *
1924ccc8d77SZyta Szpak  * @param dev
1934ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure (unused).
1944ccc8d77SZyta Szpak  *
1954ccc8d77SZyta Szpak  * @return
1964ccc8d77SZyta Szpak  *   Const pointer to the table with supported packet types.
1974ccc8d77SZyta Szpak  */
1984ccc8d77SZyta Szpak static const uint32_t *
199ba6a168aSSivaramakrishnan Venkat mvneta_dev_supported_ptypes_get(struct rte_eth_dev *dev __rte_unused,
200ba6a168aSSivaramakrishnan Venkat 				size_t *no_of_elements)
2014ccc8d77SZyta Szpak {
2024ccc8d77SZyta Szpak 	static const uint32_t ptypes[] = {
2034ccc8d77SZyta Szpak 		RTE_PTYPE_L2_ETHER,
2044ccc8d77SZyta Szpak 		RTE_PTYPE_L2_ETHER_VLAN,
2054ccc8d77SZyta Szpak 		RTE_PTYPE_L3_IPV4,
2064ccc8d77SZyta Szpak 		RTE_PTYPE_L3_IPV6,
2074ccc8d77SZyta Szpak 		RTE_PTYPE_L4_TCP,
2082e3ddb56SSivaramakrishnan Venkat 		RTE_PTYPE_L4_UDP,
2094ccc8d77SZyta Szpak 	};
2104ccc8d77SZyta Szpak 
211ba6a168aSSivaramakrishnan Venkat 	*no_of_elements = RTE_DIM(ptypes);
2124ccc8d77SZyta Szpak 	return ptypes;
2134ccc8d77SZyta Szpak }
2144ccc8d77SZyta Szpak 
2154ccc8d77SZyta Szpak /**
216112ef725SZyta Szpak  * DPDK callback to change the MTU.
217112ef725SZyta Szpak  *
218112ef725SZyta Szpak  * Setting the MTU affects hardware MRU (packets larger than the MRU
219112ef725SZyta Szpak  * will be dropped).
220112ef725SZyta Szpak  *
221112ef725SZyta Szpak  * @param dev
222112ef725SZyta Szpak  *   Pointer to Ethernet device structure.
223112ef725SZyta Szpak  * @param mtu
224112ef725SZyta Szpak  *   New MTU.
225112ef725SZyta Szpak  *
226112ef725SZyta Szpak  * @return
227112ef725SZyta Szpak  *   0 on success, negative error value otherwise.
228112ef725SZyta Szpak  */
229112ef725SZyta Szpak static int
230112ef725SZyta Szpak mvneta_mtu_set(struct rte_eth_dev *dev, uint16_t mtu)
231112ef725SZyta Szpak {
232112ef725SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
233112ef725SZyta Szpak 	uint16_t mbuf_data_size = 0; /* SW buffer size */
234112ef725SZyta Szpak 	uint16_t mru;
235112ef725SZyta Szpak 	int ret;
236112ef725SZyta Szpak 
237112ef725SZyta Szpak 	mru = MRVL_NETA_MTU_TO_MRU(mtu);
238112ef725SZyta Szpak 	/*
239112ef725SZyta Szpak 	 * min_rx_buf_size is equal to mbuf data size
240112ef725SZyta Szpak 	 * if pmd didn't set it differently
241112ef725SZyta Szpak 	 */
242112ef725SZyta Szpak 	mbuf_data_size = dev->data->min_rx_buf_size - RTE_PKTMBUF_HEADROOM;
243112ef725SZyta Szpak 	/* Prevent PMD from:
244112ef725SZyta Szpak 	 * - setting mru greater than the mbuf size resulting in
245112ef725SZyta Szpak 	 * hw and sw buffer size mismatch
246112ef725SZyta Szpak 	 * - setting mtu that requires the support of scattered packets
247112ef725SZyta Szpak 	 * when this feature has not been enabled/supported so far.
248112ef725SZyta Szpak 	 */
249112ef725SZyta Szpak 	if (!dev->data->scattered_rx &&
250112ef725SZyta Szpak 	    (mru + MRVL_NETA_PKT_OFFS > mbuf_data_size)) {
251112ef725SZyta Szpak 		mru = mbuf_data_size - MRVL_NETA_PKT_OFFS;
252112ef725SZyta Szpak 		mtu = MRVL_NETA_MRU_TO_MTU(mru);
2537be78d02SJosh Soref 		MVNETA_LOG(WARNING, "MTU too big, max MTU possible limited by"
254112ef725SZyta Szpak 			" current mbuf size: %u. Set MTU to %u, MRU to %u",
255112ef725SZyta Szpak 			mbuf_data_size, mtu, mru);
256112ef725SZyta Szpak 	}
257112ef725SZyta Szpak 
25835b2d13fSOlivier Matz 	if (mtu < RTE_ETHER_MIN_MTU || mru > MVNETA_PKT_SIZE_MAX) {
259112ef725SZyta Szpak 		MVNETA_LOG(ERR, "Invalid MTU [%u] or MRU [%u]", mtu, mru);
260112ef725SZyta Szpak 		return -EINVAL;
261112ef725SZyta Szpak 	}
262112ef725SZyta Szpak 
263112ef725SZyta Szpak 	if (!priv->ppio)
264112ef725SZyta Szpak 		/* It is OK. New MTU will be set later on mvneta_dev_start */
265112ef725SZyta Szpak 		return 0;
266112ef725SZyta Szpak 
267112ef725SZyta Szpak 	ret = neta_ppio_set_mru(priv->ppio, mru);
268112ef725SZyta Szpak 	if (ret) {
269112ef725SZyta Szpak 		MVNETA_LOG(ERR, "Failed to change MRU");
270112ef725SZyta Szpak 		return ret;
271112ef725SZyta Szpak 	}
272112ef725SZyta Szpak 
273112ef725SZyta Szpak 	ret = neta_ppio_set_mtu(priv->ppio, mtu);
274112ef725SZyta Szpak 	if (ret) {
275112ef725SZyta Szpak 		MVNETA_LOG(ERR, "Failed to change MTU");
276112ef725SZyta Szpak 		return ret;
277112ef725SZyta Szpak 	}
278112ef725SZyta Szpak 	MVNETA_LOG(INFO, "MTU changed to %u, MRU = %u", mtu, mru);
279112ef725SZyta Szpak 
280112ef725SZyta Szpak 	return 0;
281112ef725SZyta Szpak }
282112ef725SZyta Szpak 
283112ef725SZyta Szpak /**
2844ccc8d77SZyta Szpak  * DPDK callback to bring the link up.
2854ccc8d77SZyta Szpak  *
2864ccc8d77SZyta Szpak  * @param dev
2874ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
2884ccc8d77SZyta Szpak  *
2894ccc8d77SZyta Szpak  * @return
2904ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
2914ccc8d77SZyta Szpak  */
2924ccc8d77SZyta Szpak static int
2934ccc8d77SZyta Szpak mvneta_dev_set_link_up(struct rte_eth_dev *dev)
2944ccc8d77SZyta Szpak {
2954ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
2964ccc8d77SZyta Szpak 
2974ccc8d77SZyta Szpak 	if (!priv->ppio)
2984ccc8d77SZyta Szpak 		return 0;
2994ccc8d77SZyta Szpak 
3004ccc8d77SZyta Szpak 	return neta_ppio_enable(priv->ppio);
3014ccc8d77SZyta Szpak }
3024ccc8d77SZyta Szpak 
3034ccc8d77SZyta Szpak /**
3044ccc8d77SZyta Szpak  * DPDK callback to bring the link down.
3054ccc8d77SZyta Szpak  *
3064ccc8d77SZyta Szpak  * @param dev
3074ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
3084ccc8d77SZyta Szpak  *
3094ccc8d77SZyta Szpak  * @return
3104ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
3114ccc8d77SZyta Szpak  */
3124ccc8d77SZyta Szpak static int
3134ccc8d77SZyta Szpak mvneta_dev_set_link_down(struct rte_eth_dev *dev)
3144ccc8d77SZyta Szpak {
3154ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
3164ccc8d77SZyta Szpak 
3174ccc8d77SZyta Szpak 	if (!priv->ppio)
3184ccc8d77SZyta Szpak 		return 0;
3194ccc8d77SZyta Szpak 
3204ccc8d77SZyta Szpak 	return neta_ppio_disable(priv->ppio);
3214ccc8d77SZyta Szpak }
3224ccc8d77SZyta Szpak 
3234ccc8d77SZyta Szpak /**
3244ccc8d77SZyta Szpak  * DPDK callback to start the device.
3254ccc8d77SZyta Szpak  *
3264ccc8d77SZyta Szpak  * @param dev
3274ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
3284ccc8d77SZyta Szpak  *
3294ccc8d77SZyta Szpak  * @return
3304ccc8d77SZyta Szpak  *   0 on success, negative errno value on failure.
3314ccc8d77SZyta Szpak  */
3324ccc8d77SZyta Szpak static int
3334ccc8d77SZyta Szpak mvneta_dev_start(struct rte_eth_dev *dev)
3344ccc8d77SZyta Szpak {
3354ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
3364ccc8d77SZyta Szpak 	char match[MVNETA_MATCH_LEN];
3374ccc8d77SZyta Szpak 	int ret = 0, i;
3384ccc8d77SZyta Szpak 
3394ccc8d77SZyta Szpak 	if (priv->ppio)
3404ccc8d77SZyta Szpak 		return mvneta_dev_set_link_up(dev);
3414ccc8d77SZyta Szpak 
3426723c0fcSBruce Richardson 	strlcpy(match, dev->data->name, sizeof(match));
3434ccc8d77SZyta Szpak 	priv->ppio_params.match = match;
3444ccc8d77SZyta Szpak 	priv->ppio_params.inqs_params.mtu = dev->data->mtu;
3454ccc8d77SZyta Szpak 
3464ccc8d77SZyta Szpak 	ret = neta_ppio_init(&priv->ppio_params, &priv->ppio);
3474ccc8d77SZyta Szpak 	if (ret) {
3484ccc8d77SZyta Szpak 		MVNETA_LOG(ERR, "Failed to init ppio");
3494ccc8d77SZyta Szpak 		return ret;
3504ccc8d77SZyta Szpak 	}
3514ccc8d77SZyta Szpak 	priv->ppio_id = priv->ppio->port_id;
3524ccc8d77SZyta Szpak 
35315ae783aSYuri Chipchev 	mvneta_stats_reset(dev);
35415ae783aSYuri Chipchev 
3554ccc8d77SZyta Szpak 	/*
35623f3dac4SStephen Hemminger 	 * In case there are some stale uc/mc mac addresses flush them
3574ccc8d77SZyta Szpak 	 * here. It cannot be done during mvneta_dev_close() as port information
3584ccc8d77SZyta Szpak 	 * is already gone at that point (due to neta_ppio_deinit() in
3594ccc8d77SZyta Szpak 	 * mvneta_dev_stop()).
3604ccc8d77SZyta Szpak 	 */
3614ccc8d77SZyta Szpak 	if (!priv->uc_mc_flushed) {
3624ccc8d77SZyta Szpak 		ret = neta_ppio_flush_mac_addrs(priv->ppio, 0, 1);
3634ccc8d77SZyta Szpak 		if (ret) {
3644ccc8d77SZyta Szpak 			MVNETA_LOG(ERR,
3654ccc8d77SZyta Szpak 				"Failed to flush uc/mc filter list");
3664ccc8d77SZyta Szpak 			goto out;
3674ccc8d77SZyta Szpak 		}
3684ccc8d77SZyta Szpak 		priv->uc_mc_flushed = 1;
3694ccc8d77SZyta Szpak 	}
3704ccc8d77SZyta Szpak 
371ce7ea764SZyta Szpak 	ret = mvneta_alloc_rx_bufs(dev);
372ce7ea764SZyta Szpak 	if (ret)
373ce7ea764SZyta Szpak 		goto out;
374ce7ea764SZyta Szpak 
375112ef725SZyta Szpak 	ret = mvneta_mtu_set(dev, dev->data->mtu);
376112ef725SZyta Szpak 	if (ret) {
377112ef725SZyta Szpak 		MVNETA_LOG(ERR, "Failed to set MTU %d", dev->data->mtu);
378112ef725SZyta Szpak 		goto out;
379112ef725SZyta Szpak 	}
380112ef725SZyta Szpak 
3814ccc8d77SZyta Szpak 	ret = mvneta_dev_set_link_up(dev);
3824ccc8d77SZyta Szpak 	if (ret) {
3834ccc8d77SZyta Szpak 		MVNETA_LOG(ERR, "Failed to set link up");
3844ccc8d77SZyta Szpak 		goto out;
3854ccc8d77SZyta Szpak 	}
3864ccc8d77SZyta Szpak 
38711f489baSJie Hai 	/* start rx queues */
38811f489baSJie Hai 	for (i = 0; i < dev->data->nb_rx_queues; i++)
38911f489baSJie Hai 		dev->data->rx_queue_state[i] = RTE_ETH_QUEUE_STATE_STARTED;
39011f489baSJie Hai 
3914ccc8d77SZyta Szpak 	/* start tx queues */
3924ccc8d77SZyta Szpak 	for (i = 0; i < dev->data->nb_tx_queues; i++)
3934ccc8d77SZyta Szpak 		dev->data->tx_queue_state[i] = RTE_ETH_QUEUE_STATE_STARTED;
3944ccc8d77SZyta Szpak 
395ce7ea764SZyta Szpak 	mvneta_set_tx_function(dev);
396ce7ea764SZyta Szpak 
3974ccc8d77SZyta Szpak 	return 0;
3984ccc8d77SZyta Szpak 
3994ccc8d77SZyta Szpak out:
4004ccc8d77SZyta Szpak 	MVNETA_LOG(ERR, "Failed to start device");
4014ccc8d77SZyta Szpak 	neta_ppio_deinit(priv->ppio);
4024ccc8d77SZyta Szpak 	return ret;
4034ccc8d77SZyta Szpak }
4044ccc8d77SZyta Szpak 
4054ccc8d77SZyta Szpak /**
4064ccc8d77SZyta Szpak  * DPDK callback to stop the device.
4074ccc8d77SZyta Szpak  *
4084ccc8d77SZyta Szpak  * @param dev
4094ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
4104ccc8d77SZyta Szpak  */
41162024eb8SIvan Ilchenko static int
4124ccc8d77SZyta Szpak mvneta_dev_stop(struct rte_eth_dev *dev)
4134ccc8d77SZyta Szpak {
4144ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
41511f489baSJie Hai 	uint16_t i;
4164ccc8d77SZyta Szpak 
417b8f5d2aeSThomas Monjalon 	dev->data->dev_started = 0;
418b8f5d2aeSThomas Monjalon 
4194ccc8d77SZyta Szpak 	if (!priv->ppio)
42062024eb8SIvan Ilchenko 		return 0;
4214ccc8d77SZyta Szpak 
4224ccc8d77SZyta Szpak 	mvneta_dev_set_link_down(dev);
423ce7ea764SZyta Szpak 	mvneta_flush_queues(dev);
4244ccc8d77SZyta Szpak 	neta_ppio_deinit(priv->ppio);
4254ccc8d77SZyta Szpak 
4264ccc8d77SZyta Szpak 	priv->ppio = NULL;
42762024eb8SIvan Ilchenko 
42811f489baSJie Hai 	/* stop rx queues */
42911f489baSJie Hai 	for (i = 0; i < dev->data->nb_rx_queues; i++)
43011f489baSJie Hai 		dev->data->rx_queue_state[i] = RTE_ETH_QUEUE_STATE_STOPPED;
43111f489baSJie Hai 
43211f489baSJie Hai 	/* stop tx queues */
43311f489baSJie Hai 	for (i = 0; i < dev->data->nb_tx_queues; i++)
43411f489baSJie Hai 		dev->data->tx_queue_state[i] = RTE_ETH_QUEUE_STATE_STOPPED;
43511f489baSJie Hai 
43662024eb8SIvan Ilchenko 	return 0;
4374ccc8d77SZyta Szpak }
4384ccc8d77SZyta Szpak 
4394ccc8d77SZyta Szpak /**
4404ccc8d77SZyta Szpak  * DPDK callback to close the device.
4414ccc8d77SZyta Szpak  *
4424ccc8d77SZyta Szpak  * @param dev
4434ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
4444ccc8d77SZyta Szpak  */
445b142387bSThomas Monjalon static int
4464ccc8d77SZyta Szpak mvneta_dev_close(struct rte_eth_dev *dev)
4474ccc8d77SZyta Szpak {
4484ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
44962024eb8SIvan Ilchenko 	int i, ret = 0;
4504ccc8d77SZyta Szpak 
45130410493SThomas Monjalon 	if (rte_eal_process_type() != RTE_PROC_PRIMARY)
45230410493SThomas Monjalon 		return 0;
45330410493SThomas Monjalon 
4544ccc8d77SZyta Szpak 	if (priv->ppio)
45562024eb8SIvan Ilchenko 		ret = mvneta_dev_stop(dev);
456ce7ea764SZyta Szpak 
457ce7ea764SZyta Szpak 	for (i = 0; i < dev->data->nb_rx_queues; i++) {
4587483341aSXueming Li 		mvneta_rx_queue_release(dev, i);
459ce7ea764SZyta Szpak 		dev->data->rx_queues[i] = NULL;
460ce7ea764SZyta Szpak 	}
461ce7ea764SZyta Szpak 
462ce7ea764SZyta Szpak 	for (i = 0; i < dev->data->nb_tx_queues; i++) {
4637483341aSXueming Li 		mvneta_tx_queue_release(dev, i);
464ce7ea764SZyta Szpak 		dev->data->tx_queues[i] = NULL;
465ce7ea764SZyta Szpak 	}
4664b6d2743SYuri Chipchev 
4674b6d2743SYuri Chipchev 	mvneta_dev_num--;
4684b6d2743SYuri Chipchev 
4694b6d2743SYuri Chipchev 	if (mvneta_dev_num == 0) {
4704b6d2743SYuri Chipchev 		MVNETA_LOG(INFO, "Perform MUSDK deinit");
4714b6d2743SYuri Chipchev 		mvneta_neta_deinit();
4724b6d2743SYuri Chipchev 		rte_mvep_deinit(MVEP_MOD_T_NETA);
4734b6d2743SYuri Chipchev 	}
474b142387bSThomas Monjalon 
47562024eb8SIvan Ilchenko 	return ret;
4764ccc8d77SZyta Szpak }
4774ccc8d77SZyta Szpak 
4784ccc8d77SZyta Szpak /**
47988d3cb3cSZyta Szpak  * DPDK callback to retrieve physical link information.
48088d3cb3cSZyta Szpak  *
48188d3cb3cSZyta Szpak  * @param dev
48288d3cb3cSZyta Szpak  *   Pointer to Ethernet device structure.
48388d3cb3cSZyta Szpak  * @param wait_to_complete
48488d3cb3cSZyta Szpak  *   Wait for request completion (ignored).
48588d3cb3cSZyta Szpak  *
48688d3cb3cSZyta Szpak  * @return
48788d3cb3cSZyta Szpak  *   0 on success, negative error value otherwise.
48888d3cb3cSZyta Szpak  */
48988d3cb3cSZyta Szpak static int
49088d3cb3cSZyta Szpak mvneta_link_update(struct rte_eth_dev *dev, int wait_to_complete __rte_unused)
49188d3cb3cSZyta Szpak {
49288d3cb3cSZyta Szpak 	/*
49388d3cb3cSZyta Szpak 	 * TODO
49488d3cb3cSZyta Szpak 	 * once MUSDK provides necessary API use it here
49588d3cb3cSZyta Szpak 	 */
49688d3cb3cSZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
49788d3cb3cSZyta Szpak 	struct ethtool_cmd edata;
49888d3cb3cSZyta Szpak 	struct ifreq req;
49988d3cb3cSZyta Szpak 	int ret, fd, link_up;
50088d3cb3cSZyta Szpak 
50188d3cb3cSZyta Szpak 	if (!priv->ppio)
50288d3cb3cSZyta Szpak 		return -EPERM;
50388d3cb3cSZyta Szpak 
50488d3cb3cSZyta Szpak 	edata.cmd = ETHTOOL_GSET;
50588d3cb3cSZyta Szpak 
50688d3cb3cSZyta Szpak 	strcpy(req.ifr_name, dev->data->name);
50788d3cb3cSZyta Szpak 	req.ifr_data = (void *)&edata;
50888d3cb3cSZyta Szpak 
50988d3cb3cSZyta Szpak 	fd = socket(AF_INET, SOCK_DGRAM, 0);
51088d3cb3cSZyta Szpak 	if (fd == -1)
51188d3cb3cSZyta Szpak 		return -EFAULT;
51288d3cb3cSZyta Szpak 	ret = ioctl(fd, SIOCETHTOOL, &req);
51388d3cb3cSZyta Szpak 	if (ret == -1) {
51488d3cb3cSZyta Szpak 		close(fd);
51588d3cb3cSZyta Szpak 		return -EFAULT;
51688d3cb3cSZyta Szpak 	}
51788d3cb3cSZyta Szpak 
51888d3cb3cSZyta Szpak 	close(fd);
51988d3cb3cSZyta Szpak 
52088d3cb3cSZyta Szpak 	switch (ethtool_cmd_speed(&edata)) {
52188d3cb3cSZyta Szpak 	case SPEED_10:
522295968d1SFerruh Yigit 		dev->data->dev_link.link_speed = RTE_ETH_SPEED_NUM_10M;
52388d3cb3cSZyta Szpak 		break;
52488d3cb3cSZyta Szpak 	case SPEED_100:
525295968d1SFerruh Yigit 		dev->data->dev_link.link_speed = RTE_ETH_SPEED_NUM_100M;
52688d3cb3cSZyta Szpak 		break;
52788d3cb3cSZyta Szpak 	case SPEED_1000:
528295968d1SFerruh Yigit 		dev->data->dev_link.link_speed = RTE_ETH_SPEED_NUM_1G;
52988d3cb3cSZyta Szpak 		break;
53088d3cb3cSZyta Szpak 	case SPEED_2500:
531295968d1SFerruh Yigit 		dev->data->dev_link.link_speed = RTE_ETH_SPEED_NUM_2_5G;
53288d3cb3cSZyta Szpak 		break;
53388d3cb3cSZyta Szpak 	default:
534295968d1SFerruh Yigit 		dev->data->dev_link.link_speed = RTE_ETH_SPEED_NUM_NONE;
53588d3cb3cSZyta Szpak 	}
53688d3cb3cSZyta Szpak 
537295968d1SFerruh Yigit 	dev->data->dev_link.link_duplex = edata.duplex ? RTE_ETH_LINK_FULL_DUPLEX :
538295968d1SFerruh Yigit 							 RTE_ETH_LINK_HALF_DUPLEX;
539295968d1SFerruh Yigit 	dev->data->dev_link.link_autoneg = edata.autoneg ? RTE_ETH_LINK_AUTONEG :
540295968d1SFerruh Yigit 							   RTE_ETH_LINK_FIXED;
54188d3cb3cSZyta Szpak 
54288d3cb3cSZyta Szpak 	neta_ppio_get_link_state(priv->ppio, &link_up);
543295968d1SFerruh Yigit 	dev->data->dev_link.link_status = link_up ? RTE_ETH_LINK_UP : RTE_ETH_LINK_DOWN;
54488d3cb3cSZyta Szpak 
54588d3cb3cSZyta Szpak 	return 0;
54688d3cb3cSZyta Szpak }
54788d3cb3cSZyta Szpak 
54888d3cb3cSZyta Szpak /**
549cf9ccda9SZyta Szpak  * DPDK callback to enable promiscuous mode.
550cf9ccda9SZyta Szpak  *
551cf9ccda9SZyta Szpak  * @param dev
552cf9ccda9SZyta Szpak  *   Pointer to Ethernet device structure.
5539039c812SAndrew Rybchenko  *
5549039c812SAndrew Rybchenko  * @return
5559039c812SAndrew Rybchenko  *   always 0
556cf9ccda9SZyta Szpak  */
5579039c812SAndrew Rybchenko static int
558cf9ccda9SZyta Szpak mvneta_promiscuous_enable(struct rte_eth_dev *dev)
559cf9ccda9SZyta Szpak {
560cf9ccda9SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
561cf9ccda9SZyta Szpak 	int ret, en;
562cf9ccda9SZyta Szpak 
563cf9ccda9SZyta Szpak 	if (!priv->ppio)
5649039c812SAndrew Rybchenko 		return 0;
565cf9ccda9SZyta Szpak 
566cf9ccda9SZyta Szpak 	neta_ppio_get_promisc(priv->ppio, &en);
567cf9ccda9SZyta Szpak 	if (en) {
568cf9ccda9SZyta Szpak 		MVNETA_LOG(INFO, "Promiscuous already enabled");
5699039c812SAndrew Rybchenko 		return 0;
570cf9ccda9SZyta Szpak 	}
571cf9ccda9SZyta Szpak 
572cf9ccda9SZyta Szpak 	ret = neta_ppio_set_promisc(priv->ppio, 1);
573cf9ccda9SZyta Szpak 	if (ret)
574cf9ccda9SZyta Szpak 		MVNETA_LOG(ERR, "Failed to enable promiscuous mode");
5759039c812SAndrew Rybchenko 
5769039c812SAndrew Rybchenko 	return 0;
577cf9ccda9SZyta Szpak }
578cf9ccda9SZyta Szpak 
579cf9ccda9SZyta Szpak /**
580cf9ccda9SZyta Szpak  * DPDK callback to disable allmulticast mode.
581cf9ccda9SZyta Szpak  *
582cf9ccda9SZyta Szpak  * @param dev
583cf9ccda9SZyta Szpak  *   Pointer to Ethernet device structure.
5849039c812SAndrew Rybchenko  *
5859039c812SAndrew Rybchenko  * @return
5869039c812SAndrew Rybchenko  *   always 0
587cf9ccda9SZyta Szpak  */
5889039c812SAndrew Rybchenko static int
589cf9ccda9SZyta Szpak mvneta_promiscuous_disable(struct rte_eth_dev *dev)
590cf9ccda9SZyta Szpak {
591cf9ccda9SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
592cf9ccda9SZyta Szpak 	int ret, en;
593cf9ccda9SZyta Szpak 
594cf9ccda9SZyta Szpak 	if (!priv->ppio)
5959039c812SAndrew Rybchenko 		return 0;
596cf9ccda9SZyta Szpak 
597cf9ccda9SZyta Szpak 	neta_ppio_get_promisc(priv->ppio, &en);
598cf9ccda9SZyta Szpak 	if (!en) {
599cf9ccda9SZyta Szpak 		MVNETA_LOG(INFO, "Promiscuous already disabled");
6009039c812SAndrew Rybchenko 		return 0;
601cf9ccda9SZyta Szpak 	}
602cf9ccda9SZyta Szpak 
603cf9ccda9SZyta Szpak 	ret = neta_ppio_set_promisc(priv->ppio, 0);
604cf9ccda9SZyta Szpak 	if (ret)
605cf9ccda9SZyta Szpak 		MVNETA_LOG(ERR, "Failed to disable promiscuous mode");
6069039c812SAndrew Rybchenko 
6079039c812SAndrew Rybchenko 	return 0;
608cf9ccda9SZyta Szpak }
609cf9ccda9SZyta Szpak 
610cf9ccda9SZyta Szpak /**
611f3118fb0SZyta Szpak  * DPDK callback to remove a MAC address.
612f3118fb0SZyta Szpak  *
613f3118fb0SZyta Szpak  * @param dev
614f3118fb0SZyta Szpak  *   Pointer to Ethernet device structure.
615f3118fb0SZyta Szpak  * @param index
616f3118fb0SZyta Szpak  *   MAC address index.
617f3118fb0SZyta Szpak  */
618f3118fb0SZyta Szpak static void
619f3118fb0SZyta Szpak mvneta_mac_addr_remove(struct rte_eth_dev *dev, uint32_t index)
620f3118fb0SZyta Szpak {
621f3118fb0SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
62235b2d13fSOlivier Matz 	char buf[RTE_ETHER_ADDR_FMT_SIZE];
623f3118fb0SZyta Szpak 	int ret;
624f3118fb0SZyta Szpak 
625f3118fb0SZyta Szpak 	if (!priv->ppio)
626f3118fb0SZyta Szpak 		return;
627f3118fb0SZyta Szpak 
628f3118fb0SZyta Szpak 	ret = neta_ppio_remove_mac_addr(priv->ppio,
629f3118fb0SZyta Szpak 				       dev->data->mac_addrs[index].addr_bytes);
630f3118fb0SZyta Szpak 	if (ret) {
631538da7a1SOlivier Matz 		rte_ether_format_addr(buf, sizeof(buf),
632f3118fb0SZyta Szpak 				  &dev->data->mac_addrs[index]);
633f3118fb0SZyta Szpak 		MVNETA_LOG(ERR, "Failed to remove mac %s", buf);
634f3118fb0SZyta Szpak 	}
635f3118fb0SZyta Szpak }
636f3118fb0SZyta Szpak 
637f3118fb0SZyta Szpak /**
638f3118fb0SZyta Szpak  * DPDK callback to add a MAC address.
639f3118fb0SZyta Szpak  *
640f3118fb0SZyta Szpak  * @param dev
641f3118fb0SZyta Szpak  *   Pointer to Ethernet device structure.
642f3118fb0SZyta Szpak  * @param mac_addr
643f3118fb0SZyta Szpak  *   MAC address to register.
644f3118fb0SZyta Szpak  * @param index
645f3118fb0SZyta Szpak  *   MAC address index.
646f3118fb0SZyta Szpak  * @param vmdq
647f3118fb0SZyta Szpak  *   VMDq pool index to associate address with (unused).
648f3118fb0SZyta Szpak  *
649f3118fb0SZyta Szpak  * @return
650f3118fb0SZyta Szpak  *   0 on success, negative error value otherwise.
651f3118fb0SZyta Szpak  */
652f3118fb0SZyta Szpak static int
6536d13ea8eSOlivier Matz mvneta_mac_addr_add(struct rte_eth_dev *dev, struct rte_ether_addr *mac_addr,
654f3118fb0SZyta Szpak 		  uint32_t index, uint32_t vmdq __rte_unused)
655f3118fb0SZyta Szpak {
656f3118fb0SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
65735b2d13fSOlivier Matz 	char buf[RTE_ETHER_ADDR_FMT_SIZE];
658f3118fb0SZyta Szpak 	int ret;
659f3118fb0SZyta Szpak 
660f3118fb0SZyta Szpak 	if (index == 0)
661f3118fb0SZyta Szpak 		/* For setting index 0, mrvl_mac_addr_set() should be used.*/
662f3118fb0SZyta Szpak 		return -1;
663f3118fb0SZyta Szpak 
664f3118fb0SZyta Szpak 	if (!priv->ppio)
665f3118fb0SZyta Szpak 		return 0;
666f3118fb0SZyta Szpak 
667f3118fb0SZyta Szpak 	ret = neta_ppio_add_mac_addr(priv->ppio, mac_addr->addr_bytes);
668f3118fb0SZyta Szpak 	if (ret) {
669538da7a1SOlivier Matz 		rte_ether_format_addr(buf, sizeof(buf), mac_addr);
670f3118fb0SZyta Szpak 		MVNETA_LOG(ERR, "Failed to add mac %s", buf);
671f3118fb0SZyta Szpak 		return -1;
672f3118fb0SZyta Szpak 	}
673f3118fb0SZyta Szpak 
674f3118fb0SZyta Szpak 	return 0;
675f3118fb0SZyta Szpak }
676f3118fb0SZyta Szpak 
677f3118fb0SZyta Szpak /**
6784ccc8d77SZyta Szpak  * DPDK callback to set the primary MAC address.
6794ccc8d77SZyta Szpak  *
6804ccc8d77SZyta Szpak  * @param dev
6814ccc8d77SZyta Szpak  *   Pointer to Ethernet device structure.
6824ccc8d77SZyta Szpak  * @param mac_addr
6834ccc8d77SZyta Szpak  *   MAC address to register.
6844ccc8d77SZyta Szpak  */
6854ccc8d77SZyta Szpak static int
6866d13ea8eSOlivier Matz mvneta_mac_addr_set(struct rte_eth_dev *dev, struct rte_ether_addr *mac_addr)
6874ccc8d77SZyta Szpak {
6884ccc8d77SZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
6894ccc8d77SZyta Szpak 	int ret;
6904ccc8d77SZyta Szpak 
6914ccc8d77SZyta Szpak 	if (!priv->ppio)
6924ccc8d77SZyta Szpak 		return -EINVAL;
6934ccc8d77SZyta Szpak 
6944ccc8d77SZyta Szpak 	ret = neta_ppio_set_mac_addr(priv->ppio, mac_addr->addr_bytes);
6954ccc8d77SZyta Szpak 	if (ret) {
69635b2d13fSOlivier Matz 		char buf[RTE_ETHER_ADDR_FMT_SIZE];
697538da7a1SOlivier Matz 		rte_ether_format_addr(buf, sizeof(buf), mac_addr);
6984ccc8d77SZyta Szpak 		MVNETA_LOG(ERR, "Failed to set mac to %s", buf);
6994ccc8d77SZyta Szpak 	}
7004ccc8d77SZyta Szpak 	return 0;
7014ccc8d77SZyta Szpak }
7024ccc8d77SZyta Szpak 
703e9d5faffSZyta Szpak /**
704e9d5faffSZyta Szpak  * DPDK callback to get device statistics.
705e9d5faffSZyta Szpak  *
706e9d5faffSZyta Szpak  * @param dev
707e9d5faffSZyta Szpak  *   Pointer to Ethernet device structure.
708e9d5faffSZyta Szpak  * @param stats
709e9d5faffSZyta Szpak  *   Stats structure output buffer.
710e9d5faffSZyta Szpak  *
711e9d5faffSZyta Szpak  * @return
712e9d5faffSZyta Szpak  *   0 on success, negative error value otherwise.
713e9d5faffSZyta Szpak  */
714e9d5faffSZyta Szpak static int
715e9d5faffSZyta Szpak mvneta_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *stats)
716e9d5faffSZyta Szpak {
717e9d5faffSZyta Szpak 	struct mvneta_priv *priv = dev->data->dev_private;
718e9d5faffSZyta Szpak 	struct neta_ppio_statistics ppio_stats;
719e9d5faffSZyta Szpak 	unsigned int ret;
720e9d5faffSZyta Szpak 
721e9d5faffSZyta Szpak 	if (!priv->ppio)
722e9d5faffSZyta Szpak 		return -EPERM;
723e9d5faffSZyta Szpak 
724e9d5faffSZyta Szpak 	ret = neta_ppio_get_statistics(priv->ppio, &ppio_stats);
725e9d5faffSZyta Szpak 	if (unlikely(ret)) {
726e9d5faffSZyta Szpak 		MVNETA_LOG(ERR, "Failed to update port statistics");
727e9d5faffSZyta Szpak 		return ret;
728e9d5faffSZyta Szpak 	}
729e9d5faffSZyta Szpak 
730e9d5faffSZyta Szpak 	stats->ipackets += ppio_stats.rx_packets +
731e9d5faffSZyta Szpak 			ppio_stats.rx_broadcast_packets +
7323378383dSNatalie Samsonov 			ppio_stats.rx_multicast_packets -
7333378383dSNatalie Samsonov 			priv->prev_stats.ipackets;
734e9d5faffSZyta Szpak 	stats->opackets += ppio_stats.tx_packets +
735e9d5faffSZyta Szpak 			ppio_stats.tx_broadcast_packets +
7363378383dSNatalie Samsonov 			ppio_stats.tx_multicast_packets -
7373378383dSNatalie Samsonov 			priv->prev_stats.opackets;
7383378383dSNatalie Samsonov 	stats->ibytes += ppio_stats.rx_bytes - priv->prev_stats.ibytes;
7393378383dSNatalie Samsonov 	stats->obytes += ppio_stats.tx_bytes - priv->prev_stats.obytes;
740e9d5faffSZyta Szpak 	stats->imissed += ppio_stats.rx_discard +
7413378383dSNatalie Samsonov 			  ppio_stats.rx_overrun -
7423378383dSNatalie Samsonov 			  priv->prev_stats.imissed;
74309a6afe7SLiron Himi 	stats->ierrors = ppio_stats.rx_packets_err -
7443378383dSNatalie Samsonov 			priv->prev_stats.ierrors;
7453378383dSNatalie Samsonov 	stats->oerrors = ppio_stats.tx_errors - priv->prev_stats.oerrors;
746e9d5faffSZyta Szpak 
747e9d5faffSZyta Szpak 	return 0;
748e9d5faffSZyta Szpak }
749e9d5faffSZyta Szpak 
7503378383dSNatalie Samsonov /**
7513378383dSNatalie Samsonov  * DPDK callback to clear device statistics.
7523378383dSNatalie Samsonov  *
7533378383dSNatalie Samsonov  * @param dev
7543378383dSNatalie Samsonov  *   Pointer to Ethernet device structure.
7559970a9adSIgor Romanov  *
7569970a9adSIgor Romanov  * @return
7579970a9adSIgor Romanov  *   0 on success, negative error value otherwise.
7583378383dSNatalie Samsonov  */
7599970a9adSIgor Romanov static int
7603378383dSNatalie Samsonov mvneta_stats_reset(struct rte_eth_dev *dev)
7613378383dSNatalie Samsonov {
7623378383dSNatalie Samsonov 	struct mvneta_priv *priv = dev->data->dev_private;
7633378383dSNatalie Samsonov 	unsigned int ret;
7643378383dSNatalie Samsonov 
7653378383dSNatalie Samsonov 	if (!priv->ppio)
7669970a9adSIgor Romanov 		return 0;
7673378383dSNatalie Samsonov 
7683378383dSNatalie Samsonov 	ret = mvneta_stats_get(dev, &priv->prev_stats);
7693378383dSNatalie Samsonov 	if (unlikely(ret))
770cb0205ceSStephen Hemminger 		MVNETA_LOG(ERR, "Failed to reset port statistics");
7719970a9adSIgor Romanov 
7729970a9adSIgor Romanov 	return ret;
7733378383dSNatalie Samsonov }
7743378383dSNatalie Samsonov 
7753378383dSNatalie Samsonov 
7764ccc8d77SZyta Szpak static const struct eth_dev_ops mvneta_ops = {
7774ccc8d77SZyta Szpak 	.dev_configure = mvneta_dev_configure,
7784ccc8d77SZyta Szpak 	.dev_start = mvneta_dev_start,
7794ccc8d77SZyta Szpak 	.dev_stop = mvneta_dev_stop,
7804ccc8d77SZyta Szpak 	.dev_set_link_up = mvneta_dev_set_link_up,
7814ccc8d77SZyta Szpak 	.dev_set_link_down = mvneta_dev_set_link_down,
7824ccc8d77SZyta Szpak 	.dev_close = mvneta_dev_close,
78388d3cb3cSZyta Szpak 	.link_update = mvneta_link_update,
784cf9ccda9SZyta Szpak 	.promiscuous_enable = mvneta_promiscuous_enable,
785cf9ccda9SZyta Szpak 	.promiscuous_disable = mvneta_promiscuous_disable,
786f3118fb0SZyta Szpak 	.mac_addr_remove = mvneta_mac_addr_remove,
787f3118fb0SZyta Szpak 	.mac_addr_add = mvneta_mac_addr_add,
7884ccc8d77SZyta Szpak 	.mac_addr_set = mvneta_mac_addr_set,
789112ef725SZyta Szpak 	.mtu_set = mvneta_mtu_set,
790e9d5faffSZyta Szpak 	.stats_get = mvneta_stats_get,
7913378383dSNatalie Samsonov 	.stats_reset = mvneta_stats_reset,
7924ccc8d77SZyta Szpak 	.dev_infos_get = mvneta_dev_infos_get,
7934ccc8d77SZyta Szpak 	.dev_supported_ptypes_get = mvneta_dev_supported_ptypes_get,
794ce7ea764SZyta Szpak 	.rxq_info_get = mvneta_rxq_info_get,
795ce7ea764SZyta Szpak 	.txq_info_get = mvneta_txq_info_get,
796ce7ea764SZyta Szpak 	.rx_queue_setup = mvneta_rx_queue_setup,
797ce7ea764SZyta Szpak 	.rx_queue_release = mvneta_rx_queue_release,
798ce7ea764SZyta Szpak 	.tx_queue_setup = mvneta_tx_queue_setup,
799ce7ea764SZyta Szpak 	.tx_queue_release = mvneta_tx_queue_release,
8004ccc8d77SZyta Szpak };
8014ccc8d77SZyta Szpak 
8024ccc8d77SZyta Szpak /**
8034ccc8d77SZyta Szpak  * Create device representing Ethernet port.
8044ccc8d77SZyta Szpak  *
8054ccc8d77SZyta Szpak  * @param name
8064ccc8d77SZyta Szpak  *   Pointer to the port's name.
8074ccc8d77SZyta Szpak  *
8084ccc8d77SZyta Szpak  * @return
8094ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
8104ccc8d77SZyta Szpak  */
8114ccc8d77SZyta Szpak static int
8124ccc8d77SZyta Szpak mvneta_eth_dev_create(struct rte_vdev_device *vdev, const char *name)
8134ccc8d77SZyta Szpak {
8144ccc8d77SZyta Szpak 	int ret, fd = socket(AF_INET, SOCK_DGRAM, 0);
8154ccc8d77SZyta Szpak 	struct rte_eth_dev *eth_dev;
8164ccc8d77SZyta Szpak 	struct mvneta_priv *priv;
8174ccc8d77SZyta Szpak 	struct ifreq req;
8184ccc8d77SZyta Szpak 
8194ccc8d77SZyta Szpak 	eth_dev = rte_eth_dev_allocate(name);
8204ccc8d77SZyta Szpak 	if (!eth_dev)
8214ccc8d77SZyta Szpak 		return -ENOMEM;
8224ccc8d77SZyta Szpak 
8234ccc8d77SZyta Szpak 	priv = rte_zmalloc_socket(name, sizeof(*priv), 0, rte_socket_id());
8244ccc8d77SZyta Szpak 	if (!priv) {
8254ccc8d77SZyta Szpak 		ret = -ENOMEM;
826e16adf08SThomas Monjalon 		goto out_free;
8274ccc8d77SZyta Szpak 	}
828e16adf08SThomas Monjalon 	eth_dev->data->dev_private = priv;
8294ccc8d77SZyta Szpak 
8304ccc8d77SZyta Szpak 	eth_dev->data->mac_addrs =
8314ccc8d77SZyta Szpak 		rte_zmalloc("mac_addrs",
83235b2d13fSOlivier Matz 			    RTE_ETHER_ADDR_LEN * MVNETA_MAC_ADDRS_MAX, 0);
8334ccc8d77SZyta Szpak 	if (!eth_dev->data->mac_addrs) {
8344ccc8d77SZyta Szpak 		MVNETA_LOG(ERR, "Failed to allocate space for eth addrs");
8354ccc8d77SZyta Szpak 		ret = -ENOMEM;
836e16adf08SThomas Monjalon 		goto out_free;
8374ccc8d77SZyta Szpak 	}
8384ccc8d77SZyta Szpak 
8394ccc8d77SZyta Szpak 	memset(&req, 0, sizeof(req));
8404ccc8d77SZyta Szpak 	strcpy(req.ifr_name, name);
8414ccc8d77SZyta Szpak 	ret = ioctl(fd, SIOCGIFHWADDR, &req);
8424ccc8d77SZyta Szpak 	if (ret)
843e16adf08SThomas Monjalon 		goto out_free;
8444ccc8d77SZyta Szpak 
8454ccc8d77SZyta Szpak 	memcpy(eth_dev->data->mac_addrs[0].addr_bytes,
84635b2d13fSOlivier Matz 	       req.ifr_addr.sa_data, RTE_ETHER_ADDR_LEN);
8474ccc8d77SZyta Szpak 
8484ccc8d77SZyta Szpak 	eth_dev->device = &vdev->device;
849ce7ea764SZyta Szpak 	eth_dev->rx_pkt_burst = mvneta_rx_pkt_burst;
850ce7ea764SZyta Szpak 	mvneta_set_tx_function(eth_dev);
8514ccc8d77SZyta Szpak 	eth_dev->dev_ops = &mvneta_ops;
8524ccc8d77SZyta Szpak 
8534ccc8d77SZyta Szpak 	rte_eth_dev_probing_finish(eth_dev);
8544ccc8d77SZyta Szpak 	return 0;
855e16adf08SThomas Monjalon out_free:
8564ccc8d77SZyta Szpak 	rte_eth_dev_release_port(eth_dev);
8574ccc8d77SZyta Szpak 
8584ccc8d77SZyta Szpak 	return ret;
8594ccc8d77SZyta Szpak }
8604ccc8d77SZyta Szpak 
8614ccc8d77SZyta Szpak /**
8624ccc8d77SZyta Szpak  * Cleanup previously created device representing Ethernet port.
8634ccc8d77SZyta Szpak  *
8644ccc8d77SZyta Szpak  * @param eth_dev
8654ccc8d77SZyta Szpak  *   Pointer to the corresponding rte_eth_dev structure.
8664ccc8d77SZyta Szpak  */
8674ccc8d77SZyta Szpak static void
8684ccc8d77SZyta Szpak mvneta_eth_dev_destroy(struct rte_eth_dev *eth_dev)
8694ccc8d77SZyta Szpak {
8704ccc8d77SZyta Szpak 	rte_eth_dev_release_port(eth_dev);
8714ccc8d77SZyta Szpak }
8724ccc8d77SZyta Szpak 
8734ccc8d77SZyta Szpak /**
8744ccc8d77SZyta Szpak  * Cleanup previously created device representing Ethernet port.
8754ccc8d77SZyta Szpak  *
8764ccc8d77SZyta Szpak  * @param name
8774ccc8d77SZyta Szpak  *   Pointer to the port name.
8784ccc8d77SZyta Szpak  */
8794ccc8d77SZyta Szpak static void
8804ccc8d77SZyta Szpak mvneta_eth_dev_destroy_name(const char *name)
8814ccc8d77SZyta Szpak {
8824ccc8d77SZyta Szpak 	struct rte_eth_dev *eth_dev;
8834ccc8d77SZyta Szpak 
8844ccc8d77SZyta Szpak 	eth_dev = rte_eth_dev_allocated(name);
8854ccc8d77SZyta Szpak 	if (!eth_dev)
8864ccc8d77SZyta Szpak 		return;
8874ccc8d77SZyta Szpak 
8884ccc8d77SZyta Szpak 	mvneta_eth_dev_destroy(eth_dev);
8894ccc8d77SZyta Szpak }
8904ccc8d77SZyta Szpak 
8914ccc8d77SZyta Szpak /**
8924ccc8d77SZyta Szpak  * DPDK callback to register the virtual device.
8934ccc8d77SZyta Szpak  *
8944ccc8d77SZyta Szpak  * @param vdev
8954ccc8d77SZyta Szpak  *   Pointer to the virtual device.
8964ccc8d77SZyta Szpak  *
8974ccc8d77SZyta Szpak  * @return
8984ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
8994ccc8d77SZyta Szpak  */
9004ccc8d77SZyta Szpak static int
9014ccc8d77SZyta Szpak rte_pmd_mvneta_probe(struct rte_vdev_device *vdev)
9024ccc8d77SZyta Szpak {
9034ccc8d77SZyta Szpak 	struct rte_kvargs *kvlist;
9044ccc8d77SZyta Szpak 	struct mvneta_ifnames ifnames;
9054ccc8d77SZyta Szpak 	int ret = -EINVAL;
9064ccc8d77SZyta Szpak 	uint32_t i, ifnum;
9074ccc8d77SZyta Szpak 	const char *params;
9084ccc8d77SZyta Szpak 
9094ccc8d77SZyta Szpak 	params = rte_vdev_device_args(vdev);
9104ccc8d77SZyta Szpak 	if (!params)
9114ccc8d77SZyta Szpak 		return -EINVAL;
9124ccc8d77SZyta Szpak 
9134ccc8d77SZyta Szpak 	kvlist = rte_kvargs_parse(params, valid_args);
9144ccc8d77SZyta Szpak 	if (!kvlist)
9154ccc8d77SZyta Szpak 		return -EINVAL;
9164ccc8d77SZyta Szpak 
9174ccc8d77SZyta Szpak 	ifnum = rte_kvargs_count(kvlist, MVNETA_IFACE_NAME_ARG);
9184ccc8d77SZyta Szpak 	if (ifnum > RTE_DIM(ifnames.names))
9194ccc8d77SZyta Szpak 		goto out_free_kvlist;
9204ccc8d77SZyta Szpak 
9214ccc8d77SZyta Szpak 	ifnames.idx = 0;
9224ccc8d77SZyta Szpak 	rte_kvargs_process(kvlist, MVNETA_IFACE_NAME_ARG,
9234ccc8d77SZyta Szpak 			   mvneta_ifnames_get, &ifnames);
9244ccc8d77SZyta Szpak 
9254ccc8d77SZyta Szpak 	/*
9264ccc8d77SZyta Szpak 	 * The below system initialization should be done only once,
9274ccc8d77SZyta Szpak 	 * on the first provided configuration file
9284ccc8d77SZyta Szpak 	 */
9294ccc8d77SZyta Szpak 	if (mvneta_dev_num)
9304ccc8d77SZyta Szpak 		goto init_devices;
9314ccc8d77SZyta Szpak 
9324ccc8d77SZyta Szpak 	MVNETA_LOG(INFO, "Perform MUSDK initializations");
9334ccc8d77SZyta Szpak 
9344ccc8d77SZyta Szpak 	ret = rte_mvep_init(MVEP_MOD_T_NETA, kvlist);
9354ccc8d77SZyta Szpak 	if (ret)
9364ccc8d77SZyta Szpak 		goto out_free_kvlist;
9374ccc8d77SZyta Szpak 
9384ccc8d77SZyta Szpak 	ret = mvneta_neta_init();
9394ccc8d77SZyta Szpak 	if (ret) {
9404ccc8d77SZyta Szpak 		MVNETA_LOG(ERR, "Failed to init NETA!");
9414ccc8d77SZyta Szpak 		rte_mvep_deinit(MVEP_MOD_T_NETA);
9424ccc8d77SZyta Szpak 		goto out_free_kvlist;
9434ccc8d77SZyta Szpak 	}
9444ccc8d77SZyta Szpak 
9454ccc8d77SZyta Szpak init_devices:
9464ccc8d77SZyta Szpak 	for (i = 0; i < ifnum; i++) {
9474ccc8d77SZyta Szpak 		MVNETA_LOG(INFO, "Creating %s", ifnames.names[i]);
9484ccc8d77SZyta Szpak 		ret = mvneta_eth_dev_create(vdev, ifnames.names[i]);
9494ccc8d77SZyta Szpak 		if (ret)
9504ccc8d77SZyta Szpak 			goto out_cleanup;
9514b6d2743SYuri Chipchev 
9524b6d2743SYuri Chipchev 		mvneta_dev_num++;
9534ccc8d77SZyta Szpak 	}
9544ccc8d77SZyta Szpak 
9554ccc8d77SZyta Szpak 	rte_kvargs_free(kvlist);
9564ccc8d77SZyta Szpak 
9574ccc8d77SZyta Szpak 	return 0;
9584ccc8d77SZyta Szpak out_cleanup:
9594b6d2743SYuri Chipchev 	rte_pmd_mvneta_remove(vdev);
9604ccc8d77SZyta Szpak 
9614ccc8d77SZyta Szpak out_free_kvlist:
9624ccc8d77SZyta Szpak 	rte_kvargs_free(kvlist);
9634ccc8d77SZyta Szpak 
9644ccc8d77SZyta Szpak 	return ret;
9654ccc8d77SZyta Szpak }
9664ccc8d77SZyta Szpak 
9674ccc8d77SZyta Szpak /**
9684ccc8d77SZyta Szpak  * DPDK callback to remove virtual device.
9694ccc8d77SZyta Szpak  *
9704ccc8d77SZyta Szpak  * @param vdev
9714ccc8d77SZyta Szpak  *   Pointer to the removed virtual device.
9724ccc8d77SZyta Szpak  *
9734ccc8d77SZyta Szpak  * @return
9744ccc8d77SZyta Szpak  *   0 on success, negative error value otherwise.
9754ccc8d77SZyta Szpak  */
9764ccc8d77SZyta Szpak static int
9774ccc8d77SZyta Szpak rte_pmd_mvneta_remove(struct rte_vdev_device *vdev)
9784ccc8d77SZyta Szpak {
9794b6d2743SYuri Chipchev 	uint16_t port_id;
9808a5a0aadSThomas Monjalon 	int ret = 0;
9814ccc8d77SZyta Szpak 
9824b6d2743SYuri Chipchev 	RTE_ETH_FOREACH_DEV(port_id) {
9834b6d2743SYuri Chipchev 		if (rte_eth_devices[port_id].device != &vdev->device)
9844ccc8d77SZyta Szpak 			continue;
9858a5a0aadSThomas Monjalon 		ret |= rte_eth_dev_close(port_id);
9864ccc8d77SZyta Szpak 	}
9874ccc8d77SZyta Szpak 
9888a5a0aadSThomas Monjalon 	return ret == 0 ? 0 : -EIO;
9894ccc8d77SZyta Szpak }
9904ccc8d77SZyta Szpak 
9914ccc8d77SZyta Szpak static struct rte_vdev_driver pmd_mvneta_drv = {
9924ccc8d77SZyta Szpak 	.probe = rte_pmd_mvneta_probe,
9934ccc8d77SZyta Szpak 	.remove = rte_pmd_mvneta_remove,
9944ccc8d77SZyta Szpak };
9954ccc8d77SZyta Szpak 
9964ccc8d77SZyta Szpak RTE_PMD_REGISTER_VDEV(net_mvneta, pmd_mvneta_drv);
9974ccc8d77SZyta Szpak RTE_PMD_REGISTER_PARAM_STRING(net_mvneta, "iface=<ifc>");
998eeded204SDavid Marchand RTE_LOG_REGISTER_DEFAULT(mvneta_logtype, NOTICE);
999