1707 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1707 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /* Copyright (C) 2018-2020, Intel Corporation. */
 | |
| 
 | |
| /* flow director ethtool support for ice */
 | |
| 
 | |
| #include "ice.h"
 | |
| #include "ice_lib.h"
 | |
| #include "ice_flow.h"
 | |
| 
 | |
| static struct in6_addr full_ipv6_addr_mask = {
 | |
| 	.in6_u = {
 | |
| 		.u6_addr8 = {
 | |
| 			0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
 | |
| 			0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| 
 | |
| static struct in6_addr zero_ipv6_addr_mask = {
 | |
| 	.in6_u = {
 | |
| 		.u6_addr8 = {
 | |
| 			0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 			0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /* calls to ice_flow_add_prof require the number of segments in the array
 | |
|  * for segs_cnt. In this code that is one more than the index.
 | |
|  */
 | |
| #define TNL_SEG_CNT(_TNL_) ((_TNL_) + 1)
 | |
| 
 | |
| /**
 | |
|  * ice_fltr_to_ethtool_flow - convert filter type values to ethtool
 | |
|  * flow type values
 | |
|  * @flow: filter type to be converted
 | |
|  *
 | |
|  * Returns the corresponding ethtool flow type.
 | |
|  */
 | |
| static int ice_fltr_to_ethtool_flow(enum ice_fltr_ptype flow)
 | |
| {
 | |
| 	switch (flow) {
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_TCP:
 | |
| 		return TCP_V4_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_UDP:
 | |
| 		return UDP_V4_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_SCTP:
 | |
| 		return SCTP_V4_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_OTHER:
 | |
| 		return IPV4_USER_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_TCP:
 | |
| 		return TCP_V6_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_UDP:
 | |
| 		return UDP_V6_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_SCTP:
 | |
| 		return SCTP_V6_FLOW;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_OTHER:
 | |
| 		return IPV6_USER_FLOW;
 | |
| 	default:
 | |
| 		/* 0 is undefined ethtool flow */
 | |
| 		return 0;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_ethtool_flow_to_fltr - convert ethtool flow type to filter enum
 | |
|  * @eth: Ethtool flow type to be converted
 | |
|  *
 | |
|  * Returns flow enum
 | |
|  */
 | |
| static enum ice_fltr_ptype ice_ethtool_flow_to_fltr(int eth)
 | |
| {
 | |
| 	switch (eth) {
 | |
| 	case TCP_V4_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV4_TCP;
 | |
| 	case UDP_V4_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV4_UDP;
 | |
| 	case SCTP_V4_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV4_SCTP;
 | |
| 	case IPV4_USER_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV4_OTHER;
 | |
| 	case TCP_V6_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV6_TCP;
 | |
| 	case UDP_V6_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV6_UDP;
 | |
| 	case SCTP_V6_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV6_SCTP;
 | |
| 	case IPV6_USER_FLOW:
 | |
| 		return ICE_FLTR_PTYPE_NONF_IPV6_OTHER;
 | |
| 	default:
 | |
| 		return ICE_FLTR_PTYPE_NONF_NONE;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_is_mask_valid - check mask field set
 | |
|  * @mask: full mask to check
 | |
|  * @field: field for which mask should be valid
 | |
|  *
 | |
|  * If the mask is fully set return true. If it is not valid for field return
 | |
|  * false.
 | |
|  */
 | |
| static bool ice_is_mask_valid(u64 mask, u64 field)
 | |
| {
 | |
| 	return (mask & field) == field;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_get_ethtool_fdir_entry - fill ethtool structure with fdir filter data
 | |
|  * @hw: hardware structure that contains filter list
 | |
|  * @cmd: ethtool command data structure to receive the filter data
 | |
|  *
 | |
|  * Returns 0 on success and -EINVAL on failure
 | |
|  */
 | |
| int ice_get_ethtool_fdir_entry(struct ice_hw *hw, struct ethtool_rxnfc *cmd)
 | |
| {
 | |
| 	struct ethtool_rx_flow_spec *fsp;
 | |
| 	struct ice_fdir_fltr *rule;
 | |
| 	int ret = 0;
 | |
| 	u16 idx;
 | |
| 
 | |
| 	fsp = (struct ethtool_rx_flow_spec *)&cmd->fs;
 | |
| 
 | |
| 	mutex_lock(&hw->fdir_fltr_lock);
 | |
| 
 | |
| 	rule = ice_fdir_find_fltr_by_idx(hw, fsp->location);
 | |
| 
 | |
| 	if (!rule || fsp->location != rule->fltr_id) {
 | |
| 		ret = -EINVAL;
 | |
| 		goto release_lock;
 | |
| 	}
 | |
| 
 | |
| 	fsp->flow_type = ice_fltr_to_ethtool_flow(rule->flow_type);
 | |
| 
 | |
| 	memset(&fsp->m_u, 0, sizeof(fsp->m_u));
 | |
| 	memset(&fsp->m_ext, 0, sizeof(fsp->m_ext));
 | |
| 
 | |
| 	switch (fsp->flow_type) {
 | |
| 	case IPV4_USER_FLOW:
 | |
| 		fsp->h_u.usr_ip4_spec.ip_ver = ETH_RX_NFC_IP4;
 | |
| 		fsp->h_u.usr_ip4_spec.proto = 0;
 | |
| 		fsp->h_u.usr_ip4_spec.l4_4_bytes = rule->ip.v4.l4_header;
 | |
| 		fsp->h_u.usr_ip4_spec.tos = rule->ip.v4.tos;
 | |
| 		fsp->h_u.usr_ip4_spec.ip4src = rule->ip.v4.src_ip;
 | |
| 		fsp->h_u.usr_ip4_spec.ip4dst = rule->ip.v4.dst_ip;
 | |
| 		fsp->m_u.usr_ip4_spec.ip4src = rule->mask.v4.src_ip;
 | |
| 		fsp->m_u.usr_ip4_spec.ip4dst = rule->mask.v4.dst_ip;
 | |
| 		fsp->m_u.usr_ip4_spec.ip_ver = 0xFF;
 | |
| 		fsp->m_u.usr_ip4_spec.proto = 0;
 | |
| 		fsp->m_u.usr_ip4_spec.l4_4_bytes = rule->mask.v4.l4_header;
 | |
| 		fsp->m_u.usr_ip4_spec.tos = rule->mask.v4.tos;
 | |
| 		break;
 | |
| 	case TCP_V4_FLOW:
 | |
| 	case UDP_V4_FLOW:
 | |
| 	case SCTP_V4_FLOW:
 | |
| 		fsp->h_u.tcp_ip4_spec.psrc = rule->ip.v4.src_port;
 | |
| 		fsp->h_u.tcp_ip4_spec.pdst = rule->ip.v4.dst_port;
 | |
| 		fsp->h_u.tcp_ip4_spec.ip4src = rule->ip.v4.src_ip;
 | |
| 		fsp->h_u.tcp_ip4_spec.ip4dst = rule->ip.v4.dst_ip;
 | |
| 		fsp->m_u.tcp_ip4_spec.psrc = rule->mask.v4.src_port;
 | |
| 		fsp->m_u.tcp_ip4_spec.pdst = rule->mask.v4.dst_port;
 | |
| 		fsp->m_u.tcp_ip4_spec.ip4src = rule->mask.v4.src_ip;
 | |
| 		fsp->m_u.tcp_ip4_spec.ip4dst = rule->mask.v4.dst_ip;
 | |
| 		break;
 | |
| 	case IPV6_USER_FLOW:
 | |
| 		fsp->h_u.usr_ip6_spec.l4_4_bytes = rule->ip.v6.l4_header;
 | |
| 		fsp->h_u.usr_ip6_spec.tclass = rule->ip.v6.tc;
 | |
| 		fsp->h_u.usr_ip6_spec.l4_proto = rule->ip.v6.proto;
 | |
| 		memcpy(fsp->h_u.tcp_ip6_spec.ip6src, rule->ip.v6.src_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(fsp->h_u.tcp_ip6_spec.ip6dst, rule->ip.v6.dst_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(fsp->m_u.tcp_ip6_spec.ip6src, rule->mask.v6.src_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(fsp->m_u.tcp_ip6_spec.ip6dst, rule->mask.v6.dst_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		fsp->m_u.usr_ip6_spec.l4_4_bytes = rule->mask.v6.l4_header;
 | |
| 		fsp->m_u.usr_ip6_spec.tclass = rule->mask.v6.tc;
 | |
| 		fsp->m_u.usr_ip6_spec.l4_proto = rule->mask.v6.proto;
 | |
| 		break;
 | |
| 	case TCP_V6_FLOW:
 | |
| 	case UDP_V6_FLOW:
 | |
| 	case SCTP_V6_FLOW:
 | |
| 		memcpy(fsp->h_u.tcp_ip6_spec.ip6src, rule->ip.v6.src_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(fsp->h_u.tcp_ip6_spec.ip6dst, rule->ip.v6.dst_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		fsp->h_u.tcp_ip6_spec.psrc = rule->ip.v6.src_port;
 | |
| 		fsp->h_u.tcp_ip6_spec.pdst = rule->ip.v6.dst_port;
 | |
| 		memcpy(fsp->m_u.tcp_ip6_spec.ip6src,
 | |
| 		       rule->mask.v6.src_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(fsp->m_u.tcp_ip6_spec.ip6dst,
 | |
| 		       rule->mask.v6.dst_ip,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		fsp->m_u.tcp_ip6_spec.psrc = rule->mask.v6.src_port;
 | |
| 		fsp->m_u.tcp_ip6_spec.pdst = rule->mask.v6.dst_port;
 | |
| 		fsp->h_u.tcp_ip6_spec.tclass = rule->ip.v6.tc;
 | |
| 		fsp->m_u.tcp_ip6_spec.tclass = rule->mask.v6.tc;
 | |
| 		break;
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	if (rule->dest_ctl == ICE_FLTR_PRGM_DESC_DEST_DROP_PKT)
 | |
| 		fsp->ring_cookie = RX_CLS_FLOW_DISC;
 | |
| 	else
 | |
| 		fsp->ring_cookie = rule->q_index;
 | |
| 
 | |
| 	idx = ice_ethtool_flow_to_fltr(fsp->flow_type);
 | |
| 	if (idx == ICE_FLTR_PTYPE_NONF_NONE) {
 | |
| 		dev_err(ice_hw_to_dev(hw), "Missing input index for flow_type %d\n",
 | |
| 			rule->flow_type);
 | |
| 		ret = -EINVAL;
 | |
| 	}
 | |
| 
 | |
| release_lock:
 | |
| 	mutex_unlock(&hw->fdir_fltr_lock);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_get_fdir_fltr_ids - fill buffer with filter IDs of active filters
 | |
|  * @hw: hardware structure containing the filter list
 | |
|  * @cmd: ethtool command data structure
 | |
|  * @rule_locs: ethtool array passed in from OS to receive filter IDs
 | |
|  *
 | |
|  * Returns 0 as expected for success by ethtool
 | |
|  */
 | |
| int
 | |
| ice_get_fdir_fltr_ids(struct ice_hw *hw, struct ethtool_rxnfc *cmd,
 | |
| 		      u32 *rule_locs)
 | |
| {
 | |
| 	struct ice_fdir_fltr *f_rule;
 | |
| 	unsigned int cnt = 0;
 | |
| 	int val = 0;
 | |
| 
 | |
| 	/* report total rule count */
 | |
| 	cmd->data = ice_get_fdir_cnt_all(hw);
 | |
| 
 | |
| 	mutex_lock(&hw->fdir_fltr_lock);
 | |
| 
 | |
| 	list_for_each_entry(f_rule, &hw->fdir_list_head, fltr_node) {
 | |
| 		if (cnt == cmd->rule_cnt) {
 | |
| 			val = -EMSGSIZE;
 | |
| 			goto release_lock;
 | |
| 		}
 | |
| 		rule_locs[cnt] = f_rule->fltr_id;
 | |
| 		cnt++;
 | |
| 	}
 | |
| 
 | |
| release_lock:
 | |
| 	mutex_unlock(&hw->fdir_fltr_lock);
 | |
| 	if (!val)
 | |
| 		cmd->rule_cnt = cnt;
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_get_hw_prof - return the ice_fd_hw_proc associated with a flow
 | |
|  * @hw: hardware structure containing the filter list
 | |
|  * @blk: hardware block
 | |
|  * @flow: FDir flow type to release
 | |
|  */
 | |
| static struct ice_fd_hw_prof *
 | |
| ice_fdir_get_hw_prof(struct ice_hw *hw, enum ice_block blk, int flow)
 | |
| {
 | |
| 	if (blk == ICE_BLK_FD && hw->fdir_prof)
 | |
| 		return hw->fdir_prof[flow];
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_erase_flow_from_hw - remove a flow from the HW profile tables
 | |
|  * @hw: hardware structure containing the filter list
 | |
|  * @blk: hardware block
 | |
|  * @flow: FDir flow type to release
 | |
|  */
 | |
| static void
 | |
| ice_fdir_erase_flow_from_hw(struct ice_hw *hw, enum ice_block blk, int flow)
 | |
| {
 | |
| 	struct ice_fd_hw_prof *prof = ice_fdir_get_hw_prof(hw, blk, flow);
 | |
| 	int tun;
 | |
| 
 | |
| 	if (!prof)
 | |
| 		return;
 | |
| 
 | |
| 	for (tun = 0; tun < ICE_FD_HW_SEG_MAX; tun++) {
 | |
| 		u64 prof_id;
 | |
| 		int j;
 | |
| 
 | |
| 		prof_id = flow + tun * ICE_FLTR_PTYPE_MAX;
 | |
| 		for (j = 0; j < prof->cnt; j++) {
 | |
| 			u16 vsi_num;
 | |
| 
 | |
| 			if (!prof->entry_h[j][tun] || !prof->vsi_h[j])
 | |
| 				continue;
 | |
| 			vsi_num = ice_get_hw_vsi_num(hw, prof->vsi_h[j]);
 | |
| 			ice_rem_prof_id_flow(hw, blk, vsi_num, prof_id);
 | |
| 			ice_flow_rem_entry(hw, blk, prof->entry_h[j][tun]);
 | |
| 			prof->entry_h[j][tun] = 0;
 | |
| 		}
 | |
| 		ice_flow_rem_prof(hw, blk, prof_id);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_rem_flow - release the ice_flow structures for a filter type
 | |
|  * @hw: hardware structure containing the filter list
 | |
|  * @blk: hardware block
 | |
|  * @flow_type: FDir flow type to release
 | |
|  */
 | |
| static void
 | |
| ice_fdir_rem_flow(struct ice_hw *hw, enum ice_block blk,
 | |
| 		  enum ice_fltr_ptype flow_type)
 | |
| {
 | |
| 	int flow = (int)flow_type & ~FLOW_EXT;
 | |
| 	struct ice_fd_hw_prof *prof;
 | |
| 	int tun, i;
 | |
| 
 | |
| 	prof = ice_fdir_get_hw_prof(hw, blk, flow);
 | |
| 	if (!prof)
 | |
| 		return;
 | |
| 
 | |
| 	ice_fdir_erase_flow_from_hw(hw, blk, flow);
 | |
| 	for (i = 0; i < prof->cnt; i++)
 | |
| 		prof->vsi_h[i] = 0;
 | |
| 	for (tun = 0; tun < ICE_FD_HW_SEG_MAX; tun++) {
 | |
| 		if (!prof->fdir_seg[tun])
 | |
| 			continue;
 | |
| 		devm_kfree(ice_hw_to_dev(hw), prof->fdir_seg[tun]);
 | |
| 		prof->fdir_seg[tun] = NULL;
 | |
| 	}
 | |
| 	prof->cnt = 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_release_flows - release all flows in use for later replay
 | |
|  * @hw: pointer to HW instance
 | |
|  */
 | |
| void ice_fdir_release_flows(struct ice_hw *hw)
 | |
| {
 | |
| 	int flow;
 | |
| 
 | |
| 	/* release Flow Director HW table entries */
 | |
| 	for (flow = 0; flow < ICE_FLTR_PTYPE_MAX; flow++)
 | |
| 		ice_fdir_erase_flow_from_hw(hw, ICE_BLK_FD, flow);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_replay_flows - replay HW Flow Director filter info
 | |
|  * @hw: pointer to HW instance
 | |
|  */
 | |
| void ice_fdir_replay_flows(struct ice_hw *hw)
 | |
| {
 | |
| 	int flow;
 | |
| 
 | |
| 	for (flow = 0; flow < ICE_FLTR_PTYPE_MAX; flow++) {
 | |
| 		int tun;
 | |
| 
 | |
| 		if (!hw->fdir_prof[flow] || !hw->fdir_prof[flow]->cnt)
 | |
| 			continue;
 | |
| 		for (tun = 0; tun < ICE_FD_HW_SEG_MAX; tun++) {
 | |
| 			struct ice_flow_prof *hw_prof;
 | |
| 			struct ice_fd_hw_prof *prof;
 | |
| 			u64 prof_id;
 | |
| 			int j;
 | |
| 
 | |
| 			prof = hw->fdir_prof[flow];
 | |
| 			prof_id = flow + tun * ICE_FLTR_PTYPE_MAX;
 | |
| 			ice_flow_add_prof(hw, ICE_BLK_FD, ICE_FLOW_RX, prof_id,
 | |
| 					  prof->fdir_seg[tun], TNL_SEG_CNT(tun),
 | |
| 					  &hw_prof);
 | |
| 			for (j = 0; j < prof->cnt; j++) {
 | |
| 				enum ice_flow_priority prio;
 | |
| 				u64 entry_h = 0;
 | |
| 				int err;
 | |
| 
 | |
| 				prio = ICE_FLOW_PRIO_NORMAL;
 | |
| 				err = ice_flow_add_entry(hw, ICE_BLK_FD,
 | |
| 							 prof_id,
 | |
| 							 prof->vsi_h[0],
 | |
| 							 prof->vsi_h[j],
 | |
| 							 prio, prof->fdir_seg,
 | |
| 							 &entry_h);
 | |
| 				if (err) {
 | |
| 					dev_err(ice_hw_to_dev(hw), "Could not replay Flow Director, flow type %d\n",
 | |
| 						flow);
 | |
| 					continue;
 | |
| 				}
 | |
| 				prof->entry_h[j][tun] = entry_h;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_parse_rx_flow_user_data - deconstruct user-defined data
 | |
|  * @fsp: pointer to ethtool Rx flow specification
 | |
|  * @data: pointer to userdef data structure for storage
 | |
|  *
 | |
|  * Returns 0 on success, negative error value on failure
 | |
|  */
 | |
| static int
 | |
| ice_parse_rx_flow_user_data(struct ethtool_rx_flow_spec *fsp,
 | |
| 			    struct ice_rx_flow_userdef *data)
 | |
| {
 | |
| 	u64 value, mask;
 | |
| 
 | |
| 	memset(data, 0, sizeof(*data));
 | |
| 	if (!(fsp->flow_type & FLOW_EXT))
 | |
| 		return 0;
 | |
| 
 | |
| 	value = be64_to_cpu(*((__force __be64 *)fsp->h_ext.data));
 | |
| 	mask = be64_to_cpu(*((__force __be64 *)fsp->m_ext.data));
 | |
| 	if (!mask)
 | |
| 		return 0;
 | |
| 
 | |
| #define ICE_USERDEF_FLEX_WORD_M	GENMASK_ULL(15, 0)
 | |
| #define ICE_USERDEF_FLEX_OFFS_S	16
 | |
| #define ICE_USERDEF_FLEX_OFFS_M	GENMASK_ULL(31, ICE_USERDEF_FLEX_OFFS_S)
 | |
| #define ICE_USERDEF_FLEX_FLTR_M	GENMASK_ULL(31, 0)
 | |
| 
 | |
| 	/* 0x1fe is the maximum value for offsets stored in the internal
 | |
| 	 * filtering tables.
 | |
| 	 */
 | |
| #define ICE_USERDEF_FLEX_MAX_OFFS_VAL 0x1fe
 | |
| 
 | |
| 	if (!ice_is_mask_valid(mask, ICE_USERDEF_FLEX_FLTR_M) ||
 | |
| 	    value > ICE_USERDEF_FLEX_FLTR_M)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	data->flex_word = value & ICE_USERDEF_FLEX_WORD_M;
 | |
| 	data->flex_offset = (value & ICE_USERDEF_FLEX_OFFS_M) >>
 | |
| 			     ICE_USERDEF_FLEX_OFFS_S;
 | |
| 	if (data->flex_offset > ICE_USERDEF_FLEX_MAX_OFFS_VAL)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	data->flex_fltr = true;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_num_avail_fltr - return the number of unused flow director filters
 | |
|  * @hw: pointer to hardware structure
 | |
|  * @vsi: software VSI structure
 | |
|  *
 | |
|  * There are 2 filter pools: guaranteed and best effort(shared). Each VSI can
 | |
|  * use filters from either pool. The guaranteed pool is divided between VSIs.
 | |
|  * The best effort filter pool is common to all VSIs and is a device shared
 | |
|  * resource pool. The number of filters available to this VSI is the sum of
 | |
|  * the VSIs guaranteed filter pool and the global available best effort
 | |
|  * filter pool.
 | |
|  *
 | |
|  * Returns the number of available flow director filters to this VSI
 | |
|  */
 | |
| static int ice_fdir_num_avail_fltr(struct ice_hw *hw, struct ice_vsi *vsi)
 | |
| {
 | |
| 	u16 vsi_num = ice_get_hw_vsi_num(hw, vsi->idx);
 | |
| 	u16 num_guar;
 | |
| 	u16 num_be;
 | |
| 
 | |
| 	/* total guaranteed filters assigned to this VSI */
 | |
| 	num_guar = vsi->num_gfltr;
 | |
| 
 | |
| 	/* minus the guaranteed filters programed by this VSI */
 | |
| 	num_guar -= (rd32(hw, VSIQF_FD_CNT(vsi_num)) &
 | |
| 		     VSIQF_FD_CNT_FD_GCNT_M) >> VSIQF_FD_CNT_FD_GCNT_S;
 | |
| 
 | |
| 	/* total global best effort filters */
 | |
| 	num_be = hw->func_caps.fd_fltr_best_effort;
 | |
| 
 | |
| 	/* minus the global best effort filters programmed */
 | |
| 	num_be -= (rd32(hw, GLQF_FD_CNT) & GLQF_FD_CNT_FD_BCNT_M) >>
 | |
| 		   GLQF_FD_CNT_FD_BCNT_S;
 | |
| 
 | |
| 	return num_guar + num_be;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_alloc_flow_prof - allocate FDir flow profile structure(s)
 | |
|  * @hw: HW structure containing the FDir flow profile structure(s)
 | |
|  * @flow: flow type to allocate the flow profile for
 | |
|  *
 | |
|  * Allocate the fdir_prof and fdir_prof[flow] if not already created. Return 0
 | |
|  * on success and negative on error.
 | |
|  */
 | |
| static int
 | |
| ice_fdir_alloc_flow_prof(struct ice_hw *hw, enum ice_fltr_ptype flow)
 | |
| {
 | |
| 	if (!hw)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (!hw->fdir_prof) {
 | |
| 		hw->fdir_prof = devm_kcalloc(ice_hw_to_dev(hw),
 | |
| 					     ICE_FLTR_PTYPE_MAX,
 | |
| 					     sizeof(*hw->fdir_prof),
 | |
| 					     GFP_KERNEL);
 | |
| 		if (!hw->fdir_prof)
 | |
| 			return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	if (!hw->fdir_prof[flow]) {
 | |
| 		hw->fdir_prof[flow] = devm_kzalloc(ice_hw_to_dev(hw),
 | |
| 						   sizeof(**hw->fdir_prof),
 | |
| 						   GFP_KERNEL);
 | |
| 		if (!hw->fdir_prof[flow])
 | |
| 			return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_set_hw_fltr_rule - Configure HW tables to generate a FDir rule
 | |
|  * @pf: pointer to the PF structure
 | |
|  * @seg: protocol header description pointer
 | |
|  * @flow: filter enum
 | |
|  * @tun: FDir segment to program
 | |
|  */
 | |
| static int
 | |
| ice_fdir_set_hw_fltr_rule(struct ice_pf *pf, struct ice_flow_seg_info *seg,
 | |
| 			  enum ice_fltr_ptype flow, enum ice_fd_hw_seg tun)
 | |
| {
 | |
| 	struct device *dev = ice_pf_to_dev(pf);
 | |
| 	struct ice_vsi *main_vsi, *ctrl_vsi;
 | |
| 	struct ice_flow_seg_info *old_seg;
 | |
| 	struct ice_flow_prof *prof = NULL;
 | |
| 	struct ice_fd_hw_prof *hw_prof;
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	enum ice_status status;
 | |
| 	u64 entry1_h = 0;
 | |
| 	u64 entry2_h = 0;
 | |
| 	u64 prof_id;
 | |
| 	int err;
 | |
| 
 | |
| 	main_vsi = ice_get_main_vsi(pf);
 | |
| 	if (!main_vsi)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	ctrl_vsi = ice_get_ctrl_vsi(pf);
 | |
| 	if (!ctrl_vsi)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	err = ice_fdir_alloc_flow_prof(hw, flow);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	hw_prof = hw->fdir_prof[flow];
 | |
| 	old_seg = hw_prof->fdir_seg[tun];
 | |
| 	if (old_seg) {
 | |
| 		/* This flow_type already has a changed input set.
 | |
| 		 * If it matches the requested input set then we are
 | |
| 		 * done. Or, if it's different then it's an error.
 | |
| 		 */
 | |
| 		if (!memcmp(old_seg, seg, sizeof(*seg)))
 | |
| 			return -EEXIST;
 | |
| 
 | |
| 		/* if there are FDir filters using this flow,
 | |
| 		 * then return error.
 | |
| 		 */
 | |
| 		if (hw->fdir_fltr_cnt[flow]) {
 | |
| 			dev_err(dev, "Failed to add filter.  Flow director filters on each port must have the same input set.\n");
 | |
| 			return -EINVAL;
 | |
| 		}
 | |
| 
 | |
| 		if (ice_is_arfs_using_perfect_flow(hw, flow)) {
 | |
| 			dev_err(dev, "aRFS using perfect flow type %d, cannot change input set\n",
 | |
| 				flow);
 | |
| 			return -EINVAL;
 | |
| 		}
 | |
| 
 | |
| 		/* remove HW filter definition */
 | |
| 		ice_fdir_rem_flow(hw, ICE_BLK_FD, flow);
 | |
| 	}
 | |
| 
 | |
| 	/* Adding a profile, but there is only one header supported.
 | |
| 	 * That is the final parameters are 1 header (segment), no
 | |
| 	 * actions (NULL) and zero actions 0.
 | |
| 	 */
 | |
| 	prof_id = flow + tun * ICE_FLTR_PTYPE_MAX;
 | |
| 	status = ice_flow_add_prof(hw, ICE_BLK_FD, ICE_FLOW_RX, prof_id, seg,
 | |
| 				   TNL_SEG_CNT(tun), &prof);
 | |
| 	if (status)
 | |
| 		return ice_status_to_errno(status);
 | |
| 	status = ice_flow_add_entry(hw, ICE_BLK_FD, prof_id, main_vsi->idx,
 | |
| 				    main_vsi->idx, ICE_FLOW_PRIO_NORMAL,
 | |
| 				    seg, &entry1_h);
 | |
| 	if (status) {
 | |
| 		err = ice_status_to_errno(status);
 | |
| 		goto err_prof;
 | |
| 	}
 | |
| 	status = ice_flow_add_entry(hw, ICE_BLK_FD, prof_id, main_vsi->idx,
 | |
| 				    ctrl_vsi->idx, ICE_FLOW_PRIO_NORMAL,
 | |
| 				    seg, &entry2_h);
 | |
| 	if (status) {
 | |
| 		err = ice_status_to_errno(status);
 | |
| 		goto err_entry;
 | |
| 	}
 | |
| 
 | |
| 	hw_prof->fdir_seg[tun] = seg;
 | |
| 	hw_prof->entry_h[0][tun] = entry1_h;
 | |
| 	hw_prof->entry_h[1][tun] = entry2_h;
 | |
| 	hw_prof->vsi_h[0] = main_vsi->idx;
 | |
| 	hw_prof->vsi_h[1] = ctrl_vsi->idx;
 | |
| 	if (!hw_prof->cnt)
 | |
| 		hw_prof->cnt = 2;
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_entry:
 | |
| 	ice_rem_prof_id_flow(hw, ICE_BLK_FD,
 | |
| 			     ice_get_hw_vsi_num(hw, main_vsi->idx), prof_id);
 | |
| 	ice_flow_rem_entry(hw, ICE_BLK_FD, entry1_h);
 | |
| err_prof:
 | |
| 	ice_flow_rem_prof(hw, ICE_BLK_FD, prof_id);
 | |
| 	dev_err(dev, "Failed to add filter.  Flow director filters on each port must have the same input set.\n");
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_init_fdir_seg
 | |
|  * @seg: flow segment for programming
 | |
|  * @l3_proto: ICE_FLOW_SEG_HDR_IPV4 or ICE_FLOW_SEG_HDR_IPV6
 | |
|  * @l4_proto: ICE_FLOW_SEG_HDR_TCP or ICE_FLOW_SEG_HDR_UDP
 | |
|  *
 | |
|  * Set the configuration for perfect filters to the provided flow segment for
 | |
|  * programming the HW filter. This is to be called only when initializing
 | |
|  * filters as this function it assumes no filters exist.
 | |
|  */
 | |
| static int
 | |
| ice_set_init_fdir_seg(struct ice_flow_seg_info *seg,
 | |
| 		      enum ice_flow_seg_hdr l3_proto,
 | |
| 		      enum ice_flow_seg_hdr l4_proto)
 | |
| {
 | |
| 	enum ice_flow_field src_addr, dst_addr, src_port, dst_port;
 | |
| 
 | |
| 	if (!seg)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (l3_proto == ICE_FLOW_SEG_HDR_IPV4) {
 | |
| 		src_addr = ICE_FLOW_FIELD_IDX_IPV4_SA;
 | |
| 		dst_addr = ICE_FLOW_FIELD_IDX_IPV4_DA;
 | |
| 	} else if (l3_proto == ICE_FLOW_SEG_HDR_IPV6) {
 | |
| 		src_addr = ICE_FLOW_FIELD_IDX_IPV6_SA;
 | |
| 		dst_addr = ICE_FLOW_FIELD_IDX_IPV6_DA;
 | |
| 	} else {
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	if (l4_proto == ICE_FLOW_SEG_HDR_TCP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_TCP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_TCP_DST_PORT;
 | |
| 	} else if (l4_proto == ICE_FLOW_SEG_HDR_UDP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_UDP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_UDP_DST_PORT;
 | |
| 	} else {
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	ICE_FLOW_SET_HDRS(seg, l3_proto | l4_proto);
 | |
| 
 | |
| 	/* IP source address */
 | |
| 	ice_flow_set_fld(seg, src_addr, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 			 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 
 | |
| 	/* IP destination address */
 | |
| 	ice_flow_set_fld(seg, dst_addr, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 			 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 
 | |
| 	/* Layer 4 source port */
 | |
| 	ice_flow_set_fld(seg, src_port, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 			 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 
 | |
| 	/* Layer 4 destination port */
 | |
| 	ice_flow_set_fld(seg, dst_port, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 			 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_create_init_fdir_rule
 | |
|  * @pf: PF structure
 | |
|  * @flow: filter enum
 | |
|  *
 | |
|  * Return error value or 0 on success.
 | |
|  */
 | |
| static int
 | |
| ice_create_init_fdir_rule(struct ice_pf *pf, enum ice_fltr_ptype flow)
 | |
| {
 | |
| 	struct ice_flow_seg_info *seg, *tun_seg;
 | |
| 	struct device *dev = ice_pf_to_dev(pf);
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	int ret;
 | |
| 
 | |
| 	/* if there is already a filter rule for kind return -EINVAL */
 | |
| 	if (hw->fdir_prof && hw->fdir_prof[flow] &&
 | |
| 	    hw->fdir_prof[flow]->fdir_seg[0])
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	seg = devm_kzalloc(dev, sizeof(*seg), GFP_KERNEL);
 | |
| 	if (!seg)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	tun_seg = devm_kzalloc(dev, sizeof(*seg) * ICE_FD_HW_SEG_MAX,
 | |
| 			       GFP_KERNEL);
 | |
| 	if (!tun_seg) {
 | |
| 		devm_kfree(dev, seg);
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	if (flow == ICE_FLTR_PTYPE_NONF_IPV4_TCP)
 | |
| 		ret = ice_set_init_fdir_seg(seg, ICE_FLOW_SEG_HDR_IPV4,
 | |
| 					    ICE_FLOW_SEG_HDR_TCP);
 | |
| 	else if (flow == ICE_FLTR_PTYPE_NONF_IPV4_UDP)
 | |
| 		ret = ice_set_init_fdir_seg(seg, ICE_FLOW_SEG_HDR_IPV4,
 | |
| 					    ICE_FLOW_SEG_HDR_UDP);
 | |
| 	else if (flow == ICE_FLTR_PTYPE_NONF_IPV6_TCP)
 | |
| 		ret = ice_set_init_fdir_seg(seg, ICE_FLOW_SEG_HDR_IPV6,
 | |
| 					    ICE_FLOW_SEG_HDR_TCP);
 | |
| 	else if (flow == ICE_FLTR_PTYPE_NONF_IPV6_UDP)
 | |
| 		ret = ice_set_init_fdir_seg(seg, ICE_FLOW_SEG_HDR_IPV6,
 | |
| 					    ICE_FLOW_SEG_HDR_UDP);
 | |
| 	else
 | |
| 		ret = -EINVAL;
 | |
| 	if (ret)
 | |
| 		goto err_exit;
 | |
| 
 | |
| 	/* add filter for outer headers */
 | |
| 	ret = ice_fdir_set_hw_fltr_rule(pf, seg, flow, ICE_FD_HW_SEG_NON_TUN);
 | |
| 	if (ret)
 | |
| 		/* could not write filter, free memory */
 | |
| 		goto err_exit;
 | |
| 
 | |
| 	/* make tunneled filter HW entries if possible */
 | |
| 	memcpy(&tun_seg[1], seg, sizeof(*seg));
 | |
| 	ret = ice_fdir_set_hw_fltr_rule(pf, tun_seg, flow, ICE_FD_HW_SEG_TUN);
 | |
| 	if (ret)
 | |
| 		/* could not write tunnel filter, but outer header filter
 | |
| 		 * exists
 | |
| 		 */
 | |
| 		devm_kfree(dev, tun_seg);
 | |
| 
 | |
| 	set_bit(flow, hw->fdir_perfect_fltr);
 | |
| 	return ret;
 | |
| err_exit:
 | |
| 	devm_kfree(dev, tun_seg);
 | |
| 	devm_kfree(dev, seg);
 | |
| 
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_fdir_ip4_seg
 | |
|  * @seg: flow segment for programming
 | |
|  * @tcp_ip4_spec: mask data from ethtool
 | |
|  * @l4_proto: Layer 4 protocol to program
 | |
|  * @perfect_fltr: only valid on success; returns true if perfect filter,
 | |
|  *		  false if not
 | |
|  *
 | |
|  * Set the mask data into the flow segment to be used to program HW
 | |
|  * table based on provided L4 protocol for IPv4
 | |
|  */
 | |
| static int
 | |
| ice_set_fdir_ip4_seg(struct ice_flow_seg_info *seg,
 | |
| 		     struct ethtool_tcpip4_spec *tcp_ip4_spec,
 | |
| 		     enum ice_flow_seg_hdr l4_proto, bool *perfect_fltr)
 | |
| {
 | |
| 	enum ice_flow_field src_port, dst_port;
 | |
| 
 | |
| 	/* make sure we don't have any empty rule */
 | |
| 	if (!tcp_ip4_spec->psrc && !tcp_ip4_spec->ip4src &&
 | |
| 	    !tcp_ip4_spec->pdst && !tcp_ip4_spec->ip4dst)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	/* filtering on TOS not supported */
 | |
| 	if (tcp_ip4_spec->tos)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (l4_proto == ICE_FLOW_SEG_HDR_TCP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_TCP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_TCP_DST_PORT;
 | |
| 	} else if (l4_proto == ICE_FLOW_SEG_HDR_UDP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_UDP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_UDP_DST_PORT;
 | |
| 	} else if (l4_proto == ICE_FLOW_SEG_HDR_SCTP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_SCTP_DST_PORT;
 | |
| 	} else {
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| 
 | |
| 	*perfect_fltr = true;
 | |
| 	ICE_FLOW_SET_HDRS(seg, ICE_FLOW_SEG_HDR_IPV4 | l4_proto);
 | |
| 
 | |
| 	/* IP source address */
 | |
| 	if (tcp_ip4_spec->ip4src == htonl(0xFFFFFFFF))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV4_SA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!tcp_ip4_spec->ip4src)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* IP destination address */
 | |
| 	if (tcp_ip4_spec->ip4dst == htonl(0xFFFFFFFF))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV4_DA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!tcp_ip4_spec->ip4dst)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* Layer 4 source port */
 | |
| 	if (tcp_ip4_spec->psrc == htons(0xFFFF))
 | |
| 		ice_flow_set_fld(seg, src_port, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 false);
 | |
| 	else if (!tcp_ip4_spec->psrc)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* Layer 4 destination port */
 | |
| 	if (tcp_ip4_spec->pdst == htons(0xFFFF))
 | |
| 		ice_flow_set_fld(seg, dst_port, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 false);
 | |
| 	else if (!tcp_ip4_spec->pdst)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_fdir_ip4_usr_seg
 | |
|  * @seg: flow segment for programming
 | |
|  * @usr_ip4_spec: ethtool userdef packet offset
 | |
|  * @perfect_fltr: only valid on success; returns true if perfect filter,
 | |
|  *		  false if not
 | |
|  *
 | |
|  * Set the offset data into the flow segment to be used to program HW
 | |
|  * table for IPv4
 | |
|  */
 | |
| static int
 | |
| ice_set_fdir_ip4_usr_seg(struct ice_flow_seg_info *seg,
 | |
| 			 struct ethtool_usrip4_spec *usr_ip4_spec,
 | |
| 			 bool *perfect_fltr)
 | |
| {
 | |
| 	/* first 4 bytes of Layer 4 header */
 | |
| 	if (usr_ip4_spec->l4_4_bytes)
 | |
| 		return -EINVAL;
 | |
| 	if (usr_ip4_spec->tos)
 | |
| 		return -EINVAL;
 | |
| 	if (usr_ip4_spec->ip_ver)
 | |
| 		return -EINVAL;
 | |
| 	/* Filtering on Layer 4 protocol not supported */
 | |
| 	if (usr_ip4_spec->proto)
 | |
| 		return -EOPNOTSUPP;
 | |
| 	/* empty rules are not valid */
 | |
| 	if (!usr_ip4_spec->ip4src && !usr_ip4_spec->ip4dst)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	*perfect_fltr = true;
 | |
| 	ICE_FLOW_SET_HDRS(seg, ICE_FLOW_SEG_HDR_IPV4);
 | |
| 
 | |
| 	/* IP source address */
 | |
| 	if (usr_ip4_spec->ip4src == htonl(0xFFFFFFFF))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV4_SA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!usr_ip4_spec->ip4src)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* IP destination address */
 | |
| 	if (usr_ip4_spec->ip4dst == htonl(0xFFFFFFFF))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV4_DA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!usr_ip4_spec->ip4dst)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_fdir_ip6_seg
 | |
|  * @seg: flow segment for programming
 | |
|  * @tcp_ip6_spec: mask data from ethtool
 | |
|  * @l4_proto: Layer 4 protocol to program
 | |
|  * @perfect_fltr: only valid on success; returns true if perfect filter,
 | |
|  *		  false if not
 | |
|  *
 | |
|  * Set the mask data into the flow segment to be used to program HW
 | |
|  * table based on provided L4 protocol for IPv6
 | |
|  */
 | |
| static int
 | |
| ice_set_fdir_ip6_seg(struct ice_flow_seg_info *seg,
 | |
| 		     struct ethtool_tcpip6_spec *tcp_ip6_spec,
 | |
| 		     enum ice_flow_seg_hdr l4_proto, bool *perfect_fltr)
 | |
| {
 | |
| 	enum ice_flow_field src_port, dst_port;
 | |
| 
 | |
| 	/* make sure we don't have any empty rule */
 | |
| 	if (!memcmp(tcp_ip6_spec->ip6src, &zero_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)) &&
 | |
| 	    !memcmp(tcp_ip6_spec->ip6dst, &zero_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)) &&
 | |
| 	    !tcp_ip6_spec->psrc && !tcp_ip6_spec->pdst)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	/* filtering on TC not supported */
 | |
| 	if (tcp_ip6_spec->tclass)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (l4_proto == ICE_FLOW_SEG_HDR_TCP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_TCP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_TCP_DST_PORT;
 | |
| 	} else if (l4_proto == ICE_FLOW_SEG_HDR_UDP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_UDP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_UDP_DST_PORT;
 | |
| 	} else if (l4_proto == ICE_FLOW_SEG_HDR_SCTP) {
 | |
| 		src_port = ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT;
 | |
| 		dst_port = ICE_FLOW_FIELD_IDX_SCTP_DST_PORT;
 | |
| 	} else {
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	*perfect_fltr = true;
 | |
| 	ICE_FLOW_SET_HDRS(seg, ICE_FLOW_SEG_HDR_IPV6 | l4_proto);
 | |
| 
 | |
| 	if (!memcmp(tcp_ip6_spec->ip6src, &full_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV6_SA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!memcmp(tcp_ip6_spec->ip6src, &zero_ipv6_addr_mask,
 | |
| 			 sizeof(struct in6_addr)))
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (!memcmp(tcp_ip6_spec->ip6dst, &full_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV6_DA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!memcmp(tcp_ip6_spec->ip6dst, &zero_ipv6_addr_mask,
 | |
| 			 sizeof(struct in6_addr)))
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* Layer 4 source port */
 | |
| 	if (tcp_ip6_spec->psrc == htons(0xFFFF))
 | |
| 		ice_flow_set_fld(seg, src_port, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 false);
 | |
| 	else if (!tcp_ip6_spec->psrc)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* Layer 4 destination port */
 | |
| 	if (tcp_ip6_spec->pdst == htons(0xFFFF))
 | |
| 		ice_flow_set_fld(seg, dst_port, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 false);
 | |
| 	else if (!tcp_ip6_spec->pdst)
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_fdir_ip6_usr_seg
 | |
|  * @seg: flow segment for programming
 | |
|  * @usr_ip6_spec: ethtool userdef packet offset
 | |
|  * @perfect_fltr: only valid on success; returns true if perfect filter,
 | |
|  *		  false if not
 | |
|  *
 | |
|  * Set the offset data into the flow segment to be used to program HW
 | |
|  * table for IPv6
 | |
|  */
 | |
| static int
 | |
| ice_set_fdir_ip6_usr_seg(struct ice_flow_seg_info *seg,
 | |
| 			 struct ethtool_usrip6_spec *usr_ip6_spec,
 | |
| 			 bool *perfect_fltr)
 | |
| {
 | |
| 	/* filtering on Layer 4 bytes not supported */
 | |
| 	if (usr_ip6_spec->l4_4_bytes)
 | |
| 		return -EOPNOTSUPP;
 | |
| 	/* filtering on TC not supported */
 | |
| 	if (usr_ip6_spec->tclass)
 | |
| 		return -EOPNOTSUPP;
 | |
| 	/* filtering on Layer 4 protocol not supported */
 | |
| 	if (usr_ip6_spec->l4_proto)
 | |
| 		return -EOPNOTSUPP;
 | |
| 	/* empty rules are not valid */
 | |
| 	if (!memcmp(usr_ip6_spec->ip6src, &zero_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)) &&
 | |
| 	    !memcmp(usr_ip6_spec->ip6dst, &zero_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)))
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	*perfect_fltr = true;
 | |
| 	ICE_FLOW_SET_HDRS(seg, ICE_FLOW_SEG_HDR_IPV6);
 | |
| 
 | |
| 	if (!memcmp(usr_ip6_spec->ip6src, &full_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV6_SA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!memcmp(usr_ip6_spec->ip6src, &zero_ipv6_addr_mask,
 | |
| 			 sizeof(struct in6_addr)))
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (!memcmp(usr_ip6_spec->ip6dst, &full_ipv6_addr_mask,
 | |
| 		    sizeof(struct in6_addr)))
 | |
| 		ice_flow_set_fld(seg, ICE_FLOW_FIELD_IDX_IPV6_DA,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				 ICE_FLOW_FLD_OFF_INVAL, false);
 | |
| 	else if (!memcmp(usr_ip6_spec->ip6dst, &zero_ipv6_addr_mask,
 | |
| 			 sizeof(struct in6_addr)))
 | |
| 		*perfect_fltr = false;
 | |
| 	else
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_cfg_fdir_xtrct_seq - Configure extraction sequence for the given filter
 | |
|  * @pf: PF structure
 | |
|  * @fsp: pointer to ethtool Rx flow specification
 | |
|  * @user: user defined data from flow specification
 | |
|  *
 | |
|  * Returns 0 on success.
 | |
|  */
 | |
| static int
 | |
| ice_cfg_fdir_xtrct_seq(struct ice_pf *pf, struct ethtool_rx_flow_spec *fsp,
 | |
| 		       struct ice_rx_flow_userdef *user)
 | |
| {
 | |
| 	struct ice_flow_seg_info *seg, *tun_seg;
 | |
| 	struct device *dev = ice_pf_to_dev(pf);
 | |
| 	enum ice_fltr_ptype fltr_idx;
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	bool perfect_filter;
 | |
| 	int ret;
 | |
| 
 | |
| 	seg = devm_kzalloc(dev, sizeof(*seg), GFP_KERNEL);
 | |
| 	if (!seg)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	tun_seg = devm_kzalloc(dev, sizeof(*seg) * ICE_FD_HW_SEG_MAX,
 | |
| 			       GFP_KERNEL);
 | |
| 	if (!tun_seg) {
 | |
| 		devm_kfree(dev, seg);
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	switch (fsp->flow_type & ~FLOW_EXT) {
 | |
| 	case TCP_V4_FLOW:
 | |
| 		ret = ice_set_fdir_ip4_seg(seg, &fsp->m_u.tcp_ip4_spec,
 | |
| 					   ICE_FLOW_SEG_HDR_TCP,
 | |
| 					   &perfect_filter);
 | |
| 		break;
 | |
| 	case UDP_V4_FLOW:
 | |
| 		ret = ice_set_fdir_ip4_seg(seg, &fsp->m_u.tcp_ip4_spec,
 | |
| 					   ICE_FLOW_SEG_HDR_UDP,
 | |
| 					   &perfect_filter);
 | |
| 		break;
 | |
| 	case SCTP_V4_FLOW:
 | |
| 		ret = ice_set_fdir_ip4_seg(seg, &fsp->m_u.tcp_ip4_spec,
 | |
| 					   ICE_FLOW_SEG_HDR_SCTP,
 | |
| 					   &perfect_filter);
 | |
| 		break;
 | |
| 	case IPV4_USER_FLOW:
 | |
| 		ret = ice_set_fdir_ip4_usr_seg(seg, &fsp->m_u.usr_ip4_spec,
 | |
| 					       &perfect_filter);
 | |
| 		break;
 | |
| 	case TCP_V6_FLOW:
 | |
| 		ret = ice_set_fdir_ip6_seg(seg, &fsp->m_u.tcp_ip6_spec,
 | |
| 					   ICE_FLOW_SEG_HDR_TCP,
 | |
| 					   &perfect_filter);
 | |
| 		break;
 | |
| 	case UDP_V6_FLOW:
 | |
| 		ret = ice_set_fdir_ip6_seg(seg, &fsp->m_u.tcp_ip6_spec,
 | |
| 					   ICE_FLOW_SEG_HDR_UDP,
 | |
| 					   &perfect_filter);
 | |
| 		break;
 | |
| 	case SCTP_V6_FLOW:
 | |
| 		ret = ice_set_fdir_ip6_seg(seg, &fsp->m_u.tcp_ip6_spec,
 | |
| 					   ICE_FLOW_SEG_HDR_SCTP,
 | |
| 					   &perfect_filter);
 | |
| 		break;
 | |
| 	case IPV6_USER_FLOW:
 | |
| 		ret = ice_set_fdir_ip6_usr_seg(seg, &fsp->m_u.usr_ip6_spec,
 | |
| 					       &perfect_filter);
 | |
| 		break;
 | |
| 	default:
 | |
| 		ret = -EINVAL;
 | |
| 	}
 | |
| 	if (ret)
 | |
| 		goto err_exit;
 | |
| 
 | |
| 	/* tunnel segments are shifted up one. */
 | |
| 	memcpy(&tun_seg[1], seg, sizeof(*seg));
 | |
| 
 | |
| 	if (user && user->flex_fltr) {
 | |
| 		perfect_filter = false;
 | |
| 		ice_flow_add_fld_raw(seg, user->flex_offset,
 | |
| 				     ICE_FLTR_PRGM_FLEX_WORD_SIZE,
 | |
| 				     ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				     ICE_FLOW_FLD_OFF_INVAL);
 | |
| 		ice_flow_add_fld_raw(&tun_seg[1], user->flex_offset,
 | |
| 				     ICE_FLTR_PRGM_FLEX_WORD_SIZE,
 | |
| 				     ICE_FLOW_FLD_OFF_INVAL,
 | |
| 				     ICE_FLOW_FLD_OFF_INVAL);
 | |
| 	}
 | |
| 
 | |
| 	fltr_idx = ice_ethtool_flow_to_fltr(fsp->flow_type & ~FLOW_EXT);
 | |
| 
 | |
| 	assign_bit(fltr_idx, hw->fdir_perfect_fltr, perfect_filter);
 | |
| 
 | |
| 	/* add filter for outer headers */
 | |
| 	ret = ice_fdir_set_hw_fltr_rule(pf, seg, fltr_idx,
 | |
| 					ICE_FD_HW_SEG_NON_TUN);
 | |
| 	if (ret == -EEXIST) {
 | |
| 		/* Rule already exists, free memory and count as success */
 | |
| 		ret = 0;
 | |
| 		goto err_exit;
 | |
| 	} else if (ret) {
 | |
| 		/* could not write filter, free memory */
 | |
| 		goto err_exit;
 | |
| 	}
 | |
| 
 | |
| 	/* make tunneled filter HW entries if possible */
 | |
| 	memcpy(&tun_seg[1], seg, sizeof(*seg));
 | |
| 	ret = ice_fdir_set_hw_fltr_rule(pf, tun_seg, fltr_idx,
 | |
| 					ICE_FD_HW_SEG_TUN);
 | |
| 	if (ret == -EEXIST) {
 | |
| 		/* Rule already exists, free memory and count as success */
 | |
| 		devm_kfree(dev, tun_seg);
 | |
| 		ret = 0;
 | |
| 	} else if (ret) {
 | |
| 		/* could not write tunnel filter, but outer filter exists */
 | |
| 		devm_kfree(dev, tun_seg);
 | |
| 	}
 | |
| 
 | |
| 	return ret;
 | |
| 
 | |
| err_exit:
 | |
| 	devm_kfree(dev, tun_seg);
 | |
| 	devm_kfree(dev, seg);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_write_fltr - send a flow director filter to the hardware
 | |
|  * @pf: PF data structure
 | |
|  * @input: filter structure
 | |
|  * @add: true adds filter and false removed filter
 | |
|  * @is_tun: true adds inner filter on tunnel and false outer headers
 | |
|  *
 | |
|  * returns 0 on success and negative value on error
 | |
|  */
 | |
| int
 | |
| ice_fdir_write_fltr(struct ice_pf *pf, struct ice_fdir_fltr *input, bool add,
 | |
| 		    bool is_tun)
 | |
| {
 | |
| 	struct device *dev = ice_pf_to_dev(pf);
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	struct ice_fltr_desc desc;
 | |
| 	struct ice_vsi *ctrl_vsi;
 | |
| 	enum ice_status status;
 | |
| 	u8 *pkt, *frag_pkt;
 | |
| 	bool has_frag;
 | |
| 	int err;
 | |
| 
 | |
| 	ctrl_vsi = ice_get_ctrl_vsi(pf);
 | |
| 	if (!ctrl_vsi)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	pkt = devm_kzalloc(dev, ICE_FDIR_MAX_RAW_PKT_SIZE, GFP_KERNEL);
 | |
| 	if (!pkt)
 | |
| 		return -ENOMEM;
 | |
| 	frag_pkt = devm_kzalloc(dev, ICE_FDIR_MAX_RAW_PKT_SIZE, GFP_KERNEL);
 | |
| 	if (!frag_pkt) {
 | |
| 		err = -ENOMEM;
 | |
| 		goto err_free;
 | |
| 	}
 | |
| 
 | |
| 	ice_fdir_get_prgm_desc(hw, input, &desc, add);
 | |
| 	status = ice_fdir_get_gen_prgm_pkt(hw, input, pkt, false, is_tun);
 | |
| 	if (status) {
 | |
| 		err = ice_status_to_errno(status);
 | |
| 		goto err_free_all;
 | |
| 	}
 | |
| 	err = ice_prgm_fdir_fltr(ctrl_vsi, &desc, pkt);
 | |
| 	if (err)
 | |
| 		goto err_free_all;
 | |
| 
 | |
| 	/* repeat for fragment packet */
 | |
| 	has_frag = ice_fdir_has_frag(input->flow_type);
 | |
| 	if (has_frag) {
 | |
| 		/* does not return error */
 | |
| 		ice_fdir_get_prgm_desc(hw, input, &desc, add);
 | |
| 		status = ice_fdir_get_gen_prgm_pkt(hw, input, frag_pkt, true,
 | |
| 						   is_tun);
 | |
| 		if (status) {
 | |
| 			err = ice_status_to_errno(status);
 | |
| 			goto err_frag;
 | |
| 		}
 | |
| 		err = ice_prgm_fdir_fltr(ctrl_vsi, &desc, frag_pkt);
 | |
| 		if (err)
 | |
| 			goto err_frag;
 | |
| 	} else {
 | |
| 		devm_kfree(dev, frag_pkt);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_free_all:
 | |
| 	devm_kfree(dev, frag_pkt);
 | |
| err_free:
 | |
| 	devm_kfree(dev, pkt);
 | |
| 	return err;
 | |
| 
 | |
| err_frag:
 | |
| 	devm_kfree(dev, frag_pkt);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_write_all_fltr - send a flow director filter to the hardware
 | |
|  * @pf: PF data structure
 | |
|  * @input: filter structure
 | |
|  * @add: true adds filter and false removed filter
 | |
|  *
 | |
|  * returns 0 on success and negative value on error
 | |
|  */
 | |
| static int
 | |
| ice_fdir_write_all_fltr(struct ice_pf *pf, struct ice_fdir_fltr *input,
 | |
| 			bool add)
 | |
| {
 | |
| 	u16 port_num;
 | |
| 	int tun;
 | |
| 
 | |
| 	for (tun = 0; tun < ICE_FD_HW_SEG_MAX; tun++) {
 | |
| 		bool is_tun = tun == ICE_FD_HW_SEG_TUN;
 | |
| 		int err;
 | |
| 
 | |
| 		if (is_tun && !ice_get_open_tunnel_port(&pf->hw, &port_num))
 | |
| 			continue;
 | |
| 		err = ice_fdir_write_fltr(pf, input, add, is_tun);
 | |
| 		if (err)
 | |
| 			return err;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_replay_fltrs - replay filters from the HW filter list
 | |
|  * @pf: board private structure
 | |
|  */
 | |
| void ice_fdir_replay_fltrs(struct ice_pf *pf)
 | |
| {
 | |
| 	struct ice_fdir_fltr *f_rule;
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 
 | |
| 	list_for_each_entry(f_rule, &hw->fdir_list_head, fltr_node) {
 | |
| 		int err = ice_fdir_write_all_fltr(pf, f_rule, true);
 | |
| 
 | |
| 		if (err)
 | |
| 			dev_dbg(ice_pf_to_dev(pf), "Flow Director error %d, could not reprogram filter %d\n",
 | |
| 				err, f_rule->fltr_id);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_create_dflt_rules - create default perfect filters
 | |
|  * @pf: PF data structure
 | |
|  *
 | |
|  * Returns 0 for success or error.
 | |
|  */
 | |
| int ice_fdir_create_dflt_rules(struct ice_pf *pf)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	/* Create perfect TCP and UDP rules in hardware. */
 | |
| 	err = ice_create_init_fdir_rule(pf, ICE_FLTR_PTYPE_NONF_IPV4_TCP);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	err = ice_create_init_fdir_rule(pf, ICE_FLTR_PTYPE_NONF_IPV4_UDP);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	err = ice_create_init_fdir_rule(pf, ICE_FLTR_PTYPE_NONF_IPV6_TCP);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	err = ice_create_init_fdir_rule(pf, ICE_FLTR_PTYPE_NONF_IPV6_UDP);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_vsi_manage_fdir - turn on/off flow director
 | |
|  * @vsi: the VSI being changed
 | |
|  * @ena: boolean value indicating if this is an enable or disable request
 | |
|  */
 | |
| void ice_vsi_manage_fdir(struct ice_vsi *vsi, bool ena)
 | |
| {
 | |
| 	struct ice_fdir_fltr *f_rule, *tmp;
 | |
| 	struct ice_pf *pf = vsi->back;
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	enum ice_fltr_ptype flow;
 | |
| 
 | |
| 	if (ena) {
 | |
| 		set_bit(ICE_FLAG_FD_ENA, pf->flags);
 | |
| 		ice_fdir_create_dflt_rules(pf);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	mutex_lock(&hw->fdir_fltr_lock);
 | |
| 	if (!test_and_clear_bit(ICE_FLAG_FD_ENA, pf->flags))
 | |
| 		goto release_lock;
 | |
| 	list_for_each_entry_safe(f_rule, tmp, &hw->fdir_list_head, fltr_node) {
 | |
| 		/* ignore return value */
 | |
| 		ice_fdir_write_all_fltr(pf, f_rule, false);
 | |
| 		ice_fdir_update_cntrs(hw, f_rule->flow_type, false);
 | |
| 		list_del(&f_rule->fltr_node);
 | |
| 		devm_kfree(ice_hw_to_dev(hw), f_rule);
 | |
| 	}
 | |
| 
 | |
| 	if (hw->fdir_prof)
 | |
| 		for (flow = ICE_FLTR_PTYPE_NONF_NONE; flow < ICE_FLTR_PTYPE_MAX;
 | |
| 		     flow++)
 | |
| 			if (hw->fdir_prof[flow])
 | |
| 				ice_fdir_rem_flow(hw, ICE_BLK_FD, flow);
 | |
| 
 | |
| release_lock:
 | |
| 	mutex_unlock(&hw->fdir_fltr_lock);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_do_rem_flow - delete flow and possibly add perfect flow
 | |
|  * @pf: PF structure
 | |
|  * @flow_type: FDir flow type to release
 | |
|  */
 | |
| static void
 | |
| ice_fdir_do_rem_flow(struct ice_pf *pf, enum ice_fltr_ptype flow_type)
 | |
| {
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	bool need_perfect = false;
 | |
| 
 | |
| 	if (flow_type == ICE_FLTR_PTYPE_NONF_IPV4_TCP ||
 | |
| 	    flow_type == ICE_FLTR_PTYPE_NONF_IPV4_UDP ||
 | |
| 	    flow_type == ICE_FLTR_PTYPE_NONF_IPV6_TCP ||
 | |
| 	    flow_type == ICE_FLTR_PTYPE_NONF_IPV6_UDP)
 | |
| 		need_perfect = true;
 | |
| 
 | |
| 	if (need_perfect && test_bit(flow_type, hw->fdir_perfect_fltr))
 | |
| 		return;
 | |
| 
 | |
| 	ice_fdir_rem_flow(hw, ICE_BLK_FD, flow_type);
 | |
| 	if (need_perfect)
 | |
| 		ice_create_init_fdir_rule(pf, flow_type);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_update_list_entry - add or delete a filter from the filter list
 | |
|  * @pf: PF structure
 | |
|  * @input: filter structure
 | |
|  * @fltr_idx: ethtool index of filter to modify
 | |
|  *
 | |
|  * returns 0 on success and negative on errors
 | |
|  */
 | |
| static int
 | |
| ice_fdir_update_list_entry(struct ice_pf *pf, struct ice_fdir_fltr *input,
 | |
| 			   int fltr_idx)
 | |
| {
 | |
| 	struct ice_fdir_fltr *old_fltr;
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	int err = -ENOENT;
 | |
| 
 | |
| 	/* Do not update filters during reset */
 | |
| 	if (ice_is_reset_in_progress(pf->state))
 | |
| 		return -EBUSY;
 | |
| 
 | |
| 	old_fltr = ice_fdir_find_fltr_by_idx(hw, fltr_idx);
 | |
| 	if (old_fltr) {
 | |
| 		err = ice_fdir_write_all_fltr(pf, old_fltr, false);
 | |
| 		if (err)
 | |
| 			return err;
 | |
| 		ice_fdir_update_cntrs(hw, old_fltr->flow_type, false);
 | |
| 		if (!input && !hw->fdir_fltr_cnt[old_fltr->flow_type])
 | |
| 			/* we just deleted the last filter of flow_type so we
 | |
| 			 * should also delete the HW filter info.
 | |
| 			 */
 | |
| 			ice_fdir_do_rem_flow(pf, old_fltr->flow_type);
 | |
| 		list_del(&old_fltr->fltr_node);
 | |
| 		devm_kfree(ice_hw_to_dev(hw), old_fltr);
 | |
| 	}
 | |
| 	if (!input)
 | |
| 		return err;
 | |
| 	ice_fdir_list_add_fltr(hw, input);
 | |
| 	ice_fdir_update_cntrs(hw, input->flow_type, true);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_del_fdir_ethtool - delete Flow Director filter
 | |
|  * @vsi: pointer to target VSI
 | |
|  * @cmd: command to add or delete Flow Director filter
 | |
|  *
 | |
|  * Returns 0 on success and negative values for failure
 | |
|  */
 | |
| int ice_del_fdir_ethtool(struct ice_vsi *vsi, struct ethtool_rxnfc *cmd)
 | |
| {
 | |
| 	struct ethtool_rx_flow_spec *fsp =
 | |
| 		(struct ethtool_rx_flow_spec *)&cmd->fs;
 | |
| 	struct ice_pf *pf = vsi->back;
 | |
| 	struct ice_hw *hw = &pf->hw;
 | |
| 	int val;
 | |
| 
 | |
| 	if (!test_bit(ICE_FLAG_FD_ENA, pf->flags))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* Do not delete filters during reset */
 | |
| 	if (ice_is_reset_in_progress(pf->state)) {
 | |
| 		dev_err(ice_pf_to_dev(pf), "Device is resetting - deleting Flow Director filters not supported during reset\n");
 | |
| 		return -EBUSY;
 | |
| 	}
 | |
| 
 | |
| 	if (test_bit(ICE_FD_FLUSH_REQ, pf->state))
 | |
| 		return -EBUSY;
 | |
| 
 | |
| 	mutex_lock(&hw->fdir_fltr_lock);
 | |
| 	val = ice_fdir_update_list_entry(pf, NULL, fsp->location);
 | |
| 	mutex_unlock(&hw->fdir_fltr_lock);
 | |
| 
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_fdir_input_set - Set the input set for Flow Director
 | |
|  * @vsi: pointer to target VSI
 | |
|  * @fsp: pointer to ethtool Rx flow specification
 | |
|  * @input: filter structure
 | |
|  */
 | |
| static int
 | |
| ice_set_fdir_input_set(struct ice_vsi *vsi, struct ethtool_rx_flow_spec *fsp,
 | |
| 		       struct ice_fdir_fltr *input)
 | |
| {
 | |
| 	u16 dest_vsi, q_index = 0;
 | |
| 	struct ice_pf *pf;
 | |
| 	struct ice_hw *hw;
 | |
| 	int flow_type;
 | |
| 	u8 dest_ctl;
 | |
| 
 | |
| 	if (!vsi || !fsp || !input)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	pf = vsi->back;
 | |
| 	hw = &pf->hw;
 | |
| 
 | |
| 	dest_vsi = vsi->idx;
 | |
| 	if (fsp->ring_cookie == RX_CLS_FLOW_DISC) {
 | |
| 		dest_ctl = ICE_FLTR_PRGM_DESC_DEST_DROP_PKT;
 | |
| 	} else {
 | |
| 		u32 ring = ethtool_get_flow_spec_ring(fsp->ring_cookie);
 | |
| 		u8 vf = ethtool_get_flow_spec_ring_vf(fsp->ring_cookie);
 | |
| 
 | |
| 		if (vf) {
 | |
| 			dev_err(ice_pf_to_dev(pf), "Failed to add filter. Flow director filters are not supported on VF queues.\n");
 | |
| 			return -EINVAL;
 | |
| 		}
 | |
| 
 | |
| 		if (ring >= vsi->num_rxq)
 | |
| 			return -EINVAL;
 | |
| 
 | |
| 		dest_ctl = ICE_FLTR_PRGM_DESC_DEST_DIRECT_PKT_QINDEX;
 | |
| 		q_index = ring;
 | |
| 	}
 | |
| 
 | |
| 	input->fltr_id = fsp->location;
 | |
| 	input->q_index = q_index;
 | |
| 	flow_type = fsp->flow_type & ~FLOW_EXT;
 | |
| 
 | |
| 	input->dest_vsi = dest_vsi;
 | |
| 	input->dest_ctl = dest_ctl;
 | |
| 	input->fltr_status = ICE_FLTR_PRGM_DESC_FD_STATUS_FD_ID;
 | |
| 	input->cnt_index = ICE_FD_SB_STAT_IDX(hw->fd_ctr_base);
 | |
| 	input->flow_type = ice_ethtool_flow_to_fltr(flow_type);
 | |
| 
 | |
| 	if (fsp->flow_type & FLOW_EXT) {
 | |
| 		memcpy(input->ext_data.usr_def, fsp->h_ext.data,
 | |
| 		       sizeof(input->ext_data.usr_def));
 | |
| 		input->ext_data.vlan_type = fsp->h_ext.vlan_etype;
 | |
| 		input->ext_data.vlan_tag = fsp->h_ext.vlan_tci;
 | |
| 		memcpy(input->ext_mask.usr_def, fsp->m_ext.data,
 | |
| 		       sizeof(input->ext_mask.usr_def));
 | |
| 		input->ext_mask.vlan_type = fsp->m_ext.vlan_etype;
 | |
| 		input->ext_mask.vlan_tag = fsp->m_ext.vlan_tci;
 | |
| 	}
 | |
| 
 | |
| 	switch (flow_type) {
 | |
| 	case TCP_V4_FLOW:
 | |
| 	case UDP_V4_FLOW:
 | |
| 	case SCTP_V4_FLOW:
 | |
| 		input->ip.v4.dst_port = fsp->h_u.tcp_ip4_spec.pdst;
 | |
| 		input->ip.v4.src_port = fsp->h_u.tcp_ip4_spec.psrc;
 | |
| 		input->ip.v4.dst_ip = fsp->h_u.tcp_ip4_spec.ip4dst;
 | |
| 		input->ip.v4.src_ip = fsp->h_u.tcp_ip4_spec.ip4src;
 | |
| 		input->mask.v4.dst_port = fsp->m_u.tcp_ip4_spec.pdst;
 | |
| 		input->mask.v4.src_port = fsp->m_u.tcp_ip4_spec.psrc;
 | |
| 		input->mask.v4.dst_ip = fsp->m_u.tcp_ip4_spec.ip4dst;
 | |
| 		input->mask.v4.src_ip = fsp->m_u.tcp_ip4_spec.ip4src;
 | |
| 		break;
 | |
| 	case IPV4_USER_FLOW:
 | |
| 		input->ip.v4.dst_ip = fsp->h_u.usr_ip4_spec.ip4dst;
 | |
| 		input->ip.v4.src_ip = fsp->h_u.usr_ip4_spec.ip4src;
 | |
| 		input->ip.v4.l4_header = fsp->h_u.usr_ip4_spec.l4_4_bytes;
 | |
| 		input->ip.v4.proto = fsp->h_u.usr_ip4_spec.proto;
 | |
| 		input->ip.v4.ip_ver = fsp->h_u.usr_ip4_spec.ip_ver;
 | |
| 		input->ip.v4.tos = fsp->h_u.usr_ip4_spec.tos;
 | |
| 		input->mask.v4.dst_ip = fsp->m_u.usr_ip4_spec.ip4dst;
 | |
| 		input->mask.v4.src_ip = fsp->m_u.usr_ip4_spec.ip4src;
 | |
| 		input->mask.v4.l4_header = fsp->m_u.usr_ip4_spec.l4_4_bytes;
 | |
| 		input->mask.v4.proto = fsp->m_u.usr_ip4_spec.proto;
 | |
| 		input->mask.v4.ip_ver = fsp->m_u.usr_ip4_spec.ip_ver;
 | |
| 		input->mask.v4.tos = fsp->m_u.usr_ip4_spec.tos;
 | |
| 		break;
 | |
| 	case TCP_V6_FLOW:
 | |
| 	case UDP_V6_FLOW:
 | |
| 	case SCTP_V6_FLOW:
 | |
| 		memcpy(input->ip.v6.dst_ip, fsp->h_u.usr_ip6_spec.ip6dst,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(input->ip.v6.src_ip, fsp->h_u.usr_ip6_spec.ip6src,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		input->ip.v6.dst_port = fsp->h_u.tcp_ip6_spec.pdst;
 | |
| 		input->ip.v6.src_port = fsp->h_u.tcp_ip6_spec.psrc;
 | |
| 		input->ip.v6.tc = fsp->h_u.tcp_ip6_spec.tclass;
 | |
| 		memcpy(input->mask.v6.dst_ip, fsp->m_u.tcp_ip6_spec.ip6dst,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(input->mask.v6.src_ip, fsp->m_u.tcp_ip6_spec.ip6src,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		input->mask.v6.dst_port = fsp->m_u.tcp_ip6_spec.pdst;
 | |
| 		input->mask.v6.src_port = fsp->m_u.tcp_ip6_spec.psrc;
 | |
| 		input->mask.v6.tc = fsp->m_u.tcp_ip6_spec.tclass;
 | |
| 		break;
 | |
| 	case IPV6_USER_FLOW:
 | |
| 		memcpy(input->ip.v6.dst_ip, fsp->h_u.usr_ip6_spec.ip6dst,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(input->ip.v6.src_ip, fsp->h_u.usr_ip6_spec.ip6src,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		input->ip.v6.l4_header = fsp->h_u.usr_ip6_spec.l4_4_bytes;
 | |
| 		input->ip.v6.tc = fsp->h_u.usr_ip6_spec.tclass;
 | |
| 
 | |
| 		/* if no protocol requested, use IPPROTO_NONE */
 | |
| 		if (!fsp->m_u.usr_ip6_spec.l4_proto)
 | |
| 			input->ip.v6.proto = IPPROTO_NONE;
 | |
| 		else
 | |
| 			input->ip.v6.proto = fsp->h_u.usr_ip6_spec.l4_proto;
 | |
| 
 | |
| 		memcpy(input->mask.v6.dst_ip, fsp->m_u.usr_ip6_spec.ip6dst,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		memcpy(input->mask.v6.src_ip, fsp->m_u.usr_ip6_spec.ip6src,
 | |
| 		       sizeof(struct in6_addr));
 | |
| 		input->mask.v6.l4_header = fsp->m_u.usr_ip6_spec.l4_4_bytes;
 | |
| 		input->mask.v6.tc = fsp->m_u.usr_ip6_spec.tclass;
 | |
| 		input->mask.v6.proto = fsp->m_u.usr_ip6_spec.l4_proto;
 | |
| 		break;
 | |
| 	default:
 | |
| 		/* not doing un-parsed flow types */
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_add_fdir_ethtool - Add/Remove Flow Director filter
 | |
|  * @vsi: pointer to target VSI
 | |
|  * @cmd: command to add or delete Flow Director filter
 | |
|  *
 | |
|  * Returns 0 on success and negative values for failure
 | |
|  */
 | |
| int ice_add_fdir_ethtool(struct ice_vsi *vsi, struct ethtool_rxnfc *cmd)
 | |
| {
 | |
| 	struct ice_rx_flow_userdef userdata;
 | |
| 	struct ethtool_rx_flow_spec *fsp;
 | |
| 	struct ice_fdir_fltr *input;
 | |
| 	struct device *dev;
 | |
| 	struct ice_pf *pf;
 | |
| 	struct ice_hw *hw;
 | |
| 	int fltrs_needed;
 | |
| 	u16 tunnel_port;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (!vsi)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	pf = vsi->back;
 | |
| 	hw = &pf->hw;
 | |
| 	dev = ice_pf_to_dev(pf);
 | |
| 
 | |
| 	if (!test_bit(ICE_FLAG_FD_ENA, pf->flags))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	/* Do not program filters during reset */
 | |
| 	if (ice_is_reset_in_progress(pf->state)) {
 | |
| 		dev_err(dev, "Device is resetting - adding Flow Director filters not supported during reset\n");
 | |
| 		return -EBUSY;
 | |
| 	}
 | |
| 
 | |
| 	fsp = (struct ethtool_rx_flow_spec *)&cmd->fs;
 | |
| 
 | |
| 	if (ice_parse_rx_flow_user_data(fsp, &userdata))
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (fsp->flow_type & FLOW_MAC_EXT)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	ret = ice_cfg_fdir_xtrct_seq(pf, fsp, &userdata);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	if (fsp->location >= ice_get_fdir_cnt_all(hw)) {
 | |
| 		dev_err(dev, "Failed to add filter.  The maximum number of flow director filters has been reached.\n");
 | |
| 		return -ENOSPC;
 | |
| 	}
 | |
| 
 | |
| 	/* return error if not an update and no available filters */
 | |
| 	fltrs_needed = ice_get_open_tunnel_port(hw, &tunnel_port) ? 2 : 1;
 | |
| 	if (!ice_fdir_find_fltr_by_idx(hw, fsp->location) &&
 | |
| 	    ice_fdir_num_avail_fltr(hw, pf->vsi[vsi->idx]) < fltrs_needed) {
 | |
| 		dev_err(dev, "Failed to add filter.  The maximum number of flow director filters has been reached.\n");
 | |
| 		return -ENOSPC;
 | |
| 	}
 | |
| 
 | |
| 	input = devm_kzalloc(dev, sizeof(*input), GFP_KERNEL);
 | |
| 	if (!input)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	ret = ice_set_fdir_input_set(vsi, fsp, input);
 | |
| 	if (ret)
 | |
| 		goto free_input;
 | |
| 
 | |
| 	mutex_lock(&hw->fdir_fltr_lock);
 | |
| 	if (ice_fdir_is_dup_fltr(hw, input)) {
 | |
| 		ret = -EINVAL;
 | |
| 		goto release_lock;
 | |
| 	}
 | |
| 
 | |
| 	if (userdata.flex_fltr) {
 | |
| 		input->flex_fltr = true;
 | |
| 		input->flex_word = cpu_to_be16(userdata.flex_word);
 | |
| 		input->flex_offset = userdata.flex_offset;
 | |
| 	}
 | |
| 
 | |
| 	input->cnt_ena = ICE_FXD_FLTR_QW0_STAT_ENA_PKTS;
 | |
| 	input->fdid_prio = ICE_FXD_FLTR_QW1_FDID_PRI_THREE;
 | |
| 	input->comp_report = ICE_FXD_FLTR_QW0_COMP_REPORT_SW_FAIL;
 | |
| 
 | |
| 	/* input struct is added to the HW filter list */
 | |
| 	ret = ice_fdir_update_list_entry(pf, input, fsp->location);
 | |
| 	if (ret)
 | |
| 		goto release_lock;
 | |
| 
 | |
| 	ret = ice_fdir_write_all_fltr(pf, input, true);
 | |
| 	if (ret)
 | |
| 		goto remove_sw_rule;
 | |
| 
 | |
| 	goto release_lock;
 | |
| 
 | |
| remove_sw_rule:
 | |
| 	ice_fdir_update_cntrs(hw, input->flow_type, false);
 | |
| 	list_del(&input->fltr_node);
 | |
| release_lock:
 | |
| 	mutex_unlock(&hw->fdir_fltr_lock);
 | |
| free_input:
 | |
| 	if (ret)
 | |
| 		devm_kfree(dev, input);
 | |
| 
 | |
| 	return ret;
 | |
| }
 |