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