mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	octeontx2-pf: Implement ingress/egress VLAN offload
This patch implements egress VLAN offload by appending NIX_SEND_EXT_S header to NIX_SEND_HDR_S. The VLAN TCI information is specified in the NIX_SEND_EXT_S. The VLAN offload in the ingress path is implemented by configuring the NIX_RX_VTAG_ACTION_S to strip and capture the outer vlan fields. The NIX PF allocates one MCAM entry for Rx VLAN offload. Signed-off-by: Hariprasad Kelam <hkelam@marvell.com> Signed-off-by: Sunil Goutham <sgoutham@marvell.com> Signed-off-by: Naveen Mamindlapalli <naveenm@marvell.com> Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
		
							parent
							
								
									9a946def26
								
							
						
					
					
						commit
						fd9d7859db
					
				| @ -479,6 +479,19 @@ enum nix_af_status { | ||||
| 	NIX_AF_INVAL_NPA_PF_FUNC    = -419, | ||||
| 	NIX_AF_INVAL_SSO_PF_FUNC    = -420, | ||||
| 	NIX_AF_ERR_TX_VTAG_NOSPC    = -421, | ||||
| 	NIX_AF_ERR_RX_VTAG_INUSE    = -422, | ||||
| }; | ||||
| 
 | ||||
| /* For NIX RX vtag action  */ | ||||
| enum nix_rx_vtag0_type { | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE0, /* reserved for rx vlan offload */ | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE1, | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE2, | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE3, | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE4, | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE5, | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE6, | ||||
| 	NIX_AF_LFX_RX_VTAG_TYPE7, | ||||
| }; | ||||
| 
 | ||||
