18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci/* Huawei HiNIC PCI Express Linux driver 38c2ecf20Sopenharmony_ci * Copyright(c) 2017 Huawei Technologies Co., Ltd 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * This program is free software; you can redistribute it and/or modify it 68c2ecf20Sopenharmony_ci * under the terms and conditions of the GNU General Public License, 78c2ecf20Sopenharmony_ci * version 2, as published by the Free Software Foundation. 88c2ecf20Sopenharmony_ci * 98c2ecf20Sopenharmony_ci * This program is distributed in the hope it will be useful, but WITHOUT 108c2ecf20Sopenharmony_ci * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 118c2ecf20Sopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 128c2ecf20Sopenharmony_ci * for more details. 138c2ecf20Sopenharmony_ci * 148c2ecf20Sopenharmony_ci */ 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci#include <linux/kernel.h> 178c2ecf20Sopenharmony_ci#include <linux/pci.h> 188c2ecf20Sopenharmony_ci#include <linux/device.h> 198c2ecf20Sopenharmony_ci#include <linux/module.h> 208c2ecf20Sopenharmony_ci#include <linux/types.h> 218c2ecf20Sopenharmony_ci#include <linux/errno.h> 228c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 238c2ecf20Sopenharmony_ci#include <linux/etherdevice.h> 248c2ecf20Sopenharmony_ci#include <linux/netdevice.h> 258c2ecf20Sopenharmony_ci#include <linux/if_vlan.h> 268c2ecf20Sopenharmony_ci#include <linux/ethtool.h> 278c2ecf20Sopenharmony_ci#include <linux/vmalloc.h> 288c2ecf20Sopenharmony_ci#include <linux/sfp.h> 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci#include "hinic_hw_qp.h" 318c2ecf20Sopenharmony_ci#include "hinic_hw_dev.h" 328c2ecf20Sopenharmony_ci#include "hinic_port.h" 338c2ecf20Sopenharmony_ci#include "hinic_tx.h" 348c2ecf20Sopenharmony_ci#include "hinic_rx.h" 358c2ecf20Sopenharmony_ci#include "hinic_dev.h" 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_ci#define SET_LINK_STR_MAX_LEN 128 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci#define GET_SUPPORTED_MODE 0 408c2ecf20Sopenharmony_ci#define GET_ADVERTISED_MODE 1 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_ci#define ETHTOOL_ADD_SUPPORTED_SPEED_LINK_MODE(ecmd, mode) \ 438c2ecf20Sopenharmony_ci ((ecmd)->supported |= \ 448c2ecf20Sopenharmony_ci (1UL << hw_to_ethtool_link_mode_table[mode].link_mode_bit)) 458c2ecf20Sopenharmony_ci#define ETHTOOL_ADD_ADVERTISED_SPEED_LINK_MODE(ecmd, mode) \ 468c2ecf20Sopenharmony_ci ((ecmd)->advertising |= \ 478c2ecf20Sopenharmony_ci (1UL << hw_to_ethtool_link_mode_table[mode].link_mode_bit)) 488c2ecf20Sopenharmony_ci#define ETHTOOL_ADD_SUPPORTED_LINK_MODE(ecmd, mode) \ 498c2ecf20Sopenharmony_ci ((ecmd)->supported |= SUPPORTED_##mode) 508c2ecf20Sopenharmony_ci#define ETHTOOL_ADD_ADVERTISED_LINK_MODE(ecmd, mode) \ 518c2ecf20Sopenharmony_ci ((ecmd)->advertising |= ADVERTISED_##mode) 528c2ecf20Sopenharmony_ci 538c2ecf20Sopenharmony_ci#define COALESCE_PENDING_LIMIT_UNIT 8 548c2ecf20Sopenharmony_ci#define COALESCE_TIMER_CFG_UNIT 9 558c2ecf20Sopenharmony_ci#define COALESCE_ALL_QUEUE 0xFFFF 568c2ecf20Sopenharmony_ci#define COALESCE_MAX_PENDING_LIMIT (255 * COALESCE_PENDING_LIMIT_UNIT) 578c2ecf20Sopenharmony_ci#define COALESCE_MAX_TIMER_CFG (255 * COALESCE_TIMER_CFG_UNIT) 588c2ecf20Sopenharmony_ci#define OBJ_STR_MAX_LEN 32 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_cistruct hw2ethtool_link_mode { 618c2ecf20Sopenharmony_ci enum ethtool_link_mode_bit_indices link_mode_bit; 628c2ecf20Sopenharmony_ci u32 speed; 638c2ecf20Sopenharmony_ci enum hinic_link_mode hw_link_mode; 648c2ecf20Sopenharmony_ci}; 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_cistruct cmd_link_settings { 678c2ecf20Sopenharmony_ci u64 supported; 688c2ecf20Sopenharmony_ci u64 advertising; 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ci u32 speed; 718c2ecf20Sopenharmony_ci u8 duplex; 728c2ecf20Sopenharmony_ci u8 port; 738c2ecf20Sopenharmony_ci u8 autoneg; 748c2ecf20Sopenharmony_ci}; 758c2ecf20Sopenharmony_ci 768c2ecf20Sopenharmony_cistatic u32 hw_to_ethtool_speed[LINK_SPEED_LEVELS] = { 778c2ecf20Sopenharmony_ci SPEED_10, SPEED_100, 788c2ecf20Sopenharmony_ci SPEED_1000, SPEED_10000, 798c2ecf20Sopenharmony_ci SPEED_25000, SPEED_40000, 808c2ecf20Sopenharmony_ci SPEED_100000 818c2ecf20Sopenharmony_ci}; 828c2ecf20Sopenharmony_ci 838c2ecf20Sopenharmony_cistatic struct hw2ethtool_link_mode 848c2ecf20Sopenharmony_ci hw_to_ethtool_link_mode_table[HINIC_LINK_MODE_NUMBERS] = { 858c2ecf20Sopenharmony_ci { 868c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_10000baseKR_Full_BIT, 878c2ecf20Sopenharmony_ci .speed = SPEED_10000, 888c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_10GE_BASE_KR, 898c2ecf20Sopenharmony_ci }, 908c2ecf20Sopenharmony_ci { 918c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT, 928c2ecf20Sopenharmony_ci .speed = SPEED_40000, 938c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_40GE_BASE_KR4, 948c2ecf20Sopenharmony_ci }, 958c2ecf20Sopenharmony_ci { 968c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT, 978c2ecf20Sopenharmony_ci .speed = SPEED_40000, 988c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_40GE_BASE_CR4, 998c2ecf20Sopenharmony_ci }, 1008c2ecf20Sopenharmony_ci { 1018c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT, 1028c2ecf20Sopenharmony_ci .speed = SPEED_100000, 1038c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_100GE_BASE_KR4, 1048c2ecf20Sopenharmony_ci }, 1058c2ecf20Sopenharmony_ci { 1068c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT, 1078c2ecf20Sopenharmony_ci .speed = SPEED_100000, 1088c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_100GE_BASE_CR4, 1098c2ecf20Sopenharmony_ci }, 1108c2ecf20Sopenharmony_ci { 1118c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_25000baseKR_Full_BIT, 1128c2ecf20Sopenharmony_ci .speed = SPEED_25000, 1138c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_25GE_BASE_KR_S, 1148c2ecf20Sopenharmony_ci }, 1158c2ecf20Sopenharmony_ci { 1168c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_25000baseCR_Full_BIT, 1178c2ecf20Sopenharmony_ci .speed = SPEED_25000, 1188c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_25GE_BASE_CR_S, 1198c2ecf20Sopenharmony_ci }, 1208c2ecf20Sopenharmony_ci { 1218c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_25000baseKR_Full_BIT, 1228c2ecf20Sopenharmony_ci .speed = SPEED_25000, 1238c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_25GE_BASE_KR, 1248c2ecf20Sopenharmony_ci }, 1258c2ecf20Sopenharmony_ci { 1268c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_25000baseCR_Full_BIT, 1278c2ecf20Sopenharmony_ci .speed = SPEED_25000, 1288c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_25GE_BASE_CR, 1298c2ecf20Sopenharmony_ci }, 1308c2ecf20Sopenharmony_ci { 1318c2ecf20Sopenharmony_ci .link_mode_bit = ETHTOOL_LINK_MODE_1000baseKX_Full_BIT, 1328c2ecf20Sopenharmony_ci .speed = SPEED_1000, 1338c2ecf20Sopenharmony_ci .hw_link_mode = HINIC_GE_BASE_KX, 1348c2ecf20Sopenharmony_ci }, 1358c2ecf20Sopenharmony_ci}; 1368c2ecf20Sopenharmony_ci 1378c2ecf20Sopenharmony_ci#define LP_DEFAULT_TIME 5 /* seconds */ 1388c2ecf20Sopenharmony_ci#define LP_PKT_LEN 1514 1398c2ecf20Sopenharmony_ci 1408c2ecf20Sopenharmony_ci#define PORT_DOWN_ERR_IDX 0 1418c2ecf20Sopenharmony_cienum diag_test_index { 1428c2ecf20Sopenharmony_ci INTERNAL_LP_TEST = 0, 1438c2ecf20Sopenharmony_ci EXTERNAL_LP_TEST = 1, 1448c2ecf20Sopenharmony_ci DIAG_TEST_MAX = 2, 1458c2ecf20Sopenharmony_ci}; 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_cistatic void set_link_speed(struct ethtool_link_ksettings *link_ksettings, 1488c2ecf20Sopenharmony_ci enum hinic_speed speed) 1498c2ecf20Sopenharmony_ci{ 1508c2ecf20Sopenharmony_ci switch (speed) { 1518c2ecf20Sopenharmony_ci case HINIC_SPEED_10MB_LINK: 1528c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_10; 1538c2ecf20Sopenharmony_ci break; 1548c2ecf20Sopenharmony_ci 1558c2ecf20Sopenharmony_ci case HINIC_SPEED_100MB_LINK: 1568c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_100; 1578c2ecf20Sopenharmony_ci break; 1588c2ecf20Sopenharmony_ci 1598c2ecf20Sopenharmony_ci case HINIC_SPEED_1000MB_LINK: 1608c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_1000; 1618c2ecf20Sopenharmony_ci break; 1628c2ecf20Sopenharmony_ci 1638c2ecf20Sopenharmony_ci case HINIC_SPEED_10GB_LINK: 1648c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_10000; 1658c2ecf20Sopenharmony_ci break; 1668c2ecf20Sopenharmony_ci 1678c2ecf20Sopenharmony_ci case HINIC_SPEED_25GB_LINK: 1688c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_25000; 1698c2ecf20Sopenharmony_ci break; 1708c2ecf20Sopenharmony_ci 1718c2ecf20Sopenharmony_ci case HINIC_SPEED_40GB_LINK: 1728c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_40000; 1738c2ecf20Sopenharmony_ci break; 1748c2ecf20Sopenharmony_ci 1758c2ecf20Sopenharmony_ci case HINIC_SPEED_100GB_LINK: 1768c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_100000; 1778c2ecf20Sopenharmony_ci break; 1788c2ecf20Sopenharmony_ci 1798c2ecf20Sopenharmony_ci default: 1808c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_UNKNOWN; 1818c2ecf20Sopenharmony_ci break; 1828c2ecf20Sopenharmony_ci } 1838c2ecf20Sopenharmony_ci} 1848c2ecf20Sopenharmony_ci 1858c2ecf20Sopenharmony_cistatic int hinic_get_link_mode_index(enum hinic_link_mode link_mode) 1868c2ecf20Sopenharmony_ci{ 1878c2ecf20Sopenharmony_ci int i = 0; 1888c2ecf20Sopenharmony_ci 1898c2ecf20Sopenharmony_ci for (i = 0; i < HINIC_LINK_MODE_NUMBERS; i++) { 1908c2ecf20Sopenharmony_ci if (link_mode == hw_to_ethtool_link_mode_table[i].hw_link_mode) 1918c2ecf20Sopenharmony_ci break; 1928c2ecf20Sopenharmony_ci } 1938c2ecf20Sopenharmony_ci 1948c2ecf20Sopenharmony_ci return i; 1958c2ecf20Sopenharmony_ci} 1968c2ecf20Sopenharmony_ci 1978c2ecf20Sopenharmony_cistatic void hinic_add_ethtool_link_mode(struct cmd_link_settings *link_settings, 1988c2ecf20Sopenharmony_ci enum hinic_link_mode hw_link_mode, 1998c2ecf20Sopenharmony_ci u32 name) 2008c2ecf20Sopenharmony_ci{ 2018c2ecf20Sopenharmony_ci enum hinic_link_mode link_mode; 2028c2ecf20Sopenharmony_ci int idx = 0; 2038c2ecf20Sopenharmony_ci 2048c2ecf20Sopenharmony_ci for (link_mode = 0; link_mode < HINIC_LINK_MODE_NUMBERS; link_mode++) { 2058c2ecf20Sopenharmony_ci if (hw_link_mode & ((u32)1 << link_mode)) { 2068c2ecf20Sopenharmony_ci idx = hinic_get_link_mode_index(link_mode); 2078c2ecf20Sopenharmony_ci if (idx >= HINIC_LINK_MODE_NUMBERS) 2088c2ecf20Sopenharmony_ci continue; 2098c2ecf20Sopenharmony_ci 2108c2ecf20Sopenharmony_ci if (name == GET_SUPPORTED_MODE) 2118c2ecf20Sopenharmony_ci ETHTOOL_ADD_SUPPORTED_SPEED_LINK_MODE 2128c2ecf20Sopenharmony_ci (link_settings, idx); 2138c2ecf20Sopenharmony_ci else 2148c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_SPEED_LINK_MODE 2158c2ecf20Sopenharmony_ci (link_settings, idx); 2168c2ecf20Sopenharmony_ci } 2178c2ecf20Sopenharmony_ci } 2188c2ecf20Sopenharmony_ci} 2198c2ecf20Sopenharmony_ci 2208c2ecf20Sopenharmony_cistatic void hinic_link_port_type(struct cmd_link_settings *link_settings, 2218c2ecf20Sopenharmony_ci enum hinic_port_type port_type) 2228c2ecf20Sopenharmony_ci{ 2238c2ecf20Sopenharmony_ci switch (port_type) { 2248c2ecf20Sopenharmony_ci case HINIC_PORT_ELEC: 2258c2ecf20Sopenharmony_ci case HINIC_PORT_TP: 2268c2ecf20Sopenharmony_ci ETHTOOL_ADD_SUPPORTED_LINK_MODE(link_settings, TP); 2278c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(link_settings, TP); 2288c2ecf20Sopenharmony_ci link_settings->port = PORT_TP; 2298c2ecf20Sopenharmony_ci break; 2308c2ecf20Sopenharmony_ci 2318c2ecf20Sopenharmony_ci case HINIC_PORT_AOC: 2328c2ecf20Sopenharmony_ci case HINIC_PORT_FIBRE: 2338c2ecf20Sopenharmony_ci ETHTOOL_ADD_SUPPORTED_LINK_MODE(link_settings, FIBRE); 2348c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(link_settings, FIBRE); 2358c2ecf20Sopenharmony_ci link_settings->port = PORT_FIBRE; 2368c2ecf20Sopenharmony_ci break; 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci case HINIC_PORT_COPPER: 2398c2ecf20Sopenharmony_ci ETHTOOL_ADD_SUPPORTED_LINK_MODE(link_settings, FIBRE); 2408c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(link_settings, FIBRE); 2418c2ecf20Sopenharmony_ci link_settings->port = PORT_DA; 2428c2ecf20Sopenharmony_ci break; 2438c2ecf20Sopenharmony_ci 2448c2ecf20Sopenharmony_ci case HINIC_PORT_BACKPLANE: 2458c2ecf20Sopenharmony_ci ETHTOOL_ADD_SUPPORTED_LINK_MODE(link_settings, Backplane); 2468c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(link_settings, Backplane); 2478c2ecf20Sopenharmony_ci link_settings->port = PORT_NONE; 2488c2ecf20Sopenharmony_ci break; 2498c2ecf20Sopenharmony_ci 2508c2ecf20Sopenharmony_ci default: 2518c2ecf20Sopenharmony_ci link_settings->port = PORT_OTHER; 2528c2ecf20Sopenharmony_ci break; 2538c2ecf20Sopenharmony_ci } 2548c2ecf20Sopenharmony_ci} 2558c2ecf20Sopenharmony_ci 2568c2ecf20Sopenharmony_cistatic int hinic_get_link_ksettings(struct net_device *netdev, 2578c2ecf20Sopenharmony_ci struct ethtool_link_ksettings 2588c2ecf20Sopenharmony_ci *link_ksettings) 2598c2ecf20Sopenharmony_ci{ 2608c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 2618c2ecf20Sopenharmony_ci struct hinic_link_mode_cmd link_mode = { 0 }; 2628c2ecf20Sopenharmony_ci struct hinic_pause_config pause_info = { 0 }; 2638c2ecf20Sopenharmony_ci struct cmd_link_settings settings = { 0 }; 2648c2ecf20Sopenharmony_ci enum hinic_port_link_state link_state; 2658c2ecf20Sopenharmony_ci struct hinic_port_cap port_cap; 2668c2ecf20Sopenharmony_ci int err; 2678c2ecf20Sopenharmony_ci 2688c2ecf20Sopenharmony_ci ethtool_link_ksettings_zero_link_mode(link_ksettings, supported); 2698c2ecf20Sopenharmony_ci ethtool_link_ksettings_zero_link_mode(link_ksettings, advertising); 2708c2ecf20Sopenharmony_ci 2718c2ecf20Sopenharmony_ci link_ksettings->base.speed = SPEED_UNKNOWN; 2728c2ecf20Sopenharmony_ci link_ksettings->base.autoneg = AUTONEG_DISABLE; 2738c2ecf20Sopenharmony_ci link_ksettings->base.duplex = DUPLEX_UNKNOWN; 2748c2ecf20Sopenharmony_ci 2758c2ecf20Sopenharmony_ci err = hinic_port_get_cap(nic_dev, &port_cap); 2768c2ecf20Sopenharmony_ci if (err) 2778c2ecf20Sopenharmony_ci return err; 2788c2ecf20Sopenharmony_ci 2798c2ecf20Sopenharmony_ci hinic_link_port_type(&settings, port_cap.port_type); 2808c2ecf20Sopenharmony_ci link_ksettings->base.port = settings.port; 2818c2ecf20Sopenharmony_ci 2828c2ecf20Sopenharmony_ci err = hinic_port_link_state(nic_dev, &link_state); 2838c2ecf20Sopenharmony_ci if (err) 2848c2ecf20Sopenharmony_ci return err; 2858c2ecf20Sopenharmony_ci 2868c2ecf20Sopenharmony_ci if (link_state == HINIC_LINK_STATE_UP) { 2878c2ecf20Sopenharmony_ci set_link_speed(link_ksettings, port_cap.speed); 2888c2ecf20Sopenharmony_ci link_ksettings->base.duplex = 2898c2ecf20Sopenharmony_ci (port_cap.duplex == HINIC_DUPLEX_FULL) ? 2908c2ecf20Sopenharmony_ci DUPLEX_FULL : DUPLEX_HALF; 2918c2ecf20Sopenharmony_ci } 2928c2ecf20Sopenharmony_ci 2938c2ecf20Sopenharmony_ci if (!!(port_cap.autoneg_cap & HINIC_AUTONEG_SUPPORTED)) 2948c2ecf20Sopenharmony_ci ethtool_link_ksettings_add_link_mode(link_ksettings, 2958c2ecf20Sopenharmony_ci advertising, Autoneg); 2968c2ecf20Sopenharmony_ci 2978c2ecf20Sopenharmony_ci if (port_cap.autoneg_state == HINIC_AUTONEG_ACTIVE) 2988c2ecf20Sopenharmony_ci link_ksettings->base.autoneg = AUTONEG_ENABLE; 2998c2ecf20Sopenharmony_ci 3008c2ecf20Sopenharmony_ci err = hinic_get_link_mode(nic_dev->hwdev, &link_mode); 3018c2ecf20Sopenharmony_ci if (err || link_mode.supported == HINIC_SUPPORTED_UNKNOWN || 3028c2ecf20Sopenharmony_ci link_mode.advertised == HINIC_SUPPORTED_UNKNOWN) 3038c2ecf20Sopenharmony_ci return -EIO; 3048c2ecf20Sopenharmony_ci 3058c2ecf20Sopenharmony_ci hinic_add_ethtool_link_mode(&settings, link_mode.supported, 3068c2ecf20Sopenharmony_ci GET_SUPPORTED_MODE); 3078c2ecf20Sopenharmony_ci hinic_add_ethtool_link_mode(&settings, link_mode.advertised, 3088c2ecf20Sopenharmony_ci GET_ADVERTISED_MODE); 3098c2ecf20Sopenharmony_ci 3108c2ecf20Sopenharmony_ci if (!HINIC_IS_VF(nic_dev->hwdev->hwif)) { 3118c2ecf20Sopenharmony_ci err = hinic_get_hw_pause_info(nic_dev->hwdev, &pause_info); 3128c2ecf20Sopenharmony_ci if (err) 3138c2ecf20Sopenharmony_ci return err; 3148c2ecf20Sopenharmony_ci ETHTOOL_ADD_SUPPORTED_LINK_MODE(&settings, Pause); 3158c2ecf20Sopenharmony_ci if (pause_info.rx_pause && pause_info.tx_pause) { 3168c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(&settings, Pause); 3178c2ecf20Sopenharmony_ci } else if (pause_info.tx_pause) { 3188c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(&settings, Asym_Pause); 3198c2ecf20Sopenharmony_ci } else if (pause_info.rx_pause) { 3208c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(&settings, Pause); 3218c2ecf20Sopenharmony_ci ETHTOOL_ADD_ADVERTISED_LINK_MODE(&settings, Asym_Pause); 3228c2ecf20Sopenharmony_ci } 3238c2ecf20Sopenharmony_ci } 3248c2ecf20Sopenharmony_ci 3258c2ecf20Sopenharmony_ci bitmap_copy(link_ksettings->link_modes.supported, 3268c2ecf20Sopenharmony_ci (unsigned long *)&settings.supported, 3278c2ecf20Sopenharmony_ci __ETHTOOL_LINK_MODE_MASK_NBITS); 3288c2ecf20Sopenharmony_ci bitmap_copy(link_ksettings->link_modes.advertising, 3298c2ecf20Sopenharmony_ci (unsigned long *)&settings.advertising, 3308c2ecf20Sopenharmony_ci __ETHTOOL_LINK_MODE_MASK_NBITS); 3318c2ecf20Sopenharmony_ci 3328c2ecf20Sopenharmony_ci return 0; 3338c2ecf20Sopenharmony_ci} 3348c2ecf20Sopenharmony_ci 3358c2ecf20Sopenharmony_cistatic int hinic_ethtool_to_hw_speed_level(u32 speed) 3368c2ecf20Sopenharmony_ci{ 3378c2ecf20Sopenharmony_ci int i; 3388c2ecf20Sopenharmony_ci 3398c2ecf20Sopenharmony_ci for (i = 0; i < LINK_SPEED_LEVELS; i++) { 3408c2ecf20Sopenharmony_ci if (hw_to_ethtool_speed[i] == speed) 3418c2ecf20Sopenharmony_ci break; 3428c2ecf20Sopenharmony_ci } 3438c2ecf20Sopenharmony_ci 3448c2ecf20Sopenharmony_ci return i; 3458c2ecf20Sopenharmony_ci} 3468c2ecf20Sopenharmony_ci 3478c2ecf20Sopenharmony_cistatic bool hinic_is_support_speed(enum hinic_link_mode supported_link, 3488c2ecf20Sopenharmony_ci u32 speed) 3498c2ecf20Sopenharmony_ci{ 3508c2ecf20Sopenharmony_ci enum hinic_link_mode link_mode; 3518c2ecf20Sopenharmony_ci int idx; 3528c2ecf20Sopenharmony_ci 3538c2ecf20Sopenharmony_ci for (link_mode = 0; link_mode < HINIC_LINK_MODE_NUMBERS; link_mode++) { 3548c2ecf20Sopenharmony_ci if (!(supported_link & ((u32)1 << link_mode))) 3558c2ecf20Sopenharmony_ci continue; 3568c2ecf20Sopenharmony_ci 3578c2ecf20Sopenharmony_ci idx = hinic_get_link_mode_index(link_mode); 3588c2ecf20Sopenharmony_ci if (idx >= HINIC_LINK_MODE_NUMBERS) 3598c2ecf20Sopenharmony_ci continue; 3608c2ecf20Sopenharmony_ci 3618c2ecf20Sopenharmony_ci if (hw_to_ethtool_link_mode_table[idx].speed == speed) 3628c2ecf20Sopenharmony_ci return true; 3638c2ecf20Sopenharmony_ci } 3648c2ecf20Sopenharmony_ci 3658c2ecf20Sopenharmony_ci return false; 3668c2ecf20Sopenharmony_ci} 3678c2ecf20Sopenharmony_ci 3688c2ecf20Sopenharmony_cistatic bool hinic_is_speed_legal(struct hinic_dev *nic_dev, u32 speed) 3698c2ecf20Sopenharmony_ci{ 3708c2ecf20Sopenharmony_ci struct hinic_link_mode_cmd link_mode = { 0 }; 3718c2ecf20Sopenharmony_ci struct net_device *netdev = nic_dev->netdev; 3728c2ecf20Sopenharmony_ci enum nic_speed_level speed_level = 0; 3738c2ecf20Sopenharmony_ci int err; 3748c2ecf20Sopenharmony_ci 3758c2ecf20Sopenharmony_ci err = hinic_get_link_mode(nic_dev->hwdev, &link_mode); 3768c2ecf20Sopenharmony_ci if (err) 3778c2ecf20Sopenharmony_ci return false; 3788c2ecf20Sopenharmony_ci 3798c2ecf20Sopenharmony_ci if (link_mode.supported == HINIC_SUPPORTED_UNKNOWN || 3808c2ecf20Sopenharmony_ci link_mode.advertised == HINIC_SUPPORTED_UNKNOWN) 3818c2ecf20Sopenharmony_ci return false; 3828c2ecf20Sopenharmony_ci 3838c2ecf20Sopenharmony_ci speed_level = hinic_ethtool_to_hw_speed_level(speed); 3848c2ecf20Sopenharmony_ci if (speed_level >= LINK_SPEED_LEVELS || 3858c2ecf20Sopenharmony_ci !hinic_is_support_speed(link_mode.supported, speed)) { 3868c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 3878c2ecf20Sopenharmony_ci "Unsupported speed: %d\n", speed); 3888c2ecf20Sopenharmony_ci return false; 3898c2ecf20Sopenharmony_ci } 3908c2ecf20Sopenharmony_ci 3918c2ecf20Sopenharmony_ci return true; 3928c2ecf20Sopenharmony_ci} 3938c2ecf20Sopenharmony_ci 3948c2ecf20Sopenharmony_cistatic int get_link_settings_type(struct hinic_dev *nic_dev, 3958c2ecf20Sopenharmony_ci u8 autoneg, u32 speed, u32 *set_settings) 3968c2ecf20Sopenharmony_ci{ 3978c2ecf20Sopenharmony_ci struct hinic_port_cap port_cap = { 0 }; 3988c2ecf20Sopenharmony_ci int err; 3998c2ecf20Sopenharmony_ci 4008c2ecf20Sopenharmony_ci err = hinic_port_get_cap(nic_dev, &port_cap); 4018c2ecf20Sopenharmony_ci if (err) 4028c2ecf20Sopenharmony_ci return err; 4038c2ecf20Sopenharmony_ci 4048c2ecf20Sopenharmony_ci /* always set autonegotiation */ 4058c2ecf20Sopenharmony_ci if (port_cap.autoneg_cap) 4068c2ecf20Sopenharmony_ci *set_settings |= HILINK_LINK_SET_AUTONEG; 4078c2ecf20Sopenharmony_ci 4088c2ecf20Sopenharmony_ci if (autoneg == AUTONEG_ENABLE) { 4098c2ecf20Sopenharmony_ci if (!port_cap.autoneg_cap) { 4108c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, "Not support autoneg\n"); 4118c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 4128c2ecf20Sopenharmony_ci } 4138c2ecf20Sopenharmony_ci } else if (speed != (u32)SPEED_UNKNOWN) { 4148c2ecf20Sopenharmony_ci /* set speed only when autoneg is disabled */ 4158c2ecf20Sopenharmony_ci if (!hinic_is_speed_legal(nic_dev, speed)) 4168c2ecf20Sopenharmony_ci return -EINVAL; 4178c2ecf20Sopenharmony_ci *set_settings |= HILINK_LINK_SET_SPEED; 4188c2ecf20Sopenharmony_ci } else { 4198c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, "Need to set speed when autoneg is off\n"); 4208c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 4218c2ecf20Sopenharmony_ci } 4228c2ecf20Sopenharmony_ci 4238c2ecf20Sopenharmony_ci return 0; 4248c2ecf20Sopenharmony_ci} 4258c2ecf20Sopenharmony_ci 4268c2ecf20Sopenharmony_cistatic int set_link_settings_separate_cmd(struct hinic_dev *nic_dev, 4278c2ecf20Sopenharmony_ci u32 set_settings, u8 autoneg, 4288c2ecf20Sopenharmony_ci u32 speed) 4298c2ecf20Sopenharmony_ci{ 4308c2ecf20Sopenharmony_ci enum nic_speed_level speed_level = 0; 4318c2ecf20Sopenharmony_ci int err = 0; 4328c2ecf20Sopenharmony_ci 4338c2ecf20Sopenharmony_ci if (set_settings & HILINK_LINK_SET_AUTONEG) { 4348c2ecf20Sopenharmony_ci err = hinic_set_autoneg(nic_dev->hwdev, 4358c2ecf20Sopenharmony_ci (autoneg == AUTONEG_ENABLE)); 4368c2ecf20Sopenharmony_ci if (err) 4378c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, "%s autoneg failed\n", 4388c2ecf20Sopenharmony_ci (autoneg == AUTONEG_ENABLE) ? 4398c2ecf20Sopenharmony_ci "Enable" : "Disable"); 4408c2ecf20Sopenharmony_ci else 4418c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, nic_dev->netdev, "%s autoneg successfully\n", 4428c2ecf20Sopenharmony_ci (autoneg == AUTONEG_ENABLE) ? 4438c2ecf20Sopenharmony_ci "Enable" : "Disable"); 4448c2ecf20Sopenharmony_ci } 4458c2ecf20Sopenharmony_ci 4468c2ecf20Sopenharmony_ci if (!err && (set_settings & HILINK_LINK_SET_SPEED)) { 4478c2ecf20Sopenharmony_ci speed_level = hinic_ethtool_to_hw_speed_level(speed); 4488c2ecf20Sopenharmony_ci err = hinic_set_speed(nic_dev->hwdev, speed_level); 4498c2ecf20Sopenharmony_ci if (err) 4508c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, "Set speed %d failed\n", 4518c2ecf20Sopenharmony_ci speed); 4528c2ecf20Sopenharmony_ci else 4538c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, nic_dev->netdev, "Set speed %d successfully\n", 4548c2ecf20Sopenharmony_ci speed); 4558c2ecf20Sopenharmony_ci } 4568c2ecf20Sopenharmony_ci 4578c2ecf20Sopenharmony_ci return err; 4588c2ecf20Sopenharmony_ci} 4598c2ecf20Sopenharmony_ci 4608c2ecf20Sopenharmony_cistatic int hinic_set_settings_to_hw(struct hinic_dev *nic_dev, 4618c2ecf20Sopenharmony_ci u32 set_settings, u8 autoneg, u32 speed) 4628c2ecf20Sopenharmony_ci{ 4638c2ecf20Sopenharmony_ci struct hinic_link_ksettings_info settings = {0}; 4648c2ecf20Sopenharmony_ci char set_link_str[SET_LINK_STR_MAX_LEN] = {0}; 4658c2ecf20Sopenharmony_ci struct net_device *netdev = nic_dev->netdev; 4668c2ecf20Sopenharmony_ci enum nic_speed_level speed_level = 0; 4678c2ecf20Sopenharmony_ci int err; 4688c2ecf20Sopenharmony_ci 4698c2ecf20Sopenharmony_ci err = snprintf(set_link_str, SET_LINK_STR_MAX_LEN, "%s", 4708c2ecf20Sopenharmony_ci (set_settings & HILINK_LINK_SET_AUTONEG) ? 4718c2ecf20Sopenharmony_ci (autoneg ? "autong enable " : "autong disable ") : ""); 4728c2ecf20Sopenharmony_ci if (err < 0 || err >= SET_LINK_STR_MAX_LEN) { 4738c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, "Failed to snprintf link state, function return(%d) and dest_len(%d)\n", 4748c2ecf20Sopenharmony_ci err, SET_LINK_STR_MAX_LEN); 4758c2ecf20Sopenharmony_ci return -EFAULT; 4768c2ecf20Sopenharmony_ci } 4778c2ecf20Sopenharmony_ci 4788c2ecf20Sopenharmony_ci if (set_settings & HILINK_LINK_SET_SPEED) { 4798c2ecf20Sopenharmony_ci speed_level = hinic_ethtool_to_hw_speed_level(speed); 4808c2ecf20Sopenharmony_ci err = snprintf(set_link_str, SET_LINK_STR_MAX_LEN, 4818c2ecf20Sopenharmony_ci "%sspeed %d ", set_link_str, speed); 4828c2ecf20Sopenharmony_ci if (err <= 0 || err >= SET_LINK_STR_MAX_LEN) { 4838c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, "Failed to snprintf link speed, function return(%d) and dest_len(%d)\n", 4848c2ecf20Sopenharmony_ci err, SET_LINK_STR_MAX_LEN); 4858c2ecf20Sopenharmony_ci return -EFAULT; 4868c2ecf20Sopenharmony_ci } 4878c2ecf20Sopenharmony_ci } 4888c2ecf20Sopenharmony_ci 4898c2ecf20Sopenharmony_ci settings.func_id = HINIC_HWIF_FUNC_IDX(nic_dev->hwdev->hwif); 4908c2ecf20Sopenharmony_ci settings.valid_bitmap = set_settings; 4918c2ecf20Sopenharmony_ci settings.autoneg = autoneg; 4928c2ecf20Sopenharmony_ci settings.speed = speed_level; 4938c2ecf20Sopenharmony_ci 4948c2ecf20Sopenharmony_ci err = hinic_set_link_settings(nic_dev->hwdev, &settings); 4958c2ecf20Sopenharmony_ci if (err != HINIC_MGMT_CMD_UNSUPPORTED) { 4968c2ecf20Sopenharmony_ci if (err) 4978c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, "Set %s failed\n", 4988c2ecf20Sopenharmony_ci set_link_str); 4998c2ecf20Sopenharmony_ci else 5008c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, netdev, "Set %s successfully\n", 5018c2ecf20Sopenharmony_ci set_link_str); 5028c2ecf20Sopenharmony_ci 5038c2ecf20Sopenharmony_ci return err; 5048c2ecf20Sopenharmony_ci } 5058c2ecf20Sopenharmony_ci 5068c2ecf20Sopenharmony_ci return set_link_settings_separate_cmd(nic_dev, set_settings, autoneg, 5078c2ecf20Sopenharmony_ci speed); 5088c2ecf20Sopenharmony_ci} 5098c2ecf20Sopenharmony_ci 5108c2ecf20Sopenharmony_cistatic int set_link_settings(struct net_device *netdev, u8 autoneg, u32 speed) 5118c2ecf20Sopenharmony_ci{ 5128c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 5138c2ecf20Sopenharmony_ci u32 set_settings = 0; 5148c2ecf20Sopenharmony_ci int err; 5158c2ecf20Sopenharmony_ci 5168c2ecf20Sopenharmony_ci err = get_link_settings_type(nic_dev, autoneg, speed, &set_settings); 5178c2ecf20Sopenharmony_ci if (err) 5188c2ecf20Sopenharmony_ci return err; 5198c2ecf20Sopenharmony_ci 5208c2ecf20Sopenharmony_ci if (set_settings) 5218c2ecf20Sopenharmony_ci err = hinic_set_settings_to_hw(nic_dev, set_settings, 5228c2ecf20Sopenharmony_ci autoneg, speed); 5238c2ecf20Sopenharmony_ci else 5248c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, netdev, "Nothing changed, exit without setting anything\n"); 5258c2ecf20Sopenharmony_ci 5268c2ecf20Sopenharmony_ci return err; 5278c2ecf20Sopenharmony_ci} 5288c2ecf20Sopenharmony_ci 5298c2ecf20Sopenharmony_cistatic int hinic_set_link_ksettings(struct net_device *netdev, const struct 5308c2ecf20Sopenharmony_ci ethtool_link_ksettings *link_settings) 5318c2ecf20Sopenharmony_ci{ 5328c2ecf20Sopenharmony_ci /* only support to set autoneg and speed */ 5338c2ecf20Sopenharmony_ci return set_link_settings(netdev, link_settings->base.autoneg, 5348c2ecf20Sopenharmony_ci link_settings->base.speed); 5358c2ecf20Sopenharmony_ci} 5368c2ecf20Sopenharmony_ci 5378c2ecf20Sopenharmony_cistatic void hinic_get_drvinfo(struct net_device *netdev, 5388c2ecf20Sopenharmony_ci struct ethtool_drvinfo *info) 5398c2ecf20Sopenharmony_ci{ 5408c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 5418c2ecf20Sopenharmony_ci u8 mgmt_ver[HINIC_MGMT_VERSION_MAX_LEN] = {0}; 5428c2ecf20Sopenharmony_ci struct hinic_hwdev *hwdev = nic_dev->hwdev; 5438c2ecf20Sopenharmony_ci struct hinic_hwif *hwif = hwdev->hwif; 5448c2ecf20Sopenharmony_ci int err; 5458c2ecf20Sopenharmony_ci 5468c2ecf20Sopenharmony_ci strlcpy(info->driver, HINIC_DRV_NAME, sizeof(info->driver)); 5478c2ecf20Sopenharmony_ci strlcpy(info->bus_info, pci_name(hwif->pdev), sizeof(info->bus_info)); 5488c2ecf20Sopenharmony_ci 5498c2ecf20Sopenharmony_ci err = hinic_get_mgmt_version(nic_dev, mgmt_ver); 5508c2ecf20Sopenharmony_ci if (err) 5518c2ecf20Sopenharmony_ci return; 5528c2ecf20Sopenharmony_ci 5538c2ecf20Sopenharmony_ci snprintf(info->fw_version, sizeof(info->fw_version), "%s", mgmt_ver); 5548c2ecf20Sopenharmony_ci} 5558c2ecf20Sopenharmony_ci 5568c2ecf20Sopenharmony_cistatic void hinic_get_ringparam(struct net_device *netdev, 5578c2ecf20Sopenharmony_ci struct ethtool_ringparam *ring) 5588c2ecf20Sopenharmony_ci{ 5598c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 5608c2ecf20Sopenharmony_ci 5618c2ecf20Sopenharmony_ci ring->rx_max_pending = HINIC_MAX_QUEUE_DEPTH; 5628c2ecf20Sopenharmony_ci ring->tx_max_pending = HINIC_MAX_QUEUE_DEPTH; 5638c2ecf20Sopenharmony_ci ring->rx_pending = nic_dev->rq_depth; 5648c2ecf20Sopenharmony_ci ring->tx_pending = nic_dev->sq_depth; 5658c2ecf20Sopenharmony_ci} 5668c2ecf20Sopenharmony_ci 5678c2ecf20Sopenharmony_cistatic int check_ringparam_valid(struct hinic_dev *nic_dev, 5688c2ecf20Sopenharmony_ci struct ethtool_ringparam *ring) 5698c2ecf20Sopenharmony_ci{ 5708c2ecf20Sopenharmony_ci if (ring->rx_jumbo_pending || ring->rx_mini_pending) { 5718c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, 5728c2ecf20Sopenharmony_ci "Unsupported rx_jumbo_pending/rx_mini_pending\n"); 5738c2ecf20Sopenharmony_ci return -EINVAL; 5748c2ecf20Sopenharmony_ci } 5758c2ecf20Sopenharmony_ci 5768c2ecf20Sopenharmony_ci if (ring->tx_pending > HINIC_MAX_QUEUE_DEPTH || 5778c2ecf20Sopenharmony_ci ring->tx_pending < HINIC_MIN_QUEUE_DEPTH || 5788c2ecf20Sopenharmony_ci ring->rx_pending > HINIC_MAX_QUEUE_DEPTH || 5798c2ecf20Sopenharmony_ci ring->rx_pending < HINIC_MIN_QUEUE_DEPTH) { 5808c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, 5818c2ecf20Sopenharmony_ci "Queue depth out of range [%d-%d]\n", 5828c2ecf20Sopenharmony_ci HINIC_MIN_QUEUE_DEPTH, HINIC_MAX_QUEUE_DEPTH); 5838c2ecf20Sopenharmony_ci return -EINVAL; 5848c2ecf20Sopenharmony_ci } 5858c2ecf20Sopenharmony_ci 5868c2ecf20Sopenharmony_ci return 0; 5878c2ecf20Sopenharmony_ci} 5888c2ecf20Sopenharmony_ci 5898c2ecf20Sopenharmony_cistatic int hinic_set_ringparam(struct net_device *netdev, 5908c2ecf20Sopenharmony_ci struct ethtool_ringparam *ring) 5918c2ecf20Sopenharmony_ci{ 5928c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 5938c2ecf20Sopenharmony_ci u16 new_sq_depth, new_rq_depth; 5948c2ecf20Sopenharmony_ci int err; 5958c2ecf20Sopenharmony_ci 5968c2ecf20Sopenharmony_ci err = check_ringparam_valid(nic_dev, ring); 5978c2ecf20Sopenharmony_ci if (err) 5988c2ecf20Sopenharmony_ci return err; 5998c2ecf20Sopenharmony_ci 6008c2ecf20Sopenharmony_ci new_sq_depth = (u16)(1U << (u16)ilog2(ring->tx_pending)); 6018c2ecf20Sopenharmony_ci new_rq_depth = (u16)(1U << (u16)ilog2(ring->rx_pending)); 6028c2ecf20Sopenharmony_ci 6038c2ecf20Sopenharmony_ci if (new_sq_depth == nic_dev->sq_depth && 6048c2ecf20Sopenharmony_ci new_rq_depth == nic_dev->rq_depth) 6058c2ecf20Sopenharmony_ci return 0; 6068c2ecf20Sopenharmony_ci 6078c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, netdev, 6088c2ecf20Sopenharmony_ci "Change Tx/Rx ring depth from %d/%d to %d/%d\n", 6098c2ecf20Sopenharmony_ci nic_dev->sq_depth, nic_dev->rq_depth, 6108c2ecf20Sopenharmony_ci new_sq_depth, new_rq_depth); 6118c2ecf20Sopenharmony_ci 6128c2ecf20Sopenharmony_ci nic_dev->sq_depth = new_sq_depth; 6138c2ecf20Sopenharmony_ci nic_dev->rq_depth = new_rq_depth; 6148c2ecf20Sopenharmony_ci 6158c2ecf20Sopenharmony_ci if (netif_running(netdev)) { 6168c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, netdev, "Restarting netdev\n"); 6178c2ecf20Sopenharmony_ci err = hinic_close(netdev); 6188c2ecf20Sopenharmony_ci if (err) { 6198c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 6208c2ecf20Sopenharmony_ci "Failed to close netdev\n"); 6218c2ecf20Sopenharmony_ci return -EFAULT; 6228c2ecf20Sopenharmony_ci } 6238c2ecf20Sopenharmony_ci 6248c2ecf20Sopenharmony_ci err = hinic_open(netdev); 6258c2ecf20Sopenharmony_ci if (err) { 6268c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 6278c2ecf20Sopenharmony_ci "Failed to open netdev\n"); 6288c2ecf20Sopenharmony_ci return -EFAULT; 6298c2ecf20Sopenharmony_ci } 6308c2ecf20Sopenharmony_ci } 6318c2ecf20Sopenharmony_ci 6328c2ecf20Sopenharmony_ci return 0; 6338c2ecf20Sopenharmony_ci} 6348c2ecf20Sopenharmony_ci 6358c2ecf20Sopenharmony_cistatic int __hinic_get_coalesce(struct net_device *netdev, 6368c2ecf20Sopenharmony_ci struct ethtool_coalesce *coal, u16 queue) 6378c2ecf20Sopenharmony_ci{ 6388c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 6398c2ecf20Sopenharmony_ci struct hinic_intr_coal_info *rx_intr_coal_info; 6408c2ecf20Sopenharmony_ci struct hinic_intr_coal_info *tx_intr_coal_info; 6418c2ecf20Sopenharmony_ci 6428c2ecf20Sopenharmony_ci if (queue == COALESCE_ALL_QUEUE) { 6438c2ecf20Sopenharmony_ci /* get tx/rx irq0 as default parameters */ 6448c2ecf20Sopenharmony_ci rx_intr_coal_info = &nic_dev->rx_intr_coalesce[0]; 6458c2ecf20Sopenharmony_ci tx_intr_coal_info = &nic_dev->tx_intr_coalesce[0]; 6468c2ecf20Sopenharmony_ci } else { 6478c2ecf20Sopenharmony_ci if (queue >= nic_dev->num_qps) { 6488c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 6498c2ecf20Sopenharmony_ci "Invalid queue_id: %d\n", queue); 6508c2ecf20Sopenharmony_ci return -EINVAL; 6518c2ecf20Sopenharmony_ci } 6528c2ecf20Sopenharmony_ci rx_intr_coal_info = &nic_dev->rx_intr_coalesce[queue]; 6538c2ecf20Sopenharmony_ci tx_intr_coal_info = &nic_dev->tx_intr_coalesce[queue]; 6548c2ecf20Sopenharmony_ci } 6558c2ecf20Sopenharmony_ci 6568c2ecf20Sopenharmony_ci /* coalesce_timer is in unit of 9us */ 6578c2ecf20Sopenharmony_ci coal->rx_coalesce_usecs = rx_intr_coal_info->coalesce_timer_cfg * 6588c2ecf20Sopenharmony_ci COALESCE_TIMER_CFG_UNIT; 6598c2ecf20Sopenharmony_ci /* coalesced_frames is in unit of 8 */ 6608c2ecf20Sopenharmony_ci coal->rx_max_coalesced_frames = rx_intr_coal_info->pending_limt * 6618c2ecf20Sopenharmony_ci COALESCE_PENDING_LIMIT_UNIT; 6628c2ecf20Sopenharmony_ci coal->tx_coalesce_usecs = tx_intr_coal_info->coalesce_timer_cfg * 6638c2ecf20Sopenharmony_ci COALESCE_TIMER_CFG_UNIT; 6648c2ecf20Sopenharmony_ci coal->tx_max_coalesced_frames = tx_intr_coal_info->pending_limt * 6658c2ecf20Sopenharmony_ci COALESCE_PENDING_LIMIT_UNIT; 6668c2ecf20Sopenharmony_ci 6678c2ecf20Sopenharmony_ci return 0; 6688c2ecf20Sopenharmony_ci} 6698c2ecf20Sopenharmony_ci 6708c2ecf20Sopenharmony_cistatic int is_coalesce_exceed_limit(const struct ethtool_coalesce *coal) 6718c2ecf20Sopenharmony_ci{ 6728c2ecf20Sopenharmony_ci if (coal->rx_coalesce_usecs > COALESCE_MAX_TIMER_CFG || 6738c2ecf20Sopenharmony_ci coal->rx_max_coalesced_frames > COALESCE_MAX_PENDING_LIMIT || 6748c2ecf20Sopenharmony_ci coal->tx_coalesce_usecs > COALESCE_MAX_TIMER_CFG || 6758c2ecf20Sopenharmony_ci coal->tx_max_coalesced_frames > COALESCE_MAX_PENDING_LIMIT) 6768c2ecf20Sopenharmony_ci return -ERANGE; 6778c2ecf20Sopenharmony_ci 6788c2ecf20Sopenharmony_ci return 0; 6798c2ecf20Sopenharmony_ci} 6808c2ecf20Sopenharmony_ci 6818c2ecf20Sopenharmony_cistatic int set_queue_coalesce(struct hinic_dev *nic_dev, u16 q_id, 6828c2ecf20Sopenharmony_ci struct hinic_intr_coal_info *coal, 6838c2ecf20Sopenharmony_ci bool set_rx_coal) 6848c2ecf20Sopenharmony_ci{ 6858c2ecf20Sopenharmony_ci struct hinic_intr_coal_info *intr_coal = NULL; 6868c2ecf20Sopenharmony_ci struct hinic_msix_config interrupt_info = {0}; 6878c2ecf20Sopenharmony_ci struct net_device *netdev = nic_dev->netdev; 6888c2ecf20Sopenharmony_ci u16 msix_idx; 6898c2ecf20Sopenharmony_ci int err; 6908c2ecf20Sopenharmony_ci 6918c2ecf20Sopenharmony_ci intr_coal = set_rx_coal ? &nic_dev->rx_intr_coalesce[q_id] : 6928c2ecf20Sopenharmony_ci &nic_dev->tx_intr_coalesce[q_id]; 6938c2ecf20Sopenharmony_ci 6948c2ecf20Sopenharmony_ci intr_coal->coalesce_timer_cfg = coal->coalesce_timer_cfg; 6958c2ecf20Sopenharmony_ci intr_coal->pending_limt = coal->pending_limt; 6968c2ecf20Sopenharmony_ci 6978c2ecf20Sopenharmony_ci /* netdev not running or qp not in using, 6988c2ecf20Sopenharmony_ci * don't need to set coalesce to hw 6998c2ecf20Sopenharmony_ci */ 7008c2ecf20Sopenharmony_ci if (!(nic_dev->flags & HINIC_INTF_UP) || 7018c2ecf20Sopenharmony_ci q_id >= nic_dev->num_qps) 7028c2ecf20Sopenharmony_ci return 0; 7038c2ecf20Sopenharmony_ci 7048c2ecf20Sopenharmony_ci msix_idx = set_rx_coal ? nic_dev->rxqs[q_id].rq->msix_entry : 7058c2ecf20Sopenharmony_ci nic_dev->txqs[q_id].sq->msix_entry; 7068c2ecf20Sopenharmony_ci interrupt_info.msix_index = msix_idx; 7078c2ecf20Sopenharmony_ci interrupt_info.coalesce_timer_cnt = intr_coal->coalesce_timer_cfg; 7088c2ecf20Sopenharmony_ci interrupt_info.pending_cnt = intr_coal->pending_limt; 7098c2ecf20Sopenharmony_ci interrupt_info.resend_timer_cnt = intr_coal->resend_timer_cfg; 7108c2ecf20Sopenharmony_ci 7118c2ecf20Sopenharmony_ci err = hinic_set_interrupt_cfg(nic_dev->hwdev, &interrupt_info); 7128c2ecf20Sopenharmony_ci if (err) 7138c2ecf20Sopenharmony_ci netif_warn(nic_dev, drv, netdev, 7148c2ecf20Sopenharmony_ci "Failed to set %s queue%d coalesce", 7158c2ecf20Sopenharmony_ci set_rx_coal ? "rx" : "tx", q_id); 7168c2ecf20Sopenharmony_ci 7178c2ecf20Sopenharmony_ci return err; 7188c2ecf20Sopenharmony_ci} 7198c2ecf20Sopenharmony_ci 7208c2ecf20Sopenharmony_cistatic int __set_hw_coal_param(struct hinic_dev *nic_dev, 7218c2ecf20Sopenharmony_ci struct hinic_intr_coal_info *intr_coal, 7228c2ecf20Sopenharmony_ci u16 queue, bool set_rx_coal) 7238c2ecf20Sopenharmony_ci{ 7248c2ecf20Sopenharmony_ci int err; 7258c2ecf20Sopenharmony_ci u16 i; 7268c2ecf20Sopenharmony_ci 7278c2ecf20Sopenharmony_ci if (queue == COALESCE_ALL_QUEUE) { 7288c2ecf20Sopenharmony_ci for (i = 0; i < nic_dev->max_qps; i++) { 7298c2ecf20Sopenharmony_ci err = set_queue_coalesce(nic_dev, i, intr_coal, 7308c2ecf20Sopenharmony_ci set_rx_coal); 7318c2ecf20Sopenharmony_ci if (err) 7328c2ecf20Sopenharmony_ci return err; 7338c2ecf20Sopenharmony_ci } 7348c2ecf20Sopenharmony_ci } else { 7358c2ecf20Sopenharmony_ci if (queue >= nic_dev->num_qps) { 7368c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, nic_dev->netdev, 7378c2ecf20Sopenharmony_ci "Invalid queue_id: %d\n", queue); 7388c2ecf20Sopenharmony_ci return -EINVAL; 7398c2ecf20Sopenharmony_ci } 7408c2ecf20Sopenharmony_ci err = set_queue_coalesce(nic_dev, queue, intr_coal, 7418c2ecf20Sopenharmony_ci set_rx_coal); 7428c2ecf20Sopenharmony_ci if (err) 7438c2ecf20Sopenharmony_ci return err; 7448c2ecf20Sopenharmony_ci } 7458c2ecf20Sopenharmony_ci 7468c2ecf20Sopenharmony_ci return 0; 7478c2ecf20Sopenharmony_ci} 7488c2ecf20Sopenharmony_ci 7498c2ecf20Sopenharmony_cistatic int __hinic_set_coalesce(struct net_device *netdev, 7508c2ecf20Sopenharmony_ci struct ethtool_coalesce *coal, u16 queue) 7518c2ecf20Sopenharmony_ci{ 7528c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 7538c2ecf20Sopenharmony_ci struct hinic_intr_coal_info rx_intr_coal = {0}; 7548c2ecf20Sopenharmony_ci struct hinic_intr_coal_info tx_intr_coal = {0}; 7558c2ecf20Sopenharmony_ci bool set_rx_coal = false; 7568c2ecf20Sopenharmony_ci bool set_tx_coal = false; 7578c2ecf20Sopenharmony_ci int err; 7588c2ecf20Sopenharmony_ci 7598c2ecf20Sopenharmony_ci err = is_coalesce_exceed_limit(coal); 7608c2ecf20Sopenharmony_ci if (err) 7618c2ecf20Sopenharmony_ci return err; 7628c2ecf20Sopenharmony_ci 7638c2ecf20Sopenharmony_ci if (coal->rx_coalesce_usecs || coal->rx_max_coalesced_frames) { 7648c2ecf20Sopenharmony_ci rx_intr_coal.coalesce_timer_cfg = 7658c2ecf20Sopenharmony_ci (u8)(coal->rx_coalesce_usecs / COALESCE_TIMER_CFG_UNIT); 7668c2ecf20Sopenharmony_ci rx_intr_coal.pending_limt = (u8)(coal->rx_max_coalesced_frames / 7678c2ecf20Sopenharmony_ci COALESCE_PENDING_LIMIT_UNIT); 7688c2ecf20Sopenharmony_ci set_rx_coal = true; 7698c2ecf20Sopenharmony_ci } 7708c2ecf20Sopenharmony_ci 7718c2ecf20Sopenharmony_ci if (coal->tx_coalesce_usecs || coal->tx_max_coalesced_frames) { 7728c2ecf20Sopenharmony_ci tx_intr_coal.coalesce_timer_cfg = 7738c2ecf20Sopenharmony_ci (u8)(coal->tx_coalesce_usecs / COALESCE_TIMER_CFG_UNIT); 7748c2ecf20Sopenharmony_ci tx_intr_coal.pending_limt = (u8)(coal->tx_max_coalesced_frames / 7758c2ecf20Sopenharmony_ci COALESCE_PENDING_LIMIT_UNIT); 7768c2ecf20Sopenharmony_ci set_tx_coal = true; 7778c2ecf20Sopenharmony_ci } 7788c2ecf20Sopenharmony_ci 7798c2ecf20Sopenharmony_ci /* setting coalesce timer or pending limit to zero will disable 7808c2ecf20Sopenharmony_ci * coalesce 7818c2ecf20Sopenharmony_ci */ 7828c2ecf20Sopenharmony_ci if (set_rx_coal && (!rx_intr_coal.coalesce_timer_cfg || 7838c2ecf20Sopenharmony_ci !rx_intr_coal.pending_limt)) 7848c2ecf20Sopenharmony_ci netif_warn(nic_dev, drv, netdev, "RX coalesce will be disabled\n"); 7858c2ecf20Sopenharmony_ci if (set_tx_coal && (!tx_intr_coal.coalesce_timer_cfg || 7868c2ecf20Sopenharmony_ci !tx_intr_coal.pending_limt)) 7878c2ecf20Sopenharmony_ci netif_warn(nic_dev, drv, netdev, "TX coalesce will be disabled\n"); 7888c2ecf20Sopenharmony_ci 7898c2ecf20Sopenharmony_ci if (set_rx_coal) { 7908c2ecf20Sopenharmony_ci err = __set_hw_coal_param(nic_dev, &rx_intr_coal, queue, true); 7918c2ecf20Sopenharmony_ci if (err) 7928c2ecf20Sopenharmony_ci return err; 7938c2ecf20Sopenharmony_ci } 7948c2ecf20Sopenharmony_ci if (set_tx_coal) { 7958c2ecf20Sopenharmony_ci err = __set_hw_coal_param(nic_dev, &tx_intr_coal, queue, false); 7968c2ecf20Sopenharmony_ci if (err) 7978c2ecf20Sopenharmony_ci return err; 7988c2ecf20Sopenharmony_ci } 7998c2ecf20Sopenharmony_ci return 0; 8008c2ecf20Sopenharmony_ci} 8018c2ecf20Sopenharmony_ci 8028c2ecf20Sopenharmony_cistatic int hinic_get_coalesce(struct net_device *netdev, 8038c2ecf20Sopenharmony_ci struct ethtool_coalesce *coal) 8048c2ecf20Sopenharmony_ci{ 8058c2ecf20Sopenharmony_ci return __hinic_get_coalesce(netdev, coal, COALESCE_ALL_QUEUE); 8068c2ecf20Sopenharmony_ci} 8078c2ecf20Sopenharmony_ci 8088c2ecf20Sopenharmony_cistatic int hinic_set_coalesce(struct net_device *netdev, 8098c2ecf20Sopenharmony_ci struct ethtool_coalesce *coal) 8108c2ecf20Sopenharmony_ci{ 8118c2ecf20Sopenharmony_ci return __hinic_set_coalesce(netdev, coal, COALESCE_ALL_QUEUE); 8128c2ecf20Sopenharmony_ci} 8138c2ecf20Sopenharmony_ci 8148c2ecf20Sopenharmony_cistatic int hinic_get_per_queue_coalesce(struct net_device *netdev, u32 queue, 8158c2ecf20Sopenharmony_ci struct ethtool_coalesce *coal) 8168c2ecf20Sopenharmony_ci{ 8178c2ecf20Sopenharmony_ci return __hinic_get_coalesce(netdev, coal, queue); 8188c2ecf20Sopenharmony_ci} 8198c2ecf20Sopenharmony_ci 8208c2ecf20Sopenharmony_cistatic int hinic_set_per_queue_coalesce(struct net_device *netdev, u32 queue, 8218c2ecf20Sopenharmony_ci struct ethtool_coalesce *coal) 8228c2ecf20Sopenharmony_ci{ 8238c2ecf20Sopenharmony_ci return __hinic_set_coalesce(netdev, coal, queue); 8248c2ecf20Sopenharmony_ci} 8258c2ecf20Sopenharmony_ci 8268c2ecf20Sopenharmony_cistatic void hinic_get_pauseparam(struct net_device *netdev, 8278c2ecf20Sopenharmony_ci struct ethtool_pauseparam *pause) 8288c2ecf20Sopenharmony_ci{ 8298c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 8308c2ecf20Sopenharmony_ci struct hinic_pause_config pause_info = {0}; 8318c2ecf20Sopenharmony_ci struct hinic_nic_cfg *nic_cfg; 8328c2ecf20Sopenharmony_ci int err; 8338c2ecf20Sopenharmony_ci 8348c2ecf20Sopenharmony_ci nic_cfg = &nic_dev->hwdev->func_to_io.nic_cfg; 8358c2ecf20Sopenharmony_ci 8368c2ecf20Sopenharmony_ci err = hinic_get_hw_pause_info(nic_dev->hwdev, &pause_info); 8378c2ecf20Sopenharmony_ci if (!err) { 8388c2ecf20Sopenharmony_ci pause->autoneg = pause_info.auto_neg; 8398c2ecf20Sopenharmony_ci if (nic_cfg->pause_set || !pause_info.auto_neg) { 8408c2ecf20Sopenharmony_ci pause->rx_pause = nic_cfg->rx_pause; 8418c2ecf20Sopenharmony_ci pause->tx_pause = nic_cfg->tx_pause; 8428c2ecf20Sopenharmony_ci } else { 8438c2ecf20Sopenharmony_ci pause->rx_pause = pause_info.rx_pause; 8448c2ecf20Sopenharmony_ci pause->tx_pause = pause_info.tx_pause; 8458c2ecf20Sopenharmony_ci } 8468c2ecf20Sopenharmony_ci } 8478c2ecf20Sopenharmony_ci} 8488c2ecf20Sopenharmony_ci 8498c2ecf20Sopenharmony_cistatic int hinic_set_pauseparam(struct net_device *netdev, 8508c2ecf20Sopenharmony_ci struct ethtool_pauseparam *pause) 8518c2ecf20Sopenharmony_ci{ 8528c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 8538c2ecf20Sopenharmony_ci struct hinic_pause_config pause_info = {0}; 8548c2ecf20Sopenharmony_ci struct hinic_port_cap port_cap = {0}; 8558c2ecf20Sopenharmony_ci int err; 8568c2ecf20Sopenharmony_ci 8578c2ecf20Sopenharmony_ci err = hinic_port_get_cap(nic_dev, &port_cap); 8588c2ecf20Sopenharmony_ci if (err) 8598c2ecf20Sopenharmony_ci return -EIO; 8608c2ecf20Sopenharmony_ci 8618c2ecf20Sopenharmony_ci if (pause->autoneg != port_cap.autoneg_state) 8628c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 8638c2ecf20Sopenharmony_ci 8648c2ecf20Sopenharmony_ci pause_info.auto_neg = pause->autoneg; 8658c2ecf20Sopenharmony_ci pause_info.rx_pause = pause->rx_pause; 8668c2ecf20Sopenharmony_ci pause_info.tx_pause = pause->tx_pause; 8678c2ecf20Sopenharmony_ci 8688c2ecf20Sopenharmony_ci mutex_lock(&nic_dev->hwdev->func_to_io.nic_cfg.cfg_mutex); 8698c2ecf20Sopenharmony_ci err = hinic_set_hw_pause_info(nic_dev->hwdev, &pause_info); 8708c2ecf20Sopenharmony_ci if (err) { 8718c2ecf20Sopenharmony_ci mutex_unlock(&nic_dev->hwdev->func_to_io.nic_cfg.cfg_mutex); 8728c2ecf20Sopenharmony_ci return err; 8738c2ecf20Sopenharmony_ci } 8748c2ecf20Sopenharmony_ci nic_dev->hwdev->func_to_io.nic_cfg.pause_set = true; 8758c2ecf20Sopenharmony_ci nic_dev->hwdev->func_to_io.nic_cfg.auto_neg = pause->autoneg; 8768c2ecf20Sopenharmony_ci nic_dev->hwdev->func_to_io.nic_cfg.rx_pause = pause->rx_pause; 8778c2ecf20Sopenharmony_ci nic_dev->hwdev->func_to_io.nic_cfg.tx_pause = pause->tx_pause; 8788c2ecf20Sopenharmony_ci mutex_unlock(&nic_dev->hwdev->func_to_io.nic_cfg.cfg_mutex); 8798c2ecf20Sopenharmony_ci 8808c2ecf20Sopenharmony_ci return 0; 8818c2ecf20Sopenharmony_ci} 8828c2ecf20Sopenharmony_ci 8838c2ecf20Sopenharmony_cistatic void hinic_get_channels(struct net_device *netdev, 8848c2ecf20Sopenharmony_ci struct ethtool_channels *channels) 8858c2ecf20Sopenharmony_ci{ 8868c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 8878c2ecf20Sopenharmony_ci struct hinic_hwdev *hwdev = nic_dev->hwdev; 8888c2ecf20Sopenharmony_ci 8898c2ecf20Sopenharmony_ci channels->max_combined = nic_dev->max_qps; 8908c2ecf20Sopenharmony_ci channels->combined_count = hinic_hwdev_num_qps(hwdev); 8918c2ecf20Sopenharmony_ci} 8928c2ecf20Sopenharmony_ci 8938c2ecf20Sopenharmony_cistatic int hinic_set_channels(struct net_device *netdev, 8948c2ecf20Sopenharmony_ci struct ethtool_channels *channels) 8958c2ecf20Sopenharmony_ci{ 8968c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 8978c2ecf20Sopenharmony_ci unsigned int count = channels->combined_count; 8988c2ecf20Sopenharmony_ci int err; 8998c2ecf20Sopenharmony_ci 9008c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, netdev, "Set max combined queue number from %d to %d\n", 9018c2ecf20Sopenharmony_ci hinic_hwdev_num_qps(nic_dev->hwdev), count); 9028c2ecf20Sopenharmony_ci 9038c2ecf20Sopenharmony_ci if (netif_running(netdev)) { 9048c2ecf20Sopenharmony_ci netif_info(nic_dev, drv, netdev, "Restarting netdev\n"); 9058c2ecf20Sopenharmony_ci hinic_close(netdev); 9068c2ecf20Sopenharmony_ci 9078c2ecf20Sopenharmony_ci nic_dev->hwdev->nic_cap.num_qps = count; 9088c2ecf20Sopenharmony_ci 9098c2ecf20Sopenharmony_ci err = hinic_open(netdev); 9108c2ecf20Sopenharmony_ci if (err) { 9118c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 9128c2ecf20Sopenharmony_ci "Failed to open netdev\n"); 9138c2ecf20Sopenharmony_ci return -EFAULT; 9148c2ecf20Sopenharmony_ci } 9158c2ecf20Sopenharmony_ci } else { 9168c2ecf20Sopenharmony_ci nic_dev->hwdev->nic_cap.num_qps = count; 9178c2ecf20Sopenharmony_ci } 9188c2ecf20Sopenharmony_ci 9198c2ecf20Sopenharmony_ci return 0; 9208c2ecf20Sopenharmony_ci} 9218c2ecf20Sopenharmony_ci 9228c2ecf20Sopenharmony_cistatic int hinic_get_rss_hash_opts(struct hinic_dev *nic_dev, 9238c2ecf20Sopenharmony_ci struct ethtool_rxnfc *cmd) 9248c2ecf20Sopenharmony_ci{ 9258c2ecf20Sopenharmony_ci struct hinic_rss_type rss_type = { 0 }; 9268c2ecf20Sopenharmony_ci int err; 9278c2ecf20Sopenharmony_ci 9288c2ecf20Sopenharmony_ci cmd->data = 0; 9298c2ecf20Sopenharmony_ci 9308c2ecf20Sopenharmony_ci if (!(nic_dev->flags & HINIC_RSS_ENABLE)) 9318c2ecf20Sopenharmony_ci return 0; 9328c2ecf20Sopenharmony_ci 9338c2ecf20Sopenharmony_ci err = hinic_get_rss_type(nic_dev, nic_dev->rss_tmpl_idx, 9348c2ecf20Sopenharmony_ci &rss_type); 9358c2ecf20Sopenharmony_ci if (err) 9368c2ecf20Sopenharmony_ci return err; 9378c2ecf20Sopenharmony_ci 9388c2ecf20Sopenharmony_ci cmd->data = RXH_IP_SRC | RXH_IP_DST; 9398c2ecf20Sopenharmony_ci switch (cmd->flow_type) { 9408c2ecf20Sopenharmony_ci case TCP_V4_FLOW: 9418c2ecf20Sopenharmony_ci if (rss_type.tcp_ipv4) 9428c2ecf20Sopenharmony_ci cmd->data |= RXH_L4_B_0_1 | RXH_L4_B_2_3; 9438c2ecf20Sopenharmony_ci break; 9448c2ecf20Sopenharmony_ci case TCP_V6_FLOW: 9458c2ecf20Sopenharmony_ci if (rss_type.tcp_ipv6) 9468c2ecf20Sopenharmony_ci cmd->data |= RXH_L4_B_0_1 | RXH_L4_B_2_3; 9478c2ecf20Sopenharmony_ci break; 9488c2ecf20Sopenharmony_ci case UDP_V4_FLOW: 9498c2ecf20Sopenharmony_ci if (rss_type.udp_ipv4) 9508c2ecf20Sopenharmony_ci cmd->data |= RXH_L4_B_0_1 | RXH_L4_B_2_3; 9518c2ecf20Sopenharmony_ci break; 9528c2ecf20Sopenharmony_ci case UDP_V6_FLOW: 9538c2ecf20Sopenharmony_ci if (rss_type.udp_ipv6) 9548c2ecf20Sopenharmony_ci cmd->data |= RXH_L4_B_0_1 | RXH_L4_B_2_3; 9558c2ecf20Sopenharmony_ci break; 9568c2ecf20Sopenharmony_ci case IPV4_FLOW: 9578c2ecf20Sopenharmony_ci case IPV6_FLOW: 9588c2ecf20Sopenharmony_ci break; 9598c2ecf20Sopenharmony_ci default: 9608c2ecf20Sopenharmony_ci cmd->data = 0; 9618c2ecf20Sopenharmony_ci return -EINVAL; 9628c2ecf20Sopenharmony_ci } 9638c2ecf20Sopenharmony_ci 9648c2ecf20Sopenharmony_ci return 0; 9658c2ecf20Sopenharmony_ci} 9668c2ecf20Sopenharmony_ci 9678c2ecf20Sopenharmony_cistatic int set_l4_rss_hash_ops(struct ethtool_rxnfc *cmd, 9688c2ecf20Sopenharmony_ci struct hinic_rss_type *rss_type) 9698c2ecf20Sopenharmony_ci{ 9708c2ecf20Sopenharmony_ci u8 rss_l4_en = 0; 9718c2ecf20Sopenharmony_ci 9728c2ecf20Sopenharmony_ci switch (cmd->data & (RXH_L4_B_0_1 | RXH_L4_B_2_3)) { 9738c2ecf20Sopenharmony_ci case 0: 9748c2ecf20Sopenharmony_ci rss_l4_en = 0; 9758c2ecf20Sopenharmony_ci break; 9768c2ecf20Sopenharmony_ci case (RXH_L4_B_0_1 | RXH_L4_B_2_3): 9778c2ecf20Sopenharmony_ci rss_l4_en = 1; 9788c2ecf20Sopenharmony_ci break; 9798c2ecf20Sopenharmony_ci default: 9808c2ecf20Sopenharmony_ci return -EINVAL; 9818c2ecf20Sopenharmony_ci } 9828c2ecf20Sopenharmony_ci 9838c2ecf20Sopenharmony_ci switch (cmd->flow_type) { 9848c2ecf20Sopenharmony_ci case TCP_V4_FLOW: 9858c2ecf20Sopenharmony_ci rss_type->tcp_ipv4 = rss_l4_en; 9868c2ecf20Sopenharmony_ci break; 9878c2ecf20Sopenharmony_ci case TCP_V6_FLOW: 9888c2ecf20Sopenharmony_ci rss_type->tcp_ipv6 = rss_l4_en; 9898c2ecf20Sopenharmony_ci break; 9908c2ecf20Sopenharmony_ci case UDP_V4_FLOW: 9918c2ecf20Sopenharmony_ci rss_type->udp_ipv4 = rss_l4_en; 9928c2ecf20Sopenharmony_ci break; 9938c2ecf20Sopenharmony_ci case UDP_V6_FLOW: 9948c2ecf20Sopenharmony_ci rss_type->udp_ipv6 = rss_l4_en; 9958c2ecf20Sopenharmony_ci break; 9968c2ecf20Sopenharmony_ci default: 9978c2ecf20Sopenharmony_ci return -EINVAL; 9988c2ecf20Sopenharmony_ci } 9998c2ecf20Sopenharmony_ci 10008c2ecf20Sopenharmony_ci return 0; 10018c2ecf20Sopenharmony_ci} 10028c2ecf20Sopenharmony_ci 10038c2ecf20Sopenharmony_cistatic int hinic_set_rss_hash_opts(struct hinic_dev *nic_dev, 10048c2ecf20Sopenharmony_ci struct ethtool_rxnfc *cmd) 10058c2ecf20Sopenharmony_ci{ 10068c2ecf20Sopenharmony_ci struct hinic_rss_type *rss_type = &nic_dev->rss_type; 10078c2ecf20Sopenharmony_ci int err; 10088c2ecf20Sopenharmony_ci 10098c2ecf20Sopenharmony_ci if (!(nic_dev->flags & HINIC_RSS_ENABLE)) { 10108c2ecf20Sopenharmony_ci cmd->data = 0; 10118c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 10128c2ecf20Sopenharmony_ci } 10138c2ecf20Sopenharmony_ci 10148c2ecf20Sopenharmony_ci /* RSS does not support anything other than hashing 10158c2ecf20Sopenharmony_ci * to queues on src and dst IPs and ports 10168c2ecf20Sopenharmony_ci */ 10178c2ecf20Sopenharmony_ci if (cmd->data & ~(RXH_IP_SRC | RXH_IP_DST | RXH_L4_B_0_1 | 10188c2ecf20Sopenharmony_ci RXH_L4_B_2_3)) 10198c2ecf20Sopenharmony_ci return -EINVAL; 10208c2ecf20Sopenharmony_ci 10218c2ecf20Sopenharmony_ci /* We need at least the IP SRC and DEST fields for hashing */ 10228c2ecf20Sopenharmony_ci if (!(cmd->data & RXH_IP_SRC) || !(cmd->data & RXH_IP_DST)) 10238c2ecf20Sopenharmony_ci return -EINVAL; 10248c2ecf20Sopenharmony_ci 10258c2ecf20Sopenharmony_ci err = hinic_get_rss_type(nic_dev, 10268c2ecf20Sopenharmony_ci nic_dev->rss_tmpl_idx, rss_type); 10278c2ecf20Sopenharmony_ci if (err) 10288c2ecf20Sopenharmony_ci return -EFAULT; 10298c2ecf20Sopenharmony_ci 10308c2ecf20Sopenharmony_ci switch (cmd->flow_type) { 10318c2ecf20Sopenharmony_ci case TCP_V4_FLOW: 10328c2ecf20Sopenharmony_ci case TCP_V6_FLOW: 10338c2ecf20Sopenharmony_ci case UDP_V4_FLOW: 10348c2ecf20Sopenharmony_ci case UDP_V6_FLOW: 10358c2ecf20Sopenharmony_ci err = set_l4_rss_hash_ops(cmd, rss_type); 10368c2ecf20Sopenharmony_ci if (err) 10378c2ecf20Sopenharmony_ci return err; 10388c2ecf20Sopenharmony_ci break; 10398c2ecf20Sopenharmony_ci case IPV4_FLOW: 10408c2ecf20Sopenharmony_ci rss_type->ipv4 = 1; 10418c2ecf20Sopenharmony_ci break; 10428c2ecf20Sopenharmony_ci case IPV6_FLOW: 10438c2ecf20Sopenharmony_ci rss_type->ipv6 = 1; 10448c2ecf20Sopenharmony_ci break; 10458c2ecf20Sopenharmony_ci default: 10468c2ecf20Sopenharmony_ci return -EINVAL; 10478c2ecf20Sopenharmony_ci } 10488c2ecf20Sopenharmony_ci 10498c2ecf20Sopenharmony_ci err = hinic_set_rss_type(nic_dev, nic_dev->rss_tmpl_idx, 10508c2ecf20Sopenharmony_ci *rss_type); 10518c2ecf20Sopenharmony_ci if (err) 10528c2ecf20Sopenharmony_ci return -EFAULT; 10538c2ecf20Sopenharmony_ci 10548c2ecf20Sopenharmony_ci return 0; 10558c2ecf20Sopenharmony_ci} 10568c2ecf20Sopenharmony_ci 10578c2ecf20Sopenharmony_cistatic int __set_rss_rxfh(struct net_device *netdev, 10588c2ecf20Sopenharmony_ci const u32 *indir, const u8 *key) 10598c2ecf20Sopenharmony_ci{ 10608c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 10618c2ecf20Sopenharmony_ci int err; 10628c2ecf20Sopenharmony_ci 10638c2ecf20Sopenharmony_ci if (indir) { 10648c2ecf20Sopenharmony_ci if (!nic_dev->rss_indir_user) { 10658c2ecf20Sopenharmony_ci nic_dev->rss_indir_user = 10668c2ecf20Sopenharmony_ci kzalloc(sizeof(u32) * HINIC_RSS_INDIR_SIZE, 10678c2ecf20Sopenharmony_ci GFP_KERNEL); 10688c2ecf20Sopenharmony_ci if (!nic_dev->rss_indir_user) 10698c2ecf20Sopenharmony_ci return -ENOMEM; 10708c2ecf20Sopenharmony_ci } 10718c2ecf20Sopenharmony_ci 10728c2ecf20Sopenharmony_ci memcpy(nic_dev->rss_indir_user, indir, 10738c2ecf20Sopenharmony_ci sizeof(u32) * HINIC_RSS_INDIR_SIZE); 10748c2ecf20Sopenharmony_ci 10758c2ecf20Sopenharmony_ci err = hinic_rss_set_indir_tbl(nic_dev, 10768c2ecf20Sopenharmony_ci nic_dev->rss_tmpl_idx, indir); 10778c2ecf20Sopenharmony_ci if (err) 10788c2ecf20Sopenharmony_ci return -EFAULT; 10798c2ecf20Sopenharmony_ci } 10808c2ecf20Sopenharmony_ci 10818c2ecf20Sopenharmony_ci if (key) { 10828c2ecf20Sopenharmony_ci if (!nic_dev->rss_hkey_user) { 10838c2ecf20Sopenharmony_ci nic_dev->rss_hkey_user = 10848c2ecf20Sopenharmony_ci kzalloc(HINIC_RSS_KEY_SIZE * 2, GFP_KERNEL); 10858c2ecf20Sopenharmony_ci 10868c2ecf20Sopenharmony_ci if (!nic_dev->rss_hkey_user) 10878c2ecf20Sopenharmony_ci return -ENOMEM; 10888c2ecf20Sopenharmony_ci } 10898c2ecf20Sopenharmony_ci 10908c2ecf20Sopenharmony_ci memcpy(nic_dev->rss_hkey_user, key, HINIC_RSS_KEY_SIZE); 10918c2ecf20Sopenharmony_ci 10928c2ecf20Sopenharmony_ci err = hinic_rss_set_template_tbl(nic_dev, 10938c2ecf20Sopenharmony_ci nic_dev->rss_tmpl_idx, key); 10948c2ecf20Sopenharmony_ci if (err) 10958c2ecf20Sopenharmony_ci return -EFAULT; 10968c2ecf20Sopenharmony_ci } 10978c2ecf20Sopenharmony_ci 10988c2ecf20Sopenharmony_ci return 0; 10998c2ecf20Sopenharmony_ci} 11008c2ecf20Sopenharmony_ci 11018c2ecf20Sopenharmony_cistatic int hinic_get_rxnfc(struct net_device *netdev, 11028c2ecf20Sopenharmony_ci struct ethtool_rxnfc *cmd, u32 *rule_locs) 11038c2ecf20Sopenharmony_ci{ 11048c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 11058c2ecf20Sopenharmony_ci int err = 0; 11068c2ecf20Sopenharmony_ci 11078c2ecf20Sopenharmony_ci switch (cmd->cmd) { 11088c2ecf20Sopenharmony_ci case ETHTOOL_GRXRINGS: 11098c2ecf20Sopenharmony_ci cmd->data = nic_dev->num_qps; 11108c2ecf20Sopenharmony_ci break; 11118c2ecf20Sopenharmony_ci case ETHTOOL_GRXFH: 11128c2ecf20Sopenharmony_ci err = hinic_get_rss_hash_opts(nic_dev, cmd); 11138c2ecf20Sopenharmony_ci break; 11148c2ecf20Sopenharmony_ci default: 11158c2ecf20Sopenharmony_ci err = -EOPNOTSUPP; 11168c2ecf20Sopenharmony_ci break; 11178c2ecf20Sopenharmony_ci } 11188c2ecf20Sopenharmony_ci 11198c2ecf20Sopenharmony_ci return err; 11208c2ecf20Sopenharmony_ci} 11218c2ecf20Sopenharmony_ci 11228c2ecf20Sopenharmony_cistatic int hinic_set_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *cmd) 11238c2ecf20Sopenharmony_ci{ 11248c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 11258c2ecf20Sopenharmony_ci int err = 0; 11268c2ecf20Sopenharmony_ci 11278c2ecf20Sopenharmony_ci switch (cmd->cmd) { 11288c2ecf20Sopenharmony_ci case ETHTOOL_SRXFH: 11298c2ecf20Sopenharmony_ci err = hinic_set_rss_hash_opts(nic_dev, cmd); 11308c2ecf20Sopenharmony_ci break; 11318c2ecf20Sopenharmony_ci default: 11328c2ecf20Sopenharmony_ci err = -EOPNOTSUPP; 11338c2ecf20Sopenharmony_ci break; 11348c2ecf20Sopenharmony_ci } 11358c2ecf20Sopenharmony_ci 11368c2ecf20Sopenharmony_ci return err; 11378c2ecf20Sopenharmony_ci} 11388c2ecf20Sopenharmony_ci 11398c2ecf20Sopenharmony_cistatic int hinic_get_rxfh(struct net_device *netdev, 11408c2ecf20Sopenharmony_ci u32 *indir, u8 *key, u8 *hfunc) 11418c2ecf20Sopenharmony_ci{ 11428c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 11438c2ecf20Sopenharmony_ci u8 hash_engine_type = 0; 11448c2ecf20Sopenharmony_ci int err = 0; 11458c2ecf20Sopenharmony_ci 11468c2ecf20Sopenharmony_ci if (!(nic_dev->flags & HINIC_RSS_ENABLE)) 11478c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 11488c2ecf20Sopenharmony_ci 11498c2ecf20Sopenharmony_ci if (hfunc) { 11508c2ecf20Sopenharmony_ci err = hinic_rss_get_hash_engine(nic_dev, 11518c2ecf20Sopenharmony_ci nic_dev->rss_tmpl_idx, 11528c2ecf20Sopenharmony_ci &hash_engine_type); 11538c2ecf20Sopenharmony_ci if (err) 11548c2ecf20Sopenharmony_ci return -EFAULT; 11558c2ecf20Sopenharmony_ci 11568c2ecf20Sopenharmony_ci *hfunc = hash_engine_type ? ETH_RSS_HASH_TOP : ETH_RSS_HASH_XOR; 11578c2ecf20Sopenharmony_ci } 11588c2ecf20Sopenharmony_ci 11598c2ecf20Sopenharmony_ci if (indir) { 11608c2ecf20Sopenharmony_ci err = hinic_rss_get_indir_tbl(nic_dev, 11618c2ecf20Sopenharmony_ci nic_dev->rss_tmpl_idx, indir); 11628c2ecf20Sopenharmony_ci if (err) 11638c2ecf20Sopenharmony_ci return -EFAULT; 11648c2ecf20Sopenharmony_ci } 11658c2ecf20Sopenharmony_ci 11668c2ecf20Sopenharmony_ci if (key) 11678c2ecf20Sopenharmony_ci err = hinic_rss_get_template_tbl(nic_dev, 11688c2ecf20Sopenharmony_ci nic_dev->rss_tmpl_idx, key); 11698c2ecf20Sopenharmony_ci 11708c2ecf20Sopenharmony_ci return err; 11718c2ecf20Sopenharmony_ci} 11728c2ecf20Sopenharmony_ci 11738c2ecf20Sopenharmony_cistatic int hinic_set_rxfh(struct net_device *netdev, const u32 *indir, 11748c2ecf20Sopenharmony_ci const u8 *key, const u8 hfunc) 11758c2ecf20Sopenharmony_ci{ 11768c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 11778c2ecf20Sopenharmony_ci int err = 0; 11788c2ecf20Sopenharmony_ci 11798c2ecf20Sopenharmony_ci if (!(nic_dev->flags & HINIC_RSS_ENABLE)) 11808c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 11818c2ecf20Sopenharmony_ci 11828c2ecf20Sopenharmony_ci if (hfunc != ETH_RSS_HASH_NO_CHANGE) { 11838c2ecf20Sopenharmony_ci if (hfunc != ETH_RSS_HASH_TOP && hfunc != ETH_RSS_HASH_XOR) 11848c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 11858c2ecf20Sopenharmony_ci 11868c2ecf20Sopenharmony_ci nic_dev->rss_hash_engine = (hfunc == ETH_RSS_HASH_XOR) ? 11878c2ecf20Sopenharmony_ci HINIC_RSS_HASH_ENGINE_TYPE_XOR : 11888c2ecf20Sopenharmony_ci HINIC_RSS_HASH_ENGINE_TYPE_TOEP; 11898c2ecf20Sopenharmony_ci err = hinic_rss_set_hash_engine 11908c2ecf20Sopenharmony_ci (nic_dev, nic_dev->rss_tmpl_idx, 11918c2ecf20Sopenharmony_ci nic_dev->rss_hash_engine); 11928c2ecf20Sopenharmony_ci if (err) 11938c2ecf20Sopenharmony_ci return -EFAULT; 11948c2ecf20Sopenharmony_ci } 11958c2ecf20Sopenharmony_ci 11968c2ecf20Sopenharmony_ci err = __set_rss_rxfh(netdev, indir, key); 11978c2ecf20Sopenharmony_ci 11988c2ecf20Sopenharmony_ci return err; 11998c2ecf20Sopenharmony_ci} 12008c2ecf20Sopenharmony_ci 12018c2ecf20Sopenharmony_cistatic u32 hinic_get_rxfh_key_size(struct net_device *netdev) 12028c2ecf20Sopenharmony_ci{ 12038c2ecf20Sopenharmony_ci return HINIC_RSS_KEY_SIZE; 12048c2ecf20Sopenharmony_ci} 12058c2ecf20Sopenharmony_ci 12068c2ecf20Sopenharmony_cistatic u32 hinic_get_rxfh_indir_size(struct net_device *netdev) 12078c2ecf20Sopenharmony_ci{ 12088c2ecf20Sopenharmony_ci return HINIC_RSS_INDIR_SIZE; 12098c2ecf20Sopenharmony_ci} 12108c2ecf20Sopenharmony_ci 12118c2ecf20Sopenharmony_ci#define ARRAY_LEN(arr) ((int)((int)sizeof(arr) / (int)sizeof(arr[0]))) 12128c2ecf20Sopenharmony_ci 12138c2ecf20Sopenharmony_ci#define HINIC_FUNC_STAT(_stat_item) { \ 12148c2ecf20Sopenharmony_ci .name = #_stat_item, \ 12158c2ecf20Sopenharmony_ci .size = sizeof_field(struct hinic_vport_stats, _stat_item), \ 12168c2ecf20Sopenharmony_ci .offset = offsetof(struct hinic_vport_stats, _stat_item) \ 12178c2ecf20Sopenharmony_ci} 12188c2ecf20Sopenharmony_ci 12198c2ecf20Sopenharmony_cistatic struct hinic_stats hinic_function_stats[] = { 12208c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_unicast_pkts_vport), 12218c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_unicast_bytes_vport), 12228c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_multicast_pkts_vport), 12238c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_multicast_bytes_vport), 12248c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_broadcast_pkts_vport), 12258c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_broadcast_bytes_vport), 12268c2ecf20Sopenharmony_ci 12278c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_unicast_pkts_vport), 12288c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_unicast_bytes_vport), 12298c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_multicast_pkts_vport), 12308c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_multicast_bytes_vport), 12318c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_broadcast_pkts_vport), 12328c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_broadcast_bytes_vport), 12338c2ecf20Sopenharmony_ci 12348c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_discard_vport), 12358c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_discard_vport), 12368c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(tx_err_vport), 12378c2ecf20Sopenharmony_ci HINIC_FUNC_STAT(rx_err_vport), 12388c2ecf20Sopenharmony_ci}; 12398c2ecf20Sopenharmony_ci 12408c2ecf20Sopenharmony_cistatic char hinic_test_strings[][ETH_GSTRING_LEN] = { 12418c2ecf20Sopenharmony_ci "Internal lb test (on/offline)", 12428c2ecf20Sopenharmony_ci "External lb test (external_lb)", 12438c2ecf20Sopenharmony_ci}; 12448c2ecf20Sopenharmony_ci 12458c2ecf20Sopenharmony_ci#define HINIC_PORT_STAT(_stat_item) { \ 12468c2ecf20Sopenharmony_ci .name = #_stat_item, \ 12478c2ecf20Sopenharmony_ci .size = sizeof_field(struct hinic_phy_port_stats, _stat_item), \ 12488c2ecf20Sopenharmony_ci .offset = offsetof(struct hinic_phy_port_stats, _stat_item) \ 12498c2ecf20Sopenharmony_ci} 12508c2ecf20Sopenharmony_ci 12518c2ecf20Sopenharmony_cistatic struct hinic_stats hinic_port_stats[] = { 12528c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_total_pkt_num), 12538c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_total_oct_num), 12548c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_bad_pkt_num), 12558c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_bad_oct_num), 12568c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_good_pkt_num), 12578c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_good_oct_num), 12588c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_uni_pkt_num), 12598c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_multi_pkt_num), 12608c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_broad_pkt_num), 12618c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_total_pkt_num), 12628c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_total_oct_num), 12638c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_bad_pkt_num), 12648c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_bad_oct_num), 12658c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_good_pkt_num), 12668c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_good_oct_num), 12678c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_uni_pkt_num), 12688c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_multi_pkt_num), 12698c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_broad_pkt_num), 12708c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_fragment_pkt_num), 12718c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_undersize_pkt_num), 12728c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_undermin_pkt_num), 12738c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_64_oct_pkt_num), 12748c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_65_127_oct_pkt_num), 12758c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_128_255_oct_pkt_num), 12768c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_256_511_oct_pkt_num), 12778c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_512_1023_oct_pkt_num), 12788c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_1024_1518_oct_pkt_num), 12798c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_1519_2047_oct_pkt_num), 12808c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_2048_4095_oct_pkt_num), 12818c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_4096_8191_oct_pkt_num), 12828c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_8192_9216_oct_pkt_num), 12838c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_9217_12287_oct_pkt_num), 12848c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_12288_16383_oct_pkt_num), 12858c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_1519_max_good_pkt_num), 12868c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_1519_max_bad_pkt_num), 12878c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_oversize_pkt_num), 12888c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_jabber_pkt_num), 12898c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pause_num), 12908c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pkt_num), 12918c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri0_pkt_num), 12928c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri1_pkt_num), 12938c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri2_pkt_num), 12948c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri3_pkt_num), 12958c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri4_pkt_num), 12968c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri5_pkt_num), 12978c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri6_pkt_num), 12988c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_pfc_pri7_pkt_num), 12998c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_control_pkt_num), 13008c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_sym_err_pkt_num), 13018c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_fcs_err_pkt_num), 13028c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_send_app_good_pkt_num), 13038c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_rx_send_app_bad_pkt_num), 13048c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_fragment_pkt_num), 13058c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_undersize_pkt_num), 13068c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_undermin_pkt_num), 13078c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_64_oct_pkt_num), 13088c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_65_127_oct_pkt_num), 13098c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_128_255_oct_pkt_num), 13108c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_256_511_oct_pkt_num), 13118c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_512_1023_oct_pkt_num), 13128c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_1024_1518_oct_pkt_num), 13138c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_1519_2047_oct_pkt_num), 13148c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_2048_4095_oct_pkt_num), 13158c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_4096_8191_oct_pkt_num), 13168c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_8192_9216_oct_pkt_num), 13178c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_9217_12287_oct_pkt_num), 13188c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_12288_16383_oct_pkt_num), 13198c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_1519_max_good_pkt_num), 13208c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_1519_max_bad_pkt_num), 13218c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_oversize_pkt_num), 13228c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_jabber_pkt_num), 13238c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pause_num), 13248c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pkt_num), 13258c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri0_pkt_num), 13268c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri1_pkt_num), 13278c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri2_pkt_num), 13288c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri3_pkt_num), 13298c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri4_pkt_num), 13308c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri5_pkt_num), 13318c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri6_pkt_num), 13328c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_pfc_pri7_pkt_num), 13338c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_control_pkt_num), 13348c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_err_all_pkt_num), 13358c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_from_app_good_pkt_num), 13368c2ecf20Sopenharmony_ci HINIC_PORT_STAT(mac_tx_from_app_bad_pkt_num), 13378c2ecf20Sopenharmony_ci}; 13388c2ecf20Sopenharmony_ci 13398c2ecf20Sopenharmony_ci#define HINIC_TXQ_STAT(_stat_item) { \ 13408c2ecf20Sopenharmony_ci .name = "txq%d_"#_stat_item, \ 13418c2ecf20Sopenharmony_ci .size = sizeof_field(struct hinic_txq_stats, _stat_item), \ 13428c2ecf20Sopenharmony_ci .offset = offsetof(struct hinic_txq_stats, _stat_item) \ 13438c2ecf20Sopenharmony_ci} 13448c2ecf20Sopenharmony_ci 13458c2ecf20Sopenharmony_cistatic struct hinic_stats hinic_tx_queue_stats[] = { 13468c2ecf20Sopenharmony_ci HINIC_TXQ_STAT(pkts), 13478c2ecf20Sopenharmony_ci HINIC_TXQ_STAT(bytes), 13488c2ecf20Sopenharmony_ci HINIC_TXQ_STAT(tx_busy), 13498c2ecf20Sopenharmony_ci HINIC_TXQ_STAT(tx_wake), 13508c2ecf20Sopenharmony_ci HINIC_TXQ_STAT(tx_dropped), 13518c2ecf20Sopenharmony_ci HINIC_TXQ_STAT(big_frags_pkts), 13528c2ecf20Sopenharmony_ci}; 13538c2ecf20Sopenharmony_ci 13548c2ecf20Sopenharmony_ci#define HINIC_RXQ_STAT(_stat_item) { \ 13558c2ecf20Sopenharmony_ci .name = "rxq%d_"#_stat_item, \ 13568c2ecf20Sopenharmony_ci .size = sizeof_field(struct hinic_rxq_stats, _stat_item), \ 13578c2ecf20Sopenharmony_ci .offset = offsetof(struct hinic_rxq_stats, _stat_item) \ 13588c2ecf20Sopenharmony_ci} 13598c2ecf20Sopenharmony_ci 13608c2ecf20Sopenharmony_cistatic struct hinic_stats hinic_rx_queue_stats[] = { 13618c2ecf20Sopenharmony_ci HINIC_RXQ_STAT(pkts), 13628c2ecf20Sopenharmony_ci HINIC_RXQ_STAT(bytes), 13638c2ecf20Sopenharmony_ci HINIC_RXQ_STAT(errors), 13648c2ecf20Sopenharmony_ci HINIC_RXQ_STAT(csum_errors), 13658c2ecf20Sopenharmony_ci HINIC_RXQ_STAT(other_errors), 13668c2ecf20Sopenharmony_ci}; 13678c2ecf20Sopenharmony_ci 13688c2ecf20Sopenharmony_cistatic void get_drv_queue_stats(struct hinic_dev *nic_dev, u64 *data) 13698c2ecf20Sopenharmony_ci{ 13708c2ecf20Sopenharmony_ci struct hinic_txq_stats txq_stats; 13718c2ecf20Sopenharmony_ci struct hinic_rxq_stats rxq_stats; 13728c2ecf20Sopenharmony_ci u16 i = 0, j = 0, qid = 0; 13738c2ecf20Sopenharmony_ci char *p; 13748c2ecf20Sopenharmony_ci 13758c2ecf20Sopenharmony_ci for (qid = 0; qid < nic_dev->num_qps; qid++) { 13768c2ecf20Sopenharmony_ci if (!nic_dev->txqs) 13778c2ecf20Sopenharmony_ci break; 13788c2ecf20Sopenharmony_ci 13798c2ecf20Sopenharmony_ci hinic_txq_get_stats(&nic_dev->txqs[qid], &txq_stats); 13808c2ecf20Sopenharmony_ci for (j = 0; j < ARRAY_LEN(hinic_tx_queue_stats); j++, i++) { 13818c2ecf20Sopenharmony_ci p = (char *)&txq_stats + 13828c2ecf20Sopenharmony_ci hinic_tx_queue_stats[j].offset; 13838c2ecf20Sopenharmony_ci data[i] = (hinic_tx_queue_stats[j].size == 13848c2ecf20Sopenharmony_ci sizeof(u64)) ? *(u64 *)p : *(u32 *)p; 13858c2ecf20Sopenharmony_ci } 13868c2ecf20Sopenharmony_ci } 13878c2ecf20Sopenharmony_ci 13888c2ecf20Sopenharmony_ci for (qid = 0; qid < nic_dev->num_qps; qid++) { 13898c2ecf20Sopenharmony_ci if (!nic_dev->rxqs) 13908c2ecf20Sopenharmony_ci break; 13918c2ecf20Sopenharmony_ci 13928c2ecf20Sopenharmony_ci hinic_rxq_get_stats(&nic_dev->rxqs[qid], &rxq_stats); 13938c2ecf20Sopenharmony_ci for (j = 0; j < ARRAY_LEN(hinic_rx_queue_stats); j++, i++) { 13948c2ecf20Sopenharmony_ci p = (char *)&rxq_stats + 13958c2ecf20Sopenharmony_ci hinic_rx_queue_stats[j].offset; 13968c2ecf20Sopenharmony_ci data[i] = (hinic_rx_queue_stats[j].size == 13978c2ecf20Sopenharmony_ci sizeof(u64)) ? *(u64 *)p : *(u32 *)p; 13988c2ecf20Sopenharmony_ci } 13998c2ecf20Sopenharmony_ci } 14008c2ecf20Sopenharmony_ci} 14018c2ecf20Sopenharmony_ci 14028c2ecf20Sopenharmony_cistatic void hinic_get_ethtool_stats(struct net_device *netdev, 14038c2ecf20Sopenharmony_ci struct ethtool_stats *stats, u64 *data) 14048c2ecf20Sopenharmony_ci{ 14058c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 14068c2ecf20Sopenharmony_ci struct hinic_vport_stats vport_stats = {0}; 14078c2ecf20Sopenharmony_ci struct hinic_phy_port_stats *port_stats; 14088c2ecf20Sopenharmony_ci u16 i = 0, j = 0; 14098c2ecf20Sopenharmony_ci char *p; 14108c2ecf20Sopenharmony_ci int err; 14118c2ecf20Sopenharmony_ci 14128c2ecf20Sopenharmony_ci err = hinic_get_vport_stats(nic_dev, &vport_stats); 14138c2ecf20Sopenharmony_ci if (err) 14148c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 14158c2ecf20Sopenharmony_ci "Failed to get vport stats from firmware\n"); 14168c2ecf20Sopenharmony_ci 14178c2ecf20Sopenharmony_ci for (j = 0; j < ARRAY_LEN(hinic_function_stats); j++, i++) { 14188c2ecf20Sopenharmony_ci p = (char *)&vport_stats + hinic_function_stats[j].offset; 14198c2ecf20Sopenharmony_ci data[i] = (hinic_function_stats[j].size == 14208c2ecf20Sopenharmony_ci sizeof(u64)) ? *(u64 *)p : *(u32 *)p; 14218c2ecf20Sopenharmony_ci } 14228c2ecf20Sopenharmony_ci 14238c2ecf20Sopenharmony_ci port_stats = kzalloc(sizeof(*port_stats), GFP_KERNEL); 14248c2ecf20Sopenharmony_ci if (!port_stats) { 14258c2ecf20Sopenharmony_ci memset(&data[i], 0, 14268c2ecf20Sopenharmony_ci ARRAY_LEN(hinic_port_stats) * sizeof(*data)); 14278c2ecf20Sopenharmony_ci i += ARRAY_LEN(hinic_port_stats); 14288c2ecf20Sopenharmony_ci goto get_drv_stats; 14298c2ecf20Sopenharmony_ci } 14308c2ecf20Sopenharmony_ci 14318c2ecf20Sopenharmony_ci err = hinic_get_phy_port_stats(nic_dev, port_stats); 14328c2ecf20Sopenharmony_ci if (err) 14338c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 14348c2ecf20Sopenharmony_ci "Failed to get port stats from firmware\n"); 14358c2ecf20Sopenharmony_ci 14368c2ecf20Sopenharmony_ci for (j = 0; j < ARRAY_LEN(hinic_port_stats); j++, i++) { 14378c2ecf20Sopenharmony_ci p = (char *)port_stats + hinic_port_stats[j].offset; 14388c2ecf20Sopenharmony_ci data[i] = (hinic_port_stats[j].size == 14398c2ecf20Sopenharmony_ci sizeof(u64)) ? *(u64 *)p : *(u32 *)p; 14408c2ecf20Sopenharmony_ci } 14418c2ecf20Sopenharmony_ci 14428c2ecf20Sopenharmony_ci kfree(port_stats); 14438c2ecf20Sopenharmony_ci 14448c2ecf20Sopenharmony_ciget_drv_stats: 14458c2ecf20Sopenharmony_ci get_drv_queue_stats(nic_dev, data + i); 14468c2ecf20Sopenharmony_ci} 14478c2ecf20Sopenharmony_ci 14488c2ecf20Sopenharmony_cistatic int hinic_get_sset_count(struct net_device *netdev, int sset) 14498c2ecf20Sopenharmony_ci{ 14508c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 14518c2ecf20Sopenharmony_ci int count, q_num; 14528c2ecf20Sopenharmony_ci 14538c2ecf20Sopenharmony_ci switch (sset) { 14548c2ecf20Sopenharmony_ci case ETH_SS_TEST: 14558c2ecf20Sopenharmony_ci return ARRAY_LEN(hinic_test_strings); 14568c2ecf20Sopenharmony_ci case ETH_SS_STATS: 14578c2ecf20Sopenharmony_ci q_num = nic_dev->num_qps; 14588c2ecf20Sopenharmony_ci count = ARRAY_LEN(hinic_function_stats) + 14598c2ecf20Sopenharmony_ci (ARRAY_LEN(hinic_tx_queue_stats) + 14608c2ecf20Sopenharmony_ci ARRAY_LEN(hinic_rx_queue_stats)) * q_num; 14618c2ecf20Sopenharmony_ci 14628c2ecf20Sopenharmony_ci count += ARRAY_LEN(hinic_port_stats); 14638c2ecf20Sopenharmony_ci 14648c2ecf20Sopenharmony_ci return count; 14658c2ecf20Sopenharmony_ci default: 14668c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 14678c2ecf20Sopenharmony_ci } 14688c2ecf20Sopenharmony_ci} 14698c2ecf20Sopenharmony_ci 14708c2ecf20Sopenharmony_cistatic void hinic_get_strings(struct net_device *netdev, 14718c2ecf20Sopenharmony_ci u32 stringset, u8 *data) 14728c2ecf20Sopenharmony_ci{ 14738c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 14748c2ecf20Sopenharmony_ci char *p = (char *)data; 14758c2ecf20Sopenharmony_ci u16 i, j; 14768c2ecf20Sopenharmony_ci 14778c2ecf20Sopenharmony_ci switch (stringset) { 14788c2ecf20Sopenharmony_ci case ETH_SS_TEST: 14798c2ecf20Sopenharmony_ci memcpy(data, *hinic_test_strings, sizeof(hinic_test_strings)); 14808c2ecf20Sopenharmony_ci return; 14818c2ecf20Sopenharmony_ci case ETH_SS_STATS: 14828c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_LEN(hinic_function_stats); i++) { 14838c2ecf20Sopenharmony_ci memcpy(p, hinic_function_stats[i].name, 14848c2ecf20Sopenharmony_ci ETH_GSTRING_LEN); 14858c2ecf20Sopenharmony_ci p += ETH_GSTRING_LEN; 14868c2ecf20Sopenharmony_ci } 14878c2ecf20Sopenharmony_ci 14888c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_LEN(hinic_port_stats); i++) { 14898c2ecf20Sopenharmony_ci memcpy(p, hinic_port_stats[i].name, 14908c2ecf20Sopenharmony_ci ETH_GSTRING_LEN); 14918c2ecf20Sopenharmony_ci p += ETH_GSTRING_LEN; 14928c2ecf20Sopenharmony_ci } 14938c2ecf20Sopenharmony_ci 14948c2ecf20Sopenharmony_ci for (i = 0; i < nic_dev->num_qps; i++) { 14958c2ecf20Sopenharmony_ci for (j = 0; j < ARRAY_LEN(hinic_tx_queue_stats); j++) { 14968c2ecf20Sopenharmony_ci sprintf(p, hinic_tx_queue_stats[j].name, i); 14978c2ecf20Sopenharmony_ci p += ETH_GSTRING_LEN; 14988c2ecf20Sopenharmony_ci } 14998c2ecf20Sopenharmony_ci } 15008c2ecf20Sopenharmony_ci 15018c2ecf20Sopenharmony_ci for (i = 0; i < nic_dev->num_qps; i++) { 15028c2ecf20Sopenharmony_ci for (j = 0; j < ARRAY_LEN(hinic_rx_queue_stats); j++) { 15038c2ecf20Sopenharmony_ci sprintf(p, hinic_rx_queue_stats[j].name, i); 15048c2ecf20Sopenharmony_ci p += ETH_GSTRING_LEN; 15058c2ecf20Sopenharmony_ci } 15068c2ecf20Sopenharmony_ci } 15078c2ecf20Sopenharmony_ci 15088c2ecf20Sopenharmony_ci return; 15098c2ecf20Sopenharmony_ci default: 15108c2ecf20Sopenharmony_ci return; 15118c2ecf20Sopenharmony_ci } 15128c2ecf20Sopenharmony_ci} 15138c2ecf20Sopenharmony_ci 15148c2ecf20Sopenharmony_cistatic int hinic_run_lp_test(struct hinic_dev *nic_dev, u32 test_time) 15158c2ecf20Sopenharmony_ci{ 15168c2ecf20Sopenharmony_ci u8 *lb_test_rx_buf = nic_dev->lb_test_rx_buf; 15178c2ecf20Sopenharmony_ci struct net_device *netdev = nic_dev->netdev; 15188c2ecf20Sopenharmony_ci struct sk_buff *skb_tmp = NULL; 15198c2ecf20Sopenharmony_ci struct sk_buff *skb = NULL; 15208c2ecf20Sopenharmony_ci u32 cnt = test_time * 5; 15218c2ecf20Sopenharmony_ci u8 *test_data = NULL; 15228c2ecf20Sopenharmony_ci u32 i; 15238c2ecf20Sopenharmony_ci u8 j; 15248c2ecf20Sopenharmony_ci 15258c2ecf20Sopenharmony_ci skb_tmp = alloc_skb(LP_PKT_LEN, GFP_ATOMIC); 15268c2ecf20Sopenharmony_ci if (!skb_tmp) 15278c2ecf20Sopenharmony_ci return -ENOMEM; 15288c2ecf20Sopenharmony_ci 15298c2ecf20Sopenharmony_ci test_data = __skb_put(skb_tmp, LP_PKT_LEN); 15308c2ecf20Sopenharmony_ci 15318c2ecf20Sopenharmony_ci memset(test_data, 0xFF, 2 * ETH_ALEN); 15328c2ecf20Sopenharmony_ci test_data[ETH_ALEN] = 0xFE; 15338c2ecf20Sopenharmony_ci test_data[2 * ETH_ALEN] = 0x08; 15348c2ecf20Sopenharmony_ci test_data[2 * ETH_ALEN + 1] = 0x0; 15358c2ecf20Sopenharmony_ci 15368c2ecf20Sopenharmony_ci for (i = ETH_HLEN; i < LP_PKT_LEN; i++) 15378c2ecf20Sopenharmony_ci test_data[i] = i & 0xFF; 15388c2ecf20Sopenharmony_ci 15398c2ecf20Sopenharmony_ci skb_tmp->queue_mapping = 0; 15408c2ecf20Sopenharmony_ci skb_tmp->ip_summed = CHECKSUM_COMPLETE; 15418c2ecf20Sopenharmony_ci skb_tmp->dev = netdev; 15428c2ecf20Sopenharmony_ci 15438c2ecf20Sopenharmony_ci for (i = 0; i < cnt; i++) { 15448c2ecf20Sopenharmony_ci nic_dev->lb_test_rx_idx = 0; 15458c2ecf20Sopenharmony_ci memset(lb_test_rx_buf, 0, LP_PKT_CNT * LP_PKT_LEN); 15468c2ecf20Sopenharmony_ci 15478c2ecf20Sopenharmony_ci for (j = 0; j < LP_PKT_CNT; j++) { 15488c2ecf20Sopenharmony_ci skb = pskb_copy(skb_tmp, GFP_ATOMIC); 15498c2ecf20Sopenharmony_ci if (!skb) { 15508c2ecf20Sopenharmony_ci dev_kfree_skb_any(skb_tmp); 15518c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 15528c2ecf20Sopenharmony_ci "Copy skb failed for loopback test\n"); 15538c2ecf20Sopenharmony_ci return -ENOMEM; 15548c2ecf20Sopenharmony_ci } 15558c2ecf20Sopenharmony_ci 15568c2ecf20Sopenharmony_ci /* mark index for every pkt */ 15578c2ecf20Sopenharmony_ci skb->data[LP_PKT_LEN - 1] = j; 15588c2ecf20Sopenharmony_ci 15598c2ecf20Sopenharmony_ci if (hinic_lb_xmit_frame(skb, netdev)) { 15608c2ecf20Sopenharmony_ci dev_kfree_skb_any(skb); 15618c2ecf20Sopenharmony_ci dev_kfree_skb_any(skb_tmp); 15628c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 15638c2ecf20Sopenharmony_ci "Xmit pkt failed for loopback test\n"); 15648c2ecf20Sopenharmony_ci return -EBUSY; 15658c2ecf20Sopenharmony_ci } 15668c2ecf20Sopenharmony_ci } 15678c2ecf20Sopenharmony_ci 15688c2ecf20Sopenharmony_ci /* wait till all pkts received to RX buffer */ 15698c2ecf20Sopenharmony_ci msleep(200); 15708c2ecf20Sopenharmony_ci 15718c2ecf20Sopenharmony_ci for (j = 0; j < LP_PKT_CNT; j++) { 15728c2ecf20Sopenharmony_ci if (memcmp(lb_test_rx_buf + j * LP_PKT_LEN, 15738c2ecf20Sopenharmony_ci skb_tmp->data, LP_PKT_LEN - 1) || 15748c2ecf20Sopenharmony_ci (*(lb_test_rx_buf + j * LP_PKT_LEN + 15758c2ecf20Sopenharmony_ci LP_PKT_LEN - 1) != j)) { 15768c2ecf20Sopenharmony_ci dev_kfree_skb_any(skb_tmp); 15778c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 15788c2ecf20Sopenharmony_ci "Compare pkt failed in loopback test(index=0x%02x, data[%d]=0x%02x)\n", 15798c2ecf20Sopenharmony_ci j + i * LP_PKT_CNT, 15808c2ecf20Sopenharmony_ci LP_PKT_LEN - 1, 15818c2ecf20Sopenharmony_ci *(lb_test_rx_buf + j * LP_PKT_LEN + 15828c2ecf20Sopenharmony_ci LP_PKT_LEN - 1)); 15838c2ecf20Sopenharmony_ci return -EIO; 15848c2ecf20Sopenharmony_ci } 15858c2ecf20Sopenharmony_ci } 15868c2ecf20Sopenharmony_ci } 15878c2ecf20Sopenharmony_ci 15888c2ecf20Sopenharmony_ci dev_kfree_skb_any(skb_tmp); 15898c2ecf20Sopenharmony_ci return 0; 15908c2ecf20Sopenharmony_ci} 15918c2ecf20Sopenharmony_ci 15928c2ecf20Sopenharmony_cistatic int do_lp_test(struct hinic_dev *nic_dev, u32 flags, u32 test_time, 15938c2ecf20Sopenharmony_ci enum diag_test_index *test_index) 15948c2ecf20Sopenharmony_ci{ 15958c2ecf20Sopenharmony_ci struct net_device *netdev = nic_dev->netdev; 15968c2ecf20Sopenharmony_ci u8 *lb_test_rx_buf = NULL; 15978c2ecf20Sopenharmony_ci int err = 0; 15988c2ecf20Sopenharmony_ci 15998c2ecf20Sopenharmony_ci if (!(flags & ETH_TEST_FL_EXTERNAL_LB)) { 16008c2ecf20Sopenharmony_ci *test_index = INTERNAL_LP_TEST; 16018c2ecf20Sopenharmony_ci if (hinic_set_loopback_mode(nic_dev->hwdev, 16028c2ecf20Sopenharmony_ci HINIC_INTERNAL_LP_MODE, true)) { 16038c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 16048c2ecf20Sopenharmony_ci "Failed to set port loopback mode before loopback test\n"); 16058c2ecf20Sopenharmony_ci return -EIO; 16068c2ecf20Sopenharmony_ci } 16078c2ecf20Sopenharmony_ci } else { 16088c2ecf20Sopenharmony_ci *test_index = EXTERNAL_LP_TEST; 16098c2ecf20Sopenharmony_ci } 16108c2ecf20Sopenharmony_ci 16118c2ecf20Sopenharmony_ci lb_test_rx_buf = vmalloc(LP_PKT_CNT * LP_PKT_LEN); 16128c2ecf20Sopenharmony_ci if (!lb_test_rx_buf) { 16138c2ecf20Sopenharmony_ci err = -ENOMEM; 16148c2ecf20Sopenharmony_ci } else { 16158c2ecf20Sopenharmony_ci nic_dev->lb_test_rx_buf = lb_test_rx_buf; 16168c2ecf20Sopenharmony_ci nic_dev->lb_pkt_len = LP_PKT_LEN; 16178c2ecf20Sopenharmony_ci nic_dev->flags |= HINIC_LP_TEST; 16188c2ecf20Sopenharmony_ci err = hinic_run_lp_test(nic_dev, test_time); 16198c2ecf20Sopenharmony_ci nic_dev->flags &= ~HINIC_LP_TEST; 16208c2ecf20Sopenharmony_ci msleep(100); 16218c2ecf20Sopenharmony_ci vfree(lb_test_rx_buf); 16228c2ecf20Sopenharmony_ci nic_dev->lb_test_rx_buf = NULL; 16238c2ecf20Sopenharmony_ci } 16248c2ecf20Sopenharmony_ci 16258c2ecf20Sopenharmony_ci if (!(flags & ETH_TEST_FL_EXTERNAL_LB)) { 16268c2ecf20Sopenharmony_ci if (hinic_set_loopback_mode(nic_dev->hwdev, 16278c2ecf20Sopenharmony_ci HINIC_INTERNAL_LP_MODE, false)) { 16288c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 16298c2ecf20Sopenharmony_ci "Failed to cancel port loopback mode after loopback test\n"); 16308c2ecf20Sopenharmony_ci err = -EIO; 16318c2ecf20Sopenharmony_ci } 16328c2ecf20Sopenharmony_ci } 16338c2ecf20Sopenharmony_ci 16348c2ecf20Sopenharmony_ci return err; 16358c2ecf20Sopenharmony_ci} 16368c2ecf20Sopenharmony_ci 16378c2ecf20Sopenharmony_cistatic void hinic_diag_test(struct net_device *netdev, 16388c2ecf20Sopenharmony_ci struct ethtool_test *eth_test, u64 *data) 16398c2ecf20Sopenharmony_ci{ 16408c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 16418c2ecf20Sopenharmony_ci enum hinic_port_link_state link_state; 16428c2ecf20Sopenharmony_ci enum diag_test_index test_index = 0; 16438c2ecf20Sopenharmony_ci int err = 0; 16448c2ecf20Sopenharmony_ci 16458c2ecf20Sopenharmony_ci memset(data, 0, DIAG_TEST_MAX * sizeof(u64)); 16468c2ecf20Sopenharmony_ci 16478c2ecf20Sopenharmony_ci /* don't support loopback test when netdev is closed. */ 16488c2ecf20Sopenharmony_ci if (!(nic_dev->flags & HINIC_INTF_UP)) { 16498c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 16508c2ecf20Sopenharmony_ci "Do not support loopback test when netdev is closed\n"); 16518c2ecf20Sopenharmony_ci eth_test->flags |= ETH_TEST_FL_FAILED; 16528c2ecf20Sopenharmony_ci data[PORT_DOWN_ERR_IDX] = 1; 16538c2ecf20Sopenharmony_ci return; 16548c2ecf20Sopenharmony_ci } 16558c2ecf20Sopenharmony_ci 16568c2ecf20Sopenharmony_ci netif_carrier_off(netdev); 16578c2ecf20Sopenharmony_ci netif_tx_disable(netdev); 16588c2ecf20Sopenharmony_ci 16598c2ecf20Sopenharmony_ci err = do_lp_test(nic_dev, eth_test->flags, LP_DEFAULT_TIME, 16608c2ecf20Sopenharmony_ci &test_index); 16618c2ecf20Sopenharmony_ci if (err) { 16628c2ecf20Sopenharmony_ci eth_test->flags |= ETH_TEST_FL_FAILED; 16638c2ecf20Sopenharmony_ci data[test_index] = 1; 16648c2ecf20Sopenharmony_ci } 16658c2ecf20Sopenharmony_ci 16668c2ecf20Sopenharmony_ci netif_tx_wake_all_queues(netdev); 16678c2ecf20Sopenharmony_ci 16688c2ecf20Sopenharmony_ci err = hinic_port_link_state(nic_dev, &link_state); 16698c2ecf20Sopenharmony_ci if (!err && link_state == HINIC_LINK_STATE_UP) 16708c2ecf20Sopenharmony_ci netif_carrier_on(netdev); 16718c2ecf20Sopenharmony_ci 16728c2ecf20Sopenharmony_ci} 16738c2ecf20Sopenharmony_ci 16748c2ecf20Sopenharmony_cistatic int hinic_set_phys_id(struct net_device *netdev, 16758c2ecf20Sopenharmony_ci enum ethtool_phys_id_state state) 16768c2ecf20Sopenharmony_ci{ 16778c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 16788c2ecf20Sopenharmony_ci int err = 0; 16798c2ecf20Sopenharmony_ci u8 port; 16808c2ecf20Sopenharmony_ci 16818c2ecf20Sopenharmony_ci port = nic_dev->hwdev->port_id; 16828c2ecf20Sopenharmony_ci 16838c2ecf20Sopenharmony_ci switch (state) { 16848c2ecf20Sopenharmony_ci case ETHTOOL_ID_ACTIVE: 16858c2ecf20Sopenharmony_ci err = hinic_set_led_status(nic_dev->hwdev, port, 16868c2ecf20Sopenharmony_ci HINIC_LED_TYPE_LINK, 16878c2ecf20Sopenharmony_ci HINIC_LED_MODE_FORCE_2HZ); 16888c2ecf20Sopenharmony_ci if (err) 16898c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 16908c2ecf20Sopenharmony_ci "Set LED blinking in 2HZ failed\n"); 16918c2ecf20Sopenharmony_ci break; 16928c2ecf20Sopenharmony_ci 16938c2ecf20Sopenharmony_ci case ETHTOOL_ID_INACTIVE: 16948c2ecf20Sopenharmony_ci err = hinic_reset_led_status(nic_dev->hwdev, port); 16958c2ecf20Sopenharmony_ci if (err) 16968c2ecf20Sopenharmony_ci netif_err(nic_dev, drv, netdev, 16978c2ecf20Sopenharmony_ci "Reset LED to original status failed\n"); 16988c2ecf20Sopenharmony_ci break; 16998c2ecf20Sopenharmony_ci 17008c2ecf20Sopenharmony_ci default: 17018c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 17028c2ecf20Sopenharmony_ci } 17038c2ecf20Sopenharmony_ci 17048c2ecf20Sopenharmony_ci return err; 17058c2ecf20Sopenharmony_ci} 17068c2ecf20Sopenharmony_ci 17078c2ecf20Sopenharmony_cistatic int hinic_get_module_info(struct net_device *netdev, 17088c2ecf20Sopenharmony_ci struct ethtool_modinfo *modinfo) 17098c2ecf20Sopenharmony_ci{ 17108c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 17118c2ecf20Sopenharmony_ci u8 sfp_type_ext; 17128c2ecf20Sopenharmony_ci u8 sfp_type; 17138c2ecf20Sopenharmony_ci int err; 17148c2ecf20Sopenharmony_ci 17158c2ecf20Sopenharmony_ci err = hinic_get_sfp_type(nic_dev->hwdev, &sfp_type, &sfp_type_ext); 17168c2ecf20Sopenharmony_ci if (err) 17178c2ecf20Sopenharmony_ci return err; 17188c2ecf20Sopenharmony_ci 17198c2ecf20Sopenharmony_ci switch (sfp_type) { 17208c2ecf20Sopenharmony_ci case SFF8024_ID_SFP: 17218c2ecf20Sopenharmony_ci modinfo->type = ETH_MODULE_SFF_8472; 17228c2ecf20Sopenharmony_ci modinfo->eeprom_len = ETH_MODULE_SFF_8472_LEN; 17238c2ecf20Sopenharmony_ci break; 17248c2ecf20Sopenharmony_ci case SFF8024_ID_QSFP_8438: 17258c2ecf20Sopenharmony_ci modinfo->type = ETH_MODULE_SFF_8436; 17268c2ecf20Sopenharmony_ci modinfo->eeprom_len = ETH_MODULE_SFF_8436_MAX_LEN; 17278c2ecf20Sopenharmony_ci break; 17288c2ecf20Sopenharmony_ci case SFF8024_ID_QSFP_8436_8636: 17298c2ecf20Sopenharmony_ci if (sfp_type_ext >= 0x3) { 17308c2ecf20Sopenharmony_ci modinfo->type = ETH_MODULE_SFF_8636; 17318c2ecf20Sopenharmony_ci modinfo->eeprom_len = ETH_MODULE_SFF_8636_MAX_LEN; 17328c2ecf20Sopenharmony_ci 17338c2ecf20Sopenharmony_ci } else { 17348c2ecf20Sopenharmony_ci modinfo->type = ETH_MODULE_SFF_8436; 17358c2ecf20Sopenharmony_ci modinfo->eeprom_len = ETH_MODULE_SFF_8436_MAX_LEN; 17368c2ecf20Sopenharmony_ci } 17378c2ecf20Sopenharmony_ci break; 17388c2ecf20Sopenharmony_ci case SFF8024_ID_QSFP28_8636: 17398c2ecf20Sopenharmony_ci modinfo->type = ETH_MODULE_SFF_8636; 17408c2ecf20Sopenharmony_ci modinfo->eeprom_len = ETH_MODULE_SFF_8636_MAX_LEN; 17418c2ecf20Sopenharmony_ci break; 17428c2ecf20Sopenharmony_ci default: 17438c2ecf20Sopenharmony_ci netif_warn(nic_dev, drv, netdev, 17448c2ecf20Sopenharmony_ci "Optical module unknown: 0x%x\n", sfp_type); 17458c2ecf20Sopenharmony_ci return -EINVAL; 17468c2ecf20Sopenharmony_ci } 17478c2ecf20Sopenharmony_ci 17488c2ecf20Sopenharmony_ci return 0; 17498c2ecf20Sopenharmony_ci} 17508c2ecf20Sopenharmony_ci 17518c2ecf20Sopenharmony_cistatic int hinic_get_module_eeprom(struct net_device *netdev, 17528c2ecf20Sopenharmony_ci struct ethtool_eeprom *ee, u8 *data) 17538c2ecf20Sopenharmony_ci{ 17548c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 17558c2ecf20Sopenharmony_ci u8 sfp_data[STD_SFP_INFO_MAX_SIZE]; 17568c2ecf20Sopenharmony_ci u16 len; 17578c2ecf20Sopenharmony_ci int err; 17588c2ecf20Sopenharmony_ci 17598c2ecf20Sopenharmony_ci if (!ee->len || ((ee->len + ee->offset) > STD_SFP_INFO_MAX_SIZE)) 17608c2ecf20Sopenharmony_ci return -EINVAL; 17618c2ecf20Sopenharmony_ci 17628c2ecf20Sopenharmony_ci memset(data, 0, ee->len); 17638c2ecf20Sopenharmony_ci 17648c2ecf20Sopenharmony_ci err = hinic_get_sfp_eeprom(nic_dev->hwdev, sfp_data, &len); 17658c2ecf20Sopenharmony_ci if (err) 17668c2ecf20Sopenharmony_ci return err; 17678c2ecf20Sopenharmony_ci 17688c2ecf20Sopenharmony_ci memcpy(data, sfp_data + ee->offset, ee->len); 17698c2ecf20Sopenharmony_ci 17708c2ecf20Sopenharmony_ci return 0; 17718c2ecf20Sopenharmony_ci} 17728c2ecf20Sopenharmony_ci 17738c2ecf20Sopenharmony_cistatic int 17748c2ecf20Sopenharmony_cihinic_get_link_ext_state(struct net_device *netdev, 17758c2ecf20Sopenharmony_ci struct ethtool_link_ext_state_info *link_ext_state_info) 17768c2ecf20Sopenharmony_ci{ 17778c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 17788c2ecf20Sopenharmony_ci 17798c2ecf20Sopenharmony_ci if (netif_carrier_ok(netdev)) 17808c2ecf20Sopenharmony_ci return -ENODATA; 17818c2ecf20Sopenharmony_ci 17828c2ecf20Sopenharmony_ci if (nic_dev->cable_unplugged) 17838c2ecf20Sopenharmony_ci link_ext_state_info->link_ext_state = 17848c2ecf20Sopenharmony_ci ETHTOOL_LINK_EXT_STATE_NO_CABLE; 17858c2ecf20Sopenharmony_ci else if (nic_dev->module_unrecognized) 17868c2ecf20Sopenharmony_ci link_ext_state_info->link_ext_state = 17878c2ecf20Sopenharmony_ci ETHTOOL_LINK_EXT_STATE_LINK_LOGICAL_MISMATCH; 17888c2ecf20Sopenharmony_ci 17898c2ecf20Sopenharmony_ci return 0; 17908c2ecf20Sopenharmony_ci} 17918c2ecf20Sopenharmony_ci 17928c2ecf20Sopenharmony_cistatic const struct ethtool_ops hinic_ethtool_ops = { 17938c2ecf20Sopenharmony_ci .supported_coalesce_params = ETHTOOL_COALESCE_RX_USECS | 17948c2ecf20Sopenharmony_ci ETHTOOL_COALESCE_RX_MAX_FRAMES | 17958c2ecf20Sopenharmony_ci ETHTOOL_COALESCE_TX_USECS | 17968c2ecf20Sopenharmony_ci ETHTOOL_COALESCE_TX_MAX_FRAMES, 17978c2ecf20Sopenharmony_ci 17988c2ecf20Sopenharmony_ci .get_link_ksettings = hinic_get_link_ksettings, 17998c2ecf20Sopenharmony_ci .set_link_ksettings = hinic_set_link_ksettings, 18008c2ecf20Sopenharmony_ci .get_drvinfo = hinic_get_drvinfo, 18018c2ecf20Sopenharmony_ci .get_link = ethtool_op_get_link, 18028c2ecf20Sopenharmony_ci .get_link_ext_state = hinic_get_link_ext_state, 18038c2ecf20Sopenharmony_ci .get_ringparam = hinic_get_ringparam, 18048c2ecf20Sopenharmony_ci .set_ringparam = hinic_set_ringparam, 18058c2ecf20Sopenharmony_ci .get_coalesce = hinic_get_coalesce, 18068c2ecf20Sopenharmony_ci .set_coalesce = hinic_set_coalesce, 18078c2ecf20Sopenharmony_ci .get_per_queue_coalesce = hinic_get_per_queue_coalesce, 18088c2ecf20Sopenharmony_ci .set_per_queue_coalesce = hinic_set_per_queue_coalesce, 18098c2ecf20Sopenharmony_ci .get_pauseparam = hinic_get_pauseparam, 18108c2ecf20Sopenharmony_ci .set_pauseparam = hinic_set_pauseparam, 18118c2ecf20Sopenharmony_ci .get_channels = hinic_get_channels, 18128c2ecf20Sopenharmony_ci .set_channels = hinic_set_channels, 18138c2ecf20Sopenharmony_ci .get_rxnfc = hinic_get_rxnfc, 18148c2ecf20Sopenharmony_ci .set_rxnfc = hinic_set_rxnfc, 18158c2ecf20Sopenharmony_ci .get_rxfh_key_size = hinic_get_rxfh_key_size, 18168c2ecf20Sopenharmony_ci .get_rxfh_indir_size = hinic_get_rxfh_indir_size, 18178c2ecf20Sopenharmony_ci .get_rxfh = hinic_get_rxfh, 18188c2ecf20Sopenharmony_ci .set_rxfh = hinic_set_rxfh, 18198c2ecf20Sopenharmony_ci .get_sset_count = hinic_get_sset_count, 18208c2ecf20Sopenharmony_ci .get_ethtool_stats = hinic_get_ethtool_stats, 18218c2ecf20Sopenharmony_ci .get_strings = hinic_get_strings, 18228c2ecf20Sopenharmony_ci .self_test = hinic_diag_test, 18238c2ecf20Sopenharmony_ci .set_phys_id = hinic_set_phys_id, 18248c2ecf20Sopenharmony_ci .get_module_info = hinic_get_module_info, 18258c2ecf20Sopenharmony_ci .get_module_eeprom = hinic_get_module_eeprom, 18268c2ecf20Sopenharmony_ci}; 18278c2ecf20Sopenharmony_ci 18288c2ecf20Sopenharmony_cistatic const struct ethtool_ops hinicvf_ethtool_ops = { 18298c2ecf20Sopenharmony_ci .supported_coalesce_params = ETHTOOL_COALESCE_RX_USECS | 18308c2ecf20Sopenharmony_ci ETHTOOL_COALESCE_RX_MAX_FRAMES | 18318c2ecf20Sopenharmony_ci ETHTOOL_COALESCE_TX_USECS | 18328c2ecf20Sopenharmony_ci ETHTOOL_COALESCE_TX_MAX_FRAMES, 18338c2ecf20Sopenharmony_ci 18348c2ecf20Sopenharmony_ci .get_link_ksettings = hinic_get_link_ksettings, 18358c2ecf20Sopenharmony_ci .get_drvinfo = hinic_get_drvinfo, 18368c2ecf20Sopenharmony_ci .get_link = ethtool_op_get_link, 18378c2ecf20Sopenharmony_ci .get_ringparam = hinic_get_ringparam, 18388c2ecf20Sopenharmony_ci .set_ringparam = hinic_set_ringparam, 18398c2ecf20Sopenharmony_ci .get_coalesce = hinic_get_coalesce, 18408c2ecf20Sopenharmony_ci .set_coalesce = hinic_set_coalesce, 18418c2ecf20Sopenharmony_ci .get_per_queue_coalesce = hinic_get_per_queue_coalesce, 18428c2ecf20Sopenharmony_ci .set_per_queue_coalesce = hinic_set_per_queue_coalesce, 18438c2ecf20Sopenharmony_ci .get_channels = hinic_get_channels, 18448c2ecf20Sopenharmony_ci .set_channels = hinic_set_channels, 18458c2ecf20Sopenharmony_ci .get_rxnfc = hinic_get_rxnfc, 18468c2ecf20Sopenharmony_ci .set_rxnfc = hinic_set_rxnfc, 18478c2ecf20Sopenharmony_ci .get_rxfh_key_size = hinic_get_rxfh_key_size, 18488c2ecf20Sopenharmony_ci .get_rxfh_indir_size = hinic_get_rxfh_indir_size, 18498c2ecf20Sopenharmony_ci .get_rxfh = hinic_get_rxfh, 18508c2ecf20Sopenharmony_ci .set_rxfh = hinic_set_rxfh, 18518c2ecf20Sopenharmony_ci .get_sset_count = hinic_get_sset_count, 18528c2ecf20Sopenharmony_ci .get_ethtool_stats = hinic_get_ethtool_stats, 18538c2ecf20Sopenharmony_ci .get_strings = hinic_get_strings, 18548c2ecf20Sopenharmony_ci}; 18558c2ecf20Sopenharmony_ci 18568c2ecf20Sopenharmony_civoid hinic_set_ethtool_ops(struct net_device *netdev) 18578c2ecf20Sopenharmony_ci{ 18588c2ecf20Sopenharmony_ci struct hinic_dev *nic_dev = netdev_priv(netdev); 18598c2ecf20Sopenharmony_ci 18608c2ecf20Sopenharmony_ci if (!HINIC_IS_VF(nic_dev->hwdev->hwif)) 18618c2ecf20Sopenharmony_ci netdev->ethtool_ops = &hinic_ethtool_ops; 18628c2ecf20Sopenharmony_ci else 18638c2ecf20Sopenharmony_ci netdev->ethtool_ops = &hinicvf_ethtool_ops; 18648c2ecf20Sopenharmony_ci} 1865