xref: /linux/drivers/net/wireless/intel/iwlwifi/mld/ftm-initiator.c (revision 1a9239bb4253f9076b5b4b2a1a4e8d7defd77a95)
1*d1e879ecSMiri Korenblit // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2*d1e879ecSMiri Korenblit /*
3*d1e879ecSMiri Korenblit  * Copyright (C) 2025 Intel Corporation
4*d1e879ecSMiri Korenblit  */
5*d1e879ecSMiri Korenblit #include <linux/etherdevice.h>
6*d1e879ecSMiri Korenblit #include <linux/math64.h>
7*d1e879ecSMiri Korenblit #include <net/cfg80211.h>
8*d1e879ecSMiri Korenblit #include "mld.h"
9*d1e879ecSMiri Korenblit #include "iface.h"
10*d1e879ecSMiri Korenblit #include "phy.h"
11*d1e879ecSMiri Korenblit #include "iwl-io.h"
12*d1e879ecSMiri Korenblit #include "iwl-prph.h"
13*d1e879ecSMiri Korenblit #include "constants.h"
14*d1e879ecSMiri Korenblit #include "fw/api/location.h"
15*d1e879ecSMiri Korenblit #include "ftm-initiator.h"
16*d1e879ecSMiri Korenblit 
iwl_mld_ftm_cmd_common(struct iwl_mld * mld,struct ieee80211_vif * vif,struct iwl_tof_range_req_cmd * cmd,struct cfg80211_pmsr_request * req)17*d1e879ecSMiri Korenblit static void iwl_mld_ftm_cmd_common(struct iwl_mld *mld,
18*d1e879ecSMiri Korenblit 				   struct ieee80211_vif *vif,
19*d1e879ecSMiri Korenblit 				   struct iwl_tof_range_req_cmd *cmd,
20*d1e879ecSMiri Korenblit 				   struct cfg80211_pmsr_request *req)
21*d1e879ecSMiri Korenblit {
22*d1e879ecSMiri Korenblit 	int i;
23*d1e879ecSMiri Korenblit 
24*d1e879ecSMiri Korenblit 	cmd->initiator_flags =
25*d1e879ecSMiri Korenblit 		cpu_to_le32(IWL_TOF_INITIATOR_FLAGS_MACADDR_RANDOM |
26*d1e879ecSMiri Korenblit 			    IWL_TOF_INITIATOR_FLAGS_NON_ASAP_SUPPORT);
27*d1e879ecSMiri Korenblit 	cmd->request_id = req->cookie;
28*d1e879ecSMiri Korenblit 	cmd->num_of_ap = req->n_peers;
29*d1e879ecSMiri Korenblit 
30*d1e879ecSMiri Korenblit 	/* Use a large value for "no timeout". Don't use the maximum value
31*d1e879ecSMiri Korenblit 	 * because of fw limitations.
32*d1e879ecSMiri Korenblit 	 */
33*d1e879ecSMiri Korenblit 	if (req->timeout)
34*d1e879ecSMiri Korenblit 		cmd->req_timeout_ms = cpu_to_le32(min(req->timeout, 0xfffff));
35*d1e879ecSMiri Korenblit 	else
36*d1e879ecSMiri Korenblit 		cmd->req_timeout_ms = cpu_to_le32(0xfffff);
37*d1e879ecSMiri Korenblit 
38*d1e879ecSMiri Korenblit 	memcpy(cmd->macaddr_template, req->mac_addr, ETH_ALEN);
39*d1e879ecSMiri Korenblit 	for (i = 0; i < ETH_ALEN; i++)
40*d1e879ecSMiri Korenblit 		cmd->macaddr_mask[i] = ~req->mac_addr_mask[i];
41*d1e879ecSMiri Korenblit 
42*d1e879ecSMiri Korenblit 	if (vif->cfg.assoc) {
43*d1e879ecSMiri Korenblit 		memcpy(cmd->range_req_bssid, vif->bss_conf.bssid, ETH_ALEN);
44*d1e879ecSMiri Korenblit 
45*d1e879ecSMiri Korenblit 		/* AP's TSF is only relevant if associated */
46*d1e879ecSMiri Korenblit 		for (i = 0; i < req->n_peers; i++) {
47*d1e879ecSMiri Korenblit 			if (req->peers[i].report_ap_tsf) {
48*d1e879ecSMiri Korenblit 				struct iwl_mld_vif *mld_vif =
49*d1e879ecSMiri Korenblit 					iwl_mld_vif_from_mac80211(vif);
50*d1e879ecSMiri Korenblit 
51*d1e879ecSMiri Korenblit 				cmd->tsf_mac_id = cpu_to_le32(mld_vif->fw_id);
52*d1e879ecSMiri Korenblit 				return;
53*d1e879ecSMiri Korenblit 			}
54*d1e879ecSMiri Korenblit 		}
55*d1e879ecSMiri Korenblit 	} else {
56*d1e879ecSMiri Korenblit 		eth_broadcast_addr(cmd->range_req_bssid);
57*d1e879ecSMiri Korenblit 	}
58*d1e879ecSMiri Korenblit 
59*d1e879ecSMiri Korenblit 	/* Don't report AP's TSF */
60*d1e879ecSMiri Korenblit 	cmd->tsf_mac_id = cpu_to_le32(0xff);
61*d1e879ecSMiri Korenblit }
62*d1e879ecSMiri Korenblit 
63*d1e879ecSMiri Korenblit static int
iwl_mld_ftm_set_target_chandef(struct iwl_mld * mld,struct cfg80211_pmsr_request_peer * peer,struct iwl_tof_range_req_ap_entry * target)64*d1e879ecSMiri Korenblit iwl_mld_ftm_set_target_chandef(struct iwl_mld *mld,
65*d1e879ecSMiri Korenblit 			       struct cfg80211_pmsr_request_peer *peer,
66*d1e879ecSMiri Korenblit 			       struct iwl_tof_range_req_ap_entry *target)
67*d1e879ecSMiri Korenblit {
68*d1e879ecSMiri Korenblit 	u32 freq = peer->chandef.chan->center_freq;
69*d1e879ecSMiri Korenblit 
70*d1e879ecSMiri Korenblit 	target->channel_num = ieee80211_frequency_to_channel(freq);
71*d1e879ecSMiri Korenblit 
72*d1e879ecSMiri Korenblit 	switch (peer->chandef.width) {
73*d1e879ecSMiri Korenblit 	case NL80211_CHAN_WIDTH_20_NOHT:
74*d1e879ecSMiri Korenblit 		target->format_bw = IWL_LOCATION_FRAME_FORMAT_LEGACY;
75*d1e879ecSMiri Korenblit 		target->format_bw |= IWL_LOCATION_BW_20MHZ << LOCATION_BW_POS;
76*d1e879ecSMiri Korenblit 		break;
77*d1e879ecSMiri Korenblit 	case NL80211_CHAN_WIDTH_20:
78*d1e879ecSMiri Korenblit 		target->format_bw = IWL_LOCATION_FRAME_FORMAT_HT;
79*d1e879ecSMiri Korenblit 		target->format_bw |= IWL_LOCATION_BW_20MHZ << LOCATION_BW_POS;
80*d1e879ecSMiri Korenblit 		break;
81*d1e879ecSMiri Korenblit 	case NL80211_CHAN_WIDTH_40:
82*d1e879ecSMiri Korenblit 		target->format_bw = IWL_LOCATION_FRAME_FORMAT_HT;
83*d1e879ecSMiri Korenblit 		target->format_bw |= IWL_LOCATION_BW_40MHZ << LOCATION_BW_POS;
84*d1e879ecSMiri Korenblit 		break;
85*d1e879ecSMiri Korenblit 	case NL80211_CHAN_WIDTH_80:
86*d1e879ecSMiri Korenblit 		target->format_bw = IWL_LOCATION_FRAME_FORMAT_VHT;
87*d1e879ecSMiri Korenblit 		target->format_bw |= IWL_LOCATION_BW_80MHZ << LOCATION_BW_POS;
88*d1e879ecSMiri Korenblit 		break;
89*d1e879ecSMiri Korenblit 	case NL80211_CHAN_WIDTH_160:
90*d1e879ecSMiri Korenblit 		target->format_bw = IWL_LOCATION_FRAME_FORMAT_HE;
91*d1e879ecSMiri Korenblit 		target->format_bw |= IWL_LOCATION_BW_160MHZ << LOCATION_BW_POS;
92*d1e879ecSMiri Korenblit 		break;
93*d1e879ecSMiri Korenblit 	default:
94*d1e879ecSMiri Korenblit 		IWL_ERR(mld, "Unsupported BW in FTM request (%d)\n",
95*d1e879ecSMiri Korenblit 			peer->chandef.width);
96*d1e879ecSMiri Korenblit 		return -EINVAL;
97*d1e879ecSMiri Korenblit }
98*d1e879ecSMiri Korenblit 
99*d1e879ecSMiri Korenblit 	/* non EDCA based measurement must use HE preamble */
100*d1e879ecSMiri Korenblit 	if (peer->ftm.trigger_based || peer->ftm.non_trigger_based)
101*d1e879ecSMiri Korenblit 		target->format_bw |= IWL_LOCATION_FRAME_FORMAT_HE;
102*d1e879ecSMiri Korenblit 
103*d1e879ecSMiri Korenblit 	target->ctrl_ch_position =
104*d1e879ecSMiri Korenblit 		(peer->chandef.width > NL80211_CHAN_WIDTH_20) ?
105*d1e879ecSMiri Korenblit 		iwl_mld_get_fw_ctrl_pos(&peer->chandef) : 0;
106*d1e879ecSMiri Korenblit 
107*d1e879ecSMiri Korenblit 	target->band = iwl_mld_nl80211_band_to_fw(peer->chandef.chan->band);
108*d1e879ecSMiri Korenblit 	return 0;
109*d1e879ecSMiri Korenblit }
110*d1e879ecSMiri Korenblit 
111*d1e879ecSMiri Korenblit #define FTM_SET_FLAG(flag) (target->initiator_ap_flags |= \
112*d1e879ecSMiri Korenblit 			    cpu_to_le32(IWL_INITIATOR_AP_FLAGS_##flag))
113*d1e879ecSMiri Korenblit 
114*d1e879ecSMiri Korenblit static void
iwl_mld_ftm_set_target_flags(struct iwl_mld * mld,struct cfg80211_pmsr_request_peer * peer,struct iwl_tof_range_req_ap_entry * target)115*d1e879ecSMiri Korenblit iwl_mld_ftm_set_target_flags(struct iwl_mld *mld,
116*d1e879ecSMiri Korenblit 			     struct cfg80211_pmsr_request_peer *peer,
117*d1e879ecSMiri Korenblit 			     struct iwl_tof_range_req_ap_entry *target)
118*d1e879ecSMiri Korenblit {
119*d1e879ecSMiri Korenblit 	target->initiator_ap_flags = cpu_to_le32(0);
120*d1e879ecSMiri Korenblit 
121*d1e879ecSMiri Korenblit 	if (peer->ftm.asap)
122*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(ASAP);
123*d1e879ecSMiri Korenblit 
124*d1e879ecSMiri Korenblit 	if (peer->ftm.request_lci)
125*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(LCI_REQUEST);
126*d1e879ecSMiri Korenblit 
127*d1e879ecSMiri Korenblit 	if (peer->ftm.request_civicloc)
128*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(CIVIC_REQUEST);
129*d1e879ecSMiri Korenblit 
130*d1e879ecSMiri Korenblit 	if (IWL_MLD_FTM_INITIATOR_DYNACK)
131*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(DYN_ACK);
132*d1e879ecSMiri Korenblit 
133*d1e879ecSMiri Korenblit 	if (IWL_MLD_FTM_INITIATOR_ALGO == IWL_TOF_ALGO_TYPE_LINEAR_REG)
134*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(ALGO_LR);
135*d1e879ecSMiri Korenblit 	else if (IWL_MLD_FTM_INITIATOR_ALGO == IWL_TOF_ALGO_TYPE_FFT)
136*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(ALGO_FFT);
137*d1e879ecSMiri Korenblit 
138*d1e879ecSMiri Korenblit 	if (peer->ftm.trigger_based)
139*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(TB);
140*d1e879ecSMiri Korenblit 	else if (peer->ftm.non_trigger_based)
141*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(NON_TB);
142*d1e879ecSMiri Korenblit 
143*d1e879ecSMiri Korenblit 	if ((peer->ftm.trigger_based || peer->ftm.non_trigger_based) &&
144*d1e879ecSMiri Korenblit 	    peer->ftm.lmr_feedback)
145*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(LMR_FEEDBACK);
146*d1e879ecSMiri Korenblit }
147*d1e879ecSMiri Korenblit 
iwl_mld_ftm_set_sta(struct iwl_mld * mld,struct ieee80211_vif * vif,struct cfg80211_pmsr_request_peer * peer,struct iwl_tof_range_req_ap_entry * target)148*d1e879ecSMiri Korenblit static void iwl_mld_ftm_set_sta(struct iwl_mld *mld, struct ieee80211_vif *vif,
149*d1e879ecSMiri Korenblit 				struct cfg80211_pmsr_request_peer *peer,
150*d1e879ecSMiri Korenblit 				struct iwl_tof_range_req_ap_entry *target)
151*d1e879ecSMiri Korenblit {
152*d1e879ecSMiri Korenblit 	struct iwl_mld_vif *mld_vif = iwl_mld_vif_from_mac80211(vif);
153*d1e879ecSMiri Korenblit 	u32 sta_id_mask;
154*d1e879ecSMiri Korenblit 
155*d1e879ecSMiri Korenblit 	target->sta_id = IWL_INVALID_STA;
156*d1e879ecSMiri Korenblit 
157*d1e879ecSMiri Korenblit 	/* TODO: add ftm_unprotected debugfs support */
158*d1e879ecSMiri Korenblit 
159*d1e879ecSMiri Korenblit 	if (!vif->cfg.assoc || !mld_vif->ap_sta)
160*d1e879ecSMiri Korenblit 		return;
161*d1e879ecSMiri Korenblit 
162*d1e879ecSMiri Korenblit 	sta_id_mask = iwl_mld_fw_sta_id_mask(mld, mld_vif->ap_sta);
163*d1e879ecSMiri Korenblit 	if (WARN_ON(hweight32(sta_id_mask) != 1))
164*d1e879ecSMiri Korenblit 		return;
165*d1e879ecSMiri Korenblit 
166*d1e879ecSMiri Korenblit 	target->sta_id = __ffs(sta_id_mask);
167*d1e879ecSMiri Korenblit 
168*d1e879ecSMiri Korenblit 	if (mld_vif->ap_sta->mfp &&
169*d1e879ecSMiri Korenblit 	    (peer->ftm.trigger_based || peer->ftm.non_trigger_based))
170*d1e879ecSMiri Korenblit 		FTM_SET_FLAG(PMF);
171*d1e879ecSMiri Korenblit }
172*d1e879ecSMiri Korenblit 
173*d1e879ecSMiri Korenblit static int
iwl_mld_ftm_set_target(struct iwl_mld * mld,struct ieee80211_vif * vif,struct cfg80211_pmsr_request_peer * peer,struct iwl_tof_range_req_ap_entry * target)174*d1e879ecSMiri Korenblit iwl_mld_ftm_set_target(struct iwl_mld *mld, struct ieee80211_vif *vif,
175*d1e879ecSMiri Korenblit 		       struct cfg80211_pmsr_request_peer *peer,
176*d1e879ecSMiri Korenblit 		       struct iwl_tof_range_req_ap_entry *target)
177*d1e879ecSMiri Korenblit {
178*d1e879ecSMiri Korenblit 	u32 i2r_max_sts;
179*d1e879ecSMiri Korenblit 	int ret;
180*d1e879ecSMiri Korenblit 
181*d1e879ecSMiri Korenblit 	ret = iwl_mld_ftm_set_target_chandef(mld, peer, target);
182*d1e879ecSMiri Korenblit 	if (ret)
183*d1e879ecSMiri Korenblit 		return ret;
184*d1e879ecSMiri Korenblit 
185*d1e879ecSMiri Korenblit 	memcpy(target->bssid, peer->addr, ETH_ALEN);
186*d1e879ecSMiri Korenblit 	target->burst_period = cpu_to_le16(peer->ftm.burst_period);
187*d1e879ecSMiri Korenblit 	target->samples_per_burst = peer->ftm.ftms_per_burst;
188*d1e879ecSMiri Korenblit 	target->num_of_bursts = peer->ftm.num_bursts_exp;
189*d1e879ecSMiri Korenblit 	iwl_mld_ftm_set_target_flags(mld, peer, target);
190*d1e879ecSMiri Korenblit 	iwl_mld_ftm_set_sta(mld, vif, peer, target);
191*d1e879ecSMiri Korenblit 
192*d1e879ecSMiri Korenblit 	/* TODO: add secured ranging support */
193*d1e879ecSMiri Korenblit 
194*d1e879ecSMiri Korenblit 	i2r_max_sts = IWL_MLD_FTM_I2R_MAX_STS > 1 ? 1 :
195*d1e879ecSMiri Korenblit 		IWL_MLD_FTM_I2R_MAX_STS;
196*d1e879ecSMiri Korenblit 
197*d1e879ecSMiri Korenblit 	target->r2i_ndp_params = IWL_MLD_FTM_R2I_MAX_REP |
198*d1e879ecSMiri Korenblit 		(IWL_MLD_FTM_R2I_MAX_STS << IWL_LOCATION_MAX_STS_POS) |
199*d1e879ecSMiri Korenblit 		(IWL_MLD_FTM_R2I_MAX_TOTAL_LTF << IWL_LOCATION_TOTAL_LTF_POS);
200*d1e879ecSMiri Korenblit 	target->i2r_ndp_params = IWL_MLD_FTM_I2R_MAX_REP |
201*d1e879ecSMiri Korenblit 		(i2r_max_sts << IWL_LOCATION_MAX_STS_POS) |
202*d1e879ecSMiri Korenblit 		(IWL_MLD_FTM_I2R_MAX_TOTAL_LTF << IWL_LOCATION_TOTAL_LTF_POS);
203*d1e879ecSMiri Korenblit 
204*d1e879ecSMiri Korenblit 	if (peer->ftm.non_trigger_based) {
205*d1e879ecSMiri Korenblit 		target->min_time_between_msr =
206*d1e879ecSMiri Korenblit 			cpu_to_le16(IWL_MLD_FTM_NON_TB_MIN_TIME_BETWEEN_MSR);
207*d1e879ecSMiri Korenblit 		target->burst_period =
208*d1e879ecSMiri Korenblit 			cpu_to_le16(IWL_MLD_FTM_NON_TB_MAX_TIME_BETWEEN_MSR);
209*d1e879ecSMiri Korenblit 	} else {
210*d1e879ecSMiri Korenblit 		target->min_time_between_msr = cpu_to_le16(0);
211*d1e879ecSMiri Korenblit 	}
212*d1e879ecSMiri Korenblit 
213*d1e879ecSMiri Korenblit 	/* TODO: Beacon interval is currently unknown, so use the common value
214*d1e879ecSMiri Korenblit 	 * of 100 TUs.
215*d1e879ecSMiri Korenblit 	 */
216*d1e879ecSMiri Korenblit 	target->beacon_interval = cpu_to_le16(100);
217*d1e879ecSMiri Korenblit 
218*d1e879ecSMiri Korenblit 	return 0;
219*d1e879ecSMiri Korenblit }
220*d1e879ecSMiri Korenblit 
iwl_mld_ftm_start(struct iwl_mld * mld,struct ieee80211_vif * vif,struct cfg80211_pmsr_request * req)221*d1e879ecSMiri Korenblit int iwl_mld_ftm_start(struct iwl_mld *mld, struct ieee80211_vif *vif,
222*d1e879ecSMiri Korenblit 		      struct cfg80211_pmsr_request *req)
223*d1e879ecSMiri Korenblit {
224*d1e879ecSMiri Korenblit 	struct iwl_tof_range_req_cmd cmd;
225*d1e879ecSMiri Korenblit 	struct iwl_host_cmd hcmd = {
226*d1e879ecSMiri Korenblit 		.id = WIDE_ID(LOCATION_GROUP, TOF_RANGE_REQ_CMD),
227*d1e879ecSMiri Korenblit 		.dataflags[0] = IWL_HCMD_DFL_DUP,
228*d1e879ecSMiri Korenblit 		.data[0] = &cmd,
229*d1e879ecSMiri Korenblit 		.len[0] = sizeof(cmd),
230*d1e879ecSMiri Korenblit 	};
231*d1e879ecSMiri Korenblit 	u8 i;
232*d1e879ecSMiri Korenblit 	int ret;
233*d1e879ecSMiri Korenblit 
234*d1e879ecSMiri Korenblit 	lockdep_assert_wiphy(mld->wiphy);
235*d1e879ecSMiri Korenblit 
236*d1e879ecSMiri Korenblit 	if (mld->ftm_initiator.req)
237*d1e879ecSMiri Korenblit 		return -EBUSY;
238*d1e879ecSMiri Korenblit 
239*d1e879ecSMiri Korenblit 	if (req->n_peers > ARRAY_SIZE(cmd.ap))
240*d1e879ecSMiri Korenblit 		return -EINVAL;
241*d1e879ecSMiri Korenblit 
242*d1e879ecSMiri Korenblit 	memset(&cmd, 0, sizeof(cmd));
243*d1e879ecSMiri Korenblit 
244*d1e879ecSMiri Korenblit 	iwl_mld_ftm_cmd_common(mld, vif, (void *)&cmd, req);
245*d1e879ecSMiri Korenblit 
246*d1e879ecSMiri Korenblit 	for (i = 0; i < cmd.num_of_ap; i++) {
247*d1e879ecSMiri Korenblit 		struct cfg80211_pmsr_request_peer *peer = &req->peers[i];
248*d1e879ecSMiri Korenblit 		struct iwl_tof_range_req_ap_entry *target = &cmd.ap[i];
249*d1e879ecSMiri Korenblit 
250*d1e879ecSMiri Korenblit 		ret = iwl_mld_ftm_set_target(mld, vif, peer, target);
251*d1e879ecSMiri Korenblit 		if (ret)
252*d1e879ecSMiri Korenblit 			return ret;
253*d1e879ecSMiri Korenblit 	}
254*d1e879ecSMiri Korenblit 
255*d1e879ecSMiri Korenblit 	/* TODO: get the status from the response*/
256*d1e879ecSMiri Korenblit 	ret = iwl_mld_send_cmd(mld, &hcmd);
257*d1e879ecSMiri Korenblit 	if (!ret) {
258*d1e879ecSMiri Korenblit 		mld->ftm_initiator.req = req;
259*d1e879ecSMiri Korenblit 		mld->ftm_initiator.req_wdev = ieee80211_vif_to_wdev(vif);
260*d1e879ecSMiri Korenblit 	}
261*d1e879ecSMiri Korenblit 
262*d1e879ecSMiri Korenblit 	return ret;
263*d1e879ecSMiri Korenblit }
264*d1e879ecSMiri Korenblit 
iwl_mld_ftm_reset(struct iwl_mld * mld)265*d1e879ecSMiri Korenblit static void iwl_mld_ftm_reset(struct iwl_mld *mld)
266*d1e879ecSMiri Korenblit {
267*d1e879ecSMiri Korenblit 	lockdep_assert_wiphy(mld->wiphy);
268*d1e879ecSMiri Korenblit 
269*d1e879ecSMiri Korenblit 	mld->ftm_initiator.req = NULL;
270*d1e879ecSMiri Korenblit 	mld->ftm_initiator.req_wdev = NULL;
271*d1e879ecSMiri Korenblit 	memset(mld->ftm_initiator.responses, 0,
272*d1e879ecSMiri Korenblit 	       sizeof(mld->ftm_initiator.responses));
273*d1e879ecSMiri Korenblit }
274*d1e879ecSMiri Korenblit 
iwl_mld_ftm_range_resp_valid(struct iwl_mld * mld,u8 request_id,u8 num_of_aps)275*d1e879ecSMiri Korenblit static int iwl_mld_ftm_range_resp_valid(struct iwl_mld *mld, u8 request_id,
276*d1e879ecSMiri Korenblit 					u8 num_of_aps)
277*d1e879ecSMiri Korenblit {
278*d1e879ecSMiri Korenblit 	if (IWL_FW_CHECK(mld, request_id != (u8)mld->ftm_initiator.req->cookie,
279*d1e879ecSMiri Korenblit 			 "Request ID mismatch, got %u, active %u\n",
280*d1e879ecSMiri Korenblit 			 request_id, (u8)mld->ftm_initiator.req->cookie))
281*d1e879ecSMiri Korenblit 		return -EINVAL;
282*d1e879ecSMiri Korenblit 
283*d1e879ecSMiri Korenblit 	if (IWL_FW_CHECK(mld, num_of_aps > mld->ftm_initiator.req->n_peers ||
284*d1e879ecSMiri Korenblit 			 num_of_aps > IWL_TOF_MAX_APS,
285*d1e879ecSMiri Korenblit 			 "FTM range response: invalid num of APs (%u)\n",
286*d1e879ecSMiri Korenblit 			 num_of_aps))
287*d1e879ecSMiri Korenblit 		return -EINVAL;
288*d1e879ecSMiri Korenblit 
289*d1e879ecSMiri Korenblit 	return 0;
290*d1e879ecSMiri Korenblit }
291*d1e879ecSMiri Korenblit 
iwl_mld_ftm_find_peer(struct cfg80211_pmsr_request * req,const u8 * addr)292*d1e879ecSMiri Korenblit static int iwl_mld_ftm_find_peer(struct cfg80211_pmsr_request *req,
293*d1e879ecSMiri Korenblit 				 const u8 *addr)
294*d1e879ecSMiri Korenblit {
295*d1e879ecSMiri Korenblit 	for (int i = 0; i < req->n_peers; i++) {
296*d1e879ecSMiri Korenblit 		struct cfg80211_pmsr_request_peer *peer = &req->peers[i];
297*d1e879ecSMiri Korenblit 
298*d1e879ecSMiri Korenblit 		if (ether_addr_equal_unaligned(peer->addr, addr))
299*d1e879ecSMiri Korenblit 			return i;
300*d1e879ecSMiri Korenblit 	}
301*d1e879ecSMiri Korenblit 
302*d1e879ecSMiri Korenblit 	return -ENOENT;
303*d1e879ecSMiri Korenblit }
304*d1e879ecSMiri Korenblit 
iwl_mld_debug_range_resp(struct iwl_mld * mld,u8 index,struct cfg80211_pmsr_result * res)305*d1e879ecSMiri Korenblit static void iwl_mld_debug_range_resp(struct iwl_mld *mld, u8 index,
306*d1e879ecSMiri Korenblit 				     struct cfg80211_pmsr_result *res)
307*d1e879ecSMiri Korenblit {
308*d1e879ecSMiri Korenblit 	s64 rtt_avg = div_s64(res->ftm.rtt_avg * 100, 6666);
309*d1e879ecSMiri Korenblit 
310*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "entry %d\n", index);
311*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\tstatus: %d\n", res->status);
312*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\tBSSID: %pM\n", res->addr);
313*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\thost time: %llu\n", res->host_time);
314*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\tburst index: %d\n", res->ftm.burst_index);
315*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\tsuccess num: %u\n", res->ftm.num_ftmr_successes);
316*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\trssi: %d\n", res->ftm.rssi_avg);
317*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\trssi spread: %d\n", res->ftm.rssi_spread);
318*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\trtt: %lld\n", res->ftm.rtt_avg);
319*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\trtt var: %llu\n", res->ftm.rtt_variance);
320*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\trtt spread: %llu\n", res->ftm.rtt_spread);
321*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "\tdistance: %lld\n", rtt_avg);
322*d1e879ecSMiri Korenblit }
323*d1e879ecSMiri Korenblit 
iwl_mld_handle_ftm_resp_notif(struct iwl_mld * mld,struct iwl_rx_packet * pkt)324*d1e879ecSMiri Korenblit void iwl_mld_handle_ftm_resp_notif(struct iwl_mld *mld,
325*d1e879ecSMiri Korenblit 				   struct iwl_rx_packet *pkt)
326*d1e879ecSMiri Korenblit {
327*d1e879ecSMiri Korenblit 	struct iwl_tof_range_rsp_ntfy *fw_resp = (void *)pkt->data;
328*d1e879ecSMiri Korenblit 	u8 num_of_aps, last_in_batch;
329*d1e879ecSMiri Korenblit 
330*d1e879ecSMiri Korenblit 	if (IWL_FW_CHECK(mld, !mld->ftm_initiator.req,
331*d1e879ecSMiri Korenblit 			 "FTM response without a pending request\n"))
332*d1e879ecSMiri Korenblit 		return;
333*d1e879ecSMiri Korenblit 
334*d1e879ecSMiri Korenblit 	if (iwl_mld_ftm_range_resp_valid(mld, fw_resp->request_id,
335*d1e879ecSMiri Korenblit 					 fw_resp->num_of_aps))
336*d1e879ecSMiri Korenblit 		return;
337*d1e879ecSMiri Korenblit 
338*d1e879ecSMiri Korenblit 	num_of_aps = fw_resp->num_of_aps;
339*d1e879ecSMiri Korenblit 	last_in_batch = fw_resp->last_report;
340*d1e879ecSMiri Korenblit 
341*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "Range response received\n");
342*d1e879ecSMiri Korenblit 	IWL_DEBUG_INFO(mld, "request id: %llu, num of entries: %u\n",
343*d1e879ecSMiri Korenblit 		       mld->ftm_initiator.req->cookie, num_of_aps);
344*d1e879ecSMiri Korenblit 
345*d1e879ecSMiri Korenblit 	for (int i = 0; i < num_of_aps; i++) {
346*d1e879ecSMiri Korenblit 		struct cfg80211_pmsr_result result = {};
347*d1e879ecSMiri Korenblit 		struct iwl_tof_range_rsp_ap_entry_ntfy *fw_ap;
348*d1e879ecSMiri Korenblit 		int peer_idx;
349*d1e879ecSMiri Korenblit 
350*d1e879ecSMiri Korenblit 		fw_ap = &fw_resp->ap[i];
351*d1e879ecSMiri Korenblit 		result.final = fw_ap->last_burst;
352*d1e879ecSMiri Korenblit 		result.ap_tsf = le32_to_cpu(fw_ap->start_tsf);
353*d1e879ecSMiri Korenblit 		result.ap_tsf_valid = 1;
354*d1e879ecSMiri Korenblit 
355*d1e879ecSMiri Korenblit 		peer_idx = iwl_mld_ftm_find_peer(mld->ftm_initiator.req,
356*d1e879ecSMiri Korenblit 						 fw_ap->bssid);
357*d1e879ecSMiri Korenblit 		if (peer_idx < 0) {
358*d1e879ecSMiri Korenblit 			IWL_WARN(mld,
359*d1e879ecSMiri Korenblit 				 "Unknown address (%pM, target #%d) in FTM response\n",
360*d1e879ecSMiri Korenblit 				 fw_ap->bssid, i);
361*d1e879ecSMiri Korenblit 			continue;
362*d1e879ecSMiri Korenblit 		}
363*d1e879ecSMiri Korenblit 
364*d1e879ecSMiri Korenblit 		switch (fw_ap->measure_status) {
365*d1e879ecSMiri Korenblit 		case IWL_TOF_ENTRY_SUCCESS:
366*d1e879ecSMiri Korenblit 			result.status = NL80211_PMSR_STATUS_SUCCESS;
367*d1e879ecSMiri Korenblit 			break;
368*d1e879ecSMiri Korenblit 		case IWL_TOF_ENTRY_TIMING_MEASURE_TIMEOUT:
369*d1e879ecSMiri Korenblit 			result.status = NL80211_PMSR_STATUS_TIMEOUT;
370*d1e879ecSMiri Korenblit 			break;
371*d1e879ecSMiri Korenblit 		case IWL_TOF_ENTRY_NO_RESPONSE:
372*d1e879ecSMiri Korenblit 			result.status = NL80211_PMSR_STATUS_FAILURE;
373*d1e879ecSMiri Korenblit 			result.ftm.failure_reason =
374*d1e879ecSMiri Korenblit 				NL80211_PMSR_FTM_FAILURE_NO_RESPONSE;
375*d1e879ecSMiri Korenblit 			break;
376*d1e879ecSMiri Korenblit 		case IWL_TOF_ENTRY_REQUEST_REJECTED:
377*d1e879ecSMiri Korenblit 			result.status = NL80211_PMSR_STATUS_FAILURE;
378*d1e879ecSMiri Korenblit 			result.ftm.failure_reason =
379*d1e879ecSMiri Korenblit 				NL80211_PMSR_FTM_FAILURE_PEER_BUSY;
380*d1e879ecSMiri Korenblit 			result.ftm.busy_retry_time = fw_ap->refusal_period;
381*d1e879ecSMiri Korenblit 			break;
382*d1e879ecSMiri Korenblit 		default:
383*d1e879ecSMiri Korenblit 			result.status = NL80211_PMSR_STATUS_FAILURE;
384*d1e879ecSMiri Korenblit 			result.ftm.failure_reason =
385*d1e879ecSMiri Korenblit 				NL80211_PMSR_FTM_FAILURE_UNSPECIFIED;
386*d1e879ecSMiri Korenblit 			break;
387*d1e879ecSMiri Korenblit 		}
388*d1e879ecSMiri Korenblit 		memcpy(result.addr, fw_ap->bssid, ETH_ALEN);
389*d1e879ecSMiri Korenblit 
390*d1e879ecSMiri Korenblit 		/* TODO: convert the timestamp from the result to systime */
391*d1e879ecSMiri Korenblit 		result.host_time = ktime_get_boottime_ns();
392*d1e879ecSMiri Korenblit 
393*d1e879ecSMiri Korenblit 		result.type = NL80211_PMSR_TYPE_FTM;
394*d1e879ecSMiri Korenblit 		result.ftm.burst_index = mld->ftm_initiator.responses[peer_idx];
395*d1e879ecSMiri Korenblit 		mld->ftm_initiator.responses[peer_idx]++;
396*d1e879ecSMiri Korenblit 		result.ftm.rssi_avg = fw_ap->rssi;
397*d1e879ecSMiri Korenblit 		result.ftm.rssi_avg_valid = 1;
398*d1e879ecSMiri Korenblit 		result.ftm.rssi_spread = fw_ap->rssi_spread;
399*d1e879ecSMiri Korenblit 		result.ftm.rssi_spread_valid = 1;
400*d1e879ecSMiri Korenblit 		result.ftm.rtt_avg = (s32)le32_to_cpu(fw_ap->rtt);
401*d1e879ecSMiri Korenblit 		result.ftm.rtt_avg_valid = 1;
402*d1e879ecSMiri Korenblit 		result.ftm.rtt_variance = le32_to_cpu(fw_ap->rtt_variance);
403*d1e879ecSMiri Korenblit 		result.ftm.rtt_variance_valid = 1;
404*d1e879ecSMiri Korenblit 		result.ftm.rtt_spread = le32_to_cpu(fw_ap->rtt_spread);
405*d1e879ecSMiri Korenblit 		result.ftm.rtt_spread_valid = 1;
406*d1e879ecSMiri Korenblit 
407*d1e879ecSMiri Korenblit 		cfg80211_pmsr_report(mld->ftm_initiator.req_wdev,
408*d1e879ecSMiri Korenblit 				     mld->ftm_initiator.req,
409*d1e879ecSMiri Korenblit 				     &result, GFP_KERNEL);
410*d1e879ecSMiri Korenblit 
411*d1e879ecSMiri Korenblit 		if (fw_has_api(&mld->fw->ucode_capa,
412*d1e879ecSMiri Korenblit 			       IWL_UCODE_TLV_API_FTM_RTT_ACCURACY))
413*d1e879ecSMiri Korenblit 			IWL_DEBUG_INFO(mld, "RTT confidence: %u\n",
414*d1e879ecSMiri Korenblit 				       fw_ap->rttConfidence);
415*d1e879ecSMiri Korenblit 
416*d1e879ecSMiri Korenblit 		iwl_mld_debug_range_resp(mld, i, &result);
417*d1e879ecSMiri Korenblit 	}
418*d1e879ecSMiri Korenblit 
419*d1e879ecSMiri Korenblit 	if (last_in_batch) {
420*d1e879ecSMiri Korenblit 		cfg80211_pmsr_complete(mld->ftm_initiator.req_wdev,
421*d1e879ecSMiri Korenblit 				       mld->ftm_initiator.req,
422*d1e879ecSMiri Korenblit 				       GFP_KERNEL);
423*d1e879ecSMiri Korenblit 		iwl_mld_ftm_reset(mld);
424*d1e879ecSMiri Korenblit 	}
425*d1e879ecSMiri Korenblit }
426*d1e879ecSMiri Korenblit 
iwl_mld_ftm_restart_cleanup(struct iwl_mld * mld)427*d1e879ecSMiri Korenblit void iwl_mld_ftm_restart_cleanup(struct iwl_mld *mld)
428*d1e879ecSMiri Korenblit {
429*d1e879ecSMiri Korenblit 	struct cfg80211_pmsr_result result = {
430*d1e879ecSMiri Korenblit 		.status = NL80211_PMSR_STATUS_FAILURE,
431*d1e879ecSMiri Korenblit 		.final = 1,
432*d1e879ecSMiri Korenblit 		.host_time = ktime_get_boottime_ns(),
433*d1e879ecSMiri Korenblit 		.type = NL80211_PMSR_TYPE_FTM,
434*d1e879ecSMiri Korenblit 	};
435*d1e879ecSMiri Korenblit 
436*d1e879ecSMiri Korenblit 	if (!mld->ftm_initiator.req)
437*d1e879ecSMiri Korenblit 		return;
438*d1e879ecSMiri Korenblit 
439*d1e879ecSMiri Korenblit 	for (int i = 0; i < mld->ftm_initiator.req->n_peers; i++) {
440*d1e879ecSMiri Korenblit 		memcpy(result.addr, mld->ftm_initiator.req->peers[i].addr,
441*d1e879ecSMiri Korenblit 		       ETH_ALEN);
442*d1e879ecSMiri Korenblit 
443*d1e879ecSMiri Korenblit 		cfg80211_pmsr_report(mld->ftm_initiator.req_wdev,
444*d1e879ecSMiri Korenblit 				     mld->ftm_initiator.req,
445*d1e879ecSMiri Korenblit 				     &result, GFP_KERNEL);
446*d1e879ecSMiri Korenblit 	}
447*d1e879ecSMiri Korenblit 
448*d1e879ecSMiri Korenblit 	cfg80211_pmsr_complete(mld->ftm_initiator.req_wdev,
449*d1e879ecSMiri Korenblit 			       mld->ftm_initiator.req, GFP_KERNEL);
450*d1e879ecSMiri Korenblit 	iwl_mld_ftm_reset(mld);
451*d1e879ecSMiri Korenblit }
452