diff options
Diffstat (limited to 'drivers/net/ethernet/intel/ice')
29 files changed, 5410 insertions, 487 deletions
| diff --git a/drivers/net/ethernet/intel/ice/Makefile b/drivers/net/ethernet/intel/ice/Makefile index 7cb829132d28..59544b0fc086 100644 --- a/drivers/net/ethernet/intel/ice/Makefile +++ b/drivers/net/ethernet/intel/ice/Makefile @@ -17,7 +17,8 @@ ice-y := ice_main.o	\  	 ice_lib.o	\  	 ice_txrx_lib.o	\  	 ice_txrx.o	\ -	 ice_flex_pipe.o	\ +	 ice_flex_pipe.o \ +	 ice_flow.o	\  	 ice_ethtool.o  ice-$(CONFIG_PCI_IOV) += ice_virtchnl_pf.o ice_sriov.o  ice-$(CONFIG_DCB) += ice_dcb.o ice_dcb_nl.o ice_dcb_lib.o diff --git a/drivers/net/ethernet/intel/ice/ice.h b/drivers/net/ethernet/intel/ice/ice.h index f972dce8aebb..cb10abb14e11 100644 --- a/drivers/net/ethernet/intel/ice/ice.h +++ b/drivers/net/ethernet/intel/ice/ice.h @@ -174,6 +174,8 @@ struct ice_sw {  	struct ice_pf *pf;  	u16 sw_id;		/* switch ID for this switch */  	u16 bridge_mode;	/* VEB/VEPA/Port Virtualizer */ +	struct ice_vsi *dflt_vsi;	/* default VSI for this switch */ +	u8 dflt_vsi_ena:1;	/* true if above dflt_vsi is enabled */  };  enum ice_state { @@ -275,6 +277,7 @@ struct ice_vsi {  	u8 current_isup:1;		 /* Sync 'link up' logging */  	u8 stat_offsets_loaded:1;  	u8 vlan_ena:1; +	u16 num_vlan;  	/* queue information */  	u8 tx_mapping_mode;		 /* ICE_MAP_MODE_[CONTIG|SCATTER] */ @@ -462,12 +465,13 @@ static inline void ice_set_ring_xdp(struct ice_ring *ring)  static inline struct xdp_umem *ice_xsk_umem(struct ice_ring *ring)  {  	struct xdp_umem **umems = ring->vsi->xsk_umems; -	int qid = ring->q_index; +	u16 qid = ring->q_index;  	if (ice_ring_is_xdp(ring))  		qid -= ring->vsi->num_xdp_txq; -	if (!umems || !umems[qid] || !ice_is_xdp_ena_vsi(ring->vsi)) +	if (qid >= ring->vsi->num_xsk_umems || !umems || !umems[qid] || +	    !ice_is_xdp_ena_vsi(ring->vsi))  		return NULL;  	return umems[qid]; diff --git a/drivers/net/ethernet/intel/ice/ice_adminq_cmd.h b/drivers/net/ethernet/intel/ice/ice_adminq_cmd.h index 5421fc413f94..4459bc564b11 100644 --- a/drivers/net/ethernet/intel/ice/ice_adminq_cmd.h +++ b/drivers/net/ethernet/intel/ice/ice_adminq_cmd.h @@ -232,6 +232,13 @@ struct ice_aqc_get_sw_cfg_resp {   */  #define ICE_AQC_RES_TYPE_VSI_LIST_REP			0x03  #define ICE_AQC_RES_TYPE_VSI_LIST_PRUNE			0x04 +#define ICE_AQC_RES_TYPE_HASH_PROF_BLDR_PROFID		0x60 +#define ICE_AQC_RES_TYPE_HASH_PROF_BLDR_TCAM		0x61 + +#define ICE_AQC_RES_TYPE_FLAG_SCAN_BOTTOM		BIT(12) +#define ICE_AQC_RES_TYPE_FLAG_IGNORE_INDEX		BIT(13) + +#define ICE_AQC_RES_TYPE_FLAG_DEDICATED			0x00  /* Allocate Resources command (indirect 0x0208)   * Free Resources command (indirect 0x0209) @@ -1849,6 +1856,7 @@ enum ice_adminq_opc {  	/* package commands */  	ice_aqc_opc_download_pkg			= 0x0C40, +	ice_aqc_opc_update_pkg				= 0x0C42,  	ice_aqc_opc_get_pkg_info_list			= 0x0C43,  	/* debug commands */ diff --git a/drivers/net/ethernet/intel/ice/ice_base.c b/drivers/net/ethernet/intel/ice/ice_base.c index 77d6a0291e97..d8e975cceb21 100644 --- a/drivers/net/ethernet/intel/ice/ice_base.c +++ b/drivers/net/ethernet/intel/ice/ice_base.c @@ -93,7 +93,8 @@ static int ice_pf_rxq_wait(struct ice_pf *pf, int pf_q, bool ena)   * @vsi: the VSI being configured   * @v_idx: index of the vector in the VSI struct   * - * We allocate one q_vector. If allocation fails we return -ENOMEM. + * We allocate one q_vector and set default value for ITR setting associated + * with this q_vector. If allocation fails we return -ENOMEM.   */  static int ice_vsi_alloc_q_vector(struct ice_vsi *vsi, int v_idx)  { @@ -108,6 +109,8 @@ static int ice_vsi_alloc_q_vector(struct ice_vsi *vsi, int v_idx)  	q_vector->vsi = vsi;  	q_vector->v_idx = v_idx; +	q_vector->tx.itr_setting = ICE_DFLT_TX_ITR; +	q_vector->rx.itr_setting = ICE_DFLT_RX_ITR;  	if (vsi->type == ICE_VSI_VF)  		goto out;  	/* only set affinity_mask if the CPU is online */ @@ -299,6 +302,7 @@ int ice_setup_rx_ctx(struct ice_ring *ring)  	if (ring->vsi->type == ICE_VSI_PF) {  		if (!xdp_rxq_info_is_reg(&ring->xdp_rxq)) +			/* coverity[check_return] */  			xdp_rxq_info_reg(&ring->xdp_rxq, ring->netdev,  					 ring->q_index); @@ -323,7 +327,9 @@ int ice_setup_rx_ctx(struct ice_ring *ring)  			dev_info(&vsi->back->pdev->dev, "Registered XDP mem model MEM_TYPE_ZERO_COPY on Rx ring %d\n",  				 ring->q_index);  		} else { +			ring->zca.free = NULL;  			if (!xdp_rxq_info_is_reg(&ring->xdp_rxq)) +				/* coverity[check_return] */  				xdp_rxq_info_reg(&ring->xdp_rxq,  						 ring->netdev,  						 ring->q_index); @@ -674,10 +680,6 @@ void ice_cfg_itr(struct ice_hw *hw, struct ice_q_vector *q_vector)  	if (q_vector->num_ring_rx) {  		struct ice_ring_container *rc = &q_vector->rx; -		/* if this value is set then don't overwrite with default */ -		if (!rc->itr_setting) -			rc->itr_setting = ICE_DFLT_RX_ITR; -  		rc->target_itr = ITR_TO_REG(rc->itr_setting);  		rc->next_update = jiffies + 1;  		rc->current_itr = rc->target_itr; @@ -688,10 +690,6 @@ void ice_cfg_itr(struct ice_hw *hw, struct ice_q_vector *q_vector)  	if (q_vector->num_ring_tx) {  		struct ice_ring_container *rc = &q_vector->tx; -		/* if this value is set then don't overwrite with default */ -		if (!rc->itr_setting) -			rc->itr_setting = ICE_DFLT_TX_ITR; -  		rc->target_itr = ITR_TO_REG(rc->itr_setting);  		rc->next_update = jiffies + 1;  		rc->current_itr = rc->target_itr; diff --git a/drivers/net/ethernet/intel/ice/ice_common.c b/drivers/net/ethernet/intel/ice/ice_common.c index fb1d930470c7..0207e28c2682 100644 --- a/drivers/net/ethernet/intel/ice/ice_common.c +++ b/drivers/net/ethernet/intel/ice/ice_common.c @@ -4,28 +4,10 @@  #include "ice_common.h"  #include "ice_sched.h"  #include "ice_adminq_cmd.h" +#include "ice_flow.h"  #define ICE_PF_RESET_WAIT_COUNT	200 -#define ICE_PROG_FLEX_ENTRY(hw, rxdid, mdid, idx) \ -	wr32((hw), GLFLXP_RXDID_FLX_WRD_##idx(rxdid), \ -	     ((ICE_RX_OPC_MDID << \ -	       GLFLXP_RXDID_FLX_WRD_##idx##_RXDID_OPCODE_S) & \ -	      GLFLXP_RXDID_FLX_WRD_##idx##_RXDID_OPCODE_M) | \ -	     (((mdid) << GLFLXP_RXDID_FLX_WRD_##idx##_PROT_MDID_S) & \ -	      GLFLXP_RXDID_FLX_WRD_##idx##_PROT_MDID_M)) - -#define ICE_PROG_FLG_ENTRY(hw, rxdid, flg_0, flg_1, flg_2, flg_3, idx) \ -	wr32((hw), GLFLXP_RXDID_FLAGS(rxdid, idx), \ -	     (((flg_0) << GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_S) & \ -	      GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_M) | \ -	     (((flg_1) << GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_1_S) & \ -	      GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_1_M) | \ -	     (((flg_2) << GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_2_S) & \ -	      GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_2_M) | \ -	     (((flg_3) << GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_3_S) & \ -	      GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_3_M)) -  /**   * ice_set_mac_type - Sets MAC type   * @hw: pointer to the HW structure @@ -348,88 +330,6 @@ ice_aq_get_link_info(struct ice_port_info *pi, bool ena_lse,  }  /** - * ice_init_flex_flags - * @hw: pointer to the hardware structure - * @prof_id: Rx Descriptor Builder profile ID - * - * Function to initialize Rx flex flags - */ -static void ice_init_flex_flags(struct ice_hw *hw, enum ice_rxdid prof_id) -{ -	u8 idx = 0; - -	/* Flex-flag fields (0-2) are programmed with FLG64 bits with layout: -	 * flexiflags0[5:0] - TCP flags, is_packet_fragmented, is_packet_UDP_GRE -	 * flexiflags1[3:0] - Not used for flag programming -	 * flexiflags2[7:0] - Tunnel and VLAN types -	 * 2 invalid fields in last index -	 */ -	switch (prof_id) { -	/* Rx flex flags are currently programmed for the NIC profiles only. -	 * Different flag bit programming configurations can be added per -	 * profile as needed. -	 */ -	case ICE_RXDID_FLEX_NIC: -	case ICE_RXDID_FLEX_NIC_2: -		ICE_PROG_FLG_ENTRY(hw, prof_id, ICE_FLG_PKT_FRG, -				   ICE_FLG_UDP_GRE, ICE_FLG_PKT_DSI, -				   ICE_FLG_FIN, idx++); -		/* flex flag 1 is not used for flexi-flag programming, skipping -		 * these four FLG64 bits. -		 */ -		ICE_PROG_FLG_ENTRY(hw, prof_id, ICE_FLG_SYN, ICE_FLG_RST, -				   ICE_FLG_PKT_DSI, ICE_FLG_PKT_DSI, idx++); -		ICE_PROG_FLG_ENTRY(hw, prof_id, ICE_FLG_PKT_DSI, -				   ICE_FLG_PKT_DSI, ICE_FLG_EVLAN_x8100, -				   ICE_FLG_EVLAN_x9100, idx++); -		ICE_PROG_FLG_ENTRY(hw, prof_id, ICE_FLG_VLAN_x8100, -				   ICE_FLG_TNL_VLAN, ICE_FLG_TNL_MAC, -				   ICE_FLG_TNL0, idx++); -		ICE_PROG_FLG_ENTRY(hw, prof_id, ICE_FLG_TNL1, ICE_FLG_TNL2, -				   ICE_FLG_PKT_DSI, ICE_FLG_PKT_DSI, idx); -		break; - -	default: -		ice_debug(hw, ICE_DBG_INIT, -			  "Flag programming for profile ID %d not supported\n", -			  prof_id); -	} -} - -/** - * ice_init_flex_flds - * @hw: pointer to the hardware structure - * @prof_id: Rx Descriptor Builder profile ID - * - * Function to initialize flex descriptors - */ -static void ice_init_flex_flds(struct ice_hw *hw, enum ice_rxdid prof_id) -{ -	enum ice_flex_rx_mdid mdid; - -	switch (prof_id) { -	case ICE_RXDID_FLEX_NIC: -	case ICE_RXDID_FLEX_NIC_2: -		ICE_PROG_FLEX_ENTRY(hw, prof_id, ICE_RX_MDID_HASH_LOW, 0); -		ICE_PROG_FLEX_ENTRY(hw, prof_id, ICE_RX_MDID_HASH_HIGH, 1); -		ICE_PROG_FLEX_ENTRY(hw, prof_id, ICE_RX_MDID_FLOW_ID_LOWER, 2); - -		mdid = (prof_id == ICE_RXDID_FLEX_NIC_2) ? -			ICE_RX_MDID_SRC_VSI : ICE_RX_MDID_FLOW_ID_HIGH; - -		ICE_PROG_FLEX_ENTRY(hw, prof_id, mdid, 3); - -		ice_init_flex_flags(hw, prof_id); -		break; - -	default: -		ice_debug(hw, ICE_DBG_INIT, -			  "Field init for profile ID %d not supported\n", -			  prof_id); -	} -} - -/**   * ice_init_fltr_mgmt_struct - initializes filter management list and locks   * @hw: pointer to the HW struct   */ @@ -882,9 +782,6 @@ enum ice_status ice_init_hw(struct ice_hw *hw)  	if (status)  		goto err_unroll_fltr_mgmt_struct; - -	ice_init_flex_flds(hw, ICE_RXDID_FLEX_NIC); -	ice_init_flex_flds(hw, ICE_RXDID_FLEX_NIC_2);  	status = ice_init_hw_tbls(hw);  	if (status)  		goto err_unroll_fltr_mgmt_struct; @@ -1601,6 +1498,114 @@ void ice_release_res(struct ice_hw *hw, enum ice_aq_res_ids res)  }  /** + * ice_aq_alloc_free_res - command to allocate/free resources + * @hw: pointer to the HW struct + * @num_entries: number of resource entries in buffer + * @buf: Indirect buffer to hold data parameters and response + * @buf_size: size of buffer for indirect commands + * @opc: pass in the command opcode + * @cd: pointer to command details structure or NULL + * + * Helper function to allocate/free resources using the admin queue commands + */ +enum ice_status +ice_aq_alloc_free_res(struct ice_hw *hw, u16 num_entries, +		      struct ice_aqc_alloc_free_res_elem *buf, u16 buf_size, +		      enum ice_adminq_opc opc, struct ice_sq_cd *cd) +{ +	struct ice_aqc_alloc_free_res_cmd *cmd; +	struct ice_aq_desc desc; + +	cmd = &desc.params.sw_res_ctrl; + +	if (!buf) +		return ICE_ERR_PARAM; + +	if (buf_size < (num_entries * sizeof(buf->elem[0]))) +		return ICE_ERR_PARAM; + +	ice_fill_dflt_direct_cmd_desc(&desc, opc); + +	desc.flags |= cpu_to_le16(ICE_AQ_FLAG_RD); + +	cmd->num_entries = cpu_to_le16(num_entries); + +	return ice_aq_send_cmd(hw, &desc, buf, buf_size, cd); +} + +/** + * ice_alloc_hw_res - allocate resource + * @hw: pointer to the HW struct + * @type: type of resource + * @num: number of resources to allocate + * @btm: allocate from bottom + * @res: pointer to array that will receive the resources + */ +enum ice_status +ice_alloc_hw_res(struct ice_hw *hw, u16 type, u16 num, bool btm, u16 *res) +{ +	struct ice_aqc_alloc_free_res_elem *buf; +	enum ice_status status; +	u16 buf_len; + +	buf_len = struct_size(buf, elem, num - 1); +	buf = kzalloc(buf_len, GFP_KERNEL); +	if (!buf) +		return ICE_ERR_NO_MEMORY; + +	/* Prepare buffer to allocate resource. */ +	buf->num_elems = cpu_to_le16(num); +	buf->res_type = cpu_to_le16(type | ICE_AQC_RES_TYPE_FLAG_DEDICATED | +				    ICE_AQC_RES_TYPE_FLAG_IGNORE_INDEX); +	if (btm) +		buf->res_type |= cpu_to_le16(ICE_AQC_RES_TYPE_FLAG_SCAN_BOTTOM); + +	status = ice_aq_alloc_free_res(hw, 1, buf, buf_len, +				       ice_aqc_opc_alloc_res, NULL); +	if (status) +		goto ice_alloc_res_exit; + +	memcpy(res, buf->elem, sizeof(buf->elem) * num); + +ice_alloc_res_exit: +	kfree(buf); +	return status; +} + +/** + * ice_free_hw_res - free allocated HW resource + * @hw: pointer to the HW struct + * @type: type of resource to free + * @num: number of resources + * @res: pointer to array that contains the resources to free + */ +enum ice_status +ice_free_hw_res(struct ice_hw *hw, u16 type, u16 num, u16 *res) +{ +	struct ice_aqc_alloc_free_res_elem *buf; +	enum ice_status status; +	u16 buf_len; + +	buf_len = struct_size(buf, elem, num - 1); +	buf = kzalloc(buf_len, GFP_KERNEL); +	if (!buf) +		return ICE_ERR_NO_MEMORY; + +	/* Prepare buffer to free resource. */ +	buf->num_elems = cpu_to_le16(num); +	buf->res_type = cpu_to_le16(type); +	memcpy(buf->elem, res, sizeof(buf->elem) * num); + +	status = ice_aq_alloc_free_res(hw, num, buf, buf_len, +				       ice_aqc_opc_free_res, NULL); +	if (status) +		ice_debug(hw, ICE_DBG_SW, "CQ CMD Buffer:\n"); + +	kfree(buf); +	return status; +} + +/**   * ice_get_num_per_func - determine number of resources per PF   * @hw: pointer to the HW structure   * @max: value to be evenly split between each PF @@ -3510,7 +3515,10 @@ enum ice_status ice_replay_vsi(struct ice_hw *hw, u16 vsi_handle)  		if (status)  			return status;  	} - +	/* Replay per VSI all RSS configurations */ +	status = ice_replay_rss_cfg(hw, vsi_handle); +	if (status) +		return status;  	/* Replay per VSI all filters */  	status = ice_replay_vsi_all_fltr(hw, vsi_handle);  	return status; diff --git a/drivers/net/ethernet/intel/ice/ice_common.h b/drivers/net/ethernet/intel/ice/ice_common.h index b22aa561e253..b5c013fdaaf9 100644 --- a/drivers/net/ethernet/intel/ice/ice_common.h +++ b/drivers/net/ethernet/intel/ice/ice_common.h @@ -34,10 +34,18 @@ enum ice_status  ice_acquire_res(struct ice_hw *hw, enum ice_aq_res_ids res,  		enum ice_aq_res_access_type access, u32 timeout);  void ice_release_res(struct ice_hw *hw, enum ice_aq_res_ids res); +enum ice_status +ice_alloc_hw_res(struct ice_hw *hw, u16 type, u16 num, bool btm, u16 *res); +enum ice_status +ice_free_hw_res(struct ice_hw *hw, u16 type, u16 num, u16 *res);  enum ice_status ice_init_nvm(struct ice_hw *hw);  enum ice_status  ice_read_sr_buf(struct ice_hw *hw, u16 offset, u16 *words, u16 *data);  enum ice_status +ice_aq_alloc_free_res(struct ice_hw *hw, u16 num_entries, +		      struct ice_aqc_alloc_free_res_elem *buf, u16 buf_size, +		      enum ice_adminq_opc opc, struct ice_sq_cd *cd); +enum ice_status  ice_sq_send_cmd(struct ice_hw *hw, struct ice_ctl_q_info *cq,  		struct ice_aq_desc *desc, void *buf, u16 buf_size,  		struct ice_sq_cd *cd); diff --git a/drivers/net/ethernet/intel/ice/ice_dcb_lib.c b/drivers/net/ethernet/intel/ice/ice_dcb_lib.c index d3d3ec29def9..0664e5b8d130 100644 --- a/drivers/net/ethernet/intel/ice/ice_dcb_lib.c +++ b/drivers/net/ethernet/intel/ice/ice_dcb_lib.c @@ -396,6 +396,12 @@ dcb_error:  	prev_cfg->etscfg.tcbwtable[0] = ICE_TC_MAX_BW;  	prev_cfg->etscfg.tsatable[0] = ICE_IEEE_TSA_ETS;  	memcpy(&prev_cfg->etsrec, &prev_cfg->etscfg, sizeof(prev_cfg->etsrec)); +	/* Coverity warns the return code of ice_pf_dcb_cfg() is not checked +	 * here as is done for other calls to that function. That check is +	 * not necessary since this is in this function's error cleanup path. +	 * Suppress the Coverity warning with the following comment... +	 */ +	/* coverity[check_return] */  	ice_pf_dcb_cfg(pf, prev_cfg, false);  	kfree(prev_cfg);  } diff --git a/drivers/net/ethernet/intel/ice/ice_devids.h b/drivers/net/ethernet/intel/ice/ice_devids.h index f8d5c661d0ba..ce63017c56c7 100644 --- a/drivers/net/ethernet/intel/ice/ice_devids.h +++ b/drivers/net/ethernet/intel/ice/ice_devids.h @@ -11,5 +11,23 @@  #define ICE_DEV_ID_E810C_QSFP		0x1592  /* Intel(R) Ethernet Controller E810-C for SFP */  #define ICE_DEV_ID_E810C_SFP		0x1593 +/* Intel(R) Ethernet Connection E822-C for backplane */ +#define ICE_DEV_ID_E822C_BACKPLANE	0x1890 +/* Intel(R) Ethernet Connection E822-C for QSFP */ +#define ICE_DEV_ID_E822C_QSFP		0x1891 +/* Intel(R) Ethernet Connection E822-C for SFP */ +#define ICE_DEV_ID_E822C_SFP		0x1892 +/* Intel(R) Ethernet Connection E822-C/X557-AT 10GBASE-T */ +#define ICE_DEV_ID_E822C_10G_BASE_T	0x1893 +/* Intel(R) Ethernet Connection E822-C 1GbE */ +#define ICE_DEV_ID_E822C_SGMII		0x1894 +/* Intel(R) Ethernet Connection E822-X for backplane */ +#define ICE_DEV_ID_E822X_BACKPLANE	0x1897 +/* Intel(R) Ethernet Connection E822-L for SFP */ +#define ICE_DEV_ID_E822L_SFP		0x1898 +/* Intel(R) Ethernet Connection E822-L/X557-AT 10GBASE-T */ +#define ICE_DEV_ID_E822L_10G_BASE_T	0x1899 +/* Intel(R) Ethernet Connection E822-L 1GbE */ +#define ICE_DEV_ID_E822L_SGMII		0x189A  #endif /* _ICE_DEVIDS_H_ */ diff --git a/drivers/net/ethernet/intel/ice/ice_ethtool.c b/drivers/net/ethernet/intel/ice/ice_ethtool.c index 9ebd93e79aeb..90c6a3ca20c9 100644 --- a/drivers/net/ethernet/intel/ice/ice_ethtool.c +++ b/drivers/net/ethernet/intel/ice/ice_ethtool.c @@ -4,6 +4,7 @@  /* ethtool support for ice */  #include "ice.h" +#include "ice_flow.h"  #include "ice_lib.h"  #include "ice_dcb_lib.h" @@ -283,12 +284,15 @@ out:   */  static bool ice_active_vfs(struct ice_pf *pf)  { -	struct ice_vf *vf = pf->vf;  	int i; -	for (i = 0; i < pf->num_alloc_vfs; i++, vf++) +	ice_for_each_vf(pf, i) { +		struct ice_vf *vf = &pf->vf[i]; +  		if (test_bit(ICE_VF_STATE_ACTIVE, vf->vf_states))  			return true; +	} +  	return false;  } @@ -2531,6 +2535,243 @@ done:  }  /** + * ice_parse_hdrs - parses headers from RSS hash input + * @nfc: ethtool rxnfc command + * + * This function parses the rxnfc command and returns intended + * header types for RSS configuration + */ +static u32 ice_parse_hdrs(struct ethtool_rxnfc *nfc) +{ +	u32 hdrs = ICE_FLOW_SEG_HDR_NONE; + +	switch (nfc->flow_type) { +	case TCP_V4_FLOW: +		hdrs |= ICE_FLOW_SEG_HDR_TCP | ICE_FLOW_SEG_HDR_IPV4; +		break; +	case UDP_V4_FLOW: +		hdrs |= ICE_FLOW_SEG_HDR_UDP | ICE_FLOW_SEG_HDR_IPV4; +		break; +	case SCTP_V4_FLOW: +		hdrs |= ICE_FLOW_SEG_HDR_SCTP | ICE_FLOW_SEG_HDR_IPV4; +		break; +	case TCP_V6_FLOW: +		hdrs |= ICE_FLOW_SEG_HDR_TCP | ICE_FLOW_SEG_HDR_IPV6; +		break; +	case UDP_V6_FLOW: +		hdrs |= ICE_FLOW_SEG_HDR_UDP | ICE_FLOW_SEG_HDR_IPV6; +		break; +	case SCTP_V6_FLOW: +		hdrs |= ICE_FLOW_SEG_HDR_SCTP | ICE_FLOW_SEG_HDR_IPV6; +		break; +	default: +		break; +	} +	return hdrs; +} + +#define ICE_FLOW_HASH_FLD_IPV4_SA	BIT_ULL(ICE_FLOW_FIELD_IDX_IPV4_SA) +#define ICE_FLOW_HASH_FLD_IPV6_SA	BIT_ULL(ICE_FLOW_FIELD_IDX_IPV6_SA) +#define ICE_FLOW_HASH_FLD_IPV4_DA	BIT_ULL(ICE_FLOW_FIELD_IDX_IPV4_DA) +#define ICE_FLOW_HASH_FLD_IPV6_DA	BIT_ULL(ICE_FLOW_FIELD_IDX_IPV6_DA) +#define ICE_FLOW_HASH_FLD_TCP_SRC_PORT	BIT_ULL(ICE_FLOW_FIELD_IDX_TCP_SRC_PORT) +#define ICE_FLOW_HASH_FLD_TCP_DST_PORT	BIT_ULL(ICE_FLOW_FIELD_IDX_TCP_DST_PORT) +#define ICE_FLOW_HASH_FLD_UDP_SRC_PORT	BIT_ULL(ICE_FLOW_FIELD_IDX_UDP_SRC_PORT) +#define ICE_FLOW_HASH_FLD_UDP_DST_PORT	BIT_ULL(ICE_FLOW_FIELD_IDX_UDP_DST_PORT) +#define ICE_FLOW_HASH_FLD_SCTP_SRC_PORT	\ +	BIT_ULL(ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT) +#define ICE_FLOW_HASH_FLD_SCTP_DST_PORT	\ +	BIT_ULL(ICE_FLOW_FIELD_IDX_SCTP_DST_PORT) + +/** + * ice_parse_hash_flds - parses hash fields from RSS hash input + * @nfc: ethtool rxnfc command + * + * This function parses the rxnfc command and returns intended + * hash fields for RSS configuration + */ +static u64 ice_parse_hash_flds(struct ethtool_rxnfc *nfc) +{ +	u64 hfld = ICE_HASH_INVALID; + +	if (nfc->data & RXH_IP_SRC || nfc->data & RXH_IP_DST) { +		switch (nfc->flow_type) { +		case TCP_V4_FLOW: +		case UDP_V4_FLOW: +		case SCTP_V4_FLOW: +			if (nfc->data & RXH_IP_SRC) +				hfld |= ICE_FLOW_HASH_FLD_IPV4_SA; +			if (nfc->data & RXH_IP_DST) +				hfld |= ICE_FLOW_HASH_FLD_IPV4_DA; +			break; +		case TCP_V6_FLOW: +		case UDP_V6_FLOW: +		case SCTP_V6_FLOW: +			if (nfc->data & RXH_IP_SRC) +				hfld |= ICE_FLOW_HASH_FLD_IPV6_SA; +			if (nfc->data & RXH_IP_DST) +				hfld |= ICE_FLOW_HASH_FLD_IPV6_DA; +			break; +		default: +			break; +		} +	} + +	if (nfc->data & RXH_L4_B_0_1 || nfc->data & RXH_L4_B_2_3) { +		switch (nfc->flow_type) { +		case TCP_V4_FLOW: +		case TCP_V6_FLOW: +			if (nfc->data & RXH_L4_B_0_1) +				hfld |= ICE_FLOW_HASH_FLD_TCP_SRC_PORT; +			if (nfc->data & RXH_L4_B_2_3) +				hfld |= ICE_FLOW_HASH_FLD_TCP_DST_PORT; +			break; +		case UDP_V4_FLOW: +		case UDP_V6_FLOW: +			if (nfc->data & RXH_L4_B_0_1) +				hfld |= ICE_FLOW_HASH_FLD_UDP_SRC_PORT; +			if (nfc->data & RXH_L4_B_2_3) +				hfld |= ICE_FLOW_HASH_FLD_UDP_DST_PORT; +			break; +		case SCTP_V4_FLOW: +		case SCTP_V6_FLOW: +			if (nfc->data & RXH_L4_B_0_1) +				hfld |= ICE_FLOW_HASH_FLD_SCTP_SRC_PORT; +			if (nfc->data & RXH_L4_B_2_3) +				hfld |= ICE_FLOW_HASH_FLD_SCTP_DST_PORT; +			break; +		default: +			break; +		} +	} + +	return hfld; +} + +/** + * ice_set_rss_hash_opt - Enable/Disable flow types for RSS hash + * @vsi: the VSI being configured + * @nfc: ethtool rxnfc command + * + * Returns Success if the flow input set is supported. + */ +static int +ice_set_rss_hash_opt(struct ice_vsi *vsi, struct ethtool_rxnfc *nfc) +{ +	struct ice_pf *pf = vsi->back; +	enum ice_status status; +	struct device *dev; +	u64 hashed_flds; +	u32 hdrs; + +	dev = ice_pf_to_dev(pf); +	if (ice_is_safe_mode(pf)) { +		dev_dbg(dev, "Advanced RSS disabled. Package download failed, vsi num = %d\n", +			vsi->vsi_num); +		return -EINVAL; +	} + +	hashed_flds = ice_parse_hash_flds(nfc); +	if (hashed_flds == ICE_HASH_INVALID) { +		dev_dbg(dev, "Invalid hash fields, vsi num = %d\n", +			vsi->vsi_num); +		return -EINVAL; +	} + +	hdrs = ice_parse_hdrs(nfc); +	if (hdrs == ICE_FLOW_SEG_HDR_NONE) { +		dev_dbg(dev, "Header type is not valid, vsi num = %d\n", +			vsi->vsi_num); +		return -EINVAL; +	} + +	status = ice_add_rss_cfg(&pf->hw, vsi->idx, hashed_flds, hdrs); +	if (status) { +		dev_dbg(dev, "ice_add_rss_cfg failed, vsi num = %d, error = %d\n", +			vsi->vsi_num, status); +		return -EINVAL; +	} + +	return 0; +} + +/** + * ice_get_rss_hash_opt - Retrieve hash fields for a given flow-type + * @vsi: the VSI being configured + * @nfc: ethtool rxnfc command + */ +static void +ice_get_rss_hash_opt(struct ice_vsi *vsi, struct ethtool_rxnfc *nfc) +{ +	struct ice_pf *pf = vsi->back; +	struct device *dev; +	u64 hash_flds; +	u32 hdrs; + +	dev = ice_pf_to_dev(pf); + +	nfc->data = 0; +	if (ice_is_safe_mode(pf)) { +		dev_dbg(dev, "Advanced RSS disabled. Package download failed, vsi num = %d\n", +			vsi->vsi_num); +		return; +	} + +	hdrs = ice_parse_hdrs(nfc); +	if (hdrs == ICE_FLOW_SEG_HDR_NONE) { +		dev_dbg(dev, "Header type is not valid, vsi num = %d\n", +			vsi->vsi_num); +		return; +	} + +	hash_flds = ice_get_rss_cfg(&pf->hw, vsi->idx, hdrs); +	if (hash_flds == ICE_HASH_INVALID) { +		dev_dbg(dev, "No hash fields found for the given header type, vsi num = %d\n", +			vsi->vsi_num); +		return; +	} + +	if (hash_flds & ICE_FLOW_HASH_FLD_IPV4_SA || +	    hash_flds & ICE_FLOW_HASH_FLD_IPV6_SA) +		nfc->data |= (u64)RXH_IP_SRC; + +	if (hash_flds & ICE_FLOW_HASH_FLD_IPV4_DA || +	    hash_flds & ICE_FLOW_HASH_FLD_IPV6_DA) +		nfc->data |= (u64)RXH_IP_DST; + +	if (hash_flds & ICE_FLOW_HASH_FLD_TCP_SRC_PORT || +	    hash_flds & ICE_FLOW_HASH_FLD_UDP_SRC_PORT || +	    hash_flds & ICE_FLOW_HASH_FLD_SCTP_SRC_PORT) +		nfc->data |= (u64)RXH_L4_B_0_1; + +	if (hash_flds & ICE_FLOW_HASH_FLD_TCP_DST_PORT || +	    hash_flds & ICE_FLOW_HASH_FLD_UDP_DST_PORT || +	    hash_flds & ICE_FLOW_HASH_FLD_SCTP_DST_PORT) +		nfc->data |= (u64)RXH_L4_B_2_3; +} + +/** + * ice_set_rxnfc - command to set Rx flow rules. + * @netdev: network interface device structure + * @cmd: ethtool rxnfc command + * + * Returns 0 for success and negative values for errors + */ +static int ice_set_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *cmd) +{ +	struct ice_netdev_priv *np = netdev_priv(netdev); +	struct ice_vsi *vsi = np->vsi; + +	switch (cmd->cmd) { +	case ETHTOOL_SRXFH: +		return ice_set_rss_hash_opt(vsi, cmd); +	default: +		break; +	} +	return -EOPNOTSUPP; +} + +/**   * ice_get_rxnfc - command to get Rx flow classification rules   * @netdev: network interface device structure   * @cmd: ethtool rxnfc command @@ -2551,6 +2792,10 @@ ice_get_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *cmd,  		cmd->data = vsi->rss_size;  		ret = 0;  		break; +	case ETHTOOL_GRXFH: +		ice_get_rss_hash_opt(vsi, cmd); +		ret = 0; +		break;  	default:  		break;  	} @@ -3585,6 +3830,53 @@ ice_set_q_coalesce(struct ice_vsi *vsi, struct ethtool_coalesce *ec, int q_num)  }  /** + * ice_is_coalesce_param_invalid - check for unsupported coalesce parameters + * @netdev: pointer to the netdev associated with this query + * @ec: ethtool structure to fill with driver's coalesce settings + * + * Print netdev info if driver doesn't support one of the parameters + * and return error. When any parameters will be implemented, remove only + * this parameter from param array. + */ +static int +ice_is_coalesce_param_invalid(struct net_device *netdev, +			      struct ethtool_coalesce *ec) +{ +	struct ice_ethtool_not_used { +		u32 value; +		const char *name; +	} param[] = { +		{ec->stats_block_coalesce_usecs, "stats-block-usecs"}, +		{ec->rate_sample_interval, "sample-interval"}, +		{ec->pkt_rate_low, "pkt-rate-low"}, +		{ec->pkt_rate_high, "pkt-rate-high"}, +		{ec->rx_max_coalesced_frames, "rx-frames"}, +		{ec->rx_coalesce_usecs_irq, "rx-usecs-irq"}, +		{ec->rx_max_coalesced_frames_irq, "rx-frames-irq"}, +		{ec->tx_max_coalesced_frames, "tx-frames"}, +		{ec->tx_coalesce_usecs_irq, "tx-usecs-irq"}, +		{ec->tx_max_coalesced_frames_irq, "tx-frames-irq"}, +		{ec->rx_coalesce_usecs_low, "rx-usecs-low"}, +		{ec->rx_max_coalesced_frames_low, "rx-frames-low"}, +		{ec->tx_coalesce_usecs_low, "tx-usecs-low"}, +		{ec->tx_max_coalesced_frames_low, "tx-frames-low"}, +		{ec->rx_max_coalesced_frames_high, "rx-frames-high"}, +		{ec->tx_max_coalesced_frames_high, "tx-frames-high"} +	}; +	int i; + +	for (i = 0; i < ARRAY_SIZE(param); i++) { +		if (param[i].value) { +			netdev_info(netdev, "Setting %s not supported\n", +				    param[i].name); +			return -EINVAL; +		} +	} + +	return 0; +} + +/**   * __ice_set_coalesce - set ITR/INTRL values for the device   * @netdev: pointer to the netdev associated with this query   * @ec: ethtool structure to fill with driver's coalesce settings @@ -3600,6 +3892,9 @@ __ice_set_coalesce(struct net_device *netdev, struct ethtool_coalesce *ec,  	struct ice_netdev_priv *np = netdev_priv(netdev);  	struct ice_vsi *vsi = np->vsi; +	if (ice_is_coalesce_param_invalid(netdev, ec)) +		return -EINVAL; +  	if (q_num < 0) {  		int v_idx; @@ -3804,6 +4099,7 @@ static const struct ethtool_ops ice_ethtool_ops = {  	.set_priv_flags		= ice_set_priv_flags,  	.get_sset_count		= ice_get_sset_count,  	.get_rxnfc		= ice_get_rxnfc, +	.set_rxnfc		= ice_set_rxnfc,  	.get_ringparam		= ice_get_ringparam,  	.set_ringparam		= ice_set_ringparam,  	.nway_reset		= ice_nway_reset, diff --git a/drivers/net/ethernet/intel/ice/ice_flex_pipe.c b/drivers/net/ethernet/intel/ice/ice_flex_pipe.c index cbd53b586c36..99208946224c 100644 --- a/drivers/net/ethernet/intel/ice/ice_flex_pipe.c +++ b/drivers/net/ethernet/intel/ice/ice_flex_pipe.c @@ -3,6 +3,87 @@  #include "ice_common.h"  #include "ice_flex_pipe.h" +#include "ice_flow.h" + +static const u32 ice_sect_lkup[ICE_BLK_COUNT][ICE_SECT_COUNT] = { +	/* SWITCH */ +	{ +		ICE_SID_XLT0_SW, +		ICE_SID_XLT_KEY_BUILDER_SW, +		ICE_SID_XLT1_SW, +		ICE_SID_XLT2_SW, +		ICE_SID_PROFID_TCAM_SW, +		ICE_SID_PROFID_REDIR_SW, +		ICE_SID_FLD_VEC_SW, +		ICE_SID_CDID_KEY_BUILDER_SW, +		ICE_SID_CDID_REDIR_SW +	}, + +	/* ACL */ +	{ +		ICE_SID_XLT0_ACL, +		ICE_SID_XLT_KEY_BUILDER_ACL, +		ICE_SID_XLT1_ACL, +		ICE_SID_XLT2_ACL, +		ICE_SID_PROFID_TCAM_ACL, +		ICE_SID_PROFID_REDIR_ACL, +		ICE_SID_FLD_VEC_ACL, +		ICE_SID_CDID_KEY_BUILDER_ACL, +		ICE_SID_CDID_REDIR_ACL +	}, + +	/* FD */ +	{ +		ICE_SID_XLT0_FD, +		ICE_SID_XLT_KEY_BUILDER_FD, +		ICE_SID_XLT1_FD, +		ICE_SID_XLT2_FD, +		ICE_SID_PROFID_TCAM_FD, +		ICE_SID_PROFID_REDIR_FD, +		ICE_SID_FLD_VEC_FD, +		ICE_SID_CDID_KEY_BUILDER_FD, +		ICE_SID_CDID_REDIR_FD +	}, + +	/* RSS */ +	{ +		ICE_SID_XLT0_RSS, +		ICE_SID_XLT_KEY_BUILDER_RSS, +		ICE_SID_XLT1_RSS, +		ICE_SID_XLT2_RSS, +		ICE_SID_PROFID_TCAM_RSS, +		ICE_SID_PROFID_REDIR_RSS, +		ICE_SID_FLD_VEC_RSS, +		ICE_SID_CDID_KEY_BUILDER_RSS, +		ICE_SID_CDID_REDIR_RSS +	}, + +	/* PE */ +	{ +		ICE_SID_XLT0_PE, +		ICE_SID_XLT_KEY_BUILDER_PE, +		ICE_SID_XLT1_PE, +		ICE_SID_XLT2_PE, +		ICE_SID_PROFID_TCAM_PE, +		ICE_SID_PROFID_REDIR_PE, +		ICE_SID_FLD_VEC_PE, +		ICE_SID_CDID_KEY_BUILDER_PE, +		ICE_SID_CDID_REDIR_PE +	} +}; + +/** + * ice_sect_id - returns section ID + * @blk: block type + * @sect: section type + * + * This helper function returns the proper section ID given a block type and a + * section type. + */ +static u32 ice_sect_id(enum ice_block blk, enum ice_sect sect) +{ +	return ice_sect_lkup[blk][sect]; +}  /**   * ice_pkg_val_buf @@ -158,6 +239,176 @@ ice_pkg_enum_section(struct ice_seg *ice_seg, struct ice_pkg_enum *state,  	return state->sect;  } +/* Key creation */ + +#define ICE_DC_KEY	0x1	/* don't care */ +#define ICE_DC_KEYINV	0x1 +#define ICE_NM_KEY	0x0	/* never match */ +#define ICE_NM_KEYINV	0x0 +#define ICE_0_KEY	0x1	/* match 0 */ +#define ICE_0_KEYINV	0x0 +#define ICE_1_KEY	0x0	/* match 1 */ +#define ICE_1_KEYINV	0x1 + +/** + * ice_gen_key_word - generate 16-bits of a key/mask word + * @val: the value + * @valid: valid bits mask (change only the valid bits) + * @dont_care: don't care mask + * @nvr_mtch: never match mask + * @key: pointer to an array of where the resulting key portion + * @key_inv: pointer to an array of where the resulting key invert portion + * + * This function generates 16-bits from a 8-bit value, an 8-bit don't care mask + * and an 8-bit never match mask. The 16-bits of output are divided into 8 bits + * of key and 8 bits of key invert. + * + *     '0' =    b01, always match a 0 bit + *     '1' =    b10, always match a 1 bit + *     '?' =    b11, don't care bit (always matches) + *     '~' =    b00, never match bit + * + * Input: + *          val:         b0  1  0  1  0  1 + *          dont_care:   b0  0  1  1  0  0 + *          never_mtch:  b0  0  0  0  1  1 + *          ------------------------------ + * Result:  key:        b01 10 11 11 00 00 + */ +static enum ice_status +ice_gen_key_word(u8 val, u8 valid, u8 dont_care, u8 nvr_mtch, u8 *key, +		 u8 *key_inv) +{ +	u8 in_key = *key, in_key_inv = *key_inv; +	u8 i; + +	/* 'dont_care' and 'nvr_mtch' masks cannot overlap */ +	if ((dont_care ^ nvr_mtch) != (dont_care | nvr_mtch)) +		return ICE_ERR_CFG; + +	*key = 0; +	*key_inv = 0; + +	/* encode the 8 bits into 8-bit key and 8-bit key invert */ +	for (i = 0; i < 8; i++) { +		*key >>= 1; +		*key_inv >>= 1; + +		if (!(valid & 0x1)) { /* change only valid bits */ +			*key |= (in_key & 0x1) << 7; +			*key_inv |= (in_key_inv & 0x1) << 7; +		} else if (dont_care & 0x1) { /* don't care bit */ +			*key |= ICE_DC_KEY << 7; +			*key_inv |= ICE_DC_KEYINV << 7; +		} else if (nvr_mtch & 0x1) { /* never match bit */ +			*key |= ICE_NM_KEY << 7; +			*key_inv |= ICE_NM_KEYINV << 7; +		} else if (val & 0x01) { /* exact 1 match */ +			*key |= ICE_1_KEY << 7; +			*key_inv |= ICE_1_KEYINV << 7; +		} else { /* exact 0 match */ +			*key |= ICE_0_KEY << 7; +			*key_inv |= ICE_0_KEYINV << 7; +		} + +		dont_care >>= 1; +		nvr_mtch >>= 1; +		valid >>= 1; +		val >>= 1; +		in_key >>= 1; +		in_key_inv >>= 1; +	} + +	return 0; +} + +/** + * ice_bits_max_set - determine if the number of bits set is within a maximum + * @mask: pointer to the byte array which is the mask + * @size: the number of bytes in the mask + * @max: the max number of set bits + * + * This function determines if there are at most 'max' number of bits set in an + * array. Returns true if the number for bits set is <= max or will return false + * otherwise. + */ +static bool ice_bits_max_set(const u8 *mask, u16 size, u16 max) +{ +	u16 count = 0; +	u16 i; + +	/* check each byte */ +	for (i = 0; i < size; i++) { +		/* if 0, go to next byte */ +		if (!mask[i]) +			continue; + +		/* We know there is at least one set bit in this byte because of +		 * the above check; if we already have found 'max' number of +		 * bits set, then we can return failure now. +		 */ +		if (count == max) +			return false; + +		/* count the bits in this byte, checking threshold */ +		count += hweight8(mask[i]); +		if (count > max) +			return false; +	} + +	return true; +} + +/** + * ice_set_key - generate a variable sized key with multiples of 16-bits + * @key: pointer to where the key will be stored + * @size: the size of the complete key in bytes (must be even) + * @val: array of 8-bit values that makes up the value portion of the key + * @upd: array of 8-bit masks that determine what key portion to update + * @dc: array of 8-bit masks that make up the don't care mask + * @nm: array of 8-bit masks that make up the never match mask + * @off: the offset of the first byte in the key to update + * @len: the number of bytes in the key update + * + * This function generates a key from a value, a don't care mask and a never + * match mask. + * upd, dc, and nm are optional parameters, and can be NULL: + *	upd == NULL --> udp mask is all 1's (update all bits) + *	dc == NULL --> dc mask is all 0's (no don't care bits) + *	nm == NULL --> nm mask is all 0's (no never match bits) + */ +static enum ice_status +ice_set_key(u8 *key, u16 size, u8 *val, u8 *upd, u8 *dc, u8 *nm, u16 off, +	    u16 len) +{ +	u16 half_size; +	u16 i; + +	/* size must be a multiple of 2 bytes. */ +	if (size % 2) +		return ICE_ERR_CFG; + +	half_size = size / 2; +	if (off + len > half_size) +		return ICE_ERR_CFG; + +	/* Make sure at most one bit is set in the never match mask. Having more +	 * than one never match mask bit set will cause HW to consume excessive +	 * power otherwise; this is a power management efficiency check. +	 */ +#define ICE_NVR_MTCH_BITS_MAX	1 +	if (nm && !ice_bits_max_set(nm, len, ICE_NVR_MTCH_BITS_MAX)) +		return ICE_ERR_CFG; + +	for (i = 0; i < len; i++) +		if (ice_gen_key_word(val[i], upd ? upd[i] : 0xff, +				     dc ? dc[i] : 0, nm ? nm[i] : 0, +				     key + off + i, key + half_size + off + i)) +			return ICE_ERR_CFG; + +	return 0; +} +  /**   * ice_acquire_global_cfg_lock   * @hw: pointer to the HW structure @@ -205,6 +456,31 @@ static void ice_release_global_cfg_lock(struct ice_hw *hw)  }  /** + * ice_acquire_change_lock + * @hw: pointer to the HW structure + * @access: access type (read or write) + * + * This function will request ownership of the change lock. + */ +static enum ice_status +ice_acquire_change_lock(struct ice_hw *hw, enum ice_aq_res_access_type access) +{ +	return ice_acquire_res(hw, ICE_CHANGE_LOCK_RES_ID, access, +			       ICE_CHANGE_LOCK_TIMEOUT); +} + +/** + * ice_release_change_lock + * @hw: pointer to the HW structure + * + * This function will release the change lock using the proper Admin Command. + */ +static void ice_release_change_lock(struct ice_hw *hw) +{ +	ice_release_res(hw, ICE_CHANGE_LOCK_RES_ID); +} + +/**   * ice_aq_download_pkg   * @hw: pointer to the hardware structure   * @pkg_buf: the package buffer to transfer @@ -253,6 +529,54 @@ ice_aq_download_pkg(struct ice_hw *hw, struct ice_buf_hdr *pkg_buf,  }  /** + * ice_aq_update_pkg + * @hw: pointer to the hardware structure + * @pkg_buf: the package cmd buffer + * @buf_size: the size of the package cmd buffer + * @last_buf: last buffer indicator + * @error_offset: returns error offset + * @error_info: returns error information + * @cd: pointer to command details structure or NULL + * + * Update Package (0x0C42) + */ +static enum ice_status +ice_aq_update_pkg(struct ice_hw *hw, struct ice_buf_hdr *pkg_buf, u16 buf_size, +		  bool last_buf, u32 *error_offset, u32 *error_info, +		  struct ice_sq_cd *cd) +{ +	struct ice_aqc_download_pkg *cmd; +	struct ice_aq_desc desc; +	enum ice_status status; + +	if (error_offset) +		*error_offset = 0; +	if (error_info) +		*error_info = 0; + +	cmd = &desc.params.download_pkg; +	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_update_pkg); +	desc.flags |= cpu_to_le16(ICE_AQ_FLAG_RD); + +	if (last_buf) +		cmd->flags |= ICE_AQC_DOWNLOAD_PKG_LAST_BUF; + +	status = ice_aq_send_cmd(hw, &desc, pkg_buf, buf_size, cd); +	if (status == ICE_ERR_AQ_ERROR) { +		/* Read error from buffer only when the FW returned an error */ +		struct ice_aqc_download_pkg_resp *resp; + +		resp = (struct ice_aqc_download_pkg_resp *)pkg_buf; +		if (error_offset) +			*error_offset = le32_to_cpu(resp->error_offset); +		if (error_info) +			*error_info = le32_to_cpu(resp->error_info); +	} + +	return status; +} + +/**   * ice_find_seg_in_pkg   * @hw: pointer to the hardware structure   * @seg_type: the segment type to search for (i.e., SEGMENT_TYPE_CPK) @@ -287,6 +611,44 @@ ice_find_seg_in_pkg(struct ice_hw *hw, u32 seg_type,  }  /** + * ice_update_pkg + * @hw: pointer to the hardware structure + * @bufs: pointer to an array of buffers + * @count: the number of buffers in the array + * + * Obtains change lock and updates package. + */ +static enum ice_status +ice_update_pkg(struct ice_hw *hw, struct ice_buf *bufs, u32 count) +{ +	enum ice_status status; +	u32 offset, info, i; + +	status = ice_acquire_change_lock(hw, ICE_RES_WRITE); +	if (status) +		return status; + +	for (i = 0; i < count; i++) { +		struct ice_buf_hdr *bh = (struct ice_buf_hdr *)(bufs + i); +		bool last = ((i + 1) == count); + +		status = ice_aq_update_pkg(hw, bh, le16_to_cpu(bh->data_end), +					   last, &offset, &info, NULL); + +		if (status) { +			ice_debug(hw, ICE_DBG_PKG, +				  "Update pkg failed: err %d off %d inf %d\n", +				  status, offset, info); +			break; +		} +	} + +	ice_release_change_lock(hw); + +	return status; +} + +/**   * ice_dwnld_cfg_bufs   * @hw: pointer to the hardware structure   * @bufs: pointer to an array of buffers @@ -767,6 +1129,169 @@ enum ice_status ice_copy_and_init_pkg(struct ice_hw *hw, const u8 *buf, u32 len)  	return status;  } +/** + * ice_pkg_buf_alloc + * @hw: pointer to the HW structure + * + * Allocates a package buffer and returns a pointer to the buffer header. + * Note: all package contents must be in Little Endian form. + */ +static struct ice_buf_build *ice_pkg_buf_alloc(struct ice_hw *hw) +{ +	struct ice_buf_build *bld; +	struct ice_buf_hdr *buf; + +	bld = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*bld), GFP_KERNEL); +	if (!bld) +		return NULL; + +	buf = (struct ice_buf_hdr *)bld; +	buf->data_end = cpu_to_le16(offsetof(struct ice_buf_hdr, +					     section_entry)); +	return bld; +} + +/** + * ice_pkg_buf_free + * @hw: pointer to the HW structure + * @bld: pointer to pkg build (allocated by ice_pkg_buf_alloc()) + * + * Frees a package buffer + */ +static void ice_pkg_buf_free(struct ice_hw *hw, struct ice_buf_build *bld) +{ +	devm_kfree(ice_hw_to_dev(hw), bld); +} + +/** + * ice_pkg_buf_reserve_section + * @bld: pointer to pkg build (allocated by ice_pkg_buf_alloc()) + * @count: the number of sections to reserve + * + * Reserves one or more section table entries in a package buffer. This routine + * can be called multiple times as long as they are made before calling + * ice_pkg_buf_alloc_section(). Once ice_pkg_buf_alloc_section() + * is called once, the number of sections that can be allocated will not be able + * to be increased; not using all reserved sections is fine, but this will + * result in some wasted space in the buffer. + * Note: all package contents must be in Little Endian form. + */ +static enum ice_status +ice_pkg_buf_reserve_section(struct ice_buf_build *bld, u16 count) +{ +	struct ice_buf_hdr *buf; +	u16 section_count; +	u16 data_end; + +	if (!bld) +		return ICE_ERR_PARAM; + +	buf = (struct ice_buf_hdr *)&bld->buf; + +	/* already an active section, can't increase table size */ +	section_count = le16_to_cpu(buf->section_count); +	if (section_count > 0) +		return ICE_ERR_CFG; + +	if (bld->reserved_section_table_entries + count > ICE_MAX_S_COUNT) +		return ICE_ERR_CFG; +	bld->reserved_section_table_entries += count; + +	data_end = le16_to_cpu(buf->data_end) + +		   (count * sizeof(buf->section_entry[0])); +	buf->data_end = cpu_to_le16(data_end); + +	return 0; +} + +/** + * ice_pkg_buf_alloc_section + * @bld: pointer to pkg build (allocated by ice_pkg_buf_alloc()) + * @type: the section type value + * @size: the size of the section to reserve (in bytes) + * + * Reserves memory in the buffer for a section's content and updates the + * buffers' status accordingly. This routine returns a pointer to the first + * byte of the section start within the buffer, which is used to fill in the + * section contents. + * Note: all package contents must be in Little Endian form. + */ +static void * +ice_pkg_buf_alloc_section(struct ice_buf_build *bld, u32 type, u16 size) +{ +	struct ice_buf_hdr *buf; +	u16 sect_count; +	u16 data_end; + +	if (!bld || !type || !size) +		return NULL; + +	buf = (struct ice_buf_hdr *)&bld->buf; + +	/* check for enough space left in buffer */ +	data_end = le16_to_cpu(buf->data_end); + +	/* section start must align on 4 byte boundary */ +	data_end = ALIGN(data_end, 4); + +	if ((data_end + size) > ICE_MAX_S_DATA_END) +		return NULL; + +	/* check for more available section table entries */ +	sect_count = le16_to_cpu(buf->section_count); +	if (sect_count < bld->reserved_section_table_entries) { +		void *section_ptr = ((u8 *)buf) + data_end; + +		buf->section_entry[sect_count].offset = cpu_to_le16(data_end); +		buf->section_entry[sect_count].size = cpu_to_le16(size); +		buf->section_entry[sect_count].type = cpu_to_le32(type); + +		data_end += size; +		buf->data_end = cpu_to_le16(data_end); + +		buf->section_count = cpu_to_le16(sect_count + 1); +		return section_ptr; +	} + +	/* no free section table entries */ +	return NULL; +} + +/** + * ice_pkg_buf_get_active_sections + * @bld: pointer to pkg build (allocated by ice_pkg_buf_alloc()) + * + * Returns the number of active sections. Before using the package buffer + * in an update package command, the caller should make sure that there is at + * least one active section - otherwise, the buffer is not legal and should + * not be used. + * Note: all package contents must be in Little Endian form. + */ +static u16 ice_pkg_buf_get_active_sections(struct ice_buf_build *bld) +{ +	struct ice_buf_hdr *buf; + +	if (!bld) +		return 0; + +	buf = (struct ice_buf_hdr *)&bld->buf; +	return le16_to_cpu(buf->section_count); +} + +/** + * ice_pkg_buf + * @bld: pointer to pkg build (allocated by ice_pkg_buf_alloc()) + * + * Return a pointer to the buffer's header + */ +static struct ice_buf *ice_pkg_buf(struct ice_buf_build *bld) +{ +	if (!bld) +		return NULL; + +	return &bld->buf; +} +  /* PTG Management */  /** @@ -951,6 +1476,48 @@ enum ice_sid_all {  	ICE_SID_OFF_COUNT,  }; +/* Characteristic handling */ + +/** + * ice_match_prop_lst - determine if properties of two lists match + * @list1: first properties list + * @list2: second properties list + * + * Count, cookies and the order must match in order to be considered equivalent. + */ +static bool +ice_match_prop_lst(struct list_head *list1, struct list_head *list2) +{ +	struct ice_vsig_prof *tmp1; +	struct ice_vsig_prof *tmp2; +	u16 chk_count = 0; +	u16 count = 0; + +	/* compare counts */ +	list_for_each_entry(tmp1, list1, list) +		count++; +	list_for_each_entry(tmp2, list2, list) +		chk_count++; +	if (!count || count != chk_count) +		return false; + +	tmp1 = list_first_entry(list1, struct ice_vsig_prof, list); +	tmp2 = list_first_entry(list2, struct ice_vsig_prof, list); + +	/* profile cookies must compare, and in the exact same order to take +	 * into account priority +	 */ +	while (count--) { +		if (tmp2->profile_cookie != tmp1->profile_cookie) +			return false; + +		tmp1 = list_next_entry(tmp1, list); +		tmp2 = list_next_entry(tmp2, list); +	} + +	return true; +} +  /* VSIG Management */  /** @@ -999,6 +1566,117 @@ static u16 ice_vsig_alloc_val(struct ice_hw *hw, enum ice_block blk, u16 vsig)  }  /** + * ice_vsig_alloc - Finds a free entry and allocates a new VSIG + * @hw: pointer to the hardware structure + * @blk: HW block + * + * This function will iterate through the VSIG list and mark the first + * unused entry for the new VSIG entry as used and return that value. + */ +static u16 ice_vsig_alloc(struct ice_hw *hw, enum ice_block blk) +{ +	u16 i; + +	for (i = 1; i < ICE_MAX_VSIGS; i++) +		if (!hw->blk[blk].xlt2.vsig_tbl[i].in_use) +			return ice_vsig_alloc_val(hw, blk, i); + +	return ICE_DEFAULT_VSIG; +} + +/** + * ice_find_dup_props_vsig - find VSI group with a specified set of properties + * @hw: pointer to the hardware structure + * @blk: HW block + * @chs: characteristic list + * @vsig: returns the VSIG with the matching profiles, if found + * + * Each VSIG is associated with a characteristic set; i.e. all VSIs under + * a group have the same characteristic set. To check if there exists a VSIG + * which has the same characteristics as the input characteristics; this + * function will iterate through the XLT2 list and return the VSIG that has a + * matching configuration. In order to make sure that priorities are accounted + * for, the list must match exactly, including the order in which the + * characteristics are listed. + */ +static enum ice_status +ice_find_dup_props_vsig(struct ice_hw *hw, enum ice_block blk, +			struct list_head *chs, u16 *vsig) +{ +	struct ice_xlt2 *xlt2 = &hw->blk[blk].xlt2; +	u16 i; + +	for (i = 0; i < xlt2->count; i++) +		if (xlt2->vsig_tbl[i].in_use && +		    ice_match_prop_lst(chs, &xlt2->vsig_tbl[i].prop_lst)) { +			*vsig = ICE_VSIG_VALUE(i, hw->pf_id); +			return 0; +		} + +	return ICE_ERR_DOES_NOT_EXIST; +} + +/** + * ice_vsig_free - free VSI group + * @hw: pointer to the hardware structure + * @blk: HW block + * @vsig: VSIG to remove + * + * The function will remove all VSIs associated with the input VSIG and move + * them to the DEFAULT_VSIG and mark the VSIG available. + */ +static enum ice_status +ice_vsig_free(struct ice_hw *hw, enum ice_block blk, u16 vsig) +{ +	struct ice_vsig_prof *dtmp, *del; +	struct ice_vsig_vsi *vsi_cur; +	u16 idx; + +	idx = vsig & ICE_VSIG_IDX_M; +	if (idx >= ICE_MAX_VSIGS) +		return ICE_ERR_PARAM; + +	if (!hw->blk[blk].xlt2.vsig_tbl[idx].in_use) +		return ICE_ERR_DOES_NOT_EXIST; + +	hw->blk[blk].xlt2.vsig_tbl[idx].in_use = false; + +	vsi_cur = hw->blk[blk].xlt2.vsig_tbl[idx].first_vsi; +	/* If the VSIG has at least 1 VSI then iterate through the +	 * list and remove the VSIs before deleting the group. +	 */ +	if (vsi_cur) { +		/* remove all vsis associated with this VSIG XLT2 entry */ +		do { +			struct ice_vsig_vsi *tmp = vsi_cur->next_vsi; + +			vsi_cur->vsig = ICE_DEFAULT_VSIG; +			vsi_cur->changed = 1; +			vsi_cur->next_vsi = NULL; +			vsi_cur = tmp; +		} while (vsi_cur); + +		/* NULL terminate head of VSI list */ +		hw->blk[blk].xlt2.vsig_tbl[idx].first_vsi = NULL; +	} + +	/* free characteristic list */ +	list_for_each_entry_safe(del, dtmp, +				 &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +				 list) { +		list_del(&del->list); +		devm_kfree(ice_hw_to_dev(hw), del); +	} + +	/* if VSIG characteristic list was cleared for reset +	 * re-initialize the list head +	 */ +	INIT_LIST_HEAD(&hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst); + +	return 0; +} + +/**   * ice_vsig_remove_vsi - remove VSI from VSIG   * @hw: pointer to the hardware structure   * @blk: HW block @@ -1117,6 +1795,215 @@ ice_vsig_add_mv_vsi(struct ice_hw *hw, enum ice_block blk, u16 vsi, u16 vsig)  	return 0;  } +/** + * ice_find_prof_id - find profile ID for a given field vector + * @hw: pointer to the hardware structure + * @blk: HW block + * @fv: field vector to search for + * @prof_id: receives the profile ID + */ +static enum ice_status +ice_find_prof_id(struct ice_hw *hw, enum ice_block blk, +		 struct ice_fv_word *fv, u8 *prof_id) +{ +	struct ice_es *es = &hw->blk[blk].es; +	u16 off, i; + +	for (i = 0; i < es->count; i++) { +		off = i * es->fvw; + +		if (memcmp(&es->t[off], fv, es->fvw * sizeof(*fv))) +			continue; + +		*prof_id = i; +		return 0; +	} + +	return ICE_ERR_DOES_NOT_EXIST; +} + +/** + * ice_prof_id_rsrc_type - get profile ID resource type for a block type + * @blk: the block type + * @rsrc_type: pointer to variable to receive the resource type + */ +static bool ice_prof_id_rsrc_type(enum ice_block blk, u16 *rsrc_type) +{ +	switch (blk) { +	case ICE_BLK_RSS: +		*rsrc_type = ICE_AQC_RES_TYPE_HASH_PROF_BLDR_PROFID; +		break; +	default: +		return false; +	} +	return true; +} + +/** + * ice_tcam_ent_rsrc_type - get TCAM entry resource type for a block type + * @blk: the block type + * @rsrc_type: pointer to variable to receive the resource type + */ +static bool ice_tcam_ent_rsrc_type(enum ice_block blk, u16 *rsrc_type) +{ +	switch (blk) { +	case ICE_BLK_RSS: +		*rsrc_type = ICE_AQC_RES_TYPE_HASH_PROF_BLDR_TCAM; +		break; +	default: +		return false; +	} +	return true; +} + +/** + * ice_alloc_tcam_ent - allocate hardware TCAM entry + * @hw: pointer to the HW struct + * @blk: the block to allocate the TCAM for + * @tcam_idx: pointer to variable to receive the TCAM entry + * + * This function allocates a new entry in a Profile ID TCAM for a specific + * block. + */ +static enum ice_status +ice_alloc_tcam_ent(struct ice_hw *hw, enum ice_block blk, u16 *tcam_idx) +{ +	u16 res_type; + +	if (!ice_tcam_ent_rsrc_type(blk, &res_type)) +		return ICE_ERR_PARAM; + +	return ice_alloc_hw_res(hw, res_type, 1, true, tcam_idx); +} + +/** + * ice_free_tcam_ent - free hardware TCAM entry + * @hw: pointer to the HW struct + * @blk: the block from which to free the TCAM entry + * @tcam_idx: the TCAM entry to free + * + * This function frees an entry in a Profile ID TCAM for a specific block. + */ +static enum ice_status +ice_free_tcam_ent(struct ice_hw *hw, enum ice_block blk, u16 tcam_idx) +{ +	u16 res_type; + +	if (!ice_tcam_ent_rsrc_type(blk, &res_type)) +		return ICE_ERR_PARAM; + +	return ice_free_hw_res(hw, res_type, 1, &tcam_idx); +} + +/** + * ice_alloc_prof_id - allocate profile ID + * @hw: pointer to the HW struct + * @blk: the block to allocate the profile ID for + * @prof_id: pointer to variable to receive the profile ID + * + * This function allocates a new profile ID, which also corresponds to a Field + * Vector (Extraction Sequence) entry. + */ +static enum ice_status +ice_alloc_prof_id(struct ice_hw *hw, enum ice_block blk, u8 *prof_id) +{ +	enum ice_status status; +	u16 res_type; +	u16 get_prof; + +	if (!ice_prof_id_rsrc_type(blk, &res_type)) +		return ICE_ERR_PARAM; + +	status = ice_alloc_hw_res(hw, res_type, 1, false, &get_prof); +	if (!status) +		*prof_id = (u8)get_prof; + +	return status; +} + +/** + * ice_free_prof_id - free profile ID + * @hw: pointer to the HW struct + * @blk: the block from which to free the profile ID + * @prof_id: the profile ID to free + * + * This function frees a profile ID, which also corresponds to a Field Vector. + */ +static enum ice_status +ice_free_prof_id(struct ice_hw *hw, enum ice_block blk, u8 prof_id) +{ +	u16 tmp_prof_id = (u16)prof_id; +	u16 res_type; + +	if (!ice_prof_id_rsrc_type(blk, &res_type)) +		return ICE_ERR_PARAM; + +	return ice_free_hw_res(hw, res_type, 1, &tmp_prof_id); +} + +/** + * ice_prof_inc_ref - increment reference count for profile + * @hw: pointer to the HW struct + * @blk: the block from which to free the profile ID + * @prof_id: the profile ID for which to increment the reference count + */ +static enum ice_status +ice_prof_inc_ref(struct ice_hw *hw, enum ice_block blk, u8 prof_id) +{ +	if (prof_id > hw->blk[blk].es.count) +		return ICE_ERR_PARAM; + +	hw->blk[blk].es.ref_count[prof_id]++; + +	return 0; +} + +/** + * ice_write_es - write an extraction sequence to hardware + * @hw: pointer to the HW struct + * @blk: the block in which to write the extraction sequence + * @prof_id: the profile ID to write + * @fv: pointer to the extraction sequence to write - NULL to clear extraction + */ +static void +ice_write_es(struct ice_hw *hw, enum ice_block blk, u8 prof_id, +	     struct ice_fv_word *fv) +{ +	u16 off; + +	off = prof_id * hw->blk[blk].es.fvw; +	if (!fv) { +		memset(&hw->blk[blk].es.t[off], 0, +		       hw->blk[blk].es.fvw * sizeof(*fv)); +		hw->blk[blk].es.written[prof_id] = false; +	} else { +		memcpy(&hw->blk[blk].es.t[off], fv, +		       hw->blk[blk].es.fvw * sizeof(*fv)); +	} +} + +/** + * ice_prof_dec_ref - decrement reference count for profile + * @hw: pointer to the HW struct + * @blk: the block from which to free the profile ID + * @prof_id: the profile ID for which to decrement the reference count + */ +static enum ice_status +ice_prof_dec_ref(struct ice_hw *hw, enum ice_block blk, u8 prof_id) +{ +	if (prof_id > hw->blk[blk].es.count) +		return ICE_ERR_PARAM; + +	if (hw->blk[blk].es.ref_count[prof_id] > 0) { +		if (!--hw->blk[blk].es.ref_count[prof_id]) { +			ice_write_es(hw, blk, prof_id, NULL); +			return ice_free_prof_id(hw, blk, prof_id); +		} +	} + +	return 0; +} +  /* Block / table section IDs */  static const u32 ice_blk_sids[ICE_BLK_COUNT][ICE_SID_OFF_COUNT] = {  	/* SWITCH */ @@ -1374,16 +2261,85 @@ void ice_fill_blk_tbls(struct ice_hw *hw)  }  /** + * ice_free_prof_map - free profile map + * @hw: pointer to the hardware structure + * @blk_idx: HW block index + */ +static void ice_free_prof_map(struct ice_hw *hw, u8 blk_idx) +{ +	struct ice_es *es = &hw->blk[blk_idx].es; +	struct ice_prof_map *del, *tmp; + +	mutex_lock(&es->prof_map_lock); +	list_for_each_entry_safe(del, tmp, &es->prof_map, list) { +		list_del(&del->list); +		devm_kfree(ice_hw_to_dev(hw), del); +	} +	INIT_LIST_HEAD(&es->prof_map); +	mutex_unlock(&es->prof_map_lock); +} + +/** + * ice_free_flow_profs - free flow profile entries + * @hw: pointer to the hardware structure + * @blk_idx: HW block index + */ +static void ice_free_flow_profs(struct ice_hw *hw, u8 blk_idx) +{ +	struct ice_flow_prof *p, *tmp; + +	mutex_lock(&hw->fl_profs_locks[blk_idx]); +	list_for_each_entry_safe(p, tmp, &hw->fl_profs[blk_idx], l_entry) { +		list_del(&p->l_entry); +		devm_kfree(ice_hw_to_dev(hw), p); +	} +	mutex_unlock(&hw->fl_profs_locks[blk_idx]); + +	/* if driver is in reset and tables are being cleared +	 * re-initialize the flow profile list heads +	 */ +	INIT_LIST_HEAD(&hw->fl_profs[blk_idx]); +} + +/** + * ice_free_vsig_tbl - free complete VSIG table entries + * @hw: pointer to the hardware structure + * @blk: the HW block on which to free the VSIG table entries + */ +static void ice_free_vsig_tbl(struct ice_hw *hw, enum ice_block blk) +{ +	u16 i; + +	if (!hw->blk[blk].xlt2.vsig_tbl) +		return; + +	for (i = 1; i < ICE_MAX_VSIGS; i++) +		if (hw->blk[blk].xlt2.vsig_tbl[i].in_use) +			ice_vsig_free(hw, blk, i); +} + +/**   * ice_free_hw_tbls - free hardware table memory   * @hw: pointer to the hardware structure   */  void ice_free_hw_tbls(struct ice_hw *hw)  { +	struct ice_rss_cfg *r, *rt;  	u8 i;  	for (i = 0; i < ICE_BLK_COUNT; i++) { -		hw->blk[i].is_list_init = false; +		if (hw->blk[i].is_list_init) { +			struct ice_es *es = &hw->blk[i].es; + +			ice_free_prof_map(hw, i); +			mutex_destroy(&es->prof_map_lock); +			ice_free_flow_profs(hw, i); +			mutex_destroy(&hw->fl_profs_locks[i]); + +			hw->blk[i].is_list_init = false; +		} +		ice_free_vsig_tbl(hw, (enum ice_block)i);  		devm_kfree(ice_hw_to_dev(hw), hw->blk[i].xlt1.ptypes);  		devm_kfree(ice_hw_to_dev(hw), hw->blk[i].xlt1.ptg_tbl);  		devm_kfree(ice_hw_to_dev(hw), hw->blk[i].xlt1.t); @@ -1397,10 +2353,26 @@ void ice_free_hw_tbls(struct ice_hw *hw)  		devm_kfree(ice_hw_to_dev(hw), hw->blk[i].es.written);  	} +	list_for_each_entry_safe(r, rt, &hw->rss_list_head, l_entry) { +		list_del(&r->l_entry); +		devm_kfree(ice_hw_to_dev(hw), r); +	} +	mutex_destroy(&hw->rss_locks);  	memset(hw->blk, 0, sizeof(hw->blk));  }  /** + * ice_init_flow_profs - init flow profile locks and list heads + * @hw: pointer to the hardware structure + * @blk_idx: HW block index + */ +static void ice_init_flow_profs(struct ice_hw *hw, u8 blk_idx) +{ +	mutex_init(&hw->fl_profs_locks[blk_idx]); +	INIT_LIST_HEAD(&hw->fl_profs[blk_idx]); +} + +/**   * ice_clear_hw_tbls - clear HW tables and flow profiles   * @hw: pointer to the hardware structure   */ @@ -1415,6 +2387,13 @@ void ice_clear_hw_tbls(struct ice_hw *hw)  		struct ice_xlt2 *xlt2 = &hw->blk[i].xlt2;  		struct ice_es *es = &hw->blk[i].es; +		if (hw->blk[i].is_list_init) { +			ice_free_prof_map(hw, i); +			ice_free_flow_profs(hw, i); +		} + +		ice_free_vsig_tbl(hw, (enum ice_block)i); +  		memset(xlt1->ptypes, 0, xlt1->count * sizeof(*xlt1->ptypes));  		memset(xlt1->ptg_tbl, 0,  		       ICE_MAX_PTGS * sizeof(*xlt1->ptg_tbl)); @@ -1443,6 +2422,8 @@ enum ice_status ice_init_hw_tbls(struct ice_hw *hw)  {  	u8 i; +	mutex_init(&hw->rss_locks); +	INIT_LIST_HEAD(&hw->rss_list_head);  	for (i = 0; i < ICE_BLK_COUNT; i++) {  		struct ice_prof_redir *prof_redir = &hw->blk[i].prof_redir;  		struct ice_prof_tcam *prof = &hw->blk[i].prof; @@ -1454,6 +2435,9 @@ enum ice_status ice_init_hw_tbls(struct ice_hw *hw)  		if (hw->blk[i].is_list_init)  			continue; +		ice_init_flow_profs(hw, i); +		mutex_init(&es->prof_map_lock); +		INIT_LIST_HEAD(&es->prof_map);  		hw->blk[i].is_list_init = true;  		hw->blk[i].overwrite = blk_sizes[i].overwrite; @@ -1547,3 +2531,1580 @@ err:  	ice_free_hw_tbls(hw);  	return ICE_ERR_NO_MEMORY;  } + +/** + * ice_prof_gen_key - generate profile ID key + * @hw: pointer to the HW struct + * @blk: the block in which to write profile ID to + * @ptg: packet type group (PTG) portion of key + * @vsig: VSIG portion of key + * @cdid: CDID portion of key + * @flags: flag portion of key + * @vl_msk: valid mask + * @dc_msk: don't care mask + * @nm_msk: never match mask + * @key: output of profile ID key + */ +static enum ice_status +ice_prof_gen_key(struct ice_hw *hw, enum ice_block blk, u8 ptg, u16 vsig, +		 u8 cdid, u16 flags, u8 vl_msk[ICE_TCAM_KEY_VAL_SZ], +		 u8 dc_msk[ICE_TCAM_KEY_VAL_SZ], u8 nm_msk[ICE_TCAM_KEY_VAL_SZ], +		 u8 key[ICE_TCAM_KEY_SZ]) +{ +	struct ice_prof_id_key inkey; + +	inkey.xlt1 = ptg; +	inkey.xlt2_cdid = cpu_to_le16(vsig); +	inkey.flags = cpu_to_le16(flags); + +	switch (hw->blk[blk].prof.cdid_bits) { +	case 0: +		break; +	case 2: +#define ICE_CD_2_M 0xC000U +#define ICE_CD_2_S 14 +		inkey.xlt2_cdid &= ~cpu_to_le16(ICE_CD_2_M); +		inkey.xlt2_cdid |= cpu_to_le16(BIT(cdid) << ICE_CD_2_S); +		break; +	case 4: +#define ICE_CD_4_M 0xF000U +#define ICE_CD_4_S 12 +		inkey.xlt2_cdid &= ~cpu_to_le16(ICE_CD_4_M); +		inkey.xlt2_cdid |= cpu_to_le16(BIT(cdid) << ICE_CD_4_S); +		break; +	case 8: +#define ICE_CD_8_M 0xFF00U +#define ICE_CD_8_S 16 +		inkey.xlt2_cdid &= ~cpu_to_le16(ICE_CD_8_M); +		inkey.xlt2_cdid |= cpu_to_le16(BIT(cdid) << ICE_CD_8_S); +		break; +	default: +		ice_debug(hw, ICE_DBG_PKG, "Error in profile config\n"); +		break; +	} + +	return ice_set_key(key, ICE_TCAM_KEY_SZ, (u8 *)&inkey, vl_msk, dc_msk, +			   nm_msk, 0, ICE_TCAM_KEY_SZ / 2); +} + +/** + * ice_tcam_write_entry - write TCAM entry + * @hw: pointer to the HW struct + * @blk: the block in which to write profile ID to + * @idx: the entry index to write to + * @prof_id: profile ID + * @ptg: packet type group (PTG) portion of key + * @vsig: VSIG portion of key + * @cdid: CDID portion of key + * @flags: flag portion of key + * @vl_msk: valid mask + * @dc_msk: don't care mask + * @nm_msk: never match mask + */ +static enum ice_status +ice_tcam_write_entry(struct ice_hw *hw, enum ice_block blk, u16 idx, +		     u8 prof_id, u8 ptg, u16 vsig, u8 cdid, u16 flags, +		     u8 vl_msk[ICE_TCAM_KEY_VAL_SZ], +		     u8 dc_msk[ICE_TCAM_KEY_VAL_SZ], +		     u8 nm_msk[ICE_TCAM_KEY_VAL_SZ]) +{ +	struct ice_prof_tcam_entry; +	enum ice_status status; + +	status = ice_prof_gen_key(hw, blk, ptg, vsig, cdid, flags, vl_msk, +				  dc_msk, nm_msk, hw->blk[blk].prof.t[idx].key); +	if (!status) { +		hw->blk[blk].prof.t[idx].addr = cpu_to_le16(idx); +		hw->blk[blk].prof.t[idx].prof_id = prof_id; +	} + +	return status; +} + +/** + * ice_vsig_get_ref - returns number of VSIs belong to a VSIG + * @hw: pointer to the hardware structure + * @blk: HW block + * @vsig: VSIG to query + * @refs: pointer to variable to receive the reference count + */ +static enum ice_status +ice_vsig_get_ref(struct ice_hw *hw, enum ice_block blk, u16 vsig, u16 *refs) +{ +	u16 idx = vsig & ICE_VSIG_IDX_M; +	struct ice_vsig_vsi *ptr; + +	*refs = 0; + +	if (!hw->blk[blk].xlt2.vsig_tbl[idx].in_use) +		return ICE_ERR_DOES_NOT_EXIST; + +	ptr = hw->blk[blk].xlt2.vsig_tbl[idx].first_vsi; +	while (ptr) { +		(*refs)++; +		ptr = ptr->next_vsi; +	} + +	return 0; +} + +/** + * ice_has_prof_vsig - check to see if VSIG has a specific profile + * @hw: pointer to the hardware structure + * @blk: HW block + * @vsig: VSIG to check against + * @hdl: profile handle + */ +static bool +ice_has_prof_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, u64 hdl) +{ +	u16 idx = vsig & ICE_VSIG_IDX_M; +	struct ice_vsig_prof *ent; + +	list_for_each_entry(ent, &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +			    list) +		if (ent->profile_cookie == hdl) +			return true; + +	ice_debug(hw, ICE_DBG_INIT, +		  "Characteristic list for VSI group %d not found.\n", +		  vsig); +	return false; +} + +/** + * ice_prof_bld_es - build profile ID extraction sequence changes + * @hw: pointer to the HW struct + * @blk: hardware block + * @bld: the update package buffer build to add to + * @chgs: the list of changes to make in hardware + */ +static enum ice_status +ice_prof_bld_es(struct ice_hw *hw, enum ice_block blk, +		struct ice_buf_build *bld, struct list_head *chgs) +{ +	u16 vec_size = hw->blk[blk].es.fvw * sizeof(struct ice_fv_word); +	struct ice_chs_chg *tmp; + +	list_for_each_entry(tmp, chgs, list_entry) +		if (tmp->type == ICE_PTG_ES_ADD && tmp->add_prof) { +			u16 off = tmp->prof_id * hw->blk[blk].es.fvw; +			struct ice_pkg_es *p; +			u32 id; + +			id = ice_sect_id(blk, ICE_VEC_TBL); +			p = (struct ice_pkg_es *) +				ice_pkg_buf_alloc_section(bld, id, sizeof(*p) + +							  vec_size - +							  sizeof(p->es[0])); + +			if (!p) +				return ICE_ERR_MAX_LIMIT; + +			p->count = cpu_to_le16(1); +			p->offset = cpu_to_le16(tmp->prof_id); + +			memcpy(p->es, &hw->blk[blk].es.t[off], vec_size); +		} + +	return 0; +} + +/** + * ice_prof_bld_tcam - build profile ID TCAM changes + * @hw: pointer to the HW struct + * @blk: hardware block + * @bld: the update package buffer build to add to + * @chgs: the list of changes to make in hardware + */ +static enum ice_status +ice_prof_bld_tcam(struct ice_hw *hw, enum ice_block blk, +		  struct ice_buf_build *bld, struct list_head *chgs) +{ +	struct ice_chs_chg *tmp; + +	list_for_each_entry(tmp, chgs, list_entry) +		if (tmp->type == ICE_TCAM_ADD && tmp->add_tcam_idx) { +			struct ice_prof_id_section *p; +			u32 id; + +			id = ice_sect_id(blk, ICE_PROF_TCAM); +			p = (struct ice_prof_id_section *) +				ice_pkg_buf_alloc_section(bld, id, sizeof(*p)); + +			if (!p) +				return ICE_ERR_MAX_LIMIT; + +			p->count = cpu_to_le16(1); +			p->entry[0].addr = cpu_to_le16(tmp->tcam_idx); +			p->entry[0].prof_id = tmp->prof_id; + +			memcpy(p->entry[0].key, +			       &hw->blk[blk].prof.t[tmp->tcam_idx].key, +			       sizeof(hw->blk[blk].prof.t->key)); +		} + +	return 0; +} + +/** + * ice_prof_bld_xlt1 - build XLT1 changes + * @blk: hardware block + * @bld: the update package buffer build to add to + * @chgs: the list of changes to make in hardware + */ +static enum ice_status +ice_prof_bld_xlt1(enum ice_block blk, struct ice_buf_build *bld, +		  struct list_head *chgs) +{ +	struct ice_chs_chg *tmp; + +	list_for_each_entry(tmp, chgs, list_entry) +		if (tmp->type == ICE_PTG_ES_ADD && tmp->add_ptg) { +			struct ice_xlt1_section *p; +			u32 id; + +			id = ice_sect_id(blk, ICE_XLT1); +			p = (struct ice_xlt1_section *) +				ice_pkg_buf_alloc_section(bld, id, sizeof(*p)); + +			if (!p) +				return ICE_ERR_MAX_LIMIT; + +			p->count = cpu_to_le16(1); +			p->offset = cpu_to_le16(tmp->ptype); +			p->value[0] = tmp->ptg; +		} + +	return 0; +} + +/** + * ice_prof_bld_xlt2 - build XLT2 changes + * @blk: hardware block + * @bld: the update package buffer build to add to + * @chgs: the list of changes to make in hardware + */ +static enum ice_status +ice_prof_bld_xlt2(enum ice_block blk, struct ice_buf_build *bld, +		  struct list_head *chgs) +{ +	struct ice_chs_chg *tmp; + +	list_for_each_entry(tmp, chgs, list_entry) { +		struct ice_xlt2_section *p; +		u32 id; + +		switch (tmp->type) { +		case ICE_VSIG_ADD: +		case ICE_VSI_MOVE: +		case ICE_VSIG_REM: +			id = ice_sect_id(blk, ICE_XLT2); +			p = (struct ice_xlt2_section *) +				ice_pkg_buf_alloc_section(bld, id, sizeof(*p)); + +			if (!p) +				return ICE_ERR_MAX_LIMIT; + +			p->count = cpu_to_le16(1); +			p->offset = cpu_to_le16(tmp->vsi); +			p->value[0] = cpu_to_le16(tmp->vsig); +			break; +		default: +			break; +		} +	} + +	return 0; +} + +/** + * ice_upd_prof_hw - update hardware using the change list + * @hw: pointer to the HW struct + * @blk: hardware block + * @chgs: the list of changes to make in hardware + */ +static enum ice_status +ice_upd_prof_hw(struct ice_hw *hw, enum ice_block blk, +		struct list_head *chgs) +{ +	struct ice_buf_build *b; +	struct ice_chs_chg *tmp; +	enum ice_status status; +	u16 pkg_sects; +	u16 xlt1 = 0; +	u16 xlt2 = 0; +	u16 tcam = 0; +	u16 es = 0; +	u16 sects; + +	/* count number of sections we need */ +	list_for_each_entry(tmp, chgs, list_entry) { +		switch (tmp->type) { +		case ICE_PTG_ES_ADD: +			if (tmp->add_ptg) +				xlt1++; +			if (tmp->add_prof) +				es++; +			break; +		case ICE_TCAM_ADD: +			tcam++; +			break; +		case ICE_VSIG_ADD: +		case ICE_VSI_MOVE: +		case ICE_VSIG_REM: +			xlt2++; +			break; +		default: +			break; +		} +	} +	sects = xlt1 + xlt2 + tcam + es; + +	if (!sects) +		return 0; + +	/* Build update package buffer */ +	b = ice_pkg_buf_alloc(hw); +	if (!b) +		return ICE_ERR_NO_MEMORY; + +	status = ice_pkg_buf_reserve_section(b, sects); +	if (status) +		goto error_tmp; + +	/* Preserve order of table update: ES, TCAM, PTG, VSIG */ +	if (es) { +		status = ice_prof_bld_es(hw, blk, b, chgs); +		if (status) +			goto error_tmp; +	} + +	if (tcam) { +		status = ice_prof_bld_tcam(hw, blk, b, chgs); +		if (status) +			goto error_tmp; +	} + +	if (xlt1) { +		status = ice_prof_bld_xlt1(blk, b, chgs); +		if (status) +			goto error_tmp; +	} + +	if (xlt2) { +		status = ice_prof_bld_xlt2(blk, b, chgs); +		if (status) +			goto error_tmp; +	} + +	/* After package buffer build check if the section count in buffer is +	 * non-zero and matches the number of sections detected for package +	 * update. +	 */ +	pkg_sects = ice_pkg_buf_get_active_sections(b); +	if (!pkg_sects || pkg_sects != sects) { +		status = ICE_ERR_INVAL_SIZE; +		goto error_tmp; +	} + +	/* update package */ +	status = ice_update_pkg(hw, ice_pkg_buf(b), 1); +	if (status == ICE_ERR_AQ_ERROR) +		ice_debug(hw, ICE_DBG_INIT, "Unable to update HW profile\n"); + +error_tmp: +	ice_pkg_buf_free(hw, b); +	return status; +} + +/** + * ice_add_prof - add profile + * @hw: pointer to the HW struct + * @blk: hardware block + * @id: profile tracking ID + * @ptypes: array of bitmaps indicating ptypes (ICE_FLOW_PTYPE_MAX bits) + * @es: extraction sequence (length of array is determined by the block) + * + * This function registers a profile, which matches a set of PTGs with a + * particular extraction sequence. While the hardware profile is allocated + * it will not be written until the first call to ice_add_flow that specifies + * the ID value used here. + */ +enum ice_status +ice_add_prof(struct ice_hw *hw, enum ice_block blk, u64 id, u8 ptypes[], +	     struct ice_fv_word *es) +{ +	u32 bytes = DIV_ROUND_UP(ICE_FLOW_PTYPE_MAX, BITS_PER_BYTE); +	DECLARE_BITMAP(ptgs_used, ICE_XLT1_CNT); +	struct ice_prof_map *prof; +	enum ice_status status; +	u32 byte = 0; +	u8 prof_id; + +	bitmap_zero(ptgs_used, ICE_XLT1_CNT); + +	mutex_lock(&hw->blk[blk].es.prof_map_lock); + +	/* search for existing profile */ +	status = ice_find_prof_id(hw, blk, es, &prof_id); +	if (status) { +		/* allocate profile ID */ +		status = ice_alloc_prof_id(hw, blk, &prof_id); +		if (status) +			goto err_ice_add_prof; + +		/* and write new es */ +		ice_write_es(hw, blk, prof_id, es); +	} + +	ice_prof_inc_ref(hw, blk, prof_id); + +	/* add profile info */ +	prof = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*prof), GFP_KERNEL); +	if (!prof) +		goto err_ice_add_prof; + +	prof->profile_cookie = id; +	prof->prof_id = prof_id; +	prof->ptg_cnt = 0; +	prof->context = 0; + +	/* build list of ptgs */ +	while (bytes && prof->ptg_cnt < ICE_MAX_PTG_PER_PROFILE) { +		u32 bit; + +		if (!ptypes[byte]) { +			bytes--; +			byte++; +			continue; +		} + +		/* Examine 8 bits per byte */ +		for_each_set_bit(bit, (unsigned long *)&ptypes[byte], +				 BITS_PER_BYTE) { +			u16 ptype; +			u8 ptg; +			u8 m; + +			ptype = byte * BITS_PER_BYTE + bit; + +			/* The package should place all ptypes in a non-zero +			 * PTG, so the following call should never fail. +			 */ +			if (ice_ptg_find_ptype(hw, blk, ptype, &ptg)) +				continue; + +			/* If PTG is already added, skip and continue */ +			if (test_bit(ptg, ptgs_used)) +				continue; + +			set_bit(ptg, ptgs_used); +			prof->ptg[prof->ptg_cnt] = ptg; + +			if (++prof->ptg_cnt >= ICE_MAX_PTG_PER_PROFILE) +				break; + +			/* nothing left in byte, then exit */ +			m = ~((1 << (bit + 1)) - 1); +			if (!(ptypes[byte] & m)) +				break; +		} + +		bytes--; +		byte++; +	} + +	list_add(&prof->list, &hw->blk[blk].es.prof_map); +	status = 0; + +err_ice_add_prof: +	mutex_unlock(&hw->blk[blk].es.prof_map_lock); +	return status; +} + +/** + * ice_search_prof_id_low - Search for a profile tracking ID low level + * @hw: pointer to the HW struct + * @blk: hardware block + * @id: profile tracking ID + * + * This will search for a profile tracking ID which was previously added. This + * version assumes that the caller has already acquired the prof map lock. + */ +static struct ice_prof_map * +ice_search_prof_id_low(struct ice_hw *hw, enum ice_block blk, u64 id) +{ +	struct ice_prof_map *entry = NULL; +	struct ice_prof_map *map; + +	list_for_each_entry(map, &hw->blk[blk].es.prof_map, list) +		if (map->profile_cookie == id) { +			entry = map; +			break; +		} + +	return entry; +} + +/** + * ice_search_prof_id - Search for a profile tracking ID + * @hw: pointer to the HW struct + * @blk: hardware block + * @id: profile tracking ID + * + * This will search for a profile tracking ID which was previously added. + */ +static struct ice_prof_map * +ice_search_prof_id(struct ice_hw *hw, enum ice_block blk, u64 id) +{ +	struct ice_prof_map *entry; + +	mutex_lock(&hw->blk[blk].es.prof_map_lock); +	entry = ice_search_prof_id_low(hw, blk, id); +	mutex_unlock(&hw->blk[blk].es.prof_map_lock); + +	return entry; +} + +/** + * ice_vsig_prof_id_count - count profiles in a VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsig: VSIG to remove the profile from + */ +static u16 +ice_vsig_prof_id_count(struct ice_hw *hw, enum ice_block blk, u16 vsig) +{ +	u16 idx = vsig & ICE_VSIG_IDX_M, count = 0; +	struct ice_vsig_prof *p; + +	list_for_each_entry(p, &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +			    list) +		count++; + +	return count; +} + +/** + * ice_rel_tcam_idx - release a TCAM index + * @hw: pointer to the HW struct + * @blk: hardware block + * @idx: the index to release + */ +static enum ice_status +ice_rel_tcam_idx(struct ice_hw *hw, enum ice_block blk, u16 idx) +{ +	/* Masks to invoke a never match entry */ +	u8 vl_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }; +	u8 dc_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFE, 0xFF, 0xFF, 0xFF, 0xFF }; +	u8 nm_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x01, 0x00, 0x00, 0x00, 0x00 }; +	enum ice_status status; + +	/* write the TCAM entry */ +	status = ice_tcam_write_entry(hw, blk, idx, 0, 0, 0, 0, 0, vl_msk, +				      dc_msk, nm_msk); +	if (status) +		return status; + +	/* release the TCAM entry */ +	status = ice_free_tcam_ent(hw, blk, idx); + +	return status; +} + +/** + * ice_rem_prof_id - remove one profile from a VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @prof: pointer to profile structure to remove + */ +static enum ice_status +ice_rem_prof_id(struct ice_hw *hw, enum ice_block blk, +		struct ice_vsig_prof *prof) +{ +	enum ice_status status; +	u16 i; + +	for (i = 0; i < prof->tcam_count; i++) +		if (prof->tcam[i].in_use) { +			prof->tcam[i].in_use = false; +			status = ice_rel_tcam_idx(hw, blk, +						  prof->tcam[i].tcam_idx); +			if (status) +				return ICE_ERR_HW_TABLE; +		} + +	return 0; +} + +/** + * ice_rem_vsig - remove VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsig: the VSIG to remove + * @chg: the change list + */ +static enum ice_status +ice_rem_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, +	     struct list_head *chg) +{ +	u16 idx = vsig & ICE_VSIG_IDX_M; +	struct ice_vsig_vsi *vsi_cur; +	struct ice_vsig_prof *d, *t; +	enum ice_status status; + +	/* remove TCAM entries */ +	list_for_each_entry_safe(d, t, +				 &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +				 list) { +		status = ice_rem_prof_id(hw, blk, d); +		if (status) +			return status; + +		list_del(&d->list); +		devm_kfree(ice_hw_to_dev(hw), d); +	} + +	/* Move all VSIS associated with this VSIG to the default VSIG */ +	vsi_cur = hw->blk[blk].xlt2.vsig_tbl[idx].first_vsi; +	/* If the VSIG has at least 1 VSI then iterate through the list +	 * and remove the VSIs before deleting the group. +	 */ +	if (vsi_cur) +		do { +			struct ice_vsig_vsi *tmp = vsi_cur->next_vsi; +			struct ice_chs_chg *p; + +			p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), +					 GFP_KERNEL); +			if (!p) +				return ICE_ERR_NO_MEMORY; + +			p->type = ICE_VSIG_REM; +			p->orig_vsig = vsig; +			p->vsig = ICE_DEFAULT_VSIG; +			p->vsi = vsi_cur - hw->blk[blk].xlt2.vsis; + +			list_add(&p->list_entry, chg); + +			vsi_cur = tmp; +		} while (vsi_cur); + +	return ice_vsig_free(hw, blk, vsig); +} + +/** + * ice_rem_prof_id_vsig - remove a specific profile from a VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsig: VSIG to remove the profile from + * @hdl: profile handle indicating which profile to remove + * @chg: list to receive a record of changes + */ +static enum ice_status +ice_rem_prof_id_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, u64 hdl, +		     struct list_head *chg) +{ +	u16 idx = vsig & ICE_VSIG_IDX_M; +	struct ice_vsig_prof *p, *t; +	enum ice_status status; + +	list_for_each_entry_safe(p, t, +				 &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +				 list) +		if (p->profile_cookie == hdl) { +			if (ice_vsig_prof_id_count(hw, blk, vsig) == 1) +				/* this is the last profile, remove the VSIG */ +				return ice_rem_vsig(hw, blk, vsig, chg); + +			status = ice_rem_prof_id(hw, blk, p); +			if (!status) { +				list_del(&p->list); +				devm_kfree(ice_hw_to_dev(hw), p); +			} +			return status; +		} + +	return ICE_ERR_DOES_NOT_EXIST; +} + +/** + * ice_rem_flow_all - remove all flows with a particular profile + * @hw: pointer to the HW struct + * @blk: hardware block + * @id: profile tracking ID + */ +static enum ice_status +ice_rem_flow_all(struct ice_hw *hw, enum ice_block blk, u64 id) +{ +	struct ice_chs_chg *del, *tmp; +	enum ice_status status; +	struct list_head chg; +	u16 i; + +	INIT_LIST_HEAD(&chg); + +	for (i = 1; i < ICE_MAX_VSIGS; i++) +		if (hw->blk[blk].xlt2.vsig_tbl[i].in_use) { +			if (ice_has_prof_vsig(hw, blk, i, id)) { +				status = ice_rem_prof_id_vsig(hw, blk, i, id, +							      &chg); +				if (status) +					goto err_ice_rem_flow_all; +			} +		} + +	status = ice_upd_prof_hw(hw, blk, &chg); + +err_ice_rem_flow_all: +	list_for_each_entry_safe(del, tmp, &chg, list_entry) { +		list_del(&del->list_entry); +		devm_kfree(ice_hw_to_dev(hw), del); +	} + +	return status; +} + +/** + * ice_rem_prof - remove profile + * @hw: pointer to the HW struct + * @blk: hardware block + * @id: profile tracking ID + * + * This will remove the profile specified by the ID parameter, which was + * previously created through ice_add_prof. If any existing entries + * are associated with this profile, they will be removed as well. + */ +enum ice_status ice_rem_prof(struct ice_hw *hw, enum ice_block blk, u64 id) +{ +	struct ice_prof_map *pmap; +	enum ice_status status; + +	mutex_lock(&hw->blk[blk].es.prof_map_lock); + +	pmap = ice_search_prof_id_low(hw, blk, id); +	if (!pmap) { +		status = ICE_ERR_DOES_NOT_EXIST; +		goto err_ice_rem_prof; +	} + +	/* remove all flows with this profile */ +	status = ice_rem_flow_all(hw, blk, pmap->profile_cookie); +	if (status) +		goto err_ice_rem_prof; + +	/* dereference profile, and possibly remove */ +	ice_prof_dec_ref(hw, blk, pmap->prof_id); + +	list_del(&pmap->list); +	devm_kfree(ice_hw_to_dev(hw), pmap); + +err_ice_rem_prof: +	mutex_unlock(&hw->blk[blk].es.prof_map_lock); +	return status; +} + +/** + * ice_get_prof - get profile + * @hw: pointer to the HW struct + * @blk: hardware block + * @hdl: profile handle + * @chg: change list + */ +static enum ice_status +ice_get_prof(struct ice_hw *hw, enum ice_block blk, u64 hdl, +	     struct list_head *chg) +{ +	struct ice_prof_map *map; +	struct ice_chs_chg *p; +	u16 i; + +	/* Get the details on the profile specified by the handle ID */ +	map = ice_search_prof_id(hw, blk, hdl); +	if (!map) +		return ICE_ERR_DOES_NOT_EXIST; + +	for (i = 0; i < map->ptg_cnt; i++) +		if (!hw->blk[blk].es.written[map->prof_id]) { +			/* add ES to change list */ +			p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), +					 GFP_KERNEL); +			if (!p) +				goto err_ice_get_prof; + +			p->type = ICE_PTG_ES_ADD; +			p->ptype = 0; +			p->ptg = map->ptg[i]; +			p->add_ptg = 0; + +			p->add_prof = 1; +			p->prof_id = map->prof_id; + +			hw->blk[blk].es.written[map->prof_id] = true; + +			list_add(&p->list_entry, chg); +		} + +	return 0; + +err_ice_get_prof: +	/* let caller clean up the change list */ +	return ICE_ERR_NO_MEMORY; +} + +/** + * ice_get_profs_vsig - get a copy of the list of profiles from a VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsig: VSIG from which to copy the list + * @lst: output list + * + * This routine makes a copy of the list of profiles in the specified VSIG. + */ +static enum ice_status +ice_get_profs_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, +		   struct list_head *lst) +{ +	struct ice_vsig_prof *ent1, *ent2; +	u16 idx = vsig & ICE_VSIG_IDX_M; + +	list_for_each_entry(ent1, &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +			    list) { +		struct ice_vsig_prof *p; + +		/* copy to the input list */ +		p = devm_kmemdup(ice_hw_to_dev(hw), ent1, sizeof(*p), +				 GFP_KERNEL); +		if (!p) +			goto err_ice_get_profs_vsig; + +		list_add_tail(&p->list, lst); +	} + +	return 0; + +err_ice_get_profs_vsig: +	list_for_each_entry_safe(ent1, ent2, lst, list) { +		list_del(&ent1->list); +		devm_kfree(ice_hw_to_dev(hw), ent1); +	} + +	return ICE_ERR_NO_MEMORY; +} + +/** + * ice_add_prof_to_lst - add profile entry to a list + * @hw: pointer to the HW struct + * @blk: hardware block + * @lst: the list to be added to + * @hdl: profile handle of entry to add + */ +static enum ice_status +ice_add_prof_to_lst(struct ice_hw *hw, enum ice_block blk, +		    struct list_head *lst, u64 hdl) +{ +	struct ice_prof_map *map; +	struct ice_vsig_prof *p; +	u16 i; + +	map = ice_search_prof_id(hw, blk, hdl); +	if (!map) +		return ICE_ERR_DOES_NOT_EXIST; + +	p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); +	if (!p) +		return ICE_ERR_NO_MEMORY; + +	p->profile_cookie = map->profile_cookie; +	p->prof_id = map->prof_id; +	p->tcam_count = map->ptg_cnt; + +	for (i = 0; i < map->ptg_cnt; i++) { +		p->tcam[i].prof_id = map->prof_id; +		p->tcam[i].tcam_idx = ICE_INVALID_TCAM; +		p->tcam[i].ptg = map->ptg[i]; +	} + +	list_add(&p->list, lst); + +	return 0; +} + +/** + * ice_move_vsi - move VSI to another VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsi: the VSI to move + * @vsig: the VSIG to move the VSI to + * @chg: the change list + */ +static enum ice_status +ice_move_vsi(struct ice_hw *hw, enum ice_block blk, u16 vsi, u16 vsig, +	     struct list_head *chg) +{ +	enum ice_status status; +	struct ice_chs_chg *p; +	u16 orig_vsig; + +	p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); +	if (!p) +		return ICE_ERR_NO_MEMORY; + +	status = ice_vsig_find_vsi(hw, blk, vsi, &orig_vsig); +	if (!status) +		status = ice_vsig_add_mv_vsi(hw, blk, vsi, vsig); + +	if (status) { +		devm_kfree(ice_hw_to_dev(hw), p); +		return status; +	} + +	p->type = ICE_VSI_MOVE; +	p->vsi = vsi; +	p->orig_vsig = orig_vsig; +	p->vsig = vsig; + +	list_add(&p->list_entry, chg); + +	return 0; +} + +/** + * ice_prof_tcam_ena_dis - add enable or disable TCAM change + * @hw: pointer to the HW struct + * @blk: hardware block + * @enable: true to enable, false to disable + * @vsig: the VSIG of the TCAM entry + * @tcam: pointer the TCAM info structure of the TCAM to disable + * @chg: the change list + * + * This function appends an enable or disable TCAM entry in the change log + */ +static enum ice_status +ice_prof_tcam_ena_dis(struct ice_hw *hw, enum ice_block blk, bool enable, +		      u16 vsig, struct ice_tcam_inf *tcam, +		      struct list_head *chg) +{ +	enum ice_status status; +	struct ice_chs_chg *p; + +	/* Default: enable means change the low flag bit to don't care */ +	u8 dc_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x01, 0x00, 0x00, 0x00, 0x00 }; +	u8 nm_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x00, 0x00, 0x00, 0x00, 0x00 }; +	u8 vl_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x01, 0x00, 0x00, 0x00, 0x00 }; + +	/* if disabling, free the TCAM */ +	if (!enable) { +		status = ice_free_tcam_ent(hw, blk, tcam->tcam_idx); +		tcam->tcam_idx = 0; +		tcam->in_use = 0; +		return status; +	} + +	/* for re-enabling, reallocate a TCAM */ +	status = ice_alloc_tcam_ent(hw, blk, &tcam->tcam_idx); +	if (status) +		return status; + +	/* add TCAM to change list */ +	p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); +	if (!p) +		return ICE_ERR_NO_MEMORY; + +	status = ice_tcam_write_entry(hw, blk, tcam->tcam_idx, tcam->prof_id, +				      tcam->ptg, vsig, 0, 0, vl_msk, dc_msk, +				      nm_msk); +	if (status) +		goto err_ice_prof_tcam_ena_dis; + +	tcam->in_use = 1; + +	p->type = ICE_TCAM_ADD; +	p->add_tcam_idx = true; +	p->prof_id = tcam->prof_id; +	p->ptg = tcam->ptg; +	p->vsig = 0; +	p->tcam_idx = tcam->tcam_idx; + +	/* log change */ +	list_add(&p->list_entry, chg); + +	return 0; + +err_ice_prof_tcam_ena_dis: +	devm_kfree(ice_hw_to_dev(hw), p); +	return status; +} + +/** + * ice_adj_prof_priorities - adjust profile based on priorities + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsig: the VSIG for which to adjust profile priorities + * @chg: the change list + */ +static enum ice_status +ice_adj_prof_priorities(struct ice_hw *hw, enum ice_block blk, u16 vsig, +			struct list_head *chg) +{ +	DECLARE_BITMAP(ptgs_used, ICE_XLT1_CNT); +	struct ice_vsig_prof *t; +	enum ice_status status; +	u16 idx; + +	bitmap_zero(ptgs_used, ICE_XLT1_CNT); +	idx = vsig & ICE_VSIG_IDX_M; + +	/* Priority is based on the order in which the profiles are added. The +	 * newest added profile has highest priority and the oldest added +	 * profile has the lowest priority. Since the profile property list for +	 * a VSIG is sorted from newest to oldest, this code traverses the list +	 * in order and enables the first of each PTG that it finds (that is not +	 * already enabled); it also disables any duplicate PTGs that it finds +	 * in the older profiles (that are currently enabled). +	 */ + +	list_for_each_entry(t, &hw->blk[blk].xlt2.vsig_tbl[idx].prop_lst, +			    list) { +		u16 i; + +		for (i = 0; i < t->tcam_count; i++) { +			/* Scan the priorities from newest to oldest. +			 * Make sure that the newest profiles take priority. +			 */ +			if (test_bit(t->tcam[i].ptg, ptgs_used) && +			    t->tcam[i].in_use) { +				/* need to mark this PTG as never match, as it +				 * was already in use and therefore duplicate +				 * (and lower priority) +				 */ +				status = ice_prof_tcam_ena_dis(hw, blk, false, +							       vsig, +							       &t->tcam[i], +							       chg); +				if (status) +					return status; +			} else if (!test_bit(t->tcam[i].ptg, ptgs_used) && +				   !t->tcam[i].in_use) { +				/* need to enable this PTG, as it in not in use +				 * and not enabled (highest priority) +				 */ +				status = ice_prof_tcam_ena_dis(hw, blk, true, +							       vsig, +							       &t->tcam[i], +							       chg); +				if (status) +					return status; +			} + +			/* keep track of used ptgs */ +			set_bit(t->tcam[i].ptg, ptgs_used); +		} +	} + +	return 0; +} + +/** + * ice_add_prof_id_vsig - add profile to VSIG + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsig: the VSIG to which this profile is to be added + * @hdl: the profile handle indicating the profile to add + * @chg: the change list + */ +static enum ice_status +ice_add_prof_id_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, u64 hdl, +		     struct list_head *chg) +{ +	/* Masks that ignore flags */ +	u8 vl_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }; +	u8 dc_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0x00, 0x00, 0x00 }; +	u8 nm_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x00, 0x00, 0x00, 0x00, 0x00 }; +	struct ice_prof_map *map; +	struct ice_vsig_prof *t; +	struct ice_chs_chg *p; +	u16 i; + +	/* Get the details on the profile specified by the handle ID */ +	map = ice_search_prof_id(hw, blk, hdl); +	if (!map) +		return ICE_ERR_DOES_NOT_EXIST; + +	/* Error, if this VSIG already has this profile */ +	if (ice_has_prof_vsig(hw, blk, vsig, hdl)) +		return ICE_ERR_ALREADY_EXISTS; + +	/* new VSIG profile structure */ +	t = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*t), GFP_KERNEL); +	if (!t) +		return ICE_ERR_NO_MEMORY; + +	t->profile_cookie = map->profile_cookie; +	t->prof_id = map->prof_id; +	t->tcam_count = map->ptg_cnt; + +	/* create TCAM entries */ +	for (i = 0; i < map->ptg_cnt; i++) { +		enum ice_status status; +		u16 tcam_idx; + +		/* add TCAM to change list */ +		p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); +		if (!p) +			goto err_ice_add_prof_id_vsig; + +		/* allocate the TCAM entry index */ +		status = ice_alloc_tcam_ent(hw, blk, &tcam_idx); +		if (status) { +			devm_kfree(ice_hw_to_dev(hw), p); +			goto err_ice_add_prof_id_vsig; +		} + +		t->tcam[i].ptg = map->ptg[i]; +		t->tcam[i].prof_id = map->prof_id; +		t->tcam[i].tcam_idx = tcam_idx; +		t->tcam[i].in_use = true; + +		p->type = ICE_TCAM_ADD; +		p->add_tcam_idx = true; +		p->prof_id = t->tcam[i].prof_id; +		p->ptg = t->tcam[i].ptg; +		p->vsig = vsig; +		p->tcam_idx = t->tcam[i].tcam_idx; + +		/* write the TCAM entry */ +		status = ice_tcam_write_entry(hw, blk, t->tcam[i].tcam_idx, +					      t->tcam[i].prof_id, +					      t->tcam[i].ptg, vsig, 0, 0, +					      vl_msk, dc_msk, nm_msk); +		if (status) +			goto err_ice_add_prof_id_vsig; + +		/* log change */ +		list_add(&p->list_entry, chg); +	} + +	/* add profile to VSIG */ +	list_add(&t->list, +		 &hw->blk[blk].xlt2.vsig_tbl[(vsig & ICE_VSIG_IDX_M)].prop_lst); + +	return 0; + +err_ice_add_prof_id_vsig: +	/* let caller clean up the change list */ +	devm_kfree(ice_hw_to_dev(hw), t); +	return ICE_ERR_NO_MEMORY; +} + +/** + * ice_create_prof_id_vsig - add a new VSIG with a single profile + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsi: the initial VSI that will be in VSIG + * @hdl: the profile handle of the profile that will be added to the VSIG + * @chg: the change list + */ +static enum ice_status +ice_create_prof_id_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsi, u64 hdl, +			struct list_head *chg) +{ +	enum ice_status status; +	struct ice_chs_chg *p; +	u16 new_vsig; + +	p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); +	if (!p) +		return ICE_ERR_NO_MEMORY; + +	new_vsig = ice_vsig_alloc(hw, blk); +	if (!new_vsig) { +		status = ICE_ERR_HW_TABLE; +		goto err_ice_create_prof_id_vsig; +	} + +	status = ice_move_vsi(hw, blk, vsi, new_vsig, chg); +	if (status) +		goto err_ice_create_prof_id_vsig; + +	status = ice_add_prof_id_vsig(hw, blk, new_vsig, hdl, chg); +	if (status) +		goto err_ice_create_prof_id_vsig; + +	p->type = ICE_VSIG_ADD; +	p->vsi = vsi; +	p->orig_vsig = ICE_DEFAULT_VSIG; +	p->vsig = new_vsig; + +	list_add(&p->list_entry, chg); + +	return 0; + +err_ice_create_prof_id_vsig: +	/* let caller clean up the change list */ +	devm_kfree(ice_hw_to_dev(hw), p); +	return status; +} + +/** + * ice_create_vsig_from_lst - create a new VSIG with a list of profiles + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsi: the initial VSI that will be in VSIG + * @lst: the list of profile that will be added to the VSIG + * @chg: the change list + */ +static enum ice_status +ice_create_vsig_from_lst(struct ice_hw *hw, enum ice_block blk, u16 vsi, +			 struct list_head *lst, struct list_head *chg) +{ +	struct ice_vsig_prof *t; +	enum ice_status status; +	u16 vsig; + +	vsig = ice_vsig_alloc(hw, blk); +	if (!vsig) +		return ICE_ERR_HW_TABLE; + +	status = ice_move_vsi(hw, blk, vsi, vsig, chg); +	if (status) +		return status; + +	list_for_each_entry(t, lst, list) { +		status = ice_add_prof_id_vsig(hw, blk, vsig, t->profile_cookie, +					      chg); +		if (status) +			return status; +	} + +	return 0; +} + +/** + * ice_find_prof_vsig - find a VSIG with a specific profile handle + * @hw: pointer to the HW struct + * @blk: hardware block + * @hdl: the profile handle of the profile to search for + * @vsig: returns the VSIG with the matching profile + */ +static bool +ice_find_prof_vsig(struct ice_hw *hw, enum ice_block blk, u64 hdl, u16 *vsig) +{ +	struct ice_vsig_prof *t; +	enum ice_status status; +	struct list_head lst; + +	INIT_LIST_HEAD(&lst); + +	t = kzalloc(sizeof(*t), GFP_KERNEL); +	if (!t) +		return false; + +	t->profile_cookie = hdl; +	list_add(&t->list, &lst); + +	status = ice_find_dup_props_vsig(hw, blk, &lst, vsig); + +	list_del(&t->list); +	kfree(t); + +	return !status; +} + +/** + * ice_add_prof_id_flow - add profile flow + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsi: the VSI to enable with the profile specified by ID + * @hdl: profile handle + * + * Calling this function will update the hardware tables to enable the + * profile indicated by the ID parameter for the VSIs specified in the VSI + * array. Once successfully called, the flow will be enabled. + */ +enum ice_status +ice_add_prof_id_flow(struct ice_hw *hw, enum ice_block blk, u16 vsi, u64 hdl) +{ +	struct ice_vsig_prof *tmp1, *del1; +	struct ice_chs_chg *tmp, *del; +	struct list_head union_lst; +	enum ice_status status; +	struct list_head chg; +	u16 vsig; + +	INIT_LIST_HEAD(&union_lst); +	INIT_LIST_HEAD(&chg); + +	/* Get profile */ +	status = ice_get_prof(hw, blk, hdl, &chg); +	if (status) +		return status; + +	/* determine if VSI is already part of a VSIG */ +	status = ice_vsig_find_vsi(hw, blk, vsi, &vsig); +	if (!status && vsig) { +		bool only_vsi; +		u16 or_vsig; +		u16 ref; + +		/* found in VSIG */ +		or_vsig = vsig; + +		/* make sure that there is no overlap/conflict between the new +		 * characteristics and the existing ones; we don't support that +		 * scenario +		 */ +		if (ice_has_prof_vsig(hw, blk, vsig, hdl)) { +			status = ICE_ERR_ALREADY_EXISTS; +			goto err_ice_add_prof_id_flow; +		} + +		/* last VSI in the VSIG? */ +		status = ice_vsig_get_ref(hw, blk, vsig, &ref); +		if (status) +			goto err_ice_add_prof_id_flow; +		only_vsi = (ref == 1); + +		/* create a union of the current profiles and the one being +		 * added +		 */ +		status = ice_get_profs_vsig(hw, blk, vsig, &union_lst); +		if (status) +			goto err_ice_add_prof_id_flow; + +		status = ice_add_prof_to_lst(hw, blk, &union_lst, hdl); +		if (status) +			goto err_ice_add_prof_id_flow; + +		/* search for an existing VSIG with an exact charc match */ +		status = ice_find_dup_props_vsig(hw, blk, &union_lst, &vsig); +		if (!status) { +			/* move VSI to the VSIG that matches */ +			status = ice_move_vsi(hw, blk, vsi, vsig, &chg); +			if (status) +				goto err_ice_add_prof_id_flow; + +			/* VSI has been moved out of or_vsig. If the or_vsig had +			 * only that VSI it is now empty and can be removed. +			 */ +			if (only_vsi) { +				status = ice_rem_vsig(hw, blk, or_vsig, &chg); +				if (status) +					goto err_ice_add_prof_id_flow; +			} +		} else if (only_vsi) { +			/* If the original VSIG only contains one VSI, then it +			 * will be the requesting VSI. In this case the VSI is +			 * not sharing entries and we can simply add the new +			 * profile to the VSIG. +			 */ +			status = ice_add_prof_id_vsig(hw, blk, vsig, hdl, &chg); +			if (status) +				goto err_ice_add_prof_id_flow; + +			/* Adjust priorities */ +			status = ice_adj_prof_priorities(hw, blk, vsig, &chg); +			if (status) +				goto err_ice_add_prof_id_flow; +		} else { +			/* No match, so we need a new VSIG */ +			status = ice_create_vsig_from_lst(hw, blk, vsi, +							  &union_lst, &chg); +			if (status) +				goto err_ice_add_prof_id_flow; + +			/* Adjust priorities */ +			status = ice_adj_prof_priorities(hw, blk, vsig, &chg); +			if (status) +				goto err_ice_add_prof_id_flow; +		} +	} else { +		/* need to find or add a VSIG */ +		/* search for an existing VSIG with an exact charc match */ +		if (ice_find_prof_vsig(hw, blk, hdl, &vsig)) { +			/* found an exact match */ +			/* add or move VSI to the VSIG that matches */ +			status = ice_move_vsi(hw, blk, vsi, vsig, &chg); +			if (status) +				goto err_ice_add_prof_id_flow; +		} else { +			/* we did not find an exact match */ +			/* we need to add a VSIG */ +			status = ice_create_prof_id_vsig(hw, blk, vsi, hdl, +							 &chg); +			if (status) +				goto err_ice_add_prof_id_flow; +		} +	} + +	/* update hardware */ +	if (!status) +		status = ice_upd_prof_hw(hw, blk, &chg); + +err_ice_add_prof_id_flow: +	list_for_each_entry_safe(del, tmp, &chg, list_entry) { +		list_del(&del->list_entry); +		devm_kfree(ice_hw_to_dev(hw), del); +	} + +	list_for_each_entry_safe(del1, tmp1, &union_lst, list) { +		list_del(&del1->list); +		devm_kfree(ice_hw_to_dev(hw), del1); +	} + +	return status; +} + +/** + * ice_rem_prof_from_list - remove a profile from list + * @hw: pointer to the HW struct + * @lst: list to remove the profile from + * @hdl: the profile handle indicating the profile to remove + */ +static enum ice_status +ice_rem_prof_from_list(struct ice_hw *hw, struct list_head *lst, u64 hdl) +{ +	struct ice_vsig_prof *ent, *tmp; + +	list_for_each_entry_safe(ent, tmp, lst, list) +		if (ent->profile_cookie == hdl) { +			list_del(&ent->list); +			devm_kfree(ice_hw_to_dev(hw), ent); +			return 0; +		} + +	return ICE_ERR_DOES_NOT_EXIST; +} + +/** + * ice_rem_prof_id_flow - remove flow + * @hw: pointer to the HW struct + * @blk: hardware block + * @vsi: the VSI from which to remove the profile specified by ID + * @hdl: profile tracking handle + * + * Calling this function will update the hardware tables to remove the + * profile indicated by the ID parameter for the VSIs specified in the VSI + * array. Once successfully called, the flow will be disabled. + */ +enum ice_status +ice_rem_prof_id_flow(struct ice_hw *hw, enum ice_block blk, u16 vsi, u64 hdl) +{ +	struct ice_vsig_prof *tmp1, *del1; +	struct ice_chs_chg *tmp, *del; +	struct list_head chg, copy; +	enum ice_status status; +	u16 vsig; + +	INIT_LIST_HEAD(©); +	INIT_LIST_HEAD(&chg); + +	/* determine if VSI is already part of a VSIG */ +	status = ice_vsig_find_vsi(hw, blk, vsi, &vsig); +	if (!status && vsig) { +		bool last_profile; +		bool only_vsi; +		u16 ref; + +		/* found in VSIG */ +		last_profile = ice_vsig_prof_id_count(hw, blk, vsig) == 1; +		status = ice_vsig_get_ref(hw, blk, vsig, &ref); +		if (status) +			goto err_ice_rem_prof_id_flow; +		only_vsi = (ref == 1); + +		if (only_vsi) { +			/* If the original VSIG only contains one reference, +			 * which will be the requesting VSI, then the VSI is not +			 * sharing entries and we can simply remove the specific +			 * characteristics from the VSIG. +			 */ + +			if (last_profile) { +				/* If there are no profiles left for this VSIG, +				 * then simply remove the the VSIG. +				 */ +				status = ice_rem_vsig(hw, blk, vsig, &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; +			} else { +				status = ice_rem_prof_id_vsig(hw, blk, vsig, +							      hdl, &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; + +				/* Adjust priorities */ +				status = ice_adj_prof_priorities(hw, blk, vsig, +								 &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; +			} + +		} else { +			/* Make a copy of the VSIG's list of Profiles */ +			status = ice_get_profs_vsig(hw, blk, vsig, ©); +			if (status) +				goto err_ice_rem_prof_id_flow; + +			/* Remove specified profile entry from the list */ +			status = ice_rem_prof_from_list(hw, ©, hdl); +			if (status) +				goto err_ice_rem_prof_id_flow; + +			if (list_empty(©)) { +				status = ice_move_vsi(hw, blk, vsi, +						      ICE_DEFAULT_VSIG, &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; + +			} else if (!ice_find_dup_props_vsig(hw, blk, ©, +							    &vsig)) { +				/* found an exact match */ +				/* add or move VSI to the VSIG that matches */ +				/* Search for a VSIG with a matching profile +				 * list +				 */ + +				/* Found match, move VSI to the matching VSIG */ +				status = ice_move_vsi(hw, blk, vsi, vsig, &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; +			} else { +				/* since no existing VSIG supports this +				 * characteristic pattern, we need to create a +				 * new VSIG and TCAM entries +				 */ +				status = ice_create_vsig_from_lst(hw, blk, vsi, +								  ©, &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; + +				/* Adjust priorities */ +				status = ice_adj_prof_priorities(hw, blk, vsig, +								 &chg); +				if (status) +					goto err_ice_rem_prof_id_flow; +			} +		} +	} else { +		status = ICE_ERR_DOES_NOT_EXIST; +	} + +	/* update hardware tables */ +	if (!status) +		status = ice_upd_prof_hw(hw, blk, &chg); + +err_ice_rem_prof_id_flow: +	list_for_each_entry_safe(del, tmp, &chg, list_entry) { +		list_del(&del->list_entry); +		devm_kfree(ice_hw_to_dev(hw), del); +	} + +	list_for_each_entry_safe(del1, tmp1, ©, list) { +		list_del(&del1->list); +		devm_kfree(ice_hw_to_dev(hw), del1); +	} + +	return status; +} diff --git a/drivers/net/ethernet/intel/ice/ice_flex_pipe.h b/drivers/net/ethernet/intel/ice/ice_flex_pipe.h index 37eb282742d1..c7b5e1a6ea2b 100644 --- a/drivers/net/ethernet/intel/ice/ice_flex_pipe.h +++ b/drivers/net/ethernet/intel/ice/ice_flex_pipe.h @@ -18,6 +18,13 @@  #define ICE_PKG_CNT 4 +enum ice_status +ice_add_prof(struct ice_hw *hw, enum ice_block blk, u64 id, u8 ptypes[], +	     struct ice_fv_word *es); +enum ice_status +ice_add_prof_id_flow(struct ice_hw *hw, enum ice_block blk, u16 vsi, u64 hdl); +enum ice_status +ice_rem_prof_id_flow(struct ice_hw *hw, enum ice_block blk, u16 vsi, u64 hdl);  enum ice_status ice_init_pkg(struct ice_hw *hw, u8 *buff, u32 len);  enum ice_status  ice_copy_and_init_pkg(struct ice_hw *hw, const u8 *buf, u32 len); @@ -26,4 +33,6 @@ void ice_free_seg(struct ice_hw *hw);  void ice_fill_blk_tbls(struct ice_hw *hw);  void ice_clear_hw_tbls(struct ice_hw *hw);  void ice_free_hw_tbls(struct ice_hw *hw); +enum ice_status +ice_rem_prof(struct ice_hw *hw, enum ice_block blk, u64 id);  #endif /* _ICE_FLEX_PIPE_H_ */ diff --git a/drivers/net/ethernet/intel/ice/ice_flex_type.h b/drivers/net/ethernet/intel/ice/ice_flex_type.h index 5d5a7eaffa30..0fb3fe3ff3ea 100644 --- a/drivers/net/ethernet/intel/ice/ice_flex_type.h +++ b/drivers/net/ethernet/intel/ice/ice_flex_type.h @@ -3,6 +3,9 @@  #ifndef _ICE_FLEX_TYPE_H_  #define _ICE_FLEX_TYPE_H_ + +#define ICE_FV_OFFSET_INVAL	0x1FF +  /* Extraction Sequence (Field Vector) Table */  struct ice_fv_word {  	u8 prot_id; @@ -105,37 +108,57 @@ struct ice_buf_hdr {  	sizeof(struct ice_buf_hdr) - (hd_sz)) / (ent_sz))  /* ice package section IDs */ +#define ICE_SID_XLT0_SW			10 +#define ICE_SID_XLT_KEY_BUILDER_SW	11  #define ICE_SID_XLT1_SW			12  #define ICE_SID_XLT2_SW			13  #define ICE_SID_PROFID_TCAM_SW		14  #define ICE_SID_PROFID_REDIR_SW		15  #define ICE_SID_FLD_VEC_SW		16 +#define ICE_SID_CDID_KEY_BUILDER_SW	17 +#define ICE_SID_CDID_REDIR_SW		18 +#define ICE_SID_XLT0_ACL		20 +#define ICE_SID_XLT_KEY_BUILDER_ACL	21  #define ICE_SID_XLT1_ACL		22  #define ICE_SID_XLT2_ACL		23  #define ICE_SID_PROFID_TCAM_ACL		24  #define ICE_SID_PROFID_REDIR_ACL	25  #define ICE_SID_FLD_VEC_ACL		26 +#define ICE_SID_CDID_KEY_BUILDER_ACL	27 +#define ICE_SID_CDID_REDIR_ACL		28 +#define ICE_SID_XLT0_FD			30 +#define ICE_SID_XLT_KEY_BUILDER_FD	31  #define ICE_SID_XLT1_FD			32  #define ICE_SID_XLT2_FD			33  #define ICE_SID_PROFID_TCAM_FD		34  #define ICE_SID_PROFID_REDIR_FD		35  #define ICE_SID_FLD_VEC_FD		36 +#define ICE_SID_CDID_KEY_BUILDER_FD	37 +#define ICE_SID_CDID_REDIR_FD		38 +#define ICE_SID_XLT0_RSS		40 +#define ICE_SID_XLT_KEY_BUILDER_RSS	41  #define ICE_SID_XLT1_RSS		42  #define ICE_SID_XLT2_RSS		43  #define ICE_SID_PROFID_TCAM_RSS		44  #define ICE_SID_PROFID_REDIR_RSS	45  #define ICE_SID_FLD_VEC_RSS		46 +#define ICE_SID_CDID_KEY_BUILDER_RSS	47 +#define ICE_SID_CDID_REDIR_RSS		48  #define ICE_SID_RXPARSER_BOOST_TCAM	56 +#define ICE_SID_XLT0_PE			80 +#define ICE_SID_XLT_KEY_BUILDER_PE	81  #define ICE_SID_XLT1_PE			82  #define ICE_SID_XLT2_PE			83  #define ICE_SID_PROFID_TCAM_PE		84  #define ICE_SID_PROFID_REDIR_PE		85  #define ICE_SID_FLD_VEC_PE		86 +#define ICE_SID_CDID_KEY_BUILDER_PE	87 +#define ICE_SID_CDID_REDIR_PE		88  /* Label Metadata section IDs */  #define ICE_SID_LBL_FIRST		0x80000010 @@ -152,6 +175,19 @@ enum ice_block {  	ICE_BLK_COUNT  }; +enum ice_sect { +	ICE_XLT0 = 0, +	ICE_XLT_KB, +	ICE_XLT1, +	ICE_XLT2, +	ICE_PROF_TCAM, +	ICE_PROF_REDIR, +	ICE_VEC_TBL, +	ICE_CDID_KB, +	ICE_CDID_REDIR, +	ICE_SECT_COUNT +}; +  /* package labels */  struct ice_label {  	__le16 value; @@ -234,6 +270,13 @@ struct ice_prof_redir_section {  	u8 redir_value[1];  }; +/* package buffer building */ + +struct ice_buf_build { +	struct ice_buf buf; +	u16 reserved_section_table_entries; +}; +  struct ice_pkg_enum {  	struct ice_buf_table *buf_table;  	u32 buf_idx; @@ -248,6 +291,12 @@ struct ice_pkg_enum {  	void *(*handler)(u32 sect_type, void *section, u32 index, u32 *offset);  }; +struct ice_pkg_es { +	__le16 count; +	__le16 offset; +	struct ice_fv_word es[1]; +}; +  struct ice_es {  	u32 sid;  	u16 count; @@ -280,6 +329,35 @@ struct ice_ptg_ptype {  	u8 ptg;  }; +#define ICE_MAX_TCAM_PER_PROFILE	32 +#define ICE_MAX_PTG_PER_PROFILE		32 + +struct ice_prof_map { +	struct list_head list; +	u64 profile_cookie; +	u64 context; +	u8 prof_id; +	u8 ptg_cnt; +	u8 ptg[ICE_MAX_PTG_PER_PROFILE]; +}; + +#define ICE_INVALID_TCAM	0xFFFF + +struct ice_tcam_inf { +	u16 tcam_idx; +	u8 ptg; +	u8 prof_id; +	u8 in_use; +}; + +struct ice_vsig_prof { +	struct list_head list; +	u64 profile_cookie; +	u8 prof_id; +	u8 tcam_count; +	struct ice_tcam_inf tcam[ICE_MAX_TCAM_PER_PROFILE]; +}; +  struct ice_vsig_entry {  	struct list_head prop_lst;  	struct ice_vsig_vsi *first_vsi; @@ -329,6 +407,13 @@ struct ice_xlt2 {  	u16 count;  }; +/* Profile ID Management */ +struct ice_prof_id_key { +	__le16 flags; +	u8 xlt1; +	__le16 xlt2_cdid; +} __packed; +  /* Keys are made up of two values, each one-half the size of the key.   * For TCAM, the entire key is 80 bits wide (or 2, 40-bit wide values)   */ @@ -371,4 +456,31 @@ struct ice_blk_info {  	u8 is_list_init;  }; +enum ice_chg_type { +	ICE_TCAM_NONE = 0, +	ICE_PTG_ES_ADD, +	ICE_TCAM_ADD, +	ICE_VSIG_ADD, +	ICE_VSIG_REM, +	ICE_VSI_MOVE, +}; + +struct ice_chs_chg { +	struct list_head list_entry; +	enum ice_chg_type type; + +	u8 add_ptg; +	u8 add_vsig; +	u8 add_tcam_idx; +	u8 add_prof; +	u16 ptype; +	u8 ptg; +	u8 prof_id; +	u16 vsi; +	u16 vsig; +	u16 orig_vsig; +	u16 tcam_idx; +}; + +#define ICE_FLOW_PTYPE_MAX		ICE_XLT1_CNT  #endif /* _ICE_FLEX_TYPE_H_ */ diff --git a/drivers/net/ethernet/intel/ice/ice_flow.c b/drivers/net/ethernet/intel/ice/ice_flow.c new file mode 100644 index 000000000000..a05ceb59863b --- /dev/null +++ b/drivers/net/ethernet/intel/ice/ice_flow.c @@ -0,0 +1,1275 @@ +// SPDX-License-Identifier: GPL-2.0 +/* Copyright (c) 2019, Intel Corporation. */ + +#include "ice_common.h" +#include "ice_flow.h" + +/* Describe properties of a protocol header field */ +struct ice_flow_field_info { +	enum ice_flow_seg_hdr hdr; +	s16 off;	/* Offset from start of a protocol header, in bits */ +	u16 size;	/* Size of fields in bits */ +}; + +#define ICE_FLOW_FLD_INFO(_hdr, _offset_bytes, _size_bytes) { \ +	.hdr = _hdr, \ +	.off = (_offset_bytes) * BITS_PER_BYTE, \ +	.size = (_size_bytes) * BITS_PER_BYTE, \ +} + +/* Table containing properties of supported protocol header fields */ +static const +struct ice_flow_field_info ice_flds_info[ICE_FLOW_FIELD_IDX_MAX] = { +	/* IPv4 / IPv6 */ +	/* ICE_FLOW_FIELD_IDX_IPV4_SA */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_IPV4, 12, sizeof(struct in_addr)), +	/* ICE_FLOW_FIELD_IDX_IPV4_DA */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_IPV4, 16, sizeof(struct in_addr)), +	/* ICE_FLOW_FIELD_IDX_IPV6_SA */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_IPV6, 8, sizeof(struct in6_addr)), +	/* ICE_FLOW_FIELD_IDX_IPV6_DA */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_IPV6, 24, sizeof(struct in6_addr)), +	/* Transport */ +	/* ICE_FLOW_FIELD_IDX_TCP_SRC_PORT */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_TCP, 0, sizeof(__be16)), +	/* ICE_FLOW_FIELD_IDX_TCP_DST_PORT */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_TCP, 2, sizeof(__be16)), +	/* ICE_FLOW_FIELD_IDX_UDP_SRC_PORT */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_UDP, 0, sizeof(__be16)), +	/* ICE_FLOW_FIELD_IDX_UDP_DST_PORT */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_UDP, 2, sizeof(__be16)), +	/* ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_SCTP, 0, sizeof(__be16)), +	/* ICE_FLOW_FIELD_IDX_SCTP_DST_PORT */ +	ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_SCTP, 2, sizeof(__be16)), + +}; + +/* Bitmaps indicating relevant packet types for a particular protocol header + * + * Packet types for packets with an Outer/First/Single IPv4 header + */ +static const u32 ice_ptypes_ipv4_ofos[] = { +	0x1DC00000, 0x04000800, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* Packet types for packets with an Innermost/Last IPv4 header */ +static const u32 ice_ptypes_ipv4_il[] = { +	0xE0000000, 0xB807700E, 0x80000003, 0xE01DC03B, +	0x0000000E, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* Packet types for packets with an Outer/First/Single IPv6 header */ +static const u32 ice_ptypes_ipv6_ofos[] = { +	0x00000000, 0x00000000, 0x77000000, 0x10002000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* Packet types for packets with an Innermost/Last IPv6 header */ +static const u32 ice_ptypes_ipv6_il[] = { +	0x00000000, 0x03B80770, 0x000001DC, 0x0EE00000, +	0x00000770, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* UDP Packet types for non-tunneled packets or tunneled + * packets with inner UDP. + */ +static const u32 ice_ptypes_udp_il[] = { +	0x81000000, 0x20204040, 0x04000010, 0x80810102, +	0x00000040, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* Packet types for packets with an Innermost/Last TCP header */ +static const u32 ice_ptypes_tcp_il[] = { +	0x04000000, 0x80810102, 0x10000040, 0x02040408, +	0x00000102, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* Packet types for packets with an Innermost/Last SCTP header */ +static const u32 ice_ptypes_sctp_il[] = { +	0x08000000, 0x01020204, 0x20000081, 0x04080810, +	0x00000204, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +	0x00000000, 0x00000000, 0x00000000, 0x00000000, +}; + +/* Manage parameters and info. used during the creation of a flow profile */ +struct ice_flow_prof_params { +	enum ice_block blk; +	u16 entry_length; /* # of bytes formatted entry will require */ +	u8 es_cnt; +	struct ice_flow_prof *prof; + +	/* For ACL, the es[0] will have the data of ICE_RX_MDID_PKT_FLAGS_15_0 +	 * This will give us the direction flags. +	 */ +	struct ice_fv_word es[ICE_MAX_FV_WORDS]; +	DECLARE_BITMAP(ptypes, ICE_FLOW_PTYPE_MAX); +}; + +#define ICE_FLOW_SEG_HDRS_L3_MASK	\ +	(ICE_FLOW_SEG_HDR_IPV4 | ICE_FLOW_SEG_HDR_IPV6) +#define ICE_FLOW_SEG_HDRS_L4_MASK	\ +	(ICE_FLOW_SEG_HDR_TCP | ICE_FLOW_SEG_HDR_UDP | ICE_FLOW_SEG_HDR_SCTP) + +/** + * ice_flow_val_hdrs - validates packet segments for valid protocol headers + * @segs: array of one or more packet segments that describe the flow + * @segs_cnt: number of packet segments provided + */ +static enum ice_status +ice_flow_val_hdrs(struct ice_flow_seg_info *segs, u8 segs_cnt) +{ +	u8 i; + +	for (i = 0; i < segs_cnt; i++) { +		/* Multiple L3 headers */ +		if (segs[i].hdrs & ICE_FLOW_SEG_HDRS_L3_MASK && +		    !is_power_of_2(segs[i].hdrs & ICE_FLOW_SEG_HDRS_L3_MASK)) +			return ICE_ERR_PARAM; + +		/* Multiple L4 headers */ +		if (segs[i].hdrs & ICE_FLOW_SEG_HDRS_L4_MASK && +		    !is_power_of_2(segs[i].hdrs & ICE_FLOW_SEG_HDRS_L4_MASK)) +			return ICE_ERR_PARAM; +	} + +	return 0; +} + +/** + * ice_flow_proc_seg_hdrs - process protocol headers present in pkt segments + * @params: information about the flow to be processed + * + * This function identifies the packet types associated with the protocol + * headers being present in packet segments of the specified flow profile. + */ +static enum ice_status +ice_flow_proc_seg_hdrs(struct ice_flow_prof_params *params) +{ +	struct ice_flow_prof *prof; +	u8 i; + +	memset(params->ptypes, 0xff, sizeof(params->ptypes)); + +	prof = params->prof; + +	for (i = 0; i < params->prof->segs_cnt; i++) { +		const unsigned long *src; +		u32 hdrs; + +		hdrs = prof->segs[i].hdrs; + +		if (hdrs & ICE_FLOW_SEG_HDR_IPV4) { +			src = !i ? (const unsigned long *)ice_ptypes_ipv4_ofos : +				(const unsigned long *)ice_ptypes_ipv4_il; +			bitmap_and(params->ptypes, params->ptypes, src, +				   ICE_FLOW_PTYPE_MAX); +		} else if (hdrs & ICE_FLOW_SEG_HDR_IPV6) { +			src = !i ? (const unsigned long *)ice_ptypes_ipv6_ofos : +				(const unsigned long *)ice_ptypes_ipv6_il; +			bitmap_and(params->ptypes, params->ptypes, src, +				   ICE_FLOW_PTYPE_MAX); +		} + +		if (hdrs & ICE_FLOW_SEG_HDR_UDP) { +			src = (const unsigned long *)ice_ptypes_udp_il; +			bitmap_and(params->ptypes, params->ptypes, src, +				   ICE_FLOW_PTYPE_MAX); +		} else if (hdrs & ICE_FLOW_SEG_HDR_TCP) { +			bitmap_and(params->ptypes, params->ptypes, +				   (const unsigned long *)ice_ptypes_tcp_il, +				   ICE_FLOW_PTYPE_MAX); +		} else if (hdrs & ICE_FLOW_SEG_HDR_SCTP) { +			src = (const unsigned long *)ice_ptypes_sctp_il; +			bitmap_and(params->ptypes, params->ptypes, src, +				   ICE_FLOW_PTYPE_MAX); +		} +	} + +	return 0; +} + +/** + * ice_flow_xtract_fld - Create an extraction sequence entry for the given field + * @hw: pointer to the HW struct + * @params: information about the flow to be processed + * @seg: packet segment index of the field to be extracted + * @fld: ID of field to be extracted + * + * This function determines the protocol ID, offset, and size of the given + * field. It then allocates one or more extraction sequence entries for the + * given field, and fill the entries with protocol ID and offset information. + */ +static enum ice_status +ice_flow_xtract_fld(struct ice_hw *hw, struct ice_flow_prof_params *params, +		    u8 seg, enum ice_flow_field fld) +{ +	enum ice_prot_id prot_id = ICE_PROT_ID_INVAL; +	u8 fv_words = hw->blk[params->blk].es.fvw; +	struct ice_flow_fld_info *flds; +	u16 cnt, ese_bits, i; +	u16 off; + +	flds = params->prof->segs[seg].fields; + +	switch (fld) { +	case ICE_FLOW_FIELD_IDX_IPV4_SA: +	case ICE_FLOW_FIELD_IDX_IPV4_DA: +		prot_id = seg == 0 ? ICE_PROT_IPV4_OF_OR_S : ICE_PROT_IPV4_IL; +		break; +	case ICE_FLOW_FIELD_IDX_IPV6_SA: +	case ICE_FLOW_FIELD_IDX_IPV6_DA: +		prot_id = seg == 0 ? ICE_PROT_IPV6_OF_OR_S : ICE_PROT_IPV6_IL; +		break; +	case ICE_FLOW_FIELD_IDX_TCP_SRC_PORT: +	case ICE_FLOW_FIELD_IDX_TCP_DST_PORT: +		prot_id = ICE_PROT_TCP_IL; +		break; +	case ICE_FLOW_FIELD_IDX_UDP_SRC_PORT: +	case ICE_FLOW_FIELD_IDX_UDP_DST_PORT: +		prot_id = ICE_PROT_UDP_IL_OR_S; +		break; +	case ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT: +	case ICE_FLOW_FIELD_IDX_SCTP_DST_PORT: +		prot_id = ICE_PROT_SCTP_IL; +		break; +	default: +		return ICE_ERR_NOT_IMPL; +	} + +	/* Each extraction sequence entry is a word in size, and extracts a +	 * word-aligned offset from a protocol header. +	 */ +	ese_bits = ICE_FLOW_FV_EXTRACT_SZ * BITS_PER_BYTE; + +	flds[fld].xtrct.prot_id = prot_id; +	flds[fld].xtrct.off = (ice_flds_info[fld].off / ese_bits) * +		ICE_FLOW_FV_EXTRACT_SZ; +	flds[fld].xtrct.disp = (u8)(ice_flds_info[fld].off % ese_bits); +	flds[fld].xtrct.idx = params->es_cnt; + +	/* Adjust the next field-entry index after accommodating the number of +	 * entries this field consumes +	 */ +	cnt = DIV_ROUND_UP(flds[fld].xtrct.disp + ice_flds_info[fld].size, +			   ese_bits); + +	/* Fill in the extraction sequence entries needed for this field */ +	off = flds[fld].xtrct.off; +	for (i = 0; i < cnt; i++) { +		u8 idx; + +		/* Make sure the number of extraction sequence required +		 * does not exceed the block's capability +		 */ +		if (params->es_cnt >= fv_words) +			return ICE_ERR_MAX_LIMIT; + +		/* some blocks require a reversed field vector layout */ +		if (hw->blk[params->blk].es.reverse) +			idx = fv_words - params->es_cnt - 1; +		else +			idx = params->es_cnt; + +		params->es[idx].prot_id = prot_id; +		params->es[idx].off = off; +		params->es_cnt++; + +		off += ICE_FLOW_FV_EXTRACT_SZ; +	} + +	return 0; +} + +/** + * ice_flow_create_xtrct_seq - Create an extraction sequence for given segments + * @hw: pointer to the HW struct + * @params: information about the flow to be processed + * + * This function iterates through all matched fields in the given segments, and + * creates an extraction sequence for the fields. + */ +static enum ice_status +ice_flow_create_xtrct_seq(struct ice_hw *hw, +			  struct ice_flow_prof_params *params) +{ +	struct ice_flow_prof *prof = params->prof; +	enum ice_status status = 0; +	u8 i; + +	for (i = 0; i < prof->segs_cnt; i++) { +		u8 j; + +		for_each_set_bit(j, (unsigned long *)&prof->segs[i].match, +				 ICE_FLOW_FIELD_IDX_MAX) { +			status = ice_flow_xtract_fld(hw, params, i, +						     (enum ice_flow_field)j); +			if (status) +				return status; +		} +	} + +	return status; +} + +/** + * ice_flow_proc_segs - process all packet segments associated with a profile + * @hw: pointer to the HW struct + * @params: information about the flow to be processed + */ +static enum ice_status +ice_flow_proc_segs(struct ice_hw *hw, struct ice_flow_prof_params *params) +{ +	enum ice_status status; + +	status = ice_flow_proc_seg_hdrs(params); +	if (status) +		return status; + +	status = ice_flow_create_xtrct_seq(hw, params); +	if (status) +		return status; + +	switch (params->blk) { +	case ICE_BLK_RSS: +		/* Only header information is provided for RSS configuration. +		 * No further processing is needed. +		 */ +		status = 0; +		break; +	default: +		return ICE_ERR_NOT_IMPL; +	} + +	return status; +} + +#define ICE_FLOW_FIND_PROF_CHK_FLDS	0x00000001 +#define ICE_FLOW_FIND_PROF_CHK_VSI	0x00000002 +#define ICE_FLOW_FIND_PROF_NOT_CHK_DIR	0x00000004 + +/** + * ice_flow_find_prof_conds - Find a profile matching headers and conditions + * @hw: pointer to the HW struct + * @blk: classification stage + * @dir: flow direction + * @segs: array of one or more packet segments that describe the flow + * @segs_cnt: number of packet segments provided + * @vsi_handle: software VSI handle to check VSI (ICE_FLOW_FIND_PROF_CHK_VSI) + * @conds: additional conditions to be checked (ICE_FLOW_FIND_PROF_CHK_*) + */ +static struct ice_flow_prof * +ice_flow_find_prof_conds(struct ice_hw *hw, enum ice_block blk, +			 enum ice_flow_dir dir, struct ice_flow_seg_info *segs, +			 u8 segs_cnt, u16 vsi_handle, u32 conds) +{ +	struct ice_flow_prof *p, *prof = NULL; + +	mutex_lock(&hw->fl_profs_locks[blk]); +	list_for_each_entry(p, &hw->fl_profs[blk], l_entry) +		if ((p->dir == dir || conds & ICE_FLOW_FIND_PROF_NOT_CHK_DIR) && +		    segs_cnt && segs_cnt == p->segs_cnt) { +			u8 i; + +			/* Check for profile-VSI association if specified */ +			if ((conds & ICE_FLOW_FIND_PROF_CHK_VSI) && +			    ice_is_vsi_valid(hw, vsi_handle) && +			    !test_bit(vsi_handle, p->vsis)) +				continue; + +			/* Protocol headers must be checked. Matched fields are +			 * checked if specified. +			 */ +			for (i = 0; i < segs_cnt; i++) +				if (segs[i].hdrs != p->segs[i].hdrs || +				    ((conds & ICE_FLOW_FIND_PROF_CHK_FLDS) && +				     segs[i].match != p->segs[i].match)) +					break; + +			/* A match is found if all segments are matched */ +			if (i == segs_cnt) { +				prof = p; +				break; +			} +		} +	mutex_unlock(&hw->fl_profs_locks[blk]); + +	return prof; +} + +/** + * ice_flow_find_prof_id - Look up a profile with given profile ID + * @hw: pointer to the HW struct + * @blk: classification stage + * @prof_id: unique ID to identify this flow profile + */ +static struct ice_flow_prof * +ice_flow_find_prof_id(struct ice_hw *hw, enum ice_block blk, u64 prof_id) +{ +	struct ice_flow_prof *p; + +	list_for_each_entry(p, &hw->fl_profs[blk], l_entry) +		if (p->id == prof_id) +			return p; + +	return NULL; +} + +/** + * ice_flow_add_prof_sync - Add a flow profile for packet segments and fields + * @hw: pointer to the HW struct + * @blk: classification stage + * @dir: flow direction + * @prof_id: unique ID to identify this flow profile + * @segs: array of one or more packet segments that describe the flow + * @segs_cnt: number of packet segments provided + * @prof: stores the returned flow profile added + * + * Assumption: the caller has acquired the lock to the profile list + */ +static enum ice_status +ice_flow_add_prof_sync(struct ice_hw *hw, enum ice_block blk, +		       enum ice_flow_dir dir, u64 prof_id, +		       struct ice_flow_seg_info *segs, u8 segs_cnt, +		       struct ice_flow_prof **prof) +{ +	struct ice_flow_prof_params params; +	enum ice_status status; +	u8 i; + +	if (!prof) +		return ICE_ERR_BAD_PTR; + +	memset(¶ms, 0, sizeof(params)); +	params.prof = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*params.prof), +				   GFP_KERNEL); +	if (!params.prof) +		return ICE_ERR_NO_MEMORY; + +	/* initialize extraction sequence to all invalid (0xff) */ +	for (i = 0; i < ICE_MAX_FV_WORDS; i++) { +		params.es[i].prot_id = ICE_PROT_INVALID; +		params.es[i].off = ICE_FV_OFFSET_INVAL; +	} + +	params.blk = blk; +	params.prof->id = prof_id; +	params.prof->dir = dir; +	params.prof->segs_cnt = segs_cnt; + +	/* Make a copy of the segments that need to be persistent in the flow +	 * profile instance +	 */ +	for (i = 0; i < segs_cnt; i++) +		memcpy(¶ms.prof->segs[i], &segs[i], sizeof(*segs)); + +	status = ice_flow_proc_segs(hw, ¶ms); +	if (status) { +		ice_debug(hw, ICE_DBG_FLOW, +			  "Error processing a flow's packet segments\n"); +		goto out; +	} + +	/* Add a HW profile for this flow profile */ +	status = ice_add_prof(hw, blk, prof_id, (u8 *)params.ptypes, params.es); +	if (status) { +		ice_debug(hw, ICE_DBG_FLOW, "Error adding a HW flow profile\n"); +		goto out; +	} + +	INIT_LIST_HEAD(¶ms.prof->entries); +	mutex_init(¶ms.prof->entries_lock); +	*prof = params.prof; + +out: +	if (status) +		devm_kfree(ice_hw_to_dev(hw), params.prof); + +	return status; +} + +/** + * ice_flow_rem_prof_sync - remove a flow profile + * @hw: pointer to the hardware structure + * @blk: classification stage + * @prof: pointer to flow profile to remove + * + * Assumption: the caller has acquired the lock to the profile list + */ +static enum ice_status +ice_flow_rem_prof_sync(struct ice_hw *hw, enum ice_block blk, +		       struct ice_flow_prof *prof) +{ +	enum ice_status status; + +	/* Remove all hardware profiles associated with this flow profile */ +	status = ice_rem_prof(hw, blk, prof->id); +	if (!status) { +		list_del(&prof->l_entry); +		mutex_destroy(&prof->entries_lock); +		devm_kfree(ice_hw_to_dev(hw), prof); +	} + +	return status; +} + +/** + * ice_flow_assoc_prof - associate a VSI with a flow profile + * @hw: pointer to the hardware structure + * @blk: classification stage + * @prof: pointer to flow profile + * @vsi_handle: software VSI handle + * + * Assumption: the caller has acquired the lock to the profile list + * and the software VSI handle has been validated + */ +static enum ice_status +ice_flow_assoc_prof(struct ice_hw *hw, enum ice_block blk, +		    struct ice_flow_prof *prof, u16 vsi_handle) +{ +	enum ice_status status = 0; + +	if (!test_bit(vsi_handle, prof->vsis)) { +		status = ice_add_prof_id_flow(hw, blk, +					      ice_get_hw_vsi_num(hw, +								 vsi_handle), +					      prof->id); +		if (!status) +			set_bit(vsi_handle, prof->vsis); +		else +			ice_debug(hw, ICE_DBG_FLOW, +				  "HW profile add failed, %d\n", +				  status); +	} + +	return status; +} + +/** + * ice_flow_disassoc_prof - disassociate a VSI from a flow profile + * @hw: pointer to the hardware structure + * @blk: classification stage + * @prof: pointer to flow profile + * @vsi_handle: software VSI handle + * + * Assumption: the caller has acquired the lock to the profile list + * and the software VSI handle has been validated + */ +static enum ice_status +ice_flow_disassoc_prof(struct ice_hw *hw, enum ice_block blk, +		       struct ice_flow_prof *prof, u16 vsi_handle) +{ +	enum ice_status status = 0; + +	if (test_bit(vsi_handle, prof->vsis)) { +		status = ice_rem_prof_id_flow(hw, blk, +					      ice_get_hw_vsi_num(hw, +								 vsi_handle), +					      prof->id); +		if (!status) +			clear_bit(vsi_handle, prof->vsis); +		else +			ice_debug(hw, ICE_DBG_FLOW, +				  "HW profile remove failed, %d\n", +				  status); +	} + +	return status; +} + +/** + * ice_flow_add_prof - Add a flow profile for packet segments and matched fields + * @hw: pointer to the HW struct + * @blk: classification stage + * @dir: flow direction + * @prof_id: unique ID to identify this flow profile + * @segs: array of one or more packet segments that describe the flow + * @segs_cnt: number of packet segments provided + * @prof: stores the returned flow profile added + */ +static enum ice_status +ice_flow_add_prof(struct ice_hw *hw, enum ice_block blk, enum ice_flow_dir dir, +		  u64 prof_id, struct ice_flow_seg_info *segs, u8 segs_cnt, +		  struct ice_flow_prof **prof) +{ +	enum ice_status status; + +	if (segs_cnt > ICE_FLOW_SEG_MAX) +		return ICE_ERR_MAX_LIMIT; + +	if (!segs_cnt) +		return ICE_ERR_PARAM; + +	if (!segs) +		return ICE_ERR_BAD_PTR; + +	status = ice_flow_val_hdrs(segs, segs_cnt); +	if (status) +		return status; + +	mutex_lock(&hw->fl_profs_locks[blk]); + +	status = ice_flow_add_prof_sync(hw, blk, dir, prof_id, segs, segs_cnt, +					prof); +	if (!status) +		list_add(&(*prof)->l_entry, &hw->fl_profs[blk]); + +	mutex_unlock(&hw->fl_profs_locks[blk]); + +	return status; +} + +/** + * ice_flow_rem_prof - Remove a flow profile and all entries associated with it + * @hw: pointer to the HW struct + * @blk: the block for which the flow profile is to be removed + * @prof_id: unique ID of the flow profile to be removed + */ +static enum ice_status +ice_flow_rem_prof(struct ice_hw *hw, enum ice_block blk, u64 prof_id) +{ +	struct ice_flow_prof *prof; +	enum ice_status status; + +	mutex_lock(&hw->fl_profs_locks[blk]); + +	prof = ice_flow_find_prof_id(hw, blk, prof_id); +	if (!prof) { +		status = ICE_ERR_DOES_NOT_EXIST; +		goto out; +	} + +	/* prof becomes invalid after the call */ +	status = ice_flow_rem_prof_sync(hw, blk, prof); + +out: +	mutex_unlock(&hw->fl_profs_locks[blk]); + +	return status; +} + +/** + * ice_flow_set_fld_ext - specifies locations of field from entry's input buffer + * @seg: packet segment the field being set belongs to + * @fld: field to be set + * @type: type of the field + * @val_loc: if not ICE_FLOW_FLD_OFF_INVAL, location of the value to match from + *           entry's input buffer + * @mask_loc: if not ICE_FLOW_FLD_OFF_INVAL, location of mask value from entry's + *            input buffer + * @last_loc: if not ICE_FLOW_FLD_OFF_INVAL, location of last/upper value from + *            entry's input buffer + * + * This helper function stores information of a field being matched, including + * the type of the field and the locations of the value to match, the mask, and + * and the upper-bound value in the start of the input buffer for a flow entry. + * This function should only be used for fixed-size data structures. + * + * This function also opportunistically determines the protocol headers to be + * present based on the fields being set. Some fields cannot be used alone to + * determine the protocol headers present. Sometimes, fields for particular + * protocol headers are not matched. In those cases, the protocol headers + * must be explicitly set. + */ +static void +ice_flow_set_fld_ext(struct ice_flow_seg_info *seg, enum ice_flow_field fld, +		     enum ice_flow_fld_match_type type, u16 val_loc, +		     u16 mask_loc, u16 last_loc) +{ +	u64 bit = BIT_ULL(fld); + +	seg->match |= bit; +	if (type == ICE_FLOW_FLD_TYPE_RANGE) +		seg->range |= bit; + +	seg->fields[fld].type = type; +	seg->fields[fld].src.val = val_loc; +	seg->fields[fld].src.mask = mask_loc; +	seg->fields[fld].src.last = last_loc; + +	ICE_FLOW_SET_HDRS(seg, ice_flds_info[fld].hdr); +} + +/** + * ice_flow_set_fld - specifies locations of field from entry's input buffer + * @seg: packet segment the field being set belongs to + * @fld: field to be set + * @val_loc: if not ICE_FLOW_FLD_OFF_INVAL, location of the value to match from + *           entry's input buffer + * @mask_loc: if not ICE_FLOW_FLD_OFF_INVAL, location of mask value from entry's + *            input buffer + * @last_loc: if not ICE_FLOW_FLD_OFF_INVAL, location of last/upper value from + *            entry's input buffer + * @range: indicate if field being matched is to be in a range + * + * This function specifies the locations, in the form of byte offsets from the + * start of the input buffer for a flow entry, from where the value to match, + * the mask value, and upper value can be extracted. These locations are then + * stored in the flow profile. When adding a flow entry associated with the + * flow profile, these locations will be used to quickly extract the values and + * create the content of a match entry. This function should only be used for + * fixed-size data structures. + */ +static void +ice_flow_set_fld(struct ice_flow_seg_info *seg, enum ice_flow_field fld, +		 u16 val_loc, u16 mask_loc, u16 last_loc, bool range) +{ +	enum ice_flow_fld_match_type t = range ? +		ICE_FLOW_FLD_TYPE_RANGE : ICE_FLOW_FLD_TYPE_REG; + +	ice_flow_set_fld_ext(seg, fld, t, val_loc, mask_loc, last_loc); +} + +#define ICE_FLOW_RSS_SEG_HDR_L3_MASKS \ +	(ICE_FLOW_SEG_HDR_IPV4 | ICE_FLOW_SEG_HDR_IPV6) + +#define ICE_FLOW_RSS_SEG_HDR_L4_MASKS \ +	(ICE_FLOW_SEG_HDR_TCP | ICE_FLOW_SEG_HDR_UDP | ICE_FLOW_SEG_HDR_SCTP) + +#define ICE_FLOW_RSS_SEG_HDR_VAL_MASKS \ +	(ICE_FLOW_RSS_SEG_HDR_L3_MASKS | \ +	 ICE_FLOW_RSS_SEG_HDR_L4_MASKS) + +/** + * ice_flow_set_rss_seg_info - setup packet segments for RSS + * @segs: pointer to the flow field segment(s) + * @hash_fields: fields to be hashed on for the segment(s) + * @flow_hdr: protocol header fields within a packet segment + * + * Helper function to extract fields from hash bitmap and use flow + * header value to set flow field segment for further use in flow + * profile entry or removal. + */ +static enum ice_status +ice_flow_set_rss_seg_info(struct ice_flow_seg_info *segs, u64 hash_fields, +			  u32 flow_hdr) +{ +	u64 val; +	u8 i; + +	for_each_set_bit(i, (unsigned long *)&hash_fields, +			 ICE_FLOW_FIELD_IDX_MAX) +		ice_flow_set_fld(segs, (enum ice_flow_field)i, +				 ICE_FLOW_FLD_OFF_INVAL, ICE_FLOW_FLD_OFF_INVAL, +				 ICE_FLOW_FLD_OFF_INVAL, false); + +	ICE_FLOW_SET_HDRS(segs, flow_hdr); + +	if (segs->hdrs & ~ICE_FLOW_RSS_SEG_HDR_VAL_MASKS) +		return ICE_ERR_PARAM; + +	val = (u64)(segs->hdrs & ICE_FLOW_RSS_SEG_HDR_L3_MASKS); +	if (val && !is_power_of_2(val)) +		return ICE_ERR_CFG; + +	val = (u64)(segs->hdrs & ICE_FLOW_RSS_SEG_HDR_L4_MASKS); +	if (val && !is_power_of_2(val)) +		return ICE_ERR_CFG; + +	return 0; +} + +/** + * ice_rem_vsi_rss_list - remove VSI from RSS list + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * + * Remove the VSI from all RSS configurations in the list. + */ +void ice_rem_vsi_rss_list(struct ice_hw *hw, u16 vsi_handle) +{ +	struct ice_rss_cfg *r, *tmp; + +	if (list_empty(&hw->rss_list_head)) +		return; + +	mutex_lock(&hw->rss_locks); +	list_for_each_entry_safe(r, tmp, &hw->rss_list_head, l_entry) +		if (test_and_clear_bit(vsi_handle, r->vsis)) +			if (bitmap_empty(r->vsis, ICE_MAX_VSI)) { +				list_del(&r->l_entry); +				devm_kfree(ice_hw_to_dev(hw), r); +			} +	mutex_unlock(&hw->rss_locks); +} + +/** + * ice_rem_vsi_rss_cfg - remove RSS configurations associated with VSI + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * + * This function will iterate through all flow profiles and disassociate + * the VSI from that profile. If the flow profile has no VSIs it will + * be removed. + */ +enum ice_status ice_rem_vsi_rss_cfg(struct ice_hw *hw, u16 vsi_handle) +{ +	const enum ice_block blk = ICE_BLK_RSS; +	struct ice_flow_prof *p, *t; +	enum ice_status status = 0; + +	if (!ice_is_vsi_valid(hw, vsi_handle)) +		return ICE_ERR_PARAM; + +	if (list_empty(&hw->fl_profs[blk])) +		return 0; + +	mutex_lock(&hw->fl_profs_locks[blk]); +	list_for_each_entry_safe(p, t, &hw->fl_profs[blk], l_entry) +		if (test_bit(vsi_handle, p->vsis)) { +			status = ice_flow_disassoc_prof(hw, blk, p, vsi_handle); +			if (status) +				break; + +			if (bitmap_empty(p->vsis, ICE_MAX_VSI)) { +				status = ice_flow_rem_prof_sync(hw, blk, p); +				if (status) +					break; +			} +		} +	mutex_unlock(&hw->fl_profs_locks[blk]); + +	return status; +} + +/** + * ice_rem_rss_list - remove RSS configuration from list + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * @prof: pointer to flow profile + * + * Assumption: lock has already been acquired for RSS list + */ +static void +ice_rem_rss_list(struct ice_hw *hw, u16 vsi_handle, struct ice_flow_prof *prof) +{ +	struct ice_rss_cfg *r, *tmp; + +	/* Search for RSS hash fields associated to the VSI that match the +	 * hash configurations associated to the flow profile. If found +	 * remove from the RSS entry list of the VSI context and delete entry. +	 */ +	list_for_each_entry_safe(r, tmp, &hw->rss_list_head, l_entry) +		if (r->hashed_flds == prof->segs[prof->segs_cnt - 1].match && +		    r->packet_hdr == prof->segs[prof->segs_cnt - 1].hdrs) { +			clear_bit(vsi_handle, r->vsis); +			if (bitmap_empty(r->vsis, ICE_MAX_VSI)) { +				list_del(&r->l_entry); +				devm_kfree(ice_hw_to_dev(hw), r); +			} +			return; +		} +} + +/** + * ice_add_rss_list - add RSS configuration to list + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * @prof: pointer to flow profile + * + * Assumption: lock has already been acquired for RSS list + */ +static enum ice_status +ice_add_rss_list(struct ice_hw *hw, u16 vsi_handle, struct ice_flow_prof *prof) +{ +	struct ice_rss_cfg *r, *rss_cfg; + +	list_for_each_entry(r, &hw->rss_list_head, l_entry) +		if (r->hashed_flds == prof->segs[prof->segs_cnt - 1].match && +		    r->packet_hdr == prof->segs[prof->segs_cnt - 1].hdrs) { +			set_bit(vsi_handle, r->vsis); +			return 0; +		} + +	rss_cfg = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*rss_cfg), +			       GFP_KERNEL); +	if (!rss_cfg) +		return ICE_ERR_NO_MEMORY; + +	rss_cfg->hashed_flds = prof->segs[prof->segs_cnt - 1].match; +	rss_cfg->packet_hdr = prof->segs[prof->segs_cnt - 1].hdrs; +	set_bit(vsi_handle, rss_cfg->vsis); + +	list_add_tail(&rss_cfg->l_entry, &hw->rss_list_head); + +	return 0; +} + +#define ICE_FLOW_PROF_HASH_S	0 +#define ICE_FLOW_PROF_HASH_M	(0xFFFFFFFFULL << ICE_FLOW_PROF_HASH_S) +#define ICE_FLOW_PROF_HDR_S	32 +#define ICE_FLOW_PROF_HDR_M	(0x3FFFFFFFULL << ICE_FLOW_PROF_HDR_S) +#define ICE_FLOW_PROF_ENCAP_S	63 +#define ICE_FLOW_PROF_ENCAP_M	(BIT_ULL(ICE_FLOW_PROF_ENCAP_S)) + +#define ICE_RSS_OUTER_HEADERS	1 + +/* Flow profile ID format: + * [0:31] - Packet match fields + * [32:62] - Protocol header + * [63] - Encapsulation flag, 0 if non-tunneled, 1 if tunneled + */ +#define ICE_FLOW_GEN_PROFID(hash, hdr, segs_cnt) \ +	(u64)(((u64)(hash) & ICE_FLOW_PROF_HASH_M) | \ +	      (((u64)(hdr) << ICE_FLOW_PROF_HDR_S) & ICE_FLOW_PROF_HDR_M) | \ +	      ((u8)((segs_cnt) - 1) ? ICE_FLOW_PROF_ENCAP_M : 0)) + +/** + * ice_add_rss_cfg_sync - add an RSS configuration + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * @hashed_flds: hash bit fields (ICE_FLOW_HASH_*) to configure + * @addl_hdrs: protocol header fields + * @segs_cnt: packet segment count + * + * Assumption: lock has already been acquired for RSS list + */ +static enum ice_status +ice_add_rss_cfg_sync(struct ice_hw *hw, u16 vsi_handle, u64 hashed_flds, +		     u32 addl_hdrs, u8 segs_cnt) +{ +	const enum ice_block blk = ICE_BLK_RSS; +	struct ice_flow_prof *prof = NULL; +	struct ice_flow_seg_info *segs; +	enum ice_status status; + +	if (!segs_cnt || segs_cnt > ICE_FLOW_SEG_MAX) +		return ICE_ERR_PARAM; + +	segs = kcalloc(segs_cnt, sizeof(*segs), GFP_KERNEL); +	if (!segs) +		return ICE_ERR_NO_MEMORY; + +	/* Construct the packet segment info from the hashed fields */ +	status = ice_flow_set_rss_seg_info(&segs[segs_cnt - 1], hashed_flds, +					   addl_hdrs); +	if (status) +		goto exit; + +	/* Search for a flow profile that has matching headers, hash fields +	 * and has the input VSI associated to it. If found, no further +	 * operations required and exit. +	 */ +	prof = ice_flow_find_prof_conds(hw, blk, ICE_FLOW_RX, segs, segs_cnt, +					vsi_handle, +					ICE_FLOW_FIND_PROF_CHK_FLDS | +					ICE_FLOW_FIND_PROF_CHK_VSI); +	if (prof) +		goto exit; + +	/* Check if a flow profile exists with the same protocol headers and +	 * associated with the input VSI. If so disassociate the VSI from +	 * this profile. The VSI will be added to a new profile created with +	 * the protocol header and new hash field configuration. +	 */ +	prof = ice_flow_find_prof_conds(hw, blk, ICE_FLOW_RX, segs, segs_cnt, +					vsi_handle, ICE_FLOW_FIND_PROF_CHK_VSI); +	if (prof) { +		status = ice_flow_disassoc_prof(hw, blk, prof, vsi_handle); +		if (!status) +			ice_rem_rss_list(hw, vsi_handle, prof); +		else +			goto exit; + +		/* Remove profile if it has no VSIs associated */ +		if (bitmap_empty(prof->vsis, ICE_MAX_VSI)) { +			status = ice_flow_rem_prof(hw, blk, prof->id); +			if (status) +				goto exit; +		} +	} + +	/* Search for a profile that has same match fields only. If this +	 * exists then associate the VSI to this profile. +	 */ +	prof = ice_flow_find_prof_conds(hw, blk, ICE_FLOW_RX, segs, segs_cnt, +					vsi_handle, +					ICE_FLOW_FIND_PROF_CHK_FLDS); +	if (prof) { +		status = ice_flow_assoc_prof(hw, blk, prof, vsi_handle); +		if (!status) +			status = ice_add_rss_list(hw, vsi_handle, prof); +		goto exit; +	} + +	/* Create a new flow profile with generated profile and packet +	 * segment information. +	 */ +	status = ice_flow_add_prof(hw, blk, ICE_FLOW_RX, +				   ICE_FLOW_GEN_PROFID(hashed_flds, +						       segs[segs_cnt - 1].hdrs, +						       segs_cnt), +				   segs, segs_cnt, &prof); +	if (status) +		goto exit; + +	status = ice_flow_assoc_prof(hw, blk, prof, vsi_handle); +	/* If association to a new flow profile failed then this profile can +	 * be removed. +	 */ +	if (status) { +		ice_flow_rem_prof(hw, blk, prof->id); +		goto exit; +	} + +	status = ice_add_rss_list(hw, vsi_handle, prof); + +exit: +	kfree(segs); +	return status; +} + +/** + * ice_add_rss_cfg - add an RSS configuration with specified hashed fields + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * @hashed_flds: hash bit fields (ICE_FLOW_HASH_*) to configure + * @addl_hdrs: protocol header fields + * + * This function will generate a flow profile based on fields associated with + * the input fields to hash on, the flow type and use the VSI number to add + * a flow entry to the profile. + */ +enum ice_status +ice_add_rss_cfg(struct ice_hw *hw, u16 vsi_handle, u64 hashed_flds, +		u32 addl_hdrs) +{ +	enum ice_status status; + +	if (hashed_flds == ICE_HASH_INVALID || +	    !ice_is_vsi_valid(hw, vsi_handle)) +		return ICE_ERR_PARAM; + +	mutex_lock(&hw->rss_locks); +	status = ice_add_rss_cfg_sync(hw, vsi_handle, hashed_flds, addl_hdrs, +				      ICE_RSS_OUTER_HEADERS); +	mutex_unlock(&hw->rss_locks); + +	return status; +} + +/* Mapping of AVF hash bit fields to an L3-L4 hash combination. + * As the ice_flow_avf_hdr_field represent individual bit shifts in a hash, + * convert its values to their appropriate flow L3, L4 values. + */ +#define ICE_FLOW_AVF_RSS_IPV4_MASKS \ +	(BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_OTHER) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_FRAG_IPV4)) +#define ICE_FLOW_AVF_RSS_TCP_IPV4_MASKS \ +	(BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_TCP_SYN_NO_ACK) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_TCP)) +#define ICE_FLOW_AVF_RSS_UDP_IPV4_MASKS \ +	(BIT_ULL(ICE_AVF_FLOW_FIELD_UNICAST_IPV4_UDP) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_MULTICAST_IPV4_UDP) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_UDP)) +#define ICE_FLOW_AVF_RSS_ALL_IPV4_MASKS \ +	(ICE_FLOW_AVF_RSS_TCP_IPV4_MASKS | ICE_FLOW_AVF_RSS_UDP_IPV4_MASKS | \ +	 ICE_FLOW_AVF_RSS_IPV4_MASKS | BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_SCTP)) + +#define ICE_FLOW_AVF_RSS_IPV6_MASKS \ +	(BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_OTHER) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_FRAG_IPV6)) +#define ICE_FLOW_AVF_RSS_UDP_IPV6_MASKS \ +	(BIT_ULL(ICE_AVF_FLOW_FIELD_UNICAST_IPV6_UDP) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_MULTICAST_IPV6_UDP) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_UDP)) +#define ICE_FLOW_AVF_RSS_TCP_IPV6_MASKS \ +	(BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_TCP_SYN_NO_ACK) | \ +	 BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_TCP)) +#define ICE_FLOW_AVF_RSS_ALL_IPV6_MASKS \ +	(ICE_FLOW_AVF_RSS_TCP_IPV6_MASKS | ICE_FLOW_AVF_RSS_UDP_IPV6_MASKS | \ +	 ICE_FLOW_AVF_RSS_IPV6_MASKS | BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_SCTP)) + +/** + * ice_add_avf_rss_cfg - add an RSS configuration for AVF driver + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * @avf_hash: hash bit fields (ICE_AVF_FLOW_FIELD_*) to configure + * + * This function will take the hash bitmap provided by the AVF driver via a + * message, convert it to ICE-compatible values, and configure RSS flow + * profiles. + */ +enum ice_status +ice_add_avf_rss_cfg(struct ice_hw *hw, u16 vsi_handle, u64 avf_hash) +{ +	enum ice_status status = 0; +	u64 hash_flds; + +	if (avf_hash == ICE_AVF_FLOW_FIELD_INVALID || +	    !ice_is_vsi_valid(hw, vsi_handle)) +		return ICE_ERR_PARAM; + +	/* Make sure no unsupported bits are specified */ +	if (avf_hash & ~(ICE_FLOW_AVF_RSS_ALL_IPV4_MASKS | +			 ICE_FLOW_AVF_RSS_ALL_IPV6_MASKS)) +		return ICE_ERR_CFG; + +	hash_flds = avf_hash; + +	/* Always create an L3 RSS configuration for any L4 RSS configuration */ +	if (hash_flds & ICE_FLOW_AVF_RSS_ALL_IPV4_MASKS) +		hash_flds |= ICE_FLOW_AVF_RSS_IPV4_MASKS; + +	if (hash_flds & ICE_FLOW_AVF_RSS_ALL_IPV6_MASKS) +		hash_flds |= ICE_FLOW_AVF_RSS_IPV6_MASKS; + +	/* Create the corresponding RSS configuration for each valid hash bit */ +	while (hash_flds) { +		u64 rss_hash = ICE_HASH_INVALID; + +		if (hash_flds & ICE_FLOW_AVF_RSS_ALL_IPV4_MASKS) { +			if (hash_flds & ICE_FLOW_AVF_RSS_IPV4_MASKS) { +				rss_hash = ICE_FLOW_HASH_IPV4; +				hash_flds &= ~ICE_FLOW_AVF_RSS_IPV4_MASKS; +			} else if (hash_flds & +				   ICE_FLOW_AVF_RSS_TCP_IPV4_MASKS) { +				rss_hash = ICE_FLOW_HASH_IPV4 | +					ICE_FLOW_HASH_TCP_PORT; +				hash_flds &= ~ICE_FLOW_AVF_RSS_TCP_IPV4_MASKS; +			} else if (hash_flds & +				   ICE_FLOW_AVF_RSS_UDP_IPV4_MASKS) { +				rss_hash = ICE_FLOW_HASH_IPV4 | +					ICE_FLOW_HASH_UDP_PORT; +				hash_flds &= ~ICE_FLOW_AVF_RSS_UDP_IPV4_MASKS; +			} else if (hash_flds & +				   BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_SCTP)) { +				rss_hash = ICE_FLOW_HASH_IPV4 | +					ICE_FLOW_HASH_SCTP_PORT; +				hash_flds &= +					~BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_SCTP); +			} +		} else if (hash_flds & ICE_FLOW_AVF_RSS_ALL_IPV6_MASKS) { +			if (hash_flds & ICE_FLOW_AVF_RSS_IPV6_MASKS) { +				rss_hash = ICE_FLOW_HASH_IPV6; +				hash_flds &= ~ICE_FLOW_AVF_RSS_IPV6_MASKS; +			} else if (hash_flds & +				   ICE_FLOW_AVF_RSS_TCP_IPV6_MASKS) { +				rss_hash = ICE_FLOW_HASH_IPV6 | +					ICE_FLOW_HASH_TCP_PORT; +				hash_flds &= ~ICE_FLOW_AVF_RSS_TCP_IPV6_MASKS; +			} else if (hash_flds & +				   ICE_FLOW_AVF_RSS_UDP_IPV6_MASKS) { +				rss_hash = ICE_FLOW_HASH_IPV6 | +					ICE_FLOW_HASH_UDP_PORT; +				hash_flds &= ~ICE_FLOW_AVF_RSS_UDP_IPV6_MASKS; +			} else if (hash_flds & +				   BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_SCTP)) { +				rss_hash = ICE_FLOW_HASH_IPV6 | +					ICE_FLOW_HASH_SCTP_PORT; +				hash_flds &= +					~BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_SCTP); +			} +		} + +		if (rss_hash == ICE_HASH_INVALID) +			return ICE_ERR_OUT_OF_RANGE; + +		status = ice_add_rss_cfg(hw, vsi_handle, rss_hash, +					 ICE_FLOW_SEG_HDR_NONE); +		if (status) +			break; +	} + +	return status; +} + +/** + * ice_replay_rss_cfg - replay RSS configurations associated with VSI + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + */ +enum ice_status ice_replay_rss_cfg(struct ice_hw *hw, u16 vsi_handle) +{ +	enum ice_status status = 0; +	struct ice_rss_cfg *r; + +	if (!ice_is_vsi_valid(hw, vsi_handle)) +		return ICE_ERR_PARAM; + +	mutex_lock(&hw->rss_locks); +	list_for_each_entry(r, &hw->rss_list_head, l_entry) { +		if (test_bit(vsi_handle, r->vsis)) { +			status = ice_add_rss_cfg_sync(hw, vsi_handle, +						      r->hashed_flds, +						      r->packet_hdr, +						      ICE_RSS_OUTER_HEADERS); +			if (status) +				break; +		} +	} +	mutex_unlock(&hw->rss_locks); + +	return status; +} + +/** + * ice_get_rss_cfg - returns hashed fields for the given header types + * @hw: pointer to the hardware structure + * @vsi_handle: software VSI handle + * @hdrs: protocol header type + * + * This function will return the match fields of the first instance of flow + * profile having the given header types and containing input VSI + */ +u64 ice_get_rss_cfg(struct ice_hw *hw, u16 vsi_handle, u32 hdrs) +{ +	struct ice_rss_cfg *r, *rss_cfg = NULL; + +	/* verify if the protocol header is non zero and VSI is valid */ +	if (hdrs == ICE_FLOW_SEG_HDR_NONE || !ice_is_vsi_valid(hw, vsi_handle)) +		return ICE_HASH_INVALID; + +	mutex_lock(&hw->rss_locks); +	list_for_each_entry(r, &hw->rss_list_head, l_entry) +		if (test_bit(vsi_handle, r->vsis) && +		    r->packet_hdr == hdrs) { +			rss_cfg = r; +			break; +		} +	mutex_unlock(&hw->rss_locks); + +	return rss_cfg ? rss_cfg->hashed_flds : ICE_HASH_INVALID; +} diff --git a/drivers/net/ethernet/intel/ice/ice_flow.h b/drivers/net/ethernet/intel/ice/ice_flow.h new file mode 100644 index 000000000000..5558627bd5eb --- /dev/null +++ b/drivers/net/ethernet/intel/ice/ice_flow.h @@ -0,0 +1,207 @@ +/* SPDX-License-Identifier: GPL-2.0 */ +/* Copyright (c) 2019, Intel Corporation. */ + +#ifndef _ICE_FLOW_H_ +#define _ICE_FLOW_H_ + +#define ICE_FLOW_ENTRY_HANDLE_INVAL	0 +#define ICE_FLOW_FLD_OFF_INVAL		0xffff + +/* Generate flow hash field from flow field type(s) */ +#define ICE_FLOW_HASH_IPV4	\ +	(BIT_ULL(ICE_FLOW_FIELD_IDX_IPV4_SA) | \ +	 BIT_ULL(ICE_FLOW_FIELD_IDX_IPV4_DA)) +#define ICE_FLOW_HASH_IPV6	\ +	(BIT_ULL(ICE_FLOW_FIELD_IDX_IPV6_SA) | \ +	 BIT_ULL(ICE_FLOW_FIELD_IDX_IPV6_DA)) +#define ICE_FLOW_HASH_TCP_PORT	\ +	(BIT_ULL(ICE_FLOW_FIELD_IDX_TCP_SRC_PORT) | \ +	 BIT_ULL(ICE_FLOW_FIELD_IDX_TCP_DST_PORT)) +#define ICE_FLOW_HASH_UDP_PORT	\ +	(BIT_ULL(ICE_FLOW_FIELD_IDX_UDP_SRC_PORT) | \ +	 BIT_ULL(ICE_FLOW_FIELD_IDX_UDP_DST_PORT)) +#define ICE_FLOW_HASH_SCTP_PORT	\ +	(BIT_ULL(ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT) | \ +	 BIT_ULL(ICE_FLOW_FIELD_IDX_SCTP_DST_PORT)) + +#define ICE_HASH_INVALID	0 +#define ICE_HASH_TCP_IPV4	(ICE_FLOW_HASH_IPV4 | ICE_FLOW_HASH_TCP_PORT) +#define ICE_HASH_TCP_IPV6	(ICE_FLOW_HASH_IPV6 | ICE_FLOW_HASH_TCP_PORT) +#define ICE_HASH_UDP_IPV4	(ICE_FLOW_HASH_IPV4 | ICE_FLOW_HASH_UDP_PORT) +#define ICE_HASH_UDP_IPV6	(ICE_FLOW_HASH_IPV6 | ICE_FLOW_HASH_UDP_PORT) + +/* Protocol header fields within a packet segment. A segment consists of one or + * more protocol headers that make up a logical group of protocol headers. Each + * logical group of protocol headers encapsulates or is encapsulated using/by + * tunneling or encapsulation protocols for network virtualization such as GRE, + * VxLAN, etc. + */ +enum ice_flow_seg_hdr { +	ICE_FLOW_SEG_HDR_NONE		= 0x00000000, +	ICE_FLOW_SEG_HDR_IPV4		= 0x00000004, +	ICE_FLOW_SEG_HDR_IPV6		= 0x00000008, +	ICE_FLOW_SEG_HDR_TCP		= 0x00000040, +	ICE_FLOW_SEG_HDR_UDP		= 0x00000080, +	ICE_FLOW_SEG_HDR_SCTP		= 0x00000100, +}; + +enum ice_flow_field { +	/* L3 */ +	ICE_FLOW_FIELD_IDX_IPV4_SA, +	ICE_FLOW_FIELD_IDX_IPV4_DA, +	ICE_FLOW_FIELD_IDX_IPV6_SA, +	ICE_FLOW_FIELD_IDX_IPV6_DA, +	/* L4 */ +	ICE_FLOW_FIELD_IDX_TCP_SRC_PORT, +	ICE_FLOW_FIELD_IDX_TCP_DST_PORT, +	ICE_FLOW_FIELD_IDX_UDP_SRC_PORT, +	ICE_FLOW_FIELD_IDX_UDP_DST_PORT, +	ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT, +	ICE_FLOW_FIELD_IDX_SCTP_DST_PORT, +	/* The total number of enums must not exceed 64 */ +	ICE_FLOW_FIELD_IDX_MAX +}; + +/* Flow headers and fields for AVF support */ +enum ice_flow_avf_hdr_field { +	/* Values 0 - 28 are reserved for future use */ +	ICE_AVF_FLOW_FIELD_INVALID		= 0, +	ICE_AVF_FLOW_FIELD_UNICAST_IPV4_UDP	= 29, +	ICE_AVF_FLOW_FIELD_MULTICAST_IPV4_UDP, +	ICE_AVF_FLOW_FIELD_IPV4_UDP, +	ICE_AVF_FLOW_FIELD_IPV4_TCP_SYN_NO_ACK, +	ICE_AVF_FLOW_FIELD_IPV4_TCP, +	ICE_AVF_FLOW_FIELD_IPV4_SCTP, +	ICE_AVF_FLOW_FIELD_IPV4_OTHER, +	ICE_AVF_FLOW_FIELD_FRAG_IPV4, +	/* Values 37-38 are reserved */ +	ICE_AVF_FLOW_FIELD_UNICAST_IPV6_UDP	= 39, +	ICE_AVF_FLOW_FIELD_MULTICAST_IPV6_UDP, +	ICE_AVF_FLOW_FIELD_IPV6_UDP, +	ICE_AVF_FLOW_FIELD_IPV6_TCP_SYN_NO_ACK, +	ICE_AVF_FLOW_FIELD_IPV6_TCP, +	ICE_AVF_FLOW_FIELD_IPV6_SCTP, +	ICE_AVF_FLOW_FIELD_IPV6_OTHER, +	ICE_AVF_FLOW_FIELD_FRAG_IPV6, +	ICE_AVF_FLOW_FIELD_RSVD47, +	ICE_AVF_FLOW_FIELD_FCOE_OX, +	ICE_AVF_FLOW_FIELD_FCOE_RX, +	ICE_AVF_FLOW_FIELD_FCOE_OTHER, +	/* Values 51-62 are reserved */ +	ICE_AVF_FLOW_FIELD_L2_PAYLOAD		= 63, +	ICE_AVF_FLOW_FIELD_MAX +}; + +/* Supported RSS offloads  This macro is defined to support + * VIRTCHNL_OP_GET_RSS_HENA_CAPS ops. PF driver sends the RSS hardware + * capabilities to the caller of this ops. + */ +#define ICE_DEFAULT_RSS_HENA ( \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_UDP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_SCTP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_TCP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_OTHER) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_FRAG_IPV4) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_UDP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_TCP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_SCTP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_OTHER) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_FRAG_IPV6) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV4_TCP_SYN_NO_ACK) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_UNICAST_IPV4_UDP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_MULTICAST_IPV4_UDP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_IPV6_TCP_SYN_NO_ACK) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_UNICAST_IPV6_UDP) | \ +	BIT_ULL(ICE_AVF_FLOW_FIELD_MULTICAST_IPV6_UDP)) + +enum ice_flow_dir { +	ICE_FLOW_RX		= 0x02, +}; + +enum ice_flow_priority { +	ICE_FLOW_PRIO_LOW, +	ICE_FLOW_PRIO_NORMAL, +	ICE_FLOW_PRIO_HIGH +}; + +#define ICE_FLOW_SEG_MAX		2 +#define ICE_FLOW_FV_EXTRACT_SZ		2 + +#define ICE_FLOW_SET_HDRS(seg, val)	((seg)->hdrs |= (u32)(val)) + +struct ice_flow_seg_xtrct { +	u8 prot_id;	/* Protocol ID of extracted header field */ +	u16 off;	/* Starting offset of the field in header in bytes */ +	u8 idx;		/* Index of FV entry used */ +	u8 disp;	/* Displacement of field in bits fr. FV entry's start */ +}; + +enum ice_flow_fld_match_type { +	ICE_FLOW_FLD_TYPE_REG,		/* Value, mask */ +	ICE_FLOW_FLD_TYPE_RANGE,	/* Value, mask, last (upper bound) */ +	ICE_FLOW_FLD_TYPE_PREFIX,	/* IP address, prefix, size of prefix */ +	ICE_FLOW_FLD_TYPE_SIZE,		/* Value, mask, size of match */ +}; + +struct ice_flow_fld_loc { +	/* Describe offsets of field information relative to the beginning of +	 * input buffer provided when adding flow entries. +	 */ +	u16 val;	/* Offset where the value is located */ +	u16 mask;	/* Offset where the mask/prefix value is located */ +	u16 last;	/* Length or offset where the upper value is located */ +}; + +struct ice_flow_fld_info { +	enum ice_flow_fld_match_type type; +	/* Location where to retrieve data from an input buffer */ +	struct ice_flow_fld_loc src; +	/* Location where to put the data into the final entry buffer */ +	struct ice_flow_fld_loc entry; +	struct ice_flow_seg_xtrct xtrct; +}; + +struct ice_flow_seg_info { +	u32 hdrs;	/* Bitmask indicating protocol headers present */ +	u64 match;	/* Bitmask indicating header fields to be matched */ +	u64 range;	/* Bitmask indicating header fields matched as ranges */ + +	struct ice_flow_fld_info fields[ICE_FLOW_FIELD_IDX_MAX]; +}; + +struct ice_flow_prof { +	struct list_head l_entry; + +	u64 id; +	enum ice_flow_dir dir; +	u8 segs_cnt; + +	/* Keep track of flow entries associated with this flow profile */ +	struct mutex entries_lock; +	struct list_head entries; + +	struct ice_flow_seg_info segs[ICE_FLOW_SEG_MAX]; + +	/* software VSI handles referenced by this flow profile */ +	DECLARE_BITMAP(vsis, ICE_MAX_VSI); +}; + +struct ice_rss_cfg { +	struct list_head l_entry; +	/* bitmap of VSIs added to the RSS entry */ +	DECLARE_BITMAP(vsis, ICE_MAX_VSI); +	u64 hashed_flds; +	u32 packet_hdr; +}; + +enum ice_status ice_flow_rem_entry(struct ice_hw *hw, u64 entry_h); +void ice_rem_vsi_rss_list(struct ice_hw *hw, u16 vsi_handle); +enum ice_status ice_replay_rss_cfg(struct ice_hw *hw, u16 vsi_handle); +enum ice_status +ice_add_avf_rss_cfg(struct ice_hw *hw, u16 vsi_handle, u64 hashed_flds); +enum ice_status ice_rem_vsi_rss_cfg(struct ice_hw *hw, u16 vsi_handle); +enum ice_status +ice_add_rss_cfg(struct ice_hw *hw, u16 vsi_handle, u64 hashed_flds, +		u32 addl_hdrs); +u64 ice_get_rss_cfg(struct ice_hw *hw, u16 vsi_handle, u32 hdrs); +#endif /* _ICE_FLOW_H_ */ diff --git a/drivers/net/ethernet/intel/ice/ice_hw_autogen.h b/drivers/net/ethernet/intel/ice/ice_hw_autogen.h index e8f32350fed2..f2cababf2561 100644 --- a/drivers/net/ethernet/intel/ice/ice_hw_autogen.h +++ b/drivers/net/ethernet/intel/ice/ice_hw_autogen.h @@ -60,15 +60,6 @@  #define PRTDCB_GENS_DCBX_STATUS_M		ICE_M(0x7, 0)  #define GL_PREEXT_L2_PMASK0(_i)			(0x0020F0FC + ((_i) * 4))  #define GL_PREEXT_L2_PMASK1(_i)			(0x0020F108 + ((_i) * 4)) -#define GLFLXP_RXDID_FLAGS(_i, _j)		(0x0045D000 + ((_i) * 4 + (_j) * 256)) -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_S	0 -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_M	ICE_M(0x3F, 0) -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_1_S	8 -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_1_M	ICE_M(0x3F, 8) -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_2_S	16 -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_2_M	ICE_M(0x3F, 16) -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_3_S	24 -#define GLFLXP_RXDID_FLAGS_FLEXIFLAG_4N_3_M	ICE_M(0x3F, 24)  #define GLFLXP_RXDID_FLX_WRD_0(_i)		(0x0045c800 + ((_i) * 4))  #define GLFLXP_RXDID_FLX_WRD_0_PROT_MDID_S	0  #define GLFLXP_RXDID_FLX_WRD_0_PROT_MDID_M	ICE_M(0xFF, 0) diff --git a/drivers/net/ethernet/intel/ice/ice_lan_tx_rx.h b/drivers/net/ethernet/intel/ice/ice_lan_tx_rx.h index 0997d352709b..878e125d8b42 100644 --- a/drivers/net/ethernet/intel/ice/ice_lan_tx_rx.h +++ b/drivers/net/ethernet/intel/ice/ice_lan_tx_rx.h @@ -199,6 +199,14 @@ enum ice_rxdid {  /* Receive Flex Descriptor Rx opcode values */  #define ICE_RX_OPC_MDID		0x01 +/* Receive Descriptor MDID values that access packet flags */ +enum ice_flex_mdid_pkt_flags { +	ICE_RX_MDID_PKT_FLAGS_15_0	= 20, +	ICE_RX_MDID_PKT_FLAGS_31_16, +	ICE_RX_MDID_PKT_FLAGS_47_32, +	ICE_RX_MDID_PKT_FLAGS_63_48, +}; +  /* Receive Descriptor MDID values */  enum ice_flex_rx_mdid {  	ICE_RX_MDID_FLOW_ID_LOWER	= 5, diff --git a/drivers/net/ethernet/intel/ice/ice_lib.c b/drivers/net/ethernet/intel/ice/ice_lib.c index e7449248fab4..1874c9f51a32 100644 --- a/drivers/net/ethernet/intel/ice/ice_lib.c +++ b/drivers/net/ethernet/intel/ice/ice_lib.c @@ -3,6 +3,7 @@  #include "ice.h"  #include "ice_base.h" +#include "ice_flow.h"  #include "ice_lib.h"  #include "ice_dcb_lib.h" @@ -493,7 +494,28 @@ bool ice_is_safe_mode(struct ice_pf *pf)  }  /** - * ice_rss_clean - Delete RSS related VSI structures that hold user inputs + * ice_vsi_clean_rss_flow_fld - Delete RSS configuration + * @vsi: the VSI being cleaned up + * + * This function deletes RSS input set for all flows that were configured + * for this VSI + */ +static void ice_vsi_clean_rss_flow_fld(struct ice_vsi *vsi) +{ +	struct ice_pf *pf = vsi->back; +	enum ice_status status; + +	if (ice_is_safe_mode(pf)) +		return; + +	status = ice_rem_vsi_rss_cfg(&pf->hw, vsi->idx); +	if (status) +		dev_dbg(ice_pf_to_dev(pf), "ice_rem_vsi_rss_cfg failed for vsi = %d, error = %d\n", +			vsi->vsi_num, status); +} + +/** + * ice_rss_clean - Delete RSS related VSI structures and configuration   * @vsi: the VSI being removed   */  static void ice_rss_clean(struct ice_vsi *vsi) @@ -507,6 +529,11 @@ static void ice_rss_clean(struct ice_vsi *vsi)  		devm_kfree(dev, vsi->rss_hkey_user);  	if (vsi->rss_lut_user)  		devm_kfree(dev, vsi->rss_lut_user); + +	ice_vsi_clean_rss_flow_fld(vsi); +	/* remove RSS replay list */ +	if (!ice_is_safe_mode(pf)) +		ice_rem_vsi_rss_list(&pf->hw, vsi->idx);  }  /** @@ -817,12 +844,23 @@ static int ice_vsi_init(struct ice_vsi *vsi, bool init_vsi)  		ctxt->info.valid_sections |=  			cpu_to_le16(ICE_AQ_VSI_PROP_RXQ_MAP_VALID); -	/* Enable MAC Antispoof with new VSI being initialized or updated */ -	if (vsi->type == ICE_VSI_VF && pf->vf[vsi->vf_id].spoofchk) { +	/* enable/disable MAC and VLAN anti-spoof when spoofchk is on/off +	 * respectively +	 */ +	if (vsi->type == ICE_VSI_VF) {  		ctxt->info.valid_sections |=  			cpu_to_le16(ICE_AQ_VSI_PROP_SECURITY_VALID); -		ctxt->info.sec_flags |= -			ICE_AQ_VSI_SEC_FLAG_ENA_MAC_ANTI_SPOOF; +		if (pf->vf[vsi->vf_id].spoofchk) { +			ctxt->info.sec_flags |= +				ICE_AQ_VSI_SEC_FLAG_ENA_MAC_ANTI_SPOOF | +				(ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA << +				 ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S); +		} else { +			ctxt->info.sec_flags &= +				~(ICE_AQ_VSI_SEC_FLAG_ENA_MAC_ANTI_SPOOF | +				  (ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA << +				   ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S)); +		}  	}  	/* Allow control frames out of main VSI */ @@ -1076,6 +1114,115 @@ ice_vsi_cfg_rss_exit:  }  /** + * ice_vsi_set_vf_rss_flow_fld - Sets VF VSI RSS input set for different flows + * @vsi: VSI to be configured + * + * This function will only be called during the VF VSI setup. Upon successful + * completion of package download, this function will configure default RSS + * input sets for VF VSI. + */ +static void ice_vsi_set_vf_rss_flow_fld(struct ice_vsi *vsi) +{ +	struct ice_pf *pf = vsi->back; +	enum ice_status status; +	struct device *dev; + +	dev = ice_pf_to_dev(pf); +	if (ice_is_safe_mode(pf)) { +		dev_dbg(dev, "Advanced RSS disabled. Package download failed, vsi num = %d\n", +			vsi->vsi_num); +		return; +	} + +	status = ice_add_avf_rss_cfg(&pf->hw, vsi->idx, ICE_DEFAULT_RSS_HENA); +	if (status) +		dev_dbg(dev, "ice_add_avf_rss_cfg failed for vsi = %d, error = %d\n", +			vsi->vsi_num, status); +} + +/** + * ice_vsi_set_rss_flow_fld - Sets RSS input set for different flows + * @vsi: VSI to be configured + * + * This function will only be called after successful download package call + * during initialization of PF. Since the downloaded package will erase the + * RSS section, this function will configure RSS input sets for different + * flow types. The last profile added has the highest priority, therefore 2 + * tuple profiles (i.e. IPv4 src/dst) are added before 4 tuple profiles + * (i.e. IPv4 src/dst TCP src/dst port). + */ +static void ice_vsi_set_rss_flow_fld(struct ice_vsi *vsi) +{ +	u16 vsi_handle = vsi->idx, vsi_num = vsi->vsi_num; +	struct ice_pf *pf = vsi->back; +	struct ice_hw *hw = &pf->hw; +	enum ice_status status; +	struct device *dev; + +	dev = ice_pf_to_dev(pf); +	if (ice_is_safe_mode(pf)) { +		dev_dbg(dev, "Advanced RSS disabled. Package download failed, vsi num = %d\n", +			vsi_num); +		return; +	} +	/* configure RSS for IPv4 with input set IP src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_FLOW_HASH_IPV4, +				 ICE_FLOW_SEG_HDR_IPV4); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for ipv4 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for IPv6 with input set IPv6 src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_FLOW_HASH_IPV6, +				 ICE_FLOW_SEG_HDR_IPV6); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for ipv6 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for tcp4 with input set IP src/dst, TCP src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_HASH_TCP_IPV4, +				 ICE_FLOW_SEG_HDR_TCP | ICE_FLOW_SEG_HDR_IPV4); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for tcp4 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for udp4 with input set IP src/dst, UDP src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_HASH_UDP_IPV4, +				 ICE_FLOW_SEG_HDR_UDP | ICE_FLOW_SEG_HDR_IPV4); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for udp4 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for sctp4 with input set IP src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_FLOW_HASH_IPV4, +				 ICE_FLOW_SEG_HDR_SCTP | ICE_FLOW_SEG_HDR_IPV4); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for sctp4 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for tcp6 with input set IPv6 src/dst, TCP src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_HASH_TCP_IPV6, +				 ICE_FLOW_SEG_HDR_TCP | ICE_FLOW_SEG_HDR_IPV6); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for tcp6 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for udp6 with input set IPv6 src/dst, UDP src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_HASH_UDP_IPV6, +				 ICE_FLOW_SEG_HDR_UDP | ICE_FLOW_SEG_HDR_IPV6); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for udp6 flow, vsi = %d, error = %d\n", +			vsi_num, status); + +	/* configure RSS for sctp6 with input set IPv6 src/dst */ +	status = ice_add_rss_cfg(hw, vsi_handle, ICE_FLOW_HASH_IPV6, +				 ICE_FLOW_SEG_HDR_SCTP | ICE_FLOW_SEG_HDR_IPV6); +	if (status) +		dev_dbg(dev, "ice_add_rss_cfg failed for sctp6 flow, vsi = %d, error = %d\n", +			vsi_num, status); +} + +/**   * ice_add_mac_to_list - Add a MAC address filter entry to the list   * @vsi: the VSI to be forwarded to   * @add_list: pointer to the list which contains MAC filter entries @@ -1636,22 +1783,14 @@ int ice_cfg_vlan_pruning(struct ice_vsi *vsi, bool ena, bool vlan_promisc)  	ctxt->info = vsi->info; -	if (ena) { -		ctxt->info.sec_flags |= -			ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA << -			ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S; +	if (ena)  		ctxt->info.sw_flags2 |= ICE_AQ_VSI_SW_FLAG_RX_VLAN_PRUNE_ENA; -	} else { -		ctxt->info.sec_flags &= -			~(ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA << -			  ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S); +	else  		ctxt->info.sw_flags2 &= ~ICE_AQ_VSI_SW_FLAG_RX_VLAN_PRUNE_ENA; -	}  	if (!vlan_promisc)  		ctxt->info.valid_sections = -			cpu_to_le16(ICE_AQ_VSI_PROP_SECURITY_VALID | -				    ICE_AQ_VSI_PROP_SW_VALID); +			cpu_to_le16(ICE_AQ_VSI_PROP_SW_VALID);  	status = ice_update_vsi(&pf->hw, vsi->idx, ctxt, NULL);  	if (status) { @@ -1661,7 +1800,6 @@ int ice_cfg_vlan_pruning(struct ice_vsi *vsi, bool ena, bool vlan_promisc)  		goto err_out;  	} -	vsi->info.sec_flags = ctxt->info.sec_flags;  	vsi->info.sw_flags2 = ctxt->info.sw_flags2;  	kfree(ctxt); @@ -1899,8 +2037,10 @@ ice_vsi_setup(struct ice_pf *pf, struct ice_port_info *pi,  		 * receive traffic on first queue. Hence no need to capture  		 * return value  		 */ -		if (test_bit(ICE_FLAG_RSS_ENA, pf->flags)) +		if (test_bit(ICE_FLAG_RSS_ENA, pf->flags)) {  			ice_vsi_cfg_rss_lut_key(vsi); +			ice_vsi_set_rss_flow_fld(vsi); +		}  		break;  	case ICE_VSI_VF:  		/* VF driver will take care of creating netdev for this type and @@ -1924,8 +2064,10 @@ ice_vsi_setup(struct ice_pf *pf, struct ice_port_info *pi,  		 * receive traffic on first queue. Hence no need to capture  		 * return value  		 */ -		if (test_bit(ICE_FLAG_RSS_ENA, pf->flags)) +		if (test_bit(ICE_FLAG_RSS_ENA, pf->flags)) {  			ice_vsi_cfg_rss_lut_key(vsi); +			ice_vsi_set_vf_rss_flow_fld(vsi); +		}  		break;  	case ICE_VSI_LB:  		ret = ice_vsi_alloc_rings(vsi); @@ -2402,6 +2544,97 @@ int ice_vsi_release(struct ice_vsi *vsi)  }  /** + * ice_vsi_rebuild_update_coalesce - set coalesce for a q_vector + * @q_vector: pointer to q_vector which is being updated + * @coalesce: pointer to array of struct with stored coalesce + * + * Set coalesce param in q_vector and update these parameters in HW. + */ +static void +ice_vsi_rebuild_update_coalesce(struct ice_q_vector *q_vector, +				struct ice_coalesce_stored *coalesce) +{ +	struct ice_ring_container *rx_rc = &q_vector->rx; +	struct ice_ring_container *tx_rc = &q_vector->tx; +	struct ice_hw *hw = &q_vector->vsi->back->hw; + +	tx_rc->itr_setting = coalesce->itr_tx; +	rx_rc->itr_setting = coalesce->itr_rx; + +	/* dynamic ITR values will be updated during Tx/Rx */ +	if (!ITR_IS_DYNAMIC(tx_rc->itr_setting)) +		wr32(hw, GLINT_ITR(tx_rc->itr_idx, q_vector->reg_idx), +		     ITR_REG_ALIGN(tx_rc->itr_setting) >> +		     ICE_ITR_GRAN_S); +	if (!ITR_IS_DYNAMIC(rx_rc->itr_setting)) +		wr32(hw, GLINT_ITR(rx_rc->itr_idx, q_vector->reg_idx), +		     ITR_REG_ALIGN(rx_rc->itr_setting) >> +		     ICE_ITR_GRAN_S); + +	q_vector->intrl = coalesce->intrl; +	wr32(hw, GLINT_RATE(q_vector->reg_idx), +	     ice_intrl_usec_to_reg(q_vector->intrl, hw->intrl_gran)); +} + +/** + * ice_vsi_rebuild_get_coalesce - get coalesce from all q_vectors + * @vsi: VSI connected with q_vectors + * @coalesce: array of struct with stored coalesce + * + * Returns array size. + */ +static int +ice_vsi_rebuild_get_coalesce(struct ice_vsi *vsi, +			     struct ice_coalesce_stored *coalesce) +{ +	int i; + +	ice_for_each_q_vector(vsi, i) { +		struct ice_q_vector *q_vector = vsi->q_vectors[i]; + +		coalesce[i].itr_tx = q_vector->tx.itr_setting; +		coalesce[i].itr_rx = q_vector->rx.itr_setting; +		coalesce[i].intrl = q_vector->intrl; +	} + +	return vsi->num_q_vectors; +} + +/** + * ice_vsi_rebuild_set_coalesce - set coalesce from earlier saved arrays + * @vsi: VSI connected with q_vectors + * @coalesce: pointer to array of struct with stored coalesce + * @size: size of coalesce array + * + * Before this function, ice_vsi_rebuild_get_coalesce should be called to save + * ITR params in arrays. If size is 0 or coalesce wasn't stored set coalesce + * to default value. + */ +static void +ice_vsi_rebuild_set_coalesce(struct ice_vsi *vsi, +			     struct ice_coalesce_stored *coalesce, int size) +{ +	int i; + +	if ((size && !coalesce) || !vsi) +		return; + +	for (i = 0; i < size && i < vsi->num_q_vectors; i++) +		ice_vsi_rebuild_update_coalesce(vsi->q_vectors[i], +						&coalesce[i]); + +	for (; i < vsi->num_q_vectors; i++) { +		struct ice_coalesce_stored coalesce_dflt = { +			.itr_tx = ICE_DFLT_TX_ITR, +			.itr_rx = ICE_DFLT_RX_ITR, +			.intrl = 0 +		}; +		ice_vsi_rebuild_update_coalesce(vsi->q_vectors[i], +						&coalesce_dflt); +	} +} + +/**   * ice_vsi_rebuild - Rebuild VSI after reset   * @vsi: VSI to be rebuild   * @init_vsi: is this an initialization or a reconfigure of the VSI @@ -2411,6 +2644,8 @@ int ice_vsi_release(struct ice_vsi *vsi)  int ice_vsi_rebuild(struct ice_vsi *vsi, bool init_vsi)  {  	u16 max_txqs[ICE_MAX_TRAFFIC_CLASS] = { 0 }; +	struct ice_coalesce_stored *coalesce; +	int prev_num_q_vectors = 0;  	struct ice_vf *vf = NULL;  	enum ice_status status;  	struct ice_pf *pf; @@ -2423,6 +2658,11 @@ int ice_vsi_rebuild(struct ice_vsi *vsi, bool init_vsi)  	if (vsi->type == ICE_VSI_VF)  		vf = &pf->vf[vsi->vf_id]; +	coalesce = kcalloc(vsi->num_q_vectors, +			   sizeof(struct ice_coalesce_stored), GFP_KERNEL); +	if (coalesce) +		prev_num_q_vectors = ice_vsi_rebuild_get_coalesce(vsi, +								  coalesce);  	ice_rm_vsi_lan_cfg(vsi->port_info, vsi->idx);  	ice_vsi_free_q_vectors(vsi); @@ -2535,6 +2775,9 @@ int ice_vsi_rebuild(struct ice_vsi *vsi, bool init_vsi)  			return ice_schedule_reset(pf, ICE_RESET_PFR);  		}  	} +	ice_vsi_rebuild_set_coalesce(vsi, coalesce, prev_num_q_vectors); +	kfree(coalesce); +  	return 0;  err_vectors: @@ -2549,6 +2792,7 @@ err_rings:  err_vsi:  	ice_vsi_clear(vsi);  	set_bit(__ICE_RESET_FAILED, pf->state); +	kfree(coalesce);  	return ret;  } @@ -2740,3 +2984,121 @@ cfg_mac_fltr_exit:  	ice_free_fltr_list(&vsi->back->pdev->dev, &tmp_add_list);  	return status;  } + +/** + * ice_is_dflt_vsi_in_use - check if the default forwarding VSI is being used + * @sw: switch to check if its default forwarding VSI is free + * + * Return true if the default forwarding VSI is already being used, else returns + * false signalling that it's available to use. + */ +bool ice_is_dflt_vsi_in_use(struct ice_sw *sw) +{ +	return (sw->dflt_vsi && sw->dflt_vsi_ena); +} + +/** + * ice_is_vsi_dflt_vsi - check if the VSI passed in is the default VSI + * @sw: switch for the default forwarding VSI to compare against + * @vsi: VSI to compare against default forwarding VSI + * + * If this VSI passed in is the default forwarding VSI then return true, else + * return false + */ +bool ice_is_vsi_dflt_vsi(struct ice_sw *sw, struct ice_vsi *vsi) +{ +	return (sw->dflt_vsi == vsi && sw->dflt_vsi_ena); +} + +/** + * ice_set_dflt_vsi - set the default forwarding VSI + * @sw: switch used to assign the default forwarding VSI + * @vsi: VSI getting set as the default forwarding VSI on the switch + * + * If the VSI passed in is already the default VSI and it's enabled just return + * success. + * + * If there is already a default VSI on the switch and it's enabled then return + * -EEXIST since there can only be one default VSI per switch. + * + *  Otherwise try to set the VSI passed in as the switch's default VSI and + *  return the result. + */ +int ice_set_dflt_vsi(struct ice_sw *sw, struct ice_vsi *vsi) +{ +	enum ice_status status; +	struct device *dev; + +	if (!sw || !vsi) +		return -EINVAL; + +	dev = ice_pf_to_dev(vsi->back); + +	/* the VSI passed in is already the default VSI */ +	if (ice_is_vsi_dflt_vsi(sw, vsi)) { +		dev_dbg(dev, "VSI %d passed in is already the default forwarding VSI, nothing to do\n", +			vsi->vsi_num); +		return 0; +	} + +	/* another VSI is already the default VSI for this switch */ +	if (ice_is_dflt_vsi_in_use(sw)) { +		dev_err(dev, +			"Default forwarding VSI %d already in use, disable it and try again\n", +			sw->dflt_vsi->vsi_num); +		return -EEXIST; +	} + +	status = ice_cfg_dflt_vsi(&vsi->back->hw, vsi->idx, true, ICE_FLTR_RX); +	if (status) { +		dev_err(dev, +			"Failed to set VSI %d as the default forwarding VSI, error %d\n", +			vsi->vsi_num, status); +		return -EIO; +	} + +	sw->dflt_vsi = vsi; +	sw->dflt_vsi_ena = true; + +	return 0; +} + +/** + * ice_clear_dflt_vsi - clear the default forwarding VSI + * @sw: switch used to clear the default VSI + * + * If the switch has no default VSI or it's not enabled then return error. + * + * Otherwise try to clear the default VSI and return the result. + */ +int ice_clear_dflt_vsi(struct ice_sw *sw) +{ +	struct ice_vsi *dflt_vsi; +	enum ice_status status; +	struct device *dev; + +	if (!sw) +		return -EINVAL; + +	dev = ice_pf_to_dev(sw->pf); + +	dflt_vsi = sw->dflt_vsi; + +	/* there is no default VSI configured */ +	if (!ice_is_dflt_vsi_in_use(sw)) +		return -ENODEV; + +	status = ice_cfg_dflt_vsi(&dflt_vsi->back->hw, dflt_vsi->idx, false, +				  ICE_FLTR_RX); +	if (status) { +		dev_err(dev, +			"Failed to clear the default forwarding VSI %d, error %d\n", +			dflt_vsi->vsi_num, status); +		return -EIO; +	} + +	sw->dflt_vsi = NULL; +	sw->dflt_vsi_ena = false; + +	return 0; +} diff --git a/drivers/net/ethernet/intel/ice/ice_lib.h b/drivers/net/ethernet/intel/ice/ice_lib.h index 6e31e30aba39..68fd0d4505c2 100644 --- a/drivers/net/ethernet/intel/ice/ice_lib.h +++ b/drivers/net/ethernet/intel/ice/ice_lib.h @@ -103,4 +103,12 @@ enum ice_status  ice_vsi_cfg_mac_fltr(struct ice_vsi *vsi, const u8 *macaddr, bool set);  bool ice_is_safe_mode(struct ice_pf *pf); + +bool ice_is_dflt_vsi_in_use(struct ice_sw *sw); + +bool ice_is_vsi_dflt_vsi(struct ice_sw *sw, struct ice_vsi *vsi); + +int ice_set_dflt_vsi(struct ice_sw *sw, struct ice_vsi *vsi); + +int ice_clear_dflt_vsi(struct ice_sw *sw);  #endif /* !_ICE_LIB_H_ */ diff --git a/drivers/net/ethernet/intel/ice/ice_main.c b/drivers/net/ethernet/intel/ice/ice_main.c index 69bff085acf7..5ae671609f98 100644 --- a/drivers/net/ethernet/intel/ice/ice_main.c +++ b/drivers/net/ethernet/intel/ice/ice_main.c @@ -13,7 +13,7 @@  #define DRV_VERSION_MAJOR 0  #define DRV_VERSION_MINOR 8 -#define DRV_VERSION_BUILD 1 +#define DRV_VERSION_BUILD 2  #define DRV_VERSION	__stringify(DRV_VERSION_MAJOR) "." \  			__stringify(DRV_VERSION_MINOR) "." \ @@ -379,25 +379,29 @@ static int ice_vsi_sync_fltr(struct ice_vsi *vsi)  		clear_bit(ICE_VSI_FLAG_PROMISC_CHANGED, vsi->flags);  		if (vsi->current_netdev_flags & IFF_PROMISC) {  			/* Apply Rx filter rule to get traffic from wire */ -			status = ice_cfg_dflt_vsi(hw, vsi->idx, true, -						  ICE_FLTR_RX); -			if (status) { -				netdev_err(netdev, "Error setting default VSI %i Rx rule\n", -					   vsi->vsi_num); -				vsi->current_netdev_flags &= ~IFF_PROMISC; -				err = -EIO; -				goto out_promisc; +			if (!ice_is_dflt_vsi_in_use(pf->first_sw)) { +				err = ice_set_dflt_vsi(pf->first_sw, vsi); +				if (err && err != -EEXIST) { +					netdev_err(netdev, +						   "Error %d setting default VSI %i Rx rule\n", +						   err, vsi->vsi_num); +					vsi->current_netdev_flags &= +						~IFF_PROMISC; +					goto out_promisc; +				}  			}  		} else {  			/* Clear Rx filter to remove traffic from wire */ -			status = ice_cfg_dflt_vsi(hw, vsi->idx, false, -						  ICE_FLTR_RX); -			if (status) { -				netdev_err(netdev, "Error clearing default VSI %i Rx rule\n", -					   vsi->vsi_num); -				vsi->current_netdev_flags |= IFF_PROMISC; -				err = -EIO; -				goto out_promisc; +			if (ice_is_vsi_dflt_vsi(pf->first_sw, vsi)) { +				err = ice_clear_dflt_vsi(pf->first_sw); +				if (err) { +					netdev_err(netdev, +						   "Error %d clearing default VSI %i Rx rule\n", +						   err, vsi->vsi_num); +					vsi->current_netdev_flags |= +						IFF_PROMISC; +					goto out_promisc; +				}  			}  		}  	} @@ -472,7 +476,7 @@ ice_prepare_for_reset(struct ice_pf *pf)  		ice_vc_notify_reset(pf);  	/* Disable VFs until reset is completed */ -	for (i = 0; i < pf->num_alloc_vfs; i++) +	ice_for_each_vf(pf, i)  		ice_set_vf_state_qs_dis(&pf->vf[i]);  	/* clear SW filtering DB */ @@ -840,8 +844,7 @@ ice_link_event(struct ice_pf *pf, struct ice_port_info *pi, bool link_up,  	ice_vsi_link_event(vsi, link_up);  	ice_print_link_msg(vsi, link_up); -	if (pf->num_alloc_vfs) -		ice_vc_notify_link_state(pf); +	ice_vc_notify_link_state(pf);  	return result;  } @@ -1291,7 +1294,7 @@ static void ice_handle_mdd_event(struct ice_pf *pf)  	}  	/* check to see if one of the VFs caused the MDD */ -	for (i = 0; i < pf->num_alloc_vfs; i++) { +	ice_for_each_vf(pf, i) {  		struct ice_vf *vf = &pf->vf[i];  		bool vf_mdd_detected = false; @@ -2330,7 +2333,8 @@ static void ice_set_netdev_features(struct net_device *netdev)  			 NETIF_F_HW_VLAN_CTAG_TX     |  			 NETIF_F_HW_VLAN_CTAG_RX; -	tso_features = NETIF_F_TSO; +	tso_features = NETIF_F_TSO		| +		       NETIF_F_GSO_UDP_L4;  	/* set features that user can change */  	netdev->hw_features = dflt_features | csumo_features | @@ -3568,6 +3572,15 @@ static const struct pci_device_id ice_pci_tbl[] = {  	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E810C_BACKPLANE), 0 },  	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E810C_QSFP), 0 },  	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E810C_SFP), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822C_BACKPLANE), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822C_QSFP), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822C_SFP), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822C_10G_BASE_T), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822C_SGMII), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822X_BACKPLANE), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822L_SFP), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822L_10G_BASE_T), 0 }, +	{ PCI_VDEVICE(INTEL, ICE_DEV_ID_E822L_SGMII), 0 },  	/* required last entry */  	{ 0, }  }; @@ -4670,6 +4683,13 @@ static void ice_rebuild(struct ice_pf *pf, enum ice_reset_req reset_type)  		goto err_init_ctrlq;  	} +	if (pf->first_sw->dflt_vsi_ena) +		dev_info(dev, +			 "Clearing default VSI, re-enable after reset completes\n"); +	/* clear the default VSI configuration if it exists */ +	pf->first_sw->dflt_vsi = NULL; +	pf->first_sw->dflt_vsi_ena = false; +  	ice_clear_pxe_mode(hw);  	ret = ice_get_caps(hw); @@ -4825,7 +4845,7 @@ static int ice_change_mtu(struct net_device *netdev, int new_mtu)  		}  	} -	netdev_info(netdev, "changed MTU to %d\n", new_mtu); +	netdev_dbg(netdev, "changed MTU to %d\n", new_mtu);  	return 0;  } @@ -5060,42 +5080,23 @@ ice_bridge_setlink(struct net_device *dev, struct nlmsghdr *nlh,   * ice_tx_timeout - Respond to a Tx Hang   * @netdev: network interface device structure   */ -static void ice_tx_timeout(struct net_device *netdev) +static void ice_tx_timeout(struct net_device *netdev, unsigned int txqueue)  {  	struct ice_netdev_priv *np = netdev_priv(netdev);  	struct ice_ring *tx_ring = NULL;  	struct ice_vsi *vsi = np->vsi;  	struct ice_pf *pf = vsi->back; -	int hung_queue = -1;  	u32 i;  	pf->tx_timeout_count++; -	/* find the stopped queue the same way dev_watchdog() does */ -	for (i = 0; i < netdev->num_tx_queues; i++) { -		unsigned long trans_start; -		struct netdev_queue *q; - -		q = netdev_get_tx_queue(netdev, i); -		trans_start = q->trans_start; -		if (netif_xmit_stopped(q) && -		    time_after(jiffies, -			       trans_start + netdev->watchdog_timeo)) { -			hung_queue = i; -			break; -		} -	} - -	if (i == netdev->num_tx_queues) -		netdev_info(netdev, "tx_timeout: no netdev hung queue found\n"); -	else -		/* now that we have an index, find the tx_ring struct */ -		for (i = 0; i < vsi->num_txq; i++) -			if (vsi->tx_rings[i] && vsi->tx_rings[i]->desc) -				if (hung_queue == vsi->tx_rings[i]->q_index) { -					tx_ring = vsi->tx_rings[i]; -					break; -				} +	/* now that we have an index, find the tx_ring struct */ +	for (i = 0; i < vsi->num_txq; i++) +		if (vsi->tx_rings[i] && vsi->tx_rings[i]->desc) +			if (txqueue == vsi->tx_rings[i]->q_index) { +				tx_ring = vsi->tx_rings[i]; +				break; +			}  	/* Reset recovery level if enough time has elapsed after last timeout.  	 * Also ensure no new reset action happens before next timeout period. @@ -5110,19 +5111,19 @@ static void ice_tx_timeout(struct net_device *netdev)  		struct ice_hw *hw = &pf->hw;  		u32 head, val = 0; -		head = (rd32(hw, QTX_COMM_HEAD(vsi->txq_map[hung_queue])) & +		head = (rd32(hw, QTX_COMM_HEAD(vsi->txq_map[txqueue])) &  			QTX_COMM_HEAD_HEAD_M) >> QTX_COMM_HEAD_HEAD_S;  		/* Read interrupt register */  		val = rd32(hw, GLINT_DYN_CTL(tx_ring->q_vector->reg_idx));  		netdev_info(netdev, "tx_timeout: VSI_num: %d, Q %d, NTC: 0x%x, HW_HEAD: 0x%x, NTU: 0x%x, INT: 0x%x\n", -			    vsi->vsi_num, hung_queue, tx_ring->next_to_clean, +			    vsi->vsi_num, txqueue, tx_ring->next_to_clean,  			    head, tx_ring->next_to_use, val);  	}  	pf->tx_timeout_last_recovery = jiffies; -	netdev_info(netdev, "tx_timeout recovery level %d, hung_queue %d\n", -		    pf->tx_timeout_recovery_level, hung_queue); +	netdev_info(netdev, "tx_timeout recovery level %d, txqueue %d\n", +		    pf->tx_timeout_recovery_level, txqueue);  	switch (pf->tx_timeout_recovery_level) {  	case 1: diff --git a/drivers/net/ethernet/intel/ice/ice_nvm.c b/drivers/net/ethernet/intel/ice/ice_nvm.c index 57c73f613f32..7525ac50742e 100644 --- a/drivers/net/ethernet/intel/ice/ice_nvm.c +++ b/drivers/net/ethernet/intel/ice/ice_nvm.c @@ -289,6 +289,18 @@ enum ice_status ice_init_nvm(struct ice_hw *hw)  	nvm->eetrack = (eetrack_hi << 16) | eetrack_lo; +	/* the following devices do not have boot_cfg_tlv yet */ +	if (hw->device_id == ICE_DEV_ID_E822C_BACKPLANE || +	    hw->device_id == ICE_DEV_ID_E822C_QSFP || +	    hw->device_id == ICE_DEV_ID_E822C_10G_BASE_T || +	    hw->device_id == ICE_DEV_ID_E822C_SGMII || +	    hw->device_id == ICE_DEV_ID_E822C_SFP || +	    hw->device_id == ICE_DEV_ID_E822X_BACKPLANE || +	    hw->device_id == ICE_DEV_ID_E822L_SFP || +	    hw->device_id == ICE_DEV_ID_E822L_10G_BASE_T || +	    hw->device_id == ICE_DEV_ID_E822L_SGMII) +		return status; +  	status = ice_get_pfa_module_tlv(hw, &boot_cfg_tlv, &boot_cfg_tlv_len,  					ICE_SR_BOOT_CFG_PTR);  	if (status) { diff --git a/drivers/net/ethernet/intel/ice/ice_protocol_type.h b/drivers/net/ethernet/intel/ice/ice_protocol_type.h new file mode 100644 index 000000000000..71647566964e --- /dev/null +++ b/drivers/net/ethernet/intel/ice/ice_protocol_type.h @@ -0,0 +1,25 @@ +/* SPDX-License-Identifier: GPL-2.0 */ +/* Copyright (c) 2019, Intel Corporation. */ + +#ifndef _ICE_PROTOCOL_TYPE_H_ +#define _ICE_PROTOCOL_TYPE_H_ +/* Decoders for ice_prot_id: + * - F: First + * - I: Inner + * - L: Last + * - O: Outer + * - S: Single + */ +enum ice_prot_id { +	ICE_PROT_ID_INVAL	= 0, +	ICE_PROT_IPV4_OF_OR_S	= 32, +	ICE_PROT_IPV4_IL	= 33, +	ICE_PROT_IPV6_OF_OR_S	= 40, +	ICE_PROT_IPV6_IL	= 41, +	ICE_PROT_TCP_IL		= 49, +	ICE_PROT_UDP_IL_OR_S	= 53, +	ICE_PROT_SCTP_IL	= 96, +	ICE_PROT_META_ID	= 255, /* when offset == metadata */ +	ICE_PROT_INVALID	= 255  /* when offset == ICE_FV_OFFSET_INVAL */ +}; +#endif /* _ICE_PROTOCOL_TYPE_H_ */ diff --git a/drivers/net/ethernet/intel/ice/ice_status.h b/drivers/net/ethernet/intel/ice/ice_status.h index c01597885629..a9a8bc3aca42 100644 --- a/drivers/net/ethernet/intel/ice/ice_status.h +++ b/drivers/net/ethernet/intel/ice/ice_status.h @@ -26,6 +26,7 @@ enum ice_status {  	ICE_ERR_IN_USE				= -16,  	ICE_ERR_MAX_LIMIT			= -17,  	ICE_ERR_RESET_ONGOING			= -18, +	ICE_ERR_HW_TABLE			= -19,  	ICE_ERR_NVM_CHECKSUM			= -51,  	ICE_ERR_BUF_TOO_SHORT			= -52,  	ICE_ERR_NVM_BLANK_MODE			= -53, diff --git a/drivers/net/ethernet/intel/ice/ice_switch.c b/drivers/net/ethernet/intel/ice/ice_switch.c index b5a53f862a83..431266081a80 100644 --- a/drivers/net/ethernet/intel/ice/ice_switch.c +++ b/drivers/net/ethernet/intel/ice/ice_switch.c @@ -50,42 +50,6 @@ static const u8 dummy_eth_header[DUMMY_ETH_HDR_LEN] = { 0x2, 0, 0, 0, 0, 0,  	 ((n) * sizeof(((struct ice_sw_rule_vsi_list *)0)->vsi)))  /** - * ice_aq_alloc_free_res - command to allocate/free resources - * @hw: pointer to the HW struct - * @num_entries: number of resource entries in buffer - * @buf: Indirect buffer to hold data parameters and response - * @buf_size: size of buffer for indirect commands - * @opc: pass in the command opcode - * @cd: pointer to command details structure or NULL - * - * Helper function to allocate/free resources using the admin queue commands - */ -static enum ice_status -ice_aq_alloc_free_res(struct ice_hw *hw, u16 num_entries, -		      struct ice_aqc_alloc_free_res_elem *buf, u16 buf_size, -		      enum ice_adminq_opc opc, struct ice_sq_cd *cd) -{ -	struct ice_aqc_alloc_free_res_cmd *cmd; -	struct ice_aq_desc desc; - -	cmd = &desc.params.sw_res_ctrl; - -	if (!buf) -		return ICE_ERR_PARAM; - -	if (buf_size < (num_entries * sizeof(buf->elem[0]))) -		return ICE_ERR_PARAM; - -	ice_fill_dflt_direct_cmd_desc(&desc, opc); - -	desc.flags |= cpu_to_le16(ICE_AQ_FLAG_RD); - -	cmd->num_entries = cpu_to_le16(num_entries); - -	return ice_aq_send_cmd(hw, &desc, buf, buf_size, cd); -} - -/**   * ice_init_def_sw_recp - initialize the recipe book keeping tables   * @hw: pointer to the HW struct   * diff --git a/drivers/net/ethernet/intel/ice/ice_txrx.c b/drivers/net/ethernet/intel/ice/ice_txrx.c index 2c212f64d99f..fd17ace6b226 100644 --- a/drivers/net/ethernet/intel/ice/ice_txrx.c +++ b/drivers/net/ethernet/intel/ice/ice_txrx.c @@ -1071,13 +1071,16 @@ static int ice_clean_rx_irq(struct ice_ring *rx_ring, int budget)  		ice_put_rx_buf(rx_ring, rx_buf);  		continue;  construct_skb: -		if (skb) +		if (skb) {  			ice_add_rx_frag(rx_ring, rx_buf, skb, size); -		else if (ice_ring_uses_build_skb(rx_ring)) -			skb = ice_build_skb(rx_ring, rx_buf, &xdp); -		else +		} else if (likely(xdp.data)) { +			if (ice_ring_uses_build_skb(rx_ring)) +				skb = ice_build_skb(rx_ring, rx_buf, &xdp); +			else +				skb = ice_construct_skb(rx_ring, rx_buf, &xdp); +		} else {  			skb = ice_construct_skb(rx_ring, rx_buf, &xdp); - +		}  		/* exit if we failed to retrieve a buffer */  		if (!skb) {  			rx_ring->rx_stats.alloc_buf_failed++; @@ -1925,6 +1928,7 @@ int ice_tso(struct ice_tx_buf *first, struct ice_tx_offload_params *off)  	} ip;  	union {  		struct tcphdr *tcp; +		struct udphdr *udp;  		unsigned char *hdr;  	} l4;  	u64 cd_mss, cd_tso_len; @@ -1958,10 +1962,18 @@ int ice_tso(struct ice_tx_buf *first, struct ice_tx_offload_params *off)  	/* remove payload length from checksum */  	paylen = skb->len - l4_start; -	csum_replace_by_diff(&l4.tcp->check, (__force __wsum)htonl(paylen)); -	/* compute length of segmentation header */ -	off->header_len = (l4.tcp->doff * 4) + l4_start; +	if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_L4) { +		csum_replace_by_diff(&l4.udp->check, +				     (__force __wsum)htonl(paylen)); +		/* compute length of UDP segmentation header */ +		off->header_len = sizeof(l4.udp) + l4_start; +	} else { +		csum_replace_by_diff(&l4.tcp->check, +				     (__force __wsum)htonl(paylen)); +		/* compute length of TCP segmentation header */ +		off->header_len = (l4.tcp->doff * 4) + l4_start; +	}  	/* update gso_segs and bytecount */  	first->gso_segs = skb_shinfo(skb)->gso_segs; diff --git a/drivers/net/ethernet/intel/ice/ice_txrx.h b/drivers/net/ethernet/intel/ice/ice_txrx.h index a84cc0e6dd27..a86270696df1 100644 --- a/drivers/net/ethernet/intel/ice/ice_txrx.h +++ b/drivers/net/ethernet/intel/ice/ice_txrx.h @@ -341,6 +341,12 @@ struct ice_ring_container {  	u16 itr_setting;  }; +struct ice_coalesce_stored { +	u16 itr_tx; +	u16 itr_rx; +	u8 intrl; +}; +  /* iterator for handling rings in ring container */  #define ice_for_each_ring(pos, head) \  	for (pos = (head).ring; pos; pos = pos->next) diff --git a/drivers/net/ethernet/intel/ice/ice_type.h b/drivers/net/ethernet/intel/ice/ice_type.h index c4854a987130..b361ffabb0ca 100644 --- a/drivers/net/ethernet/intel/ice/ice_type.h +++ b/drivers/net/ethernet/intel/ice/ice_type.h @@ -13,6 +13,7 @@  #include "ice_controlq.h"  #include "ice_lan_tx_rx.h"  #include "ice_flex_type.h" +#include "ice_protocol_type.h"  static inline bool ice_is_tc_ena(unsigned long bitmap, u8 tc)  { @@ -41,6 +42,7 @@ static inline u32 ice_round_to_num(u32 N, u32 R)  #define ICE_DBG_QCTX		BIT_ULL(6)  #define ICE_DBG_NVM		BIT_ULL(7)  #define ICE_DBG_LAN		BIT_ULL(8) +#define ICE_DBG_FLOW		BIT_ULL(9)  #define ICE_DBG_SW		BIT_ULL(13)  #define ICE_DBG_SCHED		BIT_ULL(14)  #define ICE_DBG_PKG		BIT_ULL(16) @@ -559,6 +561,10 @@ struct ice_hw {  	/* HW block tables */  	struct ice_blk_info blk[ICE_BLK_COUNT]; +	struct mutex fl_profs_locks[ICE_BLK_COUNT];	/* lock fltr profiles */ +	struct list_head fl_profs[ICE_BLK_COUNT]; +	struct mutex rss_locks;	/* protect RSS configuration */ +	struct list_head rss_list_head;  };  /* Statistics collected by each port, VSI, VEB, and S-channel */ diff --git a/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.c b/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.c index edb374296d1f..82b1e7a4cb92 100644 --- a/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.c +++ b/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.c @@ -35,37 +35,6 @@ static int ice_check_vf_init(struct ice_pf *pf, struct ice_vf *vf)  }  /** - * ice_err_to_virt err - translate errors for VF return code - * @ice_err: error return code - */ -static enum virtchnl_status_code ice_err_to_virt_err(enum ice_status ice_err) -{ -	switch (ice_err) { -	case ICE_SUCCESS: -		return VIRTCHNL_STATUS_SUCCESS; -	case ICE_ERR_BAD_PTR: -	case ICE_ERR_INVAL_SIZE: -	case ICE_ERR_DEVICE_NOT_SUPPORTED: -	case ICE_ERR_PARAM: -	case ICE_ERR_CFG: -		return VIRTCHNL_STATUS_ERR_PARAM; -	case ICE_ERR_NO_MEMORY: -		return VIRTCHNL_STATUS_ERR_NO_MEMORY; -	case ICE_ERR_NOT_READY: -	case ICE_ERR_RESET_FAILED: -	case ICE_ERR_FW_API_VER: -	case ICE_ERR_AQ_ERROR: -	case ICE_ERR_AQ_TIMEOUT: -	case ICE_ERR_AQ_FULL: -	case ICE_ERR_AQ_NO_WORK: -	case ICE_ERR_AQ_EMPTY: -		return VIRTCHNL_STATUS_ERR_ADMIN_QUEUE_ERROR; -	default: -		return VIRTCHNL_STATUS_ERR_NOT_SUPPORTED; -	} -} - -/**   * ice_vc_vf_broadcast - Broadcast a message to all VFs on PF   * @pf: pointer to the PF structure   * @v_opcode: operation code @@ -78,10 +47,11 @@ ice_vc_vf_broadcast(struct ice_pf *pf, enum virtchnl_ops v_opcode,  		    enum virtchnl_status_code v_retval, u8 *msg, u16 msglen)  {  	struct ice_hw *hw = &pf->hw; -	struct ice_vf *vf = pf->vf;  	int i; -	for (i = 0; i < pf->num_alloc_vfs; i++, vf++) { +	ice_for_each_vf(pf, i) { +		struct ice_vf *vf = &pf->vf[i]; +  		/* Not all vfs are enabled so skip the ones that are not */  		if (!test_bit(ICE_VF_STATE_INIT, vf->vf_states) &&  		    !test_bit(ICE_VF_STATE_ACTIVE, vf->vf_states)) @@ -121,26 +91,6 @@ ice_set_pfe_link(struct ice_vf *vf, struct virtchnl_pf_event *pfe,  }  /** - * ice_set_pfe_link_forced - Force the virtchnl_pf_event link speed/status - * @vf: pointer to the VF structure - * @pfe: pointer to the virtchnl_pf_event to set link speed/status for - * @link_up: whether or not to set the link up/down - */ -static void -ice_set_pfe_link_forced(struct ice_vf *vf, struct virtchnl_pf_event *pfe, -			bool link_up) -{ -	u16 link_speed; - -	if (link_up) -		link_speed = ICE_AQ_LINK_SPEED_100GB; -	else -		link_speed = ICE_AQ_LINK_SPEED_UNKNOWN; - -	ice_set_pfe_link(vf, pfe, link_speed, link_up); -} - -/**   * ice_vc_notify_vf_link_state - Inform a VF of link status   * @vf: pointer to the VF structure   * @@ -160,13 +110,17 @@ static void ice_vc_notify_vf_link_state(struct ice_vf *vf)  	pfe.severity = PF_EVENT_SEVERITY_INFO;  	/* Always report link is down if the VF queues aren't enabled */ -	if (!vf->num_qs_ena) +	if (!vf->num_qs_ena) {  		ice_set_pfe_link(vf, &pfe, ICE_AQ_LINK_SPEED_UNKNOWN, false); -	else if (vf->link_forced) -		ice_set_pfe_link_forced(vf, &pfe, vf->link_up); -	else -		ice_set_pfe_link(vf, &pfe, ls->link_speed, ls->link_info & -				 ICE_AQ_LINK_UP); +	} else if (vf->link_forced) { +		u16 link_speed = vf->link_up ? +			ls->link_speed : ICE_AQ_LINK_SPEED_UNKNOWN; + +		ice_set_pfe_link(vf, &pfe, link_speed, vf->link_up); +	} else { +		ice_set_pfe_link(vf, &pfe, ls->link_speed, +				 ls->link_info & ICE_AQ_LINK_UP); +	}  	ice_aq_send_msg_to_vf(hw, vf->vf_id, VIRTCHNL_OP_EVENT,  			      VIRTCHNL_STATUS_SUCCESS, (u8 *)&pfe, @@ -331,7 +285,7 @@ void ice_free_vfs(struct ice_pf *pf)  		usleep_range(1000, 2000);  	/* Avoid wait time by stopping all VFs at the same time */ -	for (i = 0; i < pf->num_alloc_vfs; i++) +	ice_for_each_vf(pf, i)  		if (test_bit(ICE_VF_STATE_QS_ENA, pf->vf[i].vf_states))  			ice_dis_vf_qs(&pf->vf[i]); @@ -991,10 +945,17 @@ static void ice_cleanup_and_realloc_vf(struct ice_vf *vf)  	/* reallocate VF resources to finish resetting the VSI state */  	if (!ice_alloc_vf_res(vf)) { +		struct ice_vsi *vsi; +  		ice_ena_vf_mappings(vf);  		set_bit(ICE_VF_STATE_ACTIVE, vf->vf_states);  		clear_bit(ICE_VF_STATE_DIS, vf->vf_states); -		vf->num_vlan = 0; + +		vsi = pf->vsi[vf->lan_vsi_idx]; +		if (ice_vsi_add_vlan(vsi, 0)) +			dev_warn(ice_pf_to_dev(pf), +				 "Failed to add VLAN 0 filter for VF %d, MDD events will trigger. Reset the VF, disable spoofchk, or enable 8021q module on the guest", +				 vf->vf_id);  	}  	/* Tell the VF driver the reset is done. This needs to be done only @@ -1023,7 +984,7 @@ ice_vf_set_vsi_promisc(struct ice_vf *vf, struct ice_vsi *vsi, u8 promisc_m,  	struct ice_hw *hw;  	hw = &pf->hw; -	if (vf->num_vlan) { +	if (vsi->num_vlan) {  		status = ice_set_vlan_vsi_promisc(hw, vsi->idx, promisc_m,  						  rm_promisc);  	} else if (vf->port_vlan_id) { @@ -1070,7 +1031,7 @@ static bool ice_config_res_vfs(struct ice_pf *pf)  		ice_irq_dynamic_ena(hw, NULL, NULL);  	/* Finish resetting each VF and allocate resources */ -	for (v = 0; v < pf->num_alloc_vfs; v++) { +	ice_for_each_vf(pf, v) {  		struct ice_vf *vf = &pf->vf[v];  		vf->num_vf_qs = pf->num_vf_qps; @@ -1113,10 +1074,10 @@ bool ice_reset_all_vfs(struct ice_pf *pf, bool is_vflr)  		return false;  	/* Begin reset on all VFs at once */ -	for (v = 0; v < pf->num_alloc_vfs; v++) +	ice_for_each_vf(pf, v)  		ice_trigger_vf_reset(&pf->vf[v], is_vflr, true); -	for (v = 0; v < pf->num_alloc_vfs; v++) { +	ice_for_each_vf(pf, v) {  		struct ice_vsi *vsi;  		vf = &pf->vf[v]; @@ -1161,7 +1122,7 @@ bool ice_reset_all_vfs(struct ice_pf *pf, bool is_vflr)  		dev_warn(dev, "VF reset check timeout\n");  	/* free VF resources to begin resetting the VSI state */ -	for (v = 0; v < pf->num_alloc_vfs; v++) { +	ice_for_each_vf(pf, v) {  		vf = &pf->vf[v];  		ice_free_vf_res(vf); @@ -1273,7 +1234,7 @@ static bool ice_reset_vf(struct ice_vf *vf, bool is_vflr)  	 */  	if (test_bit(ICE_VF_STATE_UC_PROMISC, vf->vf_states) ||  	    test_bit(ICE_VF_STATE_MC_PROMISC, vf->vf_states)) { -		if (vf->port_vlan_id ||  vf->num_vlan) +		if (vf->port_vlan_id || vsi->num_vlan)  			promisc_m = ICE_UCAST_VLAN_PROMISC_BITS;  		else  			promisc_m = ICE_UCAST_PROMISC_BITS; @@ -1301,7 +1262,7 @@ void ice_vc_notify_link_state(struct ice_pf *pf)  {  	int i; -	for (i = 0; i < pf->num_alloc_vfs; i++) +	ice_for_each_vf(pf, i)  		ice_vc_notify_vf_link_state(&pf->vf[i]);  } @@ -1385,9 +1346,10 @@ static int ice_alloc_vfs(struct ice_pf *pf, u16 num_alloc_vfs)  		goto err_pci_disable_sriov;  	}  	pf->vf = vfs; +	pf->num_alloc_vfs = num_alloc_vfs;  	/* apply default profile */ -	for (i = 0; i < num_alloc_vfs; i++) { +	ice_for_each_vf(pf, i) {  		vfs[i].pf = pf;  		vfs[i].vf_sw_id = pf->first_sw;  		vfs[i].vf_id = i; @@ -1396,7 +1358,6 @@ static int ice_alloc_vfs(struct ice_pf *pf, u16 num_alloc_vfs)  		set_bit(ICE_VIRTCHNL_VF_CAP_L2, &vfs[i].vf_caps);  		vfs[i].spoofchk = true;  	} -	pf->num_alloc_vfs = num_alloc_vfs;  	/* VF resources get allocated with initialization */  	if (!ice_config_res_vfs(pf)) { @@ -1535,7 +1496,7 @@ void ice_process_vflr_event(struct ice_pf *pf)  	    !pf->num_alloc_vfs)  		return; -	for (vf_id = 0; vf_id < pf->num_alloc_vfs; vf_id++) { +	ice_for_each_vf(pf, vf_id) {  		struct ice_vf *vf = &pf->vf[vf_id];  		u32 reg_idx, bit_idx; @@ -1918,6 +1879,89 @@ error_param:  }  /** + * ice_set_vf_spoofchk + * @netdev: network interface device structure + * @vf_id: VF identifier + * @ena: flag to enable or disable feature + * + * Enable or disable VF spoof checking + */ +int ice_set_vf_spoofchk(struct net_device *netdev, int vf_id, bool ena) +{ +	struct ice_netdev_priv *np = netdev_priv(netdev); +	struct ice_pf *pf = np->vsi->back; +	struct ice_vsi_ctx *ctx; +	struct ice_vsi *vf_vsi; +	enum ice_status status; +	struct device *dev; +	struct ice_vf *vf; +	int ret = 0; + +	dev = ice_pf_to_dev(pf); +	if (ice_validate_vf_id(pf, vf_id)) +		return -EINVAL; + +	vf = &pf->vf[vf_id]; + +	if (ice_check_vf_init(pf, vf)) +		return -EBUSY; + +	vf_vsi = pf->vsi[vf->lan_vsi_idx]; +	if (!vf_vsi) { +		netdev_err(netdev, "VSI %d for VF %d is null\n", +			   vf->lan_vsi_idx, vf->vf_id); +		return -EINVAL; +	} + +	if (vf_vsi->type != ICE_VSI_VF) { +		netdev_err(netdev, +			   "Type %d of VSI %d for VF %d is no ICE_VSI_VF\n", +			   vf_vsi->type, vf_vsi->vsi_num, vf->vf_id); +		return -ENODEV; +	} + +	if (ena == vf->spoofchk) { +		dev_dbg(dev, "VF spoofchk already %s\n", ena ? "ON" : "OFF"); +		return 0; +	} + +	ctx = kzalloc(sizeof(*ctx), GFP_KERNEL); +	if (!ctx) +		return -ENOMEM; + +	ctx->info.sec_flags = vf_vsi->info.sec_flags; +	ctx->info.valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_SECURITY_VALID); +	if (ena) { +		ctx->info.sec_flags |= +			ICE_AQ_VSI_SEC_FLAG_ENA_MAC_ANTI_SPOOF | +			(ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA << +			 ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S); +	} else { +		ctx->info.sec_flags &= +			~(ICE_AQ_VSI_SEC_FLAG_ENA_MAC_ANTI_SPOOF | +			  (ICE_AQ_VSI_SEC_TX_VLAN_PRUNE_ENA << +			   ICE_AQ_VSI_SEC_TX_PRUNE_ENA_S)); +	} + +	status = ice_update_vsi(&pf->hw, vf_vsi->idx, ctx, NULL); +	if (status) { +		dev_err(dev, +			"Failed to %sable spoofchk on VF %d VSI %d\n error %d", +			ena ? "en" : "dis", vf->vf_id, vf_vsi->vsi_num, status); +		ret = -EIO; +		goto out; +	} + +	/* only update spoofchk state and VSI context on success */ +	vf_vsi->info.sec_flags = ctx->info.sec_flags; +	vf->spoofchk = ena; + +out: +	kfree(ctx); +	return ret; +} + +/**   * ice_vc_get_stats_msg   * @vf: pointer to the VF info   * @msg: pointer to the msg buffer @@ -2409,6 +2453,83 @@ static bool ice_can_vf_change_mac(struct ice_vf *vf)  }  /** + * ice_vc_add_mac_addr - attempt to add the MAC address passed in + * @vf: pointer to the VF info + * @vsi: pointer to the VF's VSI + * @mac_addr: MAC address to add + */ +static int +ice_vc_add_mac_addr(struct ice_vf *vf, struct ice_vsi *vsi, u8 *mac_addr) +{ +	struct device *dev = ice_pf_to_dev(vf->pf); +	enum ice_status status; + +	/* default unicast MAC already added */ +	if (ether_addr_equal(mac_addr, vf->dflt_lan_addr.addr)) +		return 0; + +	if (is_unicast_ether_addr(mac_addr) && !ice_can_vf_change_mac(vf)) { +		dev_err(dev, "VF attempting to override administratively set MAC address, bring down and up the VF interface to resume normal operation\n"); +		return -EPERM; +	} + +	status = ice_vsi_cfg_mac_fltr(vsi, mac_addr, true); +	if (status == ICE_ERR_ALREADY_EXISTS) { +		dev_err(dev, "MAC %pM already exists for VF %d\n", mac_addr, +			vf->vf_id); +		return -EEXIST; +	} else if (status) { +		dev_err(dev, "Failed to add MAC %pM for VF %d\n, error %d\n", +			mac_addr, vf->vf_id, status); +		return -EIO; +	} + +	/* only set dflt_lan_addr once */ +	if (is_zero_ether_addr(vf->dflt_lan_addr.addr) && +	    is_unicast_ether_addr(mac_addr)) +		ether_addr_copy(vf->dflt_lan_addr.addr, mac_addr); + +	vf->num_mac++; + +	return 0; +} + +/** + * ice_vc_del_mac_addr - attempt to delete the MAC address passed in + * @vf: pointer to the VF info + * @vsi: pointer to the VF's VSI + * @mac_addr: MAC address to delete + */ +static int +ice_vc_del_mac_addr(struct ice_vf *vf, struct ice_vsi *vsi, u8 *mac_addr) +{ +	struct device *dev = ice_pf_to_dev(vf->pf); +	enum ice_status status; + +	if (!ice_can_vf_change_mac(vf) && +	    ether_addr_equal(mac_addr, vf->dflt_lan_addr.addr)) +		return 0; + +	status = ice_vsi_cfg_mac_fltr(vsi, mac_addr, false); +	if (status == ICE_ERR_DOES_NOT_EXIST) { +		dev_err(dev, "MAC %pM does not exist for VF %d\n", mac_addr, +			vf->vf_id); +		return -ENOENT; +	} else if (status) { +		dev_err(dev, "Failed to delete MAC %pM for VF %d, error %d\n", +			mac_addr, vf->vf_id, status); +		return -EIO; +	} + +	if (ether_addr_equal(mac_addr, vf->dflt_lan_addr.addr)) +		eth_zero_addr(vf->dflt_lan_addr.addr); + +	vf->num_mac--; + +	return 0; +} + +/**   * ice_vc_handle_mac_addr_msg   * @vf: pointer to the VF info   * @msg: pointer to the msg buffer @@ -2419,23 +2540,23 @@ static bool ice_can_vf_change_mac(struct ice_vf *vf)  static int  ice_vc_handle_mac_addr_msg(struct ice_vf *vf, u8 *msg, bool set)  { +	int (*ice_vc_cfg_mac) +		(struct ice_vf *vf, struct ice_vsi *vsi, u8 *mac_addr);  	enum virtchnl_status_code v_ret = VIRTCHNL_STATUS_SUCCESS;  	struct virtchnl_ether_addr_list *al =  	    (struct virtchnl_ether_addr_list *)msg;  	struct ice_pf *pf = vf->pf;  	enum virtchnl_ops vc_op; -	enum ice_status status;  	struct ice_vsi *vsi; -	struct device *dev; -	int mac_count = 0;  	int i; -	dev = ice_pf_to_dev(pf); - -	if (set) +	if (set) {  		vc_op = VIRTCHNL_OP_ADD_ETH_ADDR; -	else +		ice_vc_cfg_mac = ice_vc_add_mac_addr; +	} else {  		vc_op = VIRTCHNL_OP_DEL_ETH_ADDR; +		ice_vc_cfg_mac = ice_vc_del_mac_addr; +	}  	if (!test_bit(ICE_VF_STATE_ACTIVE, vf->vf_states) ||  	    !ice_vc_isvalid_vsi_id(vf, al->vsi_id)) { @@ -2443,14 +2564,15 @@ ice_vc_handle_mac_addr_msg(struct ice_vf *vf, u8 *msg, bool set)  		goto handle_mac_exit;  	} +	/* If this VF is not privileged, then we can't add more than a +	 * limited number of addresses. Check to make sure that the +	 * additions do not push us over the limit. +	 */  	if (set && !ice_is_vf_trusted(vf) &&  	    (vf->num_mac + al->num_elements) > ICE_MAX_MACADDR_PER_VF) { -		dev_err(dev, +		dev_err(ice_pf_to_dev(pf),  			"Can't add more MAC addresses, because VF-%d is not trusted, switch the VF to trusted mode in order to add more functionalities\n",  			vf->vf_id); -		/* There is no need to let VF know about not being trusted -		 * to add more MAC addr, so we can just return success message. -		 */  		v_ret = VIRTCHNL_STATUS_ERR_PARAM;  		goto handle_mac_exit;  	} @@ -2462,70 +2584,22 @@ ice_vc_handle_mac_addr_msg(struct ice_vf *vf, u8 *msg, bool set)  	}  	for (i = 0; i < al->num_elements; i++) { -		u8 *maddr = al->list[i].addr; +		u8 *mac_addr = al->list[i].addr; +		int result; -		if (ether_addr_equal(maddr, vf->dflt_lan_addr.addr) || -		    is_broadcast_ether_addr(maddr)) { -			if (set) { -				/* VF is trying to add filters that the PF -				 * already added. Just continue. -				 */ -				dev_info(dev, -					 "MAC %pM already set for VF %d\n", -					 maddr, vf->vf_id); -				continue; -			} else { -				/* VF can't remove dflt_lan_addr/bcast MAC */ -				dev_err(dev, -					"VF can't remove default MAC address or MAC %pM programmed by PF for VF %d\n", -					maddr, vf->vf_id); -				continue; -			} -		} - -		/* check for the invalid cases and bail if necessary */ -		if (is_zero_ether_addr(maddr)) { -			dev_err(dev, -				"invalid MAC %pM provided for VF %d\n", -				maddr, vf->vf_id); -			v_ret = VIRTCHNL_STATUS_ERR_PARAM; -			goto handle_mac_exit; -		} - -		if (is_unicast_ether_addr(maddr) && -		    !ice_can_vf_change_mac(vf)) { -			dev_err(dev, -				"can't change unicast MAC for untrusted VF %d\n", -				vf->vf_id); -			v_ret = VIRTCHNL_STATUS_ERR_PARAM; -			goto handle_mac_exit; -		} +		if (is_broadcast_ether_addr(mac_addr) || +		    is_zero_ether_addr(mac_addr)) +			continue; -		/* program the updated filter list */ -		status = ice_vsi_cfg_mac_fltr(vsi, maddr, set); -		if (status == ICE_ERR_DOES_NOT_EXIST || -		    status == ICE_ERR_ALREADY_EXISTS) { -			dev_info(dev, -				 "can't %s MAC filters %pM for VF %d, error %d\n", -				 set ? "add" : "remove", maddr, vf->vf_id, -				 status); -		} else if (status) { -			dev_err(dev, -				"can't %s MAC filters for VF %d, error %d\n", -				set ? "add" : "remove", vf->vf_id, status); -			v_ret = ice_err_to_virt_err(status); +		result = ice_vc_cfg_mac(vf, vsi, mac_addr); +		if (result == -EEXIST || result == -ENOENT) { +			continue; +		} else if (result) { +			v_ret = VIRTCHNL_STATUS_ERR_ADMIN_QUEUE_ERROR;  			goto handle_mac_exit;  		} - -		mac_count++;  	} -	/* Track number of MAC filters programmed for the VF VSI */ -	if (set) -		vf->num_mac += mac_count; -	else -		vf->num_mac -= mac_count; -  handle_mac_exit:  	/* send the response to the VF */  	return ice_vc_send_msg_to_vf(vf, vc_op, v_ret, NULL, 0); @@ -2744,17 +2818,6 @@ static int ice_vc_process_vlan_msg(struct ice_vf *vf, u8 *msg, bool add_v)  		goto error_param;  	} -	if (add_v && !ice_is_vf_trusted(vf) && -	    vf->num_vlan >= ICE_MAX_VLAN_PER_VF) { -		dev_info(dev, -			 "VF-%d is not trusted, switch the VF to trusted mode, in order to add more VLAN addresses\n", -			 vf->vf_id); -		/* There is no need to let VF know about being not trusted, -		 * so we can just return success message here -		 */ -		goto error_param; -	} -  	for (i = 0; i < vfl->num_elements; i++) {  		if (vfl->vlan_id[i] > ICE_MAX_VLANID) {  			v_ret = VIRTCHNL_STATUS_ERR_PARAM; @@ -2771,6 +2834,17 @@ static int ice_vc_process_vlan_msg(struct ice_vf *vf, u8 *msg, bool add_v)  		goto error_param;  	} +	if (add_v && !ice_is_vf_trusted(vf) && +	    vsi->num_vlan >= ICE_MAX_VLAN_PER_VF) { +		dev_info(dev, +			 "VF-%d is not trusted, switch the VF to trusted mode, in order to add more VLAN addresses\n", +			 vf->vf_id); +		/* There is no need to let VF know about being not trusted, +		 * so we can just return success message here +		 */ +		goto error_param; +	} +  	if (vsi->info.pvid) {  		v_ret = VIRTCHNL_STATUS_ERR_PARAM;  		goto error_param; @@ -2785,7 +2859,7 @@ static int ice_vc_process_vlan_msg(struct ice_vf *vf, u8 *msg, bool add_v)  			u16 vid = vfl->vlan_id[i];  			if (!ice_is_vf_trusted(vf) && -			    vf->num_vlan >= ICE_MAX_VLAN_PER_VF) { +			    vsi->num_vlan >= ICE_MAX_VLAN_PER_VF) {  				dev_info(dev,  					 "VF-%d is not trusted, switch the VF to trusted mode, in order to add more VLAN addresses\n",  					 vf->vf_id); @@ -2796,12 +2870,20 @@ static int ice_vc_process_vlan_msg(struct ice_vf *vf, u8 *msg, bool add_v)  				goto error_param;  			} -			if (ice_vsi_add_vlan(vsi, vid)) { +			/* we add VLAN 0 by default for each VF so we can enable +			 * Tx VLAN anti-spoof without triggering MDD events so +			 * we don't need to add it again here +			 */ +			if (!vid) +				continue; + +			status = ice_vsi_add_vlan(vsi, vid); +			if (status) {  				v_ret = VIRTCHNL_STATUS_ERR_PARAM;  				goto error_param;  			} -			vf->num_vlan++; +			vsi->num_vlan++;  			/* Enable VLAN pruning when VLAN is added */  			if (!vlan_promisc) {  				status = ice_cfg_vlan_pruning(vsi, true, false); @@ -2837,21 +2919,29 @@ static int ice_vc_process_vlan_msg(struct ice_vf *vf, u8 *msg, bool add_v)  		 */  		int num_vf_vlan; -		num_vf_vlan = vf->num_vlan; +		num_vf_vlan = vsi->num_vlan;  		for (i = 0; i < vfl->num_elements && i < num_vf_vlan; i++) {  			u16 vid = vfl->vlan_id[i]; +			/* we add VLAN 0 by default for each VF so we can enable +			 * Tx VLAN anti-spoof without triggering MDD events so +			 * we don't want a VIRTCHNL request to remove it +			 */ +			if (!vid) +				continue; +  			/* Make sure ice_vsi_kill_vlan is successful before  			 * updating VLAN information  			 */ -			if (ice_vsi_kill_vlan(vsi, vid)) { +			status = ice_vsi_kill_vlan(vsi, vid); +			if (status) {  				v_ret = VIRTCHNL_STATUS_ERR_PARAM;  				goto error_param;  			} -			vf->num_vlan--; +			vsi->num_vlan--;  			/* Disable VLAN pruning when the last VLAN is removed */ -			if (!vf->num_vlan) +			if (!vsi->num_vlan)  				ice_cfg_vlan_pruning(vsi, false, false);  			/* Disable Unicast/Multicast VLAN promiscuous mode */ @@ -3165,65 +3255,6 @@ ice_get_vf_cfg(struct net_device *netdev, int vf_id, struct ifla_vf_info *ivi)  }  /** - * ice_set_vf_spoofchk - * @netdev: network interface device structure - * @vf_id: VF identifier - * @ena: flag to enable or disable feature - * - * Enable or disable VF spoof checking - */ -int ice_set_vf_spoofchk(struct net_device *netdev, int vf_id, bool ena) -{ -	struct ice_pf *pf = ice_netdev_to_pf(netdev); -	struct ice_vsi *vsi = pf->vsi[0]; -	struct ice_vsi_ctx *ctx; -	enum ice_status status; -	struct device *dev; -	struct ice_vf *vf; -	int ret = 0; - -	dev = ice_pf_to_dev(pf); -	if (ice_validate_vf_id(pf, vf_id)) -		return -EINVAL; - -	vf = &pf->vf[vf_id]; -	if (ice_check_vf_init(pf, vf)) -		return -EBUSY; - -	if (ena == vf->spoofchk) { -		dev_dbg(dev, "VF spoofchk already %s\n", -			ena ? "ON" : "OFF"); -		return 0; -	} - -	ctx = kzalloc(sizeof(*ctx), GFP_KERNEL); -	if (!ctx) -		return -ENOMEM; - -	ctx->info.valid_sections = cpu_to_le16(ICE_AQ_VSI_PROP_SECURITY_VALID); - -	if (ena) { -		ctx->info.sec_flags |= ICE_AQ_VSI_SEC_FLAG_ENA_MAC_ANTI_SPOOF; -		ctx->info.sw_flags2 |= ICE_AQ_VSI_SW_FLAG_RX_PRUNE_EN_M; -	} - -	status = ice_update_vsi(&pf->hw, vsi->idx, ctx, NULL); -	if (status) { -		dev_dbg(dev, -			"Error %d, failed to update VSI* parameters\n", status); -		ret = -EIO; -		goto out; -	} - -	vf->spoofchk = ena; -	vsi->info.sec_flags = ctx->info.sec_flags; -	vsi->info.sw_flags2 = ctx->info.sw_flags2; -out: -	kfree(ctx); -	return ret; -} - -/**   * ice_wait_on_vf_reset   * @vf: The VF being resseting   * @@ -3344,28 +3375,18 @@ int ice_set_vf_trust(struct net_device *netdev, int vf_id, bool trusted)  int ice_set_vf_link_state(struct net_device *netdev, int vf_id, int link_state)  {  	struct ice_pf *pf = ice_netdev_to_pf(netdev); -	struct virtchnl_pf_event pfe = { 0 }; -	struct ice_link_status *ls;  	struct ice_vf *vf; -	struct ice_hw *hw;  	if (ice_validate_vf_id(pf, vf_id))  		return -EINVAL;  	vf = &pf->vf[vf_id]; -	hw = &pf->hw; -	ls = &pf->hw.port_info->phy.link_info; -  	if (ice_check_vf_init(pf, vf))  		return -EBUSY; -	pfe.event = VIRTCHNL_EVENT_LINK_CHANGE; -	pfe.severity = PF_EVENT_SEVERITY_INFO; -  	switch (link_state) {  	case IFLA_VF_LINK_STATE_AUTO:  		vf->link_forced = false; -		vf->link_up = ls->link_info & ICE_AQ_LINK_UP;  		break;  	case IFLA_VF_LINK_STATE_ENABLE:  		vf->link_forced = true; @@ -3379,15 +3400,7 @@ int ice_set_vf_link_state(struct net_device *netdev, int vf_id, int link_state)  		return -EINVAL;  	} -	if (vf->link_forced) -		ice_set_pfe_link_forced(vf, &pfe, vf->link_up); -	else -		ice_set_pfe_link(vf, &pfe, ls->link_speed, vf->link_up); - -	/* Notify the VF of its new link state */ -	ice_aq_send_msg_to_vf(hw, vf->vf_id, VIRTCHNL_OP_EVENT, -			      VIRTCHNL_STATUS_SUCCESS, (u8 *)&pfe, -			      sizeof(pfe), NULL); +	ice_vc_notify_vf_link_state(vf);  	return 0;  } diff --git a/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.h b/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.h index 88aa65d5cb31..4647d636ed36 100644 --- a/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.h +++ b/drivers/net/ethernet/intel/ice/ice_virtchnl_pf.h @@ -40,6 +40,9 @@  #define ICE_DFLT_INTR_PER_VF		(ICE_DFLT_QS_PER_VF + 1)  #define ICE_MAX_VF_RESET_WAIT		15 +#define ice_for_each_vf(pf, i) \ +	for ((i) = 0; (i) < (pf)->num_alloc_vfs; (i)++) +  /* Specific VF states */  enum ice_vf_states {  	ICE_VF_STATE_INIT = 0,		/* PF is initializing VF */ @@ -91,7 +94,6 @@ struct ice_vf {  	unsigned long vf_caps;		/* VF's adv. capabilities */  	u8 num_req_qs;			/* num of queue pairs requested by VF */  	u16 num_mac; -	u16 num_vlan;  	u16 num_vf_qs;			/* num of queue configured per VF */  	u16 num_qs_ena;			/* total num of Tx/Rx queue enabled */  }; diff --git a/drivers/net/ethernet/intel/ice/ice_xsk.c b/drivers/net/ethernet/intel/ice/ice_xsk.c index cf9b8b22d24f..149dca0012ba 100644 --- a/drivers/net/ethernet/intel/ice/ice_xsk.c +++ b/drivers/net/ethernet/intel/ice/ice_xsk.c @@ -414,7 +414,8 @@ ice_xsk_umem_enable(struct ice_vsi *vsi, struct xdp_umem *umem, u16 qid)  	if (vsi->type != ICE_VSI_PF)  		return -EINVAL; -	vsi->num_xsk_umems = min_t(u16, vsi->num_rxq, vsi->num_txq); +	if (!vsi->num_xsk_umems) +		vsi->num_xsk_umems = min_t(u16, vsi->num_rxq, vsi->num_txq);  	if (qid >= vsi->num_xsk_umems)  		return -EINVAL; @@ -555,7 +556,7 @@ ice_alloc_buf_fast_zc(struct ice_ring *rx_ring, struct ice_rx_buf *rx_buf)  	rx_buf->handle = handle + umem->headroom; -	xsk_umem_discard_addr(umem); +	xsk_umem_release_addr(umem);  	return true;  } @@ -591,7 +592,7 @@ ice_alloc_buf_slow_zc(struct ice_ring *rx_ring, struct ice_rx_buf *rx_buf)  	rx_buf->handle = handle + umem->headroom; -	xsk_umem_discard_addr_rq(umem); +	xsk_umem_release_addr_rq(umem);  	return true;  } @@ -1019,8 +1020,8 @@ bool ice_clean_tx_irq_zc(struct ice_ring *xdp_ring, int budget)  	s16 ntc = xdp_ring->next_to_clean;  	struct ice_tx_desc *tx_desc;  	struct ice_tx_buf *tx_buf; -	bool xmit_done = true;  	u32 xsk_frames = 0; +	bool xmit_done;  	tx_desc = ICE_TX_DESC(xdp_ring, ntc);  	tx_buf = &xdp_ring->tx_buf[ntc]; | 
