162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/* Copyright (c) 2019, Intel Corporation. */
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#include "ice_common.h"
562306a36Sopenharmony_ci#include "ice_sched.h"
662306a36Sopenharmony_ci#include "ice_dcb.h"
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci/**
962306a36Sopenharmony_ci * ice_aq_get_lldp_mib
1062306a36Sopenharmony_ci * @hw: pointer to the HW struct
1162306a36Sopenharmony_ci * @bridge_type: type of bridge requested
1262306a36Sopenharmony_ci * @mib_type: Local, Remote or both Local and Remote MIBs
1362306a36Sopenharmony_ci * @buf: pointer to the caller-supplied buffer to store the MIB block
1462306a36Sopenharmony_ci * @buf_size: size of the buffer (in bytes)
1562306a36Sopenharmony_ci * @local_len: length of the returned Local LLDP MIB
1662306a36Sopenharmony_ci * @remote_len: length of the returned Remote LLDP MIB
1762306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
1862306a36Sopenharmony_ci *
1962306a36Sopenharmony_ci * Requests the complete LLDP MIB (entire packet). (0x0A00)
2062306a36Sopenharmony_ci */
2162306a36Sopenharmony_cistatic int
2262306a36Sopenharmony_ciice_aq_get_lldp_mib(struct ice_hw *hw, u8 bridge_type, u8 mib_type, void *buf,
2362306a36Sopenharmony_ci		    u16 buf_size, u16 *local_len, u16 *remote_len,
2462306a36Sopenharmony_ci		    struct ice_sq_cd *cd)
2562306a36Sopenharmony_ci{
2662306a36Sopenharmony_ci	struct ice_aqc_lldp_get_mib *cmd;
2762306a36Sopenharmony_ci	struct ice_aq_desc desc;
2862306a36Sopenharmony_ci	int status;
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci	cmd = &desc.params.lldp_get_mib;
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci	if (buf_size == 0 || !buf)
3362306a36Sopenharmony_ci		return -EINVAL;
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_lldp_get_mib);
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	cmd->type = mib_type & ICE_AQ_LLDP_MIB_TYPE_M;
3862306a36Sopenharmony_ci	cmd->type |= (bridge_type << ICE_AQ_LLDP_BRID_TYPE_S) &
3962306a36Sopenharmony_ci		ICE_AQ_LLDP_BRID_TYPE_M;
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci	desc.datalen = cpu_to_le16(buf_size);
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	status = ice_aq_send_cmd(hw, &desc, buf, buf_size, cd);
4462306a36Sopenharmony_ci	if (!status) {
4562306a36Sopenharmony_ci		if (local_len)
4662306a36Sopenharmony_ci			*local_len = le16_to_cpu(cmd->local_len);
4762306a36Sopenharmony_ci		if (remote_len)
4862306a36Sopenharmony_ci			*remote_len = le16_to_cpu(cmd->remote_len);
4962306a36Sopenharmony_ci	}
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	return status;
5262306a36Sopenharmony_ci}
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci/**
5562306a36Sopenharmony_ci * ice_aq_cfg_lldp_mib_change
5662306a36Sopenharmony_ci * @hw: pointer to the HW struct
5762306a36Sopenharmony_ci * @ena_update: Enable or Disable event posting
5862306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
5962306a36Sopenharmony_ci *
6062306a36Sopenharmony_ci * Enable or Disable posting of an event on ARQ when LLDP MIB
6162306a36Sopenharmony_ci * associated with the interface changes (0x0A01)
6262306a36Sopenharmony_ci */
6362306a36Sopenharmony_cistatic int
6462306a36Sopenharmony_ciice_aq_cfg_lldp_mib_change(struct ice_hw *hw, bool ena_update,
6562306a36Sopenharmony_ci			   struct ice_sq_cd *cd)
6662306a36Sopenharmony_ci{
6762306a36Sopenharmony_ci	struct ice_aqc_lldp_set_mib_change *cmd;
6862306a36Sopenharmony_ci	struct ice_aq_desc desc;
6962306a36Sopenharmony_ci
7062306a36Sopenharmony_ci	cmd = &desc.params.lldp_set_event;
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_lldp_set_mib_change);
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	if (!ena_update)
7562306a36Sopenharmony_ci		cmd->command |= ICE_AQ_LLDP_MIB_UPDATE_DIS;
7662306a36Sopenharmony_ci	else
7762306a36Sopenharmony_ci		cmd->command |= FIELD_PREP(ICE_AQ_LLDP_MIB_PENDING_M,
7862306a36Sopenharmony_ci					   ICE_AQ_LLDP_MIB_PENDING_ENABLE);
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_ci	return ice_aq_send_cmd(hw, &desc, NULL, 0, cd);
8162306a36Sopenharmony_ci}
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci/**
8462306a36Sopenharmony_ci * ice_aq_stop_lldp
8562306a36Sopenharmony_ci * @hw: pointer to the HW struct
8662306a36Sopenharmony_ci * @shutdown_lldp_agent: True if LLDP Agent needs to be Shutdown
8762306a36Sopenharmony_ci *			 False if LLDP Agent needs to be Stopped
8862306a36Sopenharmony_ci * @persist: True if Stop/Shutdown of LLDP Agent needs to be persistent across
8962306a36Sopenharmony_ci *	     reboots
9062306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
9162306a36Sopenharmony_ci *
9262306a36Sopenharmony_ci * Stop or Shutdown the embedded LLDP Agent (0x0A05)
9362306a36Sopenharmony_ci */
9462306a36Sopenharmony_ciint
9562306a36Sopenharmony_ciice_aq_stop_lldp(struct ice_hw *hw, bool shutdown_lldp_agent, bool persist,
9662306a36Sopenharmony_ci		 struct ice_sq_cd *cd)
9762306a36Sopenharmony_ci{
9862306a36Sopenharmony_ci	struct ice_aqc_lldp_stop *cmd;
9962306a36Sopenharmony_ci	struct ice_aq_desc desc;
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	cmd = &desc.params.lldp_stop;
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_lldp_stop);
10462306a36Sopenharmony_ci
10562306a36Sopenharmony_ci	if (shutdown_lldp_agent)
10662306a36Sopenharmony_ci		cmd->command |= ICE_AQ_LLDP_AGENT_SHUTDOWN;
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci	if (persist)
10962306a36Sopenharmony_ci		cmd->command |= ICE_AQ_LLDP_AGENT_PERSIST_DIS;
11062306a36Sopenharmony_ci
11162306a36Sopenharmony_ci	return ice_aq_send_cmd(hw, &desc, NULL, 0, cd);
11262306a36Sopenharmony_ci}
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ci/**
11562306a36Sopenharmony_ci * ice_aq_start_lldp
11662306a36Sopenharmony_ci * @hw: pointer to the HW struct
11762306a36Sopenharmony_ci * @persist: True if Start of LLDP Agent needs to be persistent across reboots
11862306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
11962306a36Sopenharmony_ci *
12062306a36Sopenharmony_ci * Start the embedded LLDP Agent on all ports. (0x0A06)
12162306a36Sopenharmony_ci */
12262306a36Sopenharmony_ciint ice_aq_start_lldp(struct ice_hw *hw, bool persist, struct ice_sq_cd *cd)
12362306a36Sopenharmony_ci{
12462306a36Sopenharmony_ci	struct ice_aqc_lldp_start *cmd;
12562306a36Sopenharmony_ci	struct ice_aq_desc desc;
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ci	cmd = &desc.params.lldp_start;
12862306a36Sopenharmony_ci
12962306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_lldp_start);
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	cmd->command = ICE_AQ_LLDP_AGENT_START;
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_ci	if (persist)
13462306a36Sopenharmony_ci		cmd->command |= ICE_AQ_LLDP_AGENT_PERSIST_ENA;
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci	return ice_aq_send_cmd(hw, &desc, NULL, 0, cd);
13762306a36Sopenharmony_ci}
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ci/**
14062306a36Sopenharmony_ci * ice_get_dcbx_status
14162306a36Sopenharmony_ci * @hw: pointer to the HW struct
14262306a36Sopenharmony_ci *
14362306a36Sopenharmony_ci * Get the DCBX status from the Firmware
14462306a36Sopenharmony_ci */
14562306a36Sopenharmony_cistatic u8 ice_get_dcbx_status(struct ice_hw *hw)
14662306a36Sopenharmony_ci{
14762306a36Sopenharmony_ci	u32 reg;
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci	reg = rd32(hw, PRTDCB_GENS);
15062306a36Sopenharmony_ci	return (u8)((reg & PRTDCB_GENS_DCBX_STATUS_M) >>
15162306a36Sopenharmony_ci		    PRTDCB_GENS_DCBX_STATUS_S);
15262306a36Sopenharmony_ci}
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_ci/**
15562306a36Sopenharmony_ci * ice_parse_ieee_ets_common_tlv
15662306a36Sopenharmony_ci * @buf: Data buffer to be parsed for ETS CFG/REC data
15762306a36Sopenharmony_ci * @ets_cfg: Container to store parsed data
15862306a36Sopenharmony_ci *
15962306a36Sopenharmony_ci * Parses the common data of IEEE 802.1Qaz ETS CFG/REC TLV
16062306a36Sopenharmony_ci */
16162306a36Sopenharmony_cistatic void
16262306a36Sopenharmony_ciice_parse_ieee_ets_common_tlv(u8 *buf, struct ice_dcb_ets_cfg *ets_cfg)
16362306a36Sopenharmony_ci{
16462306a36Sopenharmony_ci	u8 offset = 0;
16562306a36Sopenharmony_ci	int i;
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_ci	/* Priority Assignment Table (4 octets)
16862306a36Sopenharmony_ci	 * Octets:|    1    |    2    |    3    |    4    |
16962306a36Sopenharmony_ci	 *        -----------------------------------------
17062306a36Sopenharmony_ci	 *        |pri0|pri1|pri2|pri3|pri4|pri5|pri6|pri7|
17162306a36Sopenharmony_ci	 *        -----------------------------------------
17262306a36Sopenharmony_ci	 *   Bits:|7  4|3  0|7  4|3  0|7  4|3  0|7  4|3  0|
17362306a36Sopenharmony_ci	 *        -----------------------------------------
17462306a36Sopenharmony_ci	 */
17562306a36Sopenharmony_ci	for (i = 0; i < 4; i++) {
17662306a36Sopenharmony_ci		ets_cfg->prio_table[i * 2] =
17762306a36Sopenharmony_ci			((buf[offset] & ICE_IEEE_ETS_PRIO_1_M) >>
17862306a36Sopenharmony_ci			 ICE_IEEE_ETS_PRIO_1_S);
17962306a36Sopenharmony_ci		ets_cfg->prio_table[i * 2 + 1] =
18062306a36Sopenharmony_ci			((buf[offset] & ICE_IEEE_ETS_PRIO_0_M) >>
18162306a36Sopenharmony_ci			 ICE_IEEE_ETS_PRIO_0_S);
18262306a36Sopenharmony_ci		offset++;
18362306a36Sopenharmony_ci	}
18462306a36Sopenharmony_ci
18562306a36Sopenharmony_ci	/* TC Bandwidth Table (8 octets)
18662306a36Sopenharmony_ci	 * Octets:| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
18762306a36Sopenharmony_ci	 *        ---------------------------------
18862306a36Sopenharmony_ci	 *        |tc0|tc1|tc2|tc3|tc4|tc5|tc6|tc7|
18962306a36Sopenharmony_ci	 *        ---------------------------------
19062306a36Sopenharmony_ci	 *
19162306a36Sopenharmony_ci	 * TSA Assignment Table (8 octets)
19262306a36Sopenharmony_ci	 * Octets:| 9 | 10| 11| 12| 13| 14| 15| 16|
19362306a36Sopenharmony_ci	 *        ---------------------------------
19462306a36Sopenharmony_ci	 *        |tc0|tc1|tc2|tc3|tc4|tc5|tc6|tc7|
19562306a36Sopenharmony_ci	 *        ---------------------------------
19662306a36Sopenharmony_ci	 */
19762306a36Sopenharmony_ci	ice_for_each_traffic_class(i) {
19862306a36Sopenharmony_ci		ets_cfg->tcbwtable[i] = buf[offset];
19962306a36Sopenharmony_ci		ets_cfg->tsatable[i] = buf[ICE_MAX_TRAFFIC_CLASS + offset++];
20062306a36Sopenharmony_ci	}
20162306a36Sopenharmony_ci}
20262306a36Sopenharmony_ci
20362306a36Sopenharmony_ci/**
20462306a36Sopenharmony_ci * ice_parse_ieee_etscfg_tlv
20562306a36Sopenharmony_ci * @tlv: IEEE 802.1Qaz ETS CFG TLV
20662306a36Sopenharmony_ci * @dcbcfg: Local store to update ETS CFG data
20762306a36Sopenharmony_ci *
20862306a36Sopenharmony_ci * Parses IEEE 802.1Qaz ETS CFG TLV
20962306a36Sopenharmony_ci */
21062306a36Sopenharmony_cistatic void
21162306a36Sopenharmony_ciice_parse_ieee_etscfg_tlv(struct ice_lldp_org_tlv *tlv,
21262306a36Sopenharmony_ci			  struct ice_dcbx_cfg *dcbcfg)
21362306a36Sopenharmony_ci{
21462306a36Sopenharmony_ci	struct ice_dcb_ets_cfg *etscfg;
21562306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
21662306a36Sopenharmony_ci
21762306a36Sopenharmony_ci	/* First Octet post subtype
21862306a36Sopenharmony_ci	 * --------------------------
21962306a36Sopenharmony_ci	 * |will-|CBS  | Re-  | Max |
22062306a36Sopenharmony_ci	 * |ing  |     |served| TCs |
22162306a36Sopenharmony_ci	 * --------------------------
22262306a36Sopenharmony_ci	 * |1bit | 1bit|3 bits|3bits|
22362306a36Sopenharmony_ci	 */
22462306a36Sopenharmony_ci	etscfg = &dcbcfg->etscfg;
22562306a36Sopenharmony_ci	etscfg->willing = ((buf[0] & ICE_IEEE_ETS_WILLING_M) >>
22662306a36Sopenharmony_ci			   ICE_IEEE_ETS_WILLING_S);
22762306a36Sopenharmony_ci	etscfg->cbs = ((buf[0] & ICE_IEEE_ETS_CBS_M) >> ICE_IEEE_ETS_CBS_S);
22862306a36Sopenharmony_ci	etscfg->maxtcs = ((buf[0] & ICE_IEEE_ETS_MAXTC_M) >>
22962306a36Sopenharmony_ci			  ICE_IEEE_ETS_MAXTC_S);
23062306a36Sopenharmony_ci
23162306a36Sopenharmony_ci	/* Begin parsing at Priority Assignment Table (offset 1 in buf) */
23262306a36Sopenharmony_ci	ice_parse_ieee_ets_common_tlv(&buf[1], etscfg);
23362306a36Sopenharmony_ci}
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci/**
23662306a36Sopenharmony_ci * ice_parse_ieee_etsrec_tlv
23762306a36Sopenharmony_ci * @tlv: IEEE 802.1Qaz ETS REC TLV
23862306a36Sopenharmony_ci * @dcbcfg: Local store to update ETS REC data
23962306a36Sopenharmony_ci *
24062306a36Sopenharmony_ci * Parses IEEE 802.1Qaz ETS REC TLV
24162306a36Sopenharmony_ci */
24262306a36Sopenharmony_cistatic void
24362306a36Sopenharmony_ciice_parse_ieee_etsrec_tlv(struct ice_lldp_org_tlv *tlv,
24462306a36Sopenharmony_ci			  struct ice_dcbx_cfg *dcbcfg)
24562306a36Sopenharmony_ci{
24662306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_ci	/* Begin parsing at Priority Assignment Table (offset 1 in buf) */
24962306a36Sopenharmony_ci	ice_parse_ieee_ets_common_tlv(&buf[1], &dcbcfg->etsrec);
25062306a36Sopenharmony_ci}
25162306a36Sopenharmony_ci
25262306a36Sopenharmony_ci/**
25362306a36Sopenharmony_ci * ice_parse_ieee_pfccfg_tlv
25462306a36Sopenharmony_ci * @tlv: IEEE 802.1Qaz PFC CFG TLV
25562306a36Sopenharmony_ci * @dcbcfg: Local store to update PFC CFG data
25662306a36Sopenharmony_ci *
25762306a36Sopenharmony_ci * Parses IEEE 802.1Qaz PFC CFG TLV
25862306a36Sopenharmony_ci */
25962306a36Sopenharmony_cistatic void
26062306a36Sopenharmony_ciice_parse_ieee_pfccfg_tlv(struct ice_lldp_org_tlv *tlv,
26162306a36Sopenharmony_ci			  struct ice_dcbx_cfg *dcbcfg)
26262306a36Sopenharmony_ci{
26362306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
26462306a36Sopenharmony_ci
26562306a36Sopenharmony_ci	/* ----------------------------------------
26662306a36Sopenharmony_ci	 * |will-|MBC  | Re-  | PFC |  PFC Enable  |
26762306a36Sopenharmony_ci	 * |ing  |     |served| cap |              |
26862306a36Sopenharmony_ci	 * -----------------------------------------
26962306a36Sopenharmony_ci	 * |1bit | 1bit|2 bits|4bits| 1 octet      |
27062306a36Sopenharmony_ci	 */
27162306a36Sopenharmony_ci	dcbcfg->pfc.willing = ((buf[0] & ICE_IEEE_PFC_WILLING_M) >>
27262306a36Sopenharmony_ci			       ICE_IEEE_PFC_WILLING_S);
27362306a36Sopenharmony_ci	dcbcfg->pfc.mbc = ((buf[0] & ICE_IEEE_PFC_MBC_M) >> ICE_IEEE_PFC_MBC_S);
27462306a36Sopenharmony_ci	dcbcfg->pfc.pfccap = ((buf[0] & ICE_IEEE_PFC_CAP_M) >>
27562306a36Sopenharmony_ci			      ICE_IEEE_PFC_CAP_S);
27662306a36Sopenharmony_ci	dcbcfg->pfc.pfcena = buf[1];
27762306a36Sopenharmony_ci}
27862306a36Sopenharmony_ci
27962306a36Sopenharmony_ci/**
28062306a36Sopenharmony_ci * ice_parse_ieee_app_tlv
28162306a36Sopenharmony_ci * @tlv: IEEE 802.1Qaz APP TLV
28262306a36Sopenharmony_ci * @dcbcfg: Local store to update APP PRIO data
28362306a36Sopenharmony_ci *
28462306a36Sopenharmony_ci * Parses IEEE 802.1Qaz APP PRIO TLV
28562306a36Sopenharmony_ci */
28662306a36Sopenharmony_cistatic void
28762306a36Sopenharmony_ciice_parse_ieee_app_tlv(struct ice_lldp_org_tlv *tlv,
28862306a36Sopenharmony_ci		       struct ice_dcbx_cfg *dcbcfg)
28962306a36Sopenharmony_ci{
29062306a36Sopenharmony_ci	u16 offset = 0;
29162306a36Sopenharmony_ci	u16 typelen;
29262306a36Sopenharmony_ci	int i = 0;
29362306a36Sopenharmony_ci	u16 len;
29462306a36Sopenharmony_ci	u8 *buf;
29562306a36Sopenharmony_ci
29662306a36Sopenharmony_ci	typelen = ntohs(tlv->typelen);
29762306a36Sopenharmony_ci	len = ((typelen & ICE_LLDP_TLV_LEN_M) >> ICE_LLDP_TLV_LEN_S);
29862306a36Sopenharmony_ci	buf = tlv->tlvinfo;
29962306a36Sopenharmony_ci
30062306a36Sopenharmony_ci	/* Removing sizeof(ouisubtype) and reserved byte from len.
30162306a36Sopenharmony_ci	 * Remaining len div 3 is number of APP TLVs.
30262306a36Sopenharmony_ci	 */
30362306a36Sopenharmony_ci	len -= (sizeof(tlv->ouisubtype) + 1);
30462306a36Sopenharmony_ci
30562306a36Sopenharmony_ci	/* Move offset to App Priority Table */
30662306a36Sopenharmony_ci	offset++;
30762306a36Sopenharmony_ci
30862306a36Sopenharmony_ci	/* Application Priority Table (3 octets)
30962306a36Sopenharmony_ci	 * Octets:|         1          |    2    |    3    |
31062306a36Sopenharmony_ci	 *        -----------------------------------------
31162306a36Sopenharmony_ci	 *        |Priority|Rsrvd| Sel |    Protocol ID    |
31262306a36Sopenharmony_ci	 *        -----------------------------------------
31362306a36Sopenharmony_ci	 *   Bits:|23    21|20 19|18 16|15                0|
31462306a36Sopenharmony_ci	 *        -----------------------------------------
31562306a36Sopenharmony_ci	 */
31662306a36Sopenharmony_ci	while (offset < len) {
31762306a36Sopenharmony_ci		dcbcfg->app[i].priority = ((buf[offset] &
31862306a36Sopenharmony_ci					    ICE_IEEE_APP_PRIO_M) >>
31962306a36Sopenharmony_ci					   ICE_IEEE_APP_PRIO_S);
32062306a36Sopenharmony_ci		dcbcfg->app[i].selector = ((buf[offset] &
32162306a36Sopenharmony_ci					    ICE_IEEE_APP_SEL_M) >>
32262306a36Sopenharmony_ci					   ICE_IEEE_APP_SEL_S);
32362306a36Sopenharmony_ci		dcbcfg->app[i].prot_id = (buf[offset + 1] << 0x8) |
32462306a36Sopenharmony_ci			buf[offset + 2];
32562306a36Sopenharmony_ci		/* Move to next app */
32662306a36Sopenharmony_ci		offset += 3;
32762306a36Sopenharmony_ci		i++;
32862306a36Sopenharmony_ci		if (i >= ICE_DCBX_MAX_APPS)
32962306a36Sopenharmony_ci			break;
33062306a36Sopenharmony_ci	}
33162306a36Sopenharmony_ci
33262306a36Sopenharmony_ci	dcbcfg->numapps = i;
33362306a36Sopenharmony_ci}
33462306a36Sopenharmony_ci
33562306a36Sopenharmony_ci/**
33662306a36Sopenharmony_ci * ice_parse_ieee_tlv
33762306a36Sopenharmony_ci * @tlv: IEEE 802.1Qaz TLV
33862306a36Sopenharmony_ci * @dcbcfg: Local store to update ETS REC data
33962306a36Sopenharmony_ci *
34062306a36Sopenharmony_ci * Get the TLV subtype and send it to parsing function
34162306a36Sopenharmony_ci * based on the subtype value
34262306a36Sopenharmony_ci */
34362306a36Sopenharmony_cistatic void
34462306a36Sopenharmony_ciice_parse_ieee_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
34562306a36Sopenharmony_ci{
34662306a36Sopenharmony_ci	u32 ouisubtype;
34762306a36Sopenharmony_ci	u8 subtype;
34862306a36Sopenharmony_ci
34962306a36Sopenharmony_ci	ouisubtype = ntohl(tlv->ouisubtype);
35062306a36Sopenharmony_ci	subtype = (u8)((ouisubtype & ICE_LLDP_TLV_SUBTYPE_M) >>
35162306a36Sopenharmony_ci		       ICE_LLDP_TLV_SUBTYPE_S);
35262306a36Sopenharmony_ci	switch (subtype) {
35362306a36Sopenharmony_ci	case ICE_IEEE_SUBTYPE_ETS_CFG:
35462306a36Sopenharmony_ci		ice_parse_ieee_etscfg_tlv(tlv, dcbcfg);
35562306a36Sopenharmony_ci		break;
35662306a36Sopenharmony_ci	case ICE_IEEE_SUBTYPE_ETS_REC:
35762306a36Sopenharmony_ci		ice_parse_ieee_etsrec_tlv(tlv, dcbcfg);
35862306a36Sopenharmony_ci		break;
35962306a36Sopenharmony_ci	case ICE_IEEE_SUBTYPE_PFC_CFG:
36062306a36Sopenharmony_ci		ice_parse_ieee_pfccfg_tlv(tlv, dcbcfg);
36162306a36Sopenharmony_ci		break;
36262306a36Sopenharmony_ci	case ICE_IEEE_SUBTYPE_APP_PRI:
36362306a36Sopenharmony_ci		ice_parse_ieee_app_tlv(tlv, dcbcfg);
36462306a36Sopenharmony_ci		break;
36562306a36Sopenharmony_ci	default:
36662306a36Sopenharmony_ci		break;
36762306a36Sopenharmony_ci	}
36862306a36Sopenharmony_ci}
36962306a36Sopenharmony_ci
37062306a36Sopenharmony_ci/**
37162306a36Sopenharmony_ci * ice_parse_cee_pgcfg_tlv
37262306a36Sopenharmony_ci * @tlv: CEE DCBX PG CFG TLV
37362306a36Sopenharmony_ci * @dcbcfg: Local store to update ETS CFG data
37462306a36Sopenharmony_ci *
37562306a36Sopenharmony_ci * Parses CEE DCBX PG CFG TLV
37662306a36Sopenharmony_ci */
37762306a36Sopenharmony_cistatic void
37862306a36Sopenharmony_ciice_parse_cee_pgcfg_tlv(struct ice_cee_feat_tlv *tlv,
37962306a36Sopenharmony_ci			struct ice_dcbx_cfg *dcbcfg)
38062306a36Sopenharmony_ci{
38162306a36Sopenharmony_ci	struct ice_dcb_ets_cfg *etscfg;
38262306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
38362306a36Sopenharmony_ci	u16 offset = 0;
38462306a36Sopenharmony_ci	int i;
38562306a36Sopenharmony_ci
38662306a36Sopenharmony_ci	etscfg = &dcbcfg->etscfg;
38762306a36Sopenharmony_ci
38862306a36Sopenharmony_ci	if (tlv->en_will_err & ICE_CEE_FEAT_TLV_WILLING_M)
38962306a36Sopenharmony_ci		etscfg->willing = 1;
39062306a36Sopenharmony_ci
39162306a36Sopenharmony_ci	etscfg->cbs = 0;
39262306a36Sopenharmony_ci	/* Priority Group Table (4 octets)
39362306a36Sopenharmony_ci	 * Octets:|    1    |    2    |    3    |    4    |
39462306a36Sopenharmony_ci	 *        -----------------------------------------
39562306a36Sopenharmony_ci	 *        |pri0|pri1|pri2|pri3|pri4|pri5|pri6|pri7|
39662306a36Sopenharmony_ci	 *        -----------------------------------------
39762306a36Sopenharmony_ci	 *   Bits:|7  4|3  0|7  4|3  0|7  4|3  0|7  4|3  0|
39862306a36Sopenharmony_ci	 *        -----------------------------------------
39962306a36Sopenharmony_ci	 */
40062306a36Sopenharmony_ci	for (i = 0; i < 4; i++) {
40162306a36Sopenharmony_ci		etscfg->prio_table[i * 2] =
40262306a36Sopenharmony_ci			((buf[offset] & ICE_CEE_PGID_PRIO_1_M) >>
40362306a36Sopenharmony_ci			 ICE_CEE_PGID_PRIO_1_S);
40462306a36Sopenharmony_ci		etscfg->prio_table[i * 2 + 1] =
40562306a36Sopenharmony_ci			((buf[offset] & ICE_CEE_PGID_PRIO_0_M) >>
40662306a36Sopenharmony_ci			 ICE_CEE_PGID_PRIO_0_S);
40762306a36Sopenharmony_ci		offset++;
40862306a36Sopenharmony_ci	}
40962306a36Sopenharmony_ci
41062306a36Sopenharmony_ci	/* PG Percentage Table (8 octets)
41162306a36Sopenharmony_ci	 * Octets:| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
41262306a36Sopenharmony_ci	 *        ---------------------------------
41362306a36Sopenharmony_ci	 *        |pg0|pg1|pg2|pg3|pg4|pg5|pg6|pg7|
41462306a36Sopenharmony_ci	 *        ---------------------------------
41562306a36Sopenharmony_ci	 */
41662306a36Sopenharmony_ci	ice_for_each_traffic_class(i) {
41762306a36Sopenharmony_ci		etscfg->tcbwtable[i] = buf[offset++];
41862306a36Sopenharmony_ci
41962306a36Sopenharmony_ci		if (etscfg->prio_table[i] == ICE_CEE_PGID_STRICT)
42062306a36Sopenharmony_ci			dcbcfg->etscfg.tsatable[i] = ICE_IEEE_TSA_STRICT;
42162306a36Sopenharmony_ci		else
42262306a36Sopenharmony_ci			dcbcfg->etscfg.tsatable[i] = ICE_IEEE_TSA_ETS;
42362306a36Sopenharmony_ci	}
42462306a36Sopenharmony_ci
42562306a36Sopenharmony_ci	/* Number of TCs supported (1 octet) */
42662306a36Sopenharmony_ci	etscfg->maxtcs = buf[offset];
42762306a36Sopenharmony_ci}
42862306a36Sopenharmony_ci
42962306a36Sopenharmony_ci/**
43062306a36Sopenharmony_ci * ice_parse_cee_pfccfg_tlv
43162306a36Sopenharmony_ci * @tlv: CEE DCBX PFC CFG TLV
43262306a36Sopenharmony_ci * @dcbcfg: Local store to update PFC CFG data
43362306a36Sopenharmony_ci *
43462306a36Sopenharmony_ci * Parses CEE DCBX PFC CFG TLV
43562306a36Sopenharmony_ci */
43662306a36Sopenharmony_cistatic void
43762306a36Sopenharmony_ciice_parse_cee_pfccfg_tlv(struct ice_cee_feat_tlv *tlv,
43862306a36Sopenharmony_ci			 struct ice_dcbx_cfg *dcbcfg)
43962306a36Sopenharmony_ci{
44062306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
44162306a36Sopenharmony_ci
44262306a36Sopenharmony_ci	if (tlv->en_will_err & ICE_CEE_FEAT_TLV_WILLING_M)
44362306a36Sopenharmony_ci		dcbcfg->pfc.willing = 1;
44462306a36Sopenharmony_ci
44562306a36Sopenharmony_ci	/* ------------------------
44662306a36Sopenharmony_ci	 * | PFC Enable | PFC TCs |
44762306a36Sopenharmony_ci	 * ------------------------
44862306a36Sopenharmony_ci	 * | 1 octet    | 1 octet |
44962306a36Sopenharmony_ci	 */
45062306a36Sopenharmony_ci	dcbcfg->pfc.pfcena = buf[0];
45162306a36Sopenharmony_ci	dcbcfg->pfc.pfccap = buf[1];
45262306a36Sopenharmony_ci}
45362306a36Sopenharmony_ci
45462306a36Sopenharmony_ci/**
45562306a36Sopenharmony_ci * ice_parse_cee_app_tlv
45662306a36Sopenharmony_ci * @tlv: CEE DCBX APP TLV
45762306a36Sopenharmony_ci * @dcbcfg: Local store to update APP PRIO data
45862306a36Sopenharmony_ci *
45962306a36Sopenharmony_ci * Parses CEE DCBX APP PRIO TLV
46062306a36Sopenharmony_ci */
46162306a36Sopenharmony_cistatic void
46262306a36Sopenharmony_ciice_parse_cee_app_tlv(struct ice_cee_feat_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
46362306a36Sopenharmony_ci{
46462306a36Sopenharmony_ci	u16 len, typelen, offset = 0;
46562306a36Sopenharmony_ci	struct ice_cee_app_prio *app;
46662306a36Sopenharmony_ci	u8 i;
46762306a36Sopenharmony_ci
46862306a36Sopenharmony_ci	typelen = ntohs(tlv->hdr.typelen);
46962306a36Sopenharmony_ci	len = ((typelen & ICE_LLDP_TLV_LEN_M) >> ICE_LLDP_TLV_LEN_S);
47062306a36Sopenharmony_ci
47162306a36Sopenharmony_ci	dcbcfg->numapps = len / sizeof(*app);
47262306a36Sopenharmony_ci	if (!dcbcfg->numapps)
47362306a36Sopenharmony_ci		return;
47462306a36Sopenharmony_ci	if (dcbcfg->numapps > ICE_DCBX_MAX_APPS)
47562306a36Sopenharmony_ci		dcbcfg->numapps = ICE_DCBX_MAX_APPS;
47662306a36Sopenharmony_ci
47762306a36Sopenharmony_ci	for (i = 0; i < dcbcfg->numapps; i++) {
47862306a36Sopenharmony_ci		u8 up, selector;
47962306a36Sopenharmony_ci
48062306a36Sopenharmony_ci		app = (struct ice_cee_app_prio *)(tlv->tlvinfo + offset);
48162306a36Sopenharmony_ci		for (up = 0; up < ICE_MAX_USER_PRIORITY; up++)
48262306a36Sopenharmony_ci			if (app->prio_map & BIT(up))
48362306a36Sopenharmony_ci				break;
48462306a36Sopenharmony_ci
48562306a36Sopenharmony_ci		dcbcfg->app[i].priority = up;
48662306a36Sopenharmony_ci
48762306a36Sopenharmony_ci		/* Get Selector from lower 2 bits, and convert to IEEE */
48862306a36Sopenharmony_ci		selector = (app->upper_oui_sel & ICE_CEE_APP_SELECTOR_M);
48962306a36Sopenharmony_ci		switch (selector) {
49062306a36Sopenharmony_ci		case ICE_CEE_APP_SEL_ETHTYPE:
49162306a36Sopenharmony_ci			dcbcfg->app[i].selector = ICE_APP_SEL_ETHTYPE;
49262306a36Sopenharmony_ci			break;
49362306a36Sopenharmony_ci		case ICE_CEE_APP_SEL_TCPIP:
49462306a36Sopenharmony_ci			dcbcfg->app[i].selector = ICE_APP_SEL_TCPIP;
49562306a36Sopenharmony_ci			break;
49662306a36Sopenharmony_ci		default:
49762306a36Sopenharmony_ci			/* Keep selector as it is for unknown types */
49862306a36Sopenharmony_ci			dcbcfg->app[i].selector = selector;
49962306a36Sopenharmony_ci		}
50062306a36Sopenharmony_ci
50162306a36Sopenharmony_ci		dcbcfg->app[i].prot_id = ntohs(app->protocol);
50262306a36Sopenharmony_ci		/* Move to next app */
50362306a36Sopenharmony_ci		offset += sizeof(*app);
50462306a36Sopenharmony_ci	}
50562306a36Sopenharmony_ci}
50662306a36Sopenharmony_ci
50762306a36Sopenharmony_ci/**
50862306a36Sopenharmony_ci * ice_parse_cee_tlv
50962306a36Sopenharmony_ci * @tlv: CEE DCBX TLV
51062306a36Sopenharmony_ci * @dcbcfg: Local store to update DCBX config data
51162306a36Sopenharmony_ci *
51262306a36Sopenharmony_ci * Get the TLV subtype and send it to parsing function
51362306a36Sopenharmony_ci * based on the subtype value
51462306a36Sopenharmony_ci */
51562306a36Sopenharmony_cistatic void
51662306a36Sopenharmony_ciice_parse_cee_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
51762306a36Sopenharmony_ci{
51862306a36Sopenharmony_ci	struct ice_cee_feat_tlv *sub_tlv;
51962306a36Sopenharmony_ci	u8 subtype, feat_tlv_count = 0;
52062306a36Sopenharmony_ci	u16 len, tlvlen, typelen;
52162306a36Sopenharmony_ci	u32 ouisubtype;
52262306a36Sopenharmony_ci
52362306a36Sopenharmony_ci	ouisubtype = ntohl(tlv->ouisubtype);
52462306a36Sopenharmony_ci	subtype = (u8)((ouisubtype & ICE_LLDP_TLV_SUBTYPE_M) >>
52562306a36Sopenharmony_ci		       ICE_LLDP_TLV_SUBTYPE_S);
52662306a36Sopenharmony_ci	/* Return if not CEE DCBX */
52762306a36Sopenharmony_ci	if (subtype != ICE_CEE_DCBX_TYPE)
52862306a36Sopenharmony_ci		return;
52962306a36Sopenharmony_ci
53062306a36Sopenharmony_ci	typelen = ntohs(tlv->typelen);
53162306a36Sopenharmony_ci	tlvlen = ((typelen & ICE_LLDP_TLV_LEN_M) >> ICE_LLDP_TLV_LEN_S);
53262306a36Sopenharmony_ci	len = sizeof(tlv->typelen) + sizeof(ouisubtype) +
53362306a36Sopenharmony_ci		sizeof(struct ice_cee_ctrl_tlv);
53462306a36Sopenharmony_ci	/* Return if no CEE DCBX Feature TLVs */
53562306a36Sopenharmony_ci	if (tlvlen <= len)
53662306a36Sopenharmony_ci		return;
53762306a36Sopenharmony_ci
53862306a36Sopenharmony_ci	sub_tlv = (struct ice_cee_feat_tlv *)((char *)tlv + len);
53962306a36Sopenharmony_ci	while (feat_tlv_count < ICE_CEE_MAX_FEAT_TYPE) {
54062306a36Sopenharmony_ci		u16 sublen;
54162306a36Sopenharmony_ci
54262306a36Sopenharmony_ci		typelen = ntohs(sub_tlv->hdr.typelen);
54362306a36Sopenharmony_ci		sublen = ((typelen & ICE_LLDP_TLV_LEN_M) >> ICE_LLDP_TLV_LEN_S);
54462306a36Sopenharmony_ci		subtype = (u8)((typelen & ICE_LLDP_TLV_TYPE_M) >>
54562306a36Sopenharmony_ci			       ICE_LLDP_TLV_TYPE_S);
54662306a36Sopenharmony_ci		switch (subtype) {
54762306a36Sopenharmony_ci		case ICE_CEE_SUBTYPE_PG_CFG:
54862306a36Sopenharmony_ci			ice_parse_cee_pgcfg_tlv(sub_tlv, dcbcfg);
54962306a36Sopenharmony_ci			break;
55062306a36Sopenharmony_ci		case ICE_CEE_SUBTYPE_PFC_CFG:
55162306a36Sopenharmony_ci			ice_parse_cee_pfccfg_tlv(sub_tlv, dcbcfg);
55262306a36Sopenharmony_ci			break;
55362306a36Sopenharmony_ci		case ICE_CEE_SUBTYPE_APP_PRI:
55462306a36Sopenharmony_ci			ice_parse_cee_app_tlv(sub_tlv, dcbcfg);
55562306a36Sopenharmony_ci			break;
55662306a36Sopenharmony_ci		default:
55762306a36Sopenharmony_ci			return;	/* Invalid Sub-type return */
55862306a36Sopenharmony_ci		}
55962306a36Sopenharmony_ci		feat_tlv_count++;
56062306a36Sopenharmony_ci		/* Move to next sub TLV */
56162306a36Sopenharmony_ci		sub_tlv = (struct ice_cee_feat_tlv *)
56262306a36Sopenharmony_ci			  ((char *)sub_tlv + sizeof(sub_tlv->hdr.typelen) +
56362306a36Sopenharmony_ci			   sublen);
56462306a36Sopenharmony_ci	}
56562306a36Sopenharmony_ci}
56662306a36Sopenharmony_ci
56762306a36Sopenharmony_ci/**
56862306a36Sopenharmony_ci * ice_parse_org_tlv
56962306a36Sopenharmony_ci * @tlv: Organization specific TLV
57062306a36Sopenharmony_ci * @dcbcfg: Local store to update ETS REC data
57162306a36Sopenharmony_ci *
57262306a36Sopenharmony_ci * Currently IEEE 802.1Qaz and CEE DCBX TLV are supported, others
57362306a36Sopenharmony_ci * will be returned
57462306a36Sopenharmony_ci */
57562306a36Sopenharmony_cistatic void
57662306a36Sopenharmony_ciice_parse_org_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
57762306a36Sopenharmony_ci{
57862306a36Sopenharmony_ci	u32 ouisubtype;
57962306a36Sopenharmony_ci	u32 oui;
58062306a36Sopenharmony_ci
58162306a36Sopenharmony_ci	ouisubtype = ntohl(tlv->ouisubtype);
58262306a36Sopenharmony_ci	oui = ((ouisubtype & ICE_LLDP_TLV_OUI_M) >> ICE_LLDP_TLV_OUI_S);
58362306a36Sopenharmony_ci	switch (oui) {
58462306a36Sopenharmony_ci	case ICE_IEEE_8021QAZ_OUI:
58562306a36Sopenharmony_ci		ice_parse_ieee_tlv(tlv, dcbcfg);
58662306a36Sopenharmony_ci		break;
58762306a36Sopenharmony_ci	case ICE_CEE_DCBX_OUI:
58862306a36Sopenharmony_ci		ice_parse_cee_tlv(tlv, dcbcfg);
58962306a36Sopenharmony_ci		break;
59062306a36Sopenharmony_ci	default:
59162306a36Sopenharmony_ci		break; /* Other OUIs not supported */
59262306a36Sopenharmony_ci	}
59362306a36Sopenharmony_ci}
59462306a36Sopenharmony_ci
59562306a36Sopenharmony_ci/**
59662306a36Sopenharmony_ci * ice_lldp_to_dcb_cfg
59762306a36Sopenharmony_ci * @lldpmib: LLDPDU to be parsed
59862306a36Sopenharmony_ci * @dcbcfg: store for LLDPDU data
59962306a36Sopenharmony_ci *
60062306a36Sopenharmony_ci * Parse DCB configuration from the LLDPDU
60162306a36Sopenharmony_ci */
60262306a36Sopenharmony_cistatic int ice_lldp_to_dcb_cfg(u8 *lldpmib, struct ice_dcbx_cfg *dcbcfg)
60362306a36Sopenharmony_ci{
60462306a36Sopenharmony_ci	struct ice_lldp_org_tlv *tlv;
60562306a36Sopenharmony_ci	u16 offset = 0;
60662306a36Sopenharmony_ci	int ret = 0;
60762306a36Sopenharmony_ci	u16 typelen;
60862306a36Sopenharmony_ci	u16 type;
60962306a36Sopenharmony_ci	u16 len;
61062306a36Sopenharmony_ci
61162306a36Sopenharmony_ci	if (!lldpmib || !dcbcfg)
61262306a36Sopenharmony_ci		return -EINVAL;
61362306a36Sopenharmony_ci
61462306a36Sopenharmony_ci	/* set to the start of LLDPDU */
61562306a36Sopenharmony_ci	lldpmib += ETH_HLEN;
61662306a36Sopenharmony_ci	tlv = (struct ice_lldp_org_tlv *)lldpmib;
61762306a36Sopenharmony_ci	while (1) {
61862306a36Sopenharmony_ci		typelen = ntohs(tlv->typelen);
61962306a36Sopenharmony_ci		type = ((typelen & ICE_LLDP_TLV_TYPE_M) >> ICE_LLDP_TLV_TYPE_S);
62062306a36Sopenharmony_ci		len = ((typelen & ICE_LLDP_TLV_LEN_M) >> ICE_LLDP_TLV_LEN_S);
62162306a36Sopenharmony_ci		offset += sizeof(typelen) + len;
62262306a36Sopenharmony_ci
62362306a36Sopenharmony_ci		/* END TLV or beyond LLDPDU size */
62462306a36Sopenharmony_ci		if (type == ICE_TLV_TYPE_END || offset > ICE_LLDPDU_SIZE)
62562306a36Sopenharmony_ci			break;
62662306a36Sopenharmony_ci
62762306a36Sopenharmony_ci		switch (type) {
62862306a36Sopenharmony_ci		case ICE_TLV_TYPE_ORG:
62962306a36Sopenharmony_ci			ice_parse_org_tlv(tlv, dcbcfg);
63062306a36Sopenharmony_ci			break;
63162306a36Sopenharmony_ci		default:
63262306a36Sopenharmony_ci			break;
63362306a36Sopenharmony_ci		}
63462306a36Sopenharmony_ci
63562306a36Sopenharmony_ci		/* Move to next TLV */
63662306a36Sopenharmony_ci		tlv = (struct ice_lldp_org_tlv *)
63762306a36Sopenharmony_ci		      ((char *)tlv + sizeof(tlv->typelen) + len);
63862306a36Sopenharmony_ci	}
63962306a36Sopenharmony_ci
64062306a36Sopenharmony_ci	return ret;
64162306a36Sopenharmony_ci}
64262306a36Sopenharmony_ci
64362306a36Sopenharmony_ci/**
64462306a36Sopenharmony_ci * ice_aq_get_dcb_cfg
64562306a36Sopenharmony_ci * @hw: pointer to the HW struct
64662306a36Sopenharmony_ci * @mib_type: MIB type for the query
64762306a36Sopenharmony_ci * @bridgetype: bridge type for the query (remote)
64862306a36Sopenharmony_ci * @dcbcfg: store for LLDPDU data
64962306a36Sopenharmony_ci *
65062306a36Sopenharmony_ci * Query DCB configuration from the firmware
65162306a36Sopenharmony_ci */
65262306a36Sopenharmony_ciint
65362306a36Sopenharmony_ciice_aq_get_dcb_cfg(struct ice_hw *hw, u8 mib_type, u8 bridgetype,
65462306a36Sopenharmony_ci		   struct ice_dcbx_cfg *dcbcfg)
65562306a36Sopenharmony_ci{
65662306a36Sopenharmony_ci	u8 *lldpmib;
65762306a36Sopenharmony_ci	int ret;
65862306a36Sopenharmony_ci
65962306a36Sopenharmony_ci	/* Allocate the LLDPDU */
66062306a36Sopenharmony_ci	lldpmib = devm_kzalloc(ice_hw_to_dev(hw), ICE_LLDPDU_SIZE, GFP_KERNEL);
66162306a36Sopenharmony_ci	if (!lldpmib)
66262306a36Sopenharmony_ci		return -ENOMEM;
66362306a36Sopenharmony_ci
66462306a36Sopenharmony_ci	ret = ice_aq_get_lldp_mib(hw, bridgetype, mib_type, (void *)lldpmib,
66562306a36Sopenharmony_ci				  ICE_LLDPDU_SIZE, NULL, NULL, NULL);
66662306a36Sopenharmony_ci
66762306a36Sopenharmony_ci	if (!ret)
66862306a36Sopenharmony_ci		/* Parse LLDP MIB to get DCB configuration */
66962306a36Sopenharmony_ci		ret = ice_lldp_to_dcb_cfg(lldpmib, dcbcfg);
67062306a36Sopenharmony_ci
67162306a36Sopenharmony_ci	devm_kfree(ice_hw_to_dev(hw), lldpmib);
67262306a36Sopenharmony_ci
67362306a36Sopenharmony_ci	return ret;
67462306a36Sopenharmony_ci}
67562306a36Sopenharmony_ci
67662306a36Sopenharmony_ci/**
67762306a36Sopenharmony_ci * ice_aq_start_stop_dcbx - Start/Stop DCBX service in FW
67862306a36Sopenharmony_ci * @hw: pointer to the HW struct
67962306a36Sopenharmony_ci * @start_dcbx_agent: True if DCBX Agent needs to be started
68062306a36Sopenharmony_ci *		      False if DCBX Agent needs to be stopped
68162306a36Sopenharmony_ci * @dcbx_agent_status: FW indicates back the DCBX agent status
68262306a36Sopenharmony_ci *		       True if DCBX Agent is active
68362306a36Sopenharmony_ci *		       False if DCBX Agent is stopped
68462306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
68562306a36Sopenharmony_ci *
68662306a36Sopenharmony_ci * Start/Stop the embedded dcbx Agent. In case that this wrapper function
68762306a36Sopenharmony_ci * returns 0, caller will need to check if FW returns back the same
68862306a36Sopenharmony_ci * value as stated in dcbx_agent_status, and react accordingly. (0x0A09)
68962306a36Sopenharmony_ci */
69062306a36Sopenharmony_ciint
69162306a36Sopenharmony_ciice_aq_start_stop_dcbx(struct ice_hw *hw, bool start_dcbx_agent,
69262306a36Sopenharmony_ci		       bool *dcbx_agent_status, struct ice_sq_cd *cd)
69362306a36Sopenharmony_ci{
69462306a36Sopenharmony_ci	struct ice_aqc_lldp_stop_start_specific_agent *cmd;
69562306a36Sopenharmony_ci	struct ice_aq_desc desc;
69662306a36Sopenharmony_ci	u16 opcode;
69762306a36Sopenharmony_ci	int status;
69862306a36Sopenharmony_ci
69962306a36Sopenharmony_ci	cmd = &desc.params.lldp_agent_ctrl;
70062306a36Sopenharmony_ci
70162306a36Sopenharmony_ci	opcode = ice_aqc_opc_lldp_stop_start_specific_agent;
70262306a36Sopenharmony_ci
70362306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, opcode);
70462306a36Sopenharmony_ci
70562306a36Sopenharmony_ci	if (start_dcbx_agent)
70662306a36Sopenharmony_ci		cmd->command = ICE_AQC_START_STOP_AGENT_START_DCBX;
70762306a36Sopenharmony_ci
70862306a36Sopenharmony_ci	status = ice_aq_send_cmd(hw, &desc, NULL, 0, cd);
70962306a36Sopenharmony_ci
71062306a36Sopenharmony_ci	*dcbx_agent_status = false;
71162306a36Sopenharmony_ci
71262306a36Sopenharmony_ci	if (!status &&
71362306a36Sopenharmony_ci	    cmd->command == ICE_AQC_START_STOP_AGENT_START_DCBX)
71462306a36Sopenharmony_ci		*dcbx_agent_status = true;
71562306a36Sopenharmony_ci
71662306a36Sopenharmony_ci	return status;
71762306a36Sopenharmony_ci}
71862306a36Sopenharmony_ci
71962306a36Sopenharmony_ci/**
72062306a36Sopenharmony_ci * ice_aq_get_cee_dcb_cfg
72162306a36Sopenharmony_ci * @hw: pointer to the HW struct
72262306a36Sopenharmony_ci * @buff: response buffer that stores CEE operational configuration
72362306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
72462306a36Sopenharmony_ci *
72562306a36Sopenharmony_ci * Get CEE DCBX mode operational configuration from firmware (0x0A07)
72662306a36Sopenharmony_ci */
72762306a36Sopenharmony_cistatic int
72862306a36Sopenharmony_ciice_aq_get_cee_dcb_cfg(struct ice_hw *hw,
72962306a36Sopenharmony_ci		       struct ice_aqc_get_cee_dcb_cfg_resp *buff,
73062306a36Sopenharmony_ci		       struct ice_sq_cd *cd)
73162306a36Sopenharmony_ci{
73262306a36Sopenharmony_ci	struct ice_aq_desc desc;
73362306a36Sopenharmony_ci
73462306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_get_cee_dcb_cfg);
73562306a36Sopenharmony_ci
73662306a36Sopenharmony_ci	return ice_aq_send_cmd(hw, &desc, (void *)buff, sizeof(*buff), cd);
73762306a36Sopenharmony_ci}
73862306a36Sopenharmony_ci
73962306a36Sopenharmony_ci/**
74062306a36Sopenharmony_ci * ice_aq_set_pfc_mode - Set PFC mode
74162306a36Sopenharmony_ci * @hw: pointer to the HW struct
74262306a36Sopenharmony_ci * @pfc_mode: value of PFC mode to set
74362306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
74462306a36Sopenharmony_ci *
74562306a36Sopenharmony_ci * This AQ call configures the PFC mode to DSCP-based PFC mode or
74662306a36Sopenharmony_ci * VLAN-based PFC (0x0303)
74762306a36Sopenharmony_ci */
74862306a36Sopenharmony_ciint ice_aq_set_pfc_mode(struct ice_hw *hw, u8 pfc_mode, struct ice_sq_cd *cd)
74962306a36Sopenharmony_ci{
75062306a36Sopenharmony_ci	struct ice_aqc_set_query_pfc_mode *cmd;
75162306a36Sopenharmony_ci	struct ice_aq_desc desc;
75262306a36Sopenharmony_ci	int status;
75362306a36Sopenharmony_ci
75462306a36Sopenharmony_ci	if (pfc_mode > ICE_AQC_PFC_DSCP_BASED_PFC)
75562306a36Sopenharmony_ci		return -EINVAL;
75662306a36Sopenharmony_ci
75762306a36Sopenharmony_ci	cmd = &desc.params.set_query_pfc_mode;
75862306a36Sopenharmony_ci
75962306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_set_pfc_mode);
76062306a36Sopenharmony_ci
76162306a36Sopenharmony_ci	cmd->pfc_mode = pfc_mode;
76262306a36Sopenharmony_ci
76362306a36Sopenharmony_ci	status = ice_aq_send_cmd(hw, &desc, NULL, 0, cd);
76462306a36Sopenharmony_ci	if (status)
76562306a36Sopenharmony_ci		return status;
76662306a36Sopenharmony_ci
76762306a36Sopenharmony_ci	/* FW will write the PFC mode set back into cmd->pfc_mode, but if DCB is
76862306a36Sopenharmony_ci	 * disabled, FW will write back 0 to cmd->pfc_mode. After the AQ has
76962306a36Sopenharmony_ci	 * been executed, check if cmd->pfc_mode is what was requested. If not,
77062306a36Sopenharmony_ci	 * return an error.
77162306a36Sopenharmony_ci	 */
77262306a36Sopenharmony_ci	if (cmd->pfc_mode != pfc_mode)
77362306a36Sopenharmony_ci		return -EOPNOTSUPP;
77462306a36Sopenharmony_ci
77562306a36Sopenharmony_ci	return 0;
77662306a36Sopenharmony_ci}
77762306a36Sopenharmony_ci
77862306a36Sopenharmony_ci/**
77962306a36Sopenharmony_ci * ice_cee_to_dcb_cfg
78062306a36Sopenharmony_ci * @cee_cfg: pointer to CEE configuration struct
78162306a36Sopenharmony_ci * @pi: port information structure
78262306a36Sopenharmony_ci *
78362306a36Sopenharmony_ci * Convert CEE configuration from firmware to DCB configuration
78462306a36Sopenharmony_ci */
78562306a36Sopenharmony_cistatic void
78662306a36Sopenharmony_ciice_cee_to_dcb_cfg(struct ice_aqc_get_cee_dcb_cfg_resp *cee_cfg,
78762306a36Sopenharmony_ci		   struct ice_port_info *pi)
78862306a36Sopenharmony_ci{
78962306a36Sopenharmony_ci	u32 status, tlv_status = le32_to_cpu(cee_cfg->tlv_status);
79062306a36Sopenharmony_ci	u32 ice_aqc_cee_status_mask, ice_aqc_cee_status_shift, j;
79162306a36Sopenharmony_ci	u8 i, err, sync, oper, app_index, ice_app_sel_type;
79262306a36Sopenharmony_ci	u16 app_prio = le16_to_cpu(cee_cfg->oper_app_prio);
79362306a36Sopenharmony_ci	u16 ice_aqc_cee_app_mask, ice_aqc_cee_app_shift;
79462306a36Sopenharmony_ci	struct ice_dcbx_cfg *cmp_dcbcfg, *dcbcfg;
79562306a36Sopenharmony_ci	u16 ice_app_prot_id_type;
79662306a36Sopenharmony_ci
79762306a36Sopenharmony_ci	dcbcfg = &pi->qos_cfg.local_dcbx_cfg;
79862306a36Sopenharmony_ci	dcbcfg->dcbx_mode = ICE_DCBX_MODE_CEE;
79962306a36Sopenharmony_ci	dcbcfg->tlv_status = tlv_status;
80062306a36Sopenharmony_ci
80162306a36Sopenharmony_ci	/* CEE PG data */
80262306a36Sopenharmony_ci	dcbcfg->etscfg.maxtcs = cee_cfg->oper_num_tc;
80362306a36Sopenharmony_ci
80462306a36Sopenharmony_ci	/* Note that the FW creates the oper_prio_tc nibbles reversed
80562306a36Sopenharmony_ci	 * from those in the CEE Priority Group sub-TLV.
80662306a36Sopenharmony_ci	 */
80762306a36Sopenharmony_ci	for (i = 0; i < ICE_MAX_TRAFFIC_CLASS / 2; i++) {
80862306a36Sopenharmony_ci		dcbcfg->etscfg.prio_table[i * 2] =
80962306a36Sopenharmony_ci			((cee_cfg->oper_prio_tc[i] & ICE_CEE_PGID_PRIO_0_M) >>
81062306a36Sopenharmony_ci			 ICE_CEE_PGID_PRIO_0_S);
81162306a36Sopenharmony_ci		dcbcfg->etscfg.prio_table[i * 2 + 1] =
81262306a36Sopenharmony_ci			((cee_cfg->oper_prio_tc[i] & ICE_CEE_PGID_PRIO_1_M) >>
81362306a36Sopenharmony_ci			 ICE_CEE_PGID_PRIO_1_S);
81462306a36Sopenharmony_ci	}
81562306a36Sopenharmony_ci
81662306a36Sopenharmony_ci	ice_for_each_traffic_class(i) {
81762306a36Sopenharmony_ci		dcbcfg->etscfg.tcbwtable[i] = cee_cfg->oper_tc_bw[i];
81862306a36Sopenharmony_ci
81962306a36Sopenharmony_ci		if (dcbcfg->etscfg.prio_table[i] == ICE_CEE_PGID_STRICT) {
82062306a36Sopenharmony_ci			/* Map it to next empty TC */
82162306a36Sopenharmony_ci			dcbcfg->etscfg.prio_table[i] = cee_cfg->oper_num_tc - 1;
82262306a36Sopenharmony_ci			dcbcfg->etscfg.tsatable[i] = ICE_IEEE_TSA_STRICT;
82362306a36Sopenharmony_ci		} else {
82462306a36Sopenharmony_ci			dcbcfg->etscfg.tsatable[i] = ICE_IEEE_TSA_ETS;
82562306a36Sopenharmony_ci		}
82662306a36Sopenharmony_ci	}
82762306a36Sopenharmony_ci
82862306a36Sopenharmony_ci	/* CEE PFC data */
82962306a36Sopenharmony_ci	dcbcfg->pfc.pfcena = cee_cfg->oper_pfc_en;
83062306a36Sopenharmony_ci	dcbcfg->pfc.pfccap = ICE_MAX_TRAFFIC_CLASS;
83162306a36Sopenharmony_ci
83262306a36Sopenharmony_ci	/* CEE APP TLV data */
83362306a36Sopenharmony_ci	if (dcbcfg->app_mode == ICE_DCBX_APPS_NON_WILLING)
83462306a36Sopenharmony_ci		cmp_dcbcfg = &pi->qos_cfg.desired_dcbx_cfg;
83562306a36Sopenharmony_ci	else
83662306a36Sopenharmony_ci		cmp_dcbcfg = &pi->qos_cfg.remote_dcbx_cfg;
83762306a36Sopenharmony_ci
83862306a36Sopenharmony_ci	app_index = 0;
83962306a36Sopenharmony_ci	for (i = 0; i < 3; i++) {
84062306a36Sopenharmony_ci		if (i == 0) {
84162306a36Sopenharmony_ci			/* FCoE APP */
84262306a36Sopenharmony_ci			ice_aqc_cee_status_mask = ICE_AQC_CEE_FCOE_STATUS_M;
84362306a36Sopenharmony_ci			ice_aqc_cee_status_shift = ICE_AQC_CEE_FCOE_STATUS_S;
84462306a36Sopenharmony_ci			ice_aqc_cee_app_mask = ICE_AQC_CEE_APP_FCOE_M;
84562306a36Sopenharmony_ci			ice_aqc_cee_app_shift = ICE_AQC_CEE_APP_FCOE_S;
84662306a36Sopenharmony_ci			ice_app_sel_type = ICE_APP_SEL_ETHTYPE;
84762306a36Sopenharmony_ci			ice_app_prot_id_type = ETH_P_FCOE;
84862306a36Sopenharmony_ci		} else if (i == 1) {
84962306a36Sopenharmony_ci			/* iSCSI APP */
85062306a36Sopenharmony_ci			ice_aqc_cee_status_mask = ICE_AQC_CEE_ISCSI_STATUS_M;
85162306a36Sopenharmony_ci			ice_aqc_cee_status_shift = ICE_AQC_CEE_ISCSI_STATUS_S;
85262306a36Sopenharmony_ci			ice_aqc_cee_app_mask = ICE_AQC_CEE_APP_ISCSI_M;
85362306a36Sopenharmony_ci			ice_aqc_cee_app_shift = ICE_AQC_CEE_APP_ISCSI_S;
85462306a36Sopenharmony_ci			ice_app_sel_type = ICE_APP_SEL_TCPIP;
85562306a36Sopenharmony_ci			ice_app_prot_id_type = ISCSI_LISTEN_PORT;
85662306a36Sopenharmony_ci
85762306a36Sopenharmony_ci			for (j = 0; j < cmp_dcbcfg->numapps; j++) {
85862306a36Sopenharmony_ci				u16 prot_id = cmp_dcbcfg->app[j].prot_id;
85962306a36Sopenharmony_ci				u8 sel = cmp_dcbcfg->app[j].selector;
86062306a36Sopenharmony_ci
86162306a36Sopenharmony_ci				if  (sel == ICE_APP_SEL_TCPIP &&
86262306a36Sopenharmony_ci				     (prot_id == ISCSI_LISTEN_PORT ||
86362306a36Sopenharmony_ci				      prot_id == ICE_APP_PROT_ID_ISCSI_860)) {
86462306a36Sopenharmony_ci					ice_app_prot_id_type = prot_id;
86562306a36Sopenharmony_ci					break;
86662306a36Sopenharmony_ci				}
86762306a36Sopenharmony_ci			}
86862306a36Sopenharmony_ci		} else {
86962306a36Sopenharmony_ci			/* FIP APP */
87062306a36Sopenharmony_ci			ice_aqc_cee_status_mask = ICE_AQC_CEE_FIP_STATUS_M;
87162306a36Sopenharmony_ci			ice_aqc_cee_status_shift = ICE_AQC_CEE_FIP_STATUS_S;
87262306a36Sopenharmony_ci			ice_aqc_cee_app_mask = ICE_AQC_CEE_APP_FIP_M;
87362306a36Sopenharmony_ci			ice_aqc_cee_app_shift = ICE_AQC_CEE_APP_FIP_S;
87462306a36Sopenharmony_ci			ice_app_sel_type = ICE_APP_SEL_ETHTYPE;
87562306a36Sopenharmony_ci			ice_app_prot_id_type = ETH_P_FIP;
87662306a36Sopenharmony_ci		}
87762306a36Sopenharmony_ci
87862306a36Sopenharmony_ci		status = (tlv_status & ice_aqc_cee_status_mask) >>
87962306a36Sopenharmony_ci			 ice_aqc_cee_status_shift;
88062306a36Sopenharmony_ci		err = (status & ICE_TLV_STATUS_ERR) ? 1 : 0;
88162306a36Sopenharmony_ci		sync = (status & ICE_TLV_STATUS_SYNC) ? 1 : 0;
88262306a36Sopenharmony_ci		oper = (status & ICE_TLV_STATUS_OPER) ? 1 : 0;
88362306a36Sopenharmony_ci		/* Add FCoE/iSCSI/FIP APP if Error is False and
88462306a36Sopenharmony_ci		 * Oper/Sync is True
88562306a36Sopenharmony_ci		 */
88662306a36Sopenharmony_ci		if (!err && sync && oper) {
88762306a36Sopenharmony_ci			dcbcfg->app[app_index].priority =
88862306a36Sopenharmony_ci				(app_prio & ice_aqc_cee_app_mask) >>
88962306a36Sopenharmony_ci				ice_aqc_cee_app_shift;
89062306a36Sopenharmony_ci			dcbcfg->app[app_index].selector = ice_app_sel_type;
89162306a36Sopenharmony_ci			dcbcfg->app[app_index].prot_id = ice_app_prot_id_type;
89262306a36Sopenharmony_ci			app_index++;
89362306a36Sopenharmony_ci		}
89462306a36Sopenharmony_ci	}
89562306a36Sopenharmony_ci
89662306a36Sopenharmony_ci	dcbcfg->numapps = app_index;
89762306a36Sopenharmony_ci}
89862306a36Sopenharmony_ci
89962306a36Sopenharmony_ci/**
90062306a36Sopenharmony_ci * ice_get_ieee_or_cee_dcb_cfg
90162306a36Sopenharmony_ci * @pi: port information structure
90262306a36Sopenharmony_ci * @dcbx_mode: mode of DCBX (IEEE or CEE)
90362306a36Sopenharmony_ci *
90462306a36Sopenharmony_ci * Get IEEE or CEE mode DCB configuration from the Firmware
90562306a36Sopenharmony_ci */
90662306a36Sopenharmony_cistatic int ice_get_ieee_or_cee_dcb_cfg(struct ice_port_info *pi, u8 dcbx_mode)
90762306a36Sopenharmony_ci{
90862306a36Sopenharmony_ci	struct ice_dcbx_cfg *dcbx_cfg = NULL;
90962306a36Sopenharmony_ci	int ret;
91062306a36Sopenharmony_ci
91162306a36Sopenharmony_ci	if (!pi)
91262306a36Sopenharmony_ci		return -EINVAL;
91362306a36Sopenharmony_ci
91462306a36Sopenharmony_ci	if (dcbx_mode == ICE_DCBX_MODE_IEEE)
91562306a36Sopenharmony_ci		dcbx_cfg = &pi->qos_cfg.local_dcbx_cfg;
91662306a36Sopenharmony_ci	else if (dcbx_mode == ICE_DCBX_MODE_CEE)
91762306a36Sopenharmony_ci		dcbx_cfg = &pi->qos_cfg.desired_dcbx_cfg;
91862306a36Sopenharmony_ci
91962306a36Sopenharmony_ci	/* Get Local DCB Config in case of ICE_DCBX_MODE_IEEE
92062306a36Sopenharmony_ci	 * or get CEE DCB Desired Config in case of ICE_DCBX_MODE_CEE
92162306a36Sopenharmony_ci	 */
92262306a36Sopenharmony_ci	ret = ice_aq_get_dcb_cfg(pi->hw, ICE_AQ_LLDP_MIB_LOCAL,
92362306a36Sopenharmony_ci				 ICE_AQ_LLDP_BRID_TYPE_NEAREST_BRID, dcbx_cfg);
92462306a36Sopenharmony_ci	if (ret)
92562306a36Sopenharmony_ci		goto out;
92662306a36Sopenharmony_ci
92762306a36Sopenharmony_ci	/* Get Remote DCB Config */
92862306a36Sopenharmony_ci	dcbx_cfg = &pi->qos_cfg.remote_dcbx_cfg;
92962306a36Sopenharmony_ci	ret = ice_aq_get_dcb_cfg(pi->hw, ICE_AQ_LLDP_MIB_REMOTE,
93062306a36Sopenharmony_ci				 ICE_AQ_LLDP_BRID_TYPE_NEAREST_BRID, dcbx_cfg);
93162306a36Sopenharmony_ci	/* Don't treat ENOENT as an error for Remote MIBs */
93262306a36Sopenharmony_ci	if (pi->hw->adminq.sq_last_status == ICE_AQ_RC_ENOENT)
93362306a36Sopenharmony_ci		ret = 0;
93462306a36Sopenharmony_ci
93562306a36Sopenharmony_ciout:
93662306a36Sopenharmony_ci	return ret;
93762306a36Sopenharmony_ci}
93862306a36Sopenharmony_ci
93962306a36Sopenharmony_ci/**
94062306a36Sopenharmony_ci * ice_get_dcb_cfg
94162306a36Sopenharmony_ci * @pi: port information structure
94262306a36Sopenharmony_ci *
94362306a36Sopenharmony_ci * Get DCB configuration from the Firmware
94462306a36Sopenharmony_ci */
94562306a36Sopenharmony_ciint ice_get_dcb_cfg(struct ice_port_info *pi)
94662306a36Sopenharmony_ci{
94762306a36Sopenharmony_ci	struct ice_aqc_get_cee_dcb_cfg_resp cee_cfg;
94862306a36Sopenharmony_ci	struct ice_dcbx_cfg *dcbx_cfg;
94962306a36Sopenharmony_ci	int ret;
95062306a36Sopenharmony_ci
95162306a36Sopenharmony_ci	if (!pi)
95262306a36Sopenharmony_ci		return -EINVAL;
95362306a36Sopenharmony_ci
95462306a36Sopenharmony_ci	ret = ice_aq_get_cee_dcb_cfg(pi->hw, &cee_cfg, NULL);
95562306a36Sopenharmony_ci	if (!ret) {
95662306a36Sopenharmony_ci		/* CEE mode */
95762306a36Sopenharmony_ci		ret = ice_get_ieee_or_cee_dcb_cfg(pi, ICE_DCBX_MODE_CEE);
95862306a36Sopenharmony_ci		ice_cee_to_dcb_cfg(&cee_cfg, pi);
95962306a36Sopenharmony_ci	} else if (pi->hw->adminq.sq_last_status == ICE_AQ_RC_ENOENT) {
96062306a36Sopenharmony_ci		/* CEE mode not enabled try querying IEEE data */
96162306a36Sopenharmony_ci		dcbx_cfg = &pi->qos_cfg.local_dcbx_cfg;
96262306a36Sopenharmony_ci		dcbx_cfg->dcbx_mode = ICE_DCBX_MODE_IEEE;
96362306a36Sopenharmony_ci		ret = ice_get_ieee_or_cee_dcb_cfg(pi, ICE_DCBX_MODE_IEEE);
96462306a36Sopenharmony_ci	}
96562306a36Sopenharmony_ci
96662306a36Sopenharmony_ci	return ret;
96762306a36Sopenharmony_ci}
96862306a36Sopenharmony_ci
96962306a36Sopenharmony_ci/**
97062306a36Sopenharmony_ci * ice_get_dcb_cfg_from_mib_change
97162306a36Sopenharmony_ci * @pi: port information structure
97262306a36Sopenharmony_ci * @event: pointer to the admin queue receive event
97362306a36Sopenharmony_ci *
97462306a36Sopenharmony_ci * Set DCB configuration from received MIB Change event
97562306a36Sopenharmony_ci */
97662306a36Sopenharmony_civoid ice_get_dcb_cfg_from_mib_change(struct ice_port_info *pi,
97762306a36Sopenharmony_ci				     struct ice_rq_event_info *event)
97862306a36Sopenharmony_ci{
97962306a36Sopenharmony_ci	struct ice_dcbx_cfg *dcbx_cfg = &pi->qos_cfg.local_dcbx_cfg;
98062306a36Sopenharmony_ci	struct ice_aqc_lldp_get_mib *mib;
98162306a36Sopenharmony_ci	u8 change_type, dcbx_mode;
98262306a36Sopenharmony_ci
98362306a36Sopenharmony_ci	mib = (struct ice_aqc_lldp_get_mib *)&event->desc.params.raw;
98462306a36Sopenharmony_ci
98562306a36Sopenharmony_ci	change_type = FIELD_GET(ICE_AQ_LLDP_MIB_TYPE_M,  mib->type);
98662306a36Sopenharmony_ci	if (change_type == ICE_AQ_LLDP_MIB_REMOTE)
98762306a36Sopenharmony_ci		dcbx_cfg = &pi->qos_cfg.remote_dcbx_cfg;
98862306a36Sopenharmony_ci
98962306a36Sopenharmony_ci	dcbx_mode = FIELD_GET(ICE_AQ_LLDP_DCBX_M, mib->type);
99062306a36Sopenharmony_ci
99162306a36Sopenharmony_ci	switch (dcbx_mode) {
99262306a36Sopenharmony_ci	case ICE_AQ_LLDP_DCBX_IEEE:
99362306a36Sopenharmony_ci		dcbx_cfg->dcbx_mode = ICE_DCBX_MODE_IEEE;
99462306a36Sopenharmony_ci		ice_lldp_to_dcb_cfg(event->msg_buf, dcbx_cfg);
99562306a36Sopenharmony_ci		break;
99662306a36Sopenharmony_ci
99762306a36Sopenharmony_ci	case ICE_AQ_LLDP_DCBX_CEE:
99862306a36Sopenharmony_ci		pi->qos_cfg.desired_dcbx_cfg = pi->qos_cfg.local_dcbx_cfg;
99962306a36Sopenharmony_ci		ice_cee_to_dcb_cfg((struct ice_aqc_get_cee_dcb_cfg_resp *)
100062306a36Sopenharmony_ci				   event->msg_buf, pi);
100162306a36Sopenharmony_ci		break;
100262306a36Sopenharmony_ci	}
100362306a36Sopenharmony_ci}
100462306a36Sopenharmony_ci
100562306a36Sopenharmony_ci/**
100662306a36Sopenharmony_ci * ice_init_dcb
100762306a36Sopenharmony_ci * @hw: pointer to the HW struct
100862306a36Sopenharmony_ci * @enable_mib_change: enable MIB change event
100962306a36Sopenharmony_ci *
101062306a36Sopenharmony_ci * Update DCB configuration from the Firmware
101162306a36Sopenharmony_ci */
101262306a36Sopenharmony_ciint ice_init_dcb(struct ice_hw *hw, bool enable_mib_change)
101362306a36Sopenharmony_ci{
101462306a36Sopenharmony_ci	struct ice_qos_cfg *qos_cfg = &hw->port_info->qos_cfg;
101562306a36Sopenharmony_ci	int ret = 0;
101662306a36Sopenharmony_ci
101762306a36Sopenharmony_ci	if (!hw->func_caps.common_cap.dcb)
101862306a36Sopenharmony_ci		return -EOPNOTSUPP;
101962306a36Sopenharmony_ci
102062306a36Sopenharmony_ci	qos_cfg->is_sw_lldp = true;
102162306a36Sopenharmony_ci
102262306a36Sopenharmony_ci	/* Get DCBX status */
102362306a36Sopenharmony_ci	qos_cfg->dcbx_status = ice_get_dcbx_status(hw);
102462306a36Sopenharmony_ci
102562306a36Sopenharmony_ci	if (qos_cfg->dcbx_status == ICE_DCBX_STATUS_DONE ||
102662306a36Sopenharmony_ci	    qos_cfg->dcbx_status == ICE_DCBX_STATUS_IN_PROGRESS ||
102762306a36Sopenharmony_ci	    qos_cfg->dcbx_status == ICE_DCBX_STATUS_NOT_STARTED) {
102862306a36Sopenharmony_ci		/* Get current DCBX configuration */
102962306a36Sopenharmony_ci		ret = ice_get_dcb_cfg(hw->port_info);
103062306a36Sopenharmony_ci		if (ret)
103162306a36Sopenharmony_ci			return ret;
103262306a36Sopenharmony_ci		qos_cfg->is_sw_lldp = false;
103362306a36Sopenharmony_ci	} else if (qos_cfg->dcbx_status == ICE_DCBX_STATUS_DIS) {
103462306a36Sopenharmony_ci		return -EBUSY;
103562306a36Sopenharmony_ci	}
103662306a36Sopenharmony_ci
103762306a36Sopenharmony_ci	/* Configure the LLDP MIB change event */
103862306a36Sopenharmony_ci	if (enable_mib_change) {
103962306a36Sopenharmony_ci		ret = ice_aq_cfg_lldp_mib_change(hw, true, NULL);
104062306a36Sopenharmony_ci		if (ret)
104162306a36Sopenharmony_ci			qos_cfg->is_sw_lldp = true;
104262306a36Sopenharmony_ci	}
104362306a36Sopenharmony_ci
104462306a36Sopenharmony_ci	return ret;
104562306a36Sopenharmony_ci}
104662306a36Sopenharmony_ci
104762306a36Sopenharmony_ci/**
104862306a36Sopenharmony_ci * ice_cfg_lldp_mib_change
104962306a36Sopenharmony_ci * @hw: pointer to the HW struct
105062306a36Sopenharmony_ci * @ena_mib: enable/disable MIB change event
105162306a36Sopenharmony_ci *
105262306a36Sopenharmony_ci * Configure (disable/enable) MIB
105362306a36Sopenharmony_ci */
105462306a36Sopenharmony_ciint ice_cfg_lldp_mib_change(struct ice_hw *hw, bool ena_mib)
105562306a36Sopenharmony_ci{
105662306a36Sopenharmony_ci	struct ice_qos_cfg *qos_cfg = &hw->port_info->qos_cfg;
105762306a36Sopenharmony_ci	int ret;
105862306a36Sopenharmony_ci
105962306a36Sopenharmony_ci	if (!hw->func_caps.common_cap.dcb)
106062306a36Sopenharmony_ci		return -EOPNOTSUPP;
106162306a36Sopenharmony_ci
106262306a36Sopenharmony_ci	/* Get DCBX status */
106362306a36Sopenharmony_ci	qos_cfg->dcbx_status = ice_get_dcbx_status(hw);
106462306a36Sopenharmony_ci
106562306a36Sopenharmony_ci	if (qos_cfg->dcbx_status == ICE_DCBX_STATUS_DIS)
106662306a36Sopenharmony_ci		return -EBUSY;
106762306a36Sopenharmony_ci
106862306a36Sopenharmony_ci	ret = ice_aq_cfg_lldp_mib_change(hw, ena_mib, NULL);
106962306a36Sopenharmony_ci	if (!ret)
107062306a36Sopenharmony_ci		qos_cfg->is_sw_lldp = !ena_mib;
107162306a36Sopenharmony_ci
107262306a36Sopenharmony_ci	return ret;
107362306a36Sopenharmony_ci}
107462306a36Sopenharmony_ci
107562306a36Sopenharmony_ci/**
107662306a36Sopenharmony_ci * ice_add_ieee_ets_common_tlv
107762306a36Sopenharmony_ci * @buf: Data buffer to be populated with ice_dcb_ets_cfg data
107862306a36Sopenharmony_ci * @ets_cfg: Container for ice_dcb_ets_cfg data
107962306a36Sopenharmony_ci *
108062306a36Sopenharmony_ci * Populate the TLV buffer with ice_dcb_ets_cfg data
108162306a36Sopenharmony_ci */
108262306a36Sopenharmony_cistatic void
108362306a36Sopenharmony_ciice_add_ieee_ets_common_tlv(u8 *buf, struct ice_dcb_ets_cfg *ets_cfg)
108462306a36Sopenharmony_ci{
108562306a36Sopenharmony_ci	u8 priority0, priority1;
108662306a36Sopenharmony_ci	u8 offset = 0;
108762306a36Sopenharmony_ci	int i;
108862306a36Sopenharmony_ci
108962306a36Sopenharmony_ci	/* Priority Assignment Table (4 octets)
109062306a36Sopenharmony_ci	 * Octets:|    1    |    2    |    3    |    4    |
109162306a36Sopenharmony_ci	 *        -----------------------------------------
109262306a36Sopenharmony_ci	 *        |pri0|pri1|pri2|pri3|pri4|pri5|pri6|pri7|
109362306a36Sopenharmony_ci	 *        -----------------------------------------
109462306a36Sopenharmony_ci	 *   Bits:|7  4|3  0|7  4|3  0|7  4|3  0|7  4|3  0|
109562306a36Sopenharmony_ci	 *        -----------------------------------------
109662306a36Sopenharmony_ci	 */
109762306a36Sopenharmony_ci	for (i = 0; i < ICE_MAX_TRAFFIC_CLASS / 2; i++) {
109862306a36Sopenharmony_ci		priority0 = ets_cfg->prio_table[i * 2] & 0xF;
109962306a36Sopenharmony_ci		priority1 = ets_cfg->prio_table[i * 2 + 1] & 0xF;
110062306a36Sopenharmony_ci		buf[offset] = (priority0 << ICE_IEEE_ETS_PRIO_1_S) | priority1;
110162306a36Sopenharmony_ci		offset++;
110262306a36Sopenharmony_ci	}
110362306a36Sopenharmony_ci
110462306a36Sopenharmony_ci	/* TC Bandwidth Table (8 octets)
110562306a36Sopenharmony_ci	 * Octets:| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
110662306a36Sopenharmony_ci	 *        ---------------------------------
110762306a36Sopenharmony_ci	 *        |tc0|tc1|tc2|tc3|tc4|tc5|tc6|tc7|
110862306a36Sopenharmony_ci	 *        ---------------------------------
110962306a36Sopenharmony_ci	 *
111062306a36Sopenharmony_ci	 * TSA Assignment Table (8 octets)
111162306a36Sopenharmony_ci	 * Octets:| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
111262306a36Sopenharmony_ci	 *        ---------------------------------
111362306a36Sopenharmony_ci	 *        |tc0|tc1|tc2|tc3|tc4|tc5|tc6|tc7|
111462306a36Sopenharmony_ci	 *        ---------------------------------
111562306a36Sopenharmony_ci	 */
111662306a36Sopenharmony_ci	ice_for_each_traffic_class(i) {
111762306a36Sopenharmony_ci		buf[offset] = ets_cfg->tcbwtable[i];
111862306a36Sopenharmony_ci		buf[ICE_MAX_TRAFFIC_CLASS + offset] = ets_cfg->tsatable[i];
111962306a36Sopenharmony_ci		offset++;
112062306a36Sopenharmony_ci	}
112162306a36Sopenharmony_ci}
112262306a36Sopenharmony_ci
112362306a36Sopenharmony_ci/**
112462306a36Sopenharmony_ci * ice_add_ieee_ets_tlv - Prepare ETS TLV in IEEE format
112562306a36Sopenharmony_ci * @tlv: Fill the ETS config data in IEEE format
112662306a36Sopenharmony_ci * @dcbcfg: Local store which holds the DCB Config
112762306a36Sopenharmony_ci *
112862306a36Sopenharmony_ci * Prepare IEEE 802.1Qaz ETS CFG TLV
112962306a36Sopenharmony_ci */
113062306a36Sopenharmony_cistatic void
113162306a36Sopenharmony_ciice_add_ieee_ets_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
113262306a36Sopenharmony_ci{
113362306a36Sopenharmony_ci	struct ice_dcb_ets_cfg *etscfg;
113462306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
113562306a36Sopenharmony_ci	u8 maxtcwilling = 0;
113662306a36Sopenharmony_ci	u32 ouisubtype;
113762306a36Sopenharmony_ci	u16 typelen;
113862306a36Sopenharmony_ci
113962306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
114062306a36Sopenharmony_ci		   ICE_IEEE_ETS_TLV_LEN);
114162306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
114262306a36Sopenharmony_ci
114362306a36Sopenharmony_ci	ouisubtype = ((ICE_IEEE_8021QAZ_OUI << ICE_LLDP_TLV_OUI_S) |
114462306a36Sopenharmony_ci		      ICE_IEEE_SUBTYPE_ETS_CFG);
114562306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
114662306a36Sopenharmony_ci
114762306a36Sopenharmony_ci	/* First Octet post subtype
114862306a36Sopenharmony_ci	 * --------------------------
114962306a36Sopenharmony_ci	 * |will-|CBS  | Re-  | Max |
115062306a36Sopenharmony_ci	 * |ing  |     |served| TCs |
115162306a36Sopenharmony_ci	 * --------------------------
115262306a36Sopenharmony_ci	 * |1bit | 1bit|3 bits|3bits|
115362306a36Sopenharmony_ci	 */
115462306a36Sopenharmony_ci	etscfg = &dcbcfg->etscfg;
115562306a36Sopenharmony_ci	if (etscfg->willing)
115662306a36Sopenharmony_ci		maxtcwilling = BIT(ICE_IEEE_ETS_WILLING_S);
115762306a36Sopenharmony_ci	maxtcwilling |= etscfg->maxtcs & ICE_IEEE_ETS_MAXTC_M;
115862306a36Sopenharmony_ci	buf[0] = maxtcwilling;
115962306a36Sopenharmony_ci
116062306a36Sopenharmony_ci	/* Begin adding at Priority Assignment Table (offset 1 in buf) */
116162306a36Sopenharmony_ci	ice_add_ieee_ets_common_tlv(&buf[1], etscfg);
116262306a36Sopenharmony_ci}
116362306a36Sopenharmony_ci
116462306a36Sopenharmony_ci/**
116562306a36Sopenharmony_ci * ice_add_ieee_etsrec_tlv - Prepare ETS Recommended TLV in IEEE format
116662306a36Sopenharmony_ci * @tlv: Fill ETS Recommended TLV in IEEE format
116762306a36Sopenharmony_ci * @dcbcfg: Local store which holds the DCB Config
116862306a36Sopenharmony_ci *
116962306a36Sopenharmony_ci * Prepare IEEE 802.1Qaz ETS REC TLV
117062306a36Sopenharmony_ci */
117162306a36Sopenharmony_cistatic void
117262306a36Sopenharmony_ciice_add_ieee_etsrec_tlv(struct ice_lldp_org_tlv *tlv,
117362306a36Sopenharmony_ci			struct ice_dcbx_cfg *dcbcfg)
117462306a36Sopenharmony_ci{
117562306a36Sopenharmony_ci	struct ice_dcb_ets_cfg *etsrec;
117662306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
117762306a36Sopenharmony_ci	u32 ouisubtype;
117862306a36Sopenharmony_ci	u16 typelen;
117962306a36Sopenharmony_ci
118062306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
118162306a36Sopenharmony_ci		   ICE_IEEE_ETS_TLV_LEN);
118262306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
118362306a36Sopenharmony_ci
118462306a36Sopenharmony_ci	ouisubtype = ((ICE_IEEE_8021QAZ_OUI << ICE_LLDP_TLV_OUI_S) |
118562306a36Sopenharmony_ci		      ICE_IEEE_SUBTYPE_ETS_REC);
118662306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
118762306a36Sopenharmony_ci
118862306a36Sopenharmony_ci	etsrec = &dcbcfg->etsrec;
118962306a36Sopenharmony_ci
119062306a36Sopenharmony_ci	/* First Octet is reserved */
119162306a36Sopenharmony_ci	/* Begin adding at Priority Assignment Table (offset 1 in buf) */
119262306a36Sopenharmony_ci	ice_add_ieee_ets_common_tlv(&buf[1], etsrec);
119362306a36Sopenharmony_ci}
119462306a36Sopenharmony_ci
119562306a36Sopenharmony_ci/**
119662306a36Sopenharmony_ci * ice_add_ieee_pfc_tlv - Prepare PFC TLV in IEEE format
119762306a36Sopenharmony_ci * @tlv: Fill PFC TLV in IEEE format
119862306a36Sopenharmony_ci * @dcbcfg: Local store which holds the PFC CFG data
119962306a36Sopenharmony_ci *
120062306a36Sopenharmony_ci * Prepare IEEE 802.1Qaz PFC CFG TLV
120162306a36Sopenharmony_ci */
120262306a36Sopenharmony_cistatic void
120362306a36Sopenharmony_ciice_add_ieee_pfc_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
120462306a36Sopenharmony_ci{
120562306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
120662306a36Sopenharmony_ci	u32 ouisubtype;
120762306a36Sopenharmony_ci	u16 typelen;
120862306a36Sopenharmony_ci
120962306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
121062306a36Sopenharmony_ci		   ICE_IEEE_PFC_TLV_LEN);
121162306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
121262306a36Sopenharmony_ci
121362306a36Sopenharmony_ci	ouisubtype = ((ICE_IEEE_8021QAZ_OUI << ICE_LLDP_TLV_OUI_S) |
121462306a36Sopenharmony_ci		      ICE_IEEE_SUBTYPE_PFC_CFG);
121562306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
121662306a36Sopenharmony_ci
121762306a36Sopenharmony_ci	/* ----------------------------------------
121862306a36Sopenharmony_ci	 * |will-|MBC  | Re-  | PFC |  PFC Enable  |
121962306a36Sopenharmony_ci	 * |ing  |     |served| cap |              |
122062306a36Sopenharmony_ci	 * -----------------------------------------
122162306a36Sopenharmony_ci	 * |1bit | 1bit|2 bits|4bits| 1 octet      |
122262306a36Sopenharmony_ci	 */
122362306a36Sopenharmony_ci	if (dcbcfg->pfc.willing)
122462306a36Sopenharmony_ci		buf[0] = BIT(ICE_IEEE_PFC_WILLING_S);
122562306a36Sopenharmony_ci
122662306a36Sopenharmony_ci	if (dcbcfg->pfc.mbc)
122762306a36Sopenharmony_ci		buf[0] |= BIT(ICE_IEEE_PFC_MBC_S);
122862306a36Sopenharmony_ci
122962306a36Sopenharmony_ci	buf[0] |= dcbcfg->pfc.pfccap & 0xF;
123062306a36Sopenharmony_ci	buf[1] = dcbcfg->pfc.pfcena;
123162306a36Sopenharmony_ci}
123262306a36Sopenharmony_ci
123362306a36Sopenharmony_ci/**
123462306a36Sopenharmony_ci * ice_add_ieee_app_pri_tlv -  Prepare APP TLV in IEEE format
123562306a36Sopenharmony_ci * @tlv: Fill APP TLV in IEEE format
123662306a36Sopenharmony_ci * @dcbcfg: Local store which holds the APP CFG data
123762306a36Sopenharmony_ci *
123862306a36Sopenharmony_ci * Prepare IEEE 802.1Qaz APP CFG TLV
123962306a36Sopenharmony_ci */
124062306a36Sopenharmony_cistatic void
124162306a36Sopenharmony_ciice_add_ieee_app_pri_tlv(struct ice_lldp_org_tlv *tlv,
124262306a36Sopenharmony_ci			 struct ice_dcbx_cfg *dcbcfg)
124362306a36Sopenharmony_ci{
124462306a36Sopenharmony_ci	u16 typelen, len, offset = 0;
124562306a36Sopenharmony_ci	u8 priority, selector, i = 0;
124662306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
124762306a36Sopenharmony_ci	u32 ouisubtype;
124862306a36Sopenharmony_ci
124962306a36Sopenharmony_ci	/* No APP TLVs then just return */
125062306a36Sopenharmony_ci	if (dcbcfg->numapps == 0)
125162306a36Sopenharmony_ci		return;
125262306a36Sopenharmony_ci	ouisubtype = ((ICE_IEEE_8021QAZ_OUI << ICE_LLDP_TLV_OUI_S) |
125362306a36Sopenharmony_ci		      ICE_IEEE_SUBTYPE_APP_PRI);
125462306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
125562306a36Sopenharmony_ci
125662306a36Sopenharmony_ci	/* Move offset to App Priority Table */
125762306a36Sopenharmony_ci	offset++;
125862306a36Sopenharmony_ci	/* Application Priority Table (3 octets)
125962306a36Sopenharmony_ci	 * Octets:|         1          |    2    |    3    |
126062306a36Sopenharmony_ci	 *        -----------------------------------------
126162306a36Sopenharmony_ci	 *        |Priority|Rsrvd| Sel |    Protocol ID    |
126262306a36Sopenharmony_ci	 *        -----------------------------------------
126362306a36Sopenharmony_ci	 *   Bits:|23    21|20 19|18 16|15                0|
126462306a36Sopenharmony_ci	 *        -----------------------------------------
126562306a36Sopenharmony_ci	 */
126662306a36Sopenharmony_ci	while (i < dcbcfg->numapps) {
126762306a36Sopenharmony_ci		priority = dcbcfg->app[i].priority & 0x7;
126862306a36Sopenharmony_ci		selector = dcbcfg->app[i].selector & 0x7;
126962306a36Sopenharmony_ci		buf[offset] = (priority << ICE_IEEE_APP_PRIO_S) | selector;
127062306a36Sopenharmony_ci		buf[offset + 1] = (dcbcfg->app[i].prot_id >> 0x8) & 0xFF;
127162306a36Sopenharmony_ci		buf[offset + 2] = dcbcfg->app[i].prot_id & 0xFF;
127262306a36Sopenharmony_ci		/* Move to next app */
127362306a36Sopenharmony_ci		offset += 3;
127462306a36Sopenharmony_ci		i++;
127562306a36Sopenharmony_ci		if (i >= ICE_DCBX_MAX_APPS)
127662306a36Sopenharmony_ci			break;
127762306a36Sopenharmony_ci	}
127862306a36Sopenharmony_ci	/* len includes size of ouisubtype + 1 reserved + 3*numapps */
127962306a36Sopenharmony_ci	len = sizeof(tlv->ouisubtype) + 1 + (i * 3);
128062306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) | (len & 0x1FF));
128162306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
128262306a36Sopenharmony_ci}
128362306a36Sopenharmony_ci
128462306a36Sopenharmony_ci/**
128562306a36Sopenharmony_ci * ice_add_dscp_up_tlv - Prepare DSCP to UP TLV
128662306a36Sopenharmony_ci * @tlv: location to build the TLV data
128762306a36Sopenharmony_ci * @dcbcfg: location of data to convert to TLV
128862306a36Sopenharmony_ci */
128962306a36Sopenharmony_cistatic void
129062306a36Sopenharmony_ciice_add_dscp_up_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
129162306a36Sopenharmony_ci{
129262306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
129362306a36Sopenharmony_ci	u32 ouisubtype;
129462306a36Sopenharmony_ci	u16 typelen;
129562306a36Sopenharmony_ci	int i;
129662306a36Sopenharmony_ci
129762306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
129862306a36Sopenharmony_ci		   ICE_DSCP_UP_TLV_LEN);
129962306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
130062306a36Sopenharmony_ci
130162306a36Sopenharmony_ci	ouisubtype = (u32)((ICE_DSCP_OUI << ICE_LLDP_TLV_OUI_S) |
130262306a36Sopenharmony_ci			   ICE_DSCP_SUBTYPE_DSCP2UP);
130362306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
130462306a36Sopenharmony_ci
130562306a36Sopenharmony_ci	/* bytes 0 - 63 - IPv4 DSCP2UP LUT */
130662306a36Sopenharmony_ci	for (i = 0; i < ICE_DSCP_NUM_VAL; i++) {
130762306a36Sopenharmony_ci		/* IPv4 mapping */
130862306a36Sopenharmony_ci		buf[i] = dcbcfg->dscp_map[i];
130962306a36Sopenharmony_ci		/* IPv6 mapping */
131062306a36Sopenharmony_ci		buf[i + ICE_DSCP_IPV6_OFFSET] = dcbcfg->dscp_map[i];
131162306a36Sopenharmony_ci	}
131262306a36Sopenharmony_ci
131362306a36Sopenharmony_ci	/* byte 64 - IPv4 untagged traffic */
131462306a36Sopenharmony_ci	buf[i] = 0;
131562306a36Sopenharmony_ci
131662306a36Sopenharmony_ci	/* byte 144 - IPv6 untagged traffic */
131762306a36Sopenharmony_ci	buf[i + ICE_DSCP_IPV6_OFFSET] = 0;
131862306a36Sopenharmony_ci}
131962306a36Sopenharmony_ci
132062306a36Sopenharmony_ci#define ICE_BYTES_PER_TC	8
132162306a36Sopenharmony_ci/**
132262306a36Sopenharmony_ci * ice_add_dscp_enf_tlv - Prepare DSCP Enforcement TLV
132362306a36Sopenharmony_ci * @tlv: location to build the TLV data
132462306a36Sopenharmony_ci */
132562306a36Sopenharmony_cistatic void
132662306a36Sopenharmony_ciice_add_dscp_enf_tlv(struct ice_lldp_org_tlv *tlv)
132762306a36Sopenharmony_ci{
132862306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
132962306a36Sopenharmony_ci	u32 ouisubtype;
133062306a36Sopenharmony_ci	u16 typelen;
133162306a36Sopenharmony_ci
133262306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
133362306a36Sopenharmony_ci		   ICE_DSCP_ENF_TLV_LEN);
133462306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
133562306a36Sopenharmony_ci
133662306a36Sopenharmony_ci	ouisubtype = (u32)((ICE_DSCP_OUI << ICE_LLDP_TLV_OUI_S) |
133762306a36Sopenharmony_ci			   ICE_DSCP_SUBTYPE_ENFORCE);
133862306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
133962306a36Sopenharmony_ci
134062306a36Sopenharmony_ci	/* Allow all DSCP values to be valid for all TC's (IPv4 and IPv6) */
134162306a36Sopenharmony_ci	memset(buf, 0, 2 * (ICE_MAX_TRAFFIC_CLASS * ICE_BYTES_PER_TC));
134262306a36Sopenharmony_ci}
134362306a36Sopenharmony_ci
134462306a36Sopenharmony_ci/**
134562306a36Sopenharmony_ci * ice_add_dscp_tc_bw_tlv - Prepare DSCP BW for TC TLV
134662306a36Sopenharmony_ci * @tlv: location to build the TLV data
134762306a36Sopenharmony_ci * @dcbcfg: location of the data to convert to TLV
134862306a36Sopenharmony_ci */
134962306a36Sopenharmony_cistatic void
135062306a36Sopenharmony_ciice_add_dscp_tc_bw_tlv(struct ice_lldp_org_tlv *tlv,
135162306a36Sopenharmony_ci		       struct ice_dcbx_cfg *dcbcfg)
135262306a36Sopenharmony_ci{
135362306a36Sopenharmony_ci	struct ice_dcb_ets_cfg *etscfg;
135462306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
135562306a36Sopenharmony_ci	u32 ouisubtype;
135662306a36Sopenharmony_ci	u8 offset = 0;
135762306a36Sopenharmony_ci	u16 typelen;
135862306a36Sopenharmony_ci	int i;
135962306a36Sopenharmony_ci
136062306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
136162306a36Sopenharmony_ci		   ICE_DSCP_TC_BW_TLV_LEN);
136262306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
136362306a36Sopenharmony_ci
136462306a36Sopenharmony_ci	ouisubtype = (u32)((ICE_DSCP_OUI << ICE_LLDP_TLV_OUI_S) |
136562306a36Sopenharmony_ci			   ICE_DSCP_SUBTYPE_TCBW);
136662306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
136762306a36Sopenharmony_ci
136862306a36Sopenharmony_ci	/* First Octect after subtype
136962306a36Sopenharmony_ci	 * ----------------------------
137062306a36Sopenharmony_ci	 * | RSV | CBS | RSV | Max TCs |
137162306a36Sopenharmony_ci	 * | 1b  | 1b  | 3b  | 3b      |
137262306a36Sopenharmony_ci	 * ----------------------------
137362306a36Sopenharmony_ci	 */
137462306a36Sopenharmony_ci	etscfg = &dcbcfg->etscfg;
137562306a36Sopenharmony_ci	buf[0] = etscfg->maxtcs & ICE_IEEE_ETS_MAXTC_M;
137662306a36Sopenharmony_ci
137762306a36Sopenharmony_ci	/* bytes 1 - 4 reserved */
137862306a36Sopenharmony_ci	offset = 5;
137962306a36Sopenharmony_ci
138062306a36Sopenharmony_ci	/* TC BW table
138162306a36Sopenharmony_ci	 * bytes 0 - 7 for TC 0 - 7
138262306a36Sopenharmony_ci	 *
138362306a36Sopenharmony_ci	 * TSA Assignment table
138462306a36Sopenharmony_ci	 * bytes 8 - 15 for TC 0 - 7
138562306a36Sopenharmony_ci	 */
138662306a36Sopenharmony_ci	for (i = 0; i < ICE_MAX_TRAFFIC_CLASS; i++) {
138762306a36Sopenharmony_ci		buf[offset] = etscfg->tcbwtable[i];
138862306a36Sopenharmony_ci		buf[offset + ICE_MAX_TRAFFIC_CLASS] = etscfg->tsatable[i];
138962306a36Sopenharmony_ci		offset++;
139062306a36Sopenharmony_ci	}
139162306a36Sopenharmony_ci}
139262306a36Sopenharmony_ci
139362306a36Sopenharmony_ci/**
139462306a36Sopenharmony_ci * ice_add_dscp_pfc_tlv - Prepare DSCP PFC TLV
139562306a36Sopenharmony_ci * @tlv: Fill PFC TLV in IEEE format
139662306a36Sopenharmony_ci * @dcbcfg: Local store which holds the PFC CFG data
139762306a36Sopenharmony_ci */
139862306a36Sopenharmony_cistatic void
139962306a36Sopenharmony_ciice_add_dscp_pfc_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg)
140062306a36Sopenharmony_ci{
140162306a36Sopenharmony_ci	u8 *buf = tlv->tlvinfo;
140262306a36Sopenharmony_ci	u32 ouisubtype;
140362306a36Sopenharmony_ci	u16 typelen;
140462306a36Sopenharmony_ci
140562306a36Sopenharmony_ci	typelen = ((ICE_TLV_TYPE_ORG << ICE_LLDP_TLV_TYPE_S) |
140662306a36Sopenharmony_ci		   ICE_DSCP_PFC_TLV_LEN);
140762306a36Sopenharmony_ci	tlv->typelen = htons(typelen);
140862306a36Sopenharmony_ci
140962306a36Sopenharmony_ci	ouisubtype = (u32)((ICE_DSCP_OUI << ICE_LLDP_TLV_OUI_S) |
141062306a36Sopenharmony_ci			   ICE_DSCP_SUBTYPE_PFC);
141162306a36Sopenharmony_ci	tlv->ouisubtype = htonl(ouisubtype);
141262306a36Sopenharmony_ci
141362306a36Sopenharmony_ci	buf[0] = dcbcfg->pfc.pfccap & 0xF;
141462306a36Sopenharmony_ci	buf[1] = dcbcfg->pfc.pfcena;
141562306a36Sopenharmony_ci}
141662306a36Sopenharmony_ci
141762306a36Sopenharmony_ci/**
141862306a36Sopenharmony_ci * ice_add_dcb_tlv - Add all IEEE or DSCP TLVs
141962306a36Sopenharmony_ci * @tlv: Fill TLV data in IEEE format
142062306a36Sopenharmony_ci * @dcbcfg: Local store which holds the DCB Config
142162306a36Sopenharmony_ci * @tlvid: Type of IEEE TLV
142262306a36Sopenharmony_ci *
142362306a36Sopenharmony_ci * Add tlv information
142462306a36Sopenharmony_ci */
142562306a36Sopenharmony_cistatic void
142662306a36Sopenharmony_ciice_add_dcb_tlv(struct ice_lldp_org_tlv *tlv, struct ice_dcbx_cfg *dcbcfg,
142762306a36Sopenharmony_ci		u16 tlvid)
142862306a36Sopenharmony_ci{
142962306a36Sopenharmony_ci	if (dcbcfg->pfc_mode == ICE_QOS_MODE_VLAN) {
143062306a36Sopenharmony_ci		switch (tlvid) {
143162306a36Sopenharmony_ci		case ICE_IEEE_TLV_ID_ETS_CFG:
143262306a36Sopenharmony_ci			ice_add_ieee_ets_tlv(tlv, dcbcfg);
143362306a36Sopenharmony_ci			break;
143462306a36Sopenharmony_ci		case ICE_IEEE_TLV_ID_ETS_REC:
143562306a36Sopenharmony_ci			ice_add_ieee_etsrec_tlv(tlv, dcbcfg);
143662306a36Sopenharmony_ci			break;
143762306a36Sopenharmony_ci		case ICE_IEEE_TLV_ID_PFC_CFG:
143862306a36Sopenharmony_ci			ice_add_ieee_pfc_tlv(tlv, dcbcfg);
143962306a36Sopenharmony_ci			break;
144062306a36Sopenharmony_ci		case ICE_IEEE_TLV_ID_APP_PRI:
144162306a36Sopenharmony_ci			ice_add_ieee_app_pri_tlv(tlv, dcbcfg);
144262306a36Sopenharmony_ci			break;
144362306a36Sopenharmony_ci		default:
144462306a36Sopenharmony_ci			break;
144562306a36Sopenharmony_ci		}
144662306a36Sopenharmony_ci	} else {
144762306a36Sopenharmony_ci		/* pfc_mode == ICE_QOS_MODE_DSCP */
144862306a36Sopenharmony_ci		switch (tlvid) {
144962306a36Sopenharmony_ci		case ICE_TLV_ID_DSCP_UP:
145062306a36Sopenharmony_ci			ice_add_dscp_up_tlv(tlv, dcbcfg);
145162306a36Sopenharmony_ci			break;
145262306a36Sopenharmony_ci		case ICE_TLV_ID_DSCP_ENF:
145362306a36Sopenharmony_ci			ice_add_dscp_enf_tlv(tlv);
145462306a36Sopenharmony_ci			break;
145562306a36Sopenharmony_ci		case ICE_TLV_ID_DSCP_TC_BW:
145662306a36Sopenharmony_ci			ice_add_dscp_tc_bw_tlv(tlv, dcbcfg);
145762306a36Sopenharmony_ci			break;
145862306a36Sopenharmony_ci		case ICE_TLV_ID_DSCP_TO_PFC:
145962306a36Sopenharmony_ci			ice_add_dscp_pfc_tlv(tlv, dcbcfg);
146062306a36Sopenharmony_ci			break;
146162306a36Sopenharmony_ci		default:
146262306a36Sopenharmony_ci			break;
146362306a36Sopenharmony_ci		}
146462306a36Sopenharmony_ci	}
146562306a36Sopenharmony_ci}
146662306a36Sopenharmony_ci
146762306a36Sopenharmony_ci/**
146862306a36Sopenharmony_ci * ice_dcb_cfg_to_lldp - Convert DCB configuration to MIB format
146962306a36Sopenharmony_ci * @lldpmib: pointer to the HW struct
147062306a36Sopenharmony_ci * @miblen: length of LLDP MIB
147162306a36Sopenharmony_ci * @dcbcfg: Local store which holds the DCB Config
147262306a36Sopenharmony_ci *
147362306a36Sopenharmony_ci * Convert the DCB configuration to MIB format
147462306a36Sopenharmony_ci */
147562306a36Sopenharmony_cistatic void
147662306a36Sopenharmony_ciice_dcb_cfg_to_lldp(u8 *lldpmib, u16 *miblen, struct ice_dcbx_cfg *dcbcfg)
147762306a36Sopenharmony_ci{
147862306a36Sopenharmony_ci	u16 len, offset = 0, tlvid = ICE_TLV_ID_START;
147962306a36Sopenharmony_ci	struct ice_lldp_org_tlv *tlv;
148062306a36Sopenharmony_ci	u16 typelen;
148162306a36Sopenharmony_ci
148262306a36Sopenharmony_ci	tlv = (struct ice_lldp_org_tlv *)lldpmib;
148362306a36Sopenharmony_ci	while (1) {
148462306a36Sopenharmony_ci		ice_add_dcb_tlv(tlv, dcbcfg, tlvid++);
148562306a36Sopenharmony_ci		typelen = ntohs(tlv->typelen);
148662306a36Sopenharmony_ci		len = (typelen & ICE_LLDP_TLV_LEN_M) >> ICE_LLDP_TLV_LEN_S;
148762306a36Sopenharmony_ci		if (len)
148862306a36Sopenharmony_ci			offset += len + 2;
148962306a36Sopenharmony_ci		/* END TLV or beyond LLDPDU size */
149062306a36Sopenharmony_ci		if (tlvid >= ICE_TLV_ID_END_OF_LLDPPDU ||
149162306a36Sopenharmony_ci		    offset > ICE_LLDPDU_SIZE)
149262306a36Sopenharmony_ci			break;
149362306a36Sopenharmony_ci		/* Move to next TLV */
149462306a36Sopenharmony_ci		if (len)
149562306a36Sopenharmony_ci			tlv = (struct ice_lldp_org_tlv *)
149662306a36Sopenharmony_ci				((char *)tlv + sizeof(tlv->typelen) + len);
149762306a36Sopenharmony_ci	}
149862306a36Sopenharmony_ci	*miblen = offset;
149962306a36Sopenharmony_ci}
150062306a36Sopenharmony_ci
150162306a36Sopenharmony_ci/**
150262306a36Sopenharmony_ci * ice_set_dcb_cfg - Set the local LLDP MIB to FW
150362306a36Sopenharmony_ci * @pi: port information structure
150462306a36Sopenharmony_ci *
150562306a36Sopenharmony_ci * Set DCB configuration to the Firmware
150662306a36Sopenharmony_ci */
150762306a36Sopenharmony_ciint ice_set_dcb_cfg(struct ice_port_info *pi)
150862306a36Sopenharmony_ci{
150962306a36Sopenharmony_ci	u8 mib_type, *lldpmib = NULL;
151062306a36Sopenharmony_ci	struct ice_dcbx_cfg *dcbcfg;
151162306a36Sopenharmony_ci	struct ice_hw *hw;
151262306a36Sopenharmony_ci	u16 miblen;
151362306a36Sopenharmony_ci	int ret;
151462306a36Sopenharmony_ci
151562306a36Sopenharmony_ci	if (!pi)
151662306a36Sopenharmony_ci		return -EINVAL;
151762306a36Sopenharmony_ci
151862306a36Sopenharmony_ci	hw = pi->hw;
151962306a36Sopenharmony_ci
152062306a36Sopenharmony_ci	/* update the HW local config */
152162306a36Sopenharmony_ci	dcbcfg = &pi->qos_cfg.local_dcbx_cfg;
152262306a36Sopenharmony_ci	/* Allocate the LLDPDU */
152362306a36Sopenharmony_ci	lldpmib = devm_kzalloc(ice_hw_to_dev(hw), ICE_LLDPDU_SIZE, GFP_KERNEL);
152462306a36Sopenharmony_ci	if (!lldpmib)
152562306a36Sopenharmony_ci		return -ENOMEM;
152662306a36Sopenharmony_ci
152762306a36Sopenharmony_ci	mib_type = SET_LOCAL_MIB_TYPE_LOCAL_MIB;
152862306a36Sopenharmony_ci	if (dcbcfg->app_mode == ICE_DCBX_APPS_NON_WILLING)
152962306a36Sopenharmony_ci		mib_type |= SET_LOCAL_MIB_TYPE_CEE_NON_WILLING;
153062306a36Sopenharmony_ci
153162306a36Sopenharmony_ci	ice_dcb_cfg_to_lldp(lldpmib, &miblen, dcbcfg);
153262306a36Sopenharmony_ci	ret = ice_aq_set_lldp_mib(hw, mib_type, (void *)lldpmib, miblen,
153362306a36Sopenharmony_ci				  NULL);
153462306a36Sopenharmony_ci
153562306a36Sopenharmony_ci	devm_kfree(ice_hw_to_dev(hw), lldpmib);
153662306a36Sopenharmony_ci
153762306a36Sopenharmony_ci	return ret;
153862306a36Sopenharmony_ci}
153962306a36Sopenharmony_ci
154062306a36Sopenharmony_ci/**
154162306a36Sopenharmony_ci * ice_aq_query_port_ets - query port ETS configuration
154262306a36Sopenharmony_ci * @pi: port information structure
154362306a36Sopenharmony_ci * @buf: pointer to buffer
154462306a36Sopenharmony_ci * @buf_size: buffer size in bytes
154562306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
154662306a36Sopenharmony_ci *
154762306a36Sopenharmony_ci * query current port ETS configuration
154862306a36Sopenharmony_ci */
154962306a36Sopenharmony_cistatic int
155062306a36Sopenharmony_ciice_aq_query_port_ets(struct ice_port_info *pi,
155162306a36Sopenharmony_ci		      struct ice_aqc_port_ets_elem *buf, u16 buf_size,
155262306a36Sopenharmony_ci		      struct ice_sq_cd *cd)
155362306a36Sopenharmony_ci{
155462306a36Sopenharmony_ci	struct ice_aqc_query_port_ets *cmd;
155562306a36Sopenharmony_ci	struct ice_aq_desc desc;
155662306a36Sopenharmony_ci	int status;
155762306a36Sopenharmony_ci
155862306a36Sopenharmony_ci	if (!pi)
155962306a36Sopenharmony_ci		return -EINVAL;
156062306a36Sopenharmony_ci	cmd = &desc.params.port_ets;
156162306a36Sopenharmony_ci	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_query_port_ets);
156262306a36Sopenharmony_ci	cmd->port_teid = pi->root->info.node_teid;
156362306a36Sopenharmony_ci
156462306a36Sopenharmony_ci	status = ice_aq_send_cmd(pi->hw, &desc, buf, buf_size, cd);
156562306a36Sopenharmony_ci	return status;
156662306a36Sopenharmony_ci}
156762306a36Sopenharmony_ci
156862306a36Sopenharmony_ci/**
156962306a36Sopenharmony_ci * ice_update_port_tc_tree_cfg - update TC tree configuration
157062306a36Sopenharmony_ci * @pi: port information structure
157162306a36Sopenharmony_ci * @buf: pointer to buffer
157262306a36Sopenharmony_ci *
157362306a36Sopenharmony_ci * update the SW DB with the new TC changes
157462306a36Sopenharmony_ci */
157562306a36Sopenharmony_cistatic int
157662306a36Sopenharmony_ciice_update_port_tc_tree_cfg(struct ice_port_info *pi,
157762306a36Sopenharmony_ci			    struct ice_aqc_port_ets_elem *buf)
157862306a36Sopenharmony_ci{
157962306a36Sopenharmony_ci	struct ice_sched_node *node, *tc_node;
158062306a36Sopenharmony_ci	struct ice_aqc_txsched_elem_data elem;
158162306a36Sopenharmony_ci	u32 teid1, teid2;
158262306a36Sopenharmony_ci	int status = 0;
158362306a36Sopenharmony_ci	u8 i, j;
158462306a36Sopenharmony_ci
158562306a36Sopenharmony_ci	if (!pi)
158662306a36Sopenharmony_ci		return -EINVAL;
158762306a36Sopenharmony_ci	/* suspend the missing TC nodes */
158862306a36Sopenharmony_ci	for (i = 0; i < pi->root->num_children; i++) {
158962306a36Sopenharmony_ci		teid1 = le32_to_cpu(pi->root->children[i]->info.node_teid);
159062306a36Sopenharmony_ci		ice_for_each_traffic_class(j) {
159162306a36Sopenharmony_ci			teid2 = le32_to_cpu(buf->tc_node_teid[j]);
159262306a36Sopenharmony_ci			if (teid1 == teid2)
159362306a36Sopenharmony_ci				break;
159462306a36Sopenharmony_ci		}
159562306a36Sopenharmony_ci		if (j < ICE_MAX_TRAFFIC_CLASS)
159662306a36Sopenharmony_ci			continue;
159762306a36Sopenharmony_ci		/* TC is missing */
159862306a36Sopenharmony_ci		pi->root->children[i]->in_use = false;
159962306a36Sopenharmony_ci	}
160062306a36Sopenharmony_ci	/* add the new TC nodes */
160162306a36Sopenharmony_ci	ice_for_each_traffic_class(j) {
160262306a36Sopenharmony_ci		teid2 = le32_to_cpu(buf->tc_node_teid[j]);
160362306a36Sopenharmony_ci		if (teid2 == ICE_INVAL_TEID)
160462306a36Sopenharmony_ci			continue;
160562306a36Sopenharmony_ci		/* Is it already present in the tree ? */
160662306a36Sopenharmony_ci		for (i = 0; i < pi->root->num_children; i++) {
160762306a36Sopenharmony_ci			tc_node = pi->root->children[i];
160862306a36Sopenharmony_ci			if (!tc_node)
160962306a36Sopenharmony_ci				continue;
161062306a36Sopenharmony_ci			teid1 = le32_to_cpu(tc_node->info.node_teid);
161162306a36Sopenharmony_ci			if (teid1 == teid2) {
161262306a36Sopenharmony_ci				tc_node->tc_num = j;
161362306a36Sopenharmony_ci				tc_node->in_use = true;
161462306a36Sopenharmony_ci				break;
161562306a36Sopenharmony_ci			}
161662306a36Sopenharmony_ci		}
161762306a36Sopenharmony_ci		if (i < pi->root->num_children)
161862306a36Sopenharmony_ci			continue;
161962306a36Sopenharmony_ci		/* new TC */
162062306a36Sopenharmony_ci		status = ice_sched_query_elem(pi->hw, teid2, &elem);
162162306a36Sopenharmony_ci		if (!status)
162262306a36Sopenharmony_ci			status = ice_sched_add_node(pi, 1, &elem, NULL);
162362306a36Sopenharmony_ci		if (status)
162462306a36Sopenharmony_ci			break;
162562306a36Sopenharmony_ci		/* update the TC number */
162662306a36Sopenharmony_ci		node = ice_sched_find_node_by_teid(pi->root, teid2);
162762306a36Sopenharmony_ci		if (node)
162862306a36Sopenharmony_ci			node->tc_num = j;
162962306a36Sopenharmony_ci	}
163062306a36Sopenharmony_ci	return status;
163162306a36Sopenharmony_ci}
163262306a36Sopenharmony_ci
163362306a36Sopenharmony_ci/**
163462306a36Sopenharmony_ci * ice_query_port_ets - query port ETS configuration
163562306a36Sopenharmony_ci * @pi: port information structure
163662306a36Sopenharmony_ci * @buf: pointer to buffer
163762306a36Sopenharmony_ci * @buf_size: buffer size in bytes
163862306a36Sopenharmony_ci * @cd: pointer to command details structure or NULL
163962306a36Sopenharmony_ci *
164062306a36Sopenharmony_ci * query current port ETS configuration and update the
164162306a36Sopenharmony_ci * SW DB with the TC changes
164262306a36Sopenharmony_ci */
164362306a36Sopenharmony_ciint
164462306a36Sopenharmony_ciice_query_port_ets(struct ice_port_info *pi,
164562306a36Sopenharmony_ci		   struct ice_aqc_port_ets_elem *buf, u16 buf_size,
164662306a36Sopenharmony_ci		   struct ice_sq_cd *cd)
164762306a36Sopenharmony_ci{
164862306a36Sopenharmony_ci	int status;
164962306a36Sopenharmony_ci
165062306a36Sopenharmony_ci	mutex_lock(&pi->sched_lock);
165162306a36Sopenharmony_ci	status = ice_aq_query_port_ets(pi, buf, buf_size, cd);
165262306a36Sopenharmony_ci	if (!status)
165362306a36Sopenharmony_ci		status = ice_update_port_tc_tree_cfg(pi, buf);
165462306a36Sopenharmony_ci	mutex_unlock(&pi->sched_lock);
165562306a36Sopenharmony_ci	return status;
165662306a36Sopenharmony_ci}
1657