lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Date:   Mon, 3 Jun 2019 16:05:48 -0700
From:   Jesse Brandeburg <jesse.brandeburg@...el.com>
To:     Xue Chaojing <xuechaojing@...wei.com>
Cc:     <davem@...emloft.net>, <linux-kernel@...r.kernel.org>,
        <netdev@...r.kernel.org>, <luoshaokai@...wei.com>,
        <cloud.wangxiaoyun@...wei.com>, <chiqijun@...wei.com>,
        <wulike1@...wei.com>, jesse.brandeburg@...el.com
Subject: Re: [PATCH net-next v3] hinic: add LRO support

some review comments below...

On Mon, 3 Jun 2019 04:35:36 +0000
Xue Chaojing <xuechaojing@...wei.com> wrote:

> This patch adds LRO support for the HiNIC driver.
> 
> Signed-off-by: Xue Chaojing <xuechaojing@...wei.com>
> ---
>  .../net/ethernet/huawei/hinic/hinic_hw_dev.c  |   2 +
>  .../net/ethernet/huawei/hinic/hinic_hw_dev.h  |   8 +-
>  .../net/ethernet/huawei/hinic/hinic_hw_io.c   |  58 +++++++++
>  .../ethernet/huawei/hinic/hinic_hw_qp_ctxt.h  |   5 +
>  .../net/ethernet/huawei/hinic/hinic_hw_wqe.h  |  22 +++-
>  .../net/ethernet/huawei/hinic/hinic_main.c    |  51 +++++++-
>  .../net/ethernet/huawei/hinic/hinic_port.c    | 114 ++++++++++++++++++
>  .../net/ethernet/huawei/hinic/hinic_port.h    |  45 +++++++
>  drivers/net/ethernet/huawei/hinic/hinic_rx.c  |  46 +++++--
>  drivers/net/ethernet/huawei/hinic/hinic_rx.h  |   2 +
>  10 files changed, 340 insertions(+), 13 deletions(-)
> 
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.c b/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.c
> index 3875f39f43bb..756a7e3280bd 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.c
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.c
> @@ -313,6 +313,8 @@ static int set_hw_ioctxt(struct hinic_hwdev *hwdev, unsigned int rq_depth,
>  	hw_ioctxt.set_cmdq_depth = HW_IOCTXT_SET_CMDQ_DEPTH_DEFAULT;
>  	hw_ioctxt.cmdq_depth = 0;
>  
> +	hw_ioctxt.lro_en = 1;
> +
>  	hw_ioctxt.rq_depth  = ilog2(rq_depth);
>  
>  	hw_ioctxt.rx_buf_sz_idx = HINIC_RX_BUF_SZ_IDX;
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.h b/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.h
> index c9e621e19dd0..fba4fe82472a 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.h
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_hw_dev.h
> @@ -50,6 +50,8 @@ enum hinic_port_cmd {
>  
>  	HINIC_PORT_CMD_GET_LINK_STATE   = 24,
>  
> +	HINIC_PORT_CMD_SET_LRO		= 25,
> +
>  	HINIC_PORT_CMD_SET_RX_CSUM	= 26,
>  
>  	HINIC_PORT_CMD_SET_PORT_STATE   = 41,
> @@ -62,7 +64,11 @@ enum hinic_port_cmd {
>  
>  	HINIC_PORT_CMD_SET_TSO          = 112,
>  
> +	HINIC_PORT_CMD_SET_RQ_IQ_MAP	= 115,
> +
>  	HINIC_PORT_CMD_GET_CAP          = 170,
> +
> +	HINIC_PORT_CMD_SET_LRO_TIMER	= 244,
>  };
>  
>  enum hinic_mgmt_msg_cmd {
> @@ -106,7 +112,7 @@ struct hinic_cmd_hw_ioctxt {
>  	u8      set_cmdq_depth;
>  	u8      cmdq_depth;
>  
> -	u8      rsvd2;
> +	u8      lro_en;
>  	u8      rsvd3;
>  	u8      rsvd4;
>  	u8      rsvd5;
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_hw_io.c b/drivers/net/ethernet/huawei/hinic/hinic_hw_io.c
> index a322a22d9357..1169526323cf 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_hw_io.c
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_hw_io.c
> @@ -45,6 +45,7 @@
>  
>  enum io_cmd {
>  	IO_CMD_MODIFY_QUEUE_CTXT = 0,
> +	IO_CMD_CLEAN_QUEUE_CTXT,
>  };
>  
>  static void init_db_area_idx(struct hinic_free_db_area *free_db_area)
> @@ -210,6 +211,57 @@ static int write_qp_ctxts(struct hinic_func_to_io *func_to_io, u16 base_qpn,
>  		write_rq_ctxts(func_to_io, base_qpn, num_qps));
>  }
>  
> +static int clean_queue_offload_ctxt(struct hinic_func_to_io *func_to_io,
> +				    enum hinic_qp_ctxt_type ctxt_type)

all the other functions you add have hinic_ in front of them, how come
not this one?

> +{
> +	struct hinic_hwif *hwif = func_to_io->hwif;
> +	struct hinic_clean_queue_ctxt *ctxt_block;
> +	struct pci_dev *pdev = hwif->pdev;
> +	struct hinic_cmdq_buf cmdq_buf;
> +	u64 out_param = 0;
> +	int err;
> +
> +	err = hinic_alloc_cmdq_buf(&func_to_io->cmdqs, &cmdq_buf);
> +	if (err) {
> +		dev_err(&pdev->dev, "Failed to allocate cmdq buf\n");
> +		return err;
> +	}
> +
> +	ctxt_block = cmdq_buf.buf;
> +	ctxt_block->cmdq_hdr.num_queues = func_to_io->max_qps;
> +	ctxt_block->cmdq_hdr.queue_type = ctxt_type;
> +	ctxt_block->cmdq_hdr.addr_offset = 0;
> +
> +	/* TSO/LRO ctxt size: 0x0:0B; 0x1:160B; 0x2:200B; 0x3:240B */
> +	ctxt_block->ctxt_size = 0x3;
> +
> +	hinic_cpu_to_be32(ctxt_block, sizeof(*ctxt_block));
> +
> +	cmdq_buf.size = sizeof(*ctxt_block);
> +
> +	err = hinic_cmdq_direct_resp(&func_to_io->cmdqs, HINIC_MOD_L2NIC,
> +				     IO_CMD_CLEAN_QUEUE_CTXT,
> +				     &cmdq_buf, &out_param);
> +
> +	if (err || out_param) {
> +		dev_err(&pdev->dev, "Failed to clean queue offload ctxts, err: %d, out_param: 0x%llx\n",
> +			err, out_param);
> +
> +		err = -EFAULT;
> +	}
> +
> +	hinic_free_cmdq_buf(&func_to_io->cmdqs, &cmdq_buf);
> +
> +	return err;
> +}
> +
> +static int clean_qp_offload_ctxt(struct hinic_func_to_io *func_to_io)
> +{
> +	/* clean LRO/TSO context space */
> +	return (clean_queue_offload_ctxt(func_to_io, HINIC_QP_CTXT_TYPE_SQ) ||
> +		clean_queue_offload_ctxt(func_to_io, HINIC_QP_CTXT_TYPE_RQ));
> +}
> +
>  /**
>   * init_qp - Initialize a Queue Pair
>   * @func_to_io: func to io channel that holds the IO components
> @@ -381,6 +433,12 @@ int hinic_io_create_qps(struct hinic_func_to_io *func_to_io,
>  		goto err_write_qp_ctxts;
>  	}
>  
> +	err = clean_qp_offload_ctxt(func_to_io);
> +	if (err) {
> +		dev_err(&pdev->dev, "Failed to clean qp offload ctxts\n");

What is the user supposed to do when they see this message? Please
consider that your audience is users who are not developers.  for
messages that are only useful for developers, you're better off using
dev_dbg(...)

> +		goto err_write_qp_ctxts;
> +	}
> +
>  	return 0;
>  
>  err_write_qp_ctxts:
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_hw_qp_ctxt.h b/drivers/net/ethernet/huawei/hinic/hinic_hw_qp_ctxt.h
> index 376abf00762b..01c41dd705cb 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_hw_qp_ctxt.h
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_hw_qp_ctxt.h
> @@ -201,6 +201,11 @@ struct hinic_rq_ctxt {
>  	u32     wq_block_lo_pfn;
>  };
>  
> +struct hinic_clean_queue_ctxt {
> +	struct hinic_qp_ctxt_header	cmdq_hdr;
> +	u32				ctxt_size;
> +};
> +
>  struct hinic_sq_ctxt_block {
>  	struct hinic_qp_ctxt_header hdr;
>  	struct hinic_sq_ctxt sq_ctxt[HINIC_Q_CTXT_MAX];
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_hw_wqe.h b/drivers/net/ethernet/huawei/hinic/hinic_hw_wqe.h
> index 138941527872..ef852b7b57a3 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_hw_wqe.h
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_hw_wqe.h
> @@ -219,6 +219,26 @@
>  #define HINIC_MSS_DEFAULT		        0x3E00
>  #define HINIC_MSS_MIN		                0x50
>  
> +#define RQ_CQE_STATUS_NUM_LRO_SHIFT		16
> +#define RQ_CQE_STATUS_NUM_LRO_MASK		0xFFU
> +
> +#define RQ_CQE_STATUS_GET(val, member)		(((val) >> \
> +			RQ_CQE_STATUS_##member##_SHIFT) & \
> +			RQ_CQE_STATUS_##member##_MASK)
> +
> +#define HINIC_GET_RX_NUM_LRO(status)	\
> +		RQ_CQE_STATUS_GET(status, NUM_LRO)
> +
> +#define RQ_CQE_OFFOLAD_TYPE_PKT_TYPE_SHIFT		0
> +#define RQ_CQE_OFFOLAD_TYPE_PKT_TYPE_MASK		0xFFFU
> +
> +#define RQ_CQE_OFFOLAD_TYPE_GET(val, member)		(((val) >> \
> +				RQ_CQE_OFFOLAD_TYPE_##member##_SHIFT) & \
> +				RQ_CQE_OFFOLAD_TYPE_##member##_MASK)
> +
> +#define HINIC_GET_RX_PKT_TYPE(offload_type)	\
> +			RQ_CQE_OFFOLAD_TYPE_GET(offload_type, PKT_TYPE)
> +
>  enum hinic_l4offload_type {
>  	HINIC_L4_OFF_DISABLE            = 0,
>  	HINIC_TCP_OFFLOAD_ENABLE        = 1,
> @@ -372,7 +392,7 @@ struct hinic_rq_cqe {
>  	u32     status;
>  	u32     len;
>  
> -	u32     rsvd2;
> +	u32     offload_type;
>  	u32     rsvd3;
>  	u32     rsvd4;
>  	u32     rsvd5;
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_main.c b/drivers/net/ethernet/huawei/hinic/hinic_main.c
> index cfd3f4232cac..175fe53daa9f 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_main.c
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_main.c
> @@ -372,6 +372,35 @@ static void free_rxqs(struct hinic_dev *nic_dev)
>  	nic_dev->rxqs = NULL;
>  }
>  
> +int hinic_rx_configure(struct net_device *netdev)
> +{
> +	struct hinic_dev *nic_dev = netdev_priv(netdev);
> +	int err;
> +
> +	err = hinic_set_max_qnum(nic_dev, nic_dev->hwdev->nic_cap.max_qps);
> +	if (err) {
> +		netif_err(nic_dev, drv, nic_dev->netdev,
> +			  "Failed - hinic rx configure\n");
> +		return err;
> +	}
> +
> +	return 0;
> +}
> +
> +static int hinic_configure(struct hinic_dev *nic_dev)
> +{
> +	struct net_device *netdev = nic_dev->netdev;
> +	int err;
> +
> +	err = hinic_rx_configure(netdev);
> +	if (err) {
> +		netif_err(nic_dev, drv, netdev, "Failed to configure rx\n");
> +		return err;
> +	}
> +
> +	return 0;
> +}
> +
>  static int hinic_open(struct net_device *netdev)
>  {
>  	struct hinic_dev *nic_dev = netdev_priv(netdev);
> @@ -401,6 +430,13 @@ static int hinic_open(struct net_device *netdev)
>  		goto err_create_rxqs;
>  	}
>  
> +	err = hinic_configure(nic_dev);
> +	if (err) {
> +		netif_err(nic_dev, drv, netdev,
> +			  "Failed to hinic configure\n");

what was the point of the message here? at the very least should be
"Failed hinic_configure, err %d"

> +		goto err_port_state;
> +	}
> +
>  	num_qps = hinic_hwdev_num_qps(nic_dev->hwdev);
>  	netif_set_real_num_tx_queues(netdev, num_qps);
>  	netif_set_real_num_rx_queues(netdev, num_qps);
> @@ -805,7 +841,7 @@ static void netdev_features_init(struct net_device *netdev)
>  {
>  	netdev->hw_features = NETIF_F_SG | NETIF_F_HIGHDMA | NETIF_F_IP_CSUM |
>  			      NETIF_F_IPV6_CSUM | NETIF_F_TSO | NETIF_F_TSO6 |
> -			      NETIF_F_RXCSUM;
> +			      NETIF_F_RXCSUM | NETIF_F_LRO;
>  
>  	netdev->vlan_features = netdev->hw_features;
>  
> @@ -869,6 +905,8 @@ static int set_features(struct hinic_dev *nic_dev,
>  {
>  	netdev_features_t changed = force_change ? ~0 : pre_features ^ features;
>  	u32 csum_en = HINIC_RX_CSUM_OFFLOAD_EN;
> +	u32 lro_buf_size = 0;
> +	u32 lro_timer = 0;
>  	int err = 0;
>  
>  	if (changed & NETIF_F_TSO)
> @@ -878,6 +916,17 @@ static int set_features(struct hinic_dev *nic_dev,
>  	if (changed & NETIF_F_RXCSUM)
>  		err = hinic_set_rx_csum_offload(nic_dev, csum_en);
>  
> +	if (changed & NETIF_F_LRO) {
> +		lro_timer = HINIC_LRO_RX_TIMER_DEFAULT;
> +		lro_buf_size = HINIC_LRO_MAX_WQE_NUM_DEFAULT * HINIC_RX_BUF_SZ;

was there some reason not to just declare as a local? cppcheck would
have caught this.

> +
> +		err = hinic_set_rx_lro_state(nic_dev,
> +					     !!(features & NETIF_F_LRO),
> +					     lro_timer,
> +					     lro_buf_size /
> +					     HINIC_RX_BUF_SZ);

unwrap the previous line.

I also think you're missing some checks to make sure that if RX_CSUM is
disabled that LRO turns off.

Also, does your hardware not support the constraints of NETIF_F_GRO_HW?


> +	}
> +
>  	return err;
>  }
>  
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_port.c b/drivers/net/ethernet/huawei/hinic/hinic_port.c
> index 122c93597268..c1947b2f4462 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_port.c
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_port.c
> @@ -439,3 +439,117 @@ int hinic_set_rx_csum_offload(struct hinic_dev *nic_dev, u32 en)
>  
>  	return 0;
>  }
> +
> +int hinic_set_max_qnum(struct hinic_dev *nic_dev, u8 num_rqs)
> +{
> +	struct hinic_hwdev *hwdev = nic_dev->hwdev;
> +	struct hinic_hwif *hwif = hwdev->hwif;
> +	struct pci_dev *pdev = hwif->pdev;
> +	struct hinic_rq_num rq_num = {0};
> +	u16 out_size = sizeof(rq_num);
> +	int err;
> +
> +	rq_num.func_id = HINIC_HWIF_FUNC_IDX(hwif);
> +	rq_num.num_rqs = num_rqs;
> +	rq_num.rq_depth = ilog2(HINIC_SQ_DEPTH);
> +
> +	err = hinic_port_msg_cmd(hwdev, HINIC_PORT_CMD_SET_RQ_IQ_MAP,
> +				 &rq_num, sizeof(rq_num),
> +				 &rq_num, &out_size);
> +	if (err || !out_size || rq_num.status) {
> +		dev_err(&pdev->dev,
> +			"Failed to rxq number, ret = %d\n",
> +			rq_num.status);
> +		return -EINVAL;
> +	}
> +
> +	return 0;
> +}
> +
> +int hinic_set_rx_lro(struct hinic_dev *nic_dev, u8 ipv4_en, u8 ipv6_en,
> +		     u8 max_wqe_num)
> +{
> +	struct hinic_hwdev *hwdev = nic_dev->hwdev;
> +	struct hinic_hwif *hwif = hwdev->hwif;
> +	struct hinic_lro_config lro_cfg = {0};

I think canonical usage is { 0 };


> +	struct pci_dev *pdev = hwif->pdev;
> +	u16 out_size = sizeof(lro_cfg);
> +	int err;
> +
> +	lro_cfg.func_id = HINIC_HWIF_FUNC_IDX(hwif);
> +	lro_cfg.lro_ipv4_en = ipv4_en;
> +	lro_cfg.lro_ipv6_en = ipv6_en;
> +	lro_cfg.lro_max_wqe_num = max_wqe_num;
> +
> +	err = hinic_port_msg_cmd(hwdev, HINIC_PORT_CMD_SET_LRO,
> +				 &lro_cfg, sizeof(lro_cfg),
> +				 &lro_cfg, &out_size);
> +	if (err || !out_size || lro_cfg.status) {
> +		dev_err(&pdev->dev,
> +			"Failed to set lro offload, ret = %d\n",
> +			lro_cfg.status);
> +		return -EINVAL;
> +	}
> +
> +	return 0;
> +}
> +
> +int hinic_set_rx_lro_timer(struct hinic_dev *nic_dev, u32 timer_value)
> +{
> +	struct hinic_hwdev *hwdev = nic_dev->hwdev;
> +	struct hinic_lro_timer lro_timer = {0};

same as above

> +	struct hinic_hwif *hwif = hwdev->hwif;
> +	struct pci_dev *pdev = hwif->pdev;
> +	u16 out_size = sizeof(lro_timer);
> +	int err;
> +
> +	lro_timer.status = 0;
> +	lro_timer.type = 0;
> +	lro_timer.enable = 1;
> +	lro_timer.timer = timer_value;
> +
> +	err = hinic_port_msg_cmd(hwdev, HINIC_PORT_CMD_SET_LRO_TIMER,
> +				 &lro_timer, sizeof(lro_timer),
> +				 &lro_timer, &out_size);
> +	if (lro_timer.status == 0xFF) {
> +		/* For this case, we think status (0xFF) is OK */
> +		lro_timer.status = 0;
> +		dev_err(&pdev->dev,
> +			"Set lro timer not supported by the current FW version, it will be 1ms default\n");

If it's ok, why print a message? also, maybe dev_dbg?

> +	}
> +
> +	if (err || !out_size || lro_timer.status) {
> +		dev_err(&pdev->dev,
> +			"Failed to set lro timer, ret = %d\n",
> +			lro_timer.status);
> +
> +		return -EINVAL;
> +	}
> +
> +	return 0;
> +}
> +
> +int hinic_set_rx_lro_state(struct hinic_dev *nic_dev, u8 lro_en,
> +			   u32 lro_timer, u32 wqe_num)
> +{
> +	struct hinic_hwdev *hwdev = nic_dev->hwdev;
> +	u8 ipv4_en = 0;
> +	u8 ipv6_en = 0;

no need to initialize with the way you use below, in fact compiler will
warn of overwritten value if you turn on the right options.

> +	int err;
> +
> +	if (!hwdev)
> +		return -EINVAL;
> +
> +	ipv4_en = lro_en ? 1 : 0;
> +	ipv6_en = lro_en ? 1 : 0;
> +
> +	err = hinic_set_rx_lro(nic_dev, ipv4_en, ipv6_en, (u8)wqe_num);
> +	if (err)
> +		return err;
> +
> +	err = hinic_set_rx_lro_timer(nic_dev, lro_timer);
> +	if (err)
> +		return err;
> +
> +	return 0;
> +}
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_port.h b/drivers/net/ethernet/huawei/hinic/hinic_port.h
> index 02d896eed455..94e3f9bbd688 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_port.h
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_port.h
> @@ -22,6 +22,10 @@
>  
>  #include "hinic_dev.h"
>  
> +#define HINIC_LRO_MAX_WQE_NUM_DEFAULT     8
> +
> +#define HINIC_LRO_RX_TIMER_DEFAULT	16
> +
>  enum hinic_rx_mode {
>  	HINIC_RX_MODE_UC        = BIT(0),
>  	HINIC_RX_MODE_MC        = BIT(1),
> @@ -192,6 +196,42 @@ struct hinic_checksum_offload {
>  	u16	rsvd1;
>  	u32	rx_csum_offload;
>  };
> +
> +struct hinic_rq_num {
> +	u8	status;
> +	u8	version;
> +	u8	rsvd0[6];
> +
> +	u16	func_id;
> +	u16	rsvd1[33];
> +	u32	num_rqs;
> +	u32	rq_depth;
> +};
> +
> +struct hinic_lro_config {
> +	u8	status;
> +	u8	version;
> +	u8	rsvd0[6];
> +
> +	u16	func_id;
> +	u16	rsvd1;
> +	u8	lro_ipv4_en;
> +	u8	lro_ipv6_en;
> +	u8	lro_max_wqe_num;
> +	u8	resv2[13];
> +};
> +
> +struct hinic_lro_timer {
> +	u8	status;
> +	u8	version;
> +	u8	rsvd0[6];
> +
> +	u8	type;   /* 0: set timer value, 1: get timer value */
> +	u8	enable; /* when set lro time, enable should be 1 */
> +	u16	rsvd1;
> +	u32	timer;
> +};
> +
>  int hinic_port_add_mac(struct hinic_dev *nic_dev, const u8 *addr,
>  		       u16 vlan_id);
>  
> @@ -220,7 +260,12 @@ int hinic_port_set_func_state(struct hinic_dev *nic_dev,
>  int hinic_port_get_cap(struct hinic_dev *nic_dev,
>  		       struct hinic_port_cap *port_cap);
>  
> +int hinic_set_max_qnum(struct hinic_dev *nic_dev, u8 num_rqs);
> +
>  int hinic_port_set_tso(struct hinic_dev *nic_dev, enum hinic_tso_state state);
>  
>  int hinic_set_rx_csum_offload(struct hinic_dev *nic_dev, u32 en);
> +
> +int hinic_set_rx_lro_state(struct hinic_dev *nic_dev, u8 lro_en,
> +			   u32 lro_timer, u32 wqe_num);
>  #endif
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_rx.c b/drivers/net/ethernet/huawei/hinic/hinic_rx.c
> index b6d218768ec1..04c887d13848 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_rx.c
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_rx.c
> @@ -45,6 +45,15 @@
>  #define RX_IRQ_NO_RESEND_TIMER          0
>  #define HINIC_RX_BUFFER_WRITE           16
>  
> +#define HINIC_RX_IPV6_PKT		7
> +#define LRO_PKT_HDR_LEN_IPV4		66
> +#define LRO_PKT_HDR_LEN_IPV6		86
> +#define LRO_REPLENISH_THLD		256
> +
> +#define LRO_PKT_HDR_LEN(cqe)		\
> +	(HINIC_GET_RX_PKT_TYPE(be32_to_cpu((cqe)->offload_type)) == \
> +	 HINIC_RX_IPV6_PKT ? LRO_PKT_HDR_LEN_IPV6 : LRO_PKT_HDR_LEN_IPV4)
> +
>  /**
>   * hinic_rxq_clean_stats - Clean the statistics of specific queue
>   * @rxq: Logical Rx Queue
> @@ -90,18 +99,12 @@ static void rxq_stats_init(struct hinic_rxq *rxq)
>  	hinic_rxq_clean_stats(rxq);
>  }
>  
> -static void rx_csum(struct hinic_rxq *rxq, u16 cons_idx,
> +static void rx_csum(struct hinic_rxq *rxq, u32 status,
>  		    struct sk_buff *skb)
>  {
>  	struct net_device *netdev = rxq->netdev;
> -	struct hinic_rq_cqe *cqe;
> -	struct hinic_rq *rq;
>  	u32 csum_err;
> -	u32 status;
>  
> -	rq = rxq->rq;
> -	cqe = rq->cqe[cons_idx];
> -	status = be32_to_cpu(cqe->status);
>  	csum_err = HINIC_RQ_CQE_STATUS_GET(status, CSUM_ERR);
>  
>  	if (!(netdev->features & NETIF_F_RXCSUM))
> @@ -321,12 +324,16 @@ static int rxq_recv(struct hinic_rxq *rxq, int budget)
>  {
>  	struct hinic_qp *qp = container_of(rxq->rq, struct hinic_qp, rq);
>  	u64 pkt_len = 0, rx_bytes = 0;
> +	struct hinic_rq *rq = rxq->rq;
>  	struct hinic_rq_wqe *rq_wqe;
>  	unsigned int free_wqebbs;
> +	struct hinic_rq_cqe *cqe;
>  	int num_wqes, pkts = 0;
>  	struct hinic_sge sge;
> +	unsigned int status;
>  	struct sk_buff *skb;
> -	u16 ci;
> +	u16 ci, num_lro;
> +	u16 num_wqe = 0;
>  
>  	while (pkts < budget) {
>  		num_wqes = 0;
> @@ -336,11 +343,13 @@ static int rxq_recv(struct hinic_rxq *rxq, int budget)
>  		if (!rq_wqe)
>  			break;
>  
> +		cqe = rq->cqe[ci];
> +		status =  be32_to_cpu(cqe->status);
>  		hinic_rq_get_sge(rxq->rq, rq_wqe, ci, &sge);
>  
>  		rx_unmap_skb(rxq, hinic_sge_to_dma(&sge));
>  
> -		rx_csum(rxq, ci, skb);
> +		rx_csum(rxq, status, skb);
>  
>  		prefetch(skb->data);
>  
> @@ -354,7 +363,7 @@ static int rxq_recv(struct hinic_rxq *rxq, int budget)
>  						     HINIC_RX_BUF_SZ, ci);
>  		}
>  
> -		hinic_rq_put_wqe(rxq->rq, ci,
> +		hinic_rq_put_wqe(rq, ci,
>  				 (num_wqes + 1) * HINIC_RQ_WQE_SIZE);
>  
>  		skb_record_rx_queue(skb, qp->q_id);
> @@ -364,6 +373,21 @@ static int rxq_recv(struct hinic_rxq *rxq, int budget)
>  
>  		pkts++;
>  		rx_bytes += pkt_len;
> +
> +		num_lro = HINIC_GET_RX_NUM_LRO(status);
> +		if (num_lro) {
> +			rx_bytes += ((num_lro - 1) *
> +				     LRO_PKT_HDR_LEN(cqe));
> +
> +			num_wqe +=
> +			(u16)(pkt_len >> rxq->rx_buff_shift) +
> +			((pkt_len & (rxq->buf_len - 1)) ? 1 : 0);
> +		}
> +
> +		cqe->status = 0;
> +
> +		if (num_wqe >= LRO_REPLENISH_THLD)
> +			break;
>  	}
>  
>  	free_wqebbs = hinic_get_rq_free_wqebbs(rxq->rq);
> @@ -482,6 +506,8 @@ int hinic_init_rxq(struct hinic_rxq *rxq, struct hinic_rq *rq,
>  
>  	rxq->netdev = netdev;
>  	rxq->rq = rq;
> +	rxq->buf_len = HINIC_RX_BUF_SZ;
> +	rxq->rx_buff_shift = ilog2(HINIC_RX_BUF_SZ);
>  
>  	rxq_stats_init(rxq);
>  
> diff --git a/drivers/net/ethernet/huawei/hinic/hinic_rx.h b/drivers/net/ethernet/huawei/hinic/hinic_rx.h
> index f8ed3fa6c8ee..08e7d88382cd 100644
> --- a/drivers/net/ethernet/huawei/hinic/hinic_rx.h
> +++ b/drivers/net/ethernet/huawei/hinic/hinic_rx.h
> @@ -41,6 +41,8 @@ struct hinic_rxq {
>  	struct hinic_rxq_stats  rxq_stats;
>  
>  	char                    *irq_name;
> +	u16			buf_len;
> +	u32			rx_buff_shift;
>  
>  	struct napi_struct      napi;
>  };

Powered by blists - more mailing lists