diff options
Diffstat (limited to 'drivers/net/ethernet/cavium/thunder/nicvf_main.c')
| -rw-r--r-- | drivers/net/ethernet/cavium/thunder/nicvf_main.c | 230 | 
1 files changed, 129 insertions, 101 deletions
| diff --git a/drivers/net/ethernet/cavium/thunder/nicvf_main.c b/drivers/net/ethernet/cavium/thunder/nicvf_main.c index 3240349615bd..8a37012c9c89 100644 --- a/drivers/net/ethernet/cavium/thunder/nicvf_main.c +++ b/drivers/net/ethernet/cavium/thunder/nicvf_main.c @@ -29,10 +29,20 @@  static const struct pci_device_id nicvf_id_table[] = {  	{ PCI_DEVICE_SUB(PCI_VENDOR_ID_CAVIUM,  			 PCI_DEVICE_ID_THUNDER_NIC_VF, -			 PCI_VENDOR_ID_CAVIUM, 0xA134) }, +			 PCI_VENDOR_ID_CAVIUM, +			 PCI_SUBSYS_DEVID_88XX_NIC_VF) },  	{ PCI_DEVICE_SUB(PCI_VENDOR_ID_CAVIUM,  			 PCI_DEVICE_ID_THUNDER_PASS1_NIC_VF, -			 PCI_VENDOR_ID_CAVIUM, 0xA11E) }, +			 PCI_VENDOR_ID_CAVIUM, +			 PCI_SUBSYS_DEVID_88XX_PASS1_NIC_VF) }, +	{ PCI_DEVICE_SUB(PCI_VENDOR_ID_CAVIUM, +			 PCI_DEVICE_ID_THUNDER_NIC_VF, +			 PCI_VENDOR_ID_CAVIUM, +			 PCI_SUBSYS_DEVID_81XX_NIC_VF) }, +	{ PCI_DEVICE_SUB(PCI_VENDOR_ID_CAVIUM, +			 PCI_DEVICE_ID_THUNDER_NIC_VF, +			 PCI_VENDOR_ID_CAVIUM, +			 PCI_SUBSYS_DEVID_83XX_NIC_VF) },  	{ 0, }  /* end of table */  }; @@ -59,25 +69,6 @@ static inline u8 nicvf_netdev_qidx(struct nicvf *nic, u8 qidx)  		return qidx;  } -static inline void nicvf_set_rx_frame_cnt(struct nicvf *nic, -					  struct sk_buff *skb) -{ -	if (skb->len <= 64) -		nic->drv_stats.rx_frames_64++; -	else if (skb->len <= 127) -		nic->drv_stats.rx_frames_127++; -	else if (skb->len <= 255) -		nic->drv_stats.rx_frames_255++; -	else if (skb->len <= 511) -		nic->drv_stats.rx_frames_511++; -	else if (skb->len <= 1023) -		nic->drv_stats.rx_frames_1023++; -	else if (skb->len <= 1518) -		nic->drv_stats.rx_frames_1518++; -	else -		nic->drv_stats.rx_frames_jumbo++; -} -  /* The Cavium ThunderX network controller can *only* be found in SoCs   * containing the ThunderX ARM64 CPU implementation.  All accesses to the device   * registers on this platform are implicitly strongly ordered with respect @@ -134,15 +125,19 @@ int nicvf_send_msg_to_pf(struct nicvf *nic, union nic_mbx *mbx)  	/* Wait for previous message to be acked, timeout 2sec */  	while (!nic->pf_acked) { -		if (nic->pf_nacked) +		if (nic->pf_nacked) { +			netdev_err(nic->netdev, +				   "PF NACK to mbox msg 0x%02x from VF%d\n", +				   (mbx->msg.msg & 0xFF), nic->vf_id);  			return -EINVAL; +		}  		msleep(sleep);  		if (nic->pf_acked)  			break;  		timeout -= sleep;  		if (!timeout) {  			netdev_err(nic->netdev, -				   "PF didn't ack to mbox msg %d from VF%d\n", +				   "PF didn't ACK to mbox msg 0x%02x from VF%d\n",  				   (mbx->msg.msg & 0xFF), nic->vf_id);  			return -EBUSY;  		} @@ -352,13 +347,7 @@ static int nicvf_rss_init(struct nicvf *nic)  	rss->enable = true; -	/* Using the HW reset value for now */ -	rss->key[0] = 0xFEED0BADFEED0BADULL; -	rss->key[1] = 0xFEED0BADFEED0BADULL; -	rss->key[2] = 0xFEED0BADFEED0BADULL; -	rss->key[3] = 0xFEED0BADFEED0BADULL; -	rss->key[4] = 0xFEED0BADFEED0BADULL; - +	netdev_rss_key_fill(rss->key, RSS_HASH_KEY_SIZE * sizeof(u64));  	nicvf_set_rss_key(nic);  	rss->cfg = RSS_IP_HASH_ENA | RSS_TCP_HASH_ENA | RSS_UDP_HASH_ENA; @@ -484,9 +473,6 @@ int nicvf_set_real_num_queues(struct net_device *netdev,  static int nicvf_init_resources(struct nicvf *nic)  {  	int err; -	union nic_mbx mbx = {}; - -	mbx.msg.msg = NIC_MBOX_MSG_CFG_DONE;  	/* Enable Qset */  	nicvf_qset_config(nic, true); @@ -499,15 +485,13 @@ static int nicvf_init_resources(struct nicvf *nic)  		return err;  	} -	/* Send VF config done msg to PF */ -	nicvf_write_to_mbx(nic, &mbx); -  	return 0;  }  static void nicvf_snd_pkt_handler(struct net_device *netdev, -				  struct cmp_queue *cq, -				  struct cqe_send_t *cqe_tx, int cqe_type) +				  struct cqe_send_t *cqe_tx, +				  int cqe_type, int budget, +				  unsigned int *tx_pkts, unsigned int *tx_bytes)  {  	struct sk_buff *skb = NULL;  	struct nicvf *nic = netdev_priv(netdev); @@ -526,7 +510,7 @@ static void nicvf_snd_pkt_handler(struct net_device *netdev,  		   __func__, cqe_tx->sq_qs, cqe_tx->sq_idx,  		   cqe_tx->sqe_ptr, hdr->subdesc_cnt); -	nicvf_check_cqe_tx_errs(nic, cq, cqe_tx); +	nicvf_check_cqe_tx_errs(nic, cqe_tx);  	skb = (struct sk_buff *)sq->skbuff[cqe_tx->sqe_ptr];  	if (skb) {  		/* Check for dummy descriptor used for HW TSO offload on 88xx */ @@ -538,7 +522,9 @@ static void nicvf_snd_pkt_handler(struct net_device *netdev,  		}  		nicvf_put_sq_desc(sq, hdr->subdesc_cnt + 1);  		prefetch(skb); -		dev_consume_skb_any(skb); +		(*tx_pkts)++; +		*tx_bytes += skb->len; +		napi_consume_skb(skb, budget);  		sq->skbuff[cqe_tx->sqe_ptr] = (u64)NULL;  	} else {  		/* In case of SW TSO on 88xx, only last segment will have @@ -618,8 +604,6 @@ static void nicvf_rcv_pkt_handler(struct net_device *netdev,  		return;  	} -	nicvf_set_rx_frame_cnt(nic, skb); -  	nicvf_set_rxhash(netdev, cqe_rx, skb);  	skb_record_rx_queue(skb, rq_idx); @@ -653,6 +637,7 @@ static int nicvf_cq_intr_handler(struct net_device *netdev, u8 cq_idx,  	struct cmp_queue *cq = &qs->cq[cq_idx];  	struct cqe_rx_t *cq_desc;  	struct netdev_queue *txq; +	unsigned int tx_pkts = 0, tx_bytes = 0;  	spin_lock_bh(&cq->lock);  loop: @@ -690,8 +675,9 @@ loop:  			work_done++;  		break;  		case CQE_TYPE_SEND: -			nicvf_snd_pkt_handler(netdev, cq, -					      (void *)cq_desc, CQE_TYPE_SEND); +			nicvf_snd_pkt_handler(netdev, +					      (void *)cq_desc, CQE_TYPE_SEND, +					      budget, &tx_pkts, &tx_bytes);  			tx_done++;  		break;  		case CQE_TYPE_INVALID: @@ -720,10 +706,13 @@ done:  		netdev = nic->pnicvf->netdev;  		txq = netdev_get_tx_queue(netdev,  					  nicvf_netdev_qidx(nic, cq_idx)); +		if (tx_pkts) +			netdev_tx_completed_queue(txq, tx_pkts, tx_bytes); +  		nic = nic->pnicvf;  		if (netif_tx_queue_stopped(txq) && netif_carrier_ok(netdev)) {  			netif_tx_start_queue(txq); -			nic->drv_stats.txq_wake++; +			this_cpu_inc(nic->drv_stats->txq_wake);  			if (netif_msg_tx_err(nic))  				netdev_warn(netdev,  					    "%s: Transmit queue wakeup SQ%d\n", @@ -933,16 +922,19 @@ static int nicvf_register_interrupts(struct nicvf *nic)  	int vector;  	for_each_cq_irq(irq) -		sprintf(nic->irq_name[irq], "NICVF%d CQ%d", -			nic->vf_id, irq); +		sprintf(nic->irq_name[irq], "%s-rxtx-%d", +			nic->pnicvf->netdev->name, +			nicvf_netdev_qidx(nic, irq));  	for_each_sq_irq(irq) -		sprintf(nic->irq_name[irq], "NICVF%d SQ%d", -			nic->vf_id, irq - NICVF_INTR_ID_SQ); +		sprintf(nic->irq_name[irq], "%s-sq-%d", +			nic->pnicvf->netdev->name, +			nicvf_netdev_qidx(nic, irq - NICVF_INTR_ID_SQ));  	for_each_rbdr_irq(irq) -		sprintf(nic->irq_name[irq], "NICVF%d RBDR%d", -			nic->vf_id, irq - NICVF_INTR_ID_RBDR); +		sprintf(nic->irq_name[irq], "%s-rbdr-%d", +			nic->pnicvf->netdev->name, +			nic->sqs_mode ? (nic->sqs_id + 1) : 0);  	/* Register CQ interrupts */  	for (irq = 0; irq < nic->qs->cq_cnt; irq++) { @@ -966,8 +958,9 @@ static int nicvf_register_interrupts(struct nicvf *nic)  	}  	/* Register QS error interrupt */ -	sprintf(nic->irq_name[NICVF_INTR_ID_QS_ERR], -		"NICVF%d Qset error", nic->vf_id); +	sprintf(nic->irq_name[NICVF_INTR_ID_QS_ERR], "%s-qset-err-%d", +		nic->pnicvf->netdev->name, +		nic->sqs_mode ? (nic->sqs_id + 1) : 0);  	irq = NICVF_INTR_ID_QS_ERR;  	ret = request_irq(nic->msix_entries[irq].vector,  			  nicvf_qs_err_intr_handler, @@ -1063,7 +1056,7 @@ static netdev_tx_t nicvf_xmit(struct sk_buff *skb, struct net_device *netdev)  	if (!netif_tx_queue_stopped(txq) && !nicvf_sq_append_skb(nic, skb)) {  		netif_tx_stop_queue(txq); -		nic->drv_stats.txq_stop++; +		this_cpu_inc(nic->drv_stats->txq_stop);  		if (netif_msg_tx_err(nic))  			netdev_warn(netdev,  				    "%s: Transmit ring full, stopping SQ%d\n", @@ -1146,6 +1139,9 @@ int nicvf_stop(struct net_device *netdev)  	netif_tx_disable(netdev); +	for (qidx = 0; qidx < netdev->num_tx_queues; qidx++) +		netdev_tx_reset_queue(netdev_get_tx_queue(netdev, qidx)); +  	/* Free resources */  	nicvf_config_data_transfer(nic, false); @@ -1165,14 +1161,24 @@ int nicvf_stop(struct net_device *netdev)  	return 0;  } +static int nicvf_update_hw_max_frs(struct nicvf *nic, int mtu) +{ +	union nic_mbx mbx = {}; + +	mbx.frs.msg = NIC_MBOX_MSG_SET_MAX_FRS; +	mbx.frs.max_frs = mtu; +	mbx.frs.vf_id = nic->vf_id; + +	return nicvf_send_msg_to_pf(nic, &mbx); +} +  int nicvf_open(struct net_device *netdev)  { -	int err, qidx; +	int cpu, err, qidx;  	struct nicvf *nic = netdev_priv(netdev);  	struct queue_set *qs = nic->qs;  	struct nicvf_cq_poll *cq_poll = NULL; - -	nic->mtu = netdev->mtu; +	union nic_mbx mbx = {};  	netif_carrier_off(netdev); @@ -1196,7 +1202,7 @@ int nicvf_open(struct net_device *netdev)  	}  	/* Check if we got MAC address from PF or else generate a radom MAC */ -	if (is_zero_ether_addr(netdev->dev_addr)) { +	if (!nic->sqs_mode && is_zero_ether_addr(netdev->dev_addr)) {  		eth_hw_addr_random(netdev);  		nicvf_hw_set_mac_addr(nic, netdev);  	} @@ -1224,9 +1230,17 @@ int nicvf_open(struct net_device *netdev)  	if (nic->sqs_mode)  		nicvf_get_primary_vf_struct(nic); -	/* Configure receive side scaling */ -	if (!nic->sqs_mode) +	/* Configure receive side scaling and MTU */ +	if (!nic->sqs_mode) {  		nicvf_rss_init(nic); +		if (nicvf_update_hw_max_frs(nic, netdev->mtu)) +			goto cleanup; + +		/* Clear percpu stats */ +		for_each_possible_cpu(cpu) +			memset(per_cpu_ptr(nic->drv_stats, cpu), 0, +			       sizeof(struct nicvf_drv_stats)); +	}  	err = nicvf_register_interrupts(nic);  	if (err) @@ -1252,8 +1266,9 @@ int nicvf_open(struct net_device *netdev)  	for (qidx = 0; qidx < qs->rbdr_cnt; qidx++)  		nicvf_enable_intr(nic, NICVF_INTR_RBDR, qidx); -	nic->drv_stats.txq_stop = 0; -	nic->drv_stats.txq_wake = 0; +	/* Send VF config done msg to PF */ +	mbx.msg.msg = NIC_MBOX_MSG_CFG_DONE; +	nicvf_write_to_mbx(nic, &mbx);  	return 0;  cleanup: @@ -1273,17 +1288,6 @@ napi_del:  	return err;  } -static int nicvf_update_hw_max_frs(struct nicvf *nic, int mtu) -{ -	union nic_mbx mbx = {}; - -	mbx.frs.msg = NIC_MBOX_MSG_SET_MAX_FRS; -	mbx.frs.max_frs = mtu; -	mbx.frs.vf_id = nic->vf_id; - -	return nicvf_send_msg_to_pf(nic, &mbx); -} -  static int nicvf_change_mtu(struct net_device *netdev, int new_mtu)  {  	struct nicvf *nic = netdev_priv(netdev); @@ -1294,10 +1298,13 @@ static int nicvf_change_mtu(struct net_device *netdev, int new_mtu)  	if (new_mtu < NIC_HW_MIN_FRS)  		return -EINVAL; +	netdev->mtu = new_mtu; + +	if (!netif_running(netdev)) +		return 0; +  	if (nicvf_update_hw_max_frs(nic, new_mtu))  		return -EINVAL; -	netdev->mtu = new_mtu; -	nic->mtu = new_mtu;  	return 0;  } @@ -1355,9 +1362,10 @@ void nicvf_update_lmac_stats(struct nicvf *nic)  void nicvf_update_stats(struct nicvf *nic)  { -	int qidx; +	int qidx, cpu; +	u64 tmp_stats = 0;  	struct nicvf_hw_stats *stats = &nic->hw_stats; -	struct nicvf_drv_stats *drv_stats = &nic->drv_stats; +	struct nicvf_drv_stats *drv_stats;  	struct queue_set *qs = nic->qs;  #define GET_RX_STATS(reg) \ @@ -1380,21 +1388,33 @@ void nicvf_update_stats(struct nicvf *nic)  	stats->rx_drop_l3_bcast = GET_RX_STATS(RX_DRP_L3BCAST);  	stats->rx_drop_l3_mcast = GET_RX_STATS(RX_DRP_L3MCAST); -	stats->tx_bytes_ok = GET_TX_STATS(TX_OCTS); -	stats->tx_ucast_frames_ok = GET_TX_STATS(TX_UCAST); -	stats->tx_bcast_frames_ok = GET_TX_STATS(TX_BCAST); -	stats->tx_mcast_frames_ok = GET_TX_STATS(TX_MCAST); +	stats->tx_bytes = GET_TX_STATS(TX_OCTS); +	stats->tx_ucast_frames = GET_TX_STATS(TX_UCAST); +	stats->tx_bcast_frames = GET_TX_STATS(TX_BCAST); +	stats->tx_mcast_frames = GET_TX_STATS(TX_MCAST);  	stats->tx_drops = GET_TX_STATS(TX_DROP); -	drv_stats->tx_frames_ok = stats->tx_ucast_frames_ok + -				  stats->tx_bcast_frames_ok + -				  stats->tx_mcast_frames_ok; -	drv_stats->rx_frames_ok = stats->rx_ucast_frames + -				  stats->rx_bcast_frames + -				  stats->rx_mcast_frames; -	drv_stats->rx_drops = stats->rx_drop_red + -			      stats->rx_drop_overrun; -	drv_stats->tx_drops = stats->tx_drops; +	/* On T88 pass 2.0, the dummy SQE added for TSO notification +	 * via CQE has 'dont_send' set. Hence HW drops the pkt pointed +	 * pointed by dummy SQE and results in tx_drops counter being +	 * incremented. Subtracting it from tx_tso counter will give +	 * exact tx_drops counter. +	 */ +	if (nic->t88 && nic->hw_tso) { +		for_each_possible_cpu(cpu) { +			drv_stats = per_cpu_ptr(nic->drv_stats, cpu); +			tmp_stats += drv_stats->tx_tso; +		} +		stats->tx_drops = tmp_stats - stats->tx_drops; +	} +	stats->tx_frames = stats->tx_ucast_frames + +			   stats->tx_bcast_frames + +			   stats->tx_mcast_frames; +	stats->rx_frames = stats->rx_ucast_frames + +			   stats->rx_bcast_frames + +			   stats->rx_mcast_frames; +	stats->rx_drops = stats->rx_drop_red + +			  stats->rx_drop_overrun;  	/* Update RQ and SQ stats */  	for (qidx = 0; qidx < qs->rq_cnt; qidx++) @@ -1408,18 +1428,17 @@ static struct rtnl_link_stats64 *nicvf_get_stats64(struct net_device *netdev,  {  	struct nicvf *nic = netdev_priv(netdev);  	struct nicvf_hw_stats *hw_stats = &nic->hw_stats; -	struct nicvf_drv_stats *drv_stats = &nic->drv_stats;  	nicvf_update_stats(nic);  	stats->rx_bytes = hw_stats->rx_bytes; -	stats->rx_packets = drv_stats->rx_frames_ok; -	stats->rx_dropped = drv_stats->rx_drops; +	stats->rx_packets = hw_stats->rx_frames; +	stats->rx_dropped = hw_stats->rx_drops;  	stats->multicast = hw_stats->rx_mcast_frames; -	stats->tx_bytes = hw_stats->tx_bytes_ok; -	stats->tx_packets = drv_stats->tx_frames_ok; -	stats->tx_dropped = drv_stats->tx_drops; +	stats->tx_bytes = hw_stats->tx_bytes; +	stats->tx_packets = hw_stats->tx_frames; +	stats->tx_dropped = hw_stats->tx_drops;  	return stats;  } @@ -1432,7 +1451,7 @@ static void nicvf_tx_timeout(struct net_device *dev)  		netdev_warn(dev, "%s: Transmit timed out, resetting\n",  			    dev->name); -	nic->drv_stats.tx_timeout++; +	this_cpu_inc(nic->drv_stats->tx_timeout);  	schedule_work(&nic->reset_task);  } @@ -1533,14 +1552,13 @@ static int nicvf_probe(struct pci_dev *pdev, const struct pci_device_id *ent)  		goto err_release_regions;  	} -	qcount = MAX_CMP_QUEUES_PER_QS; +	qcount = netif_get_num_default_rss_queues();  	/* Restrict multiqset support only for host bound VFs */  	if (pdev->is_virtfn) {  		/* Set max number of queues per VF */ -		qcount = roundup(num_online_cpus(), MAX_CMP_QUEUES_PER_QS); -		qcount = min(qcount, -			     (MAX_SQS_PER_VF + 1) * MAX_CMP_QUEUES_PER_QS); +		qcount = min_t(int, num_online_cpus(), +			       (MAX_SQS_PER_VF + 1) * MAX_CMP_QUEUES_PER_QS);  	}  	netdev = alloc_etherdev_mqs(sizeof(struct nicvf), qcount, qcount); @@ -1567,6 +1585,12 @@ static int nicvf_probe(struct pci_dev *pdev, const struct pci_device_id *ent)  		goto err_free_netdev;  	} +	nic->drv_stats = netdev_alloc_pcpu_stats(struct nicvf_drv_stats); +	if (!nic->drv_stats) { +		err = -ENOMEM; +		goto err_free_netdev; +	} +  	err = nicvf_set_qset_resources(nic);  	if (err)  		goto err_free_netdev; @@ -1625,6 +1649,8 @@ err_unregister_interrupts:  	nicvf_unregister_interrupts(nic);  err_free_netdev:  	pci_set_drvdata(pdev, NULL); +	if (nic->drv_stats) +		free_percpu(nic->drv_stats);  	free_netdev(netdev);  err_release_regions:  	pci_release_regions(pdev); @@ -1652,6 +1678,8 @@ static void nicvf_remove(struct pci_dev *pdev)  		unregister_netdev(pnetdev);  	nicvf_unregister_interrupts(nic);  	pci_set_drvdata(pdev, NULL); +	if (nic->drv_stats) +		free_percpu(nic->drv_stats);  	free_netdev(netdev);  	pci_release_regions(pdev);  	pci_disable_device(pdev); |