2018-07-19 22:47:06 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
#include <linux/dma-mapping.h>
|
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/interrupt.h>
|
2019-05-24 19:19:45 +08:00
|
|
|
#ifdef CONFIG_RFS_ACCEL
|
|
|
|
#include <linux/cpu_rmap.h>
|
|
|
|
#endif
|
2017-08-02 23:59:45 +08:00
|
|
|
#include <linux/if_vlan.h>
|
2020-06-18 09:02:10 +08:00
|
|
|
#include <linux/irq.h>
|
2017-08-02 23:59:45 +08:00
|
|
|
#include <linux/ip.h>
|
|
|
|
#include <linux/ipv6.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/pci.h>
|
2018-10-20 03:15:27 +08:00
|
|
|
#include <linux/aer.h>
|
2017-08-02 23:59:45 +08:00
|
|
|
#include <linux/skbuff.h>
|
|
|
|
#include <linux/sctp.h>
|
|
|
|
#include <net/gre.h>
|
2019-06-07 10:03:04 +08:00
|
|
|
#include <net/ip6_checksum.h>
|
2017-10-17 14:51:30 +08:00
|
|
|
#include <net/pkt_cls.h>
|
2018-11-15 17:29:25 +08:00
|
|
|
#include <net/tcp.h>
|
2017-08-02 23:59:45 +08:00
|
|
|
#include <net/vxlan.h>
|
2020-08-26 13:11:50 +08:00
|
|
|
#include <net/geneve.h>
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
#include "hnae3.h"
|
|
|
|
#include "hns3_enet.h"
|
2020-01-04 10:49:24 +08:00
|
|
|
/* All hns3 tracepoints are defined by the include below, which
|
|
|
|
* must be included exactly once across the whole kernel with
|
|
|
|
* CREATE_TRACE_POINTS defined
|
|
|
|
*/
|
|
|
|
#define CREATE_TRACE_POINTS
|
|
|
|
#include "hns3_trace.h"
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2021-02-09 10:41:54 +08:00
|
|
|
#define hns3_set_field(origin, shift, val) ((origin) |= (val) << (shift))
|
2019-03-21 11:28:43 +08:00
|
|
|
#define hns3_tx_bd_count(S) DIV_ROUND_UP(S, HNS3_MAX_BD_SIZE)
|
2019-02-23 17:22:13 +08:00
|
|
|
|
2019-08-09 10:31:13 +08:00
|
|
|
#define hns3_rl_err(fmt, ...) \
|
|
|
|
do { \
|
|
|
|
if (net_ratelimit()) \
|
|
|
|
netdev_err(fmt, ##__VA_ARGS__); \
|
|
|
|
} while (0)
|
|
|
|
|
2019-06-28 19:50:07 +08:00
|
|
|
static void hns3_clear_all_ring(struct hnae3_handle *h, bool force);
|
2018-05-26 02:43:04 +08:00
|
|
|
|
2017-10-09 15:44:01 +08:00
|
|
|
static const char hns3_driver_name[] = "hns3";
|
2017-08-02 23:59:45 +08:00
|
|
|
static const char hns3_driver_string[] =
|
|
|
|
"Hisilicon Ethernet Network Driver for Hip08 Family";
|
|
|
|
static const char hns3_copyright[] = "Copyright (c) 2017 Huawei Corporation.";
|
|
|
|
static struct hnae3_client client;
|
|
|
|
|
2019-04-19 11:05:43 +08:00
|
|
|
static int debug = -1;
|
|
|
|
module_param(debug, int, 0);
|
|
|
|
MODULE_PARM_DESC(debug, " Network interface message level setting");
|
|
|
|
|
|
|
|
#define DEFAULT_MSG_LEVEL (NETIF_MSG_PROBE | NETIF_MSG_LINK | \
|
|
|
|
NETIF_MSG_IFDOWN | NETIF_MSG_IFUP)
|
|
|
|
|
2019-08-09 10:31:09 +08:00
|
|
|
#define HNS3_INNER_VLAN_TAG 1
|
|
|
|
#define HNS3_OUTER_VLAN_TAG 2
|
|
|
|
|
2020-01-15 10:46:45 +08:00
|
|
|
#define HNS3_MIN_TX_LEN 33U
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* hns3_pci_tbl - PCI Device ID Table
|
|
|
|
*
|
|
|
|
* Last entry must be all 0s
|
|
|
|
*
|
|
|
|
* { Vendor ID, Device ID, SubVendor ID, SubDevice ID,
|
|
|
|
* Class, Class Mask, private data (not used) }
|
|
|
|
*/
|
|
|
|
static const struct pci_device_id hns3_pci_tbl[] = {
|
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_GE), 0},
|
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_25GE), 0},
|
2017-09-20 18:52:50 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_25GE_RDMA),
|
2017-09-20 18:52:51 +08:00
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2017-09-20 18:52:50 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_25GE_RDMA_MACSEC),
|
2017-09-20 18:52:51 +08:00
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2017-09-20 18:52:50 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_50GE_RDMA),
|
2017-09-20 18:52:51 +08:00
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2017-09-20 18:52:50 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_50GE_RDMA_MACSEC),
|
2017-09-20 18:52:51 +08:00
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2017-09-20 18:52:50 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_100G_RDMA_MACSEC),
|
2017-09-20 18:52:51 +08:00
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2020-09-25 08:26:17 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_200G_RDMA),
|
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2020-09-25 08:26:18 +08:00
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_VF), 0},
|
|
|
|
{PCI_VDEVICE(HUAWEI, HNAE3_DEV_ID_RDMA_DCB_PFC_VF),
|
2018-08-03 17:56:30 +08:00
|
|
|
HNAE3_DEV_SUPPORT_ROCE_DCB_BITS},
|
2017-08-02 23:59:45 +08:00
|
|
|
/* required last entry */
|
|
|
|
{0, }
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(pci, hns3_pci_tbl);
|
|
|
|
|
2018-07-19 22:47:04 +08:00
|
|
|
static irqreturn_t hns3_irq_handle(int irq, void *vector)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2018-07-19 22:47:04 +08:00
|
|
|
struct hns3_enet_tqp_vector *tqp_vector = vector;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:42 +08:00
|
|
|
napi_schedule_irqoff(&tqp_vector->napi);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_nic_uninit_irq(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vectors;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
tqp_vectors = &priv->tqp_vector[i];
|
|
|
|
|
|
|
|
if (tqp_vectors->irq_init_flag != HNS3_VECTOR_INITED)
|
|
|
|
continue;
|
|
|
|
|
2019-05-24 19:19:45 +08:00
|
|
|
/* clear the affinity mask */
|
2018-09-20 01:29:47 +08:00
|
|
|
irq_set_affinity_hint(tqp_vectors->vector_irq, NULL);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* release the irq resource */
|
|
|
|
free_irq(tqp_vectors->vector_irq, tqp_vectors);
|
|
|
|
tqp_vectors->irq_init_flag = HNS3_VECTOR_NOT_INITED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_init_irq(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vectors;
|
|
|
|
int txrx_int_idx = 0;
|
|
|
|
int rx_int_idx = 0;
|
|
|
|
int tx_int_idx = 0;
|
|
|
|
unsigned int i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
tqp_vectors = &priv->tqp_vector[i];
|
|
|
|
|
|
|
|
if (tqp_vectors->irq_init_flag == HNS3_VECTOR_INITED)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (tqp_vectors->tx_group.ring && tqp_vectors->rx_group.ring) {
|
2020-01-04 10:49:26 +08:00
|
|
|
snprintf(tqp_vectors->name, HNAE3_INT_NAME_LEN,
|
|
|
|
"%s-%s-%s-%d", hns3_driver_name,
|
|
|
|
pci_name(priv->ae_handle->pdev),
|
|
|
|
"TxRx", txrx_int_idx++);
|
2017-08-02 23:59:45 +08:00
|
|
|
txrx_int_idx++;
|
|
|
|
} else if (tqp_vectors->rx_group.ring) {
|
2020-01-04 10:49:26 +08:00
|
|
|
snprintf(tqp_vectors->name, HNAE3_INT_NAME_LEN,
|
|
|
|
"%s-%s-%s-%d", hns3_driver_name,
|
|
|
|
pci_name(priv->ae_handle->pdev),
|
|
|
|
"Rx", rx_int_idx++);
|
2017-08-02 23:59:45 +08:00
|
|
|
} else if (tqp_vectors->tx_group.ring) {
|
2020-01-04 10:49:26 +08:00
|
|
|
snprintf(tqp_vectors->name, HNAE3_INT_NAME_LEN,
|
|
|
|
"%s-%s-%s-%d", hns3_driver_name,
|
|
|
|
pci_name(priv->ae_handle->pdev),
|
|
|
|
"Tx", tx_int_idx++);
|
2017-08-02 23:59:45 +08:00
|
|
|
} else {
|
|
|
|
/* Skip this unused q_vector */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
tqp_vectors->name[HNAE3_INT_NAME_LEN - 1] = '\0';
|
|
|
|
|
2020-06-18 09:02:10 +08:00
|
|
|
irq_set_status_flags(tqp_vectors->vector_irq, IRQ_NOAUTOEN);
|
2017-08-02 23:59:45 +08:00
|
|
|
ret = request_irq(tqp_vectors->vector_irq, hns3_irq_handle, 0,
|
2019-06-07 10:03:13 +08:00
|
|
|
tqp_vectors->name, tqp_vectors);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (ret) {
|
|
|
|
netdev_err(priv->netdev, "request irq(%d) fail\n",
|
|
|
|
tqp_vectors->vector_irq);
|
2019-06-13 17:12:28 +08:00
|
|
|
hns3_nic_uninit_irq(priv);
|
2017-08-02 23:59:45 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-09-20 01:29:47 +08:00
|
|
|
irq_set_affinity_hint(tqp_vectors->vector_irq,
|
|
|
|
&tqp_vectors->affinity_mask);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
tqp_vectors->irq_init_flag = HNS3_VECTOR_INITED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_mask_vector_irq(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
u32 mask_en)
|
|
|
|
{
|
|
|
|
writel(mask_en, tqp_vector->mask_addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_vector_enable(struct hns3_enet_tqp_vector *tqp_vector)
|
|
|
|
{
|
|
|
|
napi_enable(&tqp_vector->napi);
|
2020-01-04 10:49:25 +08:00
|
|
|
enable_irq(tqp_vector->vector_irq);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
/* enable vector */
|
|
|
|
hns3_mask_vector_irq(tqp_vector, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_vector_disable(struct hns3_enet_tqp_vector *tqp_vector)
|
|
|
|
{
|
|
|
|
/* disable vector */
|
|
|
|
hns3_mask_vector_irq(tqp_vector, 0);
|
|
|
|
|
|
|
|
disable_irq(tqp_vector->vector_irq);
|
|
|
|
napi_disable(&tqp_vector->napi);
|
|
|
|
}
|
|
|
|
|
2018-01-12 16:23:10 +08:00
|
|
|
void hns3_set_vector_coalesce_rl(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
u32 rl_value)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2018-01-12 16:23:10 +08:00
|
|
|
u32 rl_reg = hns3_rl_usec_to_reg(rl_value);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* this defines the configuration for RL (Interrupt Rate Limiter).
|
|
|
|
* Rl defines rate of interrupts i.e. number of interrupts-per-second
|
|
|
|
* GL and RL(Rate Limiter) are 2 ways to acheive interrupt coalescing
|
|
|
|
*/
|
2020-11-16 16:20:54 +08:00
|
|
|
if (rl_reg > 0 && !tqp_vector->tx_group.coal.adapt_enable &&
|
|
|
|
!tqp_vector->rx_group.coal.adapt_enable)
|
2018-01-12 16:23:10 +08:00
|
|
|
/* According to the hardware, the range of rl_reg is
|
|
|
|
* 0-59 and the unit is 4.
|
|
|
|
*/
|
|
|
|
rl_reg |= HNS3_INT_RL_ENABLE_MASK;
|
|
|
|
|
|
|
|
writel(rl_reg, tqp_vector->mask_addr + HNS3_VECTOR_RL_OFFSET);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hns3_set_vector_coalesce_rx_gl(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
u32 gl_value)
|
|
|
|
{
|
2020-11-16 16:20:53 +08:00
|
|
|
u32 new_val;
|
2018-01-12 16:23:10 +08:00
|
|
|
|
2020-11-16 16:20:53 +08:00
|
|
|
if (tqp_vector->rx_group.coal.unit_1us)
|
|
|
|
new_val = gl_value | HNS3_INT_GL_1US;
|
|
|
|
else
|
|
|
|
new_val = hns3_gl_usec_to_reg(gl_value);
|
|
|
|
|
|
|
|
writel(new_val, tqp_vector->mask_addr + HNS3_VECTOR_GL0_OFFSET);
|
2018-01-12 16:23:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void hns3_set_vector_coalesce_tx_gl(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
u32 gl_value)
|
|
|
|
{
|
2020-11-16 16:20:53 +08:00
|
|
|
u32 new_val;
|
|
|
|
|
|
|
|
if (tqp_vector->tx_group.coal.unit_1us)
|
|
|
|
new_val = gl_value | HNS3_INT_GL_1US;
|
|
|
|
else
|
|
|
|
new_val = hns3_gl_usec_to_reg(gl_value);
|
2018-01-12 16:23:10 +08:00
|
|
|
|
2020-11-16 16:20:53 +08:00
|
|
|
writel(new_val, tqp_vector->mask_addr + HNS3_VECTOR_GL1_OFFSET);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-11-16 16:20:51 +08:00
|
|
|
void hns3_set_vector_coalesce_tx_ql(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
u32 ql_value)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2020-11-16 16:20:51 +08:00
|
|
|
writel(ql_value, tqp_vector->mask_addr + HNS3_VECTOR_TX_QL_OFFSET);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hns3_set_vector_coalesce_rx_ql(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
u32 ql_value)
|
|
|
|
{
|
|
|
|
writel(ql_value, tqp_vector->mask_addr + HNS3_VECTOR_RX_QL_OFFSET);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_vector_coalesce_init(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(priv->ae_handle->pdev);
|
|
|
|
struct hns3_enet_coalesce *tx_coal = &tqp_vector->tx_group.coal;
|
|
|
|
struct hns3_enet_coalesce *rx_coal = &tqp_vector->rx_group.coal;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* initialize the configuration for interrupt coalescing.
|
|
|
|
* 1. GL (Interrupt Gap Limiter)
|
|
|
|
* 2. RL (Interrupt Rate Limiter)
|
2020-11-16 16:20:51 +08:00
|
|
|
* 3. QL (Interrupt Quantity Limiter)
|
2019-08-16 16:09:37 +08:00
|
|
|
*
|
|
|
|
* Default: enable interrupt coalescing self-adaptive and GL
|
2017-08-02 23:59:45 +08:00
|
|
|
*/
|
2020-11-16 16:20:54 +08:00
|
|
|
tx_coal->adapt_enable = 1;
|
|
|
|
rx_coal->adapt_enable = 1;
|
2020-11-16 16:20:51 +08:00
|
|
|
|
|
|
|
tx_coal->int_gl = HNS3_INT_GL_50K;
|
|
|
|
rx_coal->int_gl = HNS3_INT_GL_50K;
|
2018-01-12 16:23:11 +08:00
|
|
|
|
2020-11-16 16:20:51 +08:00
|
|
|
rx_coal->flow_level = HNS3_FLOW_LOW;
|
|
|
|
tx_coal->flow_level = HNS3_FLOW_LOW;
|
2018-01-12 16:23:11 +08:00
|
|
|
|
2020-11-16 16:20:53 +08:00
|
|
|
/* device version above V3(include V3), GL can configure 1us
|
|
|
|
* unit, so uses 1us unit.
|
|
|
|
*/
|
|
|
|
if (ae_dev->dev_version >= HNAE3_DEVICE_VERSION_V3) {
|
|
|
|
tx_coal->unit_1us = 1;
|
|
|
|
rx_coal->unit_1us = 1;
|
|
|
|
}
|
|
|
|
|
2020-11-16 16:20:51 +08:00
|
|
|
if (ae_dev->dev_specs.int_ql_max) {
|
|
|
|
tx_coal->ql_enable = 1;
|
|
|
|
rx_coal->ql_enable = 1;
|
|
|
|
tx_coal->int_ql_max = ae_dev->dev_specs.int_ql_max;
|
|
|
|
rx_coal->int_ql_max = ae_dev->dev_specs.int_ql_max;
|
|
|
|
tx_coal->int_ql = HNS3_INT_QL_DEFAULT_CFG;
|
|
|
|
rx_coal->int_ql = HNS3_INT_QL_DEFAULT_CFG;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-11-16 16:20:51 +08:00
|
|
|
static void
|
|
|
|
hns3_vector_coalesce_init_hw(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
struct hns3_nic_priv *priv)
|
2018-03-09 10:37:02 +08:00
|
|
|
{
|
2020-11-16 16:20:51 +08:00
|
|
|
struct hns3_enet_coalesce *tx_coal = &tqp_vector->tx_group.coal;
|
|
|
|
struct hns3_enet_coalesce *rx_coal = &tqp_vector->rx_group.coal;
|
2018-03-09 10:37:02 +08:00
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
|
2020-11-16 16:20:51 +08:00
|
|
|
hns3_set_vector_coalesce_tx_gl(tqp_vector, tx_coal->int_gl);
|
|
|
|
hns3_set_vector_coalesce_rx_gl(tqp_vector, rx_coal->int_gl);
|
2018-03-09 10:37:02 +08:00
|
|
|
hns3_set_vector_coalesce_rl(tqp_vector, h->kinfo.int_rl_setting);
|
2020-11-16 16:20:51 +08:00
|
|
|
|
|
|
|
if (tx_coal->ql_enable)
|
|
|
|
hns3_set_vector_coalesce_tx_ql(tqp_vector, tx_coal->int_ql);
|
|
|
|
|
|
|
|
if (rx_coal->ql_enable)
|
|
|
|
hns3_set_vector_coalesce_rx_ql(tqp_vector, rx_coal->int_ql);
|
2018-03-09 10:37:02 +08:00
|
|
|
}
|
|
|
|
|
2017-09-27 09:45:32 +08:00
|
|
|
static int hns3_nic_set_real_num_queue(struct net_device *netdev)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-09-27 09:45:32 +08:00
|
|
|
struct hnae3_knic_private_info *kinfo = &h->kinfo;
|
2020-12-10 11:42:06 +08:00
|
|
|
struct hnae3_tc_info *tc_info = &kinfo->tc_info;
|
2020-12-10 11:42:07 +08:00
|
|
|
unsigned int queue_size = kinfo->num_tqps;
|
2018-07-06 18:27:55 +08:00
|
|
|
int i, ret;
|
|
|
|
|
2020-12-10 11:42:07 +08:00
|
|
|
if (tc_info->num_tc <= 1 && !tc_info->mqprio_active) {
|
2018-07-06 18:27:55 +08:00
|
|
|
netdev_reset_tc(netdev);
|
|
|
|
} else {
|
2020-12-10 11:42:06 +08:00
|
|
|
ret = netdev_set_num_tc(netdev, tc_info->num_tc);
|
2018-07-06 18:27:55 +08:00
|
|
|
if (ret) {
|
|
|
|
netdev_err(netdev,
|
|
|
|
"netdev_set_num_tc fail, ret=%d!\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < HNAE3_MAX_TC; i++) {
|
2020-12-10 11:42:06 +08:00
|
|
|
if (!test_bit(i, &tc_info->tc_en))
|
2018-07-06 18:27:55 +08:00
|
|
|
continue;
|
|
|
|
|
2020-12-10 11:42:06 +08:00
|
|
|
netdev_set_tc_queue(netdev, i, tc_info->tqp_count[i],
|
|
|
|
tc_info->tqp_offset[i]);
|
2018-07-06 18:27:55 +08:00
|
|
|
}
|
|
|
|
}
|
2017-09-27 09:45:32 +08:00
|
|
|
|
|
|
|
ret = netif_set_real_num_tx_queues(netdev, queue_size);
|
|
|
|
if (ret) {
|
|
|
|
netdev_err(netdev,
|
2019-06-07 10:03:13 +08:00
|
|
|
"netif_set_real_num_tx_queues fail, ret=%d!\n", ret);
|
2017-09-27 09:45:32 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = netif_set_real_num_rx_queues(netdev, queue_size);
|
|
|
|
if (ret) {
|
|
|
|
netdev_err(netdev,
|
|
|
|
"netif_set_real_num_rx_queues fail, ret=%d!\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-03-08 19:41:54 +08:00
|
|
|
static u16 hns3_get_max_available_channels(struct hnae3_handle *h)
|
|
|
|
{
|
2018-09-27 02:28:39 +08:00
|
|
|
u16 alloc_tqps, max_rss_size, rss_size;
|
2018-03-08 19:41:54 +08:00
|
|
|
|
2018-09-27 02:28:39 +08:00
|
|
|
h->ae_algo->ops->get_tqps_and_rss_info(h, &alloc_tqps, &max_rss_size);
|
2020-12-10 11:42:06 +08:00
|
|
|
rss_size = alloc_tqps / h->kinfo.tc_info.num_tc;
|
2018-03-08 19:41:54 +08:00
|
|
|
|
2018-09-27 02:28:39 +08:00
|
|
|
return min_t(u16, rss_size, max_rss_size);
|
2018-03-08 19:41:54 +08:00
|
|
|
}
|
|
|
|
|
2018-11-07 12:06:11 +08:00
|
|
|
static void hns3_tqp_enable(struct hnae3_queue *tqp)
|
|
|
|
{
|
|
|
|
u32 rcb_reg;
|
|
|
|
|
|
|
|
rcb_reg = hns3_read_dev(tqp, HNS3_RING_EN_REG);
|
|
|
|
rcb_reg |= BIT(HNS3_RING_EN_B);
|
|
|
|
hns3_write_dev(tqp, HNS3_RING_EN_REG, rcb_reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_tqp_disable(struct hnae3_queue *tqp)
|
|
|
|
{
|
|
|
|
u32 rcb_reg;
|
|
|
|
|
|
|
|
rcb_reg = hns3_read_dev(tqp, HNS3_RING_EN_REG);
|
|
|
|
rcb_reg &= ~BIT(HNS3_RING_EN_B);
|
|
|
|
hns3_write_dev(tqp, HNS3_RING_EN_REG, rcb_reg);
|
|
|
|
}
|
|
|
|
|
2019-05-24 19:19:45 +08:00
|
|
|
static void hns3_free_rx_cpu_rmap(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_RFS_ACCEL
|
|
|
|
free_irq_cpu_rmap(netdev->rx_cpu_rmap);
|
|
|
|
netdev->rx_cpu_rmap = NULL;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_set_rx_cpu_rmap(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_RFS_ACCEL
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
if (!netdev->rx_cpu_rmap) {
|
|
|
|
netdev->rx_cpu_rmap = alloc_irq_cpu_rmap(priv->vector_num);
|
|
|
|
if (!netdev->rx_cpu_rmap)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
tqp_vector = &priv->tqp_vector[i];
|
|
|
|
ret = irq_cpu_rmap_add(netdev->rx_cpu_rmap,
|
|
|
|
tqp_vector->vector_irq);
|
|
|
|
if (ret) {
|
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_nic_net_up(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
int i, j;
|
|
|
|
int ret;
|
|
|
|
|
2018-05-26 02:43:04 +08:00
|
|
|
ret = hns3_nic_reset_all_ring(h);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2019-01-27 00:49:20 +08:00
|
|
|
clear_bit(HNS3_NIC_STATE_DOWN, &priv->state);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* enable the vectors */
|
|
|
|
for (i = 0; i < priv->vector_num; i++)
|
|
|
|
hns3_vector_enable(&priv->tqp_vector[i]);
|
|
|
|
|
2018-11-07 12:06:11 +08:00
|
|
|
/* enable rcb */
|
|
|
|
for (j = 0; j < h->kinfo.num_tqps; j++)
|
|
|
|
hns3_tqp_enable(h->kinfo.tqp[j]);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* start the ae_dev */
|
|
|
|
ret = h->ae_algo->ops->start ? h->ae_algo->ops->start(h) : 0;
|
2020-01-04 10:49:25 +08:00
|
|
|
if (ret) {
|
|
|
|
set_bit(HNS3_NIC_STATE_DOWN, &priv->state);
|
|
|
|
while (j--)
|
|
|
|
hns3_tqp_disable(h->kinfo.tqp[j]);
|
2018-11-07 12:06:11 +08:00
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
for (j = i - 1; j >= 0; j--)
|
|
|
|
hns3_vector_disable(&priv->tqp_vector[j]);
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-02-23 17:22:08 +08:00
|
|
|
static void hns3_config_xps(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector = &priv->tqp_vector[i];
|
|
|
|
struct hns3_enet_ring *ring = tqp_vector->tx_group.ring;
|
|
|
|
|
|
|
|
while (ring) {
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = netif_set_xps_queue(priv->netdev,
|
|
|
|
&tqp_vector->affinity_mask,
|
|
|
|
ring->tqp->tqp_index);
|
|
|
|
if (ret)
|
|
|
|
netdev_warn(priv->netdev,
|
|
|
|
"set xps queue failed: %d", ret);
|
|
|
|
|
|
|
|
ring = ring->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_nic_net_open(struct net_device *netdev)
|
|
|
|
{
|
2018-12-20 11:51:58 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2018-07-06 18:27:55 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
struct hnae3_knic_private_info *kinfo;
|
|
|
|
int i, ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-11-07 12:06:12 +08:00
|
|
|
if (hns3_nic_resetting(netdev))
|
|
|
|
return -EBUSY;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
netif_carrier_off(netdev);
|
|
|
|
|
2017-09-27 09:45:32 +08:00
|
|
|
ret = hns3_nic_set_real_num_queue(netdev);
|
|
|
|
if (ret)
|
2017-08-02 23:59:45 +08:00
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = hns3_nic_net_up(netdev);
|
|
|
|
if (ret) {
|
2019-06-07 10:03:13 +08:00
|
|
|
netdev_err(netdev, "net up fail, ret=%d!\n", ret);
|
2017-08-02 23:59:45 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-07-06 18:27:55 +08:00
|
|
|
kinfo = &h->kinfo;
|
2019-06-07 10:03:13 +08:00
|
|
|
for (i = 0; i < HNAE3_MAX_USER_PRIO; i++)
|
2020-12-10 11:42:06 +08:00
|
|
|
netdev_set_prio_tc_map(netdev, i, kinfo->tc_info.prio_tc[i]);
|
2018-07-06 18:27:55 +08:00
|
|
|
|
2018-12-20 11:51:58 +08:00
|
|
|
if (h->ae_algo->ops->set_timer_task)
|
|
|
|
h->ae_algo->ops->set_timer_task(priv->ae_handle, true);
|
|
|
|
|
2019-02-23 17:22:08 +08:00
|
|
|
hns3_config_xps(priv);
|
2019-07-29 10:53:27 +08:00
|
|
|
|
|
|
|
netif_dbg(h, drv, netdev, "net open\n");
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-28 19:50:07 +08:00
|
|
|
static void hns3_reset_tx_queue(struct hnae3_handle *h)
|
|
|
|
{
|
|
|
|
struct net_device *ndev = h->kinfo.netdev;
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(ndev);
|
|
|
|
struct netdev_queue *dev_queue;
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++) {
|
|
|
|
dev_queue = netdev_get_tx_queue(ndev,
|
2019-10-19 16:03:49 +08:00
|
|
|
priv->ring[i].queue_index);
|
2019-06-28 19:50:07 +08:00
|
|
|
netdev_tx_reset_queue(dev_queue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static void hns3_nic_net_down(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2018-11-07 12:06:11 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
const struct hnae3_ae_ops *ops;
|
|
|
|
int i;
|
|
|
|
|
2018-05-26 02:43:04 +08:00
|
|
|
/* disable vectors */
|
|
|
|
for (i = 0; i < priv->vector_num; i++)
|
|
|
|
hns3_vector_disable(&priv->tqp_vector[i]);
|
2018-11-07 12:06:11 +08:00
|
|
|
|
|
|
|
/* disable rcb */
|
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++)
|
|
|
|
hns3_tqp_disable(h->kinfo.tqp[i]);
|
2018-05-26 02:43:04 +08:00
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* stop ae_dev */
|
|
|
|
ops = priv->ae_handle->ae_algo->ops;
|
|
|
|
if (ops->stop)
|
|
|
|
ops->stop(priv->ae_handle);
|
|
|
|
|
2019-06-13 17:12:31 +08:00
|
|
|
/* delay ring buffer clearing to hns3_reset_notify_uninit_enet
|
|
|
|
* during reset process, because driver may not be able
|
|
|
|
* to disable the ring through firmware when downing the netdev.
|
|
|
|
*/
|
|
|
|
if (!hns3_nic_resetting(netdev))
|
2019-06-28 19:50:07 +08:00
|
|
|
hns3_clear_all_ring(priv->ae_handle, false);
|
|
|
|
|
|
|
|
hns3_reset_tx_queue(priv->ae_handle);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_net_stop(struct net_device *netdev)
|
|
|
|
{
|
2018-11-09 22:07:52 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2018-12-20 11:51:58 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2018-11-09 22:07:52 +08:00
|
|
|
|
|
|
|
if (test_and_set_bit(HNS3_NIC_STATE_DOWN, &priv->state))
|
|
|
|
return 0;
|
|
|
|
|
2019-07-29 10:53:27 +08:00
|
|
|
netif_dbg(h, drv, netdev, "net stop\n");
|
|
|
|
|
2018-12-20 11:51:58 +08:00
|
|
|
if (h->ae_algo->ops->set_timer_task)
|
|
|
|
h->ae_algo->ops->set_timer_task(priv->ae_handle, false);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
netif_carrier_off(netdev);
|
2021-04-30 17:06:20 +08:00
|
|
|
netif_tx_disable(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
hns3_nic_net_down(netdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_uc_sync(struct net_device *netdev,
|
|
|
|
const unsigned char *addr)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
if (h->ae_algo->ops->add_uc_addr)
|
|
|
|
return h->ae_algo->ops->add_uc_addr(h, addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_uc_unsync(struct net_device *netdev,
|
|
|
|
const unsigned char *addr)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
net: hns3: refactor the MAC address configure
Currently, the HNS3 driver sync and unsync MAC address in function
hns3_set_rx_mode(). For PF, it adds and deletes MAC address directly
in the path of dev_set_rx_mode(). If failed, it won't retry until
next calling of hns3_set_rx_mode(). On the other hand, if request
add and remove a same address many times at a short interval, each
request must be done one by one, can't be merged. For VF, it sends
mailbox messages to PF to request adding or deleting MAC address in
the path of function hns3_set_rx_mode(), no matter the address is
configured success.
This patch refines it by recording the MAC address in function
hns3_set_rx_mode(), and updating MAC address in the service task.
If failed, it will retry by the next calling of periodical service
task. It also uses some state to mark the state of each MAC address
in the MAC list, which can help merge configure request for a same
address. With these changes, when global reset or IMP reset occurs,
we can restore the MAC table with the MAC list.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2020-04-24 10:23:09 +08:00
|
|
|
/* need ignore the request of removing device address, because
|
|
|
|
* we store the device address and other addresses of uc list
|
|
|
|
* in the function's mac filter list.
|
|
|
|
*/
|
|
|
|
if (ether_addr_equal(addr, netdev->dev_addr))
|
|
|
|
return 0;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
if (h->ae_algo->ops->rm_uc_addr)
|
|
|
|
return h->ae_algo->ops->rm_uc_addr(h, addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_mc_sync(struct net_device *netdev,
|
|
|
|
const unsigned char *addr)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2017-08-10 17:56:14 +08:00
|
|
|
if (h->ae_algo->ops->add_mc_addr)
|
2017-08-02 23:59:45 +08:00
|
|
|
return h->ae_algo->ops->add_mc_addr(h, addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_mc_unsync(struct net_device *netdev,
|
|
|
|
const unsigned char *addr)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2017-08-10 17:56:14 +08:00
|
|
|
if (h->ae_algo->ops->rm_mc_addr)
|
2017-08-02 23:59:45 +08:00
|
|
|
return h->ae_algo->ops->rm_mc_addr(h, addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-12 22:34:04 +08:00
|
|
|
static u8 hns3_get_netdev_flags(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
u8 flags = 0;
|
|
|
|
|
|
|
|
if (netdev->flags & IFF_PROMISC) {
|
2019-01-27 00:49:14 +08:00
|
|
|
flags = HNAE3_USER_UPE | HNAE3_USER_MPE | HNAE3_BPE;
|
2018-10-12 22:34:04 +08:00
|
|
|
} else {
|
|
|
|
flags |= HNAE3_VLAN_FLTR;
|
|
|
|
if (netdev->flags & IFF_ALLMULTI)
|
|
|
|
flags |= HNAE3_USER_MPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
2017-10-09 15:44:01 +08:00
|
|
|
static void hns3_nic_set_rx_mode(struct net_device *netdev)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2018-10-12 22:34:04 +08:00
|
|
|
u8 new_flags;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-10-12 22:34:04 +08:00
|
|
|
new_flags = hns3_get_netdev_flags(netdev);
|
|
|
|
|
2020-04-24 10:23:11 +08:00
|
|
|
__dev_uc_sync(netdev, hns3_nic_uc_sync, hns3_nic_uc_unsync);
|
|
|
|
__dev_mc_sync(netdev, hns3_nic_mc_sync, hns3_nic_mc_unsync);
|
2018-10-12 22:34:04 +08:00
|
|
|
|
|
|
|
/* User mode Promisc mode enable and vlan filtering is disabled to
|
2020-04-24 10:23:11 +08:00
|
|
|
* let all packets in.
|
2018-10-12 22:34:04 +08:00
|
|
|
*/
|
|
|
|
h->netdev_flags = new_flags;
|
2020-04-24 10:23:11 +08:00
|
|
|
hns3_request_update_promisc_mode(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hns3_request_update_promisc_mode(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
const struct hnae3_ae_ops *ops = handle->ae_algo->ops;
|
|
|
|
|
|
|
|
if (ops->request_update_promisc_mode)
|
|
|
|
ops->request_update_promisc_mode(handle);
|
2018-10-12 22:34:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void hns3_enable_vlan_filter(struct net_device *netdev, bool enable)
|
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
2020-09-27 15:12:39 +08:00
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(h->pdev);
|
2018-10-12 22:34:04 +08:00
|
|
|
bool last_state;
|
|
|
|
|
2020-09-27 15:12:39 +08:00
|
|
|
if (ae_dev->dev_version >= HNAE3_DEVICE_VERSION_V2 &&
|
|
|
|
h->ae_algo->ops->enable_vlan_filter) {
|
2018-10-12 22:34:04 +08:00
|
|
|
last_state = h->netdev_flags & HNAE3_VLAN_FLTR ? true : false;
|
|
|
|
if (enable != last_state) {
|
|
|
|
netdev_info(netdev,
|
|
|
|
"%s vlan filter\n",
|
|
|
|
enable ? "enable" : "disable");
|
|
|
|
h->ae_algo->ops->enable_vlan_filter(h, enable);
|
|
|
|
}
|
2018-06-02 00:52:10 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-11-28 11:51:47 +08:00
|
|
|
static int hns3_set_tso(struct sk_buff *skb, u32 *paylen_fdop_ol4cs,
|
2021-03-29 11:57:51 +08:00
|
|
|
u16 *mss, u32 *type_cs_vlan_tso, u32 *send_bytes)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
u32 l4_offset, hdr_len;
|
|
|
|
union l3_hdr_info l3;
|
|
|
|
union l4_hdr_info l4;
|
|
|
|
u32 l4_paylen;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!skb_is_gso(skb))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = skb_cow_head(skb, 0);
|
2019-10-08 09:20:09 +08:00
|
|
|
if (unlikely(ret < 0))
|
2017-08-02 23:59:45 +08:00
|
|
|
return ret;
|
|
|
|
|
|
|
|
l3.hdr = skb_network_header(skb);
|
|
|
|
l4.hdr = skb_transport_header(skb);
|
|
|
|
|
|
|
|
/* Software should clear the IPv4's checksum field when tso is
|
|
|
|
* needed.
|
|
|
|
*/
|
|
|
|
if (l3.v4->version == 4)
|
|
|
|
l3.v4->check = 0;
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
/* tunnel packet */
|
2017-08-02 23:59:45 +08:00
|
|
|
if (skb_shinfo(skb)->gso_type & (SKB_GSO_GRE |
|
|
|
|
SKB_GSO_GRE_CSUM |
|
2020-11-28 11:51:47 +08:00
|
|
|
SKB_GSO_UDP_TUNNEL |
|
|
|
|
SKB_GSO_UDP_TUNNEL_CSUM)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
/* reset l3&l4 pointers from outer to inner headers */
|
|
|
|
l3.hdr = skb_inner_network_header(skb);
|
|
|
|
l4.hdr = skb_inner_transport_header(skb);
|
|
|
|
|
|
|
|
/* Software should clear the IPv4's checksum field when
|
|
|
|
* tso is needed.
|
|
|
|
*/
|
|
|
|
if (l3.v4->version == 4)
|
|
|
|
l3.v4->check = 0;
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
/* normal or tunnel packet */
|
2017-08-02 23:59:45 +08:00
|
|
|
l4_offset = l4.hdr - skb->data;
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
/* remove payload length from inner pseudo checksum when tso */
|
2017-08-02 23:59:45 +08:00
|
|
|
l4_paylen = skb->len - l4_offset;
|
2020-09-29 17:32:01 +08:00
|
|
|
|
|
|
|
if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_L4) {
|
|
|
|
hdr_len = sizeof(*l4.udp) + l4_offset;
|
|
|
|
csum_replace_by_diff(&l4.udp->check,
|
|
|
|
(__force __wsum)htonl(l4_paylen));
|
|
|
|
} else {
|
|
|
|
hdr_len = (l4.tcp->doff << 2) + l4_offset;
|
|
|
|
csum_replace_by_diff(&l4.tcp->check,
|
|
|
|
(__force __wsum)htonl(l4_paylen));
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2021-03-29 11:57:51 +08:00
|
|
|
*send_bytes = (skb_shinfo(skb)->gso_segs - 1) * hdr_len + skb->len;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* find the txbd field values */
|
2020-11-28 11:51:47 +08:00
|
|
|
*paylen_fdop_ol4cs = skb->len - hdr_len;
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_TSO_B, 1);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-11-28 11:51:47 +08:00
|
|
|
/* offload outer UDP header checksum */
|
|
|
|
if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM)
|
|
|
|
hns3_set_field(*paylen_fdop_ol4cs, HNS3_TXD_OL4CS_B, 1);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* get MSS for TSO */
|
|
|
|
*mss = skb_shinfo(skb)->gso_size;
|
|
|
|
|
2020-01-04 10:49:24 +08:00
|
|
|
trace_hns3_tso(skb);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-18 19:31:39 +08:00
|
|
|
static int hns3_get_l4_protocol(struct sk_buff *skb, u8 *ol4_proto,
|
|
|
|
u8 *il4_proto)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-01-31 04:55:41 +08:00
|
|
|
union l3_hdr_info l3;
|
2017-08-02 23:59:45 +08:00
|
|
|
unsigned char *l4_hdr;
|
|
|
|
unsigned char *exthdr;
|
|
|
|
u8 l4_proto_tmp;
|
|
|
|
__be16 frag_off;
|
|
|
|
|
|
|
|
/* find outer header point */
|
|
|
|
l3.hdr = skb_network_header(skb);
|
2018-05-02 02:55:58 +08:00
|
|
|
l4_hdr = skb_transport_header(skb);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
if (skb->protocol == htons(ETH_P_IPV6)) {
|
|
|
|
exthdr = l3.hdr + sizeof(*l3.v6);
|
|
|
|
l4_proto_tmp = l3.v6->nexthdr;
|
|
|
|
if (l4_hdr != exthdr)
|
|
|
|
ipv6_skip_exthdr(skb, exthdr - skb->data,
|
|
|
|
&l4_proto_tmp, &frag_off);
|
|
|
|
} else if (skb->protocol == htons(ETH_P_IP)) {
|
|
|
|
l4_proto_tmp = l3.v4->protocol;
|
2017-08-18 19:31:39 +08:00
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
*ol4_proto = l4_proto_tmp;
|
|
|
|
|
|
|
|
/* tunnel packet */
|
|
|
|
if (!skb->encapsulation) {
|
|
|
|
*il4_proto = 0;
|
2017-08-18 19:31:39 +08:00
|
|
|
return 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* find inner header point */
|
|
|
|
l3.hdr = skb_inner_network_header(skb);
|
|
|
|
l4_hdr = skb_inner_transport_header(skb);
|
|
|
|
|
|
|
|
if (l3.v6->version == 6) {
|
|
|
|
exthdr = l3.hdr + sizeof(*l3.v6);
|
|
|
|
l4_proto_tmp = l3.v6->nexthdr;
|
|
|
|
if (l4_hdr != exthdr)
|
|
|
|
ipv6_skip_exthdr(skb, exthdr - skb->data,
|
|
|
|
&l4_proto_tmp, &frag_off);
|
|
|
|
} else if (l3.v4->version == 4) {
|
|
|
|
l4_proto_tmp = l3.v4->protocol;
|
|
|
|
}
|
|
|
|
|
|
|
|
*il4_proto = l4_proto_tmp;
|
2017-08-18 19:31:39 +08:00
|
|
|
|
|
|
|
return 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2018-06-02 00:52:09 +08:00
|
|
|
/* when skb->encapsulation is 0, skb->ip_summed is CHECKSUM_PARTIAL
|
|
|
|
* and it is udp packet, which has a dest port as the IANA assigned.
|
|
|
|
* the hardware is expected to do the checksum offload, but the
|
|
|
|
* hardware will not do the checksum offload when udp dest port is
|
2021-04-30 17:06:19 +08:00
|
|
|
* 4789, 4790 or 6081.
|
2018-06-02 00:52:09 +08:00
|
|
|
*/
|
|
|
|
static bool hns3_tunnel_csum_bug(struct sk_buff *skb)
|
|
|
|
{
|
2020-11-28 11:51:49 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(skb->dev);
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(priv->ae_handle->pdev);
|
2019-01-31 04:55:41 +08:00
|
|
|
union l4_hdr_info l4;
|
2018-06-02 00:52:09 +08:00
|
|
|
|
2020-11-28 11:51:49 +08:00
|
|
|
/* device version above V3(include V3), the hardware can
|
|
|
|
* do this checksum offload.
|
|
|
|
*/
|
|
|
|
if (ae_dev->dev_version >= HNAE3_DEVICE_VERSION_V3)
|
|
|
|
return false;
|
|
|
|
|
2018-06-02 00:52:09 +08:00
|
|
|
l4.hdr = skb_transport_header(skb);
|
|
|
|
|
2019-03-22 06:51:39 +08:00
|
|
|
if (!(!skb->encapsulation &&
|
2020-08-26 13:11:50 +08:00
|
|
|
(l4.udp->dest == htons(IANA_VXLAN_UDP_PORT) ||
|
2021-04-30 17:06:19 +08:00
|
|
|
l4.udp->dest == htons(GENEVE_UDP_PORT) ||
|
|
|
|
l4.udp->dest == htons(4790))))
|
2018-06-02 00:52:09 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
skb_checksum_help(skb);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:47 +08:00
|
|
|
static void hns3_set_outer_l2l3l4(struct sk_buff *skb, u8 ol4_proto,
|
|
|
|
u32 *ol_type_vlan_len_msec)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-05-06 10:48:47 +08:00
|
|
|
u32 l2_len, l3_len, l4_len;
|
|
|
|
unsigned char *il2_hdr;
|
2019-01-31 04:55:41 +08:00
|
|
|
union l3_hdr_info l3;
|
2019-05-06 10:48:47 +08:00
|
|
|
union l4_hdr_info l4;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
l3.hdr = skb_network_header(skb);
|
2019-05-06 10:48:47 +08:00
|
|
|
l4.hdr = skb_transport_header(skb);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:47 +08:00
|
|
|
/* compute OL2 header size, defined in 2 Bytes */
|
|
|
|
l2_len = l3.hdr - skb->data;
|
|
|
|
hns3_set_field(*ol_type_vlan_len_msec, HNS3_TXD_L2LEN_S, l2_len >> 1);
|
|
|
|
|
|
|
|
/* compute OL3 header size, defined in 4 Bytes */
|
|
|
|
l3_len = l4.hdr - l3.hdr;
|
|
|
|
hns3_set_field(*ol_type_vlan_len_msec, HNS3_TXD_L3LEN_S, l3_len >> 2);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:47 +08:00
|
|
|
il2_hdr = skb_inner_mac_header(skb);
|
2019-06-07 10:03:13 +08:00
|
|
|
/* compute OL4 header size, defined in 4 Bytes */
|
2019-05-06 10:48:47 +08:00
|
|
|
l4_len = il2_hdr - l4.hdr;
|
|
|
|
hns3_set_field(*ol_type_vlan_len_msec, HNS3_TXD_L4LEN_S, l4_len >> 2);
|
|
|
|
|
|
|
|
/* define outer network header type */
|
|
|
|
if (skb->protocol == htons(ETH_P_IP)) {
|
|
|
|
if (skb_is_gso(skb))
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*ol_type_vlan_len_msec,
|
2019-05-06 10:48:47 +08:00
|
|
|
HNS3_TXD_OL3T_S,
|
|
|
|
HNS3_OL3T_IPV4_CSUM);
|
|
|
|
else
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*ol_type_vlan_len_msec,
|
2019-05-06 10:48:47 +08:00
|
|
|
HNS3_TXD_OL3T_S,
|
|
|
|
HNS3_OL3T_IPV4_NO_CSUM);
|
|
|
|
} else if (skb->protocol == htons(ETH_P_IPV6)) {
|
|
|
|
hns3_set_field(*ol_type_vlan_len_msec, HNS3_TXD_OL3T_S,
|
|
|
|
HNS3_OL3T_IPV6);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ol4_proto == IPPROTO_UDP)
|
|
|
|
hns3_set_field(*ol_type_vlan_len_msec, HNS3_TXD_TUNTYPE_S,
|
|
|
|
HNS3_TUN_MAC_IN_UDP);
|
|
|
|
else if (ol4_proto == IPPROTO_GRE)
|
|
|
|
hns3_set_field(*ol_type_vlan_len_msec, HNS3_TXD_TUNTYPE_S,
|
|
|
|
HNS3_TUN_NVGRE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_set_l2l3l4(struct sk_buff *skb, u8 ol4_proto,
|
|
|
|
u8 il4_proto, u32 *type_cs_vlan_tso,
|
|
|
|
u32 *ol_type_vlan_len_msec)
|
|
|
|
{
|
2019-05-08 18:51:35 +08:00
|
|
|
unsigned char *l2_hdr = skb->data;
|
2019-05-06 10:48:47 +08:00
|
|
|
u32 l4_proto = ol4_proto;
|
|
|
|
union l4_hdr_info l4;
|
|
|
|
union l3_hdr_info l3;
|
|
|
|
u32 l2_len, l3_len;
|
|
|
|
|
|
|
|
l4.hdr = skb_transport_header(skb);
|
|
|
|
l3.hdr = skb_network_header(skb);
|
|
|
|
|
|
|
|
/* handle encapsulation skb */
|
|
|
|
if (skb->encapsulation) {
|
|
|
|
/* If this is a not UDP/GRE encapsulation skb */
|
|
|
|
if (!(ol4_proto == IPPROTO_UDP || ol4_proto == IPPROTO_GRE)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
/* drop the skb tunnel packet if hardware don't support,
|
|
|
|
* because hardware can't calculate csum when TSO.
|
|
|
|
*/
|
|
|
|
if (skb_is_gso(skb))
|
|
|
|
return -EDOM;
|
|
|
|
|
|
|
|
/* the stack computes the IP header already,
|
|
|
|
* driver calculate l4 checksum when not TSO.
|
|
|
|
*/
|
|
|
|
skb_checksum_help(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:47 +08:00
|
|
|
hns3_set_outer_l2l3l4(skb, ol4_proto, ol_type_vlan_len_msec);
|
|
|
|
|
|
|
|
/* switch to inner header */
|
|
|
|
l2_hdr = skb_inner_mac_header(skb);
|
2017-08-02 23:59:45 +08:00
|
|
|
l3.hdr = skb_inner_network_header(skb);
|
2019-05-06 10:48:47 +08:00
|
|
|
l4.hdr = skb_inner_transport_header(skb);
|
2017-08-02 23:59:45 +08:00
|
|
|
l4_proto = il4_proto;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l3.v4->version == 4) {
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L3T_S,
|
|
|
|
HNS3_L3T_IPV4);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
/* the stack computes the IP header already, the only time we
|
|
|
|
* need the hardware to recompute it is in the case of TSO.
|
|
|
|
*/
|
|
|
|
if (skb_is_gso(skb))
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L3CS_B, 1);
|
2017-08-02 23:59:45 +08:00
|
|
|
} else if (l3.v6->version == 6) {
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L3T_S,
|
|
|
|
HNS3_L3T_IPV6);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:47 +08:00
|
|
|
/* compute inner(/normal) L2 header size, defined in 2 Bytes */
|
|
|
|
l2_len = l3.hdr - l2_hdr;
|
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L2LEN_S, l2_len >> 1);
|
|
|
|
|
|
|
|
/* compute inner(/normal) L3 header size, defined in 4 Bytes */
|
|
|
|
l3_len = l4.hdr - l3.hdr;
|
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L3LEN_S, l3_len >> 2);
|
|
|
|
|
|
|
|
/* compute inner(/normal) L4 header size, defined in 4 Bytes */
|
2017-08-02 23:59:45 +08:00
|
|
|
switch (l4_proto) {
|
|
|
|
case IPPROTO_TCP:
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4CS_B, 1);
|
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4T_S,
|
|
|
|
HNS3_L4T_TCP);
|
2019-05-06 10:48:47 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4LEN_S,
|
|
|
|
l4.tcp->doff);
|
2017-08-02 23:59:45 +08:00
|
|
|
break;
|
|
|
|
case IPPROTO_UDP:
|
2018-06-02 00:52:09 +08:00
|
|
|
if (hns3_tunnel_csum_bug(skb))
|
|
|
|
break;
|
|
|
|
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4CS_B, 1);
|
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4T_S,
|
|
|
|
HNS3_L4T_UDP);
|
2019-05-06 10:48:47 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4LEN_S,
|
|
|
|
(sizeof(struct udphdr) >> 2));
|
2017-08-02 23:59:45 +08:00
|
|
|
break;
|
|
|
|
case IPPROTO_SCTP:
|
2019-02-23 17:22:13 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4CS_B, 1);
|
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4T_S,
|
|
|
|
HNS3_L4T_SCTP);
|
2019-05-06 10:48:47 +08:00
|
|
|
hns3_set_field(*type_cs_vlan_tso, HNS3_TXD_L4LEN_S,
|
|
|
|
(sizeof(struct sctphdr) >> 2));
|
2017-08-02 23:59:45 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* drop the skb tunnel packet if hardware don't support,
|
|
|
|
* because hardware can't calculate csum when TSO.
|
|
|
|
*/
|
|
|
|
if (skb_is_gso(skb))
|
|
|
|
return -EDOM;
|
|
|
|
|
|
|
|
/* the stack computes the IP header already,
|
|
|
|
* driver calculate l4 checksum when not TSO.
|
|
|
|
*/
|
|
|
|
skb_checksum_help(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-08-09 10:31:09 +08:00
|
|
|
static int hns3_handle_vtags(struct hns3_enet_ring *tx_ring,
|
|
|
|
struct sk_buff *skb)
|
2017-12-22 12:21:48 +08:00
|
|
|
{
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
struct hnae3_handle *handle = tx_ring->tqp->handle;
|
net: hns3: refine the VLAN tag handle for port based VLAN
For DEVICE_VERSION_V2, the hardware only supports max two layer
VLAN tags, including port based tag inserted by hardware, tag in
tx buffer descriptor(get from skb->tci) and tag in packet.
For transmit packet:
If port based VLAN disabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the Outer_VLAN_TAG field
(tag near to DMAC) of tx buffer descriptor, otherwise it may
be inserted after the tag in packet.
If port based VLAN enabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the VLAN_TAG field (tag
far to DMAC) of tx buffer descriptor, otherwise it may be
conflicted with port based VLAN, and raise a hardware error.
For receive packet:
The hardware will strip the VLAN tags and fill them in the rx
buffer descriptor, no matter port based VLAN enable or not.
Because port based VLAN tag is useless for stack, so vf driver
needs to discard the port based VLAN tag get from rx buffer
descriptor when port based VLAN enabled.
So vf must know about the port based VLAN state.
For DEVICE_VERSION_V3, the hardware provides some new
configuration to improve it.
For transmit packet:
When enable tag shift mode, hardware will handle the VLAN tag
in outer_VLAN_TAG field as VLAN_TAG, so it won't conflict with
port based VLAN. And hardware also make sure the tag before
the tag in packet. So vf driver doesn't need to specify the tag
position according to the port based VLAN state anymore.
For receive packet:
When enable discard mode, hardware will strip and discard the
port based VLAN tag, so vf driver doesn't need to identify it
from rx buffer descriptor.
So modify the port based VLAN configuration, simplify the process
for vf handling the VLAN tag.
Signed-off-by: Guojia Liao <liaoguojia@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2020-12-06 12:06:15 +08:00
|
|
|
struct hnae3_ae_dev *ae_dev;
|
2019-08-09 10:31:09 +08:00
|
|
|
struct vlan_ethhdr *vhdr;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (!(skb->protocol == htons(ETH_P_8021Q) ||
|
|
|
|
skb_vlan_tag_present(skb)))
|
|
|
|
return 0;
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
|
net: hns3: refine the VLAN tag handle for port based VLAN
For DEVICE_VERSION_V2, the hardware only supports max two layer
VLAN tags, including port based tag inserted by hardware, tag in
tx buffer descriptor(get from skb->tci) and tag in packet.
For transmit packet:
If port based VLAN disabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the Outer_VLAN_TAG field
(tag near to DMAC) of tx buffer descriptor, otherwise it may
be inserted after the tag in packet.
If port based VLAN enabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the VLAN_TAG field (tag
far to DMAC) of tx buffer descriptor, otherwise it may be
conflicted with port based VLAN, and raise a hardware error.
For receive packet:
The hardware will strip the VLAN tags and fill them in the rx
buffer descriptor, no matter port based VLAN enable or not.
Because port based VLAN tag is useless for stack, so vf driver
needs to discard the port based VLAN tag get from rx buffer
descriptor when port based VLAN enabled.
So vf must know about the port based VLAN state.
For DEVICE_VERSION_V3, the hardware provides some new
configuration to improve it.
For transmit packet:
When enable tag shift mode, hardware will handle the VLAN tag
in outer_VLAN_TAG field as VLAN_TAG, so it won't conflict with
port based VLAN. And hardware also make sure the tag before
the tag in packet. So vf driver doesn't need to specify the tag
position according to the port based VLAN state anymore.
For receive packet:
When enable discard mode, hardware will strip and discard the
port based VLAN tag, so vf driver doesn't need to identify it
from rx buffer descriptor.
So modify the port based VLAN configuration, simplify the process
for vf handling the VLAN tag.
Signed-off-by: Guojia Liao <liaoguojia@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2020-12-06 12:06:15 +08:00
|
|
|
/* For HW limitation on HNAE3_DEVICE_VERSION_V2, if port based insert
|
|
|
|
* VLAN enabled, only one VLAN header is allowed in skb, otherwise it
|
|
|
|
* will cause RAS error.
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
*/
|
net: hns3: refine the VLAN tag handle for port based VLAN
For DEVICE_VERSION_V2, the hardware only supports max two layer
VLAN tags, including port based tag inserted by hardware, tag in
tx buffer descriptor(get from skb->tci) and tag in packet.
For transmit packet:
If port based VLAN disabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the Outer_VLAN_TAG field
(tag near to DMAC) of tx buffer descriptor, otherwise it may
be inserted after the tag in packet.
If port based VLAN enabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the VLAN_TAG field (tag
far to DMAC) of tx buffer descriptor, otherwise it may be
conflicted with port based VLAN, and raise a hardware error.
For receive packet:
The hardware will strip the VLAN tags and fill them in the rx
buffer descriptor, no matter port based VLAN enable or not.
Because port based VLAN tag is useless for stack, so vf driver
needs to discard the port based VLAN tag get from rx buffer
descriptor when port based VLAN enabled.
So vf must know about the port based VLAN state.
For DEVICE_VERSION_V3, the hardware provides some new
configuration to improve it.
For transmit packet:
When enable tag shift mode, hardware will handle the VLAN tag
in outer_VLAN_TAG field as VLAN_TAG, so it won't conflict with
port based VLAN. And hardware also make sure the tag before
the tag in packet. So vf driver doesn't need to specify the tag
position according to the port based VLAN state anymore.
For receive packet:
When enable discard mode, hardware will strip and discard the
port based VLAN tag, so vf driver doesn't need to identify it
from rx buffer descriptor.
So modify the port based VLAN configuration, simplify the process
for vf handling the VLAN tag.
Signed-off-by: Guojia Liao <liaoguojia@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2020-12-06 12:06:15 +08:00
|
|
|
ae_dev = pci_get_drvdata(handle->pdev);
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
if (unlikely(skb_vlan_tagged_multi(skb) &&
|
net: hns3: refine the VLAN tag handle for port based VLAN
For DEVICE_VERSION_V2, the hardware only supports max two layer
VLAN tags, including port based tag inserted by hardware, tag in
tx buffer descriptor(get from skb->tci) and tag in packet.
For transmit packet:
If port based VLAN disabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the Outer_VLAN_TAG field
(tag near to DMAC) of tx buffer descriptor, otherwise it may
be inserted after the tag in packet.
If port based VLAN enabled, and vf driver gets a VLAN tag from
skb, the VLAN tag must be filled to the VLAN_TAG field (tag
far to DMAC) of tx buffer descriptor, otherwise it may be
conflicted with port based VLAN, and raise a hardware error.
For receive packet:
The hardware will strip the VLAN tags and fill them in the rx
buffer descriptor, no matter port based VLAN enable or not.
Because port based VLAN tag is useless for stack, so vf driver
needs to discard the port based VLAN tag get from rx buffer
descriptor when port based VLAN enabled.
So vf must know about the port based VLAN state.
For DEVICE_VERSION_V3, the hardware provides some new
configuration to improve it.
For transmit packet:
When enable tag shift mode, hardware will handle the VLAN tag
in outer_VLAN_TAG field as VLAN_TAG, so it won't conflict with
port based VLAN. And hardware also make sure the tag before
the tag in packet. So vf driver doesn't need to specify the tag
position according to the port based VLAN state anymore.
For receive packet:
When enable discard mode, hardware will strip and discard the
port based VLAN tag, so vf driver doesn't need to identify it
from rx buffer descriptor.
So modify the port based VLAN configuration, simplify the process
for vf handling the VLAN tag.
Signed-off-by: Guojia Liao <liaoguojia@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2020-12-06 12:06:15 +08:00
|
|
|
ae_dev->dev_version <= HNAE3_DEVICE_VERSION_V2 &&
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
handle->port_base_vlan_state ==
|
|
|
|
HNAE3_PORT_BASE_VLAN_ENABLE))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-12-22 12:21:48 +08:00
|
|
|
if (skb->protocol == htons(ETH_P_8021Q) &&
|
2019-08-09 10:31:09 +08:00
|
|
|
!(handle->kinfo.netdev->features & NETIF_F_HW_VLAN_CTAG_TX)) {
|
2017-12-22 12:21:48 +08:00
|
|
|
/* When HW VLAN acceleration is turned off, and the stack
|
|
|
|
* sets the protocol to 802.1q, the driver just need to
|
|
|
|
* set the protocol to the encapsulated ethertype.
|
|
|
|
*/
|
|
|
|
skb->protocol = vlan_get_protocol(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (skb_vlan_tag_present(skb)) {
|
|
|
|
/* Based on hw strategy, use out_vtag in two layer tag case,
|
|
|
|
* and use inner_vtag in one tag case.
|
|
|
|
*/
|
2019-08-09 10:31:09 +08:00
|
|
|
if (skb->protocol == htons(ETH_P_8021Q) &&
|
|
|
|
handle->port_base_vlan_state ==
|
|
|
|
HNAE3_PORT_BASE_VLAN_DISABLE)
|
|
|
|
rc = HNS3_OUTER_VLAN_TAG;
|
|
|
|
else
|
|
|
|
rc = HNS3_INNER_VLAN_TAG;
|
|
|
|
|
|
|
|
skb->protocol = vlan_get_protocol(skb);
|
|
|
|
return rc;
|
2017-12-22 12:21:48 +08:00
|
|
|
}
|
|
|
|
|
2019-08-09 10:31:09 +08:00
|
|
|
rc = skb_cow_head(skb, 0);
|
|
|
|
if (unlikely(rc < 0))
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
vhdr = (struct vlan_ethhdr *)skb->data;
|
|
|
|
vhdr->h_vlan_TCI |= cpu_to_be16((skb->priority << VLAN_PRIO_SHIFT)
|
|
|
|
& VLAN_PRIO_MASK);
|
|
|
|
|
2017-12-22 12:21:48 +08:00
|
|
|
skb->protocol = vlan_get_protocol(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
/* check if the hardware is capable of checksum offloading */
|
|
|
|
static bool hns3_check_hw_tx_csum(struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(skb->dev);
|
|
|
|
|
|
|
|
/* Kindly note, due to backward compatibility of the TX descriptor,
|
|
|
|
* HW checksum of the non-IP packets and GSO packets is handled at
|
|
|
|
* different place in the following code
|
|
|
|
*/
|
2021-01-22 17:31:01 +08:00
|
|
|
if (skb_csum_is_sctp(skb) || skb_is_gso(skb) ||
|
2020-11-28 11:51:45 +08:00
|
|
|
!test_bit(HNS3_NIC_STATE_HW_TX_CSUM_ENABLE, &priv->state))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-08-09 10:31:09 +08:00
|
|
|
static int hns3_fill_skb_desc(struct hns3_enet_ring *ring,
|
2021-03-29 11:57:51 +08:00
|
|
|
struct sk_buff *skb, struct hns3_desc *desc,
|
|
|
|
struct hns3_desc_cb *desc_cb)
|
2019-08-09 10:31:09 +08:00
|
|
|
{
|
|
|
|
u32 ol_type_vlan_len_msec = 0;
|
2020-11-28 11:51:47 +08:00
|
|
|
u32 paylen_ol4cs = skb->len;
|
2019-08-09 10:31:09 +08:00
|
|
|
u32 type_cs_vlan_tso = 0;
|
2020-11-28 11:51:45 +08:00
|
|
|
u16 mss_hw_csum = 0;
|
2019-08-09 10:31:09 +08:00
|
|
|
u16 inner_vtag = 0;
|
|
|
|
u16 out_vtag = 0;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = hns3_handle_vtags(ring, skb);
|
|
|
|
if (unlikely(ret < 0)) {
|
2019-08-09 10:31:13 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_vlan_err++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
2019-08-09 10:31:09 +08:00
|
|
|
return ret;
|
|
|
|
} else if (ret == HNS3_INNER_VLAN_TAG) {
|
|
|
|
inner_vtag = skb_vlan_tag_get(skb);
|
|
|
|
inner_vtag |= (skb->priority << VLAN_PRIO_SHIFT) &
|
|
|
|
VLAN_PRIO_MASK;
|
|
|
|
hns3_set_field(type_cs_vlan_tso, HNS3_TXD_VLAN_B, 1);
|
|
|
|
} else if (ret == HNS3_OUTER_VLAN_TAG) {
|
|
|
|
out_vtag = skb_vlan_tag_get(skb);
|
|
|
|
out_vtag |= (skb->priority << VLAN_PRIO_SHIFT) &
|
|
|
|
VLAN_PRIO_MASK;
|
|
|
|
hns3_set_field(ol_type_vlan_len_msec, HNS3_TXD_OVLAN_B,
|
|
|
|
1);
|
|
|
|
}
|
|
|
|
|
2021-03-29 11:57:51 +08:00
|
|
|
desc_cb->send_bytes = skb->len;
|
|
|
|
|
2019-08-09 10:31:09 +08:00
|
|
|
if (skb->ip_summed == CHECKSUM_PARTIAL) {
|
|
|
|
u8 ol4_proto, il4_proto;
|
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
if (hns3_check_hw_tx_csum(skb)) {
|
|
|
|
/* set checksum start and offset, defined in 2 Bytes */
|
|
|
|
hns3_set_field(type_cs_vlan_tso, HNS3_TXD_CSUM_START_S,
|
|
|
|
skb_checksum_start_offset(skb) >> 1);
|
|
|
|
hns3_set_field(ol_type_vlan_len_msec,
|
|
|
|
HNS3_TXD_CSUM_OFFSET_S,
|
|
|
|
skb->csum_offset >> 1);
|
|
|
|
mss_hw_csum |= BIT(HNS3_TXD_HW_CS_B);
|
|
|
|
goto out_hw_tx_csum;
|
|
|
|
}
|
|
|
|
|
2019-08-09 10:31:09 +08:00
|
|
|
skb_reset_mac_len(skb);
|
|
|
|
|
|
|
|
ret = hns3_get_l4_protocol(skb, &ol4_proto, &il4_proto);
|
2019-10-08 09:20:09 +08:00
|
|
|
if (unlikely(ret < 0)) {
|
2019-08-09 10:31:13 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_l4_proto_err++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
2019-08-09 10:31:09 +08:00
|
|
|
return ret;
|
2019-08-09 10:31:13 +08:00
|
|
|
}
|
2019-08-09 10:31:09 +08:00
|
|
|
|
|
|
|
ret = hns3_set_l2l3l4(skb, ol4_proto, il4_proto,
|
|
|
|
&type_cs_vlan_tso,
|
|
|
|
&ol_type_vlan_len_msec);
|
2019-10-08 09:20:09 +08:00
|
|
|
if (unlikely(ret < 0)) {
|
2019-08-09 10:31:13 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_l2l3l4_err++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
2019-08-09 10:31:09 +08:00
|
|
|
return ret;
|
2019-08-09 10:31:13 +08:00
|
|
|
}
|
2019-08-09 10:31:09 +08:00
|
|
|
|
2020-11-28 11:51:47 +08:00
|
|
|
ret = hns3_set_tso(skb, &paylen_ol4cs, &mss_hw_csum,
|
2021-03-29 11:57:51 +08:00
|
|
|
&type_cs_vlan_tso, &desc_cb->send_bytes);
|
2019-10-08 09:20:09 +08:00
|
|
|
if (unlikely(ret < 0)) {
|
2019-08-09 10:31:13 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_tso_err++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
2019-08-09 10:31:09 +08:00
|
|
|
return ret;
|
2019-08-09 10:31:13 +08:00
|
|
|
}
|
2019-08-09 10:31:09 +08:00
|
|
|
}
|
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
out_hw_tx_csum:
|
2019-08-09 10:31:09 +08:00
|
|
|
/* Set txbd */
|
|
|
|
desc->tx.ol_type_vlan_len_msec =
|
|
|
|
cpu_to_le32(ol_type_vlan_len_msec);
|
|
|
|
desc->tx.type_cs_vlan_tso_len = cpu_to_le32(type_cs_vlan_tso);
|
2020-11-28 11:51:47 +08:00
|
|
|
desc->tx.paylen_ol4cs = cpu_to_le32(paylen_ol4cs);
|
2020-11-28 11:51:45 +08:00
|
|
|
desc->tx.mss_hw_csum = cpu_to_le16(mss_hw_csum);
|
2019-08-09 10:31:09 +08:00
|
|
|
desc->tx.vlan_tag = cpu_to_le16(inner_vtag);
|
|
|
|
desc->tx.outer_vlan_tag = cpu_to_le16(out_vtag);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_fill_desc(struct hns3_enet_ring *ring, void *priv,
|
2019-10-08 09:20:09 +08:00
|
|
|
unsigned int size, enum hns_desc_type type)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-10-08 09:20:09 +08:00
|
|
|
#define HNS3_LIKELY_BD_NUM 1
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_desc_cb *desc_cb = &ring->desc_cb[ring->next_to_use];
|
|
|
|
struct hns3_desc *desc = &ring->desc[ring->next_to_use];
|
2018-10-16 19:58:49 +08:00
|
|
|
struct device *dev = ring_to_dev(ring);
|
2019-07-23 11:08:25 +08:00
|
|
|
skb_frag_t *frag;
|
2018-10-16 19:58:50 +08:00
|
|
|
unsigned int frag_buf_num;
|
2019-02-23 17:22:10 +08:00
|
|
|
int k, sizeoflast;
|
2018-10-16 19:58:49 +08:00
|
|
|
dma_addr_t dma;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-07-28 10:16:48 +08:00
|
|
|
if (type == DESC_TYPE_FRAGLIST_SKB ||
|
|
|
|
type == DESC_TYPE_SKB) {
|
2020-03-28 15:09:56 +08:00
|
|
|
struct sk_buff *skb = (struct sk_buff *)priv;
|
|
|
|
|
2018-10-16 19:58:49 +08:00
|
|
|
dma = dma_map_single(dev, skb->data, size, DMA_TO_DEVICE);
|
|
|
|
} else {
|
2019-07-23 11:08:25 +08:00
|
|
|
frag = (skb_frag_t *)priv;
|
2018-10-16 19:58:49 +08:00
|
|
|
dma = skb_frag_dma_map(dev, frag, 0, size, DMA_TO_DEVICE);
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:51 +08:00
|
|
|
if (unlikely(dma_mapping_error(dev, dma))) {
|
2019-08-09 10:31:13 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
2018-10-16 19:58:49 +08:00
|
|
|
ring->stats.sw_err_cnt++;
|
2019-08-09 10:31:13 +08:00
|
|
|
u64_stats_update_end(&ring->syncp);
|
2018-10-16 19:58:49 +08:00
|
|
|
return -ENOMEM;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-07-21 19:03:51 +08:00
|
|
|
desc_cb->priv = priv;
|
2018-10-16 19:58:52 +08:00
|
|
|
desc_cb->length = size;
|
2020-07-21 19:03:51 +08:00
|
|
|
desc_cb->dma = dma;
|
|
|
|
desc_cb->type = type;
|
2018-10-16 19:58:52 +08:00
|
|
|
|
2019-04-04 16:17:53 +08:00
|
|
|
if (likely(size <= HNS3_MAX_BD_SIZE)) {
|
|
|
|
desc->addr = cpu_to_le64(dma);
|
|
|
|
desc->tx.send_size = cpu_to_le16(size);
|
|
|
|
desc->tx.bdtp_fe_sc_vld_ra_ri =
|
2019-10-08 09:20:09 +08:00
|
|
|
cpu_to_le16(BIT(HNS3_TXD_VLD_B));
|
2019-04-04 16:17:53 +08:00
|
|
|
|
2020-01-04 10:49:24 +08:00
|
|
|
trace_hns3_tx_desc(ring, ring->next_to_use);
|
2019-04-04 16:17:53 +08:00
|
|
|
ring_ptr_move_fw(ring, next_to_use);
|
2019-10-08 09:20:09 +08:00
|
|
|
return HNS3_LIKELY_BD_NUM;
|
2019-04-04 16:17:53 +08:00
|
|
|
}
|
|
|
|
|
2019-03-21 11:28:43 +08:00
|
|
|
frag_buf_num = hns3_tx_bd_count(size);
|
2020-07-21 19:03:52 +08:00
|
|
|
sizeoflast = size % HNS3_MAX_BD_SIZE;
|
2018-10-16 19:58:50 +08:00
|
|
|
sizeoflast = sizeoflast ? sizeoflast : HNS3_MAX_BD_SIZE;
|
|
|
|
|
|
|
|
/* When frag size is bigger than hardware limit, split this frag */
|
|
|
|
for (k = 0; k < frag_buf_num; k++) {
|
|
|
|
/* now, fill the descriptor */
|
|
|
|
desc->addr = cpu_to_le64(dma + HNS3_MAX_BD_SIZE * k);
|
2018-10-16 19:58:52 +08:00
|
|
|
desc->tx.send_size = cpu_to_le16((k == frag_buf_num - 1) ?
|
2019-06-07 10:03:13 +08:00
|
|
|
(u16)sizeoflast : (u16)HNS3_MAX_BD_SIZE);
|
2018-10-16 19:58:50 +08:00
|
|
|
desc->tx.bdtp_fe_sc_vld_ra_ri =
|
2019-10-08 09:20:09 +08:00
|
|
|
cpu_to_le16(BIT(HNS3_TXD_VLD_B));
|
2018-10-16 19:58:50 +08:00
|
|
|
|
2020-01-04 10:49:24 +08:00
|
|
|
trace_hns3_tx_desc(ring, ring->next_to_use);
|
2019-06-07 10:03:13 +08:00
|
|
|
/* move ring pointer to next */
|
2018-10-16 19:58:50 +08:00
|
|
|
ring_ptr_move_fw(ring, next_to_use);
|
|
|
|
|
|
|
|
desc = &ring->desc[ring->next_to_use];
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
return frag_buf_num;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
static unsigned int hns3_skb_bd_num(struct sk_buff *skb, unsigned int *bd_size,
|
|
|
|
unsigned int bd_num)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-10-08 09:20:09 +08:00
|
|
|
unsigned int size;
|
2019-08-09 10:31:14 +08:00
|
|
|
int i;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
size = skb_headlen(skb);
|
|
|
|
while (size > HNS3_MAX_BD_SIZE) {
|
|
|
|
bd_size[bd_num++] = HNS3_MAX_BD_SIZE;
|
|
|
|
size -= HNS3_MAX_BD_SIZE;
|
|
|
|
|
|
|
|
if (bd_num > HNS3_MAX_TSO_BD_NUM)
|
|
|
|
return bd_num;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
if (size) {
|
|
|
|
bd_size[bd_num++] = size;
|
|
|
|
if (bd_num > HNS3_MAX_TSO_BD_NUM)
|
|
|
|
return bd_num;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:41 +08:00
|
|
|
for (i = 0; i < skb_shinfo(skb)->nr_frags; i++) {
|
2019-07-23 11:08:25 +08:00
|
|
|
skb_frag_t *frag = &skb_shinfo(skb)->frags[i];
|
2019-10-08 09:20:09 +08:00
|
|
|
size = skb_frag_size(frag);
|
|
|
|
if (!size)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
while (size > HNS3_MAX_BD_SIZE) {
|
|
|
|
bd_size[bd_num++] = HNS3_MAX_BD_SIZE;
|
|
|
|
size -= HNS3_MAX_BD_SIZE;
|
|
|
|
|
|
|
|
if (bd_num > HNS3_MAX_TSO_BD_NUM)
|
|
|
|
return bd_num;
|
|
|
|
}
|
|
|
|
|
|
|
|
bd_size[bd_num++] = size;
|
|
|
|
if (bd_num > HNS3_MAX_TSO_BD_NUM)
|
|
|
|
return bd_num;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bd_num;
|
|
|
|
}
|
|
|
|
|
2020-09-29 17:31:59 +08:00
|
|
|
static unsigned int hns3_tx_bd_num(struct sk_buff *skb, unsigned int *bd_size,
|
2021-03-29 11:57:50 +08:00
|
|
|
u8 max_non_tso_bd_num, unsigned int bd_num,
|
|
|
|
unsigned int recursion_level)
|
2019-10-08 09:20:09 +08:00
|
|
|
{
|
2021-03-29 11:57:50 +08:00
|
|
|
#define HNS3_MAX_RECURSION_LEVEL 24
|
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
struct sk_buff *frag_skb;
|
|
|
|
|
|
|
|
/* If the total len is within the max bd limit */
|
2021-03-29 11:57:50 +08:00
|
|
|
if (likely(skb->len <= HNS3_MAX_BD_SIZE && !recursion_level &&
|
|
|
|
!skb_has_frag_list(skb) &&
|
2020-09-29 17:31:59 +08:00
|
|
|
skb_shinfo(skb)->nr_frags < max_non_tso_bd_num))
|
2019-10-08 09:20:09 +08:00
|
|
|
return skb_shinfo(skb)->nr_frags + 1U;
|
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
if (unlikely(recursion_level >= HNS3_MAX_RECURSION_LEVEL))
|
|
|
|
return UINT_MAX;
|
2019-10-08 09:20:09 +08:00
|
|
|
|
|
|
|
bd_num = hns3_skb_bd_num(skb, bd_size, bd_num);
|
|
|
|
if (!skb_has_frag_list(skb) || bd_num > HNS3_MAX_TSO_BD_NUM)
|
|
|
|
return bd_num;
|
|
|
|
|
|
|
|
skb_walk_frags(skb, frag_skb) {
|
2021-03-29 11:57:50 +08:00
|
|
|
bd_num = hns3_tx_bd_num(frag_skb, bd_size, max_non_tso_bd_num,
|
|
|
|
bd_num, recursion_level + 1);
|
2019-10-08 09:20:09 +08:00
|
|
|
if (bd_num > HNS3_MAX_TSO_BD_NUM)
|
|
|
|
return bd_num;
|
2019-05-06 10:48:41 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:41 +08:00
|
|
|
return bd_num;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:44 +08:00
|
|
|
static unsigned int hns3_gso_hdr_len(struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
if (!skb->encapsulation)
|
|
|
|
return skb_transport_offset(skb) + tcp_hdrlen(skb);
|
|
|
|
|
|
|
|
return skb_inner_transport_offset(skb) + inner_tcp_hdrlen(skb);
|
|
|
|
}
|
|
|
|
|
2020-09-29 17:31:59 +08:00
|
|
|
/* HW need every continuous max_non_tso_bd_num buffer data to be larger
|
|
|
|
* than MSS, we simplify it by ensuring skb_headlen + the first continuous
|
|
|
|
* max_non_tso_bd_num - 1 frags to be larger than gso header len + mss,
|
|
|
|
* and the remaining continuous max_non_tso_bd_num - 1 frags to be larger
|
|
|
|
* than MSS except the last max_non_tso_bd_num - 1 frags.
|
2019-05-06 10:48:44 +08:00
|
|
|
*/
|
2019-10-08 09:20:09 +08:00
|
|
|
static bool hns3_skb_need_linearized(struct sk_buff *skb, unsigned int *bd_size,
|
2020-09-29 17:31:59 +08:00
|
|
|
unsigned int bd_num, u8 max_non_tso_bd_num)
|
2019-05-06 10:48:44 +08:00
|
|
|
{
|
|
|
|
unsigned int tot_len = 0;
|
|
|
|
int i;
|
|
|
|
|
2020-09-29 17:31:59 +08:00
|
|
|
for (i = 0; i < max_non_tso_bd_num - 1U; i++)
|
2019-10-08 09:20:09 +08:00
|
|
|
tot_len += bd_size[i];
|
2019-05-06 10:48:44 +08:00
|
|
|
|
2020-09-29 17:31:59 +08:00
|
|
|
/* ensure the first max_non_tso_bd_num frags is greater than
|
|
|
|
* mss + header
|
|
|
|
*/
|
|
|
|
if (tot_len + bd_size[max_non_tso_bd_num - 1U] <
|
2019-10-08 09:20:09 +08:00
|
|
|
skb_shinfo(skb)->gso_size + hns3_gso_hdr_len(skb))
|
2019-05-06 10:48:44 +08:00
|
|
|
return true;
|
|
|
|
|
2020-09-29 17:31:59 +08:00
|
|
|
/* ensure every continuous max_non_tso_bd_num - 1 buffer is greater
|
|
|
|
* than mss except the last one.
|
2019-10-08 09:20:09 +08:00
|
|
|
*/
|
2020-09-29 17:31:59 +08:00
|
|
|
for (i = 0; i < bd_num - max_non_tso_bd_num; i++) {
|
2019-10-08 09:20:09 +08:00
|
|
|
tot_len -= bd_size[i];
|
2020-09-29 17:31:59 +08:00
|
|
|
tot_len += bd_size[i + max_non_tso_bd_num - 1U];
|
2019-05-06 10:48:44 +08:00
|
|
|
|
|
|
|
if (tot_len < skb_shinfo(skb)->gso_size)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-01-04 10:49:24 +08:00
|
|
|
void hns3_shinfo_pack(struct skb_shared_info *shinfo, __u32 *size)
|
|
|
|
{
|
2020-09-25 08:26:14 +08:00
|
|
|
int i;
|
2020-01-04 10:49:24 +08:00
|
|
|
|
|
|
|
for (i = 0; i < MAX_SKB_FRAGS; i++)
|
|
|
|
size[i] = skb_frag_size(&shinfo->frags[i]);
|
|
|
|
}
|
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
static int hns3_skb_linearize(struct hns3_enet_ring *ring,
|
|
|
|
struct sk_buff *skb,
|
|
|
|
u8 max_non_tso_bd_num,
|
|
|
|
unsigned int bd_num)
|
|
|
|
{
|
|
|
|
/* 'bd_num == UINT_MAX' means the skb' fraglist has a
|
|
|
|
* recursion level of over HNS3_MAX_RECURSION_LEVEL.
|
|
|
|
*/
|
|
|
|
if (bd_num == UINT_MAX) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.over_max_recursion++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The skb->len has exceeded the hw limitation, linearization
|
|
|
|
* will not help.
|
|
|
|
*/
|
|
|
|
if (skb->len > HNS3_MAX_TSO_SIZE ||
|
|
|
|
(!skb_is_gso(skb) && skb->len >
|
|
|
|
HNS3_MAX_NON_TSO_SIZE(max_non_tso_bd_num))) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.hw_limitation++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (__skb_linearize(skb)) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.sw_err_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:41 +08:00
|
|
|
static int hns3_nic_maybe_stop_tx(struct hns3_enet_ring *ring,
|
2019-12-05 10:12:27 +08:00
|
|
|
struct net_device *netdev,
|
2019-12-05 10:12:28 +08:00
|
|
|
struct sk_buff *skb)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-12-05 10:12:27 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2020-09-29 17:31:59 +08:00
|
|
|
u8 max_non_tso_bd_num = priv->max_non_tso_bd_num;
|
2019-10-08 09:20:09 +08:00
|
|
|
unsigned int bd_size[HNS3_MAX_TSO_BD_NUM + 1U];
|
2019-08-09 10:31:14 +08:00
|
|
|
unsigned int bd_num;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
bd_num = hns3_tx_bd_num(skb, bd_size, max_non_tso_bd_num, 0, 0);
|
2020-09-29 17:31:59 +08:00
|
|
|
if (unlikely(bd_num > max_non_tso_bd_num)) {
|
2019-10-08 09:20:09 +08:00
|
|
|
if (bd_num <= HNS3_MAX_TSO_BD_NUM && skb_is_gso(skb) &&
|
2020-09-29 17:31:59 +08:00
|
|
|
!hns3_skb_need_linearized(skb, bd_size, bd_num,
|
|
|
|
max_non_tso_bd_num)) {
|
2020-09-29 17:32:00 +08:00
|
|
|
trace_hns3_over_max_bd(skb);
|
2019-05-06 10:48:44 +08:00
|
|
|
goto out;
|
2020-01-04 10:49:24 +08:00
|
|
|
}
|
2019-05-06 10:48:44 +08:00
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
if (hns3_skb_linearize(ring, skb, max_non_tso_bd_num,
|
|
|
|
bd_num))
|
2019-01-27 00:49:19 +08:00
|
|
|
return -ENOMEM;
|
2019-05-06 10:48:41 +08:00
|
|
|
|
2019-12-05 10:12:28 +08:00
|
|
|
bd_num = hns3_tx_bd_count(skb->len);
|
2019-08-09 10:31:14 +08:00
|
|
|
|
2019-05-06 10:48:41 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_copy++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
2019-01-27 00:49:19 +08:00
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:44 +08:00
|
|
|
out:
|
2019-12-05 10:12:27 +08:00
|
|
|
if (likely(ring_space(ring) >= bd_num))
|
|
|
|
return bd_num;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-12-05 10:12:27 +08:00
|
|
|
netif_stop_subqueue(netdev, ring->queue_index);
|
|
|
|
smp_mb(); /* Memory barrier before checking ring_space */
|
|
|
|
|
|
|
|
/* Start queue in case hns3_clean_tx_ring has just made room
|
|
|
|
* available and has not seen the queue stopped state performed
|
|
|
|
* by netif_stop_subqueue above.
|
|
|
|
*/
|
|
|
|
if (ring_space(ring) >= bd_num && netif_carrier_ok(netdev) &&
|
|
|
|
!test_bit(HNS3_NIC_STATE_DOWN, &priv->state)) {
|
|
|
|
netif_start_subqueue(netdev, ring->queue_index);
|
|
|
|
return bd_num;
|
|
|
|
}
|
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_busy++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
2019-12-05 10:12:27 +08:00
|
|
|
return -EBUSY;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2018-10-16 19:58:51 +08:00
|
|
|
static void hns3_clear_desc(struct hns3_enet_ring *ring, int next_to_use_orig)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct device *dev = ring_to_dev(ring);
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ring->desc_num; i++) {
|
2020-07-21 19:03:53 +08:00
|
|
|
struct hns3_desc *desc = &ring->desc[ring->next_to_use];
|
|
|
|
|
|
|
|
memset(desc, 0, sizeof(*desc));
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* check if this is where we started */
|
|
|
|
if (ring->next_to_use == next_to_use_orig)
|
|
|
|
break;
|
|
|
|
|
2019-05-06 10:48:48 +08:00
|
|
|
/* rollback one */
|
|
|
|
ring_ptr_move_bw(ring, next_to_use);
|
|
|
|
|
2020-07-21 19:03:53 +08:00
|
|
|
if (!ring->desc_cb[ring->next_to_use].dma)
|
|
|
|
continue;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* unmap the descriptor dma address */
|
2020-03-28 15:09:56 +08:00
|
|
|
if (ring->desc_cb[ring->next_to_use].type == DESC_TYPE_SKB ||
|
|
|
|
ring->desc_cb[ring->next_to_use].type ==
|
|
|
|
DESC_TYPE_FRAGLIST_SKB)
|
2017-08-02 23:59:45 +08:00
|
|
|
dma_unmap_single(dev,
|
|
|
|
ring->desc_cb[ring->next_to_use].dma,
|
|
|
|
ring->desc_cb[ring->next_to_use].length,
|
|
|
|
DMA_TO_DEVICE);
|
2018-10-16 19:58:52 +08:00
|
|
|
else if (ring->desc_cb[ring->next_to_use].length)
|
2017-08-02 23:59:45 +08:00
|
|
|
dma_unmap_page(dev,
|
|
|
|
ring->desc_cb[ring->next_to_use].dma,
|
|
|
|
ring->desc_cb[ring->next_to_use].length,
|
|
|
|
DMA_TO_DEVICE);
|
|
|
|
|
2018-10-16 19:58:52 +08:00
|
|
|
ring->desc_cb[ring->next_to_use].length = 0;
|
2019-05-06 10:48:48 +08:00
|
|
|
ring->desc_cb[ring->next_to_use].dma = 0;
|
2020-07-21 19:03:53 +08:00
|
|
|
ring->desc_cb[ring->next_to_use].type = DESC_TYPE_UNKNOWN;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
static int hns3_fill_skb_to_desc(struct hns3_enet_ring *ring,
|
|
|
|
struct sk_buff *skb, enum hns_desc_type type)
|
|
|
|
{
|
|
|
|
unsigned int size = skb_headlen(skb);
|
2021-03-29 11:57:50 +08:00
|
|
|
struct sk_buff *frag_skb;
|
2019-10-08 09:20:09 +08:00
|
|
|
int i, ret, bd_num = 0;
|
|
|
|
|
|
|
|
if (size) {
|
|
|
|
ret = hns3_fill_desc(ring, skb, size, type);
|
|
|
|
if (unlikely(ret < 0))
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
bd_num += ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < skb_shinfo(skb)->nr_frags; i++) {
|
|
|
|
skb_frag_t *frag = &skb_shinfo(skb)->frags[i];
|
|
|
|
|
|
|
|
size = skb_frag_size(frag);
|
|
|
|
if (!size)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
ret = hns3_fill_desc(ring, frag, size, DESC_TYPE_PAGE);
|
|
|
|
if (unlikely(ret < 0))
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
bd_num += ret;
|
|
|
|
}
|
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
skb_walk_frags(skb, frag_skb) {
|
|
|
|
ret = hns3_fill_skb_to_desc(ring, frag_skb,
|
|
|
|
DESC_TYPE_FRAGLIST_SKB);
|
|
|
|
if (unlikely(ret < 0))
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
bd_num += ret;
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
return bd_num;
|
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:46 +08:00
|
|
|
static void hns3_tx_doorbell(struct hns3_enet_ring *ring, int num,
|
|
|
|
bool doorbell)
|
|
|
|
{
|
|
|
|
ring->pending_buf += num;
|
|
|
|
|
|
|
|
if (!doorbell) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_more++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ring->pending_buf)
|
|
|
|
return;
|
|
|
|
|
2020-09-16 17:33:49 +08:00
|
|
|
writel(ring->pending_buf,
|
|
|
|
ring->tqp->io_base + HNS3_RING_TX_RING_TAIL_REG);
|
2020-09-16 17:33:46 +08:00
|
|
|
ring->pending_buf = 0;
|
2020-09-16 17:33:47 +08:00
|
|
|
WRITE_ONCE(ring->last_to_use, ring->next_to_use);
|
2020-09-16 17:33:46 +08:00
|
|
|
}
|
|
|
|
|
2017-10-20 10:19:21 +08:00
|
|
|
netdev_tx_t hns3_nic_net_xmit(struct sk_buff *skb, struct net_device *netdev)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2019-10-19 16:03:49 +08:00
|
|
|
struct hns3_enet_ring *ring = &priv->ring[skb->queue_mapping];
|
2021-03-29 11:57:51 +08:00
|
|
|
struct hns3_desc_cb *desc_cb = &ring->desc_cb[ring->next_to_use];
|
2017-08-02 23:59:45 +08:00
|
|
|
struct netdev_queue *dev_queue;
|
2019-10-08 09:20:09 +08:00
|
|
|
int pre_ntu, next_to_use_head;
|
2020-09-16 17:33:46 +08:00
|
|
|
bool doorbell;
|
2017-08-02 23:59:45 +08:00
|
|
|
int ret;
|
|
|
|
|
2020-01-15 10:46:45 +08:00
|
|
|
/* Hardware can only handle short frames above 32 bytes */
|
2020-09-16 17:33:46 +08:00
|
|
|
if (skb_put_padto(skb, HNS3_MIN_TX_LEN)) {
|
|
|
|
hns3_tx_doorbell(ring, 0, !netdev_xmit_more());
|
2021-03-29 11:57:53 +08:00
|
|
|
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.sw_err_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
2020-01-15 10:46:45 +08:00
|
|
|
return NETDEV_TX_OK;
|
2020-09-16 17:33:46 +08:00
|
|
|
}
|
2020-01-15 10:46:45 +08:00
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* Prefetch the data used later */
|
|
|
|
prefetch(skb->data);
|
|
|
|
|
2019-12-05 10:12:28 +08:00
|
|
|
ret = hns3_nic_maybe_stop_tx(ring, netdev, skb);
|
2019-10-08 09:20:09 +08:00
|
|
|
if (unlikely(ret <= 0)) {
|
|
|
|
if (ret == -EBUSY) {
|
2020-09-16 17:33:46 +08:00
|
|
|
hns3_tx_doorbell(ring, 0, true);
|
2019-12-05 10:12:27 +08:00
|
|
|
return NETDEV_TX_BUSY;
|
2019-05-06 10:48:41 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
hns3_rl_err(netdev, "xmit error: %d!\n", ret);
|
2017-08-02 23:59:45 +08:00
|
|
|
goto out_err_tx_ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
next_to_use_head = ring->next_to_use;
|
|
|
|
|
2021-03-29 11:57:51 +08:00
|
|
|
ret = hns3_fill_skb_desc(ring, skb, &ring->desc[ring->next_to_use],
|
|
|
|
desc_cb);
|
2020-07-28 10:16:48 +08:00
|
|
|
if (unlikely(ret < 0))
|
|
|
|
goto fill_err;
|
|
|
|
|
2021-03-29 11:57:50 +08:00
|
|
|
/* 'ret < 0' means filling error, 'ret == 0' means skb->len is
|
|
|
|
* zero, which is unlikely, and 'ret > 0' means how many tx desc
|
|
|
|
* need to be notified to the hw.
|
|
|
|
*/
|
2019-10-08 09:20:09 +08:00
|
|
|
ret = hns3_fill_skb_to_desc(ring, skb, DESC_TYPE_SKB);
|
2021-03-29 11:57:50 +08:00
|
|
|
if (unlikely(ret <= 0))
|
2019-05-06 10:48:48 +08:00
|
|
|
goto fill_err;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-08 09:20:09 +08:00
|
|
|
pre_ntu = ring->next_to_use ? (ring->next_to_use - 1) :
|
|
|
|
(ring->desc_num - 1);
|
|
|
|
ring->desc[pre_ntu].tx.bdtp_fe_sc_vld_ra_ri |=
|
|
|
|
cpu_to_le16(BIT(HNS3_TXD_FE_B));
|
2020-01-04 10:49:24 +08:00
|
|
|
trace_hns3_tx_desc(ring, pre_ntu);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
/* Complete translate all packets */
|
2019-10-19 16:03:49 +08:00
|
|
|
dev_queue = netdev_get_tx_queue(netdev, ring->queue_index);
|
2021-03-29 11:57:51 +08:00
|
|
|
doorbell = __netdev_tx_sent_queue(dev_queue, desc_cb->send_bytes,
|
2020-09-16 17:33:46 +08:00
|
|
|
netdev_xmit_more());
|
2021-03-29 11:57:50 +08:00
|
|
|
hns3_tx_doorbell(ring, ret, doorbell);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return NETDEV_TX_OK;
|
|
|
|
|
2019-05-06 10:48:48 +08:00
|
|
|
fill_err:
|
2018-10-16 19:58:51 +08:00
|
|
|
hns3_clear_desc(ring, next_to_use_head);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
out_err_tx_ok:
|
|
|
|
dev_kfree_skb_any(skb);
|
2020-09-16 17:33:46 +08:00
|
|
|
hns3_tx_doorbell(ring, 0, !netdev_xmit_more());
|
2017-08-02 23:59:45 +08:00
|
|
|
return NETDEV_TX_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_net_set_mac_address(struct net_device *netdev, void *p)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
struct sockaddr *mac_addr = p;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!mac_addr || !is_valid_ether_addr((const u8 *)mac_addr->sa_data))
|
|
|
|
return -EADDRNOTAVAIL;
|
|
|
|
|
2018-06-02 00:52:03 +08:00
|
|
|
if (ether_addr_equal(netdev->dev_addr, mac_addr->sa_data)) {
|
|
|
|
netdev_info(netdev, "already using mac address %pM\n",
|
|
|
|
mac_addr->sa_data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:08 +08:00
|
|
|
/* For VF device, if there is a perm_addr, then the user will not
|
|
|
|
* be allowed to change the address.
|
|
|
|
*/
|
|
|
|
if (!hns3_is_phys_func(h->pdev) &&
|
|
|
|
!is_zero_ether_addr(netdev->perm_addr)) {
|
|
|
|
netdev_err(netdev, "has permanent MAC %pM, user MAC %pM not allow\n",
|
|
|
|
netdev->perm_addr, mac_addr->sa_data);
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
2018-03-10 11:29:22 +08:00
|
|
|
ret = h->ae_algo->ops->set_mac_addr(h, mac_addr->sa_data, false);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (ret) {
|
|
|
|
netdev_err(netdev, "set_mac_address fail, ret=%d!\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ether_addr_copy(netdev->dev_addr, mac_addr->sa_data);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-20 01:29:51 +08:00
|
|
|
static int hns3_nic_do_ioctl(struct net_device *netdev,
|
|
|
|
struct ifreq *ifr, int cmd)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
|
|
|
|
if (!netif_running(netdev))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (!h->ae_algo->ops->do_ioctl)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return h->ae_algo->ops->do_ioctl(h, ifr, cmd);
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_nic_set_features(struct net_device *netdev,
|
|
|
|
netdev_features_t features)
|
|
|
|
{
|
2018-01-12 16:23:16 +08:00
|
|
|
netdev_features_t changed = netdev->features ^ features;
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2017-12-22 12:21:47 +08:00
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
2019-02-02 22:39:28 +08:00
|
|
|
bool enable;
|
2017-12-22 12:21:47 +08:00
|
|
|
int ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-11-15 17:29:23 +08:00
|
|
|
if (changed & (NETIF_F_GRO_HW) && h->ae_algo->ops->set_gro_en) {
|
2019-02-02 22:39:28 +08:00
|
|
|
enable = !!(features & NETIF_F_GRO_HW);
|
|
|
|
ret = h->ae_algo->ops->set_gro_en(h, enable);
|
2018-11-15 17:29:23 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-01-12 16:23:17 +08:00
|
|
|
if ((changed & NETIF_F_HW_VLAN_CTAG_RX) &&
|
|
|
|
h->ae_algo->ops->enable_hw_strip_rxvtag) {
|
2019-02-02 22:39:28 +08:00
|
|
|
enable = !!(features & NETIF_F_HW_VLAN_CTAG_RX);
|
|
|
|
ret = h->ae_algo->ops->enable_hw_strip_rxvtag(h, enable);
|
2017-12-22 12:21:47 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-10-01 19:46:47 +08:00
|
|
|
if ((changed & NETIF_F_NTUPLE) && h->ae_algo->ops->enable_fd) {
|
2019-02-02 22:39:28 +08:00
|
|
|
enable = !!(features & NETIF_F_NTUPLE);
|
|
|
|
h->ae_algo->ops->enable_fd(h, enable);
|
2018-10-01 19:46:47 +08:00
|
|
|
}
|
|
|
|
|
2020-12-10 11:42:09 +08:00
|
|
|
if ((netdev->features & NETIF_F_HW_TC) > (features & NETIF_F_HW_TC) &&
|
|
|
|
h->ae_algo->ops->cls_flower_active(h)) {
|
|
|
|
netdev_err(netdev,
|
|
|
|
"there are offloaded TC filters active, cannot disable HW TC offload");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
netdev->features = features;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-12-19 14:57:44 +08:00
|
|
|
static netdev_features_t hns3_features_check(struct sk_buff *skb,
|
|
|
|
struct net_device *dev,
|
|
|
|
netdev_features_t features)
|
|
|
|
{
|
|
|
|
#define HNS3_MAX_HDR_LEN 480U
|
|
|
|
#define HNS3_MAX_L4_HDR_LEN 60U
|
|
|
|
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
if (skb->ip_summed != CHECKSUM_PARTIAL)
|
|
|
|
return features;
|
|
|
|
|
|
|
|
if (skb->encapsulation)
|
|
|
|
len = skb_inner_transport_header(skb) - skb->data;
|
|
|
|
else
|
|
|
|
len = skb_transport_header(skb) - skb->data;
|
|
|
|
|
|
|
|
/* Assume L4 is 60 byte as TCP is the only protocol with a
|
|
|
|
* a flexible value, and it's max len is 60 bytes.
|
|
|
|
*/
|
|
|
|
len += HNS3_MAX_L4_HDR_LEN;
|
|
|
|
|
|
|
|
/* Hardware only supports checksum on the skb with a max header
|
|
|
|
* len of 480 bytes.
|
|
|
|
*/
|
|
|
|
if (len > HNS3_MAX_HDR_LEN)
|
|
|
|
features &= ~(NETIF_F_CSUM_MASK | NETIF_F_GSO_MASK);
|
|
|
|
|
|
|
|
return features;
|
|
|
|
}
|
|
|
|
|
2018-01-09 14:50:59 +08:00
|
|
|
static void hns3_nic_get_stats64(struct net_device *netdev,
|
|
|
|
struct rtnl_link_stats64 *stats)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
int queue_num = priv->ae_handle->kinfo.num_tqps;
|
2018-01-05 18:18:10 +08:00
|
|
|
struct hnae3_handle *handle = priv->ae_handle;
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_enet_ring *ring;
|
2019-01-23 07:39:29 +08:00
|
|
|
u64 rx_length_errors = 0;
|
|
|
|
u64 rx_crc_errors = 0;
|
|
|
|
u64 rx_multicast = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
unsigned int start;
|
2019-01-23 07:39:29 +08:00
|
|
|
u64 tx_errors = 0;
|
|
|
|
u64 rx_errors = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
unsigned int idx;
|
|
|
|
u64 tx_bytes = 0;
|
|
|
|
u64 rx_bytes = 0;
|
|
|
|
u64 tx_pkts = 0;
|
|
|
|
u64 rx_pkts = 0;
|
2018-01-05 18:18:12 +08:00
|
|
|
u64 tx_drop = 0;
|
|
|
|
u64 rx_drop = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-01-05 18:18:11 +08:00
|
|
|
if (test_bit(HNS3_NIC_STATE_DOWN, &priv->state))
|
|
|
|
return;
|
|
|
|
|
2018-01-05 18:18:10 +08:00
|
|
|
handle->ae_algo->ops->update_stats(handle, &netdev->stats);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
for (idx = 0; idx < queue_num; idx++) {
|
|
|
|
/* fetch the tx stats */
|
2019-10-19 16:03:49 +08:00
|
|
|
ring = &priv->ring[idx];
|
2017-08-02 23:59:45 +08:00
|
|
|
do {
|
2017-08-18 19:31:37 +08:00
|
|
|
start = u64_stats_fetch_begin_irq(&ring->syncp);
|
2017-08-02 23:59:45 +08:00
|
|
|
tx_bytes += ring->stats.tx_bytes;
|
|
|
|
tx_pkts += ring->stats.tx_pkts;
|
2018-01-05 18:18:12 +08:00
|
|
|
tx_drop += ring->stats.sw_err_cnt;
|
2019-08-09 10:31:13 +08:00
|
|
|
tx_drop += ring->stats.tx_vlan_err;
|
|
|
|
tx_drop += ring->stats.tx_l4_proto_err;
|
|
|
|
tx_drop += ring->stats.tx_l2l3l4_err;
|
|
|
|
tx_drop += ring->stats.tx_tso_err;
|
2021-03-29 11:57:50 +08:00
|
|
|
tx_drop += ring->stats.over_max_recursion;
|
|
|
|
tx_drop += ring->stats.hw_limitation;
|
2019-01-23 07:39:29 +08:00
|
|
|
tx_errors += ring->stats.sw_err_cnt;
|
2019-08-09 10:31:13 +08:00
|
|
|
tx_errors += ring->stats.tx_vlan_err;
|
|
|
|
tx_errors += ring->stats.tx_l4_proto_err;
|
|
|
|
tx_errors += ring->stats.tx_l2l3l4_err;
|
|
|
|
tx_errors += ring->stats.tx_tso_err;
|
2021-03-29 11:57:50 +08:00
|
|
|
tx_errors += ring->stats.over_max_recursion;
|
|
|
|
tx_errors += ring->stats.hw_limitation;
|
2017-08-02 23:59:45 +08:00
|
|
|
} while (u64_stats_fetch_retry_irq(&ring->syncp, start));
|
|
|
|
|
|
|
|
/* fetch the rx stats */
|
2019-10-19 16:03:49 +08:00
|
|
|
ring = &priv->ring[idx + queue_num];
|
2017-08-02 23:59:45 +08:00
|
|
|
do {
|
2017-08-18 19:31:37 +08:00
|
|
|
start = u64_stats_fetch_begin_irq(&ring->syncp);
|
2017-08-02 23:59:45 +08:00
|
|
|
rx_bytes += ring->stats.rx_bytes;
|
|
|
|
rx_pkts += ring->stats.rx_pkts;
|
2018-01-05 18:18:12 +08:00
|
|
|
rx_drop += ring->stats.l2_err;
|
2019-01-23 07:39:29 +08:00
|
|
|
rx_errors += ring->stats.l2_err;
|
2019-06-28 19:50:15 +08:00
|
|
|
rx_errors += ring->stats.l3l4_csum_err;
|
2019-01-23 07:39:29 +08:00
|
|
|
rx_crc_errors += ring->stats.l2_err;
|
|
|
|
rx_multicast += ring->stats.rx_multicast;
|
|
|
|
rx_length_errors += ring->stats.err_pkt_len;
|
2017-08-02 23:59:45 +08:00
|
|
|
} while (u64_stats_fetch_retry_irq(&ring->syncp, start));
|
|
|
|
}
|
|
|
|
|
|
|
|
stats->tx_bytes = tx_bytes;
|
|
|
|
stats->tx_packets = tx_pkts;
|
|
|
|
stats->rx_bytes = rx_bytes;
|
|
|
|
stats->rx_packets = rx_pkts;
|
|
|
|
|
2019-01-23 07:39:29 +08:00
|
|
|
stats->rx_errors = rx_errors;
|
|
|
|
stats->multicast = rx_multicast;
|
|
|
|
stats->rx_length_errors = rx_length_errors;
|
|
|
|
stats->rx_crc_errors = rx_crc_errors;
|
2017-08-02 23:59:45 +08:00
|
|
|
stats->rx_missed_errors = netdev->stats.rx_missed_errors;
|
|
|
|
|
2019-01-23 07:39:29 +08:00
|
|
|
stats->tx_errors = tx_errors;
|
|
|
|
stats->rx_dropped = rx_drop;
|
|
|
|
stats->tx_dropped = tx_drop;
|
2017-08-02 23:59:45 +08:00
|
|
|
stats->collisions = netdev->stats.collisions;
|
|
|
|
stats->rx_over_errors = netdev->stats.rx_over_errors;
|
|
|
|
stats->rx_frame_errors = netdev->stats.rx_frame_errors;
|
|
|
|
stats->rx_fifo_errors = netdev->stats.rx_fifo_errors;
|
|
|
|
stats->tx_aborted_errors = netdev->stats.tx_aborted_errors;
|
|
|
|
stats->tx_carrier_errors = netdev->stats.tx_carrier_errors;
|
|
|
|
stats->tx_fifo_errors = netdev->stats.tx_fifo_errors;
|
|
|
|
stats->tx_heartbeat_errors = netdev->stats.tx_heartbeat_errors;
|
|
|
|
stats->tx_window_errors = netdev->stats.tx_window_errors;
|
|
|
|
stats->rx_compressed = netdev->stats.rx_compressed;
|
|
|
|
stats->tx_compressed = netdev->stats.tx_compressed;
|
|
|
|
}
|
|
|
|
|
2017-10-17 14:51:30 +08:00
|
|
|
static int hns3_setup_tc(struct net_device *netdev, void *type_data)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2017-10-17 14:51:30 +08:00
|
|
|
struct tc_mqprio_qopt_offload *mqprio_qopt = type_data;
|
2019-06-13 17:12:29 +08:00
|
|
|
struct hnae3_knic_private_info *kinfo;
|
2017-10-17 14:51:30 +08:00
|
|
|
u8 tc = mqprio_qopt->qopt.num_tc;
|
|
|
|
u16 mode = mqprio_qopt->mode;
|
|
|
|
u8 hw = mqprio_qopt->qopt.hw;
|
2019-06-13 17:12:29 +08:00
|
|
|
struct hnae3_handle *h;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2017-10-17 14:51:30 +08:00
|
|
|
if (!((hw == TC_MQPRIO_HW_OFFLOAD_TCS &&
|
|
|
|
mode == TC_MQPRIO_MODE_CHANNEL) || (!hw && tc == 0)))
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
if (tc > HNAE3_MAX_TC)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (!netdev)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-13 17:12:29 +08:00
|
|
|
h = hns3_get_handle(netdev);
|
|
|
|
kinfo = &h->kinfo;
|
|
|
|
|
2019-07-29 10:53:27 +08:00
|
|
|
netif_dbg(h, drv, netdev, "setup tc: num_tc=%u\n", tc);
|
|
|
|
|
2019-01-27 00:49:18 +08:00
|
|
|
return (kinfo->dcb_ops && kinfo->dcb_ops->setup_tc) ?
|
2020-12-10 11:42:07 +08:00
|
|
|
kinfo->dcb_ops->setup_tc(h, mqprio_qopt) : -EOPNOTSUPP;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-12-10 11:42:09 +08:00
|
|
|
static int hns3_setup_tc_cls_flower(struct hns3_nic_priv *priv,
|
|
|
|
struct flow_cls_offload *flow)
|
|
|
|
{
|
|
|
|
int tc = tc_classid_to_hwtc(priv->netdev, flow->classid);
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(priv->netdev);
|
|
|
|
|
|
|
|
switch (flow->command) {
|
|
|
|
case FLOW_CLS_REPLACE:
|
|
|
|
if (h->ae_algo->ops->add_cls_flower)
|
|
|
|
return h->ae_algo->ops->add_cls_flower(h, flow, tc);
|
|
|
|
break;
|
|
|
|
case FLOW_CLS_DESTROY:
|
|
|
|
if (h->ae_algo->ops->del_cls_flower)
|
|
|
|
return h->ae_algo->ops->del_cls_flower(h, flow);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_setup_tc_block_cb(enum tc_setup_type type, void *type_data,
|
|
|
|
void *cb_priv)
|
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = cb_priv;
|
|
|
|
|
|
|
|
if (!tc_cls_can_offload_and_chain0(priv->netdev, type_data))
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case TC_SETUP_CLSFLOWER:
|
|
|
|
return hns3_setup_tc_cls_flower(priv, type_data);
|
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static LIST_HEAD(hns3_block_cb_list);
|
|
|
|
|
2017-08-07 16:15:17 +08:00
|
|
|
static int hns3_nic_setup_tc(struct net_device *dev, enum tc_setup_type type,
|
2017-08-07 16:15:32 +08:00
|
|
|
void *type_data)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2020-12-10 11:42:09 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(dev);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case TC_SETUP_QDISC_MQPRIO:
|
|
|
|
ret = hns3_setup_tc(dev, type_data);
|
|
|
|
break;
|
|
|
|
case TC_SETUP_BLOCK:
|
|
|
|
ret = flow_block_cb_setup_simple(type_data,
|
|
|
|
&hns3_block_cb_list,
|
|
|
|
hns3_setup_tc_block_cb,
|
|
|
|
priv, priv, true);
|
|
|
|
break;
|
|
|
|
default:
|
2017-08-07 16:15:31 +08:00
|
|
|
return -EOPNOTSUPP;
|
2020-12-10 11:42:09 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-12-10 11:42:09 +08:00
|
|
|
return ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_vlan_rx_add_vid(struct net_device *netdev,
|
|
|
|
__be16 proto, u16 vid)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
int ret = -EIO;
|
|
|
|
|
|
|
|
if (h->ae_algo->ops->set_vlan_filter)
|
|
|
|
ret = h->ae_algo->ops->set_vlan_filter(h, proto, vid, false);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_vlan_rx_kill_vid(struct net_device *netdev,
|
|
|
|
__be16 proto, u16 vid)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
int ret = -EIO;
|
|
|
|
|
|
|
|
if (h->ae_algo->ops->set_vlan_filter)
|
|
|
|
ret = h->ae_algo->ops->set_vlan_filter(h, proto, vid, true);
|
|
|
|
|
2018-10-30 21:50:50 +08:00
|
|
|
return ret;
|
2018-03-21 15:49:22 +08:00
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_ndo_set_vf_vlan(struct net_device *netdev, int vf, u16 vlan,
|
|
|
|
u8 qos, __be16 vlan_proto)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
int ret = -EIO;
|
|
|
|
|
2019-07-29 10:53:27 +08:00
|
|
|
netif_dbg(h, drv, netdev,
|
2019-10-31 19:23:24 +08:00
|
|
|
"set vf vlan: vf=%d, vlan=%u, qos=%u, vlan_proto=0x%x\n",
|
|
|
|
vf, vlan, qos, ntohs(vlan_proto));
|
2019-07-29 10:53:27 +08:00
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
if (h->ae_algo->ops->set_vf_vlan_filter)
|
|
|
|
ret = h->ae_algo->ops->set_vf_vlan_filter(h, vf, vlan,
|
2019-06-07 10:03:13 +08:00
|
|
|
qos, vlan_proto);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:05 +08:00
|
|
|
static int hns3_set_vf_spoofchk(struct net_device *netdev, int vf, bool enable)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *handle = hns3_get_handle(netdev);
|
|
|
|
|
|
|
|
if (hns3_nic_resetting(netdev))
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
if (!handle->ae_algo->ops->set_vf_spoofchk)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return handle->ae_algo->ops->set_vf_spoofchk(handle, vf, enable);
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:06 +08:00
|
|
|
static int hns3_set_vf_trust(struct net_device *netdev, int vf, bool enable)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *handle = hns3_get_handle(netdev);
|
|
|
|
|
|
|
|
if (!handle->ae_algo->ops->set_vf_trust)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return handle->ae_algo->ops->set_vf_trust(handle, vf, enable);
|
|
|
|
}
|
|
|
|
|
2017-08-22 00:05:24 +08:00
|
|
|
static int hns3_nic_change_mtu(struct net_device *netdev, int new_mtu)
|
|
|
|
{
|
2017-10-09 15:43:56 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
2017-08-22 00:05:24 +08:00
|
|
|
int ret;
|
|
|
|
|
2019-04-04 16:17:57 +08:00
|
|
|
if (hns3_nic_resetting(netdev))
|
|
|
|
return -EBUSY;
|
|
|
|
|
2017-08-22 00:05:24 +08:00
|
|
|
if (!h->ae_algo->ops->set_mtu)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
2019-07-29 10:53:27 +08:00
|
|
|
netif_dbg(h, drv, netdev,
|
|
|
|
"change mtu from %u to %d\n", netdev->mtu, new_mtu);
|
|
|
|
|
2017-08-22 00:05:24 +08:00
|
|
|
ret = h->ae_algo->ops->set_mtu(h, new_mtu);
|
2018-09-27 02:28:37 +08:00
|
|
|
if (ret)
|
2017-08-22 00:05:24 +08:00
|
|
|
netdev_err(netdev, "failed to change MTU in hardware %d\n",
|
|
|
|
ret);
|
2018-09-27 02:28:37 +08:00
|
|
|
else
|
|
|
|
netdev->mtu = new_mtu;
|
2018-01-05 18:18:20 +08:00
|
|
|
|
2017-08-22 00:05:24 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-11-02 20:45:20 +08:00
|
|
|
static bool hns3_get_tx_timeo_queue_info(struct net_device *ndev)
|
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(ndev);
|
2019-04-19 11:05:42 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(ndev);
|
2019-10-31 19:23:21 +08:00
|
|
|
struct hns3_enet_ring *tx_ring;
|
2019-04-19 11:05:42 +08:00
|
|
|
struct napi_struct *napi;
|
2017-11-02 20:45:20 +08:00
|
|
|
int timeout_queue = 0;
|
|
|
|
int hw_head, hw_tail;
|
2019-04-19 11:05:42 +08:00
|
|
|
int fbd_num, fbd_oft;
|
|
|
|
int ebd_num, ebd_oft;
|
|
|
|
int bd_num, bd_err;
|
|
|
|
int ring_en, tc;
|
2017-11-02 20:45:20 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Find the stopped queue the same way the stack does */
|
2019-04-19 11:05:41 +08:00
|
|
|
for (i = 0; i < ndev->num_tx_queues; i++) {
|
2017-11-02 20:45:20 +08:00
|
|
|
struct netdev_queue *q;
|
|
|
|
unsigned long trans_start;
|
|
|
|
|
|
|
|
q = netdev_get_tx_queue(ndev, i);
|
|
|
|
trans_start = q->trans_start;
|
|
|
|
if (netif_xmit_stopped(q) &&
|
|
|
|
time_after(jiffies,
|
|
|
|
(trans_start + ndev->watchdog_timeo))) {
|
|
|
|
timeout_queue = i;
|
2019-10-31 19:23:17 +08:00
|
|
|
netdev_info(ndev, "queue state: 0x%lx, delta msecs: %u\n",
|
|
|
|
q->state,
|
|
|
|
jiffies_to_msecs(jiffies - trans_start));
|
2017-11-02 20:45:20 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == ndev->num_tx_queues) {
|
|
|
|
netdev_info(ndev,
|
|
|
|
"no netdev TX timeout queue found, timeout count: %llu\n",
|
|
|
|
priv->tx_timeout_count);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-04-19 11:05:40 +08:00
|
|
|
priv->tx_timeout_count++;
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
tx_ring = &priv->ring[timeout_queue];
|
2019-04-19 11:05:42 +08:00
|
|
|
napi = &tx_ring->tqp_vector->napi;
|
|
|
|
|
|
|
|
netdev_info(ndev,
|
|
|
|
"tx_timeout count: %llu, queue id: %d, SW_NTU: 0x%x, SW_NTC: 0x%x, napi state: %lu\n",
|
|
|
|
priv->tx_timeout_count, timeout_queue, tx_ring->next_to_use,
|
|
|
|
tx_ring->next_to_clean, napi->state);
|
|
|
|
|
|
|
|
netdev_info(ndev,
|
2020-09-16 17:33:47 +08:00
|
|
|
"tx_pkts: %llu, tx_bytes: %llu, sw_err_cnt: %llu, tx_pending: %d\n",
|
2019-04-19 11:05:42 +08:00
|
|
|
tx_ring->stats.tx_pkts, tx_ring->stats.tx_bytes,
|
2020-09-16 17:33:47 +08:00
|
|
|
tx_ring->stats.sw_err_cnt, tx_ring->pending_buf);
|
2019-04-19 11:05:42 +08:00
|
|
|
|
|
|
|
netdev_info(ndev,
|
2020-09-16 17:33:46 +08:00
|
|
|
"seg_pkt_cnt: %llu, tx_more: %llu, restart_queue: %llu, tx_busy: %llu\n",
|
|
|
|
tx_ring->stats.seg_pkt_cnt, tx_ring->stats.tx_more,
|
2019-04-19 11:05:42 +08:00
|
|
|
tx_ring->stats.restart_queue, tx_ring->stats.tx_busy);
|
|
|
|
|
|
|
|
/* When mac received many pause frames continuous, it's unable to send
|
|
|
|
* packets, which may cause tx timeout
|
|
|
|
*/
|
2019-08-09 10:31:16 +08:00
|
|
|
if (h->ae_algo->ops->get_mac_stats) {
|
|
|
|
struct hns3_mac_stats mac_stats;
|
2019-04-19 11:05:42 +08:00
|
|
|
|
2019-08-09 10:31:16 +08:00
|
|
|
h->ae_algo->ops->get_mac_stats(h, &mac_stats);
|
2019-04-19 11:05:42 +08:00
|
|
|
netdev_info(ndev, "tx_pause_cnt: %llu, rx_pause_cnt: %llu\n",
|
2019-08-09 10:31:16 +08:00
|
|
|
mac_stats.tx_pause_cnt, mac_stats.rx_pause_cnt);
|
2019-04-19 11:05:42 +08:00
|
|
|
}
|
2017-11-02 20:45:20 +08:00
|
|
|
|
|
|
|
hw_head = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_HEAD_REG);
|
|
|
|
hw_tail = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_TAIL_REG);
|
2019-04-19 11:05:42 +08:00
|
|
|
fbd_num = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_FBDNUM_REG);
|
|
|
|
fbd_oft = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_OFFSET_REG);
|
|
|
|
ebd_num = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_EBDNUM_REG);
|
|
|
|
ebd_oft = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_EBD_OFFSET_REG);
|
|
|
|
bd_num = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_BD_NUM_REG);
|
|
|
|
bd_err = readl_relaxed(tx_ring->tqp->io_base +
|
|
|
|
HNS3_RING_TX_RING_BD_ERR_REG);
|
|
|
|
ring_en = readl_relaxed(tx_ring->tqp->io_base + HNS3_RING_EN_REG);
|
|
|
|
tc = readl_relaxed(tx_ring->tqp->io_base + HNS3_RING_TX_RING_TC_REG);
|
|
|
|
|
2017-11-02 20:45:20 +08:00
|
|
|
netdev_info(ndev,
|
2019-04-19 11:05:42 +08:00
|
|
|
"BD_NUM: 0x%x HW_HEAD: 0x%x, HW_TAIL: 0x%x, BD_ERR: 0x%x, INT: 0x%x\n",
|
|
|
|
bd_num, hw_head, hw_tail, bd_err,
|
2017-11-02 20:45:20 +08:00
|
|
|
readl(tx_ring->tqp_vector->mask_addr));
|
2019-04-19 11:05:42 +08:00
|
|
|
netdev_info(ndev,
|
|
|
|
"RING_EN: 0x%x, TC: 0x%x, FBD_NUM: 0x%x FBD_OFT: 0x%x, EBD_NUM: 0x%x, EBD_OFT: 0x%x\n",
|
|
|
|
ring_en, tc, fbd_num, fbd_oft, ebd_num, ebd_oft);
|
2017-11-02 20:45:20 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
netdev: pass the stuck queue to the timeout handler
This allows incrementing the correct timeout statistic without any mess.
Down the road, devices can learn to reset just the specific queue.
The patch was generated with the following script:
use strict;
use warnings;
our $^I = '.bak';
my @work = (
["arch/m68k/emu/nfeth.c", "nfeth_tx_timeout"],
["arch/um/drivers/net_kern.c", "uml_net_tx_timeout"],
["arch/um/drivers/vector_kern.c", "vector_net_tx_timeout"],
["arch/xtensa/platforms/iss/network.c", "iss_net_tx_timeout"],
["drivers/char/pcmcia/synclink_cs.c", "hdlcdev_tx_timeout"],
["drivers/infiniband/ulp/ipoib/ipoib_main.c", "ipoib_timeout"],
["drivers/infiniband/ulp/ipoib/ipoib_main.c", "ipoib_timeout"],
["drivers/message/fusion/mptlan.c", "mpt_lan_tx_timeout"],
["drivers/misc/sgi-xp/xpnet.c", "xpnet_dev_tx_timeout"],
["drivers/net/appletalk/cops.c", "cops_timeout"],
["drivers/net/arcnet/arcdevice.h", "arcnet_timeout"],
["drivers/net/arcnet/arcnet.c", "arcnet_timeout"],
["drivers/net/arcnet/com20020.c", "arcnet_timeout"],
["drivers/net/ethernet/3com/3c509.c", "el3_tx_timeout"],
["drivers/net/ethernet/3com/3c515.c", "corkscrew_timeout"],
["drivers/net/ethernet/3com/3c574_cs.c", "el3_tx_timeout"],
["drivers/net/ethernet/3com/3c589_cs.c", "el3_tx_timeout"],
["drivers/net/ethernet/3com/3c59x.c", "vortex_tx_timeout"],
["drivers/net/ethernet/3com/3c59x.c", "vortex_tx_timeout"],
["drivers/net/ethernet/3com/typhoon.c", "typhoon_tx_timeout"],
["drivers/net/ethernet/8390/8390.h", "ei_tx_timeout"],
["drivers/net/ethernet/8390/8390.h", "eip_tx_timeout"],
["drivers/net/ethernet/8390/8390.c", "ei_tx_timeout"],
["drivers/net/ethernet/8390/8390p.c", "eip_tx_timeout"],
["drivers/net/ethernet/8390/ax88796.c", "ax_ei_tx_timeout"],
["drivers/net/ethernet/8390/axnet_cs.c", "axnet_tx_timeout"],
["drivers/net/ethernet/8390/etherh.c", "__ei_tx_timeout"],
["drivers/net/ethernet/8390/hydra.c", "__ei_tx_timeout"],
["drivers/net/ethernet/8390/mac8390.c", "__ei_tx_timeout"],
["drivers/net/ethernet/8390/mcf8390.c", "__ei_tx_timeout"],
["drivers/net/ethernet/8390/lib8390.c", "__ei_tx_timeout"],
["drivers/net/ethernet/8390/ne2k-pci.c", "ei_tx_timeout"],
["drivers/net/ethernet/8390/pcnet_cs.c", "ei_tx_timeout"],
["drivers/net/ethernet/8390/smc-ultra.c", "ei_tx_timeout"],
["drivers/net/ethernet/8390/wd.c", "ei_tx_timeout"],
["drivers/net/ethernet/8390/zorro8390.c", "__ei_tx_timeout"],
["drivers/net/ethernet/adaptec/starfire.c", "tx_timeout"],
["drivers/net/ethernet/agere/et131x.c", "et131x_tx_timeout"],
["drivers/net/ethernet/allwinner/sun4i-emac.c", "emac_timeout"],
["drivers/net/ethernet/alteon/acenic.c", "ace_watchdog"],
["drivers/net/ethernet/amazon/ena/ena_netdev.c", "ena_tx_timeout"],
["drivers/net/ethernet/amd/7990.h", "lance_tx_timeout"],
["drivers/net/ethernet/amd/7990.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/a2065.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/am79c961a.c", "am79c961_timeout"],
["drivers/net/ethernet/amd/amd8111e.c", "amd8111e_tx_timeout"],
["drivers/net/ethernet/amd/ariadne.c", "ariadne_tx_timeout"],
["drivers/net/ethernet/amd/atarilance.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/au1000_eth.c", "au1000_tx_timeout"],
["drivers/net/ethernet/amd/declance.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/lance.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/mvme147.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/ni65.c", "ni65_timeout"],
["drivers/net/ethernet/amd/nmclan_cs.c", "mace_tx_timeout"],
["drivers/net/ethernet/amd/pcnet32.c", "pcnet32_tx_timeout"],
["drivers/net/ethernet/amd/sunlance.c", "lance_tx_timeout"],
["drivers/net/ethernet/amd/xgbe/xgbe-drv.c", "xgbe_tx_timeout"],
["drivers/net/ethernet/apm/xgene-v2/main.c", "xge_timeout"],
["drivers/net/ethernet/apm/xgene/xgene_enet_main.c", "xgene_enet_timeout"],
["drivers/net/ethernet/apple/macmace.c", "mace_tx_timeout"],
["drivers/net/ethernet/atheros/ag71xx.c", "ag71xx_tx_timeout"],
["drivers/net/ethernet/atheros/alx/main.c", "alx_tx_timeout"],
["drivers/net/ethernet/atheros/atl1c/atl1c_main.c", "atl1c_tx_timeout"],
["drivers/net/ethernet/atheros/atl1e/atl1e_main.c", "atl1e_tx_timeout"],
["drivers/net/ethernet/atheros/atlx/atl.c", "atlx_tx_timeout"],
["drivers/net/ethernet/atheros/atlx/atl1.c", "atlx_tx_timeout"],
["drivers/net/ethernet/atheros/atlx/atl2.c", "atl2_tx_timeout"],
["drivers/net/ethernet/broadcom/b44.c", "b44_tx_timeout"],
["drivers/net/ethernet/broadcom/bcmsysport.c", "bcm_sysport_tx_timeout"],
["drivers/net/ethernet/broadcom/bnx2.c", "bnx2_tx_timeout"],
["drivers/net/ethernet/broadcom/bnx2x/bnx2x_cmn.h", "bnx2x_tx_timeout"],
["drivers/net/ethernet/broadcom/bnx2x/bnx2x_cmn.c", "bnx2x_tx_timeout"],
["drivers/net/ethernet/broadcom/bnx2x/bnx2x_main.c", "bnx2x_tx_timeout"],
["drivers/net/ethernet/broadcom/bnxt/bnxt.c", "bnxt_tx_timeout"],
["drivers/net/ethernet/broadcom/genet/bcmgenet.c", "bcmgenet_timeout"],
["drivers/net/ethernet/broadcom/sb1250-mac.c", "sbmac_tx_timeout"],
["drivers/net/ethernet/broadcom/tg3.c", "tg3_tx_timeout"],
["drivers/net/ethernet/calxeda/xgmac.c", "xgmac_tx_timeout"],
["drivers/net/ethernet/cavium/liquidio/lio_main.c", "liquidio_tx_timeout"],
["drivers/net/ethernet/cavium/liquidio/lio_vf_main.c", "liquidio_tx_timeout"],
["drivers/net/ethernet/cavium/liquidio/lio_vf_rep.c", "lio_vf_rep_tx_timeout"],
["drivers/net/ethernet/cavium/thunder/nicvf_main.c", "nicvf_tx_timeout"],
["drivers/net/ethernet/cirrus/cs89x0.c", "net_timeout"],
["drivers/net/ethernet/cisco/enic/enic_main.c", "enic_tx_timeout"],
["drivers/net/ethernet/cisco/enic/enic_main.c", "enic_tx_timeout"],
["drivers/net/ethernet/cortina/gemini.c", "gmac_tx_timeout"],
["drivers/net/ethernet/davicom/dm9000.c", "dm9000_timeout"],
["drivers/net/ethernet/dec/tulip/de2104x.c", "de_tx_timeout"],
["drivers/net/ethernet/dec/tulip/tulip_core.c", "tulip_tx_timeout"],
["drivers/net/ethernet/dec/tulip/winbond-840.c", "tx_timeout"],
["drivers/net/ethernet/dlink/dl2k.c", "rio_tx_timeout"],
["drivers/net/ethernet/dlink/sundance.c", "tx_timeout"],
["drivers/net/ethernet/emulex/benet/be_main.c", "be_tx_timeout"],
["drivers/net/ethernet/ethoc.c", "ethoc_tx_timeout"],
["drivers/net/ethernet/faraday/ftgmac100.c", "ftgmac100_tx_timeout"],
["drivers/net/ethernet/fealnx.c", "fealnx_tx_timeout"],
["drivers/net/ethernet/freescale/dpaa/dpaa_eth.c", "dpaa_tx_timeout"],
["drivers/net/ethernet/freescale/fec_main.c", "fec_timeout"],
["drivers/net/ethernet/freescale/fec_mpc52xx.c", "mpc52xx_fec_tx_timeout"],
["drivers/net/ethernet/freescale/fs_enet/fs_enet-main.c", "fs_timeout"],
["drivers/net/ethernet/freescale/gianfar.c", "gfar_timeout"],
["drivers/net/ethernet/freescale/ucc_geth.c", "ucc_geth_timeout"],
["drivers/net/ethernet/fujitsu/fmvj18x_cs.c", "fjn_tx_timeout"],
["drivers/net/ethernet/google/gve/gve_main.c", "gve_tx_timeout"],
["drivers/net/ethernet/hisilicon/hip04_eth.c", "hip04_timeout"],
["drivers/net/ethernet/hisilicon/hix5hd2_gmac.c", "hix5hd2_net_timeout"],
["drivers/net/ethernet/hisilicon/hns/hns_enet.c", "hns_nic_net_timeout"],
["drivers/net/ethernet/hisilicon/hns3/hns3_enet.c", "hns3_nic_net_timeout"],
["drivers/net/ethernet/huawei/hinic/hinic_main.c", "hinic_tx_timeout"],
["drivers/net/ethernet/i825xx/82596.c", "i596_tx_timeout"],
["drivers/net/ethernet/i825xx/ether1.c", "ether1_timeout"],
["drivers/net/ethernet/i825xx/lib82596.c", "i596_tx_timeout"],
["drivers/net/ethernet/i825xx/sun3_82586.c", "sun3_82586_timeout"],
["drivers/net/ethernet/ibm/ehea/ehea_main.c", "ehea_tx_watchdog"],
["drivers/net/ethernet/ibm/emac/core.c", "emac_tx_timeout"],
["drivers/net/ethernet/ibm/emac/core.c", "emac_tx_timeout"],
["drivers/net/ethernet/ibm/ibmvnic.c", "ibmvnic_tx_timeout"],
["drivers/net/ethernet/intel/e100.c", "e100_tx_timeout"],
["drivers/net/ethernet/intel/e1000/e1000_main.c", "e1000_tx_timeout"],
["drivers/net/ethernet/intel/e1000e/netdev.c", "e1000_tx_timeout"],
["drivers/net/ethernet/intel/fm10k/fm10k_netdev.c", "fm10k_tx_timeout"],
["drivers/net/ethernet/intel/i40e/i40e_main.c", "i40e_tx_timeout"],
["drivers/net/ethernet/intel/iavf/iavf_main.c", "iavf_tx_timeout"],
["drivers/net/ethernet/intel/ice/ice_main.c", "ice_tx_timeout"],
["drivers/net/ethernet/intel/ice/ice_main.c", "ice_tx_timeout"],
["drivers/net/ethernet/intel/igb/igb_main.c", "igb_tx_timeout"],
["drivers/net/ethernet/intel/igbvf/netdev.c", "igbvf_tx_timeout"],
["drivers/net/ethernet/intel/ixgb/ixgb_main.c", "ixgb_tx_timeout"],
["drivers/net/ethernet/intel/ixgbe/ixgbe_debugfs.c", "adapter->netdev->netdev_ops->ndo_tx_timeout(adapter->netdev);"],
["drivers/net/ethernet/intel/ixgbe/ixgbe_main.c", "ixgbe_tx_timeout"],
["drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c", "ixgbevf_tx_timeout"],
["drivers/net/ethernet/jme.c", "jme_tx_timeout"],
["drivers/net/ethernet/korina.c", "korina_tx_timeout"],
["drivers/net/ethernet/lantiq_etop.c", "ltq_etop_tx_timeout"],
["drivers/net/ethernet/marvell/mv643xx_eth.c", "mv643xx_eth_tx_timeout"],
["drivers/net/ethernet/marvell/pxa168_eth.c", "pxa168_eth_tx_timeout"],
["drivers/net/ethernet/marvell/skge.c", "skge_tx_timeout"],
["drivers/net/ethernet/marvell/sky2.c", "sky2_tx_timeout"],
["drivers/net/ethernet/marvell/sky2.c", "sky2_tx_timeout"],
["drivers/net/ethernet/mediatek/mtk_eth_soc.c", "mtk_tx_timeout"],
["drivers/net/ethernet/mellanox/mlx4/en_netdev.c", "mlx4_en_tx_timeout"],
["drivers/net/ethernet/mellanox/mlx4/en_netdev.c", "mlx4_en_tx_timeout"],
["drivers/net/ethernet/mellanox/mlx5/core/en_main.c", "mlx5e_tx_timeout"],
["drivers/net/ethernet/micrel/ks8842.c", "ks8842_tx_timeout"],
["drivers/net/ethernet/micrel/ksz884x.c", "netdev_tx_timeout"],
["drivers/net/ethernet/microchip/enc28j60.c", "enc28j60_tx_timeout"],
["drivers/net/ethernet/microchip/encx24j600.c", "encx24j600_tx_timeout"],
["drivers/net/ethernet/natsemi/sonic.h", "sonic_tx_timeout"],
["drivers/net/ethernet/natsemi/sonic.c", "sonic_tx_timeout"],
["drivers/net/ethernet/natsemi/jazzsonic.c", "sonic_tx_timeout"],
["drivers/net/ethernet/natsemi/macsonic.c", "sonic_tx_timeout"],
["drivers/net/ethernet/natsemi/natsemi.c", "ns_tx_timeout"],
["drivers/net/ethernet/natsemi/ns83820.c", "ns83820_tx_timeout"],
["drivers/net/ethernet/natsemi/xtsonic.c", "sonic_tx_timeout"],
["drivers/net/ethernet/neterion/s2io.h", "s2io_tx_watchdog"],
["drivers/net/ethernet/neterion/s2io.c", "s2io_tx_watchdog"],
["drivers/net/ethernet/neterion/vxge/vxge-main.c", "vxge_tx_watchdog"],
["drivers/net/ethernet/netronome/nfp/nfp_net_common.c", "nfp_net_tx_timeout"],
["drivers/net/ethernet/nvidia/forcedeth.c", "nv_tx_timeout"],
["drivers/net/ethernet/nvidia/forcedeth.c", "nv_tx_timeout"],
["drivers/net/ethernet/oki-semi/pch_gbe/pch_gbe_main.c", "pch_gbe_tx_timeout"],
["drivers/net/ethernet/packetengines/hamachi.c", "hamachi_tx_timeout"],
["drivers/net/ethernet/packetengines/yellowfin.c", "yellowfin_tx_timeout"],
["drivers/net/ethernet/pensando/ionic/ionic_lif.c", "ionic_tx_timeout"],
["drivers/net/ethernet/qlogic/netxen/netxen_nic_main.c", "netxen_tx_timeout"],
["drivers/net/ethernet/qlogic/qla3xxx.c", "ql3xxx_tx_timeout"],
["drivers/net/ethernet/qlogic/qlcnic/qlcnic_main.c", "qlcnic_tx_timeout"],
["drivers/net/ethernet/qualcomm/emac/emac.c", "emac_tx_timeout"],
["drivers/net/ethernet/qualcomm/qca_spi.c", "qcaspi_netdev_tx_timeout"],
["drivers/net/ethernet/qualcomm/qca_uart.c", "qcauart_netdev_tx_timeout"],
["drivers/net/ethernet/rdc/r6040.c", "r6040_tx_timeout"],
["drivers/net/ethernet/realtek/8139cp.c", "cp_tx_timeout"],
["drivers/net/ethernet/realtek/8139too.c", "rtl8139_tx_timeout"],
["drivers/net/ethernet/realtek/atp.c", "tx_timeout"],
["drivers/net/ethernet/realtek/r8169_main.c", "rtl8169_tx_timeout"],
["drivers/net/ethernet/renesas/ravb_main.c", "ravb_tx_timeout"],
["drivers/net/ethernet/renesas/sh_eth.c", "sh_eth_tx_timeout"],
["drivers/net/ethernet/renesas/sh_eth.c", "sh_eth_tx_timeout"],
["drivers/net/ethernet/samsung/sxgbe/sxgbe_main.c", "sxgbe_tx_timeout"],
["drivers/net/ethernet/seeq/ether3.c", "ether3_timeout"],
["drivers/net/ethernet/seeq/sgiseeq.c", "timeout"],
["drivers/net/ethernet/sfc/efx.c", "efx_watchdog"],
["drivers/net/ethernet/sfc/falcon/efx.c", "ef4_watchdog"],
["drivers/net/ethernet/sgi/ioc3-eth.c", "ioc3_timeout"],
["drivers/net/ethernet/sgi/meth.c", "meth_tx_timeout"],
["drivers/net/ethernet/silan/sc92031.c", "sc92031_tx_timeout"],
["drivers/net/ethernet/sis/sis190.c", "sis190_tx_timeout"],
["drivers/net/ethernet/sis/sis900.c", "sis900_tx_timeout"],
["drivers/net/ethernet/smsc/epic100.c", "epic_tx_timeout"],
["drivers/net/ethernet/smsc/smc911x.c", "smc911x_timeout"],
["drivers/net/ethernet/smsc/smc9194.c", "smc_timeout"],
["drivers/net/ethernet/smsc/smc91c92_cs.c", "smc_tx_timeout"],
["drivers/net/ethernet/smsc/smc91x.c", "smc_timeout"],
["drivers/net/ethernet/stmicro/stmmac/stmmac_main.c", "stmmac_tx_timeout"],
["drivers/net/ethernet/sun/cassini.c", "cas_tx_timeout"],
["drivers/net/ethernet/sun/ldmvsw.c", "sunvnet_tx_timeout_common"],
["drivers/net/ethernet/sun/niu.c", "niu_tx_timeout"],
["drivers/net/ethernet/sun/sunbmac.c", "bigmac_tx_timeout"],
["drivers/net/ethernet/sun/sungem.c", "gem_tx_timeout"],
["drivers/net/ethernet/sun/sunhme.c", "happy_meal_tx_timeout"],
["drivers/net/ethernet/sun/sunqe.c", "qe_tx_timeout"],
["drivers/net/ethernet/sun/sunvnet.c", "sunvnet_tx_timeout_common"],
["drivers/net/ethernet/sun/sunvnet_common.c", "sunvnet_tx_timeout_common"],
["drivers/net/ethernet/sun/sunvnet_common.h", "sunvnet_tx_timeout_common"],
["drivers/net/ethernet/synopsys/dwc-xlgmac-net.c", "xlgmac_tx_timeout"],
["drivers/net/ethernet/ti/cpmac.c", "cpmac_tx_timeout"],
["drivers/net/ethernet/ti/cpsw.c", "cpsw_ndo_tx_timeout"],
["drivers/net/ethernet/ti/cpsw_priv.c", "cpsw_ndo_tx_timeout"],
["drivers/net/ethernet/ti/cpsw_priv.h", "cpsw_ndo_tx_timeout"],
["drivers/net/ethernet/ti/davinci_emac.c", "emac_dev_tx_timeout"],
["drivers/net/ethernet/ti/netcp_core.c", "netcp_ndo_tx_timeout"],
["drivers/net/ethernet/ti/tlan.c", "tlan_tx_timeout"],
["drivers/net/ethernet/toshiba/ps3_gelic_net.h", "gelic_net_tx_timeout"],
["drivers/net/ethernet/toshiba/ps3_gelic_net.c", "gelic_net_tx_timeout"],
["drivers/net/ethernet/toshiba/ps3_gelic_wireless.c", "gelic_net_tx_timeout"],
["drivers/net/ethernet/toshiba/spider_net.c", "spider_net_tx_timeout"],
["drivers/net/ethernet/toshiba/tc35815.c", "tc35815_tx_timeout"],
["drivers/net/ethernet/via/via-rhine.c", "rhine_tx_timeout"],
["drivers/net/ethernet/wiznet/w5100.c", "w5100_tx_timeout"],
["drivers/net/ethernet/wiznet/w5300.c", "w5300_tx_timeout"],
["drivers/net/ethernet/xilinx/xilinx_emaclite.c", "xemaclite_tx_timeout"],
["drivers/net/ethernet/xircom/xirc2ps_cs.c", "xirc_tx_timeout"],
["drivers/net/fjes/fjes_main.c", "fjes_tx_retry"],
["drivers/net/slip/slip.c", "sl_tx_timeout"],
["include/linux/usb/usbnet.h", "usbnet_tx_timeout"],
["drivers/net/usb/aqc111.c", "usbnet_tx_timeout"],
["drivers/net/usb/asix_devices.c", "usbnet_tx_timeout"],
["drivers/net/usb/asix_devices.c", "usbnet_tx_timeout"],
["drivers/net/usb/asix_devices.c", "usbnet_tx_timeout"],
["drivers/net/usb/ax88172a.c", "usbnet_tx_timeout"],
["drivers/net/usb/ax88179_178a.c", "usbnet_tx_timeout"],
["drivers/net/usb/catc.c", "catc_tx_timeout"],
["drivers/net/usb/cdc_mbim.c", "usbnet_tx_timeout"],
["drivers/net/usb/cdc_ncm.c", "usbnet_tx_timeout"],
["drivers/net/usb/dm9601.c", "usbnet_tx_timeout"],
["drivers/net/usb/hso.c", "hso_net_tx_timeout"],
["drivers/net/usb/int51x1.c", "usbnet_tx_timeout"],
["drivers/net/usb/ipheth.c", "ipheth_tx_timeout"],
["drivers/net/usb/kaweth.c", "kaweth_tx_timeout"],
["drivers/net/usb/lan78xx.c", "lan78xx_tx_timeout"],
["drivers/net/usb/mcs7830.c", "usbnet_tx_timeout"],
["drivers/net/usb/pegasus.c", "pegasus_tx_timeout"],
["drivers/net/usb/qmi_wwan.c", "usbnet_tx_timeout"],
["drivers/net/usb/r8152.c", "rtl8152_tx_timeout"],
["drivers/net/usb/rndis_host.c", "usbnet_tx_timeout"],
["drivers/net/usb/rtl8150.c", "rtl8150_tx_timeout"],
["drivers/net/usb/sierra_net.c", "usbnet_tx_timeout"],
["drivers/net/usb/smsc75xx.c", "usbnet_tx_timeout"],
["drivers/net/usb/smsc95xx.c", "usbnet_tx_timeout"],
["drivers/net/usb/sr9700.c", "usbnet_tx_timeout"],
["drivers/net/usb/sr9800.c", "usbnet_tx_timeout"],
["drivers/net/usb/usbnet.c", "usbnet_tx_timeout"],
["drivers/net/vmxnet3/vmxnet3_drv.c", "vmxnet3_tx_timeout"],
["drivers/net/wan/cosa.c", "cosa_net_timeout"],
["drivers/net/wan/farsync.c", "fst_tx_timeout"],
["drivers/net/wan/fsl_ucc_hdlc.c", "uhdlc_tx_timeout"],
["drivers/net/wan/lmc/lmc_main.c", "lmc_driver_timeout"],
["drivers/net/wan/x25_asy.c", "x25_asy_timeout"],
["drivers/net/wimax/i2400m/netdev.c", "i2400m_tx_timeout"],
["drivers/net/wireless/intel/ipw2x00/ipw2100.c", "ipw2100_tx_timeout"],
["drivers/net/wireless/intersil/hostap/hostap_main.c", "prism2_tx_timeout"],
["drivers/net/wireless/intersil/hostap/hostap_main.c", "prism2_tx_timeout"],
["drivers/net/wireless/intersil/hostap/hostap_main.c", "prism2_tx_timeout"],
["drivers/net/wireless/intersil/orinoco/main.c", "orinoco_tx_timeout"],
["drivers/net/wireless/intersil/orinoco/orinoco_usb.c", "orinoco_tx_timeout"],
["drivers/net/wireless/intersil/orinoco/orinoco.h", "orinoco_tx_timeout"],
["drivers/net/wireless/intersil/prism54/islpci_dev.c", "islpci_eth_tx_timeout"],
["drivers/net/wireless/intersil/prism54/islpci_eth.c", "islpci_eth_tx_timeout"],
["drivers/net/wireless/intersil/prism54/islpci_eth.h", "islpci_eth_tx_timeout"],
["drivers/net/wireless/marvell/mwifiex/main.c", "mwifiex_tx_timeout"],
["drivers/net/wireless/quantenna/qtnfmac/core.c", "qtnf_netdev_tx_timeout"],
["drivers/net/wireless/quantenna/qtnfmac/core.h", "qtnf_netdev_tx_timeout"],
["drivers/net/wireless/rndis_wlan.c", "usbnet_tx_timeout"],
["drivers/net/wireless/wl3501_cs.c", "wl3501_tx_timeout"],
["drivers/net/wireless/zydas/zd1201.c", "zd1201_tx_timeout"],
["drivers/s390/net/qeth_core.h", "qeth_tx_timeout"],
["drivers/s390/net/qeth_core_main.c", "qeth_tx_timeout"],
["drivers/s390/net/qeth_l2_main.c", "qeth_tx_timeout"],
["drivers/s390/net/qeth_l2_main.c", "qeth_tx_timeout"],
["drivers/s390/net/qeth_l3_main.c", "qeth_tx_timeout"],
["drivers/s390/net/qeth_l3_main.c", "qeth_tx_timeout"],
["drivers/staging/ks7010/ks_wlan_net.c", "ks_wlan_tx_timeout"],
["drivers/staging/qlge/qlge_main.c", "qlge_tx_timeout"],
["drivers/staging/rtl8192e/rtl8192e/rtl_core.c", "_rtl92e_tx_timeout"],
["drivers/staging/rtl8192u/r8192U_core.c", "tx_timeout"],
["drivers/staging/unisys/visornic/visornic_main.c", "visornic_xmit_timeout"],
["drivers/staging/wlan-ng/p80211netdev.c", "p80211knetdev_tx_timeout"],
["drivers/tty/n_gsm.c", "gsm_mux_net_tx_timeout"],
["drivers/tty/synclink.c", "hdlcdev_tx_timeout"],
["drivers/tty/synclink_gt.c", "hdlcdev_tx_timeout"],
["drivers/tty/synclinkmp.c", "hdlcdev_tx_timeout"],
["net/atm/lec.c", "lec_tx_timeout"],
["net/bluetooth/bnep/netdev.c", "bnep_net_timeout"]
);
for my $p (@work) {
my @pair = @$p;
my $file = $pair[0];
my $func = $pair[1];
print STDERR $file , ": ", $func,"\n";
our @ARGV = ($file);
while (<ARGV>) {
if (m/($func\s*\(struct\s+net_device\s+\*[A-Za-z_]?[A-Za-z-0-9_]*)(\))/) {
print STDERR "found $1+$2 in $file\n";
}
if (s/($func\s*\(struct\s+net_device\s+\*[A-Za-z_]?[A-Za-z-0-9_]*)(\))/$1, unsigned int txqueue$2/) {
print STDERR "$func found in $file\n";
}
print;
}
}
where the list of files and functions is simply from:
git grep ndo_tx_timeout, with manual addition of headers
in the rare cases where the function is from a header,
then manually changing the few places which actually
call ndo_tx_timeout.
Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
Acked-by: Heiner Kallweit <hkallweit1@gmail.com>
Acked-by: Jakub Kicinski <jakub.kicinski@netronome.com>
Acked-by: Shannon Nelson <snelson@pensando.io>
Reviewed-by: Martin Habets <mhabets@solarflare.com>
changes from v9:
fixup a forward declaration
changes from v9:
more leftovers from v3 change
changes from v8:
fix up a missing direct call to timeout
rebased on net-next
changes from v7:
fixup leftovers from v3 change
changes from v6:
fix typo in rtl driver
changes from v5:
add missing files (allow any net device argument name)
changes from v4:
add a missing driver header
changes from v3:
change queue # to unsigned
Changes from v2:
added headers
Changes from v1:
Fix errors found by kbuild:
generalize the pattern a bit, to pick up
a couple of instances missed by the previous
version.
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-12-10 22:23:51 +08:00
|
|
|
static void hns3_nic_net_timeout(struct net_device *ndev, unsigned int txqueue)
|
2017-11-02 20:45:20 +08:00
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(ndev);
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
|
|
|
|
if (!hns3_get_tx_timeo_queue_info(ndev))
|
|
|
|
return;
|
|
|
|
|
2018-11-07 12:06:14 +08:00
|
|
|
/* request the reset, and let the hclge to determine
|
|
|
|
* which reset level should be done
|
|
|
|
*/
|
2017-11-02 20:45:20 +08:00
|
|
|
if (h->ae_algo->ops->reset_event)
|
2018-10-20 03:15:27 +08:00
|
|
|
h->ae_algo->ops->reset_event(h->pdev, h);
|
2017-11-02 20:45:20 +08:00
|
|
|
}
|
|
|
|
|
2019-05-24 19:19:47 +08:00
|
|
|
#ifdef CONFIG_RFS_ACCEL
|
|
|
|
static int hns3_rx_flow_steer(struct net_device *dev, const struct sk_buff *skb,
|
|
|
|
u16 rxq_index, u32 flow_id)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(dev);
|
|
|
|
struct flow_keys fkeys;
|
|
|
|
|
|
|
|
if (!h->ae_algo->ops->add_arfs_entry)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
if (skb->encapsulation)
|
|
|
|
return -EPROTONOSUPPORT;
|
|
|
|
|
|
|
|
if (!skb_flow_dissect_flow_keys(skb, &fkeys, 0))
|
|
|
|
return -EPROTONOSUPPORT;
|
|
|
|
|
|
|
|
if ((fkeys.basic.n_proto != htons(ETH_P_IP) &&
|
|
|
|
fkeys.basic.n_proto != htons(ETH_P_IPV6)) ||
|
|
|
|
(fkeys.basic.ip_proto != IPPROTO_TCP &&
|
|
|
|
fkeys.basic.ip_proto != IPPROTO_UDP))
|
|
|
|
return -EPROTONOSUPPORT;
|
|
|
|
|
|
|
|
return h->ae_algo->ops->add_arfs_entry(h, rxq_index, flow_id, &fkeys);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-10-08 09:20:04 +08:00
|
|
|
static int hns3_nic_get_vf_config(struct net_device *ndev, int vf,
|
|
|
|
struct ifla_vf_info *ivf)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(ndev);
|
|
|
|
|
|
|
|
if (!h->ae_algo->ops->get_vf_config)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return h->ae_algo->ops->get_vf_config(h, vf, ivf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_set_vf_link_state(struct net_device *ndev, int vf,
|
|
|
|
int link_state)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(ndev);
|
|
|
|
|
|
|
|
if (!h->ae_algo->ops->set_vf_link_state)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return h->ae_algo->ops->set_vf_link_state(h, vf, link_state);
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:07 +08:00
|
|
|
static int hns3_nic_set_vf_rate(struct net_device *ndev, int vf,
|
|
|
|
int min_tx_rate, int max_tx_rate)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(ndev);
|
|
|
|
|
|
|
|
if (!h->ae_algo->ops->set_vf_rate)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return h->ae_algo->ops->set_vf_rate(h, vf, min_tx_rate, max_tx_rate,
|
|
|
|
false);
|
|
|
|
}
|
|
|
|
|
2019-10-08 09:20:08 +08:00
|
|
|
static int hns3_nic_set_vf_mac(struct net_device *netdev, int vf_id, u8 *mac)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
|
|
|
|
if (!h->ae_algo->ops->set_vf_mac)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
if (is_multicast_ether_addr(mac)) {
|
|
|
|
netdev_err(netdev,
|
|
|
|
"Invalid MAC:%pM specified. Could not set MAC\n",
|
|
|
|
mac);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return h->ae_algo->ops->set_vf_mac(h, vf_id, mac);
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static const struct net_device_ops hns3_nic_netdev_ops = {
|
|
|
|
.ndo_open = hns3_nic_net_open,
|
|
|
|
.ndo_stop = hns3_nic_net_stop,
|
|
|
|
.ndo_start_xmit = hns3_nic_net_xmit,
|
2017-11-02 20:45:20 +08:00
|
|
|
.ndo_tx_timeout = hns3_nic_net_timeout,
|
2017-08-02 23:59:45 +08:00
|
|
|
.ndo_set_mac_address = hns3_nic_net_set_mac_address,
|
2018-09-20 01:29:51 +08:00
|
|
|
.ndo_do_ioctl = hns3_nic_do_ioctl,
|
2017-08-22 00:05:24 +08:00
|
|
|
.ndo_change_mtu = hns3_nic_change_mtu,
|
2017-08-02 23:59:45 +08:00
|
|
|
.ndo_set_features = hns3_nic_set_features,
|
2019-12-19 14:57:44 +08:00
|
|
|
.ndo_features_check = hns3_features_check,
|
2017-08-02 23:59:45 +08:00
|
|
|
.ndo_get_stats64 = hns3_nic_get_stats64,
|
|
|
|
.ndo_setup_tc = hns3_nic_setup_tc,
|
|
|
|
.ndo_set_rx_mode = hns3_nic_set_rx_mode,
|
|
|
|
.ndo_vlan_rx_add_vid = hns3_vlan_rx_add_vid,
|
|
|
|
.ndo_vlan_rx_kill_vid = hns3_vlan_rx_kill_vid,
|
|
|
|
.ndo_set_vf_vlan = hns3_ndo_set_vf_vlan,
|
2019-10-08 09:20:05 +08:00
|
|
|
.ndo_set_vf_spoofchk = hns3_set_vf_spoofchk,
|
2019-10-08 09:20:06 +08:00
|
|
|
.ndo_set_vf_trust = hns3_set_vf_trust,
|
2019-05-24 19:19:47 +08:00
|
|
|
#ifdef CONFIG_RFS_ACCEL
|
|
|
|
.ndo_rx_flow_steer = hns3_rx_flow_steer,
|
|
|
|
#endif
|
2019-10-08 09:20:04 +08:00
|
|
|
.ndo_get_vf_config = hns3_nic_get_vf_config,
|
|
|
|
.ndo_set_vf_link_state = hns3_nic_set_vf_link_state,
|
2019-10-08 09:20:07 +08:00
|
|
|
.ndo_set_vf_rate = hns3_nic_set_vf_rate,
|
2019-10-08 09:20:08 +08:00
|
|
|
.ndo_set_vf_mac = hns3_nic_set_vf_mac,
|
2017-08-02 23:59:45 +08:00
|
|
|
};
|
|
|
|
|
2019-04-19 11:05:47 +08:00
|
|
|
bool hns3_is_phys_func(struct pci_dev *pdev)
|
2018-05-16 02:20:05 +08:00
|
|
|
{
|
|
|
|
u32 dev_id = pdev->device;
|
|
|
|
|
|
|
|
switch (dev_id) {
|
|
|
|
case HNAE3_DEV_ID_GE:
|
|
|
|
case HNAE3_DEV_ID_25GE:
|
|
|
|
case HNAE3_DEV_ID_25GE_RDMA:
|
|
|
|
case HNAE3_DEV_ID_25GE_RDMA_MACSEC:
|
|
|
|
case HNAE3_DEV_ID_50GE_RDMA:
|
|
|
|
case HNAE3_DEV_ID_50GE_RDMA_MACSEC:
|
|
|
|
case HNAE3_DEV_ID_100G_RDMA_MACSEC:
|
2020-09-25 08:26:17 +08:00
|
|
|
case HNAE3_DEV_ID_200G_RDMA:
|
2018-05-16 02:20:05 +08:00
|
|
|
return true;
|
2020-09-25 08:26:18 +08:00
|
|
|
case HNAE3_DEV_ID_VF:
|
|
|
|
case HNAE3_DEV_ID_RDMA_DCB_PFC_VF:
|
2018-05-16 02:20:05 +08:00
|
|
|
return false;
|
|
|
|
default:
|
2019-10-31 19:23:23 +08:00
|
|
|
dev_warn(&pdev->dev, "un-recognized pci device-id %u",
|
2018-05-16 02:20:05 +08:00
|
|
|
dev_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_disable_sriov(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
/* If our VFs are assigned we cannot shut down SR-IOV
|
|
|
|
* without causing issues, so just leave the hardware
|
|
|
|
* available but disabled
|
|
|
|
*/
|
|
|
|
if (pci_vfs_assigned(pdev)) {
|
|
|
|
dev_warn(&pdev->dev,
|
|
|
|
"disabling driver while VFs are assigned\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pci_disable_sriov(pdev);
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* hns3_probe - Device initialization routine
|
|
|
|
* @pdev: PCI device information struct
|
|
|
|
* @ent: entry in hns3_pci_tbl
|
|
|
|
*
|
|
|
|
* hns3_probe initializes a PF identified by a pci_dev structure.
|
|
|
|
* The OS initialization, configuring of the PF private structure,
|
|
|
|
* and a hardware reset occur.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, negative on failure
|
|
|
|
*/
|
|
|
|
static int hns3_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev;
|
|
|
|
int ret;
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
ae_dev = devm_kzalloc(&pdev->dev, sizeof(*ae_dev), GFP_KERNEL);
|
2020-01-21 16:42:13 +08:00
|
|
|
if (!ae_dev)
|
|
|
|
return -ENOMEM;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
ae_dev->pdev = pdev;
|
2017-09-20 18:52:50 +08:00
|
|
|
ae_dev->flag = ent->driver_data;
|
2017-08-02 23:59:45 +08:00
|
|
|
pci_set_drvdata(pdev, ae_dev);
|
|
|
|
|
2019-01-31 04:55:44 +08:00
|
|
|
ret = hnae3_register_ae_dev(ae_dev);
|
2020-06-18 09:02:07 +08:00
|
|
|
if (ret)
|
2019-01-31 04:55:44 +08:00
|
|
|
pci_set_drvdata(pdev, NULL);
|
2018-05-16 02:20:05 +08:00
|
|
|
|
2019-01-31 04:55:44 +08:00
|
|
|
return ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* hns3_remove - Device removal routine
|
|
|
|
* @pdev: PCI device information struct
|
|
|
|
*/
|
|
|
|
static void hns3_remove(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
|
|
|
|
2018-05-16 02:20:05 +08:00
|
|
|
if (hns3_is_phys_func(pdev) && IS_ENABLED(CONFIG_PCI_IOV))
|
|
|
|
hns3_disable_sriov(pdev);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
hnae3_unregister_ae_dev(ae_dev);
|
2019-01-27 00:49:11 +08:00
|
|
|
pci_set_drvdata(pdev, NULL);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2018-05-16 02:20:13 +08:00
|
|
|
/**
|
|
|
|
* hns3_pci_sriov_configure
|
|
|
|
* @pdev: pointer to a pci_dev structure
|
|
|
|
* @num_vfs: number of VFs to allocate
|
|
|
|
*
|
|
|
|
* Enable or change the number of VFs. Called when the user updates the number
|
|
|
|
* of VFs in sysfs.
|
|
|
|
**/
|
2018-05-19 23:53:15 +08:00
|
|
|
static int hns3_pci_sriov_configure(struct pci_dev *pdev, int num_vfs)
|
2018-05-16 02:20:13 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!(hns3_is_phys_func(pdev) && IS_ENABLED(CONFIG_PCI_IOV))) {
|
|
|
|
dev_warn(&pdev->dev, "Can not config SRIOV\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (num_vfs) {
|
|
|
|
ret = pci_enable_sriov(pdev, num_vfs);
|
|
|
|
if (ret)
|
|
|
|
dev_err(&pdev->dev, "SRIOV enable failed %d\n", ret);
|
2018-05-19 23:53:15 +08:00
|
|
|
else
|
|
|
|
return num_vfs;
|
2018-05-16 02:20:13 +08:00
|
|
|
} else if (!pci_vfs_assigned(pdev)) {
|
|
|
|
pci_disable_sriov(pdev);
|
|
|
|
} else {
|
|
|
|
dev_warn(&pdev->dev,
|
|
|
|
"Unable to free VFs because some are assigned to VMs.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-03 18:21:49 +08:00
|
|
|
static void hns3_shutdown(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
|
|
|
|
|
|
|
hnae3_unregister_ae_dev(ae_dev);
|
|
|
|
pci_set_drvdata(pdev, NULL);
|
|
|
|
|
|
|
|
if (system_state == SYSTEM_POWER_OFF)
|
|
|
|
pci_set_power_state(pdev, PCI_D3hot);
|
|
|
|
}
|
|
|
|
|
2021-04-08 11:40:05 +08:00
|
|
|
static int __maybe_unused hns3_suspend(struct device *dev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = dev_get_drvdata(dev);
|
|
|
|
|
2021-04-10 00:37:26 +08:00
|
|
|
if (ae_dev && hns3_is_phys_func(ae_dev->pdev)) {
|
2021-04-08 11:40:05 +08:00
|
|
|
dev_info(dev, "Begin to suspend.\n");
|
2021-04-10 00:37:26 +08:00
|
|
|
if (ae_dev->ops && ae_dev->ops->reset_prepare)
|
2021-04-08 11:40:05 +08:00
|
|
|
ae_dev->ops->reset_prepare(ae_dev, HNAE3_FUNC_RESET);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __maybe_unused hns3_resume(struct device *dev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = dev_get_drvdata(dev);
|
|
|
|
|
2021-04-10 00:37:26 +08:00
|
|
|
if (ae_dev && hns3_is_phys_func(ae_dev->pdev)) {
|
2021-04-08 11:40:05 +08:00
|
|
|
dev_info(dev, "Begin to resume.\n");
|
2021-04-10 00:37:26 +08:00
|
|
|
if (ae_dev->ops && ae_dev->ops->reset_done)
|
2021-04-08 11:40:05 +08:00
|
|
|
ae_dev->ops->reset_done(ae_dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-20 03:15:26 +08:00
|
|
|
static pci_ers_result_t hns3_error_detected(struct pci_dev *pdev,
|
|
|
|
pci_channel_state_t state)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
|
|
|
pci_ers_result_t ret;
|
|
|
|
|
2021-02-09 10:41:55 +08:00
|
|
|
dev_info(&pdev->dev, "PCI error detected, state(=%u)!!\n", state);
|
2018-10-20 03:15:26 +08:00
|
|
|
|
|
|
|
if (state == pci_channel_io_perm_failure)
|
|
|
|
return PCI_ERS_RESULT_DISCONNECT;
|
|
|
|
|
2019-06-03 10:09:17 +08:00
|
|
|
if (!ae_dev || !ae_dev->ops) {
|
2018-10-20 03:15:26 +08:00
|
|
|
dev_err(&pdev->dev,
|
2019-06-03 10:09:17 +08:00
|
|
|
"Can't recover - error happened before device initialized\n");
|
2018-10-20 03:15:26 +08:00
|
|
|
return PCI_ERS_RESULT_NONE;
|
|
|
|
}
|
|
|
|
|
2018-12-08 05:08:02 +08:00
|
|
|
if (ae_dev->ops->handle_hw_ras_error)
|
|
|
|
ret = ae_dev->ops->handle_hw_ras_error(ae_dev);
|
2018-10-20 03:15:26 +08:00
|
|
|
else
|
|
|
|
return PCI_ERS_RESULT_NONE;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-10-20 03:15:27 +08:00
|
|
|
static pci_ers_result_t hns3_slot_reset(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
2019-06-17 19:42:14 +08:00
|
|
|
const struct hnae3_ae_ops *ops;
|
2019-06-13 17:12:21 +08:00
|
|
|
enum hnae3_reset_type reset_type;
|
2018-10-20 03:15:27 +08:00
|
|
|
struct device *dev = &pdev->dev;
|
|
|
|
|
2019-06-03 10:09:17 +08:00
|
|
|
if (!ae_dev || !ae_dev->ops)
|
|
|
|
return PCI_ERS_RESULT_NONE;
|
|
|
|
|
2019-06-17 19:42:14 +08:00
|
|
|
ops = ae_dev->ops;
|
2018-10-20 03:15:27 +08:00
|
|
|
/* request the reset */
|
2019-09-11 10:40:38 +08:00
|
|
|
if (ops->reset_event && ops->get_reset_level &&
|
|
|
|
ops->set_default_reset_request) {
|
2019-06-20 16:52:41 +08:00
|
|
|
if (ae_dev->hw_err_reset_req) {
|
2019-06-13 17:12:21 +08:00
|
|
|
reset_type = ops->get_reset_level(ae_dev,
|
|
|
|
&ae_dev->hw_err_reset_req);
|
|
|
|
ops->set_default_reset_request(ae_dev, reset_type);
|
|
|
|
dev_info(dev, "requesting reset due to PCI error\n");
|
|
|
|
ops->reset_event(pdev, NULL);
|
|
|
|
}
|
2019-03-10 14:47:51 +08:00
|
|
|
|
2018-10-20 03:15:27 +08:00
|
|
|
return PCI_ERS_RESULT_RECOVERED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PCI_ERS_RESULT_DISCONNECT;
|
|
|
|
}
|
|
|
|
|
2018-11-09 22:07:54 +08:00
|
|
|
static void hns3_reset_prepare(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
|
|
|
|
2020-03-07 11:42:49 +08:00
|
|
|
dev_info(&pdev->dev, "FLR prepare\n");
|
2021-04-08 11:40:04 +08:00
|
|
|
if (ae_dev && ae_dev->ops && ae_dev->ops->reset_prepare)
|
|
|
|
ae_dev->ops->reset_prepare(ae_dev, HNAE3_FLR_RESET);
|
2018-11-09 22:07:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_reset_done(struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
|
|
|
|
2020-03-07 11:42:49 +08:00
|
|
|
dev_info(&pdev->dev, "FLR done\n");
|
2021-04-08 11:40:04 +08:00
|
|
|
if (ae_dev && ae_dev->ops && ae_dev->ops->reset_done)
|
|
|
|
ae_dev->ops->reset_done(ae_dev);
|
2018-11-09 22:07:54 +08:00
|
|
|
}
|
|
|
|
|
2018-10-20 03:15:26 +08:00
|
|
|
static const struct pci_error_handlers hns3_err_handler = {
|
|
|
|
.error_detected = hns3_error_detected,
|
2018-10-20 03:15:27 +08:00
|
|
|
.slot_reset = hns3_slot_reset,
|
2018-11-09 22:07:54 +08:00
|
|
|
.reset_prepare = hns3_reset_prepare,
|
|
|
|
.reset_done = hns3_reset_done,
|
2018-10-20 03:15:26 +08:00
|
|
|
};
|
|
|
|
|
2021-04-08 11:40:05 +08:00
|
|
|
static SIMPLE_DEV_PM_OPS(hns3_pm_ops, hns3_suspend, hns3_resume);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static struct pci_driver hns3_driver = {
|
|
|
|
.name = hns3_driver_name,
|
|
|
|
.id_table = hns3_pci_tbl,
|
|
|
|
.probe = hns3_probe,
|
|
|
|
.remove = hns3_remove,
|
2018-09-03 18:21:49 +08:00
|
|
|
.shutdown = hns3_shutdown,
|
2021-04-08 11:40:05 +08:00
|
|
|
.driver.pm = &hns3_pm_ops,
|
2018-05-16 02:20:13 +08:00
|
|
|
.sriov_configure = hns3_pci_sriov_configure,
|
2018-10-20 03:15:26 +08:00
|
|
|
.err_handler = &hns3_err_handler,
|
2017-08-02 23:59:45 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* set default feature to hns3 */
|
|
|
|
static void hns3_set_default_feature(struct net_device *netdev)
|
|
|
|
{
|
2018-08-15 00:13:19 +08:00
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
struct pci_dev *pdev = h->pdev;
|
2020-09-27 15:12:39 +08:00
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
2018-08-15 00:13:19 +08:00
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
netdev->priv_flags |= IFF_UNICAST_FLT;
|
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
netdev->hw_enc_features |= NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_GSO |
|
2017-08-02 23:59:45 +08:00
|
|
|
NETIF_F_GRO | NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_GSO_GRE |
|
|
|
|
NETIF_F_GSO_GRE_CSUM | NETIF_F_GSO_UDP_TUNNEL |
|
2020-11-28 11:51:46 +08:00
|
|
|
NETIF_F_SCTP_CRC | NETIF_F_TSO_MANGLEID | NETIF_F_FRAGLIST;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
netdev->gso_partial_features |= NETIF_F_GSO_GRE_CSUM;
|
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
netdev->features |= NETIF_F_HW_VLAN_CTAG_FILTER |
|
2017-12-22 12:21:47 +08:00
|
|
|
NETIF_F_HW_VLAN_CTAG_TX | NETIF_F_HW_VLAN_CTAG_RX |
|
2017-08-02 23:59:45 +08:00
|
|
|
NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_GSO |
|
|
|
|
NETIF_F_GRO | NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_GSO_GRE |
|
|
|
|
NETIF_F_GSO_GRE_CSUM | NETIF_F_GSO_UDP_TUNNEL |
|
2020-11-28 11:51:46 +08:00
|
|
|
NETIF_F_SCTP_CRC | NETIF_F_FRAGLIST;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
netdev->vlan_features |= NETIF_F_RXCSUM |
|
2017-08-02 23:59:45 +08:00
|
|
|
NETIF_F_SG | NETIF_F_GSO | NETIF_F_GRO |
|
|
|
|
NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_GSO_GRE |
|
|
|
|
NETIF_F_GSO_GRE_CSUM | NETIF_F_GSO_UDP_TUNNEL |
|
2020-11-28 11:51:46 +08:00
|
|
|
NETIF_F_SCTP_CRC | NETIF_F_FRAGLIST;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
netdev->hw_features |= NETIF_F_HW_VLAN_CTAG_TX |
|
|
|
|
NETIF_F_HW_VLAN_CTAG_RX |
|
2017-08-02 23:59:45 +08:00
|
|
|
NETIF_F_RXCSUM | NETIF_F_SG | NETIF_F_GSO |
|
|
|
|
NETIF_F_GRO | NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_GSO_GRE |
|
|
|
|
NETIF_F_GSO_GRE_CSUM | NETIF_F_GSO_UDP_TUNNEL |
|
2020-11-28 11:51:46 +08:00
|
|
|
NETIF_F_SCTP_CRC | NETIF_F_FRAGLIST;
|
2018-08-15 00:13:19 +08:00
|
|
|
|
2020-09-27 15:12:39 +08:00
|
|
|
if (ae_dev->dev_version >= HNAE3_DEVICE_VERSION_V2) {
|
2019-02-02 22:39:35 +08:00
|
|
|
netdev->hw_features |= NETIF_F_GRO_HW;
|
2018-11-15 17:29:23 +08:00
|
|
|
netdev->features |= NETIF_F_GRO_HW;
|
2018-10-01 19:46:47 +08:00
|
|
|
|
|
|
|
if (!(h->flags & HNAE3_SUPPORT_VF)) {
|
|
|
|
netdev->hw_features |= NETIF_F_NTUPLE;
|
|
|
|
netdev->features |= NETIF_F_NTUPLE;
|
|
|
|
}
|
|
|
|
}
|
2020-09-29 17:32:01 +08:00
|
|
|
|
|
|
|
if (test_bit(HNAE3_DEV_SUPPORT_UDP_GSO_B, ae_dev->caps)) {
|
|
|
|
netdev->hw_features |= NETIF_F_GSO_UDP_L4;
|
|
|
|
netdev->features |= NETIF_F_GSO_UDP_L4;
|
|
|
|
netdev->vlan_features |= NETIF_F_GSO_UDP_L4;
|
|
|
|
netdev->hw_enc_features |= NETIF_F_GSO_UDP_L4;
|
|
|
|
}
|
2020-11-28 11:51:45 +08:00
|
|
|
|
|
|
|
if (test_bit(HNAE3_DEV_SUPPORT_HW_TX_CSUM_B, ae_dev->caps)) {
|
|
|
|
netdev->hw_features |= NETIF_F_HW_CSUM;
|
|
|
|
netdev->features |= NETIF_F_HW_CSUM;
|
|
|
|
netdev->vlan_features |= NETIF_F_HW_CSUM;
|
|
|
|
netdev->hw_enc_features |= NETIF_F_HW_CSUM;
|
|
|
|
} else {
|
|
|
|
netdev->hw_features |= NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM;
|
|
|
|
netdev->features |= NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM;
|
|
|
|
netdev->vlan_features |= NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM;
|
|
|
|
netdev->hw_enc_features |= NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM;
|
|
|
|
}
|
2020-11-28 11:51:47 +08:00
|
|
|
|
|
|
|
if (test_bit(HNAE3_DEV_SUPPORT_UDP_TUNNEL_CSUM_B, ae_dev->caps)) {
|
|
|
|
netdev->hw_features |= NETIF_F_GSO_UDP_TUNNEL_CSUM;
|
|
|
|
netdev->features |= NETIF_F_GSO_UDP_TUNNEL_CSUM;
|
|
|
|
netdev->vlan_features |= NETIF_F_GSO_UDP_TUNNEL_CSUM;
|
|
|
|
netdev->hw_enc_features |= NETIF_F_GSO_UDP_TUNNEL_CSUM;
|
|
|
|
}
|
2020-12-10 11:42:09 +08:00
|
|
|
|
|
|
|
if (test_bit(HNAE3_DEV_SUPPORT_FD_FORWARD_TC_B, ae_dev->caps)) {
|
|
|
|
netdev->hw_features |= NETIF_F_HW_TC;
|
|
|
|
netdev->features |= NETIF_F_HW_TC;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_alloc_buffer(struct hns3_enet_ring *ring,
|
|
|
|
struct hns3_desc_cb *cb)
|
|
|
|
{
|
2019-07-29 10:53:30 +08:00
|
|
|
unsigned int order = hns3_page_order(ring);
|
2017-08-02 23:59:45 +08:00
|
|
|
struct page *p;
|
|
|
|
|
|
|
|
p = dev_alloc_pages(order);
|
|
|
|
if (!p)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
cb->priv = p;
|
|
|
|
cb->page_offset = 0;
|
|
|
|
cb->reuse_flag = 0;
|
|
|
|
cb->buf = page_address(p);
|
2019-07-29 10:53:30 +08:00
|
|
|
cb->length = hns3_page_size(ring);
|
2017-08-02 23:59:45 +08:00
|
|
|
cb->type = DESC_TYPE_PAGE;
|
2020-09-16 17:33:45 +08:00
|
|
|
page_ref_add(p, USHRT_MAX - 1);
|
|
|
|
cb->pagecnt_bias = USHRT_MAX;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_free_buffer(struct hns3_enet_ring *ring,
|
2020-09-16 17:33:50 +08:00
|
|
|
struct hns3_desc_cb *cb, int budget)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
if (cb->type == DESC_TYPE_SKB)
|
2020-09-16 17:33:50 +08:00
|
|
|
napi_consume_skb(cb->priv, budget);
|
2020-09-16 17:33:45 +08:00
|
|
|
else if (!HNAE3_IS_TX_RING(ring) && cb->pagecnt_bias)
|
|
|
|
__page_frag_cache_drain(cb->priv, cb->pagecnt_bias);
|
2017-08-02 23:59:45 +08:00
|
|
|
memset(cb, 0, sizeof(*cb));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_map_buffer(struct hns3_enet_ring *ring, struct hns3_desc_cb *cb)
|
|
|
|
{
|
|
|
|
cb->dma = dma_map_page(ring_to_dev(ring), cb->priv, 0,
|
|
|
|
cb->length, ring_to_dma_dir(ring));
|
|
|
|
|
2018-09-21 23:41:44 +08:00
|
|
|
if (unlikely(dma_mapping_error(ring_to_dev(ring), cb->dma)))
|
2017-08-02 23:59:45 +08:00
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_unmap_buffer(struct hns3_enet_ring *ring,
|
|
|
|
struct hns3_desc_cb *cb)
|
|
|
|
{
|
2020-03-28 15:09:56 +08:00
|
|
|
if (cb->type == DESC_TYPE_SKB || cb->type == DESC_TYPE_FRAGLIST_SKB)
|
2017-08-02 23:59:45 +08:00
|
|
|
dma_unmap_single(ring_to_dev(ring), cb->dma, cb->length,
|
|
|
|
ring_to_dma_dir(ring));
|
2018-10-16 19:58:52 +08:00
|
|
|
else if (cb->length)
|
2017-08-02 23:59:45 +08:00
|
|
|
dma_unmap_page(ring_to_dev(ring), cb->dma, cb->length,
|
|
|
|
ring_to_dma_dir(ring));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_buffer_detach(struct hns3_enet_ring *ring, int i)
|
|
|
|
{
|
|
|
|
hns3_unmap_buffer(ring, &ring->desc_cb[i]);
|
|
|
|
ring->desc[i].addr = 0;
|
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:50 +08:00
|
|
|
static void hns3_free_buffer_detach(struct hns3_enet_ring *ring, int i,
|
|
|
|
int budget)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hns3_desc_cb *cb = &ring->desc_cb[i];
|
|
|
|
|
|
|
|
if (!ring->desc_cb[i].dma)
|
|
|
|
return;
|
|
|
|
|
|
|
|
hns3_buffer_detach(ring, i);
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_free_buffer(ring, cb, budget);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_free_buffers(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ring->desc_num; i++)
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_free_buffer_detach(ring, i, 0);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* free desc along with its attached buffer */
|
|
|
|
static void hns3_free_desc(struct hns3_enet_ring *ring)
|
|
|
|
{
|
2018-07-02 15:50:25 +08:00
|
|
|
int size = ring->desc_num * sizeof(ring->desc[0]);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
hns3_free_buffers(ring);
|
|
|
|
|
2018-07-02 15:50:25 +08:00
|
|
|
if (ring->desc) {
|
|
|
|
dma_free_coherent(ring_to_dev(ring), size,
|
|
|
|
ring->desc, ring->desc_dma_addr);
|
|
|
|
ring->desc = NULL;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_alloc_desc(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
int size = ring->desc_num * sizeof(ring->desc[0]);
|
|
|
|
|
cross-tree: phase out dma_zalloc_coherent()
We already need to zero out memory for dma_alloc_coherent(), as such
using dma_zalloc_coherent() is superflous. Phase it out.
This change was generated with the following Coccinelle SmPL patch:
@ replace_dma_zalloc_coherent @
expression dev, size, data, handle, flags;
@@
-dma_zalloc_coherent(dev, size, handle, flags)
+dma_alloc_coherent(dev, size, handle, flags)
Suggested-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Luis Chamberlain <mcgrof@kernel.org>
[hch: re-ran the script on the latest tree]
Signed-off-by: Christoph Hellwig <hch@lst.de>
2019-01-04 16:23:09 +08:00
|
|
|
ring->desc = dma_alloc_coherent(ring_to_dev(ring), size,
|
|
|
|
&ring->desc_dma_addr, GFP_KERNEL);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (!ring->desc)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-06-18 09:02:09 +08:00
|
|
|
static int hns3_alloc_and_map_buffer(struct hns3_enet_ring *ring,
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_desc_cb *cb)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = hns3_alloc_buffer(ring, cb);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = hns3_map_buffer(ring, cb);
|
|
|
|
if (ret)
|
|
|
|
goto out_with_buf;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_with_buf:
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_free_buffer(ring, cb, 0);
|
2017-08-02 23:59:45 +08:00
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-06-18 09:02:09 +08:00
|
|
|
static int hns3_alloc_and_attach_buffer(struct hns3_enet_ring *ring, int i)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2020-06-18 09:02:09 +08:00
|
|
|
int ret = hns3_alloc_and_map_buffer(ring, &ring->desc_cb[i]);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ring->desc[i].addr = cpu_to_le64(ring->desc_cb[i].dma);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate memory for raw pkg, and map with dma */
|
|
|
|
static int hns3_alloc_ring_buffers(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
int i, j, ret;
|
|
|
|
|
|
|
|
for (i = 0; i < ring->desc_num; i++) {
|
2020-06-18 09:02:09 +08:00
|
|
|
ret = hns3_alloc_and_attach_buffer(ring, i);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (ret)
|
|
|
|
goto out_buffer_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_buffer_fail:
|
|
|
|
for (j = i - 1; j >= 0; j--)
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_free_buffer_detach(ring, j, 0);
|
2017-08-02 23:59:45 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
/* detach a in-used buffer and replace with a reserved one */
|
2017-08-02 23:59:45 +08:00
|
|
|
static void hns3_replace_buffer(struct hns3_enet_ring *ring, int i,
|
|
|
|
struct hns3_desc_cb *res_cb)
|
|
|
|
{
|
2017-10-23 19:51:01 +08:00
|
|
|
hns3_unmap_buffer(ring, &ring->desc_cb[i]);
|
2017-08-02 23:59:45 +08:00
|
|
|
ring->desc_cb[i] = *res_cb;
|
|
|
|
ring->desc[i].addr = cpu_to_le64(ring->desc_cb[i].dma);
|
2018-05-26 02:43:02 +08:00
|
|
|
ring->desc[i].rx.bd_base_info = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_reuse_buffer(struct hns3_enet_ring *ring, int i)
|
|
|
|
{
|
|
|
|
ring->desc_cb[i].reuse_flag = 0;
|
2019-06-07 10:03:13 +08:00
|
|
|
ring->desc[i].addr = cpu_to_le64(ring->desc_cb[i].dma +
|
|
|
|
ring->desc_cb[i].page_offset);
|
2018-05-26 02:43:02 +08:00
|
|
|
ring->desc[i].rx.bd_base_info = 0;
|
2020-06-18 09:02:11 +08:00
|
|
|
|
|
|
|
dma_sync_single_for_device(ring_to_dev(ring),
|
|
|
|
ring->desc_cb[i].dma + ring->desc_cb[i].page_offset,
|
|
|
|
hns3_buf_size(ring),
|
|
|
|
DMA_FROM_DEVICE);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:47 +08:00
|
|
|
static bool hns3_nic_reclaim_desc(struct hns3_enet_ring *ring,
|
2020-09-16 17:33:50 +08:00
|
|
|
int *bytes, int *pkts, int budget)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2020-09-16 17:33:47 +08:00
|
|
|
/* pair with ring->last_to_use update in hns3_tx_doorbell(),
|
|
|
|
* smp_store_release() is not used in hns3_tx_doorbell() because
|
|
|
|
* the doorbell operation already have the needed barrier operation.
|
|
|
|
*/
|
|
|
|
int ltu = smp_load_acquire(&ring->last_to_use);
|
2019-04-25 20:42:45 +08:00
|
|
|
int ntc = ring->next_to_clean;
|
|
|
|
struct hns3_desc_cb *desc_cb;
|
2020-09-16 17:33:47 +08:00
|
|
|
bool reclaimed = false;
|
|
|
|
struct hns3_desc *desc;
|
|
|
|
|
|
|
|
while (ltu != ntc) {
|
|
|
|
desc = &ring->desc[ntc];
|
|
|
|
|
|
|
|
if (le16_to_cpu(desc->tx.bdtp_fe_sc_vld_ra_ri) &
|
|
|
|
BIT(HNS3_TXD_VLD_B))
|
|
|
|
break;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:49 +08:00
|
|
|
desc_cb = &ring->desc_cb[ntc];
|
2021-03-29 11:57:51 +08:00
|
|
|
|
|
|
|
if (desc_cb->type == DESC_TYPE_SKB) {
|
|
|
|
(*pkts)++;
|
|
|
|
(*bytes) += desc_cb->send_bytes;
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:49 +08:00
|
|
|
/* desc_cb will be cleaned, after hnae3_free_buffer_detach */
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_free_buffer_detach(ring, ntc, budget);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-05-06 10:48:49 +08:00
|
|
|
if (++ntc == ring->desc_num)
|
|
|
|
ntc = 0;
|
|
|
|
|
|
|
|
/* Issue prefetch for next Tx descriptor */
|
|
|
|
prefetch(&ring->desc_cb[ntc]);
|
2020-09-16 17:33:47 +08:00
|
|
|
reclaimed = true;
|
2019-05-06 10:48:49 +08:00
|
|
|
}
|
2019-04-25 20:42:45 +08:00
|
|
|
|
2020-09-16 17:33:47 +08:00
|
|
|
if (unlikely(!reclaimed))
|
|
|
|
return false;
|
|
|
|
|
2019-04-25 20:42:45 +08:00
|
|
|
/* This smp_store_release() pairs with smp_load_acquire() in
|
|
|
|
* ring_space called by hns3_nic_net_xmit.
|
|
|
|
*/
|
|
|
|
smp_store_release(&ring->next_to_clean, ntc);
|
2020-09-16 17:33:47 +08:00
|
|
|
return true;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:50 +08:00
|
|
|
void hns3_clean_tx_ring(struct hns3_enet_ring *ring, int budget)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-10-19 16:03:52 +08:00
|
|
|
struct net_device *netdev = ring_to_netdev(ring);
|
2018-09-21 23:41:43 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
struct netdev_queue *dev_queue;
|
|
|
|
int bytes, pkts;
|
|
|
|
|
|
|
|
bytes = 0;
|
|
|
|
pkts = 0;
|
2020-09-16 17:33:47 +08:00
|
|
|
|
2020-09-16 17:33:50 +08:00
|
|
|
if (unlikely(!hns3_nic_reclaim_desc(ring, &bytes, &pkts, budget)))
|
2020-09-16 17:33:47 +08:00
|
|
|
return;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
ring->tqp_vector->tx_group.total_bytes += bytes;
|
|
|
|
ring->tqp_vector->tx_group.total_packets += pkts;
|
|
|
|
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.tx_bytes += bytes;
|
|
|
|
ring->stats.tx_pkts += pkts;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
|
|
|
dev_queue = netdev_get_tx_queue(netdev, ring->tqp->tqp_index);
|
|
|
|
netdev_tx_completed_queue(dev_queue, pkts, bytes);
|
|
|
|
|
2019-12-05 10:12:27 +08:00
|
|
|
if (unlikely(netif_carrier_ok(netdev) &&
|
2019-10-08 09:20:09 +08:00
|
|
|
ring_space(ring) > HNS3_MAX_TSO_BD_NUM)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
/* Make sure that anybody stopping the queue after this
|
|
|
|
* sees the new next_to_clean.
|
|
|
|
*/
|
|
|
|
smp_mb();
|
2018-09-21 23:41:43 +08:00
|
|
|
if (netif_tx_queue_stopped(dev_queue) &&
|
|
|
|
!test_bit(HNS3_NIC_STATE_DOWN, &priv->state)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
netif_tx_wake_queue(dev_queue);
|
|
|
|
ring->stats.restart_queue++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_desc_unused(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
int ntc = ring->next_to_clean;
|
|
|
|
int ntu = ring->next_to_use;
|
|
|
|
|
|
|
|
return ((ntc >= ntu) ? 0 : ring->desc_num) + ntc - ntu;
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
static void hns3_nic_alloc_rx_buffers(struct hns3_enet_ring *ring,
|
|
|
|
int cleand_count)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hns3_desc_cb *desc_cb;
|
|
|
|
struct hns3_desc_cb res_cbs;
|
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
for (i = 0; i < cleand_count; i++) {
|
|
|
|
desc_cb = &ring->desc_cb[ring->next_to_use];
|
|
|
|
if (desc_cb->reuse_flag) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.reuse_pg_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
|
|
|
hns3_reuse_buffer(ring, ring->next_to_use);
|
|
|
|
} else {
|
2020-06-18 09:02:09 +08:00
|
|
|
ret = hns3_alloc_and_map_buffer(ring, &res_cbs);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (ret) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.sw_err_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
2019-10-19 16:03:52 +08:00
|
|
|
hns3_rl_err(ring_to_netdev(ring),
|
2019-08-09 10:31:13 +08:00
|
|
|
"alloc rx buffer failed: %d\n",
|
|
|
|
ret);
|
2017-08-02 23:59:45 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
hns3_replace_buffer(ring, ring->next_to_use, &res_cbs);
|
2019-05-06 10:48:43 +08:00
|
|
|
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.non_reuse_pg++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ring_ptr_move_fw(ring, next_to_use);
|
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:49 +08:00
|
|
|
writel(i, ring->tqp->io_base + HNS3_RING_RX_RING_HEAD_REG);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:45 +08:00
|
|
|
static bool hns3_can_reuse_page(struct hns3_desc_cb *cb)
|
|
|
|
{
|
|
|
|
return (page_count(cb->priv) - cb->pagecnt_bias) == 1;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static void hns3_nic_reuse_page(struct sk_buff *skb, int i,
|
|
|
|
struct hns3_enet_ring *ring, int pull_len,
|
|
|
|
struct hns3_desc_cb *desc_cb)
|
|
|
|
{
|
2019-05-06 10:48:50 +08:00
|
|
|
struct hns3_desc *desc = &ring->desc[ring->next_to_clean];
|
|
|
|
int size = le16_to_cpu(desc->rx.size);
|
2019-07-29 10:53:30 +08:00
|
|
|
u32 truesize = hns3_buf_size(ring);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-09-16 17:33:45 +08:00
|
|
|
desc_cb->pagecnt_bias--;
|
2017-08-02 23:59:45 +08:00
|
|
|
skb_add_rx_frag(skb, i, desc_cb->priv, desc_cb->page_offset + pull_len,
|
2018-03-10 11:29:26 +08:00
|
|
|
size - pull_len, truesize);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2021-02-02 21:31:35 +08:00
|
|
|
/* Avoid re-using remote and pfmemalloc pages, or the stack is still
|
|
|
|
* using the page when page_offset rollback to zero, flag default
|
|
|
|
* unreuse
|
2019-05-06 10:48:50 +08:00
|
|
|
*/
|
2021-02-02 21:31:35 +08:00
|
|
|
if (!dev_page_is_reusable(desc_cb->priv) ||
|
2020-09-16 17:33:45 +08:00
|
|
|
(!desc_cb->page_offset && !hns3_can_reuse_page(desc_cb))) {
|
|
|
|
__page_frag_cache_drain(desc_cb->priv, desc_cb->pagecnt_bias);
|
2017-08-02 23:59:45 +08:00
|
|
|
return;
|
2020-09-16 17:33:45 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
/* Move offset up to the next cache line */
|
|
|
|
desc_cb->page_offset += truesize;
|
|
|
|
|
2019-07-29 10:53:30 +08:00
|
|
|
if (desc_cb->page_offset + truesize <= hns3_page_size(ring)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
desc_cb->reuse_flag = 1;
|
2020-09-16 17:33:45 +08:00
|
|
|
} else if (hns3_can_reuse_page(desc_cb)) {
|
2019-05-06 10:48:50 +08:00
|
|
|
desc_cb->reuse_flag = 1;
|
|
|
|
desc_cb->page_offset = 0;
|
2020-09-16 17:33:45 +08:00
|
|
|
} else if (desc_cb->pagecnt_bias) {
|
|
|
|
__page_frag_cache_drain(desc_cb->priv, desc_cb->pagecnt_bias);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!desc_cb->pagecnt_bias)) {
|
|
|
|
page_ref_add(desc_cb->priv, USHRT_MAX);
|
|
|
|
desc_cb->pagecnt_bias = USHRT_MAX;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
static int hns3_gro_complete(struct sk_buff *skb, u32 l234info)
|
2019-04-14 09:47:40 +08:00
|
|
|
{
|
|
|
|
__be16 type = skb->protocol;
|
|
|
|
struct tcphdr *th;
|
|
|
|
int depth = 0;
|
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
while (eth_type_vlan(type)) {
|
2019-04-14 09:47:40 +08:00
|
|
|
struct vlan_hdr *vh;
|
|
|
|
|
|
|
|
if ((depth + VLAN_HLEN) > skb_headlen(skb))
|
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
vh = (struct vlan_hdr *)(skb->data + depth);
|
|
|
|
type = vh->h_vlan_encapsulated_proto;
|
|
|
|
depth += VLAN_HLEN;
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
skb_set_network_header(skb, depth);
|
|
|
|
|
2019-04-14 09:47:40 +08:00
|
|
|
if (type == htons(ETH_P_IP)) {
|
2019-06-07 10:03:04 +08:00
|
|
|
const struct iphdr *iph = ip_hdr(skb);
|
|
|
|
|
2019-04-14 09:47:40 +08:00
|
|
|
depth += sizeof(struct iphdr);
|
2019-06-07 10:03:04 +08:00
|
|
|
skb_set_transport_header(skb, depth);
|
|
|
|
th = tcp_hdr(skb);
|
|
|
|
th->check = ~tcp_v4_check(skb->len - depth, iph->saddr,
|
|
|
|
iph->daddr, 0);
|
2019-04-14 09:47:40 +08:00
|
|
|
} else if (type == htons(ETH_P_IPV6)) {
|
2019-06-07 10:03:04 +08:00
|
|
|
const struct ipv6hdr *iph = ipv6_hdr(skb);
|
|
|
|
|
2019-04-14 09:47:40 +08:00
|
|
|
depth += sizeof(struct ipv6hdr);
|
2019-06-07 10:03:04 +08:00
|
|
|
skb_set_transport_header(skb, depth);
|
|
|
|
th = tcp_hdr(skb);
|
|
|
|
th->check = ~tcp_v6_check(skb->len - depth, &iph->saddr,
|
|
|
|
&iph->daddr, 0);
|
2019-04-14 09:47:40 +08:00
|
|
|
} else {
|
2019-08-09 10:31:13 +08:00
|
|
|
hns3_rl_err(skb->dev,
|
|
|
|
"Error: FW GRO supports only IPv4/IPv6, not 0x%04x, depth: %d\n",
|
|
|
|
be16_to_cpu(type), depth);
|
2019-04-14 09:47:40 +08:00
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
skb_shinfo(skb)->gso_segs = NAPI_GRO_CB(skb)->count;
|
|
|
|
if (th->cwr)
|
|
|
|
skb_shinfo(skb)->gso_type |= SKB_GSO_TCP_ECN;
|
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
if (l234info & BIT(HNS3_RXD_GRO_FIXID_B))
|
|
|
|
skb_shinfo(skb)->gso_type |= SKB_GSO_TCP_FIXEDID;
|
2019-04-14 09:47:40 +08:00
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
skb->csum_start = (unsigned char *)th - skb->head;
|
|
|
|
skb->csum_offset = offsetof(struct tcphdr, check);
|
|
|
|
skb->ip_summed = CHECKSUM_PARTIAL;
|
2020-01-04 10:49:24 +08:00
|
|
|
|
|
|
|
trace_hns3_gro(skb);
|
|
|
|
|
2019-04-14 09:47:40 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-11-28 11:51:44 +08:00
|
|
|
static void hns3_checksum_complete(struct hns3_enet_ring *ring,
|
|
|
|
struct sk_buff *skb, u32 l234info)
|
|
|
|
{
|
|
|
|
u32 lo, hi;
|
|
|
|
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.csum_complete++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
skb->ip_summed = CHECKSUM_COMPLETE;
|
|
|
|
lo = hnae3_get_field(l234info, HNS3_RXD_L2_CSUM_L_M,
|
|
|
|
HNS3_RXD_L2_CSUM_L_S);
|
|
|
|
hi = hnae3_get_field(l234info, HNS3_RXD_L2_CSUM_H_M,
|
|
|
|
HNS3_RXD_L2_CSUM_H_S);
|
|
|
|
skb->csum = csum_unfold((__force __sum16)(lo | hi << 8));
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static void hns3_rx_checksum(struct hns3_enet_ring *ring, struct sk_buff *skb,
|
2019-05-06 10:48:45 +08:00
|
|
|
u32 l234info, u32 bd_base_info, u32 ol_info)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-10-19 16:03:52 +08:00
|
|
|
struct net_device *netdev = ring_to_netdev(ring);
|
2017-08-02 23:59:45 +08:00
|
|
|
int l3_type, l4_type;
|
|
|
|
int ol4_type;
|
|
|
|
|
|
|
|
skb->ip_summed = CHECKSUM_NONE;
|
|
|
|
|
|
|
|
skb_checksum_none_assert(skb);
|
|
|
|
|
|
|
|
if (!(netdev->features & NETIF_F_RXCSUM))
|
|
|
|
return;
|
|
|
|
|
2020-11-28 11:51:44 +08:00
|
|
|
if (l234info & BIT(HNS3_RXD_L2_CSUM_B)) {
|
|
|
|
hns3_checksum_complete(ring, skb, l234info);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* check if hardware has done checksum */
|
2019-02-23 17:22:14 +08:00
|
|
|
if (!(bd_base_info & BIT(HNS3_RXD_L3L4P_B)))
|
2017-08-02 23:59:45 +08:00
|
|
|
return;
|
|
|
|
|
2019-03-06 16:12:34 +08:00
|
|
|
if (unlikely(l234info & (BIT(HNS3_RXD_L3E_B) | BIT(HNS3_RXD_L4E_B) |
|
|
|
|
BIT(HNS3_RXD_OL3E_B) |
|
2019-02-23 17:22:14 +08:00
|
|
|
BIT(HNS3_RXD_OL4E_B)))) {
|
2017-08-02 23:59:45 +08:00
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.l3l4_csum_err++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:48:45 +08:00
|
|
|
ol4_type = hnae3_get_field(ol_info, HNS3_RXD_OL4ID_M,
|
2018-07-02 15:50:26 +08:00
|
|
|
HNS3_RXD_OL4ID_S);
|
2017-08-02 23:59:45 +08:00
|
|
|
switch (ol4_type) {
|
|
|
|
case HNS3_OL4_TYPE_MAC_IN_UDP:
|
|
|
|
case HNS3_OL4_TYPE_NVGRE:
|
|
|
|
skb->csum_level = 1;
|
2020-08-24 06:36:59 +08:00
|
|
|
fallthrough;
|
2017-08-02 23:59:45 +08:00
|
|
|
case HNS3_OL4_TYPE_NO_TUN:
|
2019-02-23 17:22:10 +08:00
|
|
|
l3_type = hnae3_get_field(l234info, HNS3_RXD_L3ID_M,
|
|
|
|
HNS3_RXD_L3ID_S);
|
|
|
|
l4_type = hnae3_get_field(l234info, HNS3_RXD_L4ID_M,
|
|
|
|
HNS3_RXD_L4ID_S);
|
2017-08-02 23:59:45 +08:00
|
|
|
/* Can checksum ipv4 or ipv6 + UDP/TCP/SCTP packets */
|
2018-06-28 12:12:22 +08:00
|
|
|
if ((l3_type == HNS3_L3_TYPE_IPV4 ||
|
|
|
|
l3_type == HNS3_L3_TYPE_IPV6) &&
|
|
|
|
(l4_type == HNS3_L4_TYPE_UDP ||
|
|
|
|
l4_type == HNS3_L4_TYPE_TCP ||
|
|
|
|
l4_type == HNS3_L4_TYPE_SCTP))
|
2017-08-02 23:59:45 +08:00
|
|
|
skb->ip_summed = CHECKSUM_UNNECESSARY;
|
|
|
|
break;
|
2018-09-21 23:41:42 +08:00
|
|
|
default:
|
|
|
|
break;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-20 10:19:21 +08:00
|
|
|
static void hns3_rx_skb(struct hns3_enet_ring *ring, struct sk_buff *skb)
|
|
|
|
{
|
2018-11-15 17:29:24 +08:00
|
|
|
if (skb_has_frag_list(skb))
|
|
|
|
napi_gro_flush(&ring->tqp_vector->napi, false);
|
|
|
|
|
2017-10-20 10:19:21 +08:00
|
|
|
napi_gro_receive(&ring->tqp_vector->napi, skb);
|
|
|
|
}
|
|
|
|
|
2018-10-06 01:03:29 +08:00
|
|
|
static bool hns3_parse_vlan_tag(struct hns3_enet_ring *ring,
|
|
|
|
struct hns3_desc *desc, u32 l234info,
|
|
|
|
u16 *vlan_tag)
|
2018-05-26 02:42:58 +08:00
|
|
|
{
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
struct hnae3_handle *handle = ring->tqp->handle;
|
2018-05-26 02:42:58 +08:00
|
|
|
struct pci_dev *pdev = ring->tqp->handle->pdev;
|
2020-09-27 15:12:39 +08:00
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
2018-05-26 02:42:58 +08:00
|
|
|
|
2020-09-27 15:12:39 +08:00
|
|
|
if (unlikely(ae_dev->dev_version < HNAE3_DEVICE_VERSION_V2)) {
|
2018-10-06 01:03:29 +08:00
|
|
|
*vlan_tag = le16_to_cpu(desc->rx.ot_vlan_tag);
|
|
|
|
if (!(*vlan_tag & VLAN_VID_MASK))
|
|
|
|
*vlan_tag = le16_to_cpu(desc->rx.vlan_tag);
|
2018-05-26 02:42:58 +08:00
|
|
|
|
2018-10-06 01:03:29 +08:00
|
|
|
return (*vlan_tag != 0);
|
2018-05-26 02:42:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#define HNS3_STRP_OUTER_VLAN 0x1
|
|
|
|
#define HNS3_STRP_INNER_VLAN 0x2
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
#define HNS3_STRP_BOTH 0x3
|
2018-05-26 02:42:58 +08:00
|
|
|
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
/* Hardware always insert VLAN tag into RX descriptor when
|
|
|
|
* remove the tag from packet, driver needs to determine
|
|
|
|
* reporting which tag to stack.
|
|
|
|
*/
|
2018-07-02 15:50:26 +08:00
|
|
|
switch (hnae3_get_field(l234info, HNS3_RXD_STRP_TAGP_M,
|
|
|
|
HNS3_RXD_STRP_TAGP_S)) {
|
2018-05-26 02:42:58 +08:00
|
|
|
case HNS3_STRP_OUTER_VLAN:
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
if (handle->port_base_vlan_state !=
|
|
|
|
HNAE3_PORT_BASE_VLAN_DISABLE)
|
|
|
|
return false;
|
|
|
|
|
2018-10-06 01:03:29 +08:00
|
|
|
*vlan_tag = le16_to_cpu(desc->rx.ot_vlan_tag);
|
|
|
|
return true;
|
2018-05-26 02:42:58 +08:00
|
|
|
case HNS3_STRP_INNER_VLAN:
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
if (handle->port_base_vlan_state !=
|
|
|
|
HNAE3_PORT_BASE_VLAN_DISABLE)
|
|
|
|
return false;
|
|
|
|
|
2018-10-06 01:03:29 +08:00
|
|
|
*vlan_tag = le16_to_cpu(desc->rx.vlan_tag);
|
net: hns3: fix VLAN offload handle for VLAN inserted by port
Currently, in TX direction, driver implements the TX VLAN offload
by checking the VLAN header in skb, and filling it into TX descriptor.
Usually it works well, but if enable inserting VLAN header based on
port, it may conflict when out_tag field of TX descriptor is already
used, and cause RAS error.
In RX direction, hardware supports stripping max two VLAN headers.
For vlan_tci in skb can only store one VLAN tag, when RX VLAN offload
enabled, driver tells hardware to strip one VLAN header from RX
packet; when RX VLAN offload disabled, driver tells hardware not to
strip VLAN header from RX packet. Now if port based insert VLAN
enabled, all RX packets will have the port based VLAN header. This
header is useless for stack, driver needs to ask hardware to strip
it. Unfortunately, hardware can't drop this VLAN header, and always
fill it into RX descriptor, so driver has to identify and drop it.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-04-14 09:47:36 +08:00
|
|
|
return true;
|
|
|
|
case HNS3_STRP_BOTH:
|
|
|
|
if (handle->port_base_vlan_state ==
|
|
|
|
HNAE3_PORT_BASE_VLAN_DISABLE)
|
|
|
|
*vlan_tag = le16_to_cpu(desc->rx.ot_vlan_tag);
|
|
|
|
else
|
|
|
|
*vlan_tag = le16_to_cpu(desc->rx.vlan_tag);
|
|
|
|
|
2018-10-06 01:03:29 +08:00
|
|
|
return true;
|
2018-05-26 02:42:58 +08:00
|
|
|
default:
|
2018-10-06 01:03:29 +08:00
|
|
|
return false;
|
2018-05-26 02:42:58 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-16 17:33:48 +08:00
|
|
|
static void hns3_rx_ring_move_fw(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
ring->desc[ring->next_to_clean].rx.bd_base_info &=
|
|
|
|
cpu_to_le32(~BIT(HNS3_RXD_VLD_B));
|
|
|
|
ring->next_to_clean += 1;
|
|
|
|
|
|
|
|
if (unlikely(ring->next_to_clean == ring->desc_num))
|
|
|
|
ring->next_to_clean = 0;
|
|
|
|
}
|
|
|
|
|
2019-07-04 22:04:27 +08:00
|
|
|
static int hns3_alloc_skb(struct hns3_enet_ring *ring, unsigned int length,
|
2018-11-15 17:29:22 +08:00
|
|
|
unsigned char *va)
|
|
|
|
{
|
|
|
|
struct hns3_desc_cb *desc_cb = &ring->desc_cb[ring->next_to_clean];
|
2019-10-19 16:03:52 +08:00
|
|
|
struct net_device *netdev = ring_to_netdev(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
struct sk_buff *skb;
|
|
|
|
|
|
|
|
ring->skb = napi_alloc_skb(&ring->tqp_vector->napi, HNS3_RX_HEAD_SIZE);
|
|
|
|
skb = ring->skb;
|
|
|
|
if (unlikely(!skb)) {
|
2019-08-09 10:31:13 +08:00
|
|
|
hns3_rl_err(netdev, "alloc rx skb fail\n");
|
2018-11-15 17:29:22 +08:00
|
|
|
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.sw_err_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2020-01-04 10:49:24 +08:00
|
|
|
trace_hns3_rx_desc(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
prefetchw(skb->data);
|
|
|
|
|
|
|
|
ring->pending_buf = 1;
|
2018-11-15 17:29:24 +08:00
|
|
|
ring->frag_num = 0;
|
|
|
|
ring->tail_skb = NULL;
|
2018-11-15 17:29:22 +08:00
|
|
|
if (length <= HNS3_RX_HEAD_SIZE) {
|
|
|
|
memcpy(__skb_put(skb, length), va, ALIGN(length, sizeof(long)));
|
|
|
|
|
2021-02-02 21:31:35 +08:00
|
|
|
/* We can reuse buffer as-is, just make sure it is reusable */
|
|
|
|
if (dev_page_is_reusable(desc_cb->priv))
|
2018-11-15 17:29:22 +08:00
|
|
|
desc_cb->reuse_flag = 1;
|
|
|
|
else /* This page cannot be reused so discard it */
|
2020-09-16 17:33:45 +08:00
|
|
|
__page_frag_cache_drain(desc_cb->priv,
|
|
|
|
desc_cb->pagecnt_bias);
|
2018-11-15 17:29:22 +08:00
|
|
|
|
2020-09-16 17:33:48 +08:00
|
|
|
hns3_rx_ring_move_fw(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.seg_pkt_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
2019-04-22 23:55:48 +08:00
|
|
|
ring->pull_len = eth_get_headlen(netdev, va, HNS3_RX_HEAD_SIZE);
|
2018-11-15 17:29:22 +08:00
|
|
|
__skb_put(skb, ring->pull_len);
|
2018-11-15 17:29:24 +08:00
|
|
|
hns3_nic_reuse_page(skb, ring->frag_num++, ring, ring->pull_len,
|
2018-11-15 17:29:22 +08:00
|
|
|
desc_cb);
|
2020-09-16 17:33:48 +08:00
|
|
|
hns3_rx_ring_move_fw(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
|
2019-12-19 14:57:40 +08:00
|
|
|
return 0;
|
2018-11-15 17:29:22 +08:00
|
|
|
}
|
|
|
|
|
2019-12-19 14:57:40 +08:00
|
|
|
static int hns3_add_frag(struct hns3_enet_ring *ring)
|
2018-11-15 17:29:22 +08:00
|
|
|
{
|
2019-10-19 16:03:54 +08:00
|
|
|
struct sk_buff *skb = ring->skb;
|
|
|
|
struct sk_buff *head_skb = skb;
|
2018-11-15 17:29:24 +08:00
|
|
|
struct sk_buff *new_skb;
|
2018-11-15 17:29:22 +08:00
|
|
|
struct hns3_desc_cb *desc_cb;
|
2019-12-19 14:57:40 +08:00
|
|
|
struct hns3_desc *desc;
|
2018-11-15 17:29:22 +08:00
|
|
|
u32 bd_base_info;
|
|
|
|
|
2019-12-19 14:57:40 +08:00
|
|
|
do {
|
2018-11-15 17:29:22 +08:00
|
|
|
desc = &ring->desc[ring->next_to_clean];
|
|
|
|
desc_cb = &ring->desc_cb[ring->next_to_clean];
|
|
|
|
bd_base_info = le32_to_cpu(desc->rx.bd_base_info);
|
2019-03-06 11:26:37 +08:00
|
|
|
/* make sure HW write desc complete */
|
|
|
|
dma_rmb();
|
2019-02-23 17:22:14 +08:00
|
|
|
if (!(bd_base_info & BIT(HNS3_RXD_VLD_B)))
|
2018-11-15 17:29:22 +08:00
|
|
|
return -ENXIO;
|
|
|
|
|
2018-11-15 17:29:24 +08:00
|
|
|
if (unlikely(ring->frag_num >= MAX_SKB_FRAGS)) {
|
2019-10-19 16:03:55 +08:00
|
|
|
new_skb = napi_alloc_skb(&ring->tqp_vector->napi, 0);
|
2018-11-15 17:29:24 +08:00
|
|
|
if (unlikely(!new_skb)) {
|
2019-10-19 16:03:52 +08:00
|
|
|
hns3_rl_err(ring_to_netdev(ring),
|
2019-08-09 10:31:13 +08:00
|
|
|
"alloc rx fraglist skb fail\n");
|
2018-11-15 17:29:24 +08:00
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
ring->frag_num = 0;
|
|
|
|
|
|
|
|
if (ring->tail_skb) {
|
|
|
|
ring->tail_skb->next = new_skb;
|
|
|
|
ring->tail_skb = new_skb;
|
|
|
|
} else {
|
|
|
|
skb_shinfo(skb)->frag_list = new_skb;
|
|
|
|
ring->tail_skb = new_skb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ring->tail_skb) {
|
2019-07-29 10:53:30 +08:00
|
|
|
head_skb->truesize += hns3_buf_size(ring);
|
2018-11-15 17:29:24 +08:00
|
|
|
head_skb->data_len += le16_to_cpu(desc->rx.size);
|
|
|
|
head_skb->len += le16_to_cpu(desc->rx.size);
|
|
|
|
skb = ring->tail_skb;
|
|
|
|
}
|
|
|
|
|
2020-06-18 09:02:11 +08:00
|
|
|
dma_sync_single_for_cpu(ring_to_dev(ring),
|
|
|
|
desc_cb->dma + desc_cb->page_offset,
|
|
|
|
hns3_buf_size(ring),
|
|
|
|
DMA_FROM_DEVICE);
|
|
|
|
|
2018-11-15 17:29:24 +08:00
|
|
|
hns3_nic_reuse_page(skb, ring->frag_num++, ring, 0, desc_cb);
|
2020-01-04 10:49:24 +08:00
|
|
|
trace_hns3_rx_desc(ring);
|
2020-09-16 17:33:48 +08:00
|
|
|
hns3_rx_ring_move_fw(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
ring->pending_buf++;
|
2019-12-19 14:57:40 +08:00
|
|
|
} while (!(bd_base_info & BIT(HNS3_RXD_FE_B)));
|
2018-11-15 17:29:22 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-14 09:47:40 +08:00
|
|
|
static int hns3_set_gro_and_checksum(struct hns3_enet_ring *ring,
|
|
|
|
struct sk_buff *skb, u32 l234info,
|
2019-05-06 10:48:45 +08:00
|
|
|
u32 bd_base_info, u32 ol_info)
|
2018-11-15 17:29:25 +08:00
|
|
|
{
|
|
|
|
u32 l3_type;
|
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
skb_shinfo(skb)->gso_size = hnae3_get_field(bd_base_info,
|
|
|
|
HNS3_RXD_GRO_SIZE_M,
|
|
|
|
HNS3_RXD_GRO_SIZE_S);
|
2018-11-15 17:29:25 +08:00
|
|
|
/* if there is no HW GRO, do not set gro params */
|
2019-06-07 10:03:04 +08:00
|
|
|
if (!skb_shinfo(skb)->gso_size) {
|
2019-05-06 10:48:45 +08:00
|
|
|
hns3_rx_checksum(ring, skb, l234info, bd_base_info, ol_info);
|
2019-04-14 09:47:40 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2018-11-15 17:29:25 +08:00
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
NAPI_GRO_CB(skb)->count = hnae3_get_field(l234info,
|
|
|
|
HNS3_RXD_GRO_COUNT_M,
|
|
|
|
HNS3_RXD_GRO_COUNT_S);
|
2018-11-15 17:29:25 +08:00
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
l3_type = hnae3_get_field(l234info, HNS3_RXD_L3ID_M, HNS3_RXD_L3ID_S);
|
2018-11-15 17:29:25 +08:00
|
|
|
if (l3_type == HNS3_L3_TYPE_IPV4)
|
|
|
|
skb_shinfo(skb)->gso_type = SKB_GSO_TCPV4;
|
|
|
|
else if (l3_type == HNS3_L3_TYPE_IPV6)
|
|
|
|
skb_shinfo(skb)->gso_type = SKB_GSO_TCPV6;
|
|
|
|
else
|
2019-04-14 09:47:40 +08:00
|
|
|
return -EFAULT;
|
2018-11-15 17:29:25 +08:00
|
|
|
|
2019-06-07 10:03:04 +08:00
|
|
|
return hns3_gro_complete(skb, l234info);
|
2018-11-15 17:29:25 +08:00
|
|
|
}
|
|
|
|
|
2018-10-11 03:05:37 +08:00
|
|
|
static void hns3_set_rx_skb_rss_type(struct hns3_enet_ring *ring,
|
2019-04-25 20:42:47 +08:00
|
|
|
struct sk_buff *skb, u32 rss_hash)
|
2018-10-11 03:05:37 +08:00
|
|
|
{
|
|
|
|
struct hnae3_handle *handle = ring->tqp->handle;
|
|
|
|
enum pkt_hash_types rss_type;
|
|
|
|
|
2019-04-25 20:42:47 +08:00
|
|
|
if (rss_hash)
|
2018-10-11 03:05:37 +08:00
|
|
|
rss_type = handle->kinfo.rss_type;
|
|
|
|
else
|
|
|
|
rss_type = PKT_HASH_TYPE_NONE;
|
|
|
|
|
2019-04-25 20:42:47 +08:00
|
|
|
skb_set_hash(skb, rss_hash, rss_type);
|
2018-10-11 03:05:37 +08:00
|
|
|
}
|
|
|
|
|
2019-04-25 20:42:47 +08:00
|
|
|
static int hns3_handle_bdinfo(struct hns3_enet_ring *ring, struct sk_buff *skb)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-10-19 16:03:52 +08:00
|
|
|
struct net_device *netdev = ring_to_netdev(ring);
|
2019-01-23 07:39:28 +08:00
|
|
|
enum hns3_pkt_l2t_type l2_frame_type;
|
2019-05-06 10:48:45 +08:00
|
|
|
u32 bd_base_info, l234info, ol_info;
|
2019-04-25 20:42:47 +08:00
|
|
|
struct hns3_desc *desc;
|
2019-04-14 09:47:40 +08:00
|
|
|
unsigned int len;
|
2019-04-25 20:42:47 +08:00
|
|
|
int pre_ntc, ret;
|
|
|
|
|
|
|
|
/* bdinfo handled below is only valid on the last BD of the
|
|
|
|
* current packet, and ring->next_to_clean indicates the first
|
|
|
|
* descriptor of next packet, so need - 1 below.
|
|
|
|
*/
|
|
|
|
pre_ntc = ring->next_to_clean ? (ring->next_to_clean - 1) :
|
|
|
|
(ring->desc_num - 1);
|
|
|
|
desc = &ring->desc[pre_ntc];
|
|
|
|
bd_base_info = le32_to_cpu(desc->rx.bd_base_info);
|
|
|
|
l234info = le32_to_cpu(desc->rx.l234_info);
|
2019-05-06 10:48:45 +08:00
|
|
|
ol_info = le32_to_cpu(desc->rx.ol_info);
|
2019-04-14 09:47:40 +08:00
|
|
|
|
|
|
|
/* Based on hw strategy, the tag offloaded will be stored at
|
|
|
|
* ot_vlan_tag in two layer tag case, and stored at vlan_tag
|
|
|
|
* in one layer tag case.
|
|
|
|
*/
|
|
|
|
if (netdev->features & NETIF_F_HW_VLAN_CTAG_RX) {
|
|
|
|
u16 vlan_tag;
|
|
|
|
|
|
|
|
if (hns3_parse_vlan_tag(ring, desc, l234info, &vlan_tag))
|
|
|
|
__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q),
|
|
|
|
vlan_tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!desc->rx.pkt_len || (l234info & (BIT(HNS3_RXD_TRUNCAT_B) |
|
|
|
|
BIT(HNS3_RXD_L2E_B))))) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
if (l234info & BIT(HNS3_RXD_L2E_B))
|
|
|
|
ring->stats.l2_err++;
|
|
|
|
else
|
|
|
|
ring->stats.err_pkt_len++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = skb->len;
|
|
|
|
|
|
|
|
/* Do update ip stack process */
|
|
|
|
skb->protocol = eth_type_trans(skb, netdev);
|
|
|
|
|
|
|
|
/* This is needed in order to enable forwarding support */
|
2019-05-06 10:48:45 +08:00
|
|
|
ret = hns3_set_gro_and_checksum(ring, skb, l234info,
|
|
|
|
bd_base_info, ol_info);
|
2019-04-14 09:47:40 +08:00
|
|
|
if (unlikely(ret)) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.rx_err_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
l2_frame_type = hnae3_get_field(l234info, HNS3_RXD_DMAC_M,
|
|
|
|
HNS3_RXD_DMAC_S);
|
|
|
|
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.rx_pkts++;
|
|
|
|
ring->stats.rx_bytes += len;
|
|
|
|
|
|
|
|
if (l2_frame_type == HNS3_L2_TYPE_MULTICAST)
|
|
|
|
ring->stats.rx_multicast++;
|
|
|
|
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
|
|
|
|
ring->tqp_vector->rx_group.total_bytes += len;
|
2019-04-25 20:42:47 +08:00
|
|
|
|
|
|
|
hns3_set_rx_skb_rss_type(ring, skb, le32_to_cpu(desc->rx.rss_hash));
|
2019-04-14 09:47:40 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-19 16:03:54 +08:00
|
|
|
static int hns3_handle_rx_bd(struct hns3_enet_ring *ring)
|
2019-04-14 09:47:40 +08:00
|
|
|
{
|
2018-11-15 17:29:22 +08:00
|
|
|
struct sk_buff *skb = ring->skb;
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_desc_cb *desc_cb;
|
|
|
|
struct hns3_desc *desc;
|
2019-07-04 22:04:27 +08:00
|
|
|
unsigned int length;
|
2017-08-02 23:59:45 +08:00
|
|
|
u32 bd_base_info;
|
2018-11-15 17:29:22 +08:00
|
|
|
int ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
desc = &ring->desc[ring->next_to_clean];
|
|
|
|
desc_cb = &ring->desc_cb[ring->next_to_clean];
|
|
|
|
|
|
|
|
prefetch(desc);
|
|
|
|
|
2020-09-16 17:33:48 +08:00
|
|
|
if (!skb) {
|
|
|
|
bd_base_info = le32_to_cpu(desc->rx.bd_base_info);
|
|
|
|
/* Check valid BD */
|
|
|
|
if (unlikely(!(bd_base_info & BIT(HNS3_RXD_VLD_B))))
|
|
|
|
return -ENXIO;
|
|
|
|
|
|
|
|
dma_rmb();
|
|
|
|
length = le16_to_cpu(desc->rx.size);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-06-18 09:02:08 +08:00
|
|
|
ring->va = desc_cb->buf + desc_cb->page_offset;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-06-18 09:02:11 +08:00
|
|
|
dma_sync_single_for_cpu(ring_to_dev(ring),
|
|
|
|
desc_cb->dma + desc_cb->page_offset,
|
|
|
|
hns3_buf_size(ring),
|
|
|
|
DMA_FROM_DEVICE);
|
|
|
|
|
2020-09-16 17:33:48 +08:00
|
|
|
/* Prefetch first cache line of first page.
|
|
|
|
* Idea is to cache few bytes of the header of the packet.
|
|
|
|
* Our L1 Cache line size is 64B so need to prefetch twice to make
|
|
|
|
* it 128B. But in actual we can have greater size of caches with
|
|
|
|
* 128B Level 1 cache lines. In such a case, single fetch would
|
|
|
|
* suffice to cache in the relevant part of the header.
|
|
|
|
*/
|
|
|
|
net_prefetch(ring->va);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-11-15 17:29:22 +08:00
|
|
|
ret = hns3_alloc_skb(ring, length, ring->va);
|
2019-10-19 16:03:54 +08:00
|
|
|
skb = ring->skb;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-11-15 17:29:22 +08:00
|
|
|
if (ret < 0) /* alloc buffer fail */
|
|
|
|
return ret;
|
2019-12-19 14:57:40 +08:00
|
|
|
if (!(bd_base_info & BIT(HNS3_RXD_FE_B))) { /* need add frag */
|
|
|
|
ret = hns3_add_frag(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
} else {
|
2019-12-19 14:57:40 +08:00
|
|
|
ret = hns3_add_frag(ring);
|
2018-11-15 17:29:22 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
2019-12-19 14:57:40 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-12-19 14:57:40 +08:00
|
|
|
/* As the head data may be changed when GRO enable, copy
|
|
|
|
* the head data in after other data rx completed
|
|
|
|
*/
|
|
|
|
if (skb->len > HNS3_RX_HEAD_SIZE)
|
2018-11-15 17:29:22 +08:00
|
|
|
memcpy(skb->data, ring->va,
|
|
|
|
ALIGN(ring->pull_len, sizeof(long)));
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-04-25 20:42:47 +08:00
|
|
|
ret = hns3_handle_bdinfo(ring, skb);
|
2019-04-14 09:47:40 +08:00
|
|
|
if (unlikely(ret)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
dev_kfree_skb_any(skb);
|
2019-04-14 09:47:40 +08:00
|
|
|
return ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-05-24 19:19:47 +08:00
|
|
|
skb_record_rx_queue(skb, ring->tqp->tqp_index);
|
2017-08-02 23:59:45 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
int hns3_clean_rx_ring(struct hns3_enet_ring *ring, int budget,
|
|
|
|
void (*rx_fn)(struct hns3_enet_ring *, struct sk_buff *))
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
#define RCB_NOF_ALLOC_RX_BUFF_ONCE 16
|
2019-04-25 20:42:46 +08:00
|
|
|
int unused_count = hns3_desc_unused(ring);
|
2019-08-01 11:55:39 +08:00
|
|
|
int recv_pkts = 0;
|
2020-09-16 17:33:48 +08:00
|
|
|
int err;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-04-25 20:42:46 +08:00
|
|
|
unused_count -= ring->pending_buf;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-09-16 17:33:48 +08:00
|
|
|
while (recv_pkts < budget) {
|
2017-08-02 23:59:45 +08:00
|
|
|
/* Reuse or realloc buffers */
|
2019-08-01 11:55:39 +08:00
|
|
|
if (unused_count >= RCB_NOF_ALLOC_RX_BUFF_ONCE) {
|
|
|
|
hns3_nic_alloc_rx_buffers(ring, unused_count);
|
2018-11-15 17:29:22 +08:00
|
|
|
unused_count = hns3_desc_unused(ring) -
|
|
|
|
ring->pending_buf;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Poll one pkt */
|
2019-10-19 16:03:54 +08:00
|
|
|
err = hns3_handle_rx_bd(ring);
|
|
|
|
/* Do not get FE for the packet or failed to alloc skb */
|
|
|
|
if (unlikely(!ring->skb || err == -ENXIO)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
goto out;
|
2019-10-19 16:03:54 +08:00
|
|
|
} else if (likely(!err)) {
|
|
|
|
rx_fn(ring, ring->skb);
|
|
|
|
recv_pkts++;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-08-01 11:55:39 +08:00
|
|
|
unused_count += ring->pending_buf;
|
2018-11-15 17:29:22 +08:00
|
|
|
ring->skb = NULL;
|
|
|
|
ring->pending_buf = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
/* Make all data has been write before submit */
|
2019-08-01 11:55:39 +08:00
|
|
|
if (unused_count > 0)
|
|
|
|
hns3_nic_alloc_rx_buffers(ring, unused_count);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return recv_pkts;
|
|
|
|
}
|
|
|
|
|
2019-06-07 10:03:06 +08:00
|
|
|
static bool hns3_get_new_flow_lvl(struct hns3_enet_ring_group *ring_group)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-06-07 10:03:06 +08:00
|
|
|
#define HNS3_RX_LOW_BYTE_RATE 10000
|
|
|
|
#define HNS3_RX_MID_BYTE_RATE 20000
|
|
|
|
#define HNS3_RX_ULTRA_PACKET_RATE 40
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
enum hns3_flow_level_range new_flow_level;
|
2019-06-07 10:03:06 +08:00
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
|
|
|
int packets_per_msecs, bytes_per_msecs;
|
2018-03-21 15:49:26 +08:00
|
|
|
u32 time_passed_ms;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-06-07 10:03:06 +08:00
|
|
|
tqp_vector = ring_group->ring->tqp_vector;
|
2018-03-21 15:49:26 +08:00
|
|
|
time_passed_ms =
|
|
|
|
jiffies_to_msecs(jiffies - tqp_vector->last_jiffies);
|
|
|
|
if (!time_passed_ms)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
do_div(ring_group->total_packets, time_passed_ms);
|
|
|
|
packets_per_msecs = ring_group->total_packets;
|
|
|
|
|
|
|
|
do_div(ring_group->total_bytes, time_passed_ms);
|
|
|
|
bytes_per_msecs = ring_group->total_bytes;
|
|
|
|
|
2019-06-07 10:03:06 +08:00
|
|
|
new_flow_level = ring_group->coal.flow_level;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-06-07 10:03:06 +08:00
|
|
|
/* Simple throttlerate management
|
|
|
|
* 0-10MB/s lower (50000 ints/s)
|
|
|
|
* 10-20MB/s middle (20000 ints/s)
|
|
|
|
* 20-1249MB/s high (18000 ints/s)
|
|
|
|
* > 40000pps ultra (8000 ints/s)
|
|
|
|
*/
|
2017-08-02 23:59:45 +08:00
|
|
|
switch (new_flow_level) {
|
|
|
|
case HNS3_FLOW_LOW:
|
2018-03-21 15:49:26 +08:00
|
|
|
if (bytes_per_msecs > HNS3_RX_LOW_BYTE_RATE)
|
2017-08-02 23:59:45 +08:00
|
|
|
new_flow_level = HNS3_FLOW_MID;
|
|
|
|
break;
|
|
|
|
case HNS3_FLOW_MID:
|
2018-03-21 15:49:26 +08:00
|
|
|
if (bytes_per_msecs > HNS3_RX_MID_BYTE_RATE)
|
2017-08-02 23:59:45 +08:00
|
|
|
new_flow_level = HNS3_FLOW_HIGH;
|
2018-03-21 15:49:26 +08:00
|
|
|
else if (bytes_per_msecs <= HNS3_RX_LOW_BYTE_RATE)
|
2017-08-02 23:59:45 +08:00
|
|
|
new_flow_level = HNS3_FLOW_LOW;
|
|
|
|
break;
|
|
|
|
case HNS3_FLOW_HIGH:
|
|
|
|
case HNS3_FLOW_ULTRA:
|
|
|
|
default:
|
2018-03-21 15:49:26 +08:00
|
|
|
if (bytes_per_msecs <= HNS3_RX_MID_BYTE_RATE)
|
2017-08-02 23:59:45 +08:00
|
|
|
new_flow_level = HNS3_FLOW_MID;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-03-21 15:49:26 +08:00
|
|
|
if (packets_per_msecs > HNS3_RX_ULTRA_PACKET_RATE &&
|
|
|
|
&tqp_vector->rx_group == ring_group)
|
2017-08-02 23:59:45 +08:00
|
|
|
new_flow_level = HNS3_FLOW_ULTRA;
|
|
|
|
|
2019-06-07 10:03:06 +08:00
|
|
|
ring_group->total_bytes = 0;
|
|
|
|
ring_group->total_packets = 0;
|
|
|
|
ring_group->coal.flow_level = new_flow_level;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool hns3_get_new_int_gl(struct hns3_enet_ring_group *ring_group)
|
|
|
|
{
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
|
|
|
u16 new_int_gl;
|
|
|
|
|
|
|
|
if (!ring_group->ring)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
tqp_vector = ring_group->ring->tqp_vector;
|
|
|
|
if (!tqp_vector->last_jiffies)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (ring_group->total_packets == 0) {
|
|
|
|
ring_group->coal.int_gl = HNS3_INT_GL_50K;
|
|
|
|
ring_group->coal.flow_level = HNS3_FLOW_LOW;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hns3_get_new_flow_lvl(ring_group))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
new_int_gl = ring_group->coal.int_gl;
|
|
|
|
switch (ring_group->coal.flow_level) {
|
2017-08-02 23:59:45 +08:00
|
|
|
case HNS3_FLOW_LOW:
|
|
|
|
new_int_gl = HNS3_INT_GL_50K;
|
|
|
|
break;
|
|
|
|
case HNS3_FLOW_MID:
|
|
|
|
new_int_gl = HNS3_INT_GL_20K;
|
|
|
|
break;
|
|
|
|
case HNS3_FLOW_HIGH:
|
|
|
|
new_int_gl = HNS3_INT_GL_18K;
|
|
|
|
break;
|
|
|
|
case HNS3_FLOW_ULTRA:
|
|
|
|
new_int_gl = HNS3_INT_GL_8K;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-03-09 10:37:03 +08:00
|
|
|
if (new_int_gl != ring_group->coal.int_gl) {
|
|
|
|
ring_group->coal.int_gl = new_int_gl;
|
2017-08-02 23:59:45 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_update_new_int_gl(struct hns3_enet_tqp_vector *tqp_vector)
|
|
|
|
{
|
2018-01-12 16:23:12 +08:00
|
|
|
struct hns3_enet_ring_group *rx_group = &tqp_vector->rx_group;
|
|
|
|
struct hns3_enet_ring_group *tx_group = &tqp_vector->tx_group;
|
|
|
|
bool rx_update, tx_update;
|
|
|
|
|
2018-12-18 19:37:54 +08:00
|
|
|
/* update param every 1000ms */
|
|
|
|
if (time_before(jiffies,
|
|
|
|
tqp_vector->last_jiffies + msecs_to_jiffies(1000)))
|
2018-03-21 15:49:25 +08:00
|
|
|
return;
|
|
|
|
|
2020-11-16 16:20:54 +08:00
|
|
|
if (rx_group->coal.adapt_enable) {
|
2018-01-12 16:23:12 +08:00
|
|
|
rx_update = hns3_get_new_int_gl(rx_group);
|
|
|
|
if (rx_update)
|
|
|
|
hns3_set_vector_coalesce_rx_gl(tqp_vector,
|
2018-03-09 10:37:03 +08:00
|
|
|
rx_group->coal.int_gl);
|
2018-01-12 16:23:12 +08:00
|
|
|
}
|
|
|
|
|
2020-11-16 16:20:54 +08:00
|
|
|
if (tx_group->coal.adapt_enable) {
|
2019-02-02 22:39:26 +08:00
|
|
|
tx_update = hns3_get_new_int_gl(tx_group);
|
2018-01-12 16:23:12 +08:00
|
|
|
if (tx_update)
|
|
|
|
hns3_set_vector_coalesce_tx_gl(tqp_vector,
|
2018-03-09 10:37:03 +08:00
|
|
|
tx_group->coal.int_gl);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
2018-03-21 15:49:25 +08:00
|
|
|
|
2018-03-21 15:49:26 +08:00
|
|
|
tqp_vector->last_jiffies = jiffies;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_common_poll(struct napi_struct *napi, int budget)
|
|
|
|
{
|
2018-11-09 22:07:52 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(napi->dev);
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_enet_ring *ring;
|
|
|
|
int rx_pkt_total = 0;
|
|
|
|
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector =
|
|
|
|
container_of(napi, struct hns3_enet_tqp_vector, napi);
|
|
|
|
bool clean_complete = true;
|
2019-04-04 16:17:53 +08:00
|
|
|
int rx_budget = budget;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-11-09 22:07:52 +08:00
|
|
|
if (unlikely(test_bit(HNS3_NIC_STATE_DOWN, &priv->state))) {
|
|
|
|
napi_complete(napi);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
/* Since the actual Tx work is minimal, we can give the Tx a larger
|
|
|
|
* budget and be more aggressive about cleaning up the Tx descriptors.
|
|
|
|
*/
|
2018-09-20 01:29:49 +08:00
|
|
|
hns3_for_each_ring(ring, tqp_vector->tx_group)
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_clean_tx_ring(ring, budget);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
/* make sure rx ring budget not smaller than 1 */
|
2019-04-04 16:17:53 +08:00
|
|
|
if (tqp_vector->num_tqps > 1)
|
|
|
|
rx_budget = max(budget / tqp_vector->num_tqps, 1);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
hns3_for_each_ring(ring, tqp_vector->rx_group) {
|
2017-10-20 10:19:21 +08:00
|
|
|
int rx_cleaned = hns3_clean_rx_ring(ring, rx_budget,
|
|
|
|
hns3_rx_skb);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (rx_cleaned >= rx_budget)
|
|
|
|
clean_complete = false;
|
|
|
|
|
|
|
|
rx_pkt_total += rx_cleaned;
|
|
|
|
}
|
|
|
|
|
|
|
|
tqp_vector->rx_group.total_packets += rx_pkt_total;
|
|
|
|
|
|
|
|
if (!clean_complete)
|
|
|
|
return budget;
|
|
|
|
|
2018-12-18 19:37:50 +08:00
|
|
|
if (napi_complete(napi) &&
|
|
|
|
likely(!test_bit(HNS3_NIC_STATE_DOWN, &priv->state))) {
|
2018-11-09 22:07:52 +08:00
|
|
|
hns3_update_new_int_gl(tqp_vector);
|
|
|
|
hns3_mask_vector_irq(tqp_vector, 1);
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return rx_pkt_total;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_get_vector_ring_chain(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
struct hnae3_ring_chain_node *head)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = tqp_vector->handle->pdev;
|
|
|
|
struct hnae3_ring_chain_node *cur_chain = head;
|
|
|
|
struct hnae3_ring_chain_node *chain;
|
|
|
|
struct hns3_enet_ring *tx_ring;
|
|
|
|
struct hns3_enet_ring *rx_ring;
|
|
|
|
|
|
|
|
tx_ring = tqp_vector->tx_group.ring;
|
|
|
|
if (tx_ring) {
|
|
|
|
cur_chain->tqp_index = tx_ring->tqp->tqp_index;
|
2018-07-02 15:50:26 +08:00
|
|
|
hnae3_set_bit(cur_chain->flag, HNAE3_RING_TYPE_B,
|
|
|
|
HNAE3_RING_TYPE_TX);
|
|
|
|
hnae3_set_field(cur_chain->int_gl_idx, HNAE3_RING_GL_IDX_M,
|
|
|
|
HNAE3_RING_GL_IDX_S, HNAE3_RING_GL_TX);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
cur_chain->next = NULL;
|
|
|
|
|
|
|
|
while (tx_ring->next) {
|
|
|
|
tx_ring = tx_ring->next;
|
|
|
|
|
|
|
|
chain = devm_kzalloc(&pdev->dev, sizeof(*chain),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!chain)
|
2018-10-30 21:50:44 +08:00
|
|
|
goto err_free_chain;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
cur_chain->next = chain;
|
|
|
|
chain->tqp_index = tx_ring->tqp->tqp_index;
|
2018-07-02 15:50:26 +08:00
|
|
|
hnae3_set_bit(chain->flag, HNAE3_RING_TYPE_B,
|
|
|
|
HNAE3_RING_TYPE_TX);
|
|
|
|
hnae3_set_field(chain->int_gl_idx,
|
|
|
|
HNAE3_RING_GL_IDX_M,
|
|
|
|
HNAE3_RING_GL_IDX_S,
|
|
|
|
HNAE3_RING_GL_TX);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
cur_chain = chain;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rx_ring = tqp_vector->rx_group.ring;
|
|
|
|
if (!tx_ring && rx_ring) {
|
|
|
|
cur_chain->next = NULL;
|
|
|
|
cur_chain->tqp_index = rx_ring->tqp->tqp_index;
|
2018-07-02 15:50:26 +08:00
|
|
|
hnae3_set_bit(cur_chain->flag, HNAE3_RING_TYPE_B,
|
|
|
|
HNAE3_RING_TYPE_RX);
|
|
|
|
hnae3_set_field(cur_chain->int_gl_idx, HNAE3_RING_GL_IDX_M,
|
|
|
|
HNAE3_RING_GL_IDX_S, HNAE3_RING_GL_RX);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
rx_ring = rx_ring->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (rx_ring) {
|
|
|
|
chain = devm_kzalloc(&pdev->dev, sizeof(*chain), GFP_KERNEL);
|
|
|
|
if (!chain)
|
2018-10-30 21:50:44 +08:00
|
|
|
goto err_free_chain;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
cur_chain->next = chain;
|
|
|
|
chain->tqp_index = rx_ring->tqp->tqp_index;
|
2018-07-02 15:50:26 +08:00
|
|
|
hnae3_set_bit(chain->flag, HNAE3_RING_TYPE_B,
|
|
|
|
HNAE3_RING_TYPE_RX);
|
|
|
|
hnae3_set_field(chain->int_gl_idx, HNAE3_RING_GL_IDX_M,
|
|
|
|
HNAE3_RING_GL_IDX_S, HNAE3_RING_GL_RX);
|
2018-01-12 16:23:15 +08:00
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
cur_chain = chain;
|
|
|
|
|
|
|
|
rx_ring = rx_ring->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2018-10-30 21:50:44 +08:00
|
|
|
|
|
|
|
err_free_chain:
|
|
|
|
cur_chain = head->next;
|
|
|
|
while (cur_chain) {
|
|
|
|
chain = cur_chain->next;
|
2018-12-18 19:37:48 +08:00
|
|
|
devm_kfree(&pdev->dev, cur_chain);
|
2018-10-30 21:50:44 +08:00
|
|
|
cur_chain = chain;
|
|
|
|
}
|
2018-12-18 19:37:48 +08:00
|
|
|
head->next = NULL;
|
2018-10-30 21:50:44 +08:00
|
|
|
|
|
|
|
return -ENOMEM;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_free_vector_ring_chain(struct hns3_enet_tqp_vector *tqp_vector,
|
|
|
|
struct hnae3_ring_chain_node *head)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = tqp_vector->handle->pdev;
|
|
|
|
struct hnae3_ring_chain_node *chain_tmp, *chain;
|
|
|
|
|
|
|
|
chain = head->next;
|
|
|
|
|
|
|
|
while (chain) {
|
|
|
|
chain_tmp = chain->next;
|
|
|
|
devm_kfree(&pdev->dev, chain);
|
|
|
|
chain = chain_tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_add_ring_to_group(struct hns3_enet_ring_group *group,
|
|
|
|
struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
ring->next = group->ring;
|
|
|
|
group->ring = ring;
|
|
|
|
|
|
|
|
group->count++;
|
|
|
|
}
|
|
|
|
|
2018-09-20 01:29:47 +08:00
|
|
|
static void hns3_nic_set_cpumask(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = priv->ae_handle->pdev;
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
|
|
|
int num_vectors = priv->vector_num;
|
|
|
|
int numa_node;
|
|
|
|
int vector_i;
|
|
|
|
|
|
|
|
numa_node = dev_to_node(&pdev->dev);
|
|
|
|
|
|
|
|
for (vector_i = 0; vector_i < num_vectors; vector_i++) {
|
|
|
|
tqp_vector = &priv->tqp_vector[vector_i];
|
|
|
|
cpumask_set_cpu(cpumask_local_spread(vector_i, numa_node),
|
|
|
|
&tqp_vector->affinity_mask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_nic_init_vector_data(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
2020-09-25 08:26:14 +08:00
|
|
|
int ret;
|
2018-10-30 21:50:43 +08:00
|
|
|
int i;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-09-20 01:29:47 +08:00
|
|
|
hns3_nic_set_cpumask(priv);
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
tqp_vector = &priv->tqp_vector[i];
|
2020-11-16 16:20:51 +08:00
|
|
|
hns3_vector_coalesce_init_hw(tqp_vector, priv);
|
2018-03-09 10:37:02 +08:00
|
|
|
tqp_vector->num_tqps = 0;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++) {
|
|
|
|
u16 vector_i = i % priv->vector_num;
|
|
|
|
u16 tqp_num = h->kinfo.num_tqps;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
tqp_vector = &priv->tqp_vector[vector_i];
|
|
|
|
|
|
|
|
hns3_add_ring_to_group(&tqp_vector->tx_group,
|
2019-10-19 16:03:49 +08:00
|
|
|
&priv->ring[i]);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
hns3_add_ring_to_group(&tqp_vector->rx_group,
|
2019-10-19 16:03:49 +08:00
|
|
|
&priv->ring[i + tqp_num]);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
priv->ring[i].tqp_vector = tqp_vector;
|
|
|
|
priv->ring[i + tqp_num].tqp_vector = tqp_vector;
|
2018-03-09 10:37:02 +08:00
|
|
|
tqp_vector->num_tqps++;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
2021-04-06 01:28:25 +08:00
|
|
|
struct hnae3_ring_chain_node vector_ring_chain;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
tqp_vector = &priv->tqp_vector[i];
|
|
|
|
|
|
|
|
tqp_vector->rx_group.total_bytes = 0;
|
|
|
|
tqp_vector->rx_group.total_packets = 0;
|
|
|
|
tqp_vector->tx_group.total_bytes = 0;
|
|
|
|
tqp_vector->tx_group.total_packets = 0;
|
|
|
|
tqp_vector->handle = h;
|
|
|
|
|
|
|
|
ret = hns3_get_vector_ring_chain(tqp_vector,
|
|
|
|
&vector_ring_chain);
|
|
|
|
if (ret)
|
2018-12-18 19:37:48 +08:00
|
|
|
goto map_ring_fail;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
ret = h->ae_algo->ops->map_ring_to_vector(h,
|
|
|
|
tqp_vector->vector_irq, &vector_ring_chain);
|
|
|
|
|
|
|
|
hns3_free_vector_ring_chain(tqp_vector, &vector_ring_chain);
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
if (ret)
|
2018-10-30 21:50:43 +08:00
|
|
|
goto map_ring_fail;
|
2018-03-09 10:37:02 +08:00
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
netif_napi_add(priv->netdev, &tqp_vector->napi,
|
|
|
|
hns3_nic_common_poll, NAPI_POLL_WEIGHT);
|
|
|
|
}
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
return 0;
|
2018-10-30 21:50:43 +08:00
|
|
|
|
|
|
|
map_ring_fail:
|
|
|
|
while (i--)
|
|
|
|
netif_napi_del(&priv->tqp_vector[i].napi);
|
|
|
|
|
|
|
|
return ret;
|
2018-03-09 10:37:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_nic_alloc_vector_data(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
|
|
|
struct hnae3_vector_info *vector;
|
|
|
|
struct pci_dev *pdev = h->pdev;
|
|
|
|
u16 tqp_num = h->kinfo.num_tqps;
|
|
|
|
u16 vector_num;
|
|
|
|
int ret = 0;
|
|
|
|
u16 i;
|
|
|
|
|
|
|
|
/* RSS size, cpu online and vector_num should be the same */
|
|
|
|
/* Should consider 2p/4p later */
|
|
|
|
vector_num = min_t(u16, num_online_cpus(), tqp_num);
|
2018-12-20 11:52:01 +08:00
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
vector = devm_kcalloc(&pdev->dev, vector_num, sizeof(*vector),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!vector)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2019-06-07 10:03:13 +08:00
|
|
|
/* save the actual available vector number */
|
2018-03-09 10:37:02 +08:00
|
|
|
vector_num = h->ae_algo->ops->get_vector(h, vector_num, vector);
|
|
|
|
|
|
|
|
priv->vector_num = vector_num;
|
|
|
|
priv->tqp_vector = (struct hns3_enet_tqp_vector *)
|
|
|
|
devm_kcalloc(&pdev->dev, vector_num, sizeof(*priv->tqp_vector),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!priv->tqp_vector) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
tqp_vector = &priv->tqp_vector[i];
|
|
|
|
tqp_vector->idx = i;
|
|
|
|
tqp_vector->mask_addr = vector[i].io_addr;
|
|
|
|
tqp_vector->vector_irq = vector[i].vector;
|
2020-11-16 16:20:51 +08:00
|
|
|
hns3_vector_coalesce_init(tqp_vector, priv);
|
2018-03-09 10:37:02 +08:00
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
out:
|
|
|
|
devm_kfree(&pdev->dev, vector);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
static void hns3_clear_ring_group(struct hns3_enet_ring_group *group)
|
|
|
|
{
|
|
|
|
group->ring = NULL;
|
|
|
|
group->count = 0;
|
|
|
|
}
|
|
|
|
|
2019-01-31 04:55:47 +08:00
|
|
|
static void hns3_nic_uninit_vector_data(struct hns3_nic_priv *priv)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hnae3_ring_chain_node vector_ring_chain;
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
2019-01-31 04:55:47 +08:00
|
|
|
int i;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
tqp_vector = &priv->tqp_vector[i];
|
|
|
|
|
2019-01-18 16:13:03 +08:00
|
|
|
if (!tqp_vector->rx_group.ring && !tqp_vector->tx_group.ring)
|
|
|
|
continue;
|
|
|
|
|
2019-12-19 14:57:46 +08:00
|
|
|
/* Since the mapping can be overwritten, when fail to get the
|
|
|
|
* chain between vector and ring, we should go on to deal with
|
|
|
|
* the remaining options.
|
|
|
|
*/
|
|
|
|
if (hns3_get_vector_ring_chain(tqp_vector, &vector_ring_chain))
|
|
|
|
dev_warn(priv->dev, "failed to get ring chain\n");
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-01-31 04:55:47 +08:00
|
|
|
h->ae_algo->ops->unmap_ring_from_vector(h,
|
2017-08-02 23:59:45 +08:00
|
|
|
tqp_vector->vector_irq, &vector_ring_chain);
|
|
|
|
|
|
|
|
hns3_free_vector_ring_chain(tqp_vector, &vector_ring_chain);
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
hns3_clear_ring_group(&tqp_vector->rx_group);
|
|
|
|
hns3_clear_ring_group(&tqp_vector->tx_group);
|
2017-08-02 23:59:45 +08:00
|
|
|
netif_napi_del(&priv->tqp_vector[i].napi);
|
|
|
|
}
|
2018-03-09 10:37:02 +08:00
|
|
|
}
|
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
static void hns3_nic_dealloc_vector_data(struct hns3_nic_priv *priv)
|
2018-03-09 10:37:02 +08:00
|
|
|
{
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
struct pci_dev *pdev = h->pdev;
|
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
for (i = 0; i < priv->vector_num; i++) {
|
|
|
|
struct hns3_enet_tqp_vector *tqp_vector;
|
|
|
|
|
|
|
|
tqp_vector = &priv->tqp_vector[i];
|
|
|
|
ret = h->ae_algo->ops->put_vector(h, tqp_vector->vector_irq);
|
|
|
|
if (ret)
|
2020-01-04 10:49:25 +08:00
|
|
|
return;
|
2018-03-09 10:37:02 +08:00
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
devm_kfree(&pdev->dev, priv->tqp_vector);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
static void hns3_ring_get_cfg(struct hnae3_queue *q, struct hns3_nic_priv *priv,
|
|
|
|
unsigned int ring_type)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
int queue_num = priv->ae_handle->kinfo.num_tqps;
|
|
|
|
struct hns3_enet_ring *ring;
|
2019-02-23 17:22:15 +08:00
|
|
|
int desc_num;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
if (ring_type == HNAE3_RING_TYPE_TX) {
|
2019-10-19 16:03:49 +08:00
|
|
|
ring = &priv->ring[q->tqp_index];
|
2019-02-23 17:22:15 +08:00
|
|
|
desc_num = priv->ae_handle->kinfo.num_tx_desc;
|
2019-10-19 16:03:49 +08:00
|
|
|
ring->queue_index = q->tqp_index;
|
2017-08-02 23:59:45 +08:00
|
|
|
} else {
|
2019-10-19 16:03:49 +08:00
|
|
|
ring = &priv->ring[q->tqp_index + queue_num];
|
2019-02-23 17:22:15 +08:00
|
|
|
desc_num = priv->ae_handle->kinfo.num_rx_desc;
|
2019-10-19 16:03:49 +08:00
|
|
|
ring->queue_index = q->tqp_index;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2018-07-02 15:50:26 +08:00
|
|
|
hnae3_set_bit(ring->flag, HNAE3_RING_TYPE_B, ring_type);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
ring->tqp = q;
|
|
|
|
ring->desc = NULL;
|
|
|
|
ring->desc_cb = NULL;
|
|
|
|
ring->dev = priv->dev;
|
|
|
|
ring->desc_dma_addr = 0;
|
|
|
|
ring->buf_size = q->buf_size;
|
2019-01-18 16:13:03 +08:00
|
|
|
ring->desc_num = desc_num;
|
2017-08-02 23:59:45 +08:00
|
|
|
ring->next_to_use = 0;
|
|
|
|
ring->next_to_clean = 0;
|
2020-09-16 17:33:47 +08:00
|
|
|
ring->last_to_use = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
static void hns3_queue_to_ring(struct hnae3_queue *tqp,
|
|
|
|
struct hns3_nic_priv *priv)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
2019-10-19 16:03:49 +08:00
|
|
|
hns3_ring_get_cfg(tqp, priv, HNAE3_RING_TYPE_TX);
|
|
|
|
hns3_ring_get_cfg(tqp, priv, HNAE3_RING_TYPE_RX);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_get_ring_config(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
struct pci_dev *pdev = h->pdev;
|
2019-10-19 16:03:49 +08:00
|
|
|
int i;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
priv->ring = devm_kzalloc(&pdev->dev,
|
|
|
|
array3_size(h->kinfo.num_tqps,
|
|
|
|
sizeof(*priv->ring), 2),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!priv->ring)
|
2017-08-02 23:59:45 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++)
|
|
|
|
hns3_queue_to_ring(h->kinfo.tqp[i], priv);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-12-22 12:21:41 +08:00
|
|
|
static void hns3_put_ring_config(struct hns3_nic_priv *priv)
|
|
|
|
{
|
2019-10-19 16:03:49 +08:00
|
|
|
if (!priv->ring)
|
2019-04-25 20:42:54 +08:00
|
|
|
return;
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
devm_kfree(priv->dev, priv->ring);
|
|
|
|
priv->ring = NULL;
|
2017-12-22 12:21:41 +08:00
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_alloc_ring_memory(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (ring->desc_num <= 0 || ring->buf_size <= 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2019-05-06 10:48:52 +08:00
|
|
|
ring->desc_cb = devm_kcalloc(ring_to_dev(ring), ring->desc_num,
|
|
|
|
sizeof(ring->desc_cb[0]), GFP_KERNEL);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (!ring->desc_cb) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = hns3_alloc_desc(ring);
|
|
|
|
if (ret)
|
|
|
|
goto out_with_desc_cb;
|
|
|
|
|
|
|
|
if (!HNAE3_IS_TX_RING(ring)) {
|
|
|
|
ret = hns3_alloc_ring_buffers(ring);
|
|
|
|
if (ret)
|
|
|
|
goto out_with_desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_with_desc:
|
|
|
|
hns3_free_desc(ring);
|
|
|
|
out_with_desc_cb:
|
2019-05-06 10:48:52 +08:00
|
|
|
devm_kfree(ring_to_dev(ring), ring->desc_cb);
|
2017-08-02 23:59:45 +08:00
|
|
|
ring->desc_cb = NULL;
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-08-01 11:55:37 +08:00
|
|
|
void hns3_fini_ring(struct hns3_enet_ring *ring)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
hns3_free_desc(ring);
|
2019-05-06 10:48:52 +08:00
|
|
|
devm_kfree(ring_to_dev(ring), ring->desc_cb);
|
2017-08-02 23:59:45 +08:00
|
|
|
ring->desc_cb = NULL;
|
|
|
|
ring->next_to_clean = 0;
|
|
|
|
ring->next_to_use = 0;
|
2020-09-16 17:33:47 +08:00
|
|
|
ring->last_to_use = 0;
|
2019-01-23 07:39:33 +08:00
|
|
|
ring->pending_buf = 0;
|
|
|
|
if (ring->skb) {
|
|
|
|
dev_kfree_skb_any(ring->skb);
|
|
|
|
ring->skb = NULL;
|
|
|
|
}
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
2017-10-09 15:44:01 +08:00
|
|
|
static int hns3_buf_size2type(u32 buf_size)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
int bd_size_type;
|
|
|
|
|
|
|
|
switch (buf_size) {
|
|
|
|
case 512:
|
|
|
|
bd_size_type = HNS3_BD_SIZE_512_TYPE;
|
|
|
|
break;
|
|
|
|
case 1024:
|
|
|
|
bd_size_type = HNS3_BD_SIZE_1024_TYPE;
|
|
|
|
break;
|
|
|
|
case 2048:
|
|
|
|
bd_size_type = HNS3_BD_SIZE_2048_TYPE;
|
|
|
|
break;
|
|
|
|
case 4096:
|
|
|
|
bd_size_type = HNS3_BD_SIZE_4096_TYPE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
bd_size_type = HNS3_BD_SIZE_2048_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bd_size_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_init_ring_hw(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
dma_addr_t dma = ring->desc_dma_addr;
|
|
|
|
struct hnae3_queue *q = ring->tqp;
|
|
|
|
|
|
|
|
if (!HNAE3_IS_TX_RING(ring)) {
|
2019-06-07 10:03:13 +08:00
|
|
|
hns3_write_dev(q, HNS3_RING_RX_RING_BASEADDR_L_REG, (u32)dma);
|
2017-08-02 23:59:45 +08:00
|
|
|
hns3_write_dev(q, HNS3_RING_RX_RING_BASEADDR_H_REG,
|
|
|
|
(u32)((dma >> 31) >> 1));
|
|
|
|
|
|
|
|
hns3_write_dev(q, HNS3_RING_RX_RING_BD_LEN_REG,
|
|
|
|
hns3_buf_size2type(ring->buf_size));
|
|
|
|
hns3_write_dev(q, HNS3_RING_RX_RING_BD_NUM_REG,
|
|
|
|
ring->desc_num / 8 - 1);
|
|
|
|
} else {
|
|
|
|
hns3_write_dev(q, HNS3_RING_TX_RING_BASEADDR_L_REG,
|
|
|
|
(u32)dma);
|
|
|
|
hns3_write_dev(q, HNS3_RING_TX_RING_BASEADDR_H_REG,
|
|
|
|
(u32)((dma >> 31) >> 1));
|
|
|
|
|
|
|
|
hns3_write_dev(q, HNS3_RING_TX_RING_BD_NUM_REG,
|
|
|
|
ring->desc_num / 8 - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-15 00:13:18 +08:00
|
|
|
static void hns3_init_tx_ring_tc(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hnae3_knic_private_info *kinfo = &priv->ae_handle->kinfo;
|
2020-12-10 11:42:06 +08:00
|
|
|
struct hnae3_tc_info *tc_info = &kinfo->tc_info;
|
2018-08-15 00:13:18 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < HNAE3_MAX_TC; i++) {
|
|
|
|
int j;
|
|
|
|
|
2020-12-10 11:42:06 +08:00
|
|
|
if (!test_bit(i, &tc_info->tc_en))
|
2018-08-15 00:13:18 +08:00
|
|
|
continue;
|
|
|
|
|
2020-12-10 11:42:06 +08:00
|
|
|
for (j = 0; j < tc_info->tqp_count[i]; j++) {
|
2018-08-15 00:13:18 +08:00
|
|
|
struct hnae3_queue *q;
|
|
|
|
|
2020-12-10 11:42:06 +08:00
|
|
|
q = priv->ring[tc_info->tqp_offset[i] + j].tqp;
|
|
|
|
hns3_write_dev(q, HNS3_RING_TX_RING_TC_REG, i);
|
2018-08-15 00:13:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-10 16:42:04 +08:00
|
|
|
int hns3_init_all_ring(struct hns3_nic_priv *priv)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
int ring_num = h->kinfo.num_tqps * 2;
|
|
|
|
int i, j;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
for (i = 0; i < ring_num; i++) {
|
2019-10-19 16:03:49 +08:00
|
|
|
ret = hns3_alloc_ring_memory(&priv->ring[i]);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (ret) {
|
|
|
|
dev_err(priv->dev,
|
|
|
|
"Alloc ring memory fail! ret=%d\n", ret);
|
|
|
|
goto out_when_alloc_ring_memory;
|
|
|
|
}
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
u64_stats_init(&priv->ring[i].syncp);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_when_alloc_ring_memory:
|
|
|
|
for (j = i - 1; j >= 0; j--)
|
2019-10-19 16:03:49 +08:00
|
|
|
hns3_fini_ring(&priv->ring[j]);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2021-02-09 10:41:58 +08:00
|
|
|
static void hns3_uninit_all_ring(struct hns3_nic_priv *priv)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++) {
|
2019-10-19 16:03:49 +08:00
|
|
|
hns3_fini_ring(&priv->ring[i]);
|
|
|
|
hns3_fini_ring(&priv->ring[i + h->kinfo.num_tqps]);
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set mac addr if it is configured. or leave it to the AE driver */
|
2019-10-08 09:20:08 +08:00
|
|
|
static int hns3_init_mac_addr(struct net_device *netdev)
|
2017-08-02 23:59:45 +08:00
|
|
|
{
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
struct hnae3_handle *h = priv->ae_handle;
|
|
|
|
u8 mac_addr_temp[ETH_ALEN];
|
2018-10-30 21:50:50 +08:00
|
|
|
int ret = 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-10-08 09:20:08 +08:00
|
|
|
if (h->ae_algo->ops->get_mac_addr)
|
2017-08-02 23:59:45 +08:00
|
|
|
h->ae_algo->ops->get_mac_addr(h, mac_addr_temp);
|
|
|
|
|
|
|
|
/* Check if the MAC address is valid, if not get a random one */
|
2019-10-08 09:20:08 +08:00
|
|
|
if (!is_valid_ether_addr(mac_addr_temp)) {
|
2017-08-02 23:59:45 +08:00
|
|
|
eth_hw_addr_random(netdev);
|
|
|
|
dev_warn(priv->dev, "using random MAC address %pM\n",
|
|
|
|
netdev->dev_addr);
|
net: hns3: refactor the MAC address configure
Currently, the HNS3 driver sync and unsync MAC address in function
hns3_set_rx_mode(). For PF, it adds and deletes MAC address directly
in the path of dev_set_rx_mode(). If failed, it won't retry until
next calling of hns3_set_rx_mode(). On the other hand, if request
add and remove a same address many times at a short interval, each
request must be done one by one, can't be merged. For VF, it sends
mailbox messages to PF to request adding or deleting MAC address in
the path of function hns3_set_rx_mode(), no matter the address is
configured success.
This patch refines it by recording the MAC address in function
hns3_set_rx_mode(), and updating MAC address in the service task.
If failed, it will retry by the next calling of periodical service
task. It also uses some state to mark the state of each MAC address
in the MAC list, which can help merge configure request for a same
address. With these changes, when global reset or IMP reset occurs,
we can restore the MAC table with the MAC list.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2020-04-24 10:23:09 +08:00
|
|
|
} else if (!ether_addr_equal(netdev->dev_addr, mac_addr_temp)) {
|
2019-10-08 09:20:08 +08:00
|
|
|
ether_addr_copy(netdev->dev_addr, mac_addr_temp);
|
|
|
|
ether_addr_copy(netdev->perm_addr, mac_addr_temp);
|
net: hns3: refactor the MAC address configure
Currently, the HNS3 driver sync and unsync MAC address in function
hns3_set_rx_mode(). For PF, it adds and deletes MAC address directly
in the path of dev_set_rx_mode(). If failed, it won't retry until
next calling of hns3_set_rx_mode(). On the other hand, if request
add and remove a same address many times at a short interval, each
request must be done one by one, can't be merged. For VF, it sends
mailbox messages to PF to request adding or deleting MAC address in
the path of function hns3_set_rx_mode(), no matter the address is
configured success.
This patch refines it by recording the MAC address in function
hns3_set_rx_mode(), and updating MAC address in the service task.
If failed, it will retry by the next calling of periodical service
task. It also uses some state to mark the state of each MAC address
in the MAC list, which can help merge configure request for a same
address. With these changes, when global reset or IMP reset occurs,
we can restore the MAC table with the MAC list.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2020-04-24 10:23:09 +08:00
|
|
|
} else {
|
|
|
|
return 0;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
2017-09-20 00:17:13 +08:00
|
|
|
|
|
|
|
if (h->ae_algo->ops->set_mac_addr)
|
2018-10-30 21:50:50 +08:00
|
|
|
ret = h->ae_algo->ops->set_mac_addr(h, netdev->dev_addr, true);
|
2017-09-20 00:17:13 +08:00
|
|
|
|
2018-10-30 21:50:50 +08:00
|
|
|
return ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
net: hns3: Fix NULL deref when unloading driver
When the driver is unloading, if there is a calling of ndo_open occurs
between phy_disconnect() and unregister_netdev(), it will end up
causing the kernel to eventually hit a NULL deref:
[14942.417828] Unable to handle kernel NULL pointer dereference at virtual address 0000000000000048
[14942.529878] Mem abort info:
[14942.551166] ESR = 0x96000006
[14942.567070] Exception class = DABT (current EL), IL = 32 bits
[14942.623081] SET = 0, FnV = 0
[14942.639112] EA = 0, S1PTW = 0
[14942.643628] Data abort info:
[14942.659227] ISV = 0, ISS = 0x00000006
[14942.674870] CM = 0, WnR = 0
[14942.679449] user pgtable: 4k pages, 48-bit VAs, pgdp = 00000000224ad6ad
[14942.695595] [0000000000000048] pgd=00000021e6673003, pud=00000021dbf01003, pmd=0000000000000000
[14942.723163] Internal error: Oops: 96000006 [#1] PREEMPT SMP
[14942.729358] Modules linked in: hns3(O) hclge(O) pv680_mii(O) hnae3(O) [last unloaded: hclge]
[14942.738907] CPU: 1 PID: 26629 Comm: kworker/u4:13 Tainted: G O 4.18.0-rc1-12928-ga960791-dirty #145
[14942.749491] Hardware name: Huawei Technologies Co., Ltd. D05/D05, BIOS Hi1620 FPGA TB BOOT BIOS B763 08/17/2018
[14942.760392] Workqueue: events_power_efficient phy_state_machine
[14942.766644] pstate: 80c00009 (Nzcv daif +PAN +UAO)
[14942.771918] pc : test_and_set_bit+0x18/0x38
[14942.776589] lr : netif_carrier_off+0x24/0x70
[14942.781033] sp : ffff0000121abd20
[14942.784518] x29: ffff0000121abd20 x28: 0000000000000000
[14942.790208] x27: ffff0000164d3cd8 x26: ffff8021da68b7b8
[14942.795832] x25: 0000000000000000 x24: ffff8021eb407800
[14942.801445] x23: 0000000000000000 x22: 0000000000000000
[14942.807046] x21: 0000000000000001 x20: 0000000000000000
[14942.812672] x19: 0000000000000000 x18: ffff000009781708
[14942.818284] x17: 00000000004970e8 x16: ffff00000816ad48
[14942.823900] x15: 0000000000000000 x14: 0000000000000008
[14942.829528] x13: 0000000000000000 x12: 0000000000000f65
[14942.835149] x11: 0000000000000001 x10: 00000000000009d0
[14942.840753] x9 : ffff0000121abaa0 x8 : 0000000000000000
[14942.846360] x7 : ffff000009781708 x6 : 0000000000000003
[14942.851970] x5 : 0000000000000020 x4 : 0000000000000004
[14942.857575] x3 : 0000000000000002 x2 : 0000000000000001
[14942.863180] x1 : 0000000000000048 x0 : 0000000000000000
[14942.868875] Process kworker/u4:13 (pid: 26629, stack limit = 0x00000000c909dbf3)
[14942.876464] Call trace:
[14942.879200] test_and_set_bit+0x18/0x38
[14942.883376] phy_link_change+0x38/0x78
[14942.887378] phy_state_machine+0x3dc/0x4f8
[14942.891968] process_one_work+0x158/0x470
[14942.896223] worker_thread+0x50/0x470
[14942.900219] kthread+0x104/0x130
[14942.903905] ret_from_fork+0x10/0x1c
[14942.907755] Code: d2800022 8b400c21 f9800031 9ac32044 (c85f7c22)
[14942.914185] ---[ end trace 968c9e12eb740b23 ]---
So this patch fixes it by modifying the timing to do phy_connect_direct()
and phy_disconnect().
Fixes: 256727da7395 ("net: hns3: Add MDIO support to HNS3 Ethernet driver for hip08 SoC")
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Peng Li <lipeng321@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-01-31 04:55:46 +08:00
|
|
|
static int hns3_init_phy(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (h->ae_algo->ops->mac_connect_phy)
|
|
|
|
ret = h->ae_algo->ops->mac_connect_phy(h);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_uninit_phy(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
|
|
|
|
if (h->ae_algo->ops->mac_disconnect_phy)
|
|
|
|
h->ae_algo->ops->mac_disconnect_phy(h);
|
|
|
|
}
|
|
|
|
|
2018-11-18 11:19:12 +08:00
|
|
|
static int hns3_client_start(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
if (!handle->ae_algo->ops->client_start)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return handle->ae_algo->ops->client_start(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_client_stop(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
if (!handle->ae_algo->ops->client_stop)
|
|
|
|
return;
|
|
|
|
|
|
|
|
handle->ae_algo->ops->client_stop(handle);
|
|
|
|
}
|
|
|
|
|
2019-04-19 11:05:43 +08:00
|
|
|
static void hns3_info_show(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
struct hnae3_knic_private_info *kinfo = &priv->ae_handle->kinfo;
|
|
|
|
|
|
|
|
dev_info(priv->dev, "MAC address: %pM\n", priv->netdev->dev_addr);
|
2019-10-31 19:23:23 +08:00
|
|
|
dev_info(priv->dev, "Task queue pairs numbers: %u\n", kinfo->num_tqps);
|
|
|
|
dev_info(priv->dev, "RSS size: %u\n", kinfo->rss_size);
|
|
|
|
dev_info(priv->dev, "Allocated RSS size: %u\n", kinfo->req_rss_size);
|
|
|
|
dev_info(priv->dev, "RX buffer length: %u\n", kinfo->rx_buf_len);
|
|
|
|
dev_info(priv->dev, "Desc num per TX queue: %u\n", kinfo->num_tx_desc);
|
|
|
|
dev_info(priv->dev, "Desc num per RX queue: %u\n", kinfo->num_rx_desc);
|
2020-12-10 11:42:06 +08:00
|
|
|
dev_info(priv->dev, "Total number of enabled TCs: %u\n",
|
|
|
|
kinfo->tc_info.num_tc);
|
2019-10-31 19:23:23 +08:00
|
|
|
dev_info(priv->dev, "Max mtu size: %u\n", priv->netdev->max_mtu);
|
2019-04-19 11:05:43 +08:00
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
static int hns3_client_init(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
struct pci_dev *pdev = handle->pdev;
|
2020-09-29 17:31:59 +08:00
|
|
|
struct hnae3_ae_dev *ae_dev = pci_get_drvdata(pdev);
|
2018-09-27 02:28:39 +08:00
|
|
|
u16 alloc_tqps, max_rss_size;
|
2017-08-02 23:59:45 +08:00
|
|
|
struct hns3_nic_priv *priv;
|
|
|
|
struct net_device *netdev;
|
|
|
|
int ret;
|
|
|
|
|
2018-09-27 02:28:39 +08:00
|
|
|
handle->ae_algo->ops->get_tqps_and_rss_info(handle, &alloc_tqps,
|
|
|
|
&max_rss_size);
|
|
|
|
netdev = alloc_etherdev_mq(sizeof(struct hns3_nic_priv), alloc_tqps);
|
2017-08-02 23:59:45 +08:00
|
|
|
if (!netdev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
priv = netdev_priv(netdev);
|
|
|
|
priv->dev = &pdev->dev;
|
|
|
|
priv->netdev = netdev;
|
|
|
|
priv->ae_handle = handle;
|
2017-11-02 20:45:20 +08:00
|
|
|
priv->tx_timeout_count = 0;
|
2020-09-29 17:31:59 +08:00
|
|
|
priv->max_non_tso_bd_num = ae_dev->dev_specs.max_non_tso_bd_num;
|
2019-01-27 00:49:20 +08:00
|
|
|
set_bit(HNS3_NIC_STATE_DOWN, &priv->state);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2019-04-19 11:05:43 +08:00
|
|
|
handle->msg_enable = netif_msg_init(debug, DEFAULT_MSG_LEVEL);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
handle->kinfo.netdev = netdev;
|
|
|
|
handle->priv = (void *)priv;
|
|
|
|
|
2019-10-08 09:20:08 +08:00
|
|
|
hns3_init_mac_addr(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
hns3_set_default_feature(netdev);
|
|
|
|
|
|
|
|
netdev->watchdog_timeo = HNS3_TX_TIMEOUT;
|
|
|
|
netdev->priv_flags |= IFF_UNICAST_FLT;
|
|
|
|
netdev->netdev_ops = &hns3_nic_netdev_ops;
|
|
|
|
SET_NETDEV_DEV(netdev, &pdev->dev);
|
|
|
|
hns3_ethtool_set_ops(netdev);
|
|
|
|
|
|
|
|
/* Carrier off reporting is important to ethtool even BEFORE open */
|
|
|
|
netif_carrier_off(netdev);
|
|
|
|
|
|
|
|
ret = hns3_get_ring_config(priv);
|
|
|
|
if (ret) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out_get_ring_cfg;
|
|
|
|
}
|
|
|
|
|
2018-03-09 10:37:02 +08:00
|
|
|
ret = hns3_nic_alloc_vector_data(priv);
|
|
|
|
if (ret) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out_alloc_vector_data;
|
|
|
|
}
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
ret = hns3_nic_init_vector_data(priv);
|
|
|
|
if (ret) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out_init_vector_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = hns3_init_all_ring(priv);
|
|
|
|
if (ret) {
|
|
|
|
ret = -ENOMEM;
|
2019-10-19 16:03:49 +08:00
|
|
|
goto out_init_ring;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
|
net: hns3: Fix NULL deref when unloading driver
When the driver is unloading, if there is a calling of ndo_open occurs
between phy_disconnect() and unregister_netdev(), it will end up
causing the kernel to eventually hit a NULL deref:
[14942.417828] Unable to handle kernel NULL pointer dereference at virtual address 0000000000000048
[14942.529878] Mem abort info:
[14942.551166] ESR = 0x96000006
[14942.567070] Exception class = DABT (current EL), IL = 32 bits
[14942.623081] SET = 0, FnV = 0
[14942.639112] EA = 0, S1PTW = 0
[14942.643628] Data abort info:
[14942.659227] ISV = 0, ISS = 0x00000006
[14942.674870] CM = 0, WnR = 0
[14942.679449] user pgtable: 4k pages, 48-bit VAs, pgdp = 00000000224ad6ad
[14942.695595] [0000000000000048] pgd=00000021e6673003, pud=00000021dbf01003, pmd=0000000000000000
[14942.723163] Internal error: Oops: 96000006 [#1] PREEMPT SMP
[14942.729358] Modules linked in: hns3(O) hclge(O) pv680_mii(O) hnae3(O) [last unloaded: hclge]
[14942.738907] CPU: 1 PID: 26629 Comm: kworker/u4:13 Tainted: G O 4.18.0-rc1-12928-ga960791-dirty #145
[14942.749491] Hardware name: Huawei Technologies Co., Ltd. D05/D05, BIOS Hi1620 FPGA TB BOOT BIOS B763 08/17/2018
[14942.760392] Workqueue: events_power_efficient phy_state_machine
[14942.766644] pstate: 80c00009 (Nzcv daif +PAN +UAO)
[14942.771918] pc : test_and_set_bit+0x18/0x38
[14942.776589] lr : netif_carrier_off+0x24/0x70
[14942.781033] sp : ffff0000121abd20
[14942.784518] x29: ffff0000121abd20 x28: 0000000000000000
[14942.790208] x27: ffff0000164d3cd8 x26: ffff8021da68b7b8
[14942.795832] x25: 0000000000000000 x24: ffff8021eb407800
[14942.801445] x23: 0000000000000000 x22: 0000000000000000
[14942.807046] x21: 0000000000000001 x20: 0000000000000000
[14942.812672] x19: 0000000000000000 x18: ffff000009781708
[14942.818284] x17: 00000000004970e8 x16: ffff00000816ad48
[14942.823900] x15: 0000000000000000 x14: 0000000000000008
[14942.829528] x13: 0000000000000000 x12: 0000000000000f65
[14942.835149] x11: 0000000000000001 x10: 00000000000009d0
[14942.840753] x9 : ffff0000121abaa0 x8 : 0000000000000000
[14942.846360] x7 : ffff000009781708 x6 : 0000000000000003
[14942.851970] x5 : 0000000000000020 x4 : 0000000000000004
[14942.857575] x3 : 0000000000000002 x2 : 0000000000000001
[14942.863180] x1 : 0000000000000048 x0 : 0000000000000000
[14942.868875] Process kworker/u4:13 (pid: 26629, stack limit = 0x00000000c909dbf3)
[14942.876464] Call trace:
[14942.879200] test_and_set_bit+0x18/0x38
[14942.883376] phy_link_change+0x38/0x78
[14942.887378] phy_state_machine+0x3dc/0x4f8
[14942.891968] process_one_work+0x158/0x470
[14942.896223] worker_thread+0x50/0x470
[14942.900219] kthread+0x104/0x130
[14942.903905] ret_from_fork+0x10/0x1c
[14942.907755] Code: d2800022 8b400c21 f9800031 9ac32044 (c85f7c22)
[14942.914185] ---[ end trace 968c9e12eb740b23 ]---
So this patch fixes it by modifying the timing to do phy_connect_direct()
and phy_disconnect().
Fixes: 256727da7395 ("net: hns3: Add MDIO support to HNS3 Ethernet driver for hip08 SoC")
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Peng Li <lipeng321@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-01-31 04:55:46 +08:00
|
|
|
ret = hns3_init_phy(netdev);
|
|
|
|
if (ret)
|
|
|
|
goto out_init_phy;
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
ret = register_netdev(netdev);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(priv->dev, "probe register netdev fail!\n");
|
|
|
|
goto out_reg_netdev_fail;
|
|
|
|
}
|
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
/* the device can work without cpu rmap, only aRFS needs it */
|
|
|
|
ret = hns3_set_rx_cpu_rmap(netdev);
|
|
|
|
if (ret)
|
|
|
|
dev_warn(priv->dev, "set rx cpu rmap fail, ret=%d\n", ret);
|
|
|
|
|
|
|
|
ret = hns3_nic_init_irq(priv);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(priv->dev, "init irq failed! ret=%d\n", ret);
|
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
goto out_init_irq_fail;
|
|
|
|
}
|
|
|
|
|
2018-11-18 11:19:12 +08:00
|
|
|
ret = hns3_client_start(handle);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(priv->dev, "hns3_client_start fail! ret=%d\n", ret);
|
2019-06-19 02:54:22 +08:00
|
|
|
goto out_client_start;
|
2018-11-18 11:19:12 +08:00
|
|
|
}
|
|
|
|
|
2017-09-27 09:45:30 +08:00
|
|
|
hns3_dcbnl_setup(handle);
|
|
|
|
|
2018-11-22 22:09:41 +08:00
|
|
|
hns3_dbg_init(handle);
|
|
|
|
|
2021-02-05 16:32:47 +08:00
|
|
|
netdev->max_mtu = HNS3_MAX_MTU(ae_dev->dev_specs.max_frm_size);
|
2017-08-22 00:05:24 +08:00
|
|
|
|
2020-11-28 11:51:45 +08:00
|
|
|
if (test_bit(HNAE3_DEV_SUPPORT_HW_TX_CSUM_B, ae_dev->caps))
|
|
|
|
set_bit(HNS3_NIC_STATE_HW_TX_CSUM_ENABLE, &priv->state);
|
|
|
|
|
2018-11-07 12:06:07 +08:00
|
|
|
set_bit(HNS3_NIC_STATE_INITED, &priv->state);
|
|
|
|
|
2020-12-06 12:06:14 +08:00
|
|
|
if (ae_dev->dev_version >= HNAE3_DEVICE_VERSION_V3)
|
|
|
|
set_bit(HNAE3_PFLAG_LIMIT_PROMISC, &handle->supported_pflags);
|
|
|
|
|
2019-04-19 11:05:43 +08:00
|
|
|
if (netif_msg_drv(handle))
|
|
|
|
hns3_info_show(priv);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
return ret;
|
|
|
|
|
2019-02-23 17:22:19 +08:00
|
|
|
out_client_start:
|
2020-01-04 10:49:25 +08:00
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
hns3_nic_uninit_irq(priv);
|
|
|
|
out_init_irq_fail:
|
2019-02-23 17:22:19 +08:00
|
|
|
unregister_netdev(netdev);
|
2017-08-02 23:59:45 +08:00
|
|
|
out_reg_netdev_fail:
|
net: hns3: Fix NULL deref when unloading driver
When the driver is unloading, if there is a calling of ndo_open occurs
between phy_disconnect() and unregister_netdev(), it will end up
causing the kernel to eventually hit a NULL deref:
[14942.417828] Unable to handle kernel NULL pointer dereference at virtual address 0000000000000048
[14942.529878] Mem abort info:
[14942.551166] ESR = 0x96000006
[14942.567070] Exception class = DABT (current EL), IL = 32 bits
[14942.623081] SET = 0, FnV = 0
[14942.639112] EA = 0, S1PTW = 0
[14942.643628] Data abort info:
[14942.659227] ISV = 0, ISS = 0x00000006
[14942.674870] CM = 0, WnR = 0
[14942.679449] user pgtable: 4k pages, 48-bit VAs, pgdp = 00000000224ad6ad
[14942.695595] [0000000000000048] pgd=00000021e6673003, pud=00000021dbf01003, pmd=0000000000000000
[14942.723163] Internal error: Oops: 96000006 [#1] PREEMPT SMP
[14942.729358] Modules linked in: hns3(O) hclge(O) pv680_mii(O) hnae3(O) [last unloaded: hclge]
[14942.738907] CPU: 1 PID: 26629 Comm: kworker/u4:13 Tainted: G O 4.18.0-rc1-12928-ga960791-dirty #145
[14942.749491] Hardware name: Huawei Technologies Co., Ltd. D05/D05, BIOS Hi1620 FPGA TB BOOT BIOS B763 08/17/2018
[14942.760392] Workqueue: events_power_efficient phy_state_machine
[14942.766644] pstate: 80c00009 (Nzcv daif +PAN +UAO)
[14942.771918] pc : test_and_set_bit+0x18/0x38
[14942.776589] lr : netif_carrier_off+0x24/0x70
[14942.781033] sp : ffff0000121abd20
[14942.784518] x29: ffff0000121abd20 x28: 0000000000000000
[14942.790208] x27: ffff0000164d3cd8 x26: ffff8021da68b7b8
[14942.795832] x25: 0000000000000000 x24: ffff8021eb407800
[14942.801445] x23: 0000000000000000 x22: 0000000000000000
[14942.807046] x21: 0000000000000001 x20: 0000000000000000
[14942.812672] x19: 0000000000000000 x18: ffff000009781708
[14942.818284] x17: 00000000004970e8 x16: ffff00000816ad48
[14942.823900] x15: 0000000000000000 x14: 0000000000000008
[14942.829528] x13: 0000000000000000 x12: 0000000000000f65
[14942.835149] x11: 0000000000000001 x10: 00000000000009d0
[14942.840753] x9 : ffff0000121abaa0 x8 : 0000000000000000
[14942.846360] x7 : ffff000009781708 x6 : 0000000000000003
[14942.851970] x5 : 0000000000000020 x4 : 0000000000000004
[14942.857575] x3 : 0000000000000002 x2 : 0000000000000001
[14942.863180] x1 : 0000000000000048 x0 : 0000000000000000
[14942.868875] Process kworker/u4:13 (pid: 26629, stack limit = 0x00000000c909dbf3)
[14942.876464] Call trace:
[14942.879200] test_and_set_bit+0x18/0x38
[14942.883376] phy_link_change+0x38/0x78
[14942.887378] phy_state_machine+0x3dc/0x4f8
[14942.891968] process_one_work+0x158/0x470
[14942.896223] worker_thread+0x50/0x470
[14942.900219] kthread+0x104/0x130
[14942.903905] ret_from_fork+0x10/0x1c
[14942.907755] Code: d2800022 8b400c21 f9800031 9ac32044 (c85f7c22)
[14942.914185] ---[ end trace 968c9e12eb740b23 ]---
So this patch fixes it by modifying the timing to do phy_connect_direct()
and phy_disconnect().
Fixes: 256727da7395 ("net: hns3: Add MDIO support to HNS3 Ethernet driver for hip08 SoC")
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: Peng Li <lipeng321@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-01-31 04:55:46 +08:00
|
|
|
hns3_uninit_phy(netdev);
|
|
|
|
out_init_phy:
|
|
|
|
hns3_uninit_all_ring(priv);
|
2019-10-19 16:03:49 +08:00
|
|
|
out_init_ring:
|
2019-01-31 04:55:47 +08:00
|
|
|
hns3_nic_uninit_vector_data(priv);
|
2017-08-02 23:59:45 +08:00
|
|
|
out_init_vector_data:
|
2018-03-09 10:37:02 +08:00
|
|
|
hns3_nic_dealloc_vector_data(priv);
|
|
|
|
out_alloc_vector_data:
|
2019-10-19 16:03:49 +08:00
|
|
|
priv->ring = NULL;
|
2017-08-02 23:59:45 +08:00
|
|
|
out_get_ring_cfg:
|
|
|
|
priv->ae_handle = NULL;
|
|
|
|
free_netdev(netdev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_client_uninit(struct hnae3_handle *handle, bool reset)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = handle->kinfo.netdev;
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
|
|
|
|
if (netdev->reg_state != NETREG_UNINITIALIZED)
|
|
|
|
unregister_netdev(netdev);
|
|
|
|
|
2019-04-06 15:43:28 +08:00
|
|
|
hns3_client_stop(handle);
|
|
|
|
|
2019-05-28 17:03:01 +08:00
|
|
|
hns3_uninit_phy(netdev);
|
|
|
|
|
2018-11-07 12:06:07 +08:00
|
|
|
if (!test_and_clear_bit(HNS3_NIC_STATE_INITED, &priv->state)) {
|
|
|
|
netdev_warn(netdev, "already uninitialized\n");
|
|
|
|
goto out_netdev_free;
|
|
|
|
}
|
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
|
|
|
|
hns3_nic_uninit_irq(priv);
|
|
|
|
|
2019-06-28 19:50:07 +08:00
|
|
|
hns3_clear_all_ring(handle, true);
|
2018-05-26 02:43:04 +08:00
|
|
|
|
2019-01-31 04:55:47 +08:00
|
|
|
hns3_nic_uninit_vector_data(priv);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
hns3_nic_dealloc_vector_data(priv);
|
2018-03-09 10:37:02 +08:00
|
|
|
|
2021-02-09 10:41:58 +08:00
|
|
|
hns3_uninit_all_ring(priv);
|
2017-08-02 23:59:45 +08:00
|
|
|
|
2018-03-09 10:37:00 +08:00
|
|
|
hns3_put_ring_config(priv);
|
|
|
|
|
2018-11-07 12:06:07 +08:00
|
|
|
out_netdev_free:
|
2020-07-06 19:26:01 +08:00
|
|
|
hns3_dbg_uninit(handle);
|
2017-08-02 23:59:45 +08:00
|
|
|
free_netdev(netdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_link_status_change(struct hnae3_handle *handle, bool linkup)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = handle->kinfo.netdev;
|
|
|
|
|
|
|
|
if (!netdev)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (linkup) {
|
|
|
|
netif_tx_wake_all_queues(netdev);
|
2020-07-28 10:16:49 +08:00
|
|
|
netif_carrier_on(netdev);
|
2019-04-19 11:05:43 +08:00
|
|
|
if (netif_msg_link(handle))
|
|
|
|
netdev_info(netdev, "link up\n");
|
2017-08-02 23:59:45 +08:00
|
|
|
} else {
|
|
|
|
netif_carrier_off(netdev);
|
|
|
|
netif_tx_stop_all_queues(netdev);
|
2019-04-19 11:05:43 +08:00
|
|
|
if (netif_msg_link(handle))
|
|
|
|
netdev_info(netdev, "link down\n");
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-10 00:24:40 +08:00
|
|
|
static void hns3_clear_tx_ring(struct hns3_enet_ring *ring)
|
2017-11-02 20:45:21 +08:00
|
|
|
{
|
2018-05-10 00:24:40 +08:00
|
|
|
while (ring->next_to_clean != ring->next_to_use) {
|
2018-05-26 02:43:04 +08:00
|
|
|
ring->desc[ring->next_to_clean].tx.bdtp_fe_sc_vld_ra_ri = 0;
|
2020-09-16 17:33:50 +08:00
|
|
|
hns3_free_buffer_detach(ring, ring->next_to_clean, 0);
|
2018-05-10 00:24:40 +08:00
|
|
|
ring_ptr_move_fw(ring, next_to_clean);
|
|
|
|
}
|
2020-09-16 17:33:46 +08:00
|
|
|
|
|
|
|
ring->pending_buf = 0;
|
2018-05-10 00:24:40 +08:00
|
|
|
}
|
|
|
|
|
2018-05-26 02:43:04 +08:00
|
|
|
static int hns3_clear_rx_ring(struct hns3_enet_ring *ring)
|
|
|
|
{
|
|
|
|
struct hns3_desc_cb res_cbs;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
while (ring->next_to_use != ring->next_to_clean) {
|
|
|
|
/* When a buffer is not reused, it's memory has been
|
|
|
|
* freed in hns3_handle_rx_bd or will be freed by
|
|
|
|
* stack, so we need to replace the buffer here.
|
|
|
|
*/
|
|
|
|
if (!ring->desc_cb[ring->next_to_use].reuse_flag) {
|
2020-06-18 09:02:09 +08:00
|
|
|
ret = hns3_alloc_and_map_buffer(ring, &res_cbs);
|
2018-05-26 02:43:04 +08:00
|
|
|
if (ret) {
|
|
|
|
u64_stats_update_begin(&ring->syncp);
|
|
|
|
ring->stats.sw_err_cnt++;
|
|
|
|
u64_stats_update_end(&ring->syncp);
|
|
|
|
/* if alloc new buffer fail, exit directly
|
|
|
|
* and reclear in up flow.
|
|
|
|
*/
|
2019-10-19 16:03:52 +08:00
|
|
|
netdev_warn(ring_to_netdev(ring),
|
2018-05-26 02:43:04 +08:00
|
|
|
"reserve buffer map failed, ret = %d\n",
|
|
|
|
ret);
|
|
|
|
return ret;
|
|
|
|
}
|
2019-06-07 10:03:13 +08:00
|
|
|
hns3_replace_buffer(ring, ring->next_to_use, &res_cbs);
|
2018-05-26 02:43:04 +08:00
|
|
|
}
|
|
|
|
ring_ptr_move_fw(ring, next_to_use);
|
|
|
|
}
|
|
|
|
|
2019-04-14 09:47:45 +08:00
|
|
|
/* Free the pending skb in rx ring */
|
|
|
|
if (ring->skb) {
|
|
|
|
dev_kfree_skb_any(ring->skb);
|
|
|
|
ring->skb = NULL;
|
|
|
|
ring->pending_buf = 0;
|
|
|
|
}
|
|
|
|
|
2018-05-26 02:43:04 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_force_clear_rx_ring(struct hns3_enet_ring *ring)
|
2018-05-10 00:24:40 +08:00
|
|
|
{
|
|
|
|
while (ring->next_to_use != ring->next_to_clean) {
|
|
|
|
/* When a buffer is not reused, it's memory has been
|
|
|
|
* freed in hns3_handle_rx_bd or will be freed by
|
|
|
|
* stack, so only need to unmap the buffer here.
|
|
|
|
*/
|
|
|
|
if (!ring->desc_cb[ring->next_to_use].reuse_flag) {
|
|
|
|
hns3_unmap_buffer(ring,
|
|
|
|
&ring->desc_cb[ring->next_to_use]);
|
|
|
|
ring->desc_cb[ring->next_to_use].dma = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ring_ptr_move_fw(ring, next_to_use);
|
|
|
|
}
|
2017-11-02 20:45:21 +08:00
|
|
|
}
|
|
|
|
|
2019-06-28 19:50:07 +08:00
|
|
|
static void hns3_clear_all_ring(struct hnae3_handle *h, bool force)
|
2017-11-02 20:45:21 +08:00
|
|
|
{
|
|
|
|
struct net_device *ndev = h->kinfo.netdev;
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(ndev);
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++) {
|
|
|
|
struct hns3_enet_ring *ring;
|
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
ring = &priv->ring[i];
|
2018-05-10 00:24:40 +08:00
|
|
|
hns3_clear_tx_ring(ring);
|
2017-11-02 20:45:21 +08:00
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
ring = &priv->ring[i + h->kinfo.num_tqps];
|
2018-05-26 02:43:04 +08:00
|
|
|
/* Continue to clear other rings even if clearing some
|
|
|
|
* rings failed.
|
|
|
|
*/
|
2019-06-28 19:50:07 +08:00
|
|
|
if (force)
|
|
|
|
hns3_force_clear_rx_ring(ring);
|
|
|
|
else
|
|
|
|
hns3_clear_rx_ring(ring);
|
2017-11-02 20:45:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-26 02:43:04 +08:00
|
|
|
int hns3_nic_reset_all_ring(struct hnae3_handle *h)
|
|
|
|
{
|
|
|
|
struct net_device *ndev = h->kinfo.netdev;
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(ndev);
|
|
|
|
struct hns3_enet_ring *rx_ring;
|
|
|
|
int i, j;
|
|
|
|
int ret;
|
|
|
|
|
2021-03-29 11:57:49 +08:00
|
|
|
ret = h->ae_algo->ops->reset_queue(h);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2018-10-30 21:50:50 +08:00
|
|
|
|
2021-03-29 11:57:49 +08:00
|
|
|
for (i = 0; i < h->kinfo.num_tqps; i++) {
|
2019-10-19 16:03:49 +08:00
|
|
|
hns3_init_ring_hw(&priv->ring[i]);
|
2018-05-26 02:43:04 +08:00
|
|
|
|
|
|
|
/* We need to clear tx ring here because self test will
|
|
|
|
* use the ring and will not run down before up
|
|
|
|
*/
|
2019-10-19 16:03:49 +08:00
|
|
|
hns3_clear_tx_ring(&priv->ring[i]);
|
|
|
|
priv->ring[i].next_to_clean = 0;
|
|
|
|
priv->ring[i].next_to_use = 0;
|
2020-09-16 17:33:47 +08:00
|
|
|
priv->ring[i].last_to_use = 0;
|
2018-05-26 02:43:04 +08:00
|
|
|
|
2019-10-19 16:03:49 +08:00
|
|
|
rx_ring = &priv->ring[i + h->kinfo.num_tqps];
|
2018-05-26 02:43:04 +08:00
|
|
|
hns3_init_ring_hw(rx_ring);
|
|
|
|
ret = hns3_clear_rx_ring(rx_ring);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* We can not know the hardware head and tail when this
|
|
|
|
* function is called in reset flow, so we reuse all desc.
|
|
|
|
*/
|
|
|
|
for (j = 0; j < rx_ring->desc_num; j++)
|
|
|
|
hns3_reuse_buffer(rx_ring, j);
|
|
|
|
|
|
|
|
rx_ring->next_to_clean = 0;
|
|
|
|
rx_ring->next_to_use = 0;
|
|
|
|
}
|
|
|
|
|
2018-08-15 00:13:18 +08:00
|
|
|
hns3_init_tx_ring_tc(priv);
|
|
|
|
|
2018-05-26 02:43:04 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-27 02:28:40 +08:00
|
|
|
static void hns3_store_coal(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
/* ethtool only support setting and querying one coal
|
2019-08-16 16:09:37 +08:00
|
|
|
* configuration for now, so save the vector 0' coal
|
|
|
|
* configuration here in order to restore it.
|
2018-09-27 02:28:40 +08:00
|
|
|
*/
|
|
|
|
memcpy(&priv->tx_coal, &priv->tqp_vector[0].tx_group.coal,
|
|
|
|
sizeof(struct hns3_enet_coalesce));
|
|
|
|
memcpy(&priv->rx_coal, &priv->tqp_vector[0].rx_group.coal,
|
|
|
|
sizeof(struct hns3_enet_coalesce));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hns3_restore_coal(struct hns3_nic_priv *priv)
|
|
|
|
{
|
|
|
|
u16 vector_num = priv->vector_num;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < vector_num; i++) {
|
|
|
|
memcpy(&priv->tqp_vector[i].tx_group.coal, &priv->tx_coal,
|
|
|
|
sizeof(struct hns3_enet_coalesce));
|
|
|
|
memcpy(&priv->tqp_vector[i].rx_group.coal, &priv->rx_coal,
|
|
|
|
sizeof(struct hns3_enet_coalesce));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-02 20:45:21 +08:00
|
|
|
static int hns3_reset_notify_down_enet(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
struct hnae3_knic_private_info *kinfo = &handle->kinfo;
|
|
|
|
struct net_device *ndev = kinfo->netdev;
|
2018-11-07 12:06:12 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(ndev);
|
|
|
|
|
|
|
|
if (test_and_set_bit(HNS3_NIC_STATE_RESETTING, &priv->state))
|
|
|
|
return 0;
|
2017-11-02 20:45:21 +08:00
|
|
|
|
|
|
|
if (!netif_running(ndev))
|
2018-07-16 23:36:23 +08:00
|
|
|
return 0;
|
2017-11-02 20:45:21 +08:00
|
|
|
|
|
|
|
return hns3_nic_net_stop(ndev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_reset_notify_up_enet(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
struct hnae3_knic_private_info *kinfo = &handle->kinfo;
|
2018-11-07 12:06:12 +08:00
|
|
|
struct hns3_nic_priv *priv = netdev_priv(kinfo->netdev);
|
2017-11-02 20:45:21 +08:00
|
|
|
int ret = 0;
|
|
|
|
|
2021-04-29 16:34:52 +08:00
|
|
|
if (!test_bit(HNS3_NIC_STATE_INITED, &priv->state)) {
|
|
|
|
netdev_err(kinfo->netdev, "device is not initialized yet\n");
|
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
|
2018-12-31 10:58:29 +08:00
|
|
|
clear_bit(HNS3_NIC_STATE_RESETTING, &priv->state);
|
|
|
|
|
2017-11-02 20:45:21 +08:00
|
|
|
if (netif_running(kinfo->netdev)) {
|
2018-12-31 10:58:29 +08:00
|
|
|
ret = hns3_nic_net_open(kinfo->netdev);
|
2017-11-02 20:45:21 +08:00
|
|
|
if (ret) {
|
2018-12-31 10:58:29 +08:00
|
|
|
set_bit(HNS3_NIC_STATE_RESETTING, &priv->state);
|
2017-11-02 20:45:21 +08:00
|
|
|
netdev_err(kinfo->netdev,
|
2019-06-07 10:03:13 +08:00
|
|
|
"net up fail, ret=%d!\n", ret);
|
2017-11-02 20:45:21 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_reset_notify_init_enet(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = handle->kinfo.netdev;
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Carrier off reporting is important to ethtool even BEFORE open */
|
|
|
|
netif_carrier_off(netdev);
|
|
|
|
|
2019-01-18 16:13:03 +08:00
|
|
|
ret = hns3_get_ring_config(priv);
|
2018-11-09 22:07:55 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2019-01-18 16:13:03 +08:00
|
|
|
ret = hns3_nic_alloc_vector_data(priv);
|
|
|
|
if (ret)
|
|
|
|
goto err_put_ring;
|
|
|
|
|
2018-09-27 02:28:40 +08:00
|
|
|
hns3_restore_coal(priv);
|
|
|
|
|
2017-11-02 20:45:21 +08:00
|
|
|
ret = hns3_nic_init_vector_data(priv);
|
|
|
|
if (ret)
|
2018-11-09 22:07:55 +08:00
|
|
|
goto err_dealloc_vector;
|
2017-11-02 20:45:21 +08:00
|
|
|
|
|
|
|
ret = hns3_init_all_ring(priv);
|
2018-11-09 22:07:55 +08:00
|
|
|
if (ret)
|
|
|
|
goto err_uninit_vector;
|
2017-11-02 20:45:21 +08:00
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
/* the device can work without cpu rmap, only aRFS needs it */
|
|
|
|
ret = hns3_set_rx_cpu_rmap(netdev);
|
|
|
|
if (ret)
|
|
|
|
dev_warn(priv->dev, "set rx cpu rmap fail, ret=%d\n", ret);
|
|
|
|
|
|
|
|
ret = hns3_nic_init_irq(priv);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(priv->dev, "init irq failed! ret=%d\n", ret);
|
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
goto err_init_irq_fail;
|
|
|
|
}
|
|
|
|
|
net: hns3: refactor the MAC address configure
Currently, the HNS3 driver sync and unsync MAC address in function
hns3_set_rx_mode(). For PF, it adds and deletes MAC address directly
in the path of dev_set_rx_mode(). If failed, it won't retry until
next calling of hns3_set_rx_mode(). On the other hand, if request
add and remove a same address many times at a short interval, each
request must be done one by one, can't be merged. For VF, it sends
mailbox messages to PF to request adding or deleting MAC address in
the path of function hns3_set_rx_mode(), no matter the address is
configured success.
This patch refines it by recording the MAC address in function
hns3_set_rx_mode(), and updating MAC address in the service task.
If failed, it will retry by the next calling of periodical service
task. It also uses some state to mark the state of each MAC address
in the MAC list, which can help merge configure request for a same
address. With these changes, when global reset or IMP reset occurs,
we can restore the MAC table with the MAC list.
Signed-off-by: Jian Shen <shenjian15@huawei.com>
Signed-off-by: Huazhong Tan <tanhuazhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2020-04-24 10:23:09 +08:00
|
|
|
if (!hns3_is_phys_func(handle->pdev))
|
|
|
|
hns3_init_mac_addr(netdev);
|
|
|
|
|
2019-04-06 15:43:26 +08:00
|
|
|
ret = hns3_client_start(handle);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(priv->dev, "hns3_client_start fail! ret=%d\n", ret);
|
2020-01-04 10:49:25 +08:00
|
|
|
goto err_client_start_fail;
|
2019-04-06 15:43:26 +08:00
|
|
|
}
|
|
|
|
|
2018-11-07 12:06:07 +08:00
|
|
|
set_bit(HNS3_NIC_STATE_INITED, &priv->state);
|
|
|
|
|
2018-11-09 22:07:55 +08:00
|
|
|
return ret;
|
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
err_client_start_fail:
|
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
hns3_nic_uninit_irq(priv);
|
|
|
|
err_init_irq_fail:
|
2019-04-06 15:43:26 +08:00
|
|
|
hns3_uninit_all_ring(priv);
|
2018-11-09 22:07:55 +08:00
|
|
|
err_uninit_vector:
|
|
|
|
hns3_nic_uninit_vector_data(priv);
|
|
|
|
err_dealloc_vector:
|
|
|
|
hns3_nic_dealloc_vector_data(priv);
|
2019-01-18 16:13:03 +08:00
|
|
|
err_put_ring:
|
|
|
|
hns3_put_ring_config(priv);
|
2018-11-09 22:07:55 +08:00
|
|
|
|
2017-11-02 20:45:21 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_reset_notify_uninit_enet(struct hnae3_handle *handle)
|
|
|
|
{
|
|
|
|
struct net_device *netdev = handle->kinfo.netdev;
|
|
|
|
struct hns3_nic_priv *priv = netdev_priv(netdev);
|
|
|
|
|
2019-04-04 16:17:58 +08:00
|
|
|
if (!test_and_clear_bit(HNS3_NIC_STATE_INITED, &priv->state)) {
|
2018-11-07 12:06:07 +08:00
|
|
|
netdev_warn(netdev, "already uninitialized\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
hns3_free_rx_cpu_rmap(netdev);
|
|
|
|
hns3_nic_uninit_irq(priv);
|
2019-06-28 19:50:07 +08:00
|
|
|
hns3_clear_all_ring(handle, true);
|
|
|
|
hns3_reset_tx_queue(priv->ae_handle);
|
2017-11-02 20:45:21 +08:00
|
|
|
|
2019-01-31 04:55:47 +08:00
|
|
|
hns3_nic_uninit_vector_data(priv);
|
2017-11-02 20:45:21 +08:00
|
|
|
|
2018-09-27 02:28:40 +08:00
|
|
|
hns3_store_coal(priv);
|
|
|
|
|
2020-01-04 10:49:25 +08:00
|
|
|
hns3_nic_dealloc_vector_data(priv);
|
2018-11-09 22:07:55 +08:00
|
|
|
|
2021-02-09 10:41:58 +08:00
|
|
|
hns3_uninit_all_ring(priv);
|
2017-11-02 20:45:21 +08:00
|
|
|
|
2019-01-18 16:13:03 +08:00
|
|
|
hns3_put_ring_config(priv);
|
|
|
|
|
2021-02-11 03:28:41 +08:00
|
|
|
return 0;
|
2017-11-02 20:45:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int hns3_reset_notify(struct hnae3_handle *handle,
|
|
|
|
enum hnae3_reset_notify_type type)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case HNAE3_UP_CLIENT:
|
2018-01-19 23:20:53 +08:00
|
|
|
ret = hns3_reset_notify_up_enet(handle);
|
|
|
|
break;
|
2017-11-02 20:45:21 +08:00
|
|
|
case HNAE3_DOWN_CLIENT:
|
|
|
|
ret = hns3_reset_notify_down_enet(handle);
|
|
|
|
break;
|
|
|
|
case HNAE3_INIT_CLIENT:
|
|
|
|
ret = hns3_reset_notify_init_enet(handle);
|
|
|
|
break;
|
|
|
|
case HNAE3_UNINIT_CLIENT:
|
|
|
|
ret = hns3_reset_notify_uninit_enet(handle);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-09-11 10:40:34 +08:00
|
|
|
static int hns3_change_channels(struct hnae3_handle *handle, u32 new_tqp_num,
|
|
|
|
bool rxfh_configured)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = handle->ae_algo->ops->set_channels(handle, new_tqp_num,
|
|
|
|
rxfh_configured);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(&handle->pdev->dev,
|
|
|
|
"Change tqp num(%u) fail.\n", new_tqp_num);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = hns3_reset_notify(handle, HNAE3_INIT_CLIENT);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = hns3_reset_notify(handle, HNAE3_UP_CLIENT);
|
|
|
|
if (ret)
|
|
|
|
hns3_reset_notify(handle, HNAE3_UNINIT_CLIENT);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-12-22 12:21:41 +08:00
|
|
|
int hns3_set_channels(struct net_device *netdev,
|
|
|
|
struct ethtool_channels *ch)
|
|
|
|
{
|
|
|
|
struct hnae3_handle *h = hns3_get_handle(netdev);
|
|
|
|
struct hnae3_knic_private_info *kinfo = &h->kinfo;
|
2019-01-23 07:39:30 +08:00
|
|
|
bool rxfh_configured = netif_is_rxfh_configured(netdev);
|
2017-12-22 12:21:41 +08:00
|
|
|
u32 new_tqp_num = ch->combined_count;
|
|
|
|
u16 org_tqp_num;
|
|
|
|
int ret;
|
|
|
|
|
2019-07-29 10:53:22 +08:00
|
|
|
if (hns3_nic_resetting(netdev))
|
|
|
|
return -EBUSY;
|
|
|
|
|
2017-12-22 12:21:41 +08:00
|
|
|
if (ch->rx_count || ch->tx_count)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2020-12-10 11:42:07 +08:00
|
|
|
if (kinfo->tc_info.mqprio_active) {
|
|
|
|
dev_err(&netdev->dev,
|
|
|
|
"it's not allowed to set channels via ethtool when MQPRIO mode is on\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-03-08 19:41:54 +08:00
|
|
|
if (new_tqp_num > hns3_get_max_available_channels(h) ||
|
2019-01-18 16:13:08 +08:00
|
|
|
new_tqp_num < 1) {
|
2017-12-22 12:21:41 +08:00
|
|
|
dev_err(&netdev->dev,
|
2019-10-31 19:23:23 +08:00
|
|
|
"Change tqps fail, the tqp range is from 1 to %u",
|
2018-03-08 19:41:54 +08:00
|
|
|
hns3_get_max_available_channels(h));
|
2017-12-22 12:21:41 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-01-18 16:13:08 +08:00
|
|
|
if (kinfo->rss_size == new_tqp_num)
|
2017-12-22 12:21:41 +08:00
|
|
|
return 0;
|
|
|
|
|
2019-07-29 10:53:27 +08:00
|
|
|
netif_dbg(h, drv, netdev,
|
|
|
|
"set channels: tqp_num=%u, rxfh=%d\n",
|
|
|
|
new_tqp_num, rxfh_configured);
|
|
|
|
|
2019-01-18 16:13:06 +08:00
|
|
|
ret = hns3_reset_notify(h, HNAE3_DOWN_CLIENT);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2018-03-09 10:37:02 +08:00
|
|
|
|
2019-01-18 16:13:06 +08:00
|
|
|
ret = hns3_reset_notify(h, HNAE3_UNINIT_CLIENT);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2017-12-22 12:21:41 +08:00
|
|
|
|
|
|
|
org_tqp_num = h->kinfo.num_tqps;
|
2019-09-11 10:40:34 +08:00
|
|
|
ret = hns3_change_channels(h, new_tqp_num, rxfh_configured);
|
2017-12-22 12:21:41 +08:00
|
|
|
if (ret) {
|
2019-09-11 10:40:34 +08:00
|
|
|
int ret1;
|
|
|
|
|
|
|
|
netdev_warn(netdev,
|
|
|
|
"Change channels fail, revert to old value\n");
|
|
|
|
ret1 = hns3_change_channels(h, org_tqp_num, rxfh_configured);
|
|
|
|
if (ret1) {
|
|
|
|
netdev_err(netdev,
|
|
|
|
"revert to old channel fail\n");
|
|
|
|
return ret1;
|
2017-12-22 12:21:41 +08:00
|
|
|
}
|
2019-09-11 10:40:34 +08:00
|
|
|
|
2019-01-18 16:13:06 +08:00
|
|
|
return ret;
|
2019-09-11 10:40:34 +08:00
|
|
|
}
|
2017-12-22 12:21:41 +08:00
|
|
|
|
2019-09-11 10:40:34 +08:00
|
|
|
return 0;
|
2017-12-22 12:21:41 +08:00
|
|
|
}
|
|
|
|
|
2019-08-28 22:23:13 +08:00
|
|
|
static const struct hns3_hw_error_info hns3_hw_err[] = {
|
|
|
|
{ .type = HNAE3_PPU_POISON_ERROR,
|
|
|
|
.msg = "PPU poison" },
|
|
|
|
{ .type = HNAE3_CMDQ_ECC_ERROR,
|
|
|
|
.msg = "IMP CMDQ error" },
|
|
|
|
{ .type = HNAE3_IMP_RD_POISON_ERROR,
|
|
|
|
.msg = "IMP RD poison" },
|
2020-09-25 08:26:15 +08:00
|
|
|
{ .type = HNAE3_ROCEE_AXI_RESP_ERROR,
|
|
|
|
.msg = "ROCEE AXI RESP error" },
|
2019-08-28 22:23:13 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static void hns3_process_hw_error(struct hnae3_handle *handle,
|
|
|
|
enum hnae3_hw_error_type type)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(hns3_hw_err); i++) {
|
|
|
|
if (hns3_hw_err[i].type == type) {
|
|
|
|
dev_err(&handle->pdev->dev, "Detected %s!\n",
|
|
|
|
hns3_hw_err[i].msg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-09 15:44:01 +08:00
|
|
|
static const struct hnae3_client_ops client_ops = {
|
2017-08-02 23:59:45 +08:00
|
|
|
.init_instance = hns3_client_init,
|
|
|
|
.uninit_instance = hns3_client_uninit,
|
|
|
|
.link_status_change = hns3_link_status_change,
|
2017-11-02 20:45:21 +08:00
|
|
|
.reset_notify = hns3_reset_notify,
|
2019-08-28 22:23:13 +08:00
|
|
|
.process_hw_error = hns3_process_hw_error,
|
2017-08-02 23:59:45 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* hns3_init_module - Driver registration routine
|
|
|
|
* hns3_init_module is the first routine called when the driver is
|
|
|
|
* loaded. All it does is register with the PCI subsystem.
|
|
|
|
*/
|
|
|
|
static int __init hns3_init_module(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
pr_info("%s: %s - version\n", hns3_driver_name, hns3_driver_string);
|
|
|
|
pr_info("%s: %s\n", hns3_driver_name, hns3_copyright);
|
|
|
|
|
|
|
|
client.type = HNAE3_CLIENT_KNIC;
|
2020-01-21 16:42:09 +08:00
|
|
|
snprintf(client.name, HNAE3_CLIENT_NAME_LENGTH, "%s",
|
2017-08-02 23:59:45 +08:00
|
|
|
hns3_driver_name);
|
|
|
|
|
|
|
|
client.ops = &client_ops;
|
|
|
|
|
2018-05-19 23:53:18 +08:00
|
|
|
INIT_LIST_HEAD(&client.node);
|
|
|
|
|
2018-11-22 22:09:41 +08:00
|
|
|
hns3_dbg_register_debugfs(hns3_driver_name);
|
|
|
|
|
2017-08-02 23:59:45 +08:00
|
|
|
ret = hnae3_register_client(&client);
|
|
|
|
if (ret)
|
2018-11-22 22:09:41 +08:00
|
|
|
goto err_reg_client;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
ret = pci_register_driver(&hns3_driver);
|
|
|
|
if (ret)
|
2018-11-22 22:09:41 +08:00
|
|
|
goto err_reg_driver;
|
2017-08-02 23:59:45 +08:00
|
|
|
|
|
|
|
return ret;
|
2018-11-22 22:09:41 +08:00
|
|
|
|
|
|
|
err_reg_driver:
|
|
|
|
hnae3_unregister_client(&client);
|
|
|
|
err_reg_client:
|
|
|
|
hns3_dbg_unregister_debugfs();
|
|
|
|
return ret;
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
module_init(hns3_init_module);
|
|
|
|
|
|
|
|
/* hns3_exit_module - Driver exit cleanup routine
|
|
|
|
* hns3_exit_module is called just before the driver is removed
|
|
|
|
* from memory.
|
|
|
|
*/
|
|
|
|
static void __exit hns3_exit_module(void)
|
|
|
|
{
|
|
|
|
pci_unregister_driver(&hns3_driver);
|
|
|
|
hnae3_unregister_client(&client);
|
2018-11-22 22:09:41 +08:00
|
|
|
hns3_dbg_unregister_debugfs();
|
2017-08-02 23:59:45 +08:00
|
|
|
}
|
|
|
|
module_exit(hns3_exit_module);
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION("HNS3: Hisilicon Ethernet Driver");
|
|
|
|
MODULE_AUTHOR("Huawei Tech. Co., Ltd.");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_ALIAS("pci:hns-nic");
|