xref: /freebsd/sys/contrib/dev/iwlwifi/mld/notif.c (revision b2bd08185e4984c70179c195f712cef5a136d21b)
1 // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2 /*
3  * Copyright (C) 2024-2025 Intel Corporation
4  */
5 
6 #include "mld.h"
7 #include "notif.h"
8 #include "scan.h"
9 #include "iface.h"
10 #include "mlo.h"
11 #include "iwl-trans.h"
12 #include "fw/file.h"
13 #include "fw/dbg.h"
14 #include "fw/api/cmdhdr.h"
15 #include "fw/api/mac-cfg.h"
16 #include "session-protect.h"
17 #include "fw/api/time-event.h"
18 #include "fw/api/tx.h"
19 #include "fw/api/rs.h"
20 #include "fw/api/offload.h"
21 #include "fw/api/stats.h"
22 #include "fw/api/rfi.h"
23 #include "fw/api/coex.h"
24 
25 #include "mcc.h"
26 #include "link.h"
27 #include "tx.h"
28 #include "rx.h"
29 #include "tlc.h"
30 #include "agg.h"
31 #include "mac80211.h"
32 #include "thermal.h"
33 #include "roc.h"
34 #include "stats.h"
35 #include "coex.h"
36 #include "time_sync.h"
37 #include "ftm-initiator.h"
38 
39 /* Please use this in an increasing order of the versions */
40 #define CMD_VER_ENTRY(_ver, _struct)			\
41 	{ .size = sizeof(struct _struct), .ver = _ver },
42 #define CMD_VERSIONS(name, ...)				\
43 	static const struct iwl_notif_struct_size	\
44 	iwl_notif_struct_sizes_##name[] = { __VA_ARGS__ };
45 
46 #define RX_HANDLER_NO_OBJECT(_grp, _cmd, _name, _context)		\
47 	{.cmd_id = WIDE_ID(_grp, _cmd),					\
48 	 .context = _context,						\
49 	 .fn = iwl_mld_handle_##_name,					\
50 	 .sizes = iwl_notif_struct_sizes_##_name,			\
51 	 .n_sizes = ARRAY_SIZE(iwl_notif_struct_sizes_##_name),		\
52 	},
53 
54 /* Use this for Rx handlers that do not need notification validation */
55 #define RX_HANDLER_NO_VAL(_grp, _cmd, _name, _context)			\
56 	{.cmd_id = WIDE_ID(_grp, _cmd),					\
57 	 .context = _context,						\
58 	 .fn = iwl_mld_handle_##_name,					\
59 	},
60 
61 #define RX_HANDLER_VAL_FN(_grp, _cmd, _name, _context)			\
62 	{ .cmd_id = WIDE_ID(_grp, _cmd),				\
63 	  .context = _context,						\
64 	  .fn = iwl_mld_handle_##_name,					\
65 	  .val_fn = iwl_mld_validate_##_name,				\
66 	},
67 
68 #define DEFINE_SIMPLE_CANCELLATION(name, notif_struct, id_member)		\
69 static bool iwl_mld_cancel_##name##_notif(struct iwl_mld *mld,			\
70 					  struct iwl_rx_packet *pkt,		\
71 					  u32 obj_id)				\
72 {										\
73 	const struct notif_struct *notif = (const void *)pkt->data;		\
74 										\
75 	return obj_id == _Generic((notif)->id_member,				\
76 				  __le32: le32_to_cpu((notif)->id_member),	\
77 				  __le16: le16_to_cpu((notif)->id_member),	\
78 				  u8: (notif)->id_member);			\
79 }
80 
81 /* Currently only defined for the RX_HANDLER_SIZES options. Use this for
82  * notifications that belong to a specific object, and that should be
83  * canceled when the object is removed
84  */
85 #define RX_HANDLER_OF_OBJ(_grp, _cmd, _name, _obj_type)			\
86 	{.cmd_id = WIDE_ID(_grp, _cmd),					\
87 	/* Only async handlers can be canceled */			\
88 	 .context = RX_HANDLER_ASYNC,					\
89 	 .fn = iwl_mld_handle_##_name,					\
90 	 .sizes = iwl_notif_struct_sizes_##_name,			\
91 	 .n_sizes = ARRAY_SIZE(iwl_notif_struct_sizes_##_name),		\
92 	 .obj_type = IWL_MLD_OBJECT_TYPE_##_obj_type,			\
93 	 .cancel = iwl_mld_cancel_##_name,				\
94 	 },
95 
96 #define RX_HANDLER_OF_LINK(_grp, _cmd, _name)				\
97 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, LINK)			\
98 
99 #define RX_HANDLER_OF_VIF(_grp, _cmd, _name)				\
100 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, VIF)			\
101 
102 #define RX_HANDLER_OF_STA(_grp, _cmd, _name)				\
103 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, STA)			\
104 
105 #define RX_HANDLER_OF_ROC(_grp, _cmd, _name)				\
106 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, ROC)
107 
108 #define RX_HANDLER_OF_SCAN(_grp, _cmd, _name)				\
109 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, SCAN)
110 
111 #define RX_HANDLER_OF_FTM_REQ(_grp, _cmd, _name)				\
112 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, FTM_REQ)
113 
114 #define RX_HANDLER_OF_NAN(_grp, _cmd, _name)				\
115 	RX_HANDLER_OF_OBJ(_grp, _cmd, _name, NAN)
116 
iwl_mld_handle_mfuart_notif(struct iwl_mld * mld,struct iwl_rx_packet * pkt)117 static void iwl_mld_handle_mfuart_notif(struct iwl_mld *mld,
118 					struct iwl_rx_packet *pkt)
119 {
120 	struct iwl_mfuart_load_notif *mfuart_notif = (void *)pkt->data;
121 
122 	IWL_DEBUG_INFO(mld,
123 		       "MFUART: installed ver: 0x%08x, external ver: 0x%08x\n",
124 		       le32_to_cpu(mfuart_notif->installed_ver),
125 		       le32_to_cpu(mfuart_notif->external_ver));
126 	IWL_DEBUG_INFO(mld,
127 		       "MFUART: status: 0x%08x, duration: 0x%08x image size: 0x%08x\n",
128 		       le32_to_cpu(mfuart_notif->status),
129 		       le32_to_cpu(mfuart_notif->duration),
130 		       le32_to_cpu(mfuart_notif->image_size));
131 }
132 
iwl_mld_mu_mimo_iface_iterator(void * _data,u8 * mac,struct ieee80211_vif * vif)133 static void iwl_mld_mu_mimo_iface_iterator(void *_data, u8 *mac,
134 					   struct ieee80211_vif *vif)
135 {
136 	struct ieee80211_bss_conf *bss_conf = &vif->bss_conf;
137 	unsigned int link_id = 0;
138 
139 	if (WARN(hweight16(vif->active_links) > 1,
140 		 "no support for this notif while in EMLSR 0x%x\n",
141 		 vif->active_links))
142 		return;
143 
144 	if (ieee80211_vif_is_mld(vif)) {
145 		link_id = __ffs(vif->active_links);
146 		bss_conf = link_conf_dereference_check(vif, link_id);
147 	}
148 
149 	if (!WARN_ON(!bss_conf) && bss_conf->mu_mimo_owner) {
150 		const struct iwl_mu_group_mgmt_notif *notif = _data;
151 
152 		BUILD_BUG_ON(sizeof(notif->membership_status) !=
153 			     WLAN_MEMBERSHIP_LEN);
154 		BUILD_BUG_ON(sizeof(notif->user_position) !=
155 			     WLAN_USER_POSITION_LEN);
156 
157 		/* MU-MIMO Group Id action frame is little endian. We treat
158 		 * the data received from firmware as if it came from the
159 		 * action frame, so no conversion is needed.
160 		 */
161 		ieee80211_update_mu_groups(vif, link_id,
162 #if defined(__linux__)
163 					   (u8 *)&notif->membership_status,
164 					   (u8 *)&notif->user_position);
165 #elif defined(__FreeBSD__)
166 					   (const u8 *)&notif->membership_status,
167 					   (const u8 *)&notif->user_position);
168 #endif
169 	}
170 }
171 
172 /* This handler is called in SYNC mode because it needs to be serialized with
173  * Rx as specified in ieee80211_update_mu_groups()'s documentation.
174  */
iwl_mld_handle_mu_mimo_grp_notif(struct iwl_mld * mld,struct iwl_rx_packet * pkt)175 static void iwl_mld_handle_mu_mimo_grp_notif(struct iwl_mld *mld,
176 					     struct iwl_rx_packet *pkt)
177 {
178 	struct iwl_mu_group_mgmt_notif *notif = (void *)pkt->data;
179 
180 	ieee80211_iterate_active_interfaces_atomic(mld->hw,
181 						   IEEE80211_IFACE_ITER_NORMAL,
182 						   iwl_mld_mu_mimo_iface_iterator,
183 						   notif);
184 }
185 
186 static void
iwl_mld_handle_channel_switch_start_notif(struct iwl_mld * mld,struct iwl_rx_packet * pkt)187 iwl_mld_handle_channel_switch_start_notif(struct iwl_mld *mld,
188 					  struct iwl_rx_packet *pkt)
189 {
190 	struct iwl_channel_switch_start_notif *notif = (void *)pkt->data;
191 	u32 link_id = le32_to_cpu(notif->link_id);
192 	struct ieee80211_bss_conf *link_conf =
193 		iwl_mld_fw_id_to_link_conf(mld, link_id);
194 	struct ieee80211_vif *vif;
195 
196 	if (WARN_ON(!link_conf))
197 		return;
198 
199 	vif = link_conf->vif;
200 
201 	IWL_DEBUG_INFO(mld,
202 		       "CSA Start Notification with vif type: %d, link_id: %d\n",
203 		       vif->type,
204 		       link_conf->link_id);
205 
206 	switch (vif->type) {
207 	case NL80211_IFTYPE_AP:
208 		/* We don't support canceling a CSA as it was advertised
209 		 * by the AP itself
210 		 */
211 		if (!link_conf->csa_active)
212 			return;
213 
214 		ieee80211_csa_finish(vif, link_conf->link_id);
215 		break;
216 	case NL80211_IFTYPE_STATION:
217 		if (!link_conf->csa_active) {
218 			/* Either unexpected cs notif or mac80211 chose to
219 			 * ignore, for example in channel switch to same channel
220 			 */
221 			struct iwl_cancel_channel_switch_cmd cmd = {
222 				.id = cpu_to_le32(link_id),
223 			};
224 
225 			if (iwl_mld_send_cmd_pdu(mld,
226 						 WIDE_ID(MAC_CONF_GROUP,
227 							 CANCEL_CHANNEL_SWITCH_CMD),
228 						 &cmd))
229 				IWL_ERR(mld,
230 					"Failed to cancel the channel switch\n");
231 			return;
232 		}
233 
234 		ieee80211_chswitch_done(vif, true, link_conf->link_id);
235 		break;
236 
237 	default:
238 		WARN(1, "CSA on invalid vif type: %d", vif->type);
239 	}
240 }
241 
242 static void
iwl_mld_handle_channel_switch_error_notif(struct iwl_mld * mld,struct iwl_rx_packet * pkt)243 iwl_mld_handle_channel_switch_error_notif(struct iwl_mld *mld,
244 					  struct iwl_rx_packet *pkt)
245 {
246 	struct iwl_channel_switch_error_notif *notif = (void *)pkt->data;
247 	struct ieee80211_bss_conf *link_conf;
248 	struct ieee80211_vif *vif;
249 	u32 link_id = le32_to_cpu(notif->link_id);
250 	u32 csa_err_mask = le32_to_cpu(notif->csa_err_mask);
251 
252 	link_conf = iwl_mld_fw_id_to_link_conf(mld, link_id);
253 	if (WARN_ON(!link_conf))
254 		return;
255 
256 	vif = link_conf->vif;
257 
258 	IWL_DEBUG_INFO(mld, "FW reports CSA error: id=%u, csa_err_mask=%u\n",
259 		       link_id, csa_err_mask);
260 
261 	if (csa_err_mask & (CS_ERR_COUNT_ERROR |
262 			    CS_ERR_LONG_DELAY_AFTER_CS |
263 			    CS_ERR_TX_BLOCK_TIMER_EXPIRED))
264 		ieee80211_channel_switch_disconnect(vif);
265 }
266 
iwl_mld_handle_beacon_notification(struct iwl_mld * mld,struct iwl_rx_packet * pkt)267 static void iwl_mld_handle_beacon_notification(struct iwl_mld *mld,
268 					       struct iwl_rx_packet *pkt)
269 {
270 	struct iwl_extended_beacon_notif *beacon = (void *)pkt->data;
271 
272 	mld->ibss_manager = !!beacon->ibss_mgr_status;
273 }
274 
275 /**
276  * DOC: Notification versioning
277  *
278  * The firmware's notifications change from time to time. In order to
279  * differentiate between different versions of the same notification, the
280  * firmware advertises the version of each notification.
281  * Here are listed all the notifications that are supported. Several versions
282  * of the same notification can be allowed at the same time:
283  *
284  * CMD_VERSION(my_multi_version_notif,
285  *	       CMD_VER_ENTRY(1, iwl_my_multi_version_notif_ver1)
286  *	       CMD_VER_ENTRY(2, iwl_my_multi_version_notif_ver2)
287  *
288  * etc...
289  *
290  * The driver will enforce that the notification coming from the firmware
291  * has its version listed here and it'll also enforce that the firmware sent
292  * at least enough bytes to cover the structure listed in the CMD_VER_ENTRY.
293  */
294 
295 CMD_VERSIONS(scan_start_notif,
296 	     CMD_VER_ENTRY(1, iwl_umac_scan_start))
297 CMD_VERSIONS(scan_complete_notif,
298 	     CMD_VER_ENTRY(1, iwl_umac_scan_complete))
299 CMD_VERSIONS(scan_iter_complete_notif,
300 	     CMD_VER_ENTRY(2, iwl_umac_scan_iter_complete_notif))
301 CMD_VERSIONS(channel_survey_notif,
302 	     CMD_VER_ENTRY(1, iwl_umac_scan_channel_survey_notif))
303 CMD_VERSIONS(mfuart_notif,
304 	     CMD_VER_ENTRY(2, iwl_mfuart_load_notif))
305 CMD_VERSIONS(update_mcc,
306 	     CMD_VER_ENTRY(1, iwl_mcc_chub_notif))
307 CMD_VERSIONS(session_prot_notif,
308 	     CMD_VER_ENTRY(3, iwl_session_prot_notif))
309 CMD_VERSIONS(missed_beacon_notif,
310 	     CMD_VER_ENTRY(5, iwl_missed_beacons_notif))
311 CMD_VERSIONS(tx_resp_notif,
312 	     CMD_VER_ENTRY(8, iwl_tx_resp)
313 	     CMD_VER_ENTRY(9, iwl_tx_resp))
314 CMD_VERSIONS(compressed_ba_notif,
315 	     CMD_VER_ENTRY(5, iwl_compressed_ba_notif)
316 	     CMD_VER_ENTRY(6, iwl_compressed_ba_notif)
317 	     CMD_VER_ENTRY(7, iwl_compressed_ba_notif))
318 CMD_VERSIONS(tlc_notif,
319 	     CMD_VER_ENTRY(3, iwl_tlc_update_notif)
320 	     CMD_VER_ENTRY(4, iwl_tlc_update_notif))
321 CMD_VERSIONS(mu_mimo_grp_notif,
322 	     CMD_VER_ENTRY(1, iwl_mu_group_mgmt_notif))
323 CMD_VERSIONS(channel_switch_start_notif,
324 	     CMD_VER_ENTRY(3, iwl_channel_switch_start_notif))
325 CMD_VERSIONS(channel_switch_error_notif,
326 	     CMD_VER_ENTRY(2, iwl_channel_switch_error_notif))
327 CMD_VERSIONS(ct_kill_notif,
328 	     CMD_VER_ENTRY(2, ct_kill_notif))
329 CMD_VERSIONS(temp_notif,
330 	     CMD_VER_ENTRY(2, iwl_dts_measurement_notif))
331 CMD_VERSIONS(roc_notif,
332 	     CMD_VER_ENTRY(1, iwl_roc_notif))
333 CMD_VERSIONS(probe_resp_data_notif,
334 	     CMD_VER_ENTRY(1, iwl_probe_resp_data_notif))
335 CMD_VERSIONS(datapath_monitor_notif,
336 	     CMD_VER_ENTRY(1, iwl_datapath_monitor_notif))
337 CMD_VERSIONS(stats_oper_notif,
338 	     CMD_VER_ENTRY(3, iwl_system_statistics_notif_oper))
339 CMD_VERSIONS(stats_oper_part1_notif,
340 	     CMD_VER_ENTRY(4, iwl_system_statistics_part1_notif_oper))
341 CMD_VERSIONS(bt_coex_notif,
342 	     CMD_VER_ENTRY(1, iwl_bt_coex_profile_notif))
343 CMD_VERSIONS(beacon_notification,
344 	     CMD_VER_ENTRY(6, iwl_extended_beacon_notif))
345 CMD_VERSIONS(emlsr_mode_notif,
346 	     CMD_VER_ENTRY(2, iwl_esr_mode_notif))
347 CMD_VERSIONS(emlsr_trans_fail_notif,
348 	     CMD_VER_ENTRY(1, iwl_esr_trans_fail_notif))
349 CMD_VERSIONS(uapsd_misbehaving_ap_notif,
350 	     CMD_VER_ENTRY(1, iwl_uapsd_misbehaving_ap_notif))
351 CMD_VERSIONS(time_msmt_notif,
352 	     CMD_VER_ENTRY(1, iwl_time_msmt_notify))
353 CMD_VERSIONS(time_sync_confirm_notif,
354 	     CMD_VER_ENTRY(1, iwl_time_msmt_cfm_notify))
355 CMD_VERSIONS(ftm_resp_notif, CMD_VER_ENTRY(10, iwl_tof_range_rsp_ntfy))
356 CMD_VERSIONS(beacon_filter_notif, CMD_VER_ENTRY(2, iwl_beacon_filter_notif))
357 CMD_VERSIONS(nan_cluster_notif, CMD_VER_ENTRY(1, iwl_nan_cluster_notif))
358 CMD_VERSIONS(nan_dw_end_notif, CMD_VER_ENTRY(1, iwl_nan_dw_end_notif))
359 
360 DEFINE_SIMPLE_CANCELLATION(session_prot, iwl_session_prot_notif, mac_link_id)
361 DEFINE_SIMPLE_CANCELLATION(tlc, iwl_tlc_update_notif, sta_id)
362 DEFINE_SIMPLE_CANCELLATION(channel_switch_start,
363 			   iwl_channel_switch_start_notif, link_id)
364 DEFINE_SIMPLE_CANCELLATION(channel_switch_error,
365 			   iwl_channel_switch_error_notif, link_id)
366 DEFINE_SIMPLE_CANCELLATION(datapath_monitor, iwl_datapath_monitor_notif,
367 			   link_id)
368 DEFINE_SIMPLE_CANCELLATION(roc, iwl_roc_notif, activity)
369 DEFINE_SIMPLE_CANCELLATION(scan_complete, iwl_umac_scan_complete, uid)
370 DEFINE_SIMPLE_CANCELLATION(scan_start, iwl_umac_scan_start, uid)
371 DEFINE_SIMPLE_CANCELLATION(probe_resp_data, iwl_probe_resp_data_notif,
372 			   mac_id)
373 DEFINE_SIMPLE_CANCELLATION(uapsd_misbehaving_ap, iwl_uapsd_misbehaving_ap_notif,
374 			   mac_id)
375 DEFINE_SIMPLE_CANCELLATION(ftm_resp, iwl_tof_range_rsp_ntfy, request_id)
376 DEFINE_SIMPLE_CANCELLATION(beacon_filter, iwl_beacon_filter_notif, link_id)
377 
378 /**
379  * DOC: Handlers for fw notifications
380  *
381  * Here are listed the notifications IDs (including the group ID), the handler
382  * of the notification and how it should be called:
383  *
384  *  - RX_HANDLER_SYNC: will be called as part of the Rx path
385  *  - RX_HANDLER_ASYNC: will be handled in a working with the wiphy_lock held
386  *
387  * This means that if the firmware sends two notifications A and B in that
388  * order and notification A is RX_HANDLER_ASYNC and notification is
389  * RX_HANDLER_SYNC, the handler of B will likely be called before the handler
390  * of A.
391  *
392  * This list should be in order of frequency for performance purposes.
393  * The handler can be one from two contexts, see &iwl_rx_handler_context
394  *
395  * A handler can declare that it relies on a specific object in which case it
396  * can be cancelled in case the object is deleted. In order to use this
397  * mechanism, a cancellation function is needed. The cancellation function must
398  * receive an object id (the index of that object in the firmware) and a
399  * notification payload. It'll return true if that specific notification should
400  * be cancelled upon the obliteration of the specific instance of the object.
401  *
402  * DEFINE_SIMPLE_CANCELLATION allows to easily create a cancellation function
403  * that wills simply return true if a given object id matches the object id in
404  * the firmware notification.
405  */
406 
407 VISIBLE_IF_IWLWIFI_KUNIT
408 const struct iwl_rx_handler iwl_mld_rx_handlers[] = {
409 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP, TX_CMD, tx_resp_notif,
410 			     RX_HANDLER_SYNC)
411 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP, BA_NOTIF, compressed_ba_notif,
412 			     RX_HANDLER_SYNC)
413 	RX_HANDLER_OF_SCAN(LEGACY_GROUP, SCAN_START_NOTIFICATION_UMAC,
414 			   scan_start_notif)
415 	RX_HANDLER_OF_SCAN(LEGACY_GROUP, SCAN_COMPLETE_UMAC,
416 			   scan_complete_notif)
417 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP, SCAN_ITERATION_COMPLETE_UMAC,
418 			     scan_iter_complete_notif,
419 			     RX_HANDLER_SYNC)
420 	RX_HANDLER_NO_VAL(LEGACY_GROUP, MATCH_FOUND_NOTIFICATION,
421 			  match_found_notif, RX_HANDLER_SYNC)
422 
423 	RX_HANDLER_NO_OBJECT(SCAN_GROUP, CHANNEL_SURVEY_NOTIF,
424 			     channel_survey_notif,
425 			     RX_HANDLER_ASYNC)
426 
427 	RX_HANDLER_NO_OBJECT(STATISTICS_GROUP, STATISTICS_OPER_NOTIF,
428 			     stats_oper_notif, RX_HANDLER_ASYNC)
429 	RX_HANDLER_NO_OBJECT(STATISTICS_GROUP, STATISTICS_OPER_PART1_NOTIF,
430 			     stats_oper_part1_notif, RX_HANDLER_ASYNC)
431 
432 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP, MFUART_LOAD_NOTIFICATION,
433 			     mfuart_notif, RX_HANDLER_SYNC)
434 
435 	RX_HANDLER_NO_OBJECT(PHY_OPS_GROUP, DTS_MEASUREMENT_NOTIF_WIDE,
436 			     temp_notif, RX_HANDLER_ASYNC)
437 	RX_HANDLER_OF_LINK(MAC_CONF_GROUP, SESSION_PROTECTION_NOTIF,
438 			   session_prot_notif)
439 	RX_HANDLER_OF_LINK(MAC_CONF_GROUP, MISSED_BEACONS_NOTIF,
440 			   missed_beacon_notif)
441 	RX_HANDLER_OF_STA(DATA_PATH_GROUP, TLC_MNG_UPDATE_NOTIF, tlc_notif)
442 	RX_HANDLER_OF_LINK(MAC_CONF_GROUP, CHANNEL_SWITCH_START_NOTIF,
443 			   channel_switch_start_notif)
444 	RX_HANDLER_OF_LINK(MAC_CONF_GROUP, CHANNEL_SWITCH_ERROR_NOTIF,
445 			   channel_switch_error_notif)
446 	RX_HANDLER_OF_ROC(MAC_CONF_GROUP, ROC_NOTIF, roc_notif)
447 	RX_HANDLER_NO_OBJECT(DATA_PATH_GROUP, MU_GROUP_MGMT_NOTIF,
448 			     mu_mimo_grp_notif, RX_HANDLER_SYNC)
449 	RX_HANDLER_OF_VIF(MAC_CONF_GROUP, PROBE_RESPONSE_DATA_NOTIF,
450 			  probe_resp_data_notif)
451 	RX_HANDLER_NO_OBJECT(PHY_OPS_GROUP, CT_KILL_NOTIFICATION,
452 			     ct_kill_notif, RX_HANDLER_ASYNC)
453 	RX_HANDLER_OF_LINK(DATA_PATH_GROUP, MONITOR_NOTIF,
454 			   datapath_monitor_notif)
455 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP, MCC_CHUB_UPDATE_CMD, update_mcc,
456 			     RX_HANDLER_ASYNC)
457 	RX_HANDLER_NO_OBJECT(BT_COEX_GROUP, PROFILE_NOTIF,
458 			     bt_coex_notif, RX_HANDLER_ASYNC)
459 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP, BEACON_NOTIFICATION,
460 			     beacon_notification, RX_HANDLER_ASYNC)
461 	RX_HANDLER_NO_OBJECT(DATA_PATH_GROUP, ESR_MODE_NOTIF,
462 			     emlsr_mode_notif, RX_HANDLER_ASYNC)
463 	RX_HANDLER_NO_OBJECT(MAC_CONF_GROUP, EMLSR_TRANS_FAIL_NOTIF,
464 			     emlsr_trans_fail_notif, RX_HANDLER_ASYNC)
465 	RX_HANDLER_OF_VIF(LEGACY_GROUP, PSM_UAPSD_AP_MISBEHAVING_NOTIFICATION,
466 			  uapsd_misbehaving_ap_notif)
467 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP,
468 			     WNM_80211V_TIMING_MEASUREMENT_NOTIFICATION,
469 			     time_msmt_notif, RX_HANDLER_SYNC)
470 	RX_HANDLER_NO_OBJECT(LEGACY_GROUP,
471 			     WNM_80211V_TIMING_MEASUREMENT_CONFIRM_NOTIFICATION,
472 			     time_sync_confirm_notif, RX_HANDLER_ASYNC)
473 	RX_HANDLER_OF_LINK(DATA_PATH_GROUP, BEACON_FILTER_IN_NOTIF,
474 			   beacon_filter_notif)
475 	RX_HANDLER_OF_FTM_REQ(LOCATION_GROUP, TOF_RANGE_RESPONSE_NOTIF,
476 			      ftm_resp_notif)
477 	RX_HANDLER_OF_NAN(MAC_CONF_GROUP, NAN_JOINED_CLUSTER_NOTIF,
478 			  nan_cluster_notif)
479 	RX_HANDLER_OF_NAN(MAC_CONF_GROUP, NAN_DW_END_NOTIF,
480 			  nan_dw_end_notif)
481 };
482 EXPORT_SYMBOL_IF_IWLWIFI_KUNIT(iwl_mld_rx_handlers);
483 
484 #if IS_ENABLED(CONFIG_IWLWIFI_KUNIT_TESTS)
485 const unsigned int iwl_mld_rx_handlers_num = ARRAY_SIZE(iwl_mld_rx_handlers);
486 EXPORT_SYMBOL_IF_IWLWIFI_KUNIT(iwl_mld_rx_handlers_num);
487 #endif
488 
489 static bool
iwl_mld_notif_is_valid(struct iwl_mld * mld,struct iwl_rx_packet * pkt,const struct iwl_rx_handler * handler)490 iwl_mld_notif_is_valid(struct iwl_mld *mld, struct iwl_rx_packet *pkt,
491 		       const struct iwl_rx_handler *handler)
492 {
493 	unsigned int size = iwl_rx_packet_payload_len(pkt);
494 	size_t notif_ver;
495 
496 	/* If n_sizes == 0, it indicates that a validation function may be used
497 	 * or that no validation is required.
498 	 */
499 	if (!handler->n_sizes) {
500 		if (handler->val_fn)
501 			return handler->val_fn(mld, pkt);
502 		return true;
503 	}
504 
505 	notif_ver = iwl_fw_lookup_notif_ver(mld->fw,
506 					    iwl_cmd_groupid(handler->cmd_id),
507 					    iwl_cmd_opcode(handler->cmd_id),
508 					    IWL_FW_CMD_VER_UNKNOWN);
509 
510 	for (int i = 0; i < handler->n_sizes; i++) {
511 		if (handler->sizes[i].ver != notif_ver)
512 			continue;
513 
514 		if (IWL_FW_CHECK(mld, size < handler->sizes[i].size,
515 				 "unexpected notification 0x%04x size %d, need %d\n",
516 				 handler->cmd_id, size, handler->sizes[i].size))
517 			return false;
518 		return true;
519 	}
520 
521 	IWL_FW_CHECK_FAILED(mld,
522 			    "notif 0x%04x ver %zu missing expected size, use version %u size\n",
523 			    handler->cmd_id, notif_ver,
524 			    handler->sizes[handler->n_sizes - 1].ver);
525 
526 	return size < handler->sizes[handler->n_sizes - 1].size;
527 }
528 
529 struct iwl_async_handler_entry {
530 	struct list_head list;
531 	struct iwl_rx_cmd_buffer rxb;
532 	const struct iwl_rx_handler *rx_h;
533 };
534 
535 static void
iwl_mld_log_async_handler_op(struct iwl_mld * mld,const char * op,struct iwl_rx_cmd_buffer * rxb)536 iwl_mld_log_async_handler_op(struct iwl_mld *mld, const char *op,
537 			     struct iwl_rx_cmd_buffer *rxb)
538 {
539 	struct iwl_rx_packet *pkt = rxb_addr(rxb);
540 
541 	IWL_DEBUG_HC(mld,
542 		     "%s async handler for notif %s (%.2x.%2x, seq 0x%x)\n",
543 		     op, iwl_get_cmd_string(mld->trans,
544 		     WIDE_ID(pkt->hdr.group_id, pkt->hdr.cmd)),
545 		     pkt->hdr.group_id, pkt->hdr.cmd,
546 		     le16_to_cpu(pkt->hdr.sequence));
547 }
548 
iwl_mld_rx_notif(struct iwl_mld * mld,struct iwl_rx_cmd_buffer * rxb,struct iwl_rx_packet * pkt)549 static void iwl_mld_rx_notif(struct iwl_mld *mld,
550 			     struct iwl_rx_cmd_buffer *rxb,
551 			     struct iwl_rx_packet *pkt)
552 {
553 	union iwl_dbg_tlv_tp_data tp_data = { .fw_pkt = pkt };
554 
555 	for (int i = 0; i < ARRAY_SIZE(iwl_mld_rx_handlers); i++) {
556 		const struct iwl_rx_handler *rx_h = &iwl_mld_rx_handlers[i];
557 		struct iwl_async_handler_entry *entry;
558 
559 		if (rx_h->cmd_id != WIDE_ID(pkt->hdr.group_id, pkt->hdr.cmd))
560 			continue;
561 
562 		if (!iwl_mld_notif_is_valid(mld, pkt, rx_h))
563 			return;
564 
565 		if (rx_h->context == RX_HANDLER_SYNC) {
566 			rx_h->fn(mld, pkt);
567 			break;
568 		}
569 
570 		entry = kzalloc_obj(*entry, GFP_ATOMIC);
571 		/* we can't do much... */
572 		if (!entry)
573 			return;
574 
575 		/* Set the async handler entry */
576 		entry->rxb._page = rxb_steal_page(rxb);
577 		entry->rxb._offset = rxb->_offset;
578 		entry->rxb._rx_page_order = rxb->_rx_page_order;
579 
580 		entry->rx_h = rx_h;
581 
582 		/* Add it to the list and queue the work */
583 		spin_lock(&mld->async_handlers_lock);
584 		list_add_tail(&entry->list, &mld->async_handlers_list);
585 		spin_unlock(&mld->async_handlers_lock);
586 
587 		wiphy_work_queue(mld->hw->wiphy,
588 				 &mld->async_handlers_wk);
589 
590 		iwl_mld_log_async_handler_op(mld, "Queued", rxb);
591 		break;
592 	}
593 
594 	iwl_notification_wait_notify(&mld->notif_wait, pkt);
595 	iwl_dbg_tlv_time_point(&mld->fwrt,
596 			       IWL_FW_INI_TIME_POINT_FW_RSP_OR_NOTIF, &tp_data);
597 }
598 
iwl_mld_rx(struct iwl_op_mode * op_mode,struct napi_struct * napi,struct iwl_rx_cmd_buffer * rxb)599 void iwl_mld_rx(struct iwl_op_mode *op_mode, struct napi_struct *napi,
600 		struct iwl_rx_cmd_buffer *rxb)
601 {
602 	struct iwl_rx_packet *pkt = rxb_addr(rxb);
603 	struct iwl_mld *mld = IWL_OP_MODE_GET_MLD(op_mode);
604 	u16 cmd_id = WIDE_ID(pkt->hdr.group_id, pkt->hdr.cmd);
605 
606 	if (likely(cmd_id == WIDE_ID(LEGACY_GROUP, REPLY_RX_MPDU_CMD)))
607 		iwl_mld_rx_mpdu(mld, napi, rxb, 0);
608 	else if (cmd_id == WIDE_ID(LEGACY_GROUP, FRAME_RELEASE))
609 		iwl_mld_handle_frame_release_notif(mld, napi, pkt, 0);
610 	else if (cmd_id == WIDE_ID(LEGACY_GROUP, BAR_FRAME_RELEASE))
611 		iwl_mld_handle_bar_frame_release_notif(mld, napi, pkt, 0);
612 	else if (unlikely(cmd_id == WIDE_ID(DATA_PATH_GROUP,
613 					    RX_QUEUES_NOTIFICATION)))
614 		iwl_mld_handle_rx_queues_sync_notif(mld, napi, pkt, 0);
615 	else if (cmd_id == WIDE_ID(DATA_PATH_GROUP, PHY_AIR_SNIFFER_NOTIF))
616 		iwl_mld_handle_phy_air_sniffer_notif(mld, napi, pkt);
617 	else
618 		iwl_mld_rx_notif(mld, rxb, pkt);
619 }
620 
iwl_mld_rx_rss(struct iwl_op_mode * op_mode,struct napi_struct * napi,struct iwl_rx_cmd_buffer * rxb,unsigned int queue)621 void iwl_mld_rx_rss(struct iwl_op_mode *op_mode, struct napi_struct *napi,
622 		    struct iwl_rx_cmd_buffer *rxb, unsigned int queue)
623 {
624 	struct iwl_rx_packet *pkt = rxb_addr(rxb);
625 	struct iwl_mld *mld = IWL_OP_MODE_GET_MLD(op_mode);
626 	u16 cmd_id = WIDE_ID(pkt->hdr.group_id, pkt->hdr.cmd);
627 
628 	if (unlikely(queue >= mld->trans->info.num_rxqs))
629 		return;
630 
631 	if (likely(cmd_id == WIDE_ID(LEGACY_GROUP, REPLY_RX_MPDU_CMD)))
632 		iwl_mld_rx_mpdu(mld, napi, rxb, queue);
633 	else if (unlikely(cmd_id == WIDE_ID(DATA_PATH_GROUP,
634 					    RX_QUEUES_NOTIFICATION)))
635 		iwl_mld_handle_rx_queues_sync_notif(mld, napi, pkt, queue);
636 	else if (unlikely(cmd_id == WIDE_ID(LEGACY_GROUP, FRAME_RELEASE)))
637 		iwl_mld_handle_frame_release_notif(mld, napi, pkt, queue);
638 }
639 
iwl_mld_delete_handlers(struct iwl_mld * mld,const u16 * cmds,int n_cmds)640 void iwl_mld_delete_handlers(struct iwl_mld *mld, const u16 *cmds, int n_cmds)
641 {
642 	struct iwl_async_handler_entry *entry, *tmp;
643 
644 	spin_lock_bh(&mld->async_handlers_lock);
645 	list_for_each_entry_safe(entry, tmp, &mld->async_handlers_list, list) {
646 		bool match = false;
647 
648 		for (int i = 0; i < n_cmds; i++) {
649 			if (entry->rx_h->cmd_id == cmds[i]) {
650 				match = true;
651 				break;
652 			}
653 		}
654 
655 		if (!match)
656 			continue;
657 
658 		iwl_mld_log_async_handler_op(mld, "Delete", &entry->rxb);
659 		iwl_free_rxb(&entry->rxb);
660 		list_del(&entry->list);
661 		kfree(entry);
662 	}
663 	spin_unlock_bh(&mld->async_handlers_lock);
664 }
665 
iwl_mld_async_handlers_wk(struct wiphy * wiphy,struct wiphy_work * wk)666 void iwl_mld_async_handlers_wk(struct wiphy *wiphy, struct wiphy_work *wk)
667 {
668 	struct iwl_mld *mld =
669 		container_of(wk, struct iwl_mld, async_handlers_wk);
670 	struct iwl_async_handler_entry *entry, *tmp;
671 	LIST_HEAD(local_list);
672 
673 	/* Sync with Rx path with a lock. Remove all the entries from this
674 	 * list, add them to a local one (lock free), and then handle them.
675 	 */
676 	spin_lock_bh(&mld->async_handlers_lock);
677 	list_splice_init(&mld->async_handlers_list, &local_list);
678 	spin_unlock_bh(&mld->async_handlers_lock);
679 
680 	list_for_each_entry_safe(entry, tmp, &local_list, list) {
681 		iwl_mld_log_async_handler_op(mld, "Handle", &entry->rxb);
682 		entry->rx_h->fn(mld, rxb_addr(&entry->rxb));
683 		iwl_free_rxb(&entry->rxb);
684 		list_del(&entry->list);
685 		kfree(entry);
686 	}
687 }
688 
iwl_mld_cancel_async_notifications(struct iwl_mld * mld)689 void iwl_mld_cancel_async_notifications(struct iwl_mld *mld)
690 {
691 	struct iwl_async_handler_entry *entry, *tmp;
692 
693 	lockdep_assert_wiphy(mld->wiphy);
694 
695 	wiphy_work_cancel(mld->wiphy, &mld->async_handlers_wk);
696 
697 	spin_lock_bh(&mld->async_handlers_lock);
698 	list_for_each_entry_safe(entry, tmp, &mld->async_handlers_list, list) {
699 		iwl_mld_log_async_handler_op(mld, "Purged", &entry->rxb);
700 		iwl_free_rxb(&entry->rxb);
701 		list_del(&entry->list);
702 		kfree(entry);
703 	}
704 	spin_unlock_bh(&mld->async_handlers_lock);
705 }
706 
iwl_mld_cancel_notifications_of_object(struct iwl_mld * mld,enum iwl_mld_object_type obj_type,u32 obj_id)707 void iwl_mld_cancel_notifications_of_object(struct iwl_mld *mld,
708 					    enum iwl_mld_object_type obj_type,
709 					    u32 obj_id)
710 {
711 	struct iwl_async_handler_entry *entry, *tmp;
712 	LIST_HEAD(cancel_list);
713 
714 	lockdep_assert_wiphy(mld->wiphy);
715 
716 	if (WARN_ON(obj_type == IWL_MLD_OBJECT_TYPE_NONE))
717 		return;
718 
719 	/* Sync with RX path and remove matching entries from the async list */
720 	spin_lock_bh(&mld->async_handlers_lock);
721 	list_for_each_entry_safe(entry, tmp, &mld->async_handlers_list, list) {
722 		const struct iwl_rx_handler *rx_h = entry->rx_h;
723 
724 		if (rx_h->obj_type != obj_type || WARN_ON(!rx_h->cancel))
725 			continue;
726 
727 		if (rx_h->cancel(mld, rxb_addr(&entry->rxb), obj_id)) {
728 			iwl_mld_log_async_handler_op(mld, "Cancel", &entry->rxb);
729 			list_del(&entry->list);
730 			list_add_tail(&entry->list, &cancel_list);
731 		}
732 	}
733 
734 	spin_unlock_bh(&mld->async_handlers_lock);
735 
736 	/* Free the matching entries outside of the spinlock */
737 	list_for_each_entry_safe(entry, tmp, &cancel_list, list) {
738 		iwl_free_rxb(&entry->rxb);
739 		list_del(&entry->list);
740 		kfree(entry);
741 	}
742 }
743