| /* For NIX LF context alloc and init */ | ||||
|  | ||||
| @ -1984,7 +1984,8 @@ static int nix_rx_vtag_cfg(struct rvu *rvu, int nixlf, int blkaddr, | ||||
| { | ||||
| 	u64 regval = req->vtag_size; | ||||
| 
 | ||||
| 	if (req->rx.vtag_type > 7 || req->vtag_size > VTAGSIZE_T8) | ||||
| 	if (req->rx.vtag_type > NIX_AF_LFX_RX_VTAG_TYPE7 || | ||||
| 	    req->vtag_size > VTAGSIZE_T8) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (req->rx.capture_vtag) | ||||
|  | ||||
| @ -191,10 +191,14 @@ int otx2_set_mac_address(struct net_device *netdev, void *p) | ||||
| 	if (!is_valid_ether_addr(addr->sa_data)) | ||||
| 		return -EADDRNOTAVAIL; | ||||
| 
 | ||||
| 	if (!otx2_hw_set_mac_addr(pfvf, addr->sa_data)) | ||||
| 	if (!otx2_hw_set_mac_addr(pfvf, addr->sa_data)) { | ||||
| 		memcpy(netdev->dev_addr, addr->sa_data, netdev->addr_len); | ||||
| 	else | ||||
| 		/* update dmac field in vlan offload rule */ | ||||
| 		if (pfvf->flags & OTX2_FLAG_RX_VLAN_SUPPORT) | ||||
| 			otx2_install_rxvlan_offload_flow(pfvf); | ||||
| 	} else { | ||||
| 		return -EPERM; | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| @ -240,10 +240,13 @@ struct otx2_flow_config { | ||||
| 	u32			nr_flows; | ||||
| #define OTX2_MAX_NTUPLE_FLOWS	32 | ||||
| #define OTX2_MAX_UNICAST_FLOWS	8 | ||||
| #define OTX2_MAX_VLAN_FLOWS	1 | ||||
| #define OTX2_MCAM_COUNT		(OTX2_MAX_NTUPLE_FLOWS + \ | ||||
| 				 OTX2_MAX_UNICAST_FLOWS) | ||||
| 				 OTX2_MAX_UNICAST_FLOWS + \ | ||||
| 				 OTX2_MAX_VLAN_FLOWS) | ||||
| 	u32			ntuple_offset; | ||||
| 	u32			unicast_offset; | ||||
| 	u32			rx_vlan_offset; | ||||
| 	u32                     ntuple_max_flows; | ||||
| 	struct list_head	flow_list; | ||||
| }; | ||||
| @ -261,6 +264,7 @@ struct otx2_nic { | ||||
| #define OTX2_FLAG_MCAM_ENTRIES_ALLOC		BIT_ULL(3) | ||||
| #define OTX2_FLAG_NTUPLE_SUPPORT		BIT_ULL(4) | ||||
| #define OTX2_FLAG_UCAST_FLTR_SUPPORT		BIT_ULL(5) | ||||
| #define OTX2_FLAG_RX_VLAN_SUPPORT		BIT_ULL(6) | ||||
| #define OTX2_FLAG_RX_PAUSE_ENABLED		BIT_ULL(9) | ||||
| #define OTX2_FLAG_TX_PAUSE_ENABLED		BIT_ULL(10) | ||||
| 	u64			flags; | ||||
| @ -687,5 +691,7 @@ int otx2_prepare_flow_request(struct ethtool_rx_flow_spec *fsp, | ||||
| 			      struct npc_install_flow_req *req); | ||||
| int otx2_del_macfilter(struct net_device *netdev, const u8 *mac); | ||||
| int otx2_add_macfilter(struct net_device *netdev, const u8 *mac); | ||||
| int otx2_enable_rxvlan(struct otx2_nic *pf, bool enable); | ||||
| int otx2_install_rxvlan_offload_flow(struct otx2_nic *pfvf); | ||||
| 
 | ||||
| #endif /* OTX2_COMMON_H */ | ||||
|  | ||||
| @ -58,8 +58,11 @@ int otx2_alloc_mcam_entries(struct otx2_nic *pfvf) | ||||
| 		flow_cfg->ntuple_offset = 0; | ||||
| 		flow_cfg->unicast_offset = flow_cfg->ntuple_offset + | ||||
| 						OTX2_MAX_NTUPLE_FLOWS; | ||||
| 		flow_cfg->rx_vlan_offset = flow_cfg->unicast_offset + | ||||
| 						OTX2_MAX_UNICAST_FLOWS; | ||||
| 		pfvf->flags |= OTX2_FLAG_NTUPLE_SUPPORT; | ||||
| 		pfvf->flags |= OTX2_FLAG_UCAST_FLTR_SUPPORT; | ||||
| 		pfvf->flags |= OTX2_FLAG_RX_VLAN_SUPPORT; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = 0; i < rsp->count; i++) | ||||
| @ -711,3 +714,102 @@ int otx2_destroy_mcam_flows(struct otx2_nic *pfvf) | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| int otx2_install_rxvlan_offload_flow(struct otx2_nic *pfvf) | ||||
| { | ||||
| 	struct otx2_flow_config *flow_cfg = pfvf->flow_cfg; | ||||
| 	struct npc_install_flow_req *req; | ||||
| 	int err; | ||||
| 
 | ||||
| 	mutex_lock(&pfvf->mbox.lock); | ||||
| 	req = otx2_mbox_alloc_msg_npc_install_flow(&pfvf->mbox); | ||||
| 	if (!req) { | ||||
| 		mutex_unlock(&pfvf->mbox.lock); | ||||
| 		return -ENOMEM; | ||||
| 	} | ||||
| 
 | ||||
| 	req->entry = flow_cfg->entry[flow_cfg->rx_vlan_offset]; | ||||
| 	req->intf = NIX_INTF_RX; | ||||
| 	ether_addr_copy(req->packet.dmac, pfvf->netdev->dev_addr); | ||||
| 	eth_broadcast_addr((u8 *)&req->mask.dmac); | ||||
| 	req->channel = pfvf->hw.rx_chan_base; | ||||
| 	req->op = NIX_RX_ACTION_DEFAULT; | ||||
| 	req->features = BIT_ULL(NPC_OUTER_VID) | BIT_ULL(NPC_DMAC); | ||||
| 	req->vtag0_valid = true; | ||||
| 	req->vtag0_type = NIX_AF_LFX_RX_VTAG_TYPE0; | ||||
| 
 | ||||
| 	/* Send message to AF */ | ||||
| 	err = otx2_sync_mbox_msg(&pfvf->mbox); | ||||
| 	mutex_unlock(&pfvf->mbox.lock); | ||||
| 	return err; | ||||
| } | ||||
| 
 | ||||
| static int otx2_delete_rxvlan_offload_flow(struct otx2_nic *pfvf) | ||||
| { | ||||
| 	struct otx2_flow_config *flow_cfg = pfvf->flow_cfg; | ||||
| 	struct npc_delete_flow_req *req; | ||||
| 	int err; | ||||
| 
 | ||||
| 	mutex_lock(&pfvf->mbox.lock); | ||||
| 	req = otx2_mbox_alloc_msg_npc_delete_flow(&pfvf->mbox); | ||||
| 	if (!req) { | ||||
| 		mutex_unlock(&pfvf->mbox.lock); | ||||
| 		return -ENOMEM; | ||||
| 	} | ||||
| 
 | ||||
| 	req->entry = flow_cfg->entry[flow_cfg->rx_vlan_offset]; | ||||
| 	/* Send message to AF */ | ||||
| 	err = otx2_sync_mbox_msg(&pfvf->mbox); | ||||
| 	mutex_unlock(&pfvf->mbox.lock); | ||||
| 	return err; | ||||
| } | ||||
| 
 | ||||
| int otx2_enable_rxvlan(struct otx2_nic *pf, bool enable) | ||||
| { | ||||
| 	struct nix_vtag_config *req; | ||||
| 	struct mbox_msghdr *rsp_hdr; | ||||
| 	int err; | ||||
| 
 | ||||
| 	/* Dont have enough mcam entries */ | ||||
| 	if (!(pf->flags & OTX2_FLAG_RX_VLAN_SUPPORT)) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	if (enable) { | ||||
| 		err = otx2_install_rxvlan_offload_flow(pf); | ||||
| 		if (err) | ||||
| 			return err; | ||||
| 	} else { | ||||
| 		err = otx2_delete_rxvlan_offload_flow(pf); | ||||
| 		if (err) | ||||
| 			return err; | ||||
| 	} | ||||
| 
 | ||||
| 	mutex_lock(&pf->mbox.lock); | ||||
| 	req = otx2_mbox_alloc_msg_nix_vtag_cfg(&pf->mbox); | ||||
| 	if (!req) { | ||||
| 		mutex_unlock(&pf->mbox.lock); | ||||
| 		return -ENOMEM; | ||||
| 	} | ||||
| 
 | ||||
| 	/* config strip, capture and size */ | ||||
| 	req->vtag_size = VTAGSIZE_T4; | ||||
| 	req->cfg_type = 1; /* rx vlan cfg */ | ||||
| 	req->rx.vtag_type = NIX_AF_LFX_RX_VTAG_TYPE0; | ||||
| 	req->rx.strip_vtag = enable; | ||||
| 	req->rx.capture_vtag = enable; | ||||
| 
 | ||||
| 	err = otx2_sync_mbox_msg(&pf->mbox); | ||||
| 	if (err) { | ||||
| 		mutex_unlock(&pf->mbox.lock); | ||||
| 		return err; | ||||
| 	} | ||||
| 
 | ||||
| 	rsp_hdr = otx2_mbox_get_rsp(&pf->mbox.mbox, 0, &req->hdr); | ||||
| 	if (IS_ERR(rsp_hdr)) { | ||||
| 		mutex_unlock(&pf->mbox.lock); | ||||
| 		return PTR_ERR(rsp_hdr); | ||||
| 	} | ||||
| 
 | ||||
| 	mutex_unlock(&pf->mbox.lock); | ||||
| 	return rsp_hdr->rc; | ||||
| } | ||||
|  | ||||
| @ -1566,6 +1566,9 @@ int otx2_open(struct net_device *netdev) | ||||
| 
 | ||||
| 	otx2_set_cints_affinity(pf); | ||||
| 
 | ||||
| 	if (pf->flags & OTX2_FLAG_RX_VLAN_SUPPORT) | ||||
| 		otx2_enable_rxvlan(pf, true); | ||||
| 
 | ||||
| 	/* When reinitializing enable time stamping if it is enabled before */ | ||||
| 	if (pf->flags & OTX2_FLAG_TX_TSTAMP_ENABLED) { | ||||
| 		pf->flags &= ~OTX2_FLAG_TX_TSTAMP_ENABLED; | ||||
| @ -1763,6 +1766,10 @@ static int otx2_set_features(struct net_device *netdev, | ||||
| 		return otx2_cgx_config_loopback(pf, | ||||
| 						features & NETIF_F_LOOPBACK); | ||||
| 
 | ||||
| 	if ((changed & NETIF_F_HW_VLAN_CTAG_RX) && netif_running(netdev)) | ||||
| 		return otx2_enable_rxvlan(pf, | ||||
| 					  features & NETIF_F_HW_VLAN_CTAG_RX); | ||||
| 
 | ||||
| 	if ((changed & NETIF_F_NTUPLE) && !ntuple) | ||||
| 		otx2_destroy_ntuple_flows(pf); | ||||
| 
 | ||||
| @ -2138,6 +2145,15 @@ static int otx2_probe(struct pci_dev *pdev, const struct pci_device_id *id) | ||||
| 	if (pf->flags & OTX2_FLAG_UCAST_FLTR_SUPPORT) | ||||
| 		netdev->priv_flags |= IFF_UNICAST_FLT; | ||||
| 
 | ||||
| 	/* Support TSO on tag interface */ | ||||
| 	netdev->vlan_features |= netdev->features; | ||||
| 	netdev->hw_features  |= NETIF_F_HW_VLAN_CTAG_TX | | ||||
| 				NETIF_F_HW_VLAN_STAG_TX; | ||||
| 	if (pf->flags & OTX2_FLAG_RX_VLAN_SUPPORT) | ||||
| 		netdev->hw_features |= NETIF_F_HW_VLAN_CTAG_RX | | ||||
| 				       NETIF_F_HW_VLAN_STAG_RX; | ||||
| 	netdev->features |= netdev->hw_features; | ||||
| 
 | ||||
| 	netdev->gso_max_segs = OTX2_MAX_GSO_SEGS; | ||||
| 	netdev->watchdog_timeo = OTX2_TX_TIMEOUT; | ||||
| 
 | ||||
|  | ||||
| @ -556,6 +556,19 @@ static void otx2_sqe_add_ext(struct otx2_nic *pfvf, struct otx2_snd_queue *sq, | ||||
| 		ext->tstmp = 1; | ||||
| 	} | ||||
| 
 | ||||
| #define OTX2_VLAN_PTR_OFFSET     (ETH_HLEN - ETH_TLEN) | ||||
| 	if (skb_vlan_tag_present(skb)) { | ||||
| 		if (skb->vlan_proto == htons(ETH_P_8021Q)) { | ||||
| 			ext->vlan1_ins_ena = 1; | ||||
| 			ext->vlan1_ins_ptr = OTX2_VLAN_PTR_OFFSET; | ||||
| 			ext->vlan1_ins_tci = skb_vlan_tag_get(skb); | ||||
| 		} else if (skb->vlan_proto == htons(ETH_P_8021AD)) { | ||||
| 			ext->vlan0_ins_ena = 1; | ||||
| 			ext->vlan0_ins_ptr = OTX2_VLAN_PTR_OFFSET; | ||||
| 			ext->vlan0_ins_tci = skb_vlan_tag_get(skb); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	*offset += sizeof(*ext); | ||||
| } | ||||
| 
 | ||||
| @ -871,6 +884,9 @@ bool otx2_sq_append_skb(struct net_device *netdev, struct otx2_snd_queue *sq, | ||||
| 	} | ||||
| 
 | ||||
| 	if (skb_shinfo(skb)->gso_size && !is_hw_tso_supported(pfvf, skb)) { | ||||
| 		/* Insert vlan tag before giving pkt to tso */ | ||||
| 		if (skb_vlan_tag_present(skb)) | ||||
| 			skb = __vlan_hwaccel_push_inside(skb); | ||||
| 		otx2_sq_append_tso(pfvf, sq, skb, qidx); | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| @ -558,6 +558,11 @@ static int otx2vf_probe(struct pci_dev *pdev, const struct pci_device_id *id) | ||||
| 			      NETIF_F_SG | NETIF_F_TSO | NETIF_F_TSO6 | | ||||
| 			      NETIF_F_GSO_UDP_L4; | ||||
| 	netdev->features = netdev->hw_features; | ||||
| 	/* Support TSO on tag interface */ | ||||
| 	netdev->vlan_features |= netdev->features; | ||||
| 	netdev->hw_features  |= NETIF_F_HW_VLAN_CTAG_TX | | ||||
| 				NETIF_F_HW_VLAN_STAG_TX; | ||||
| 	netdev->features |= netdev->hw_features; | ||||
| 
 | ||||
| 	netdev->gso_max_segs = OTX2_MAX_GSO_SEGS; | ||||
| 	netdev->watchdog_timeo = OTX2_TX_TIMEOUT; | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Hariprasad Kelam
						Hariprasad Kelam