xref: /linux/drivers/net/wireless/intel/iwlwifi/mvm/link.c (revision 173b0b5b0e865348684c02bd9cb1d22b5d46e458)
1 // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2 /*
3  * Copyright (C) 2022 - 2024 Intel Corporation
4  */
5 #include "mvm.h"
6 #include "time-event.h"
7 
8 static u32 iwl_mvm_get_free_fw_link_id(struct iwl_mvm *mvm,
9 				       struct iwl_mvm_vif *mvm_vif)
10 {
11 	u32 link_id;
12 
13 	lockdep_assert_held(&mvm->mutex);
14 
15 	link_id = ffz(mvm->fw_link_ids_map);
16 
17 	/* this case can happen if there're deactivated but not removed links */
18 	if (link_id > IWL_MVM_FW_MAX_LINK_ID)
19 		return IWL_MVM_FW_LINK_ID_INVALID;
20 
21 	mvm->fw_link_ids_map |= BIT(link_id);
22 	return link_id;
23 }
24 
25 static void iwl_mvm_release_fw_link_id(struct iwl_mvm *mvm, u32 link_id)
26 {
27 	lockdep_assert_held(&mvm->mutex);
28 
29 	if (!WARN_ON(link_id > IWL_MVM_FW_MAX_LINK_ID))
30 		mvm->fw_link_ids_map &= ~BIT(link_id);
31 }
32 
33 static int iwl_mvm_link_cmd_send(struct iwl_mvm *mvm,
34 				 struct iwl_link_config_cmd *cmd,
35 				 enum iwl_ctxt_action action)
36 {
37 	int ret;
38 
39 	cmd->action = cpu_to_le32(action);
40 	ret = iwl_mvm_send_cmd_pdu(mvm,
41 				   WIDE_ID(MAC_CONF_GROUP, LINK_CONFIG_CMD), 0,
42 				   sizeof(*cmd), cmd);
43 	if (ret)
44 		IWL_ERR(mvm, "Failed to send LINK_CONFIG_CMD (action:%d): %d\n",
45 			action, ret);
46 	return ret;
47 }
48 
49 int iwl_mvm_set_link_mapping(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
50 			     struct ieee80211_bss_conf *link_conf)
51 {
52 	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
53 	struct iwl_mvm_vif_link_info *link_info =
54 		mvmvif->link[link_conf->link_id];
55 
56 	if (link_info->fw_link_id == IWL_MVM_FW_LINK_ID_INVALID) {
57 		link_info->fw_link_id = iwl_mvm_get_free_fw_link_id(mvm,
58 								    mvmvif);
59 		if (link_info->fw_link_id >=
60 		    ARRAY_SIZE(mvm->link_id_to_link_conf))
61 			return -EINVAL;
62 
63 		rcu_assign_pointer(mvm->link_id_to_link_conf[link_info->fw_link_id],
64 				   link_conf);
65 	}
66 
67 	return 0;
68 }
69 
70 int iwl_mvm_add_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
71 		     struct ieee80211_bss_conf *link_conf)
72 {
73 	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
74 	unsigned int link_id = link_conf->link_id;
75 	struct iwl_mvm_vif_link_info *link_info = mvmvif->link[link_id];
76 	struct iwl_link_config_cmd cmd = {};
77 	unsigned int cmd_id = WIDE_ID(MAC_CONF_GROUP, LINK_CONFIG_CMD);
78 	u8 cmd_ver = iwl_fw_lookup_cmd_ver(mvm->fw, cmd_id, 1);
79 	int ret;
80 
81 	if (WARN_ON_ONCE(!link_info))
82 		return -EINVAL;
83 
84 	ret = iwl_mvm_set_link_mapping(mvm, vif, link_conf);
85 	if (ret)
86 		return ret;
87 
88 	/* Update SF - Disable if needed. if this fails, SF might still be on
89 	 * while many macs are bound, which is forbidden - so fail the binding.
90 	 */
91 	if (iwl_mvm_sf_update(mvm, vif, false))
92 		return -EINVAL;
93 
94 	cmd.link_id = cpu_to_le32(link_info->fw_link_id);
95 	cmd.mac_id = cpu_to_le32(mvmvif->id);
96 	cmd.spec_link_id = link_conf->link_id;
97 	WARN_ON_ONCE(link_info->phy_ctxt);
98 	cmd.phy_id = cpu_to_le32(FW_CTXT_INVALID);
99 
100 	memcpy(cmd.local_link_addr, link_conf->addr, ETH_ALEN);
101 
102 	if (vif->type == NL80211_IFTYPE_ADHOC && link_conf->bssid)
103 		memcpy(cmd.ibss_bssid_addr, link_conf->bssid, ETH_ALEN);
104 
105 	if (cmd_ver < 2)
106 		cmd.listen_lmac = cpu_to_le32(link_info->listen_lmac);
107 
108 	return iwl_mvm_link_cmd_send(mvm, &cmd, FW_CTXT_ACTION_ADD);
109 }
110 
111 int iwl_mvm_link_changed(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
112 			 struct ieee80211_bss_conf *link_conf,
113 			 u32 changes, bool active)
114 {
115 	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
116 	unsigned int link_id = link_conf->link_id;
117 	struct iwl_mvm_vif_link_info *link_info = mvmvif->link[link_id];
118 	struct iwl_mvm_phy_ctxt *phyctxt;
119 	struct iwl_link_config_cmd cmd = {};
120 	u32 ht_flag, flags = 0, flags_mask = 0;
121 	int ret;
122 	unsigned int cmd_id = WIDE_ID(MAC_CONF_GROUP, LINK_CONFIG_CMD);
123 	u8 cmd_ver = iwl_fw_lookup_cmd_ver(mvm->fw, cmd_id, 1);
124 
125 	if (WARN_ON_ONCE(!link_info ||
126 			 link_info->fw_link_id == IWL_MVM_FW_LINK_ID_INVALID))
127 		return -EINVAL;
128 
129 	if (changes & LINK_CONTEXT_MODIFY_ACTIVE) {
130 		/* When activating a link, phy context should be valid;
131 		 * when deactivating a link, it also should be valid since
132 		 * the link was active before. So, do nothing in this case.
133 		 * Since a link is added first with FW_CTXT_INVALID, then we
134 		 * can get here in case it's removed before it was activated.
135 		 */
136 		if (!link_info->phy_ctxt)
137 			return 0;
138 
139 		/* Catch early if driver tries to activate or deactivate a link
140 		 * twice.
141 		 */
142 		WARN_ON_ONCE(active == link_info->active);
143 
144 		/* When deactivating a link session protection should
145 		 * be stopped
146 		 */
147 		if (!active && vif->type == NL80211_IFTYPE_STATION)
148 			iwl_mvm_stop_session_protection(mvm, vif);
149 	}
150 
151 	cmd.link_id = cpu_to_le32(link_info->fw_link_id);
152 
153 	/* The phy_id, link address and listen_lmac can be modified only until
154 	 * the link becomes active, otherwise they will be ignored.
155 	 */
156 	phyctxt = link_info->phy_ctxt;
157 	if (phyctxt)
158 		cmd.phy_id = cpu_to_le32(phyctxt->id);
159 	else
160 		cmd.phy_id = cpu_to_le32(FW_CTXT_INVALID);
161 	cmd.mac_id = cpu_to_le32(mvmvif->id);
162 
163 	memcpy(cmd.local_link_addr, link_conf->addr, ETH_ALEN);
164 
165 	cmd.active = cpu_to_le32(active);
166 
167 	if (vif->type == NL80211_IFTYPE_ADHOC && link_conf->bssid)
168 		memcpy(cmd.ibss_bssid_addr, link_conf->bssid, ETH_ALEN);
169 
170 	iwl_mvm_set_fw_basic_rates(mvm, vif, link_conf,
171 				   &cmd.cck_rates, &cmd.ofdm_rates);
172 
173 	cmd.cck_short_preamble = cpu_to_le32(link_conf->use_short_preamble);
174 	cmd.short_slot = cpu_to_le32(link_conf->use_short_slot);
175 
176 	/* The fw does not distinguish between ht and fat */
177 	ht_flag = LINK_PROT_FLG_HT_PROT | LINK_PROT_FLG_FAT_PROT;
178 	iwl_mvm_set_fw_protection_flags(mvm, vif, link_conf,
179 					&cmd.protection_flags,
180 					ht_flag, LINK_PROT_FLG_TGG_PROTECT);
181 
182 	iwl_mvm_set_fw_qos_params(mvm, vif, link_conf, cmd.ac,
183 				  &cmd.qos_flags);
184 
185 
186 	cmd.bi = cpu_to_le32(link_conf->beacon_int);
187 	cmd.dtim_interval = cpu_to_le32(link_conf->beacon_int *
188 					link_conf->dtim_period);
189 
190 	if (!link_conf->he_support || iwlwifi_mod_params.disable_11ax ||
191 	    (vif->type == NL80211_IFTYPE_STATION && !vif->cfg.assoc)) {
192 		changes &= ~LINK_CONTEXT_MODIFY_HE_PARAMS;
193 		goto send_cmd;
194 	}
195 
196 	cmd.htc_trig_based_pkt_ext = link_conf->htc_trig_based_pkt_ext;
197 
198 	if (link_conf->uora_exists) {
199 		cmd.rand_alloc_ecwmin =
200 			link_conf->uora_ocw_range & 0x7;
201 		cmd.rand_alloc_ecwmax =
202 			(link_conf->uora_ocw_range >> 3) & 0x7;
203 	}
204 
205 	/* TODO  how to set ndp_fdbk_buff_th_exp? */
206 
207 	if (iwl_mvm_set_fw_mu_edca_params(mvm, mvmvif->link[link_id],
208 					  &cmd.trig_based_txf[0])) {
209 		flags |= LINK_FLG_MU_EDCA_CW;
210 		flags_mask |= LINK_FLG_MU_EDCA_CW;
211 	}
212 
213 	if (changes & LINK_CONTEXT_MODIFY_EHT_PARAMS) {
214 		struct ieee80211_chanctx_conf *ctx;
215 		struct cfg80211_chan_def *def = NULL;
216 
217 		rcu_read_lock();
218 		ctx = rcu_dereference(link_conf->chanctx_conf);
219 		if (ctx)
220 			def = iwl_mvm_chanctx_def(mvm, ctx);
221 
222 		if (iwlwifi_mod_params.disable_11be ||
223 		    !link_conf->eht_support || !def ||
224 		    iwl_fw_lookup_cmd_ver(mvm->fw, PHY_CONTEXT_CMD, 1) >= 6)
225 			changes &= ~LINK_CONTEXT_MODIFY_EHT_PARAMS;
226 		else
227 			cmd.puncture_mask = cpu_to_le16(def->punctured);
228 		rcu_read_unlock();
229 	}
230 
231 	cmd.bss_color = link_conf->he_bss_color.color;
232 
233 	if (!link_conf->he_bss_color.enabled) {
234 		flags |= LINK_FLG_BSS_COLOR_DIS;
235 		flags_mask |= LINK_FLG_BSS_COLOR_DIS;
236 	}
237 
238 	cmd.frame_time_rts_th = cpu_to_le16(link_conf->frame_time_rts_th);
239 
240 	/* Block 26-tone RU OFDMA transmissions */
241 	if (link_info->he_ru_2mhz_block) {
242 		flags |= LINK_FLG_RU_2MHZ_BLOCK;
243 		flags_mask |= LINK_FLG_RU_2MHZ_BLOCK;
244 	}
245 
246 	if (link_conf->nontransmitted) {
247 		ether_addr_copy(cmd.ref_bssid_addr,
248 				link_conf->transmitter_bssid);
249 		cmd.bssid_index = link_conf->bssid_index;
250 	}
251 
252 send_cmd:
253 	cmd.modify_mask = cpu_to_le32(changes);
254 	cmd.flags = cpu_to_le32(flags);
255 	cmd.flags_mask = cpu_to_le32(flags_mask);
256 	cmd.spec_link_id = link_conf->link_id;
257 	if (cmd_ver < 2)
258 		cmd.listen_lmac = cpu_to_le32(link_info->listen_lmac);
259 
260 	ret = iwl_mvm_link_cmd_send(mvm, &cmd, FW_CTXT_ACTION_MODIFY);
261 	if (!ret && (changes & LINK_CONTEXT_MODIFY_ACTIVE))
262 		link_info->active = active;
263 
264 	return ret;
265 }
266 
267 int iwl_mvm_unset_link_mapping(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
268 			       struct ieee80211_bss_conf *link_conf)
269 {
270 	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
271 	struct iwl_mvm_vif_link_info *link_info =
272 		mvmvif->link[link_conf->link_id];
273 
274 	/* mac80211 thought we have the link, but it was never configured */
275 	if (WARN_ON(!link_info ||
276 		    link_info->fw_link_id >=
277 		    ARRAY_SIZE(mvm->link_id_to_link_conf)))
278 		return -EINVAL;
279 
280 	RCU_INIT_POINTER(mvm->link_id_to_link_conf[link_info->fw_link_id],
281 			 NULL);
282 	return 0;
283 }
284 
285 int iwl_mvm_remove_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
286 			struct ieee80211_bss_conf *link_conf)
287 {
288 	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
289 	unsigned int link_id = link_conf->link_id;
290 	struct iwl_mvm_vif_link_info *link_info = mvmvif->link[link_id];
291 	struct iwl_link_config_cmd cmd = {};
292 	int ret;
293 
294 	ret = iwl_mvm_unset_link_mapping(mvm, vif, link_conf);
295 	if (ret)
296 		return 0;
297 
298 	cmd.link_id = cpu_to_le32(link_info->fw_link_id);
299 	iwl_mvm_release_fw_link_id(mvm, link_info->fw_link_id);
300 	link_info->fw_link_id = IWL_MVM_FW_LINK_ID_INVALID;
301 	cmd.spec_link_id = link_conf->link_id;
302 	cmd.phy_id = cpu_to_le32(FW_CTXT_INVALID);
303 
304 	ret = iwl_mvm_link_cmd_send(mvm, &cmd, FW_CTXT_ACTION_REMOVE);
305 
306 	if (!ret)
307 		if (iwl_mvm_sf_update(mvm, vif, true))
308 			IWL_ERR(mvm, "Failed to update SF state\n");
309 
310 	return ret;
311 }
312 
313 /* link should be deactivated before removal, so in most cases we need to
314  * perform these two operations together
315  */
316 int iwl_mvm_disable_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
317 			 struct ieee80211_bss_conf *link_conf)
318 {
319 	int ret;
320 
321 	ret = iwl_mvm_link_changed(mvm, vif, link_conf,
322 				   LINK_CONTEXT_MODIFY_ACTIVE, false);
323 	if (ret)
324 		return ret;
325 
326 	ret = iwl_mvm_remove_link(mvm, vif, link_conf);
327 	if (ret)
328 		return ret;
329 
330 	return ret;
331 }
332