1  /*
2   * Copyright (c) 2017-2021 The Linux Foundation. All rights reserved.
3   * Copyright (c) 2021-2024 Qualcomm Innovation Center, Inc. All rights reserved.
4   *
5   * Permission to use, copy, modify, and/or distribute this software for
6   * any purpose with or without fee is hereby granted, provided that the
7   * above copyright notice and this permission notice appear in all
8   * copies.
9   *
10   * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
11   * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
12   * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
13   * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
14   * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
15   * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
16   * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17   * PERFORMANCE OF THIS SOFTWARE.
18   */
19  
20  /**
21   * DOC: This file contains off channel tx API definitions
22   */
23  
24  #include <wmi_unified_api.h>
25  #include <wlan_mgmt_txrx_utils_api.h>
26  #include <wlan_objmgr_psoc_obj.h>
27  #include <wlan_objmgr_peer_obj.h>
28  #include <wlan_utility.h>
29  #include <scheduler_api.h>
30  #include "wlan_p2p_public_struct.h"
31  #include "wlan_p2p_tgt_api.h"
32  #include "wlan_p2p_ucfg_api.h"
33  #include "wlan_p2p_roc.h"
34  #include "wlan_p2p_main.h"
35  #include "wlan_p2p_off_chan_tx.h"
36  #include "wlan_osif_request_manager.h"
37  #include <wlan_mlme_main.h>
38  #include "wlan_mlme_api.h"
39  #include <wlan_cm_api.h>
40  #include <wlan_mlo_mgr_sta.h>
41  
42  /**
43   * p2p_psoc_get_tx_ops() - get p2p tx ops
44   * @psoc:        psoc object
45   *
46   * This function returns p2p tx ops callbacks.
47   *
48   * Return: wlan_lmac_if_p2p_tx_ops
49   */
50  static inline struct wlan_lmac_if_p2p_tx_ops *
p2p_psoc_get_tx_ops(struct wlan_objmgr_psoc * psoc)51  p2p_psoc_get_tx_ops(struct wlan_objmgr_psoc *psoc)
52  {
53  	return &psoc->soc_cb.tx_ops->p2p;
54  }
55  
56  /**
57   * p2p_tx_context_check_valid() - check tx action context
58   * @tx_ctx:         tx context
59   *
60   * This function check if tx action context and parameters are valid.
61   *
62   * Return: QDF_STATUS_SUCCESS - in case of success
63   */
p2p_tx_context_check_valid(struct tx_action_context * tx_ctx)64  static QDF_STATUS p2p_tx_context_check_valid(struct tx_action_context *tx_ctx)
65  {
66  	struct wlan_objmgr_psoc *psoc;
67  	struct p2p_soc_priv_obj *p2p_soc_obj;
68  
69  	if (!tx_ctx) {
70  		p2p_err("null tx action context");
71  		return QDF_STATUS_E_INVAL;
72  	}
73  
74  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
75  	if (!p2p_soc_obj) {
76  		p2p_err("null p2p soc private object");
77  		return QDF_STATUS_E_INVAL;
78  	}
79  
80  	psoc = p2p_soc_obj->soc;
81  	if (!psoc) {
82  		p2p_err("null p2p soc object");
83  		return QDF_STATUS_E_INVAL;
84  	}
85  
86  	if (!tx_ctx->buf) {
87  		p2p_err("null tx buffer");
88  		return QDF_STATUS_E_INVAL;
89  	}
90  
91  	return QDF_STATUS_SUCCESS;
92  }
93  
94  /**
95   * p2p_vdev_check_valid() - check vdev and vdev mode
96   * @tx_ctx:         tx context
97   *
98   * This function check if vdev and vdev mode are valid. It will drop
99   * probe response in sta mode.
100   *
101   * Return: QDF_STATUS_SUCCESS - in case of success
102   */
103  #ifdef SUPPORT_P2P_BY_ONE_INTF_WLAN
p2p_vdev_check_valid(struct tx_action_context * tx_ctx)104  static QDF_STATUS p2p_vdev_check_valid(struct tx_action_context *tx_ctx)
105  {
106  	enum QDF_OPMODE mode;
107  	struct wlan_objmgr_vdev *vdev;
108  	struct wlan_objmgr_psoc *psoc;
109  	struct p2p_soc_priv_obj *p2p_soc_obj;
110  	QDF_STATUS status = QDF_STATUS_SUCCESS;
111  
112  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
113  	psoc = p2p_soc_obj->soc;
114  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
115  			psoc, tx_ctx->vdev_id, WLAN_P2P_ID);
116  	if (!vdev) {
117  		p2p_err("null vdev object");
118  		return QDF_STATUS_E_INVAL;
119  	}
120  
121  	mode = wlan_vdev_mlme_get_opmode(vdev);
122  	p2p_debug("vdev mode:%d", mode);
123  
124  	/* drop probe response/disassoc/deauth for go, sap */
125  	if ((mode == QDF_SAP_MODE ||
126  	     mode == QDF_P2P_GO_MODE) &&
127  	    ((tx_ctx->frame_info.sub_type == P2P_MGMT_PROBE_RSP) ||
128  	     (tx_ctx->frame_info.sub_type == P2P_MGMT_DISASSOC) ||
129  	     (tx_ctx->frame_info.sub_type == P2P_MGMT_DEAUTH))) {
130  		p2p_debug("drop frame, mode:%d, sub type:%d", mode,
131  			  tx_ctx->frame_info.sub_type);
132  		status = QDF_STATUS_E_FAILURE;
133  	}
134  
135  	/* drop action frame for sap */
136  	if ((mode == QDF_SAP_MODE) &&
137  	    (tx_ctx->frame_info.sub_type == P2P_MGMT_ACTION) &&
138  	    (tx_ctx->frame_info.public_action_type ==
139  	     P2P_PUBLIC_ACTION_NOT_SUPPORT) &&
140  	    (tx_ctx->frame_info.action_type == P2P_ACTION_NOT_SUPPORT) &&
141  	    !(wlan_vdev_mlme_feat_cap_get(vdev, WLAN_VDEV_F_SON) &&
142  	      !tx_ctx->off_chan)) {
143  		p2p_debug("drop action frame for SAP");
144  		status = QDF_STATUS_E_FAILURE;
145  	}
146  
147  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
148  
149  	return status;
150  }
151  #else
p2p_vdev_check_valid(struct tx_action_context * tx_ctx)152  static QDF_STATUS p2p_vdev_check_valid(struct tx_action_context *tx_ctx)
153  {
154  	enum QDF_OPMODE mode;
155  	struct wlan_objmgr_vdev *vdev;
156  	struct wlan_objmgr_psoc *psoc;
157  	struct p2p_soc_priv_obj *p2p_soc_obj;
158  	QDF_STATUS status = QDF_STATUS_SUCCESS;
159  
160  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
161  	psoc = p2p_soc_obj->soc;
162  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
163  			psoc, tx_ctx->vdev_id, WLAN_P2P_ID);
164  	if (!vdev) {
165  		p2p_err("null vdev object");
166  		return QDF_STATUS_E_INVAL;
167  	}
168  
169  	mode = wlan_vdev_mlme_get_opmode(vdev);
170  	p2p_debug("vdev mode:%d", mode);
171  
172  	/* drop probe response/disassoc/deauth for sta, go, sap */
173  	if ((mode == QDF_STA_MODE &&
174  	     tx_ctx->frame_info.sub_type == P2P_MGMT_PROBE_RSP) ||
175  	    ((mode == QDF_SAP_MODE || mode == QDF_P2P_GO_MODE) &&
176  	     ((tx_ctx->frame_info.sub_type == P2P_MGMT_PROBE_RSP) ||
177  	     (tx_ctx->frame_info.sub_type == P2P_MGMT_DISASSOC) ||
178  	     (tx_ctx->frame_info.sub_type == P2P_MGMT_DEAUTH)))) {
179  		p2p_debug("drop frame, mode:%d, sub type:%d", mode,
180  			  tx_ctx->frame_info.sub_type);
181  		status = QDF_STATUS_E_FAILURE;
182  	}
183  
184  	/* drop ation frame for sap */
185  	if ((mode == QDF_SAP_MODE) &&
186  	    (tx_ctx->frame_info.sub_type == P2P_MGMT_ACTION) &&
187  	    (tx_ctx->frame_info.public_action_type ==
188  	     P2P_PUBLIC_ACTION_NOT_SUPPORT) &&
189  	    (tx_ctx->frame_info.action_type == P2P_ACTION_NOT_SUPPORT) &&
190  	    !(wlan_vdev_mlme_feat_cap_get(vdev, WLAN_VDEV_F_SON) &&
191  	      !tx_ctx->off_chan)) {
192  		p2p_debug("drop action frame for SAP");
193  		status = QDF_STATUS_E_FAILURE;
194  	}
195  
196  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
197  
198  	return status;
199  }
200  #endif /* SUPPORT_P2P_BY_ONE_INTF_WLAN */
201  
202  /**
203   * p2p_check_and_update_channel() - check and update tx channel
204   * @tx_ctx:         tx context
205   *
206   * This function checks and updates tx channel if channel is 0 in tx context.
207   * It will update channel to current roc channel if vdev mode is
208   * P2P DEVICE/CLIENT/GO.
209   *
210   * Return: QDF_STATUS_SUCCESS - in case of success
211   */
p2p_check_and_update_channel(struct tx_action_context * tx_ctx)212  static QDF_STATUS p2p_check_and_update_channel(struct tx_action_context *tx_ctx)
213  {
214  	enum QDF_OPMODE mode;
215  	struct wlan_objmgr_vdev *vdev;
216  	struct wlan_objmgr_psoc *psoc;
217  	struct p2p_soc_priv_obj *p2p_soc_obj;
218  	struct p2p_roc_context *curr_roc_ctx;
219  
220  	if (!tx_ctx || tx_ctx->chan_freq) {
221  		p2p_err("NULL tx ctx or channel valid");
222  		return QDF_STATUS_E_INVAL;
223  	}
224  
225  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
226  	psoc = p2p_soc_obj->soc;
227  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
228  			psoc, tx_ctx->vdev_id, WLAN_P2P_ID);
229  	if (!vdev) {
230  		p2p_err("null vdev object");
231  		return QDF_STATUS_E_INVAL;
232  	}
233  
234  	mode = wlan_vdev_mlme_get_opmode(vdev);
235  	curr_roc_ctx = p2p_find_current_roc_ctx(p2p_soc_obj);
236  
237  	if (curr_roc_ctx &&
238  	    (mode == QDF_P2P_DEVICE_MODE ||
239  	     mode == QDF_P2P_CLIENT_MODE ||
240  	     mode == QDF_P2P_GO_MODE))
241  		tx_ctx->chan_freq = curr_roc_ctx->chan_freq;
242  
243  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
244  
245  	return QDF_STATUS_SUCCESS;
246  }
247  
248  /**
249   * p2p_get_p2pie_ptr() - get the pointer to p2p ie
250   * @ie:      source ie
251   * @ie_len:  source ie length
252   *
253   * This function finds out p2p ie by p2p oui and return the pointer.
254   *
255   * Return: pointer to p2p ie
256   */
p2p_get_p2pie_ptr(const uint8_t * ie,uint16_t ie_len)257  const uint8_t *p2p_get_p2pie_ptr(const uint8_t *ie, uint16_t ie_len)
258  {
259  	return wlan_get_vendor_ie_ptr_from_oui(P2P_OUI,
260  			P2P_OUI_SIZE, ie, ie_len);
261  }
262  
263  /**
264   * p2p_get_p2pie_from_probe_rsp() - get the pointer to p2p ie from
265   * probe response
266   * @tx_ctx:      tx context
267   *
268   * This function finds out p2p ie and return the pointer if it is a
269   * probe response frame.
270   *
271   * Return: pointer to p2p ie
272   */
p2p_get_p2pie_from_probe_rsp(struct tx_action_context * tx_ctx)273  static const uint8_t *p2p_get_p2pie_from_probe_rsp(
274  	struct tx_action_context *tx_ctx)
275  {
276  	const uint8_t *ie;
277  	const uint8_t *p2p_ie;
278  	const uint8_t *tmp_p2p_ie = NULL;
279  	uint16_t ie_len;
280  
281  	if (tx_ctx->buf_len <= PROBE_RSP_IE_OFFSET) {
282  		p2p_err("Invalid header len for probe response");
283  		return NULL;
284  	}
285  
286  	ie = tx_ctx->buf + PROBE_RSP_IE_OFFSET;
287  	ie_len = tx_ctx->buf_len - PROBE_RSP_IE_OFFSET;
288  	p2p_ie = p2p_get_p2pie_ptr(ie, ie_len);
289  	while ((p2p_ie) &&
290  		(WLAN_MAX_IE_LEN == p2p_ie[1])) {
291  		ie_len = tx_ctx->buf_len - (p2p_ie - tx_ctx->buf);
292  		if (ie_len > 2) {
293  			ie = p2p_ie + WLAN_MAX_IE_LEN + 2;
294  			tmp_p2p_ie = p2p_get_p2pie_ptr(ie, ie_len);
295  		}
296  
297  		if (tmp_p2p_ie) {
298  			p2p_ie = tmp_p2p_ie;
299  			tmp_p2p_ie = NULL;
300  		} else {
301  			break;
302  		}
303  	}
304  
305  	return p2p_ie;
306  }
307  
308  /**
309   * p2p_get_presence_noa_attr() - get the pointer to noa attr
310   * @pies:      source ie
311   * @length:    source ie length
312   *
313   * This function finds out noa attr by noa eid and return the pointer.
314   *
315   * Return: pointer to noa attr
316   */
p2p_get_presence_noa_attr(const uint8_t * pies,int length)317  static const uint8_t *p2p_get_presence_noa_attr(const uint8_t *pies, int length)
318  {
319  	int left = length;
320  	const uint8_t *ptr = pies;
321  	uint8_t elem_id;
322  	uint16_t elem_len;
323  
324  	p2p_debug("pies:%pK, length:%d", pies, length);
325  
326  	while (left >= 3) {
327  		elem_id = ptr[0];
328  		elem_len = ((uint16_t) ptr[1]) | (ptr[2] << 8);
329  
330  		left -= 3;
331  		if (elem_len > left) {
332  			p2p_err("****Invalid IEs, elem_len=%d left=%d*****",
333  				elem_len, left);
334  			return NULL;
335  		}
336  		if (elem_id == P2P_NOA_ATTR)
337  			return ptr;
338  
339  		left -= elem_len;
340  		ptr += (elem_len + 3);
341  	}
342  
343  	return NULL;
344  }
345  
346  /**
347   * p2p_get_noa_attr_stream_in_mult_p2p_ies() - get the pointer to noa
348   * attr from multi p2p ie
349   * @noa_stream:      noa stream
350   * @noa_len:         noa stream length
351   * @overflow_len:    overflow length
352   *
353   * This function finds out noa attr from multi p2p ies.
354   *
355   * Return: noa length
356   */
p2p_get_noa_attr_stream_in_mult_p2p_ies(uint8_t * noa_stream,uint8_t noa_len,uint8_t overflow_len)357  static uint8_t p2p_get_noa_attr_stream_in_mult_p2p_ies(uint8_t *noa_stream,
358  	uint8_t noa_len, uint8_t overflow_len)
359  {
360  	uint8_t overflow_p2p_stream[P2P_MAX_NOA_ATTR_LEN];
361  
362  	p2p_debug("noa_stream:%pK, noa_len:%d, overflow_len:%d",
363  		noa_stream, noa_len, overflow_len);
364  	if ((noa_len <= (P2P_MAX_NOA_ATTR_LEN + P2P_IE_HEADER_LEN)) &&
365  	    (noa_len >= overflow_len) &&
366  	    (overflow_len <= P2P_MAX_NOA_ATTR_LEN)) {
367  		qdf_mem_copy(overflow_p2p_stream,
368  			     noa_stream + noa_len - overflow_len,
369  			     overflow_len);
370  		noa_stream[noa_len - overflow_len] =
371  			P2P_EID_VENDOR;
372  		noa_stream[noa_len - overflow_len + 1] =
373  			overflow_len + P2P_OUI_SIZE;
374  		qdf_mem_copy(noa_stream + noa_len - overflow_len + 2,
375  			P2P_OUI, P2P_OUI_SIZE);
376  		qdf_mem_copy(noa_stream + noa_len + 2 + P2P_OUI_SIZE -
377  			overflow_len, overflow_p2p_stream,
378  			overflow_len);
379  	}
380  
381  	return noa_len + P2P_IE_HEADER_LEN;
382  }
383  
384  /**
385   * p2p_get_vdev_noa_info() - get vdev noa information
386   * @tx_ctx:         tx context
387   *
388   * This function gets vdev noa information
389   *
390   * Return: pointer to noa information
391   */
p2p_get_vdev_noa_info(struct tx_action_context * tx_ctx)392  static struct p2p_noa_info *p2p_get_vdev_noa_info(
393  	struct tx_action_context *tx_ctx)
394  {
395  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
396  	struct p2p_soc_priv_obj *p2p_soc_obj;
397  	struct wlan_objmgr_vdev *vdev;
398  	struct wlan_objmgr_psoc *psoc;
399  	enum QDF_OPMODE mode;
400  	struct p2p_noa_info *noa_info = NULL;
401  
402  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
403  	psoc = p2p_soc_obj->soc;
404  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(psoc,
405  			tx_ctx->vdev_id, WLAN_P2P_ID);
406  	if (!vdev) {
407  		p2p_err("vdev obj is NULL");
408  		return NULL;
409  	}
410  
411  	mode = wlan_vdev_mlme_get_opmode(vdev);
412  	p2p_debug("vdev mode:%d", mode);
413  	if (mode != QDF_P2P_GO_MODE) {
414  		p2p_debug("invalid p2p vdev mode:%d", mode);
415  		goto fail;
416  	}
417  
418  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(vdev,
419  			WLAN_UMAC_COMP_P2P);
420  
421  	if (!p2p_vdev_obj || !(p2p_vdev_obj->noa_info)) {
422  		p2p_debug("null noa info");
423  		goto fail;
424  	}
425  
426  	noa_info = p2p_vdev_obj->noa_info;
427  
428  fail:
429  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
430  
431  	return noa_info;
432  }
433  
434  /**
435   * p2p_get_noa_attr_stream() - get noa stream from p2p vdev object
436   * @tx_ctx:         tx context
437   * @pnoa_stream:    pointer to noa stream
438   *
439   * This function finds out noa stream from p2p vdev object
440   *
441   * Return: noa stream length
442   */
p2p_get_noa_attr_stream(struct tx_action_context * tx_ctx,uint8_t * pnoa_stream)443  static uint8_t p2p_get_noa_attr_stream(
444  	struct tx_action_context *tx_ctx, uint8_t *pnoa_stream)
445  {
446  	struct p2p_noa_info *noa_info;
447  	struct noa_descriptor *noa_desc_0;
448  	struct noa_descriptor *noa_desc_1;
449  	uint8_t *pbody = pnoa_stream;
450  	uint8_t len = 0;
451  
452  	noa_info = p2p_get_vdev_noa_info(tx_ctx);
453  	if (!noa_info) {
454  		p2p_debug("not valid noa information");
455  		return 0;
456  	}
457  
458  	noa_desc_0 = &(noa_info->noa_desc[0]);
459  	noa_desc_1 = &(noa_info->noa_desc[1]);
460  	if ((!(noa_desc_0->duration)) &&
461  		(!(noa_desc_1->duration)) &&
462  		(!noa_info->opps_ps)) {
463  		p2p_debug("opps ps and duration are 0");
464  		return 0;
465  	}
466  
467  	pbody[0] = P2P_NOA_ATTR;
468  	pbody[3] = noa_info->index;
469  	pbody[4] = noa_info->ct_window | (noa_info->opps_ps << 7);
470  	len = 5;
471  	pbody += len;
472  
473  	if (noa_desc_0->duration) {
474  		*pbody = noa_desc_0->type_count;
475  		pbody += 1;
476  		len += 1;
477  
478  		*((uint32_t *) (pbody)) = noa_desc_0->duration;
479  		pbody += sizeof(uint32_t);
480  		len += 4;
481  
482  		*((uint32_t *) (pbody)) = noa_desc_0->interval;
483  		pbody += sizeof(uint32_t);
484  		len += 4;
485  
486  		*((uint32_t *) (pbody)) = noa_desc_0->start_time;
487  		pbody += sizeof(uint32_t);
488  		len += 4;
489  	}
490  
491  	if (noa_desc_1->duration) {
492  		*pbody = noa_desc_1->type_count;
493  		pbody += 1;
494  		len += 1;
495  
496  		*((uint32_t *) (pbody)) = noa_desc_1->duration;
497  		pbody += sizeof(uint32_t);
498  		len += 4;
499  
500  		*((uint32_t *) (pbody)) = noa_desc_1->interval;
501  		pbody += sizeof(uint32_t);
502  		len += 4;
503  
504  		*((uint32_t *) (pbody)) = noa_desc_1->start_time;
505  		pbody += sizeof(uint32_t);
506  		len += 4;
507  	}
508  
509  	pbody = pnoa_stream + 1;
510  	 /* one byte for Attr and 2 bytes for length */
511  	*((uint16_t *) (pbody)) = len - 3;
512  
513  	return len;
514  }
515  
516  /**
517   * p2p_update_noa_stream() - update noa stream
518   * @tx_ctx:       tx context
519   * @p2p_ie:       pointer to p2p ie
520   * @noa_attr:     pointer to noa attr
521   * @total_len:    pointer to total length of ie
522   * @noa_stream:   noa stream
523   *
524   * This function updates noa stream.
525   *
526   * Return: noa stream length
527   */
p2p_update_noa_stream(struct tx_action_context * tx_ctx,uint8_t * p2p_ie,const uint8_t * noa_attr,uint32_t * total_len,uint8_t * noa_stream)528  static uint16_t p2p_update_noa_stream(struct tx_action_context *tx_ctx,
529  	uint8_t *p2p_ie, const uint8_t *noa_attr, uint32_t *total_len,
530  	uint8_t *noa_stream)
531  {
532  	uint16_t noa_len;
533  	uint16_t overflow_len;
534  	uint8_t orig_len;
535  	uint32_t nbytes_copy;
536  	uint32_t buf_len = *total_len;
537  
538  	noa_len = p2p_get_noa_attr_stream(tx_ctx, noa_stream);
539  	if (noa_len <= 0) {
540  		p2p_debug("do not find out noa attr");
541  		return 0;
542  	}
543  
544  	orig_len = p2p_ie[1];
545  	if (noa_attr) {
546  		noa_len = noa_attr[1] | (noa_attr[2] << 8);
547  		orig_len -= (noa_len + 1 + 2);
548  		buf_len -= (noa_len + 1 + 2);
549  		p2p_ie[1] = orig_len;
550  	}
551  
552  	if ((p2p_ie[1] + noa_len) > WLAN_MAX_IE_LEN) {
553  		overflow_len = p2p_ie[1] + noa_len -
554  				WLAN_MAX_IE_LEN;
555  		noa_len = p2p_get_noa_attr_stream_in_mult_p2p_ies(
556  				noa_stream, noa_len, overflow_len);
557  		p2p_ie[1] = WLAN_MAX_IE_LEN;
558  	} else {
559  		/* increment the length of P2P IE */
560  		p2p_ie[1] += noa_len;
561  	}
562  
563  	*total_len = buf_len;
564  	nbytes_copy = (p2p_ie + orig_len + 2) - tx_ctx->buf;
565  
566  	p2p_debug("noa_len=%d orig_len=%d p2p_ie=%pK buf_len=%d nbytes copy=%d ",
567  		noa_len, orig_len, p2p_ie, buf_len, nbytes_copy);
568  
569  	return noa_len;
570  }
571  
572  /**
573   * p2p_set_ht_caps() - set ht capability
574   * @tx_ctx:         tx context
575   * @num_bytes:      number bytes
576   *
577   * This function sets ht capability
578   *
579   * Return: None
580   */
p2p_set_ht_caps(struct tx_action_context * tx_ctx,uint32_t num_bytes)581  static void p2p_set_ht_caps(struct tx_action_context *tx_ctx,
582  	uint32_t num_bytes)
583  {
584  }
585  
586  /**
587   * p2p_get_next_seq_num() - get next sequence number to fill mac header
588   * @peer:   PEER object
589   * @tx_ctx: tx context
590   * @ta : transmitter address
591   *
592   * API to get next sequence number of action frame for the peer.
593   *
594   * Return: Next sequence number of the peer
595   */
p2p_get_next_seq_num(struct wlan_objmgr_peer * peer,struct tx_action_context * tx_ctx,uint8_t * ta)596  static uint16_t p2p_get_next_seq_num(struct wlan_objmgr_peer *peer,
597  				     struct tx_action_context *tx_ctx,
598  				     uint8_t *ta)
599  {
600  	uint16_t random_num, random_num_bitmask = 0x03FF;
601  	uint16_t seq_num, seq_num_bitmask = 0x0FFF;
602  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
603  	struct wlan_objmgr_vdev *vdev;
604  	bool is_new_random_ta;
605  
606  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(tx_ctx->p2p_soc_obj->soc,
607  						    tx_ctx->vdev_id,
608  						    WLAN_P2P_ID);
609  	if (!vdev) {
610  		p2p_debug("vdev is null");
611  		return false;
612  	}
613  
614  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
615  						vdev, WLAN_UMAC_COMP_P2P);
616  	if (!p2p_vdev_obj) {
617  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
618  		p2p_debug("p2p vdev object is NULL");
619  		return false;
620  	}
621  
622  	is_new_random_ta = qdf_mem_cmp(p2p_vdev_obj->prev_action_frame_addr2,
623  				       ta, QDF_MAC_ADDR_SIZE);
624  	if (is_new_random_ta &&
625  	    tx_ctx->p2p_soc_obj->param.is_random_seq_num_enabled) {
626  		/**
627  		 * Increment the previous sequence number with 10-bits
628  		 * random number to get the next sequence number.
629  		 */
630  
631  		qdf_get_random_bytes(&random_num, sizeof(random_num));
632  		random_num &= random_num_bitmask;
633  		seq_num = (peer->peer_mlme.seq_num + random_num) &
634  			  seq_num_bitmask;
635  		peer->peer_mlme.seq_num = seq_num;
636  
637  		qdf_mem_copy(p2p_vdev_obj->prev_action_frame_addr2, ta,
638  			     QDF_MAC_ADDR_SIZE);
639  
640  	} else {
641  		seq_num = wlan_peer_mlme_get_next_seq_num(peer);
642  	}
643  
644  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
645  
646  	return seq_num;
647  }
648  
649  /**
650   * p2p_populate_mac_header() - update sequence number
651   * @tx_ctx:      tx context
652   *
653   * This function updates sequence number of this mgmt frame
654   *
655   * Return: QDF_STATUS_SUCCESS - in case of success
656   */
p2p_populate_mac_header(struct tx_action_context * tx_ctx)657  static QDF_STATUS p2p_populate_mac_header(
658  	struct tx_action_context *tx_ctx)
659  {
660  	struct wlan_seq_ctl *seq_ctl;
661  	struct wlan_frame_hdr *wh;
662  	struct wlan_objmgr_peer *peer;
663  	struct wlan_objmgr_psoc *psoc;
664  	void *mac_addr;
665  	uint16_t seq_num;
666  	uint8_t pdev_id;
667  	struct wlan_objmgr_vdev *vdev;
668  	enum QDF_OPMODE opmode;
669  
670  	psoc = tx_ctx->p2p_soc_obj->soc;
671  
672  	wh = (struct wlan_frame_hdr *)tx_ctx->buf;
673  	/*
674  	 * Remove the WEP bit if already set, p2p_populate_rmf_field will set it
675  	 * if required.
676  	 */
677  	wh->i_fc[1] &= ~IEEE80211_FC1_WEP;
678  	mac_addr = wh->i_addr1;
679  	pdev_id = wlan_get_pdev_id_from_vdev_id(psoc, tx_ctx->vdev_id,
680  						WLAN_P2P_ID);
681  	peer = wlan_objmgr_get_peer(psoc, pdev_id, mac_addr, WLAN_P2P_ID);
682  	if (!peer) {
683  		mac_addr = wh->i_addr2;
684  		peer = wlan_objmgr_get_peer(psoc, pdev_id, mac_addr,
685  					    WLAN_P2P_ID);
686  	}
687  	if (!peer) {
688  		vdev = wlan_objmgr_get_vdev_by_id_from_psoc(psoc,
689  							    tx_ctx->vdev_id,
690  							    WLAN_P2P_ID);
691  		if (vdev) {
692  			opmode = wlan_vdev_mlme_get_opmode(vdev);
693  			/*
694  			 * For NAN iface, retrieves mac address from vdev
695  			 * as it is self peer. Also, rand_mac_tx would be
696  			 * false as tx channel is not available.
697  			 */
698  			if (opmode == QDF_NAN_DISC_MODE ||
699  			    tx_ctx->rand_mac_tx) {
700  				mac_addr = wlan_vdev_mlme_get_macaddr(vdev);
701  				peer = wlan_objmgr_get_peer(psoc, pdev_id,
702  							    mac_addr,
703  							    WLAN_P2P_ID);
704  			}
705  
706  			wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
707  		}
708  	}
709  	if (!peer) {
710  		p2p_err("no valid peer");
711  		return QDF_STATUS_E_INVAL;
712  	}
713  	seq_num = (uint16_t)p2p_get_next_seq_num(peer, tx_ctx, wh->i_addr2);
714  	seq_ctl = (struct wlan_seq_ctl *)(tx_ctx->buf +
715  			WLAN_SEQ_CTL_OFFSET);
716  	seq_ctl->seq_num_lo = (seq_num & WLAN_LOW_SEQ_NUM_MASK);
717  	seq_ctl->seq_num_hi = ((seq_num & WLAN_HIGH_SEQ_NUM_MASK) >>
718  				WLAN_HIGH_SEQ_NUM_OFFSET);
719  	p2p_debug("seq num: %d", seq_num);
720  
721  	wlan_objmgr_peer_release_ref(peer, WLAN_P2P_ID);
722  
723  	return QDF_STATUS_SUCCESS;
724  }
725  
726  /**
727   * p2p_get_frame_type_str() - parse frame type to string
728   * @frame_info: frame information
729   *
730   * This function parse frame type to string.
731   *
732   * Return: command string
733   */
p2p_get_frame_type_str(struct p2p_frame_info * frame_info)734  static char *p2p_get_frame_type_str(struct p2p_frame_info *frame_info)
735  {
736  	if (frame_info->type == P2P_FRAME_NOT_SUPPORT)
737  		return "Not support frame";
738  
739  	if (frame_info->sub_type == P2P_MGMT_NOT_SUPPORT)
740  		return "Not support sub frame";
741  
742  	if (frame_info->action_type == P2P_ACTION_PRESENCE_REQ)
743  		return "P2P action presence request";
744  	if (frame_info->action_type == P2P_ACTION_PRESENCE_RSP)
745  		return "P2P action presence response";
746  
747  	switch (frame_info->public_action_type) {
748  	case P2P_PUBLIC_ACTION_NEG_REQ:
749  		return "GO negotiation request frame";
750  	case P2P_PUBLIC_ACTION_NEG_RSP:
751  		return "GO negotiation response frame";
752  	case P2P_PUBLIC_ACTION_NEG_CNF:
753  		return "GO negotiation confirm frame";
754  	case P2P_PUBLIC_ACTION_INVIT_REQ:
755  		return "P2P invitation request";
756  	case P2P_PUBLIC_ACTION_INVIT_RSP:
757  		return "P2P invitation response";
758  	case P2P_PUBLIC_ACTION_DEV_DIS_REQ:
759  		return "Device discoverability request";
760  	case P2P_PUBLIC_ACTION_DEV_DIS_RSP:
761  		return "Device discoverability response";
762  	case P2P_PUBLIC_ACTION_PROV_DIS_REQ:
763  		return "Provision discovery request";
764  	case P2P_PUBLIC_ACTION_PROV_DIS_RSP:
765  		return "Provision discovery response";
766  	case P2P_PUBLIC_ACTION_GAS_INIT_REQ:
767  		return "GAS init request";
768  	case P2P_PUBLIC_ACTION_GAS_INIT_RSP:
769  		return "GAS init response";
770  	case P2P_PUBLIC_ACTION_GAS_COMB_REQ:
771  		return "GAS come back request";
772  	case P2P_PUBLIC_ACTION_GAS_COMB_RSP:
773  		return "GAS come back response";
774  	case P2P_PUBLIC_ACTION_WNM_BTM_REQ:
775  		return "BTM request";
776  	case P2P_PUBLIC_ACTION_RRM_BEACON_REQ:
777  		return "BEACON request";
778  	case P2P_PUBLIC_ACTION_RRM_NEIGHBOR_RSP:
779  		return "NEIGHBOR response";
780  	default:
781  		return "Other frame";
782  	}
783  }
784  
785  /**
786   * p2p_init_frame_info() - init frame information structure
787   * @frame_info:     pointer to frame information
788   *
789   * This function init frame information structure.
790   *
791   * Return: None
792   */
p2p_init_frame_info(struct p2p_frame_info * frame_info)793  static void p2p_init_frame_info(struct p2p_frame_info *frame_info)
794  {
795  	frame_info->type = P2P_FRAME_NOT_SUPPORT;
796  	frame_info->sub_type = P2P_MGMT_NOT_SUPPORT;
797  	frame_info->public_action_type =
798  				P2P_PUBLIC_ACTION_NOT_SUPPORT;
799  	frame_info->action_type = P2P_ACTION_NOT_SUPPORT;
800  }
801  
802  /**
803   * p2p_get_frame_info() - get frame information from packet
804   * @data_buf:          data buffer address
805   * @length:            buffer length
806   * @frame_info:        frame information
807   *
808   * This function gets frame information from packet.
809   *
810   * Return: QDF_STATUS_SUCCESS - in case of success
811   */
p2p_get_frame_info(uint8_t * data_buf,uint32_t length,struct p2p_frame_info * frame_info)812  QDF_STATUS p2p_get_frame_info(uint8_t *data_buf, uint32_t length,
813  			      struct p2p_frame_info *frame_info)
814  {
815  	uint8_t type;
816  	uint8_t sub_type;
817  	uint8_t action_type;
818  	uint8_t *buf = data_buf;
819  
820  	p2p_init_frame_info(frame_info);
821  
822  	if (length < P2P_ACTION_OFFSET + 1) {
823  		p2p_err("invalid p2p mgmt hdr len");
824  		return QDF_STATUS_E_INVAL;
825  	}
826  
827  	type = P2P_GET_TYPE_FRM_FC(buf[0]);
828  	sub_type = P2P_GET_SUBTYPE_FRM_FC(buf[0]);
829  	if (type != P2P_FRAME_MGMT) {
830  		p2p_err("just support mgmt frame");
831  		return QDF_STATUS_E_FAILURE;
832  	}
833  
834  	frame_info->type = type;
835  	frame_info->sub_type = sub_type;
836  
837  	if (sub_type != P2P_MGMT_ACTION)
838  		return QDF_STATUS_SUCCESS;
839  
840  	buf += P2P_ACTION_OFFSET;
841  	if (length > P2P_PUBLIC_ACTION_FRAME_TYPE_OFFSET) {
842  		switch (buf[0]) {
843  		case P2P_PUBLIC_ACTION_FRAME:
844  			if (buf[1] == P2P_PUBLIC_ACTION_VENDOR_SPECIFIC &&
845  			    !qdf_mem_cmp(&buf[2], P2P_OUI, P2P_OUI_SIZE)) {
846  				buf = data_buf +
847  					P2P_PUBLIC_ACTION_FRAME_TYPE_OFFSET;
848  				action_type = buf[0];
849  				if (action_type > P2P_PUBLIC_ACTION_PROV_DIS_RSP)
850  					frame_info->public_action_type =
851  						P2P_PUBLIC_ACTION_NOT_SUPPORT;
852  				else
853  					frame_info->public_action_type =
854  						action_type;
855  			}
856  			break;
857  		case WNM_ACTION_FRAME:
858  			if (buf[1] == WNM_BSS_TM_REQUEST) {
859  				action_type = buf[0];
860  				frame_info->public_action_type =
861  					P2P_PUBLIC_ACTION_WNM_BTM_REQ;
862  			}
863  			break;
864  		case RRM_ACTION_FRAME:
865  			if (buf[1] == RRM_RADIO_MEASURE_REQ) {
866  				action_type = buf[0];
867  				frame_info->public_action_type =
868  					P2P_PUBLIC_ACTION_RRM_BEACON_REQ;
869  			} else if (buf[1] == RRM_NEIGHBOR_RPT) {
870  				action_type = buf[0];
871  				frame_info->public_action_type =
872  					P2P_PUBLIC_ACTION_RRM_NEIGHBOR_RSP;
873  			}
874  			break;
875  		default:
876  			break;
877  		}
878  	} else if (length > P2P_ACTION_FRAME_TYPE_OFFSET &&
879  		   buf[0] == P2P_ACTION_VENDOR_SPECIFIC_CATEGORY &&
880  		   !qdf_mem_cmp(&buf[1], P2P_OUI, P2P_OUI_SIZE)) {
881  		buf = data_buf +
882  			P2P_ACTION_FRAME_TYPE_OFFSET;
883  		action_type = buf[0];
884  		if (action_type == P2P_ACTION_PRESENCE_REQ)
885  			frame_info->action_type =
886  				P2P_ACTION_PRESENCE_REQ;
887  		if (action_type == P2P_ACTION_PRESENCE_RSP)
888  			frame_info->action_type =
889  				P2P_ACTION_PRESENCE_RSP;
890  	} else {
891  		p2p_debug("this is not vendor specific p2p action frame");
892  		return QDF_STATUS_SUCCESS;
893  	}
894  
895  	p2p_debug("%s", p2p_get_frame_type_str(frame_info));
896  
897  	return QDF_STATUS_SUCCESS;
898  }
899  
p2p_is_action_frame_of_p2p_type(uint8_t * data_buf,uint32_t length)900  bool p2p_is_action_frame_of_p2p_type(uint8_t *data_buf, uint32_t length)
901  {
902  	struct p2p_frame_info frame_info = {0};
903  	QDF_STATUS status;
904  
905  	status = p2p_get_frame_info(data_buf, length, &frame_info);
906  	if (QDF_IS_STATUS_ERROR(status))
907  		return false;
908  
909  	if (frame_info.public_action_type != P2P_PUBLIC_ACTION_NOT_SUPPORT ||
910  	    frame_info.action_type != P2P_ACTION_NOT_SUPPORT)
911  		return true;
912  
913  	return false;
914  }
915  
916  #ifdef WLAN_FEATURE_P2P_DEBUG
917  /**
918   * p2p_tx_update_connection_status() - Update P2P connection status
919   * with tx frame
920   * @p2p_soc_obj:        P2P soc private object
921   * @tx_frame_info:      frame information
922   * @mac_to:              Pointer to dest MAC address
923   *
924   * This function updates P2P connection status with tx frame.
925   *
926   * Return: QDF_STATUS_SUCCESS - in case of success
927   */
p2p_tx_update_connection_status(struct p2p_soc_priv_obj * p2p_soc_obj,struct p2p_frame_info * tx_frame_info,uint8_t * mac_to)928  static QDF_STATUS p2p_tx_update_connection_status(
929  	struct p2p_soc_priv_obj *p2p_soc_obj,
930  	struct p2p_frame_info *tx_frame_info,
931  	uint8_t *mac_to)
932  {
933  	if (!p2p_soc_obj || !tx_frame_info || !mac_to) {
934  		p2p_err("invalid p2p_soc_obj:%pK or tx_frame_info:%pK or mac_to:%pK",
935  			p2p_soc_obj, tx_frame_info, mac_to);
936  		return QDF_STATUS_E_INVAL;
937  	}
938  
939  	if (tx_frame_info->public_action_type !=
940  		P2P_PUBLIC_ACTION_NOT_SUPPORT)
941  		p2p_debug("%s ---> OTA to " QDF_MAC_ADDR_FMT,
942  			  p2p_get_frame_type_str(tx_frame_info),
943  			  QDF_MAC_ADDR_REF(mac_to));
944  
945  	if ((tx_frame_info->public_action_type ==
946  	     P2P_PUBLIC_ACTION_PROV_DIS_REQ) ||
947  	    (tx_frame_info->public_action_type ==
948  	     P2P_PUBLIC_ACTION_INVIT_REQ) ||
949  	    (tx_frame_info->public_action_type ==
950  	     P2P_PUBLIC_ACTION_NEG_REQ) ||
951  	     (tx_frame_info->public_action_type ==
952  	     P2P_PUBLIC_ACTION_NEG_RSP)) {
953  		p2p_status_update(p2p_soc_obj, P2P_GO_NEG_PROCESS);
954  		p2p_debug("[P2P State]Inactive state to GO negotiation progress state");
955  	} else if ((tx_frame_info->public_action_type ==
956  		    P2P_PUBLIC_ACTION_NEG_CNF) ||
957  		   (tx_frame_info->public_action_type ==
958  		    P2P_PUBLIC_ACTION_INVIT_RSP)) {
959  		p2p_status_update(p2p_soc_obj, P2P_GO_NEG_COMPLETED);
960  		p2p_debug("[P2P State]GO nego progress to GO nego completed state");
961  	}
962  
963  	return QDF_STATUS_SUCCESS;
964  }
965  
966  /**
967   * p2p_rx_update_connection_status() - Update P2P connection status
968   * with rx frame
969   * @p2p_soc_obj:        P2P soc private object
970   * @rx_frame_info:      frame information
971   * @mac_from:            Pointer to source MAC address
972   *
973   * This function updates P2P connection status with rx frame.
974   *
975   * Return: QDF_STATUS_SUCCESS - in case of success
976   */
p2p_rx_update_connection_status(struct p2p_soc_priv_obj * p2p_soc_obj,struct p2p_frame_info * rx_frame_info,uint8_t * mac_from)977  static QDF_STATUS p2p_rx_update_connection_status(
978  	struct p2p_soc_priv_obj *p2p_soc_obj,
979  	struct p2p_frame_info *rx_frame_info,
980  	uint8_t *mac_from)
981  {
982  	if (!p2p_soc_obj || !rx_frame_info || !mac_from) {
983  		p2p_err("invalid p2p_soc_obj:%pK or rx_frame_info:%pK, mac_from:%pK",
984  			p2p_soc_obj, rx_frame_info, mac_from);
985  		return QDF_STATUS_E_INVAL;
986  	}
987  
988  	if (rx_frame_info->public_action_type !=
989  		P2P_PUBLIC_ACTION_NOT_SUPPORT)
990  		p2p_info_rl("%s <--- OTA from " QDF_MAC_ADDR_FMT,
991  			    p2p_get_frame_type_str(rx_frame_info),
992  			    QDF_MAC_ADDR_REF(mac_from));
993  
994  	if ((rx_frame_info->public_action_type ==
995  			P2P_PUBLIC_ACTION_PROV_DIS_REQ) ||
996  	    (rx_frame_info->public_action_type ==
997  			P2P_PUBLIC_ACTION_NEG_REQ) ||
998  	    (rx_frame_info->public_action_type ==
999  			P2P_PUBLIC_ACTION_NEG_RSP)) {
1000  		p2p_status_update(p2p_soc_obj, P2P_GO_NEG_PROCESS);
1001  		p2p_info_rl("[P2P State]Inactive state to GO negotiation progress state");
1002  	} else if (((rx_frame_info->public_action_type ==
1003  		     P2P_PUBLIC_ACTION_NEG_CNF) ||
1004  		   (rx_frame_info->public_action_type ==
1005  		     P2P_PUBLIC_ACTION_INVIT_RSP)) &&
1006  		   (p2p_soc_obj->connection_status ==
1007  		    P2P_GO_NEG_PROCESS)) {
1008  		p2p_status_update(p2p_soc_obj, P2P_GO_NEG_COMPLETED);
1009  		p2p_info_rl("[P2P State]GO negotiation progress to GO negotiation completed state");
1010  	} else if ((rx_frame_info->public_action_type ==
1011  		    P2P_PUBLIC_ACTION_INVIT_REQ) &&
1012  		   (p2p_soc_obj->connection_status == P2P_NOT_ACTIVE)) {
1013  		p2p_status_update(p2p_soc_obj, P2P_GO_NEG_COMPLETED);
1014  		p2p_info_rl("[P2P State]Inactive state to GO negotiation completed state Autonomous GO formation");
1015  	}
1016  
1017  	return QDF_STATUS_SUCCESS;
1018  }
1019  #else
p2p_tx_update_connection_status(struct p2p_soc_priv_obj * p2p_soc_obj,struct p2p_frame_info * tx_frame_info,uint8_t * mac_to)1020  static QDF_STATUS p2p_tx_update_connection_status(
1021  	struct p2p_soc_priv_obj *p2p_soc_obj,
1022  	struct p2p_frame_info *tx_frame_info,
1023  	uint8_t *mac_to)
1024  {
1025  	return QDF_STATUS_SUCCESS;
1026  }
1027  
p2p_rx_update_connection_status(struct p2p_soc_priv_obj * p2p_soc_obj,struct p2p_frame_info * rx_frame_info,uint8_t * mac_from)1028  static QDF_STATUS p2p_rx_update_connection_status(
1029  	struct p2p_soc_priv_obj *p2p_soc_obj,
1030  	struct p2p_frame_info *rx_frame_info,
1031  	uint8_t *mac_from)
1032  {
1033  	return QDF_STATUS_SUCCESS;
1034  }
1035  #endif
1036  
1037  /**
1038   * p2p_packet_alloc() - allocate qdf nbuf
1039   * @size:         buffe size
1040   * @data:         pointer to qdf nbuf data point
1041   * @ppPacket:     pointer to qdf nbuf point
1042   *
1043   * This function allocates qdf nbuf.
1044   *
1045   * Return: QDF_STATUS_SUCCESS - in case of success
1046   */
p2p_packet_alloc(uint16_t size,void ** data,qdf_nbuf_t * ppPacket)1047  static QDF_STATUS p2p_packet_alloc(uint16_t size, void **data,
1048  	qdf_nbuf_t *ppPacket)
1049  {
1050  	QDF_STATUS status = QDF_STATUS_E_FAILURE;
1051  	qdf_nbuf_t nbuf;
1052  
1053  	nbuf = qdf_nbuf_alloc(NULL,
1054  			roundup(size + P2P_TX_PKT_MIN_HEADROOM, 4),
1055  			P2P_TX_PKT_MIN_HEADROOM, sizeof(uint32_t),
1056  			false);
1057  
1058  	if (nbuf) {
1059  		qdf_nbuf_put_tail(nbuf, size);
1060  		qdf_nbuf_set_protocol(nbuf, ETH_P_CONTROL);
1061  		*ppPacket = nbuf;
1062  		*data = qdf_nbuf_data(nbuf);
1063  		qdf_mem_zero(*data, size);
1064  		status = QDF_STATUS_SUCCESS;
1065  	}
1066  
1067  	return status;
1068  }
1069  
1070  /**
1071   * p2p_send_tx_conf() - send tx confirm
1072   * @tx_ctx:        tx context
1073   * @status:        tx status
1074   *
1075   * This function send tx confirm to osif
1076   *
1077   * Return: QDF_STATUS_SUCCESS - pointer to tx context
1078   */
p2p_send_tx_conf(struct tx_action_context * tx_ctx,bool status)1079  static QDF_STATUS p2p_send_tx_conf(struct tx_action_context *tx_ctx,
1080  	bool status)
1081  {
1082  	struct p2p_tx_cnf tx_cnf;
1083  	struct p2p_soc_priv_obj *p2p_soc_obj;
1084  	struct p2p_start_param *start_param;
1085  
1086  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
1087  
1088  	if (!p2p_soc_obj || !(p2p_soc_obj->start_param)) {
1089  		p2p_err("Invalid p2p soc object or start parameters");
1090  		return QDF_STATUS_E_INVAL;
1091  	}
1092  
1093  	start_param = p2p_soc_obj->start_param;
1094  	if (!(start_param->tx_cnf_cb)) {
1095  		p2p_err("no tx confirm callback");
1096  		return QDF_STATUS_E_INVAL;
1097  	}
1098  
1099  	if (tx_ctx->no_ack)
1100  		tx_cnf.action_cookie = 0;
1101  	else
1102  		tx_cnf.action_cookie = (uint64_t)tx_ctx->id;
1103  
1104  	tx_cnf.vdev_id = tx_ctx->vdev_id;
1105  	tx_cnf.buf = tx_ctx->buf;
1106  	tx_cnf.buf_len = tx_ctx->buf_len;
1107  	tx_cnf.status = status ? 0 : 1;
1108  
1109  	p2p_debug("soc:%pK, vdev_id:%d, action_cookie:%llx, len:%d, status:%d, buf:%pK",
1110  		p2p_soc_obj->soc, tx_cnf.vdev_id,
1111  		tx_cnf.action_cookie, tx_cnf.buf_len,
1112  		tx_cnf.status, tx_cnf.buf);
1113  
1114  	p2p_rand_mac_tx_done(p2p_soc_obj->soc, tx_ctx);
1115  
1116  	start_param->tx_cnf_cb(start_param->tx_cnf_cb_data, &tx_cnf);
1117  
1118  	return QDF_STATUS_SUCCESS;
1119  }
1120  
1121  /**
1122   * p2p_get_hw_retry_count() - Get hw tx retry count from config store
1123   * @psoc:          psoc object
1124   * @tx_ctx:        tx context
1125   *
1126   * This function return the hw tx retry count for p2p action frame.
1127   * 0 value means target will use fw default mgmt tx retry count 15.
1128   *
1129   * Return: frame hw tx retry count
1130   */
p2p_get_hw_retry_count(struct wlan_objmgr_psoc * psoc,struct tx_action_context * tx_ctx)1131  static uint8_t p2p_get_hw_retry_count(struct wlan_objmgr_psoc *psoc,
1132  				      struct tx_action_context *tx_ctx)
1133  {
1134  	if (tx_ctx->frame_info.type != P2P_FRAME_MGMT)
1135  		return 0;
1136  
1137  	if (tx_ctx->frame_info.sub_type != P2P_MGMT_ACTION)
1138  		return 0;
1139  
1140  	switch (tx_ctx->frame_info.public_action_type) {
1141  	case P2P_PUBLIC_ACTION_NEG_REQ:
1142  		return wlan_mlme_get_mgmt_hw_tx_retry_count(
1143  					psoc,
1144  					CFG_GO_NEGOTIATION_REQ_FRAME_TYPE);
1145  	case P2P_PUBLIC_ACTION_INVIT_REQ:
1146  		return wlan_mlme_get_mgmt_hw_tx_retry_count(
1147  					psoc,
1148  					CFG_P2P_INVITATION_REQ_FRAME_TYPE);
1149  	case P2P_PUBLIC_ACTION_PROV_DIS_REQ:
1150  		return wlan_mlme_get_mgmt_hw_tx_retry_count(
1151  					psoc,
1152  					CFG_PROVISION_DISCOVERY_REQ_FRAME_TYPE);
1153  	default:
1154  		return 0;
1155  	}
1156  }
1157  
1158  #define GET_HW_RETRY_LIMIT(count) QDF_GET_BITS(count, 0, 4)
1159  #define GET_HW_RETRY_LIMIT_EXT(count) QDF_GET_BITS(count, 4, 3)
1160  
1161  /**
1162   * p2p_mgmt_set_hw_retry_count() - Set mgmt hw tx retry count
1163   * @psoc:          psoc object
1164   * @tx_ctx:        tx context
1165   * @mgmt_param:    mgmt frame tx parameter
1166   *
1167   * This function will set mgmt frame hw tx retry count to tx parameter
1168   *
1169   * Return: void
1170   */
1171  static void
p2p_mgmt_set_hw_retry_count(struct wlan_objmgr_psoc * psoc,struct tx_action_context * tx_ctx,struct wmi_mgmt_params * mgmt_param)1172  p2p_mgmt_set_hw_retry_count(struct wlan_objmgr_psoc *psoc,
1173  			    struct tx_action_context *tx_ctx,
1174  			    struct wmi_mgmt_params *mgmt_param)
1175  {
1176  	uint8_t retry_count = p2p_get_hw_retry_count(psoc, tx_ctx);
1177  
1178  	mgmt_param->tx_param.retry_limit = GET_HW_RETRY_LIMIT(retry_count);
1179  	mgmt_param->tx_param.retry_limit_ext =
1180  					GET_HW_RETRY_LIMIT_EXT(retry_count);
1181  	if (mgmt_param->tx_param.retry_limit ||
1182  	    mgmt_param->tx_param.retry_limit_ext)
1183  		mgmt_param->tx_params_valid = true;
1184  }
1185  
1186  /**
1187   * p2p_mgmt_tx() - call mgmt tx api
1188   * @tx_ctx:        tx context
1189   * @buf_len:       buffer length
1190   * @packet:        pointer to qdf nbuf
1191   * @frame:         pointer to qdf nbuf data
1192   *
1193   * This function call mgmt tx api to tx this action frame.
1194   *
1195   * Return: QDF_STATUS_SUCCESS - in case of success
1196   */
p2p_mgmt_tx(struct tx_action_context * tx_ctx,uint32_t buf_len,qdf_nbuf_t packet,uint8_t * frame)1197  static QDF_STATUS p2p_mgmt_tx(struct tx_action_context *tx_ctx,
1198  	uint32_t buf_len, qdf_nbuf_t packet, uint8_t *frame)
1199  {
1200  	QDF_STATUS status;
1201  	mgmt_tx_download_comp_cb tx_comp_cb;
1202  	mgmt_ota_comp_cb tx_ota_comp_cb;
1203  	struct wlan_frame_hdr *wh;
1204  	struct wlan_objmgr_peer *peer = NULL;
1205  	struct wmi_mgmt_params mgmt_param = { 0 };
1206  	struct wlan_objmgr_psoc *psoc;
1207  	void *mac_addr;
1208  	uint8_t pdev_id;
1209  	struct wlan_objmgr_vdev *vdev;
1210  	enum QDF_OPMODE opmode;
1211  	bool mlo_link_agnostic;
1212  
1213  	psoc = tx_ctx->p2p_soc_obj->soc;
1214  	mgmt_param.tx_frame = packet;
1215  	mgmt_param.frm_len = buf_len;
1216  	mgmt_param.vdev_id = tx_ctx->vdev_id;
1217  	mgmt_param.pdata = frame;
1218  	mgmt_param.chanfreq = tx_ctx->chan_freq;
1219  	mgmt_param.qdf_ctx = wlan_psoc_get_qdf_dev(psoc);
1220  	if (!(mgmt_param.qdf_ctx)) {
1221  		p2p_err("qdf ctx is null");
1222  		return QDF_STATUS_E_INVAL;
1223  	}
1224  	p2p_mgmt_set_hw_retry_count(psoc, tx_ctx, &mgmt_param);
1225  
1226  	wh = (struct wlan_frame_hdr *)frame;
1227  	mac_addr = wh->i_addr1;
1228  	pdev_id = wlan_get_pdev_id_from_vdev_id(psoc, tx_ctx->vdev_id,
1229  						WLAN_P2P_ID);
1230  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(psoc, tx_ctx->vdev_id,
1231  						    WLAN_P2P_ID);
1232  	if (!vdev) {
1233  		p2p_err("VDEV null");
1234  		return QDF_STATUS_E_INVAL;
1235  	}
1236  
1237  	mlo_link_agnostic = wlan_get_mlo_link_agnostic_flag(vdev, mac_addr);
1238  
1239  	peer = wlan_objmgr_get_peer(psoc, pdev_id,  mac_addr, WLAN_P2P_ID);
1240  	if (!peer) {
1241  		mac_addr = wh->i_addr2;
1242  		peer = wlan_objmgr_get_peer(psoc, pdev_id, mac_addr,
1243  					    WLAN_P2P_ID);
1244  	}
1245  	if (!peer) {
1246  		opmode = wlan_vdev_mlme_get_opmode(vdev);
1247  		/*
1248  		 * For NAN iface, retrieves mac address from vdev
1249  		 * as it is self peer. Also, rand_mac_tx would be
1250  		 * false as tx channel is not available.
1251  		 */
1252  		if (opmode == QDF_NAN_DISC_MODE || tx_ctx->rand_mac_tx) {
1253  			mac_addr = wlan_vdev_mlme_get_mldaddr(vdev);
1254  			/* for non-MLO case, mld address will zero */
1255  			if (qdf_is_macaddr_zero(
1256  				(struct qdf_mac_addr *)mac_addr))
1257  				mac_addr = wlan_vdev_mlme_get_macaddr(vdev);
1258  
1259  			peer = wlan_objmgr_get_peer(psoc, pdev_id, mac_addr,
1260  						    WLAN_P2P_ID);
1261  		}
1262  	}
1263  	if (!peer) {
1264  		p2p_err("no valid peer");
1265  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
1266  		return QDF_STATUS_E_INVAL;
1267  	}
1268  
1269  	if (tx_ctx->no_ack) {
1270  		tx_comp_cb = tgt_p2p_mgmt_download_comp_cb;
1271  		tx_ota_comp_cb = NULL;
1272  	} else {
1273  		tx_comp_cb = NULL;
1274  		tx_ota_comp_cb = tgt_p2p_mgmt_ota_comp_cb;
1275  	}
1276  
1277  	p2p_debug("length:%d, chanfreq:%d retry count:%d(%d, %d)",
1278  		  mgmt_param.frm_len, mgmt_param.chanfreq,
1279  		  (mgmt_param.tx_param.retry_limit_ext << 4) |
1280  		  mgmt_param.tx_param.retry_limit,
1281  		  mgmt_param.tx_param.retry_limit,
1282  		  mgmt_param.tx_param.retry_limit_ext);
1283  
1284  	tx_ctx->nbuf = packet;
1285  
1286  	if (mlo_is_mld_sta(vdev) && tx_ctx->frame_info.type == P2P_FRAME_MGMT &&
1287  	    tx_ctx->frame_info.sub_type == P2P_MGMT_ACTION &&
1288  	    mlo_link_agnostic) {
1289  		mgmt_param.mlo_link_agnostic = true;
1290  	}
1291  	status = wlan_mgmt_txrx_mgmt_frame_tx(peer, tx_ctx->p2p_soc_obj,
1292  			packet, tx_comp_cb, tx_ota_comp_cb,
1293  			WLAN_UMAC_COMP_P2P, &mgmt_param);
1294  
1295  	wlan_objmgr_peer_release_ref(peer, WLAN_P2P_ID);
1296  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
1297  
1298  	return status;
1299  }
1300  
1301  /**
1302   * p2p_roc_req_for_tx_action() - new a roc request for tx
1303   * @tx_ctx:        tx context
1304   *
1305   * This function new a roc request for tx and call roc api to process
1306   * this new roc request.
1307   *
1308   * Return: QDF_STATUS_SUCCESS - in case of success
1309   */
p2p_roc_req_for_tx_action(struct tx_action_context * tx_ctx)1310  static QDF_STATUS p2p_roc_req_for_tx_action(
1311  	struct tx_action_context *tx_ctx)
1312  {
1313  	struct p2p_soc_priv_obj *p2p_soc_obj;
1314  	struct p2p_roc_context *roc_ctx;
1315  	QDF_STATUS status;
1316  
1317  	roc_ctx = qdf_mem_malloc(sizeof(struct p2p_roc_context));
1318  	if (!roc_ctx)
1319  		return QDF_STATUS_E_NOMEM;
1320  
1321  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
1322  	roc_ctx->p2p_soc_obj = p2p_soc_obj;
1323  	roc_ctx->vdev_id = tx_ctx->vdev_id;
1324  	roc_ctx->chan_freq = tx_ctx->chan_freq;
1325  	roc_ctx->duration = tx_ctx->duration;
1326  	roc_ctx->roc_state = ROC_STATE_IDLE;
1327  	roc_ctx->roc_type = OFF_CHANNEL_TX;
1328  	roc_ctx->tx_ctx = tx_ctx;
1329  	roc_ctx->id = tx_ctx->id;
1330  	tx_ctx->roc_cookie = (uintptr_t)roc_ctx;
1331  
1332  	p2p_debug("create roc request for off channel tx, tx ctx:%pK, roc ctx:%pK",
1333  		tx_ctx, roc_ctx);
1334  
1335  	status = p2p_process_roc_req(roc_ctx);
1336  	if (status != QDF_STATUS_SUCCESS) {
1337  		p2p_err("request roc for tx action frrame fail");
1338  		return status;
1339  	}
1340  
1341  	status = qdf_list_insert_back(&p2p_soc_obj->tx_q_roc,
1342  		&tx_ctx->node);
1343  	if (status != QDF_STATUS_SUCCESS)
1344  		p2p_err("Failed to insert off chan tx context to wait roc req queue");
1345  
1346  	return status;
1347  }
1348  
1349  /**
1350   * p2p_find_tx_ctx() - find tx context by cookie
1351   * @p2p_soc_obj:        p2p soc object
1352   * @cookie:          cookie to this p2p tx context
1353   * @is_roc_q:        it is in waiting for roc queue
1354   * @is_ack_q:        it is in waiting for ack queue
1355   *
1356   * This function finds out tx context by cookie.
1357   *
1358   * Return: pointer to tx context
1359   */
p2p_find_tx_ctx(struct p2p_soc_priv_obj * p2p_soc_obj,uint64_t cookie,bool * is_roc_q,bool * is_ack_q)1360  static struct tx_action_context *p2p_find_tx_ctx(
1361  	struct p2p_soc_priv_obj *p2p_soc_obj, uint64_t cookie,
1362  	bool *is_roc_q, bool *is_ack_q)
1363  {
1364  	struct tx_action_context *cur_tx_ctx;
1365  	qdf_list_node_t *p_node;
1366  	QDF_STATUS status;
1367  	*is_roc_q = false;
1368  	*is_ack_q = false;
1369  
1370  	p2p_debug("Start to find tx ctx, p2p soc_obj:%pK, cookie:%llx",
1371  		p2p_soc_obj, cookie);
1372  
1373  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_roc, &p_node);
1374  	while (QDF_IS_STATUS_SUCCESS(status)) {
1375  		cur_tx_ctx = qdf_container_of(p_node,
1376  				struct tx_action_context, node);
1377  		if ((uintptr_t) cur_tx_ctx == cookie) {
1378  			*is_roc_q = true;
1379  			p2p_debug("find tx ctx, cookie:%llx", cookie);
1380  			return cur_tx_ctx;
1381  		}
1382  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_roc,
1383  						p_node, &p_node);
1384  	}
1385  
1386  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_ack, &p_node);
1387  	while (QDF_IS_STATUS_SUCCESS(status)) {
1388  		cur_tx_ctx = qdf_container_of(p_node,
1389  				struct tx_action_context, node);
1390  		if ((uintptr_t) cur_tx_ctx == cookie) {
1391  			*is_ack_q = true;
1392  			p2p_debug("find tx ctx, cookie:%llx", cookie);
1393  			return cur_tx_ctx;
1394  		}
1395  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_ack,
1396  						p_node, &p_node);
1397  	}
1398  
1399  	return NULL;
1400  }
1401  
1402  /**
1403   * p2p_find_tx_ctx_by_roc() - find tx context by roc
1404   * @p2p_soc_obj:        p2p soc object
1405   * @cookie:          cookie to roc context
1406   *
1407   * This function finds out tx context by roc context.
1408   *
1409   * Return: pointer to tx context
1410   */
p2p_find_tx_ctx_by_roc(struct p2p_soc_priv_obj * p2p_soc_obj,uint64_t cookie)1411  static struct tx_action_context *p2p_find_tx_ctx_by_roc(
1412  	struct p2p_soc_priv_obj *p2p_soc_obj, uint64_t cookie)
1413  {
1414  	struct tx_action_context *cur_tx_ctx;
1415  	qdf_list_node_t *p_node;
1416  	QDF_STATUS status;
1417  
1418  	p2p_debug("Start to find tx ctx, p2p soc_obj:%pK, cookie:%llx",
1419  		p2p_soc_obj, cookie);
1420  
1421  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_roc, &p_node);
1422  	while (QDF_IS_STATUS_SUCCESS(status)) {
1423  		cur_tx_ctx = qdf_container_of(p_node,
1424  					struct tx_action_context, node);
1425  		if (cur_tx_ctx->roc_cookie == cookie) {
1426  			p2p_debug("find tx ctx, cookie:%llx", cookie);
1427  			return cur_tx_ctx;
1428  		}
1429  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_roc,
1430  						p_node, &p_node);
1431  	}
1432  
1433  	return NULL;
1434  }
1435  
1436  /**
1437   * p2p_move_tx_context_to_ack_queue() - move tx context to tx_q_ack
1438   * @tx_ctx:        tx context
1439   *
1440   * This function moves tx context to waiting for ack queue.
1441   *
1442   * Return: QDF_STATUS_SUCCESS - in case of success
1443   */
p2p_move_tx_context_to_ack_queue(struct tx_action_context * tx_ctx)1444  static QDF_STATUS p2p_move_tx_context_to_ack_queue(
1445  	struct tx_action_context *tx_ctx)
1446  {
1447  	bool is_roc_q = false;
1448  	bool is_ack_q = false;
1449  	struct p2p_soc_priv_obj *p2p_soc_obj = tx_ctx->p2p_soc_obj;
1450  	struct tx_action_context *cur_tx_ctx;
1451  	QDF_STATUS status;
1452  
1453  	cur_tx_ctx = p2p_find_tx_ctx(p2p_soc_obj, (uintptr_t)tx_ctx,
1454  					&is_roc_q, &is_ack_q);
1455  	if (cur_tx_ctx) {
1456  		if (is_roc_q) {
1457  			p2p_debug("find in wait for roc queue");
1458  			status = qdf_list_remove_node(
1459  				&p2p_soc_obj->tx_q_roc,
1460  				(qdf_list_node_t *)tx_ctx);
1461  			if (status != QDF_STATUS_SUCCESS)
1462  				p2p_err("Failed to remove off chan tx context from wait roc req queue");
1463  		}
1464  
1465  		if (is_ack_q) {
1466  			p2p_debug("Already in waiting for ack queue");
1467  			return QDF_STATUS_SUCCESS;
1468  		}
1469  	}
1470  
1471  	status = qdf_list_insert_back(
1472  				&p2p_soc_obj->tx_q_ack,
1473  				&tx_ctx->node);
1474  	if (status != QDF_STATUS_SUCCESS)
1475  		p2p_err("Failed to insert off chan tx context to wait ack req queue");
1476  
1477  	return status;
1478  }
1479  
1480  /**
1481   * p2p_extend_roc_timer() - extend roc timer
1482   * @p2p_soc_obj:   p2p soc private object
1483   * @frame_info:    pointer to frame information
1484   *
1485   * This function extends roc timer for some of p2p public action frame.
1486   *
1487   * Return: QDF_STATUS_SUCCESS - in case of success
1488   */
p2p_extend_roc_timer(struct p2p_soc_priv_obj * p2p_soc_obj,struct p2p_frame_info * frame_info)1489  static QDF_STATUS p2p_extend_roc_timer(
1490  	struct p2p_soc_priv_obj *p2p_soc_obj,
1491  	struct p2p_frame_info *frame_info)
1492  {
1493  	struct p2p_roc_context *curr_roc_ctx;
1494  	uint32_t extend_time;
1495  
1496  	curr_roc_ctx = p2p_find_current_roc_ctx(p2p_soc_obj);
1497  	if (!curr_roc_ctx) {
1498  		p2p_debug("no running roc request currently");
1499  		return QDF_STATUS_SUCCESS;
1500  	}
1501  
1502  	if (!frame_info) {
1503  		p2p_err("invalid frame information");
1504  		return QDF_STATUS_E_INVAL;
1505  	}
1506  
1507  	switch (frame_info->public_action_type) {
1508  	case P2P_PUBLIC_ACTION_NEG_REQ:
1509  	case P2P_PUBLIC_ACTION_NEG_RSP:
1510  		extend_time = 2 * P2P_ACTION_FRAME_DEFAULT_WAIT;
1511  		break;
1512  	case P2P_PUBLIC_ACTION_INVIT_REQ:
1513  	case P2P_PUBLIC_ACTION_DEV_DIS_REQ:
1514  		extend_time = P2P_ACTION_FRAME_DEFAULT_WAIT;
1515  		break;
1516  	default:
1517  		extend_time = 0;
1518  		break;
1519  	}
1520  
1521  	if (extend_time) {
1522  		p2p_debug("extend roc timer, duration:%d", extend_time);
1523  		curr_roc_ctx->duration = extend_time;
1524  		return p2p_restart_roc_timer(curr_roc_ctx);
1525  	}
1526  
1527  	return QDF_STATUS_SUCCESS;
1528  }
1529  
1530  /**
1531   * p2p_adjust_tx_wait() - adjust tx wait
1532   * @tx_ctx:        tx context
1533   *
1534   * This function adjust wait time of this tx context
1535   *
1536   * Return: None
1537   */
p2p_adjust_tx_wait(struct tx_action_context * tx_ctx)1538  static void p2p_adjust_tx_wait(struct tx_action_context *tx_ctx)
1539  {
1540  	struct p2p_frame_info *frame_info;
1541  
1542  	frame_info = &(tx_ctx->frame_info);
1543  	switch (frame_info->public_action_type) {
1544  	case P2P_PUBLIC_ACTION_NEG_RSP:
1545  	case P2P_PUBLIC_ACTION_PROV_DIS_RSP:
1546  		tx_ctx->duration += P2P_ACTION_FRAME_RSP_WAIT;
1547  		break;
1548  	case P2P_PUBLIC_ACTION_NEG_CNF:
1549  	case P2P_PUBLIC_ACTION_INVIT_RSP:
1550  		tx_ctx->duration += P2P_ACTION_FRAME_ACK_WAIT;
1551  		break;
1552  	default:
1553  		break;
1554  	}
1555  }
1556  
1557  /**
1558   * p2p_remove_tx_context() - remove tx ctx from queue
1559   * @tx_ctx:        tx context
1560   *
1561   * This function remove tx context from waiting for roc queue or
1562   * waiting for ack queue.
1563   *
1564   * Return: QDF_STATUS_SUCCESS - in case of success
1565   */
p2p_remove_tx_context(struct tx_action_context * tx_ctx)1566  static QDF_STATUS p2p_remove_tx_context(
1567  	struct tx_action_context *tx_ctx)
1568  {
1569  	bool is_roc_q = false;
1570  	bool is_ack_q = false;
1571  	struct tx_action_context *cur_tx_ctx;
1572  	uint64_t cookie = (uintptr_t)tx_ctx;
1573  	struct p2p_soc_priv_obj *p2p_soc_obj = tx_ctx->p2p_soc_obj;
1574  	QDF_STATUS status = QDF_STATUS_E_FAILURE;
1575  
1576  	p2p_debug("tx context:%pK", tx_ctx);
1577  
1578  	cur_tx_ctx = p2p_find_tx_ctx(p2p_soc_obj, cookie, &is_roc_q,
1579  					&is_ack_q);
1580  
1581  	/* for not off channel tx case, won't find from queue */
1582  	if (!cur_tx_ctx) {
1583  		p2p_debug("Do not find tx context from queue");
1584  		goto end;
1585  	}
1586  
1587  	if (is_roc_q) {
1588  		status = qdf_list_remove_node(
1589  			&p2p_soc_obj->tx_q_roc,
1590  			(qdf_list_node_t *)cur_tx_ctx);
1591  		if (status != QDF_STATUS_SUCCESS)
1592  			p2p_err("Failed to  tx context from wait roc req queue");
1593  	}
1594  
1595  	if (is_ack_q) {
1596  		status = qdf_list_remove_node(
1597  			&p2p_soc_obj->tx_q_ack,
1598  			(qdf_list_node_t *)cur_tx_ctx);
1599  		if (status != QDF_STATUS_SUCCESS)
1600  			p2p_err("Failed to  tx context from wait ack req queue");
1601  	}
1602  
1603  end:
1604  	if (!tx_ctx->roc_cookie)
1605  		qdf_idr_remove(&p2p_soc_obj->p2p_idr, tx_ctx->id);
1606  	qdf_mem_free(tx_ctx->buf);
1607  	qdf_mem_free(tx_ctx);
1608  
1609  	return status;
1610  }
1611  
1612  /**
1613   * p2p_tx_timeout() - Callback for tx timeout
1614   * @pdata: pointer to tx context
1615   *
1616   * This function is callback for tx time out.
1617   *
1618   * Return: None
1619   */
p2p_tx_timeout(void * pdata)1620  static void p2p_tx_timeout(void *pdata)
1621  {
1622  	QDF_STATUS status, ret;
1623  	qdf_list_node_t *p_node;
1624  	struct tx_action_context *tx_ctx;
1625  	struct p2p_soc_priv_obj *p2p_soc_obj;
1626  
1627  	p2p_info("pdata:%pK", pdata);
1628  	p2p_soc_obj = (struct p2p_soc_priv_obj *)pdata;
1629  	if (!p2p_soc_obj) {
1630  		p2p_err("null p2p soc obj");
1631  		return;
1632  	}
1633  
1634  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_ack, &p_node);
1635  	while (QDF_IS_STATUS_SUCCESS(status)) {
1636  		tx_ctx = qdf_container_of(p_node,
1637  					  struct tx_action_context, node);
1638  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_ack,
1639  					    p_node, &p_node);
1640  		if (QDF_TIMER_STATE_STOPPED ==
1641  		    qdf_mc_timer_get_current_state(&tx_ctx->tx_timer)) {
1642  			ret = qdf_list_remove_node(&p2p_soc_obj->tx_q_ack,
1643  						   &tx_ctx->node);
1644  			if (ret == QDF_STATUS_SUCCESS) {
1645  				qdf_mc_timer_destroy(&tx_ctx->tx_timer);
1646  				p2p_send_tx_conf(tx_ctx, false);
1647  				qdf_mem_free(tx_ctx->buf);
1648  				qdf_mem_free(tx_ctx);
1649  			} else
1650  				p2p_err("remove %pK from roc_q fail",
1651  					tx_ctx);
1652  		}
1653  	}
1654  }
1655  
1656  /**
1657   * p2p_enable_tx_timer() - enable tx timer
1658   * @tx_ctx:         tx context
1659   *
1660   * This function enable tx timer for action frame required ota tx.
1661   *
1662   * Return: QDF_STATUS_SUCCESS - in case of success
1663   */
p2p_enable_tx_timer(struct tx_action_context * tx_ctx)1664  static QDF_STATUS p2p_enable_tx_timer(struct tx_action_context *tx_ctx)
1665  {
1666  	QDF_STATUS status;
1667  
1668  	status = qdf_mc_timer_init(&tx_ctx->tx_timer,
1669  				   QDF_TIMER_TYPE_SW, p2p_tx_timeout,
1670  				   tx_ctx->p2p_soc_obj);
1671  	if (status != QDF_STATUS_SUCCESS) {
1672  		p2p_err("failed to init tx timer tx_ctx:%pK", tx_ctx);
1673  		return status;
1674  	}
1675  
1676  	status = qdf_mc_timer_start(&tx_ctx->tx_timer,
1677  				    P2P_ACTION_FRAME_TX_TIMEOUT);
1678  	if (status != QDF_STATUS_SUCCESS)
1679  		p2p_err("tx timer start failed tx_ctx:%pK", tx_ctx);
1680  
1681  	return status;
1682  }
1683  
1684  /**
1685   * p2p_disable_tx_timer() - disable tx timer
1686   * @tx_ctx:         tx context
1687   *
1688   * This function disable tx timer for action frame required ota tx.
1689   *
1690   * Return: QDF_STATUS_SUCCESS - in case of success
1691   */
p2p_disable_tx_timer(struct tx_action_context * tx_ctx)1692  static QDF_STATUS p2p_disable_tx_timer(struct tx_action_context *tx_ctx)
1693  {
1694  	QDF_STATUS status;
1695  
1696  	p2p_debug("tx context:%pK", tx_ctx);
1697  
1698  	status = qdf_mc_timer_stop(&tx_ctx->tx_timer);
1699  	if (status != QDF_STATUS_SUCCESS)
1700  		p2p_err("Failed to stop tx timer, status:%d", status);
1701  
1702  	status = qdf_mc_timer_destroy(&tx_ctx->tx_timer);
1703  	if (status != QDF_STATUS_SUCCESS)
1704  		p2p_err("Failed to destroy tx timer, status:%d", status);
1705  
1706  	return status;
1707  }
1708  
1709  /**
1710   * p2p_populate_rmf_field() - populate unicast rmf frame
1711   * @tx_ctx: tx_action_context
1712   * @size: input size of frame, and output new size
1713   * @ppbuf: input frame ptr, and output new frame
1714   * @ppkt: input pkt, output new pkt.
1715   *
1716   * This function allocates new pkt for rmf frame. The
1717   * new frame has extra space for ccmp field.
1718   *
1719   * Return: QDF_STATUS_SUCCESS - in case of success
1720   */
p2p_populate_rmf_field(struct tx_action_context * tx_ctx,uint32_t * size,uint8_t ** ppbuf,qdf_nbuf_t * ppkt)1721  static QDF_STATUS p2p_populate_rmf_field(struct tx_action_context *tx_ctx,
1722  	uint32_t *size, uint8_t **ppbuf, qdf_nbuf_t *ppkt)
1723  {
1724  	struct wlan_frame_hdr *wh, *rmf_wh;
1725  	struct action_frm_hdr *action_hdr;
1726  	QDF_STATUS status = QDF_STATUS_SUCCESS;
1727  	qdf_nbuf_t pkt = NULL;
1728  	uint8_t *frame;
1729  	uint32_t frame_len;
1730  	struct p2p_soc_priv_obj *p2p_soc_obj;
1731  	uint8_t action_category;
1732  
1733  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
1734  
1735  	if (tx_ctx->frame_info.sub_type != P2P_MGMT_ACTION ||
1736  	    !p2p_soc_obj->p2p_cb.is_mgmt_protected)
1737  		return QDF_STATUS_SUCCESS;
1738  	if (*size < (sizeof(struct wlan_frame_hdr) +
1739  	    sizeof(struct action_frm_hdr))) {
1740  		return QDF_STATUS_E_INVAL;
1741  	}
1742  
1743  	wh = (struct wlan_frame_hdr *)(*ppbuf);
1744  	action_hdr = (struct action_frm_hdr *)(*ppbuf + sizeof(*wh));
1745  
1746  	/*
1747  	 * For Action frame which are not handled, the resp is sent back to the
1748  	 * source without change, except that MSB of the Category set to 1, so
1749  	 * to get the actual action category we need to ignore the MSB.
1750  	 */
1751  	action_category = action_hdr->action_category & 0x7f;
1752  	if (!wlan_mgmt_is_rmf_mgmt_action_frame(action_category)) {
1753  		p2p_debug("non rmf act frame 0x%x category %x",
1754  			  tx_ctx->frame_info.sub_type,
1755  			  action_hdr->action_category);
1756  		return QDF_STATUS_SUCCESS;
1757  	}
1758  
1759  	if (!p2p_soc_obj->p2p_cb.is_mgmt_protected(
1760  		tx_ctx->vdev_id, wh->i_addr1)) {
1761  		p2p_debug("non rmf connection vdev %d "QDF_MAC_ADDR_FMT,
1762  			  tx_ctx->vdev_id, QDF_MAC_ADDR_REF(wh->i_addr1));
1763  		return QDF_STATUS_SUCCESS;
1764  	}
1765  	if (!qdf_is_macaddr_group((struct qdf_mac_addr *)wh->i_addr1) &&
1766  	    !qdf_is_macaddr_broadcast((struct qdf_mac_addr *)wh->i_addr1)) {
1767  		uint8_t mic_len, mic_hdr_len, pdev_id;
1768  
1769  		pdev_id =
1770  			wlan_get_pdev_id_from_vdev_id(tx_ctx->p2p_soc_obj->soc,
1771  						      tx_ctx->vdev_id,
1772  						      WLAN_P2P_ID);
1773  		status = mlme_get_peer_mic_len(p2p_soc_obj->soc, pdev_id,
1774  					       wh->i_addr1, &mic_len,
1775  					       &mic_hdr_len);
1776  		if (QDF_IS_STATUS_ERROR(status)) {
1777  			p2p_err("Failed to get peer mic length.");
1778  			return status;
1779  		}
1780  
1781  		frame_len = *size + mic_hdr_len + mic_len;
1782  		status = p2p_packet_alloc((uint16_t)frame_len, (void **)&frame,
1783  			 &pkt);
1784  		if (status != QDF_STATUS_SUCCESS) {
1785  			p2p_err("Failed to allocate %d bytes for rmf frame.",
1786  				frame_len);
1787  			return QDF_STATUS_E_NOMEM;
1788  		}
1789  
1790  		qdf_mem_copy(frame, wh, sizeof(*wh));
1791  		qdf_mem_copy(frame + sizeof(*wh) + mic_hdr_len,
1792  			     *ppbuf + sizeof(*wh),
1793  			     *size - sizeof(*wh));
1794  		rmf_wh = (struct wlan_frame_hdr *)frame;
1795  		(rmf_wh)->i_fc[1] |= IEEE80211_FC1_WEP;
1796  		p2p_debug("set protection 0x%x cat %d "QDF_MAC_ADDR_FMT,
1797  			  tx_ctx->frame_info.sub_type,
1798  			  action_hdr->action_category,
1799  			  QDF_MAC_ADDR_REF(wh->i_addr1));
1800  
1801  		qdf_nbuf_free(*ppkt);
1802  		*ppbuf = frame;
1803  		*ppkt = pkt;
1804  		*size = frame_len;
1805  		/*
1806  		 * Some target which support sending mgmt frame based on htt
1807  		 * would DMA write this PMF tx frame buffer, it may cause smmu
1808  		 * check permission fault, set a flag to do bi-direction DMA
1809  		 * map, normal tx unmap is enough for this case.
1810  		 */
1811  		QDF_NBUF_CB_TX_DMA_BI_MAP(pkt) = 1;
1812  	}
1813  
1814  	return status;
1815  }
1816  
1817  /**
1818   * p2p_execute_tx_action_frame() - execute tx action frame
1819   * @tx_ctx:        tx context
1820   *
1821   * This function modify p2p ie and tx this action frame.
1822   *
1823   * Return: QDF_STATUS_SUCCESS - in case of success
1824   */
p2p_execute_tx_action_frame(struct tx_action_context * tx_ctx)1825  static QDF_STATUS p2p_execute_tx_action_frame(
1826  	struct tx_action_context *tx_ctx)
1827  {
1828  	uint8_t *frame;
1829  	qdf_nbuf_t packet;
1830  	QDF_STATUS status;
1831  	uint8_t noa_len = 0;
1832  	uint8_t noa_stream[P2P_NOA_STREAM_ARR_SIZE];
1833  	uint8_t orig_len = 0;
1834  	const uint8_t *ie;
1835  	uint8_t ie_len;
1836  	uint8_t *p2p_ie = NULL;
1837  	const uint8_t *presence_noa_attr = NULL;
1838  	uint32_t nbytes_copy;
1839  	uint32_t buf_len = tx_ctx->buf_len;
1840  	struct p2p_frame_info *frame_info;
1841  	struct wlan_objmgr_vdev *vdev;
1842  
1843  	frame_info = &(tx_ctx->frame_info);
1844  	if (frame_info->sub_type == P2P_MGMT_PROBE_RSP) {
1845  		p2p_ie = (uint8_t *)p2p_get_p2pie_from_probe_rsp(tx_ctx);
1846  	} else if (frame_info->action_type ==
1847  			P2P_ACTION_PRESENCE_RSP) {
1848  		ie = tx_ctx->buf +
1849  			P2P_PUBLIC_ACTION_FRAME_TYPE_OFFSET;
1850  		ie_len = tx_ctx->buf_len -
1851  			P2P_PUBLIC_ACTION_FRAME_TYPE_OFFSET;
1852  		p2p_ie = (uint8_t *)p2p_get_p2pie_ptr(ie, ie_len);
1853  		if (p2p_ie) {
1854  			/* extract the presence of NoA attribute inside
1855  			 * P2P IE */
1856  			ie = p2p_ie + P2P_IE_HEADER_LEN;
1857  			ie_len = p2p_ie[1];
1858  			presence_noa_attr = p2p_get_presence_noa_attr(
1859  						ie, ie_len);
1860  		}
1861  	} else if (frame_info->type == P2P_FRAME_MGMT &&
1862  		   frame_info->sub_type == P2P_MGMT_ACTION) {
1863  		vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
1864  				tx_ctx->p2p_soc_obj->soc, tx_ctx->vdev_id,
1865  				WLAN_P2P_ID);
1866  
1867  		if (vdev) {
1868  			wlan_mlo_update_action_frame_from_user(vdev,
1869  							       tx_ctx->buf,
1870  							       tx_ctx->buf_len);
1871  			wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
1872  		}
1873  	}
1874  
1875  	if ((frame_info->sub_type != P2P_MGMT_NOT_SUPPORT) &&
1876  		p2p_ie) {
1877  		orig_len = p2p_ie[1];
1878  		noa_len = p2p_update_noa_stream(tx_ctx, p2p_ie,
1879  				presence_noa_attr, &buf_len,
1880  				noa_stream);
1881  		buf_len += noa_len;
1882  	}
1883  
1884  	if (frame_info->sub_type == P2P_MGMT_PROBE_RSP)
1885  		p2p_set_ht_caps(tx_ctx, buf_len);
1886  
1887  		/* Ok-- try to allocate some memory: */
1888  	status = p2p_packet_alloc((uint16_t) buf_len, (void **)&frame,
1889  			&packet);
1890  	if (status != QDF_STATUS_SUCCESS) {
1891  		p2p_err("Failed to allocate %d bytes for a Probe Request.",
1892  			buf_len);
1893  		return status;
1894  	}
1895  
1896  	/*
1897  	 * Add sequence number to action frames
1898  	 * Frames are handed over in .11 format by supplicant already
1899  	 */
1900  	p2p_populate_mac_header(tx_ctx);
1901  
1902  	if ((noa_len > 0) && p2p_ie
1903  		&& (noa_len < (P2P_MAX_NOA_ATTR_LEN +
1904  				P2P_IE_HEADER_LEN))) {
1905  		/* Add 2 bytes for length and Arribute field */
1906  		nbytes_copy = (p2p_ie + orig_len + 2) - tx_ctx->buf;
1907  		qdf_mem_copy(frame, tx_ctx->buf, nbytes_copy);
1908  		qdf_mem_copy((frame + nbytes_copy), noa_stream,
1909  				noa_len);
1910  		qdf_mem_copy((frame + nbytes_copy + noa_len),
1911  			tx_ctx->buf + nbytes_copy,
1912  			buf_len - nbytes_copy - noa_len);
1913  	} else {
1914  		qdf_mem_copy(frame, tx_ctx->buf, buf_len);
1915  	}
1916  
1917  	status = p2p_populate_rmf_field(tx_ctx, &buf_len, &frame, &packet);
1918  	if (status != QDF_STATUS_SUCCESS) {
1919  		p2p_err("failed to populate rmf frame");
1920  		qdf_nbuf_free(packet);
1921  		return status;
1922  	}
1923  
1924  	status = p2p_mgmt_tx(tx_ctx, buf_len, packet, frame);
1925  	if (status == QDF_STATUS_SUCCESS) {
1926  		if (tx_ctx->no_ack) {
1927  			p2p_send_tx_conf(tx_ctx, true);
1928  			p2p_remove_tx_context(tx_ctx);
1929  		} else {
1930  			p2p_enable_tx_timer(tx_ctx);
1931  			p2p_move_tx_context_to_ack_queue(tx_ctx);
1932  		}
1933  	} else {
1934  		p2p_err("failed to tx mgmt frame");
1935  		qdf_nbuf_free(packet);
1936  	}
1937  
1938  	return status;
1939  }
1940  
p2p_find_tx_ctx_by_nbuf(struct p2p_soc_priv_obj * p2p_soc_obj,void * nbuf)1941  struct tx_action_context *p2p_find_tx_ctx_by_nbuf(
1942  	struct p2p_soc_priv_obj *p2p_soc_obj, void *nbuf)
1943  {
1944  	struct tx_action_context *cur_tx_ctx;
1945  	qdf_list_node_t *p_node;
1946  	QDF_STATUS status;
1947  
1948  	if (!p2p_soc_obj) {
1949  		p2p_err("invalid p2p soc object");
1950  		return NULL;
1951  	}
1952  
1953  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_ack, &p_node);
1954  	while (QDF_IS_STATUS_SUCCESS(status)) {
1955  		cur_tx_ctx =
1956  			qdf_container_of(p_node, struct tx_action_context, node);
1957  		if (cur_tx_ctx->nbuf == nbuf) {
1958  			p2p_debug("find tx ctx, nbuf:%pK", nbuf);
1959  			return cur_tx_ctx;
1960  		}
1961  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_ack,
1962  						p_node, &p_node);
1963  	}
1964  
1965  	return NULL;
1966  }
1967  
p2p_dump_tx_queue(struct p2p_soc_priv_obj * p2p_soc_obj)1968  void p2p_dump_tx_queue(struct p2p_soc_priv_obj *p2p_soc_obj)
1969  {
1970  	struct tx_action_context *tx_ctx;
1971  	qdf_list_node_t *p_node;
1972  	QDF_STATUS status;
1973  
1974  	p2p_debug("dump tx queue wait for roc, p2p soc obj:%pK, size:%d",
1975  		p2p_soc_obj, qdf_list_size(&p2p_soc_obj->tx_q_roc));
1976  
1977  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_roc, &p_node);
1978  	while (QDF_IS_STATUS_SUCCESS(status)) {
1979  		tx_ctx = qdf_container_of(p_node,
1980  				struct tx_action_context, node);
1981  		p2p_debug("p2p soc object:%pK, tx ctx:%pK, vdev_id:%d, "
1982  			  "scan_id:%d, roc_cookie:%llx, freq:%d, buf:%pK, "
1983  			  "len:%d, off_chan:%d, cck:%d, ack:%d, duration:%d",
1984  			  p2p_soc_obj, tx_ctx,
1985  			  tx_ctx->vdev_id, tx_ctx->scan_id,
1986  			  tx_ctx->roc_cookie, tx_ctx->chan_freq,
1987  			  tx_ctx->buf, tx_ctx->buf_len,
1988  			  tx_ctx->off_chan, tx_ctx->no_cck,
1989  			  tx_ctx->no_ack, tx_ctx->duration);
1990  
1991  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_roc,
1992  						p_node, &p_node);
1993  	}
1994  
1995  	p2p_debug("dump tx queue wait for ack, size:%d",
1996  		qdf_list_size(&p2p_soc_obj->tx_q_ack));
1997  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_ack, &p_node);
1998  	while (QDF_IS_STATUS_SUCCESS(status)) {
1999  		tx_ctx = qdf_container_of(p_node,
2000  				struct tx_action_context, node);
2001  		p2p_debug("p2p soc object:%pK, tx_ctx:%pK, vdev_id:%d, "
2002  			  "scan_id:%d, roc_cookie:%llx, freq:%d, buf:%pK, "
2003  			  "len:%d, off_chan:%d, cck:%d, ack:%d, duration:%d",
2004  			  p2p_soc_obj, tx_ctx,
2005  			  tx_ctx->vdev_id, tx_ctx->scan_id,
2006  			  tx_ctx->roc_cookie, tx_ctx->chan_freq,
2007  			  tx_ctx->buf, tx_ctx->buf_len,
2008  			  tx_ctx->off_chan, tx_ctx->no_cck,
2009  			  tx_ctx->no_ack, tx_ctx->duration);
2010  
2011  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_ack,
2012  						p_node, &p_node);
2013  	}
2014  }
2015  
p2p_ready_to_tx_frame(struct p2p_soc_priv_obj * p2p_soc_obj,uint64_t cookie)2016  QDF_STATUS p2p_ready_to_tx_frame(struct p2p_soc_priv_obj *p2p_soc_obj,
2017  	uint64_t cookie)
2018  {
2019  	struct tx_action_context *cur_tx_ctx;
2020  	QDF_STATUS status = QDF_STATUS_SUCCESS;
2021  
2022  	cur_tx_ctx = p2p_find_tx_ctx_by_roc(p2p_soc_obj, cookie);
2023  
2024  	while (cur_tx_ctx) {
2025  		p2p_debug("tx_ctx:%pK", cur_tx_ctx);
2026  		status = p2p_execute_tx_action_frame(cur_tx_ctx);
2027  		if (status != QDF_STATUS_SUCCESS) {
2028  			p2p_send_tx_conf(cur_tx_ctx, false);
2029  			p2p_remove_tx_context(cur_tx_ctx);
2030  		}
2031  		cur_tx_ctx = p2p_find_tx_ctx_by_roc(p2p_soc_obj, cookie);
2032  	}
2033  
2034  	return status;
2035  }
2036  
p2p_cleanup_tx_sync(struct p2p_soc_priv_obj * p2p_soc_obj,struct wlan_objmgr_vdev * vdev)2037  QDF_STATUS p2p_cleanup_tx_sync(
2038  	struct p2p_soc_priv_obj *p2p_soc_obj,
2039  	struct wlan_objmgr_vdev *vdev)
2040  {
2041  	struct scheduler_msg msg = {0};
2042  	struct p2p_cleanup_param *param;
2043  	QDF_STATUS status;
2044  	uint32_t vdev_id;
2045  
2046  	if (!p2p_soc_obj) {
2047  		p2p_err("p2p soc context is NULL");
2048  		return QDF_STATUS_E_FAILURE;
2049  	}
2050  
2051  	p2p_debug("p2p_soc_obj:%pK, vdev:%pK", p2p_soc_obj, vdev);
2052  	param = qdf_mem_malloc(sizeof(*param));
2053  	if (!param)
2054  		return QDF_STATUS_E_NOMEM;
2055  
2056  	param->p2p_soc_obj = p2p_soc_obj;
2057  	if (vdev)
2058  		vdev_id = (uint32_t)wlan_vdev_get_id(vdev);
2059  	else
2060  		vdev_id = P2P_INVALID_VDEV_ID;
2061  	param->vdev_id = vdev_id;
2062  	qdf_event_reset(&p2p_soc_obj->cleanup_tx_done);
2063  	msg.type = P2P_CLEANUP_TX;
2064  	msg.bodyptr = param;
2065  	msg.callback = p2p_process_cmd;
2066  	status = scheduler_post_message(QDF_MODULE_ID_P2P,
2067  					QDF_MODULE_ID_P2P,
2068  					QDF_MODULE_ID_OS_IF, &msg);
2069  	if (status != QDF_STATUS_SUCCESS) {
2070  		qdf_mem_free(param);
2071  		return status;
2072  	}
2073  
2074  	status = qdf_wait_single_event(
2075  			&p2p_soc_obj->cleanup_tx_done,
2076  			P2P_WAIT_CLEANUP_ROC);
2077  
2078  	if (status != QDF_STATUS_SUCCESS)
2079  		p2p_err("wait for cleanup tx timeout, %d", status);
2080  
2081  	return status;
2082  }
2083  
p2p_process_cleanup_tx_queue(struct p2p_cleanup_param * param)2084  QDF_STATUS p2p_process_cleanup_tx_queue(struct p2p_cleanup_param *param)
2085  {
2086  	struct tx_action_context *curr_tx_ctx;
2087  	qdf_list_node_t *p_node;
2088  	struct p2p_soc_priv_obj *p2p_soc_obj;
2089  	uint32_t vdev_id;
2090  	QDF_STATUS status, ret;
2091  
2092  	if (!param || !(param->p2p_soc_obj)) {
2093  		p2p_err("Invalid cleanup param");
2094  		return QDF_STATUS_E_FAILURE;
2095  	}
2096  
2097  	p2p_soc_obj = param->p2p_soc_obj;
2098  	vdev_id = param->vdev_id;
2099  
2100  	p2p_debug("clean up tx queue wait for roc, size:%d, vdev_id:%d",
2101  		  qdf_list_size(&p2p_soc_obj->tx_q_roc), vdev_id);
2102  
2103  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_roc, &p_node);
2104  	while (QDF_IS_STATUS_SUCCESS(status)) {
2105  		curr_tx_ctx = qdf_container_of(p_node,
2106  					struct tx_action_context, node);
2107  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_roc,
2108  					    p_node, &p_node);
2109  		if ((vdev_id == P2P_INVALID_VDEV_ID) ||
2110  		    (vdev_id == curr_tx_ctx->vdev_id)) {
2111  			ret = qdf_list_remove_node(&p2p_soc_obj->tx_q_roc,
2112  						   &curr_tx_ctx->node);
2113  			if (ret == QDF_STATUS_SUCCESS) {
2114  				p2p_send_tx_conf(curr_tx_ctx, false);
2115  				qdf_mem_free(curr_tx_ctx->buf);
2116  				qdf_mem_free(curr_tx_ctx);
2117  			} else
2118  				p2p_err("remove %pK from roc_q fail",
2119  					curr_tx_ctx);
2120  		}
2121  	}
2122  
2123  	p2p_debug("clean up tx queue wait for ack, size:%d",
2124  		  qdf_list_size(&p2p_soc_obj->tx_q_ack));
2125  
2126  	status = qdf_list_peek_front(&p2p_soc_obj->tx_q_ack, &p_node);
2127  	while (QDF_IS_STATUS_SUCCESS(status)) {
2128  		curr_tx_ctx = qdf_container_of(p_node,
2129  					struct tx_action_context, node);
2130  		status = qdf_list_peek_next(&p2p_soc_obj->tx_q_ack,
2131  					    p_node, &p_node);
2132  		if ((vdev_id == P2P_INVALID_VDEV_ID) ||
2133  		    (vdev_id == curr_tx_ctx->vdev_id)) {
2134  			ret = qdf_list_remove_node(&p2p_soc_obj->tx_q_ack,
2135  						   &curr_tx_ctx->node);
2136  			if (ret == QDF_STATUS_SUCCESS) {
2137  				p2p_disable_tx_timer(curr_tx_ctx);
2138  				p2p_send_tx_conf(curr_tx_ctx, false);
2139  				qdf_mem_free(curr_tx_ctx->buf);
2140  				qdf_mem_free(curr_tx_ctx);
2141  			} else
2142  				p2p_err("remove %pK from roc_q fail",
2143  					curr_tx_ctx);
2144  		}
2145  	}
2146  
2147  	qdf_event_set(&p2p_soc_obj->cleanup_tx_done);
2148  
2149  	return QDF_STATUS_SUCCESS;
2150  }
2151  
p2p_check_random_mac(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * random_mac_addr)2152  bool p2p_check_random_mac(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2153  			  uint8_t *random_mac_addr)
2154  {
2155  	uint32_t i = 0;
2156  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2157  	struct wlan_objmgr_vdev *vdev;
2158  
2159  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id, WLAN_P2P_ID);
2160  	if (!vdev) {
2161  		p2p_debug("vdev is null");
2162  		return false;
2163  	}
2164  
2165  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2166  				vdev, WLAN_UMAC_COMP_P2P);
2167  	if (!p2p_vdev_obj) {
2168  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2169  		p2p_debug("p2p vdev object is NULL");
2170  		return false;
2171  	}
2172  
2173  	qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2174  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
2175  		if ((p2p_vdev_obj->random_mac[i].in_use) &&
2176  		    (!qdf_mem_cmp(p2p_vdev_obj->random_mac[i].addr,
2177  				  random_mac_addr, QDF_MAC_ADDR_SIZE))) {
2178  			qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2179  			wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2180  			return true;
2181  		}
2182  	}
2183  	qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2184  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2185  
2186  	return false;
2187  }
2188  
2189  /**
2190   * find_action_frame_cookie() - Checks for action cookie in cookie list
2191   * @cookie_list: List of cookies
2192   * @rnd_cookie: Cookie to be searched
2193   *
2194   * Return: If search is successful return pointer to action_frame_cookie
2195   * object in which cookie item is encapsulated.
2196   */
2197  static struct action_frame_cookie *
find_action_frame_cookie(qdf_list_t * cookie_list,uint64_t rnd_cookie)2198  find_action_frame_cookie(qdf_list_t *cookie_list, uint64_t rnd_cookie)
2199  {
2200  	struct action_frame_cookie *action_cookie;
2201  
2202  	qdf_list_for_each(cookie_list, action_cookie, cookie_node) {
2203  		if (action_cookie->cookie == rnd_cookie)
2204  			return action_cookie;
2205  	}
2206  
2207  	return NULL;
2208  }
2209  
2210  /**
2211   * allocate_action_frame_cookie() - Allocate and add action cookie to
2212   * given list
2213   * @cookie_list: List of cookies
2214   * @rnd_cookie: Cookie to be added
2215   *
2216   * Return: If allocation and addition is successful return pointer to
2217   * action_frame_cookie object in which cookie item is encapsulated.
2218   */
2219  static struct action_frame_cookie *
allocate_action_frame_cookie(qdf_list_t * cookie_list,uint64_t rnd_cookie)2220  allocate_action_frame_cookie(qdf_list_t *cookie_list, uint64_t rnd_cookie)
2221  {
2222  	struct action_frame_cookie *action_cookie;
2223  
2224  	action_cookie = qdf_mem_malloc(sizeof(*action_cookie));
2225  	if (!action_cookie)
2226  		return NULL;
2227  
2228  	action_cookie->cookie = rnd_cookie;
2229  	qdf_list_insert_front(cookie_list, &action_cookie->cookie_node);
2230  
2231  	return action_cookie;
2232  }
2233  
2234  /**
2235   * delete_action_frame_cookie() - Delete the cookie from given list
2236   * @cookie_list: List of cookies
2237   * @action_cookie: Cookie to be deleted
2238   *
2239   * This function deletes the cookie item from given list and corresponding
2240   * object in which it is encapsulated.
2241   *
2242   * Return: None
2243   */
2244  static void
delete_action_frame_cookie(qdf_list_t * cookie_list,struct action_frame_cookie * action_cookie)2245  delete_action_frame_cookie(qdf_list_t *cookie_list,
2246  			   struct action_frame_cookie *action_cookie)
2247  {
2248  	qdf_list_remove_node(cookie_list, &action_cookie->cookie_node);
2249  	qdf_mem_free(action_cookie);
2250  }
2251  
2252  /**
2253   * delete_all_action_frame_cookie() - Delete all the cookies to given list
2254   * @cookie_list: List of cookies
2255   *
2256   * This function deletes all the cookies from from given list.
2257   *
2258   * Return: None
2259   */
2260  static void
delete_all_action_frame_cookie(qdf_list_t * cookie_list)2261  delete_all_action_frame_cookie(qdf_list_t *cookie_list)
2262  {
2263  	qdf_list_node_t *node = NULL;
2264  
2265  	p2p_debug("Delete cookie list %pK, size %d", cookie_list,
2266  		  qdf_list_size(cookie_list));
2267  
2268  	while (!qdf_list_empty(cookie_list)) {
2269  		qdf_list_remove_front(cookie_list, &node);
2270  		qdf_mem_free(node);
2271  	}
2272  }
2273  
2274  /**
2275   * append_action_frame_cookie() - Append action cookie to given list
2276   * @cookie_list: List of cookies
2277   * @rnd_cookie: Cookie to be append
2278   *
2279   * This is a wrapper function which invokes allocate_action_frame_cookie
2280   * if the cookie to be added is not duplicate
2281   *
2282   * Return: true - for successful case
2283   *             false - failed.
2284   */
2285  static bool
append_action_frame_cookie(qdf_list_t * cookie_list,uint64_t rnd_cookie)2286  append_action_frame_cookie(qdf_list_t *cookie_list, uint64_t rnd_cookie)
2287  {
2288  	struct action_frame_cookie *action_cookie;
2289  
2290  	/*
2291  	 * There should be no mac entry with empty cookie list,
2292  	 * check and ignore if duplicate
2293  	 */
2294  	action_cookie = find_action_frame_cookie(cookie_list, rnd_cookie);
2295  	if (action_cookie)
2296  		/* random mac address is already programmed */
2297  		return true;
2298  
2299  	/* insert new cookie in cookie list */
2300  	action_cookie = allocate_action_frame_cookie(cookie_list, rnd_cookie);
2301  	if (!action_cookie)
2302  		return false;
2303  
2304  	return true;
2305  }
2306  
2307  /**
2308   * p2p_add_random_mac() - add or append random mac to given vdev rand mac list
2309   * @soc: soc object
2310   * @vdev_id: vdev id
2311   * @mac: mac addr to be added or append
2312   * @freq: frequency
2313   * @rnd_cookie: random mac mgmt tx cookie
2314   *
2315   * This function will add or append the mac addr entry to vdev random mac list.
2316   * Once the mac addr filter is not needed, it can be removed by
2317   * p2p_del_random_mac.
2318   *
2319   * Return: QDF_STATUS_E_EXISTS - append to existing list
2320   *             QDF_STATUS_SUCCESS - add a new entry.
2321   *             other : failed to add the mac address entry.
2322   */
2323  static QDF_STATUS
p2p_add_random_mac(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * mac,uint32_t freq,uint64_t rnd_cookie)2324  p2p_add_random_mac(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2325  		   uint8_t *mac, uint32_t freq, uint64_t rnd_cookie)
2326  {
2327  	uint32_t i;
2328  	uint32_t first_unused = MAX_RANDOM_MAC_ADDRS;
2329  	struct action_frame_cookie *action_cookie;
2330  	int32_t append_ret;
2331  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2332  	struct wlan_objmgr_vdev *vdev;
2333  
2334  	p2p_debug("random_mac:vdev %d mac_addr:"QDF_MAC_ADDR_FMT" rnd_cookie=%llu freq = %u",
2335  		  vdev_id, QDF_MAC_ADDR_REF(mac), rnd_cookie, freq);
2336  
2337  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id, WLAN_P2P_ID);
2338  	if (!vdev) {
2339  		p2p_debug("vdev is null");
2340  
2341  		return QDF_STATUS_E_INVAL;
2342  	}
2343  
2344  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2345  				vdev, WLAN_UMAC_COMP_P2P);
2346  	if (!p2p_vdev_obj) {
2347  		p2p_debug("random_mac:p2p vdev object is NULL");
2348  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2349  
2350  		return QDF_STATUS_E_INVAL;
2351  	}
2352  
2353  	qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2354  	/*
2355  	 * Following loop checks whether random mac entry is already
2356  	 * present, if present get the index of matched entry else
2357  	 * get the first unused slot to store this new random mac
2358  	 */
2359  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
2360  		if (!p2p_vdev_obj->random_mac[i].in_use) {
2361  			if (first_unused == MAX_RANDOM_MAC_ADDRS)
2362  				first_unused = i;
2363  			continue;
2364  		}
2365  
2366  		if (!qdf_mem_cmp(p2p_vdev_obj->random_mac[i].addr, mac,
2367  				 QDF_MAC_ADDR_SIZE))
2368  			break;
2369  	}
2370  
2371  	if (i != MAX_RANDOM_MAC_ADDRS) {
2372  		append_ret = append_action_frame_cookie(
2373  				&p2p_vdev_obj->random_mac[i].cookie_list,
2374  				rnd_cookie);
2375  		qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2376  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2377  		p2p_debug("random_mac:append %d vdev %d freq %d "QDF_MAC_ADDR_FMT" rnd_cookie %llu",
2378  			  append_ret, vdev_id, freq, QDF_MAC_ADDR_REF(mac), rnd_cookie);
2379  		if (!append_ret) {
2380  			p2p_debug("random_mac:failed to append rnd_cookie");
2381  			return QDF_STATUS_E_NOMEM;
2382  		}
2383  
2384  		return QDF_STATUS_E_EXISTS;
2385  	}
2386  
2387  	if (first_unused == MAX_RANDOM_MAC_ADDRS) {
2388  		qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2389  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2390  		p2p_debug("random_mac:Reached the limit of Max random addresses");
2391  
2392  		return QDF_STATUS_E_RESOURCES;
2393  	}
2394  
2395  	/* get the first unused buf and store new random mac */
2396  	i = first_unused;
2397  
2398  	action_cookie = allocate_action_frame_cookie(
2399  				&p2p_vdev_obj->random_mac[i].cookie_list,
2400  				rnd_cookie);
2401  	if (!action_cookie) {
2402  		qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2403  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2404  		p2p_err("random_mac:failed to alloc rnd cookie");
2405  
2406  		return QDF_STATUS_E_NOMEM;
2407  	}
2408  	qdf_mem_copy(p2p_vdev_obj->random_mac[i].addr, mac, QDF_MAC_ADDR_SIZE);
2409  	p2p_vdev_obj->random_mac[i].in_use = true;
2410  	p2p_vdev_obj->random_mac[i].freq = freq;
2411  	qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2412  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2413  	p2p_debug("random_mac:add vdev %d freq %d "QDF_MAC_ADDR_FMT" rnd_cookie %llu",
2414  		  vdev_id, freq, QDF_MAC_ADDR_REF(mac), rnd_cookie);
2415  
2416  	return QDF_STATUS_SUCCESS;
2417  }
2418  
2419  QDF_STATUS
p2p_del_random_mac(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint64_t rnd_cookie)2420  p2p_del_random_mac(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2421  		   uint64_t rnd_cookie)
2422  {
2423  	uint32_t i;
2424  	struct action_frame_cookie *action_cookie;
2425  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2426  	struct wlan_objmgr_vdev *vdev;
2427  
2428  	p2p_debug("random_mac:vdev %d cookie %llu", vdev_id, rnd_cookie);
2429  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id,
2430  						    WLAN_P2P_ID);
2431  	if (!vdev) {
2432  		p2p_debug("vdev is null");
2433  		return QDF_STATUS_E_INVAL;
2434  	}
2435  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2436  				vdev, WLAN_UMAC_COMP_P2P);
2437  	if (!p2p_vdev_obj) {
2438  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2439  		p2p_debug("p2p vdev object is NULL");
2440  		return QDF_STATUS_E_INVAL;
2441  	}
2442  
2443  	qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2444  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
2445  		struct action_frame_random_mac *random_mac;
2446  		qdf_freq_t freq;
2447  		uint8_t addr[QDF_MAC_ADDR_SIZE];
2448  
2449  		random_mac = &p2p_vdev_obj->random_mac[i];
2450  		if (!random_mac->in_use)
2451  			continue;
2452  
2453  		action_cookie = find_action_frame_cookie(
2454  				&random_mac->cookie_list, rnd_cookie);
2455  		if (!action_cookie)
2456  			continue;
2457  
2458  		delete_action_frame_cookie(
2459  			&random_mac->cookie_list,
2460  			action_cookie);
2461  
2462  		if (qdf_list_empty(&random_mac->cookie_list)) {
2463  			random_mac->in_use = false;
2464  			freq = random_mac->freq;
2465  			qdf_mem_copy(addr, random_mac->addr, QDF_MAC_ADDR_SIZE);
2466  			qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2467  			if (qdf_mc_timer_get_current_state(
2468  					&random_mac->clear_timer) ==
2469  			    QDF_TIMER_STATE_RUNNING) {
2470  				p2p_debug("random_mac:stop timer on vdev %d addr " QDF_MAC_ADDR_FMT,
2471  					  vdev_id, QDF_MAC_ADDR_REF(addr));
2472  				qdf_mc_timer_stop(&random_mac->clear_timer);
2473  			}
2474  
2475  			p2p_clear_mac_filter(
2476  					wlan_vdev_get_psoc(p2p_vdev_obj->vdev),
2477  					vdev_id, addr, freq);
2478  
2479  			qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2480  			p2p_debug("random_mac:noref on vdev %d addr "QDF_MAC_ADDR_FMT,
2481  				  vdev_id, QDF_MAC_ADDR_REF(addr));
2482  		}
2483  		break;
2484  	}
2485  	qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2486  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2487  
2488  	return QDF_STATUS_SUCCESS;
2489  }
2490  
2491  QDF_STATUS
p2p_random_mac_handle_tx_done(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint64_t rnd_cookie,uint32_t duration)2492  p2p_random_mac_handle_tx_done(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2493  			      uint64_t rnd_cookie, uint32_t duration)
2494  {
2495  	uint32_t i;
2496  	struct action_frame_cookie *action_cookie;
2497  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2498  	struct wlan_objmgr_vdev *vdev;
2499  
2500  	p2p_debug("random_mac:vdev %d cookie %llu duration %d", vdev_id,
2501  		  rnd_cookie, duration);
2502  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id, WLAN_P2P_ID);
2503  	if (!vdev) {
2504  		p2p_debug("vdev is null");
2505  		return QDF_STATUS_E_INVAL;
2506  	}
2507  
2508  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2509  						vdev, WLAN_UMAC_COMP_P2P);
2510  	if (!p2p_vdev_obj) {
2511  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2512  		p2p_debug("p2p vdev object is NULL");
2513  		return QDF_STATUS_E_INVAL;
2514  	}
2515  
2516  	qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2517  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
2518  		struct action_frame_random_mac *random_mac;
2519  		qdf_freq_t freq;
2520  		uint8_t addr[QDF_MAC_ADDR_SIZE];
2521  
2522  		random_mac = &p2p_vdev_obj->random_mac[i];
2523  		if (!random_mac->in_use)
2524  			continue;
2525  		action_cookie = find_action_frame_cookie(
2526  					&random_mac->cookie_list, rnd_cookie);
2527  		if (!action_cookie)
2528  			continue;
2529  
2530  		/* If duration is zero then remove the cookie and also remove
2531  		 * the filter from firmware.
2532  		 */
2533  		if (!duration) {
2534  			delete_action_frame_cookie(&random_mac->cookie_list,
2535  						   action_cookie);
2536  			p2p_debug("random mac:clear mac addr on vdev %d addr " QDF_MAC_ADDR_FMT,
2537  				  vdev_id, QDF_MAC_ADDR_REF(random_mac->addr));
2538  
2539  			if (qdf_list_empty(&random_mac->cookie_list)) {
2540  				random_mac->in_use = false;
2541  				freq = random_mac->freq;
2542  				qdf_mem_copy(addr, random_mac->addr,
2543  					     QDF_MAC_ADDR_SIZE);
2544  				qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2545  				p2p_clear_mac_filter(
2546  					wlan_vdev_get_psoc(p2p_vdev_obj->vdev),
2547  					vdev_id, addr, freq);
2548  				qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2549  			}
2550  		} else {
2551  			/* As duration is non zero start the timer for this
2552  			 * duration. while the timer is running if tx cancel
2553  			 * comes from supplicant then cookie will be removed
2554  			 * and random mac filter will be removed from firmware.
2555  			 * same thing will happen if timer expires without tx
2556  			 * cancel from supplicant
2557  			 */
2558  			qdf_mem_copy(addr, random_mac->addr, QDF_MAC_ADDR_SIZE);
2559  
2560  			qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2561  			if (qdf_mc_timer_get_current_state(
2562  				&random_mac->clear_timer) ==
2563  				QDF_TIMER_STATE_RUNNING)
2564  				qdf_mc_timer_stop(&random_mac->clear_timer);
2565  			qdf_mc_timer_start(&random_mac->clear_timer, duration);
2566  			p2p_debug("random_mac:start timer on vdev %d addr " QDF_MAC_ADDR_FMT,
2567  				  vdev_id, QDF_MAC_ADDR_REF(addr));
2568  
2569  			qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2570  		}
2571  	}
2572  	qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2573  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2574  
2575  	return QDF_STATUS_SUCCESS;
2576  }
2577  
p2p_del_all_rand_mac_vdev(struct wlan_objmgr_vdev * vdev)2578  void p2p_del_all_rand_mac_vdev(struct wlan_objmgr_vdev *vdev)
2579  {
2580  	int32_t i;
2581  	uint32_t freq;
2582  	uint8_t addr[QDF_MAC_ADDR_SIZE];
2583  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2584  
2585  	if (!vdev)
2586  		return;
2587  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2588  				vdev, WLAN_UMAC_COMP_P2P);
2589  	if (!p2p_vdev_obj)
2590  		return;
2591  
2592  	qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2593  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
2594  		struct action_frame_cookie *action_cookie;
2595  		struct action_frame_cookie *action_cookie_next;
2596  
2597  		if (!p2p_vdev_obj->random_mac[i].in_use)
2598  			continue;
2599  
2600  		/* empty the list and clear random addr */
2601  		qdf_list_for_each_del(&p2p_vdev_obj->random_mac[i].cookie_list,
2602  				      action_cookie, action_cookie_next,
2603  				      cookie_node) {
2604  			qdf_list_remove_node(
2605  				&p2p_vdev_obj->random_mac[i].cookie_list,
2606  				&action_cookie->cookie_node);
2607  			qdf_mem_free(action_cookie);
2608  		}
2609  
2610  		p2p_vdev_obj->random_mac[i].in_use = false;
2611  		freq = p2p_vdev_obj->random_mac[i].freq;
2612  		qdf_mem_copy(addr, p2p_vdev_obj->random_mac[i].addr,
2613  			     QDF_MAC_ADDR_SIZE);
2614  		qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2615  		qdf_mc_timer_stop(&p2p_vdev_obj->random_mac[i].clear_timer);
2616  		p2p_clear_mac_filter(wlan_vdev_get_psoc(vdev),
2617  				     wlan_vdev_get_id(vdev), addr, freq);
2618  		p2p_debug("random_mac:delall vdev %d freq %d addr "QDF_MAC_ADDR_FMT,
2619  			  wlan_vdev_get_id(vdev), freq, QDF_MAC_ADDR_REF(addr));
2620  
2621  		qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
2622  	}
2623  	qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
2624  }
2625  
2626  static void
p2p_del_rand_mac_vdev_enum_handler(struct wlan_objmgr_psoc * psoc,void * obj,void * arg)2627  p2p_del_rand_mac_vdev_enum_handler(struct wlan_objmgr_psoc *psoc,
2628  				   void *obj, void *arg)
2629  {
2630  	struct wlan_objmgr_vdev *vdev = obj;
2631  
2632  	if (!vdev) {
2633  		p2p_err("random_mac:invalid vdev");
2634  		return;
2635  	}
2636  
2637  	if (!p2p_is_vdev_support_rand_mac(vdev))
2638  		return;
2639  
2640  	p2p_del_all_rand_mac_vdev(vdev);
2641  }
2642  
p2p_del_all_rand_mac_soc(struct wlan_objmgr_psoc * soc)2643  void p2p_del_all_rand_mac_soc(struct wlan_objmgr_psoc *soc)
2644  {
2645  	if (!soc) {
2646  		p2p_err("random_mac:soc object is NULL");
2647  		return;
2648  	}
2649  
2650  	wlan_objmgr_iterate_obj_list(soc, WLAN_VDEV_OP,
2651  				     p2p_del_rand_mac_vdev_enum_handler,
2652  				     NULL, 0, WLAN_P2P_ID);
2653  }
2654  
2655  /**
2656   * p2p_is_random_mac() - check mac addr is random mac for vdev
2657   * @soc: soc object
2658   * @vdev_id: vdev id
2659   * @mac: mac addr to be added or append
2660   *
2661   * This function will check the source mac addr same as vdev's mac addr or not.
2662   * If not same, then the source mac addr should be random mac addr.
2663   *
2664   * Return: true if mac is random mac, otherwise false
2665   */
2666  static bool
p2p_is_random_mac(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * mac)2667  p2p_is_random_mac(struct wlan_objmgr_psoc *soc, uint32_t vdev_id, uint8_t *mac)
2668  {
2669  	bool ret = false;
2670  	struct wlan_objmgr_vdev *vdev;
2671  
2672  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id, WLAN_P2P_ID);
2673  	if (!vdev) {
2674  		p2p_debug("random_mac:vdev is null");
2675  		return false;
2676  	}
2677  
2678  	if (qdf_mem_cmp(wlan_vdev_mlme_get_macaddr(vdev),
2679  			mac, QDF_MAC_ADDR_SIZE))
2680  		ret = true;
2681  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2682  
2683  	return ret;
2684  }
2685  
p2p_set_mac_filter_callback(bool result,void * context)2686  static void p2p_set_mac_filter_callback(bool result, void *context)
2687  {
2688  	struct osif_request *request;
2689  	struct random_mac_priv *priv;
2690  
2691  	p2p_debug("random_mac:set random mac filter result %d", result);
2692  	request = osif_request_get(context);
2693  	if (!request) {
2694  		p2p_err("random_mac:invalid response");
2695  		return;
2696  	}
2697  
2698  	priv = osif_request_priv(request);
2699  	priv->result = result;
2700  
2701  	osif_request_complete(request);
2702  	osif_request_put(request);
2703  }
2704  
p2p_process_set_rand_mac_rsp(struct p2p_mac_filter_rsp * resp)2705  QDF_STATUS p2p_process_set_rand_mac_rsp(struct p2p_mac_filter_rsp *resp)
2706  {
2707  	struct wlan_objmgr_psoc *soc;
2708  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2709  	struct wlan_objmgr_vdev *vdev;
2710  
2711  	if (!resp || !resp->p2p_soc_obj || !resp->p2p_soc_obj->soc) {
2712  		p2p_debug("random_mac:set_filter_req is null");
2713  		return QDF_STATUS_E_INVAL;
2714  	}
2715  	p2p_debug("random_mac:process rsp on vdev %d status %d", resp->vdev_id,
2716  		  resp->status);
2717  	soc = resp->p2p_soc_obj->soc;
2718  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, resp->vdev_id,
2719  						    WLAN_P2P_ID);
2720  	if (!vdev) {
2721  		p2p_debug("random_mac:vdev is null vdev %d", resp->vdev_id);
2722  		return QDF_STATUS_E_INVAL;
2723  	}
2724  
2725  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2726  				vdev, WLAN_UMAC_COMP_P2P);
2727  	if (!p2p_vdev_obj) {
2728  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2729  		p2p_debug("random_mac:p2p_vdev_obj is null vdev %d",
2730  			  resp->vdev_id);
2731  		return QDF_STATUS_E_INVAL;
2732  	}
2733  	if (!p2p_vdev_obj->pending_req.soc) {
2734  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2735  		p2p_debug("random_mac:no pending set req for vdev %d",
2736  			  resp->vdev_id);
2737  		return QDF_STATUS_E_INVAL;
2738  	}
2739  
2740  	p2p_debug("random_mac:get pending req on vdev %d set %d mac filter "QDF_MAC_ADDR_FMT" freq %d",
2741  		  p2p_vdev_obj->pending_req.vdev_id,
2742  		  p2p_vdev_obj->pending_req.set,
2743  		  QDF_MAC_ADDR_REF(p2p_vdev_obj->pending_req.mac),
2744  		  p2p_vdev_obj->pending_req.freq);
2745  	if (p2p_vdev_obj->pending_req.cb)
2746  		p2p_vdev_obj->pending_req.cb(
2747  			!!resp->status, p2p_vdev_obj->pending_req.req_cookie);
2748  
2749  	qdf_mem_zero(&p2p_vdev_obj->pending_req,
2750  		     sizeof(p2p_vdev_obj->pending_req));
2751  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2752  
2753  	return QDF_STATUS_SUCCESS;
2754  }
2755  
2756  QDF_STATUS
p2p_process_set_rand_mac(struct p2p_set_mac_filter_req * set_filter_req)2757  p2p_process_set_rand_mac(struct p2p_set_mac_filter_req *set_filter_req)
2758  {
2759  	struct wlan_objmgr_psoc *soc;
2760  	struct wlan_lmac_if_p2p_tx_ops *p2p_ops;
2761  	QDF_STATUS status = QDF_STATUS_E_FAILURE;
2762  	struct set_rx_mac_filter param;
2763  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2764  	struct wlan_objmgr_vdev *vdev;
2765  
2766  	if (!set_filter_req || !set_filter_req->soc) {
2767  		p2p_debug("random_mac:set_filter_req is null");
2768  		return QDF_STATUS_E_INVAL;
2769  	}
2770  	p2p_debug("random_mac:vdev %d set %d mac filter "QDF_MAC_ADDR_FMT" freq %d",
2771  		  set_filter_req->vdev_id, set_filter_req->set,
2772  		  QDF_MAC_ADDR_REF(set_filter_req->mac), set_filter_req->freq);
2773  
2774  	soc = set_filter_req->soc;
2775  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
2776  				soc, set_filter_req->vdev_id, WLAN_P2P_ID);
2777  	if (!vdev) {
2778  		p2p_debug("random_mac:vdev is null vdev %d",
2779  			  set_filter_req->vdev_id);
2780  		goto get_vdev_failed;
2781  	}
2782  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
2783  				vdev, WLAN_UMAC_COMP_P2P);
2784  	if (!p2p_vdev_obj) {
2785  		p2p_debug("random_mac:p2p_vdev_obj is null vdev %d",
2786  			  set_filter_req->vdev_id);
2787  		goto get_p2p_obj_failed;
2788  	}
2789  	if (p2p_vdev_obj->pending_req.soc) {
2790  		p2p_debug("random_mac:Busy on vdev %d set %d mac filter "QDF_MAC_ADDR_FMT" freq %d",
2791  			  p2p_vdev_obj->pending_req.vdev_id,
2792  			  p2p_vdev_obj->pending_req.set,
2793  			  QDF_MAC_ADDR_REF(p2p_vdev_obj->pending_req.mac),
2794  			  p2p_vdev_obj->pending_req.freq);
2795  		goto get_p2p_obj_failed;
2796  	}
2797  
2798  	p2p_ops = p2p_psoc_get_tx_ops(soc);
2799  	if (p2p_ops && p2p_ops->set_mac_addr_rx_filter_cmd) {
2800  		qdf_mem_zero(&param,  sizeof(param));
2801  		param.vdev_id = set_filter_req->vdev_id;
2802  		qdf_mem_copy(param.mac, set_filter_req->mac,
2803  			     QDF_MAC_ADDR_SIZE);
2804  		param.freq = set_filter_req->freq;
2805  		param.set = set_filter_req->set;
2806  		status = p2p_ops->set_mac_addr_rx_filter_cmd(soc, &param);
2807  		if (status == QDF_STATUS_SUCCESS && set_filter_req->set)
2808  			qdf_mem_copy(&p2p_vdev_obj->pending_req,
2809  				     set_filter_req, sizeof(*set_filter_req));
2810  		p2p_debug("random_mac:p2p set mac addr rx filter, status:%d",
2811  			  status);
2812  	}
2813  
2814  get_p2p_obj_failed:
2815  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2816  
2817  get_vdev_failed:
2818  	if (status != QDF_STATUS_SUCCESS &&
2819  	    set_filter_req->cb)
2820  		set_filter_req->cb(false, set_filter_req->req_cookie);
2821  
2822  	return status;
2823  }
2824  
2825  /**
2826   * p2p_set_mac_filter() - send set mac addr filter cmd
2827   * @soc: soc
2828   * @vdev_id: vdev id
2829   * @mac: mac addr
2830   * @freq: freq
2831   * @set: set or clear
2832   * @cb: callback func to be called when the request completed.
2833   * @req_cookie: cookie to be returned
2834   *
2835   * This function send set random mac addr filter command to p2p component
2836   * msg core
2837   *
2838   * Return: QDF_STATUS_SUCCESS - if sent successfully.
2839   *         otherwise : failed.
2840   */
2841  static QDF_STATUS
p2p_set_mac_filter(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * mac,uint32_t freq,bool set,p2p_request_mgr_callback_t cb,void * req_cookie)2842  p2p_set_mac_filter(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2843  		   uint8_t *mac, uint32_t freq, bool set,
2844  		   p2p_request_mgr_callback_t cb, void *req_cookie)
2845  {
2846  	struct p2p_set_mac_filter_req *set_filter_req;
2847  	struct scheduler_msg msg = {0};
2848  	QDF_STATUS status;
2849  
2850  	p2p_debug("random_mac:vdev %d freq %d set %d "QDF_MAC_ADDR_FMT,
2851  		  vdev_id, freq, set, QDF_MAC_ADDR_REF(mac));
2852  
2853  	set_filter_req = qdf_mem_malloc(sizeof(*set_filter_req));
2854  	if (!set_filter_req)
2855  		return QDF_STATUS_E_NOMEM;
2856  
2857  	set_filter_req->soc = soc;
2858  	set_filter_req->vdev_id = vdev_id;
2859  	set_filter_req->freq = freq;
2860  	qdf_mem_copy(set_filter_req->mac, mac, QDF_MAC_ADDR_SIZE);
2861  	set_filter_req->set = set;
2862  	set_filter_req->cb = cb;
2863  	set_filter_req->req_cookie = req_cookie;
2864  
2865  	msg.type = P2P_SET_RANDOM_MAC;
2866  	msg.bodyptr = set_filter_req;
2867  	msg.callback = p2p_process_cmd;
2868  	status = scheduler_post_message(QDF_MODULE_ID_P2P, QDF_MODULE_ID_P2P,
2869  					QDF_MODULE_ID_OS_IF, &msg);
2870  	if (status != QDF_STATUS_SUCCESS)
2871  		qdf_mem_free(set_filter_req);
2872  
2873  	return status;
2874  }
2875  
2876  QDF_STATUS
p2p_clear_mac_filter(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * mac,uint32_t freq)2877  p2p_clear_mac_filter(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2878  		     uint8_t *mac, uint32_t freq)
2879  {
2880  	return p2p_set_mac_filter(soc, vdev_id, mac, freq, false, NULL, NULL);
2881  }
2882  
2883  bool
p2p_is_vdev_support_rand_mac(struct wlan_objmgr_vdev * vdev)2884  p2p_is_vdev_support_rand_mac(struct wlan_objmgr_vdev *vdev)
2885  {
2886  	enum QDF_OPMODE mode;
2887  
2888  	mode = wlan_vdev_mlme_get_opmode(vdev);
2889  	if (mode == QDF_STA_MODE ||
2890  	    mode == QDF_P2P_CLIENT_MODE ||
2891  	    mode == QDF_P2P_DEVICE_MODE ||
2892  	    mode == QDF_NAN_DISC_MODE)
2893  		return true;
2894  	return false;
2895  }
2896  
2897  /**
2898   * p2p_is_vdev_support_rand_mac_by_id() - check vdev type support random mac
2899   * mgmt tx or not
2900   * @soc: soc obj
2901   * @vdev_id: vdev id
2902   *
2903   * Return: true: support random mac mgmt tx
2904   *             false: not support random mac mgmt tx.
2905   */
2906  static bool
p2p_is_vdev_support_rand_mac_by_id(struct wlan_objmgr_psoc * soc,uint32_t vdev_id)2907  p2p_is_vdev_support_rand_mac_by_id(struct wlan_objmgr_psoc *soc,
2908  				   uint32_t vdev_id)
2909  {
2910  	struct wlan_objmgr_vdev *vdev;
2911  	bool ret = false;
2912  
2913  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id, WLAN_P2P_ID);
2914  	if (!vdev)
2915  		return false;
2916  	ret = p2p_is_vdev_support_rand_mac(vdev);
2917  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
2918  
2919  	return ret;
2920  }
2921  
2922  /**
2923   * p2p_set_rand_mac() - set random mac address rx filter
2924   * @soc: soc
2925   * @vdev_id: vdev id
2926   * @mac: mac addr
2927   * @freq: freq
2928   * @rnd_cookie: cookie to be returned
2929   *
2930   * This function will post msg to p2p core to set random mac addr rx filter.
2931   * It will wait the respone and return the result to caller.
2932   *
2933   * Return: true: set successfully
2934   *             false: failed
2935   */
2936  static bool
p2p_set_rand_mac(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * mac,uint32_t freq,uint64_t rnd_cookie)2937  p2p_set_rand_mac(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
2938  		 uint8_t *mac, uint32_t freq, uint64_t rnd_cookie)
2939  {
2940  	bool ret = false;
2941  	int err;
2942  	QDF_STATUS status;
2943  	struct osif_request *request;
2944  	static const struct osif_request_params params = {
2945  		.priv_size = sizeof(struct random_mac_priv),
2946  		.timeout_ms = WLAN_WAIT_TIME_SET_RND,
2947  	};
2948  	void *req_cookie;
2949  	struct random_mac_priv *priv;
2950  
2951  	request = osif_request_alloc(&params);
2952  	if (!request) {
2953  		p2p_err("Request allocation failure");
2954  		return false;
2955  	}
2956  
2957  	req_cookie = osif_request_cookie(request);
2958  
2959  	status = p2p_set_mac_filter(soc, vdev_id, mac, freq, true,
2960  				    p2p_set_mac_filter_callback, req_cookie);
2961  	if (status != QDF_STATUS_SUCCESS) {
2962  		p2p_err("random_mac:set mac filter failure %d", status);
2963  	} else {
2964  		err = osif_request_wait_for_response(request);
2965  		if (err) {
2966  			p2p_err("random_mac:timeout for set mac filter %d",
2967  				err);
2968  		} else {
2969  			priv = osif_request_priv(request);
2970  			ret = priv->result;
2971  			p2p_debug("random_mac:vdev %d freq %d result %d "QDF_MAC_ADDR_FMT" rnd_cookie %llu",
2972  				  vdev_id, freq, priv->result,
2973  				  QDF_MAC_ADDR_REF(mac), rnd_cookie);
2974  		}
2975  	}
2976  	osif_request_put(request);
2977  
2978  	return ret;
2979  }
2980  
2981  /**
2982   * p2p_mac_clear_timeout() - clear random mac filter timeout
2983   * @context: timer context
2984   *
2985   * This function will clear the mac addr rx filter in target if no
2986   * reference to it.
2987   *
2988   * Return: void
2989   */
p2p_mac_clear_timeout(void * context)2990  static void p2p_mac_clear_timeout(void *context)
2991  {
2992  	struct action_frame_random_mac *random_mac = context;
2993  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
2994  	uint32_t freq;
2995  	uint8_t addr[QDF_MAC_ADDR_SIZE];
2996  	uint32_t vdev_id;
2997  
2998  	if (!random_mac || !random_mac->p2p_vdev_obj) {
2999  		p2p_err("invalid context for mac_clear timeout");
3000  		return;
3001  	}
3002  	p2p_vdev_obj = random_mac->p2p_vdev_obj;
3003  	if (!p2p_vdev_obj || !p2p_vdev_obj->vdev)
3004  		return;
3005  
3006  	qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
3007  
3008  	delete_all_action_frame_cookie(&random_mac->cookie_list);
3009  	random_mac->in_use = false;
3010  	freq = random_mac->freq;
3011  	qdf_mem_copy(addr, random_mac->addr, QDF_MAC_ADDR_SIZE);
3012  	qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
3013  
3014  	vdev_id = wlan_vdev_get_id(p2p_vdev_obj->vdev);
3015  	p2p_debug("random_mac:clear timeout vdev %d " QDF_MAC_ADDR_FMT " freq %d",
3016  		  vdev_id, QDF_MAC_ADDR_REF(addr), freq);
3017  
3018  	p2p_clear_mac_filter(wlan_vdev_get_psoc(p2p_vdev_obj->vdev),
3019  			     vdev_id, addr, freq);
3020  }
3021  
3022  /**
3023   * p2p_request_random_mac() - request random mac mgmt tx
3024   * @soc: soc
3025   * @vdev_id: vdev id
3026   * @mac: mac addr
3027   * @freq: freq
3028   * @rnd_cookie: cookie to be returned
3029   * @duration: duration of tx timeout
3030   *
3031   * This function will add/append the random mac addr filter entry to vdev.
3032   * If it is new added entry, it will request to set filter in target.
3033   *
3034   * Return: QDF_STATUS_SUCCESS: request successfully
3035   *             other: failed
3036   */
3037  static QDF_STATUS
p2p_request_random_mac(struct wlan_objmgr_psoc * soc,uint32_t vdev_id,uint8_t * mac,uint32_t freq,uint64_t rnd_cookie,uint32_t duration)3038  p2p_request_random_mac(struct wlan_objmgr_psoc *soc, uint32_t vdev_id,
3039  		       uint8_t *mac, uint32_t freq, uint64_t rnd_cookie,
3040  		       uint32_t duration)
3041  {
3042  	QDF_STATUS status;
3043  	uint32_t i;
3044  	struct wlan_objmgr_vdev *vdev;
3045  	struct p2p_vdev_priv_obj *p2p_vdev_obj;
3046  
3047  	status = p2p_add_random_mac(soc, vdev_id, mac, freq, rnd_cookie);
3048  	if (status == QDF_STATUS_E_EXISTS)
3049  		return QDF_STATUS_SUCCESS;
3050  
3051  	else if (status != QDF_STATUS_SUCCESS)
3052  		return status;
3053  
3054  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(soc, vdev_id, WLAN_P2P_ID);
3055  	if (!vdev) {
3056  		p2p_debug("vdev is null");
3057  		return QDF_STATUS_E_INVAL;
3058  	}
3059  
3060  	p2p_vdev_obj = wlan_objmgr_vdev_get_comp_private_obj(
3061  						vdev, WLAN_UMAC_COMP_P2P);
3062  	if (!p2p_vdev_obj) {
3063  		p2p_debug("random_mac:p2p vdev object is NULL");
3064  		wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
3065  		return QDF_STATUS_E_INVAL;
3066  	}
3067  
3068  	if (!p2p_set_rand_mac(soc, vdev_id, mac, freq, rnd_cookie)) {
3069  		p2p_debug("random mac: failed to set rand mac address");
3070  
3071  		qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
3072  		for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
3073  			if (!qdf_mem_cmp(p2p_vdev_obj->random_mac[i].addr, mac,
3074  					 QDF_MAC_ADDR_SIZE)) {
3075  				qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
3076  				p2p_mac_clear_timeout(
3077  						&p2p_vdev_obj->random_mac[i]);
3078  				status = QDF_STATUS_SUCCESS;
3079  				qdf_spin_lock(&p2p_vdev_obj->random_mac_lock);
3080  				break;
3081  			}
3082  		}
3083  		qdf_spin_unlock(&p2p_vdev_obj->random_mac_lock);
3084  	}
3085  
3086  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
3087  	return status;
3088  }
3089  
p2p_rand_mac_tx(struct wlan_objmgr_pdev * pdev,struct tx_action_context * tx_action)3090  void p2p_rand_mac_tx(struct wlan_objmgr_pdev *pdev,
3091  		     struct tx_action_context *tx_action)
3092  {
3093  	struct wlan_objmgr_psoc *soc;
3094  	struct wlan_objmgr_vdev *vdev;
3095  	QDF_STATUS status;
3096  	bool is_vdev_up;
3097  
3098  	if (!tx_action || !tx_action->p2p_soc_obj ||
3099  	    !tx_action->p2p_soc_obj->soc)
3100  		return;
3101  	soc = tx_action->p2p_soc_obj->soc;
3102  
3103  	vdev = wlan_objmgr_get_vdev_by_id_from_pdev(pdev, tx_action->vdev_id,
3104  						    WLAN_P2P_ID);
3105  	if (!vdev) {
3106  		p2p_err("vdev is null id:%d", tx_action->vdev_id);
3107  		return;
3108  	}
3109  
3110  	/*
3111  	 * For PASN authentication frames, fw may request PASN authentication
3112  	 * with source address same as vdev mac address when vdev is not already
3113  	 * started. Allow RX_FILTER configuration for vdev mac address also if
3114  	 * vdev is not started to prevent PASN authentication frame drops.
3115  	 */
3116  	is_vdev_up = QDF_IS_STATUS_SUCCESS(wlan_vdev_is_up(vdev));
3117  	if (!tx_action->no_ack && tx_action->chan_freq &&
3118  	    tx_action->buf_len > MIN_MAC_HEADER_LEN &&
3119  	    p2p_is_vdev_support_rand_mac_by_id(soc, tx_action->vdev_id) &&
3120  	    (p2p_is_random_mac(soc, tx_action->vdev_id,
3121  			      &tx_action->buf[SRC_MAC_ADDR_OFFSET]) ||
3122  	     !is_vdev_up)) {
3123  		status = p2p_request_random_mac(
3124  			soc, tx_action->vdev_id,
3125  			&tx_action->buf[SRC_MAC_ADDR_OFFSET],
3126  			tx_action->chan_freq,
3127  			tx_action->id,
3128  			tx_action->duration);
3129  		if (status == QDF_STATUS_SUCCESS)
3130  			tx_action->rand_mac_tx = true;
3131  		else
3132  			tx_action->rand_mac_tx = false;
3133  	}
3134  
3135  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
3136  }
3137  
3138  void
p2p_rand_mac_tx_done(struct wlan_objmgr_psoc * soc,struct tx_action_context * tx_ctx)3139  p2p_rand_mac_tx_done(struct wlan_objmgr_psoc *soc,
3140  		     struct tx_action_context *tx_ctx)
3141  {
3142  	if (!tx_ctx || !tx_ctx->rand_mac_tx || !soc)
3143  		return;
3144  
3145  	p2p_random_mac_handle_tx_done(soc, tx_ctx->vdev_id, tx_ctx->id,
3146  				      tx_ctx->duration);
3147  }
3148  
p2p_init_random_mac_vdev(struct p2p_vdev_priv_obj * p2p_vdev_obj)3149  void p2p_init_random_mac_vdev(struct p2p_vdev_priv_obj *p2p_vdev_obj)
3150  {
3151  	int32_t i;
3152  
3153  	qdf_spinlock_create(&p2p_vdev_obj->random_mac_lock);
3154  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
3155  		qdf_mem_zero(&p2p_vdev_obj->random_mac[i],
3156  			     sizeof(struct action_frame_random_mac));
3157  		p2p_vdev_obj->random_mac[i].in_use = false;
3158  		p2p_vdev_obj->random_mac[i].p2p_vdev_obj = p2p_vdev_obj;
3159  		qdf_list_create(&p2p_vdev_obj->random_mac[i].cookie_list, 0);
3160  		qdf_mc_timer_init(&p2p_vdev_obj->random_mac[i].clear_timer,
3161  				  QDF_TIMER_TYPE_SW, p2p_mac_clear_timeout,
3162  				  &p2p_vdev_obj->random_mac[i]);
3163  	}
3164  }
3165  
p2p_deinit_random_mac_vdev(struct p2p_vdev_priv_obj * p2p_vdev_obj)3166  void p2p_deinit_random_mac_vdev(struct p2p_vdev_priv_obj *p2p_vdev_obj)
3167  {
3168  	int32_t i;
3169  
3170  	p2p_del_all_rand_mac_vdev(p2p_vdev_obj->vdev);
3171  	for (i = 0; i < MAX_RANDOM_MAC_ADDRS; i++) {
3172  		qdf_mc_timer_destroy(&p2p_vdev_obj->random_mac[i].clear_timer);
3173  		qdf_list_destroy(&p2p_vdev_obj->random_mac[i].cookie_list);
3174  	}
3175  	qdf_spinlock_destroy(&p2p_vdev_obj->random_mac_lock);
3176  }
3177  
p2p_process_mgmt_tx(struct tx_action_context * tx_ctx)3178  QDF_STATUS p2p_process_mgmt_tx(struct tx_action_context *tx_ctx)
3179  {
3180  	struct p2p_soc_priv_obj *p2p_soc_obj;
3181  	struct p2p_roc_context *curr_roc_ctx;
3182  	uint8_t *mac_to;
3183  	enum QDF_OPMODE mode;
3184  	struct wlan_objmgr_vdev *vdev;
3185  	QDF_STATUS status;
3186  	bool is_vdev_connected = false;
3187  
3188  	status = p2p_tx_context_check_valid(tx_ctx);
3189  	if (status != QDF_STATUS_SUCCESS) {
3190  		p2p_err("invalid tx action context");
3191  		if (tx_ctx) {
3192  			if (tx_ctx->buf) {
3193  				p2p_send_tx_conf(tx_ctx, false);
3194  				qdf_mem_free(tx_ctx->buf);
3195  			}
3196  			qdf_mem_free(tx_ctx);
3197  		}
3198  		return QDF_STATUS_E_INVAL;
3199  	}
3200  
3201  	p2p_soc_obj = tx_ctx->p2p_soc_obj;
3202  
3203  	p2p_debug("soc:%pK, tx_ctx:%pK, vdev_id:%d, scan_id:%d, "
3204  		  "roc_cookie:%llx, freq:%d, buf:%pK, len:%d, "
3205  		  "off_chan:%d, cck:%d, ack:%d, duration:%d",
3206  		  p2p_soc_obj->soc, tx_ctx, tx_ctx->vdev_id,
3207  		  tx_ctx->scan_id, tx_ctx->roc_cookie, tx_ctx->chan_freq,
3208  		  tx_ctx->buf, tx_ctx->buf_len, tx_ctx->off_chan,
3209  		  tx_ctx->no_cck, tx_ctx->no_ack, tx_ctx->duration);
3210  
3211  	status = p2p_get_frame_info(tx_ctx->buf, tx_ctx->buf_len,
3212  			&(tx_ctx->frame_info));
3213  	if (status != QDF_STATUS_SUCCESS) {
3214  		p2p_err("unsupported frame");
3215  		status = QDF_STATUS_E_INVAL;
3216  		goto fail;
3217  	}
3218  
3219  	/* update P2P connection status with tx frame info */
3220  	mac_to = &(tx_ctx->buf[DST_MAC_ADDR_OFFSET]);
3221  	p2p_tx_update_connection_status(p2p_soc_obj,
3222  		&(tx_ctx->frame_info), mac_to);
3223  
3224  	status = p2p_vdev_check_valid(tx_ctx);
3225  	if (status != QDF_STATUS_SUCCESS) {
3226  		p2p_debug("invalid vdev or vdev mode");
3227  		status = QDF_STATUS_E_INVAL;
3228  		goto fail;
3229  	}
3230  
3231  	/* Do not wait for ack for probe response */
3232  	if (tx_ctx->frame_info.sub_type == P2P_MGMT_PROBE_RSP &&
3233  		!(tx_ctx->no_ack)) {
3234  		p2p_debug("Force set no ack to 1");
3235  		tx_ctx->no_ack = 1;
3236  	}
3237  
3238  	vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
3239  			p2p_soc_obj->soc, tx_ctx->vdev_id, WLAN_P2P_ID);
3240  	if (!vdev) {
3241  		p2p_err("null vdev object");
3242  		goto fail;
3243  	}
3244  
3245  	mode = wlan_vdev_mlme_get_opmode(vdev);
3246  	if (mode == QDF_STA_MODE)
3247  		is_vdev_connected = wlan_cm_is_vdev_connected(vdev);
3248  
3249  	wlan_objmgr_vdev_release_ref(vdev, WLAN_P2P_ID);
3250  
3251  	if (!tx_ctx->off_chan || !tx_ctx->chan_freq) {
3252  		if (!tx_ctx->chan_freq)
3253  			p2p_check_and_update_channel(tx_ctx);
3254  		if (!tx_ctx->chan_freq && mode == QDF_STA_MODE &&
3255  		    !is_vdev_connected) {
3256  			p2p_debug("chan freq is zero, drop tx mgmt frame");
3257  			goto fail;
3258  		}
3259  		status = p2p_execute_tx_action_frame(tx_ctx);
3260  		if (status != QDF_STATUS_SUCCESS) {
3261  			p2p_err("execute tx fail");
3262  			goto fail;
3263  		} else
3264  			return QDF_STATUS_SUCCESS;
3265  	}
3266  
3267  	/* For off channel tx case */
3268  	curr_roc_ctx = p2p_find_current_roc_ctx(p2p_soc_obj);
3269  	if (curr_roc_ctx && (curr_roc_ctx->chan_freq == tx_ctx->chan_freq)) {
3270  		if ((curr_roc_ctx->roc_state == ROC_STATE_REQUESTED) ||
3271  		    (curr_roc_ctx->roc_state == ROC_STATE_STARTED)) {
3272  			tx_ctx->roc_cookie = (uintptr_t)curr_roc_ctx;
3273  			status = qdf_list_insert_back(
3274  					&p2p_soc_obj->tx_q_roc,
3275  					&tx_ctx->node);
3276  			if (status != QDF_STATUS_SUCCESS) {
3277  				p2p_err("Failed to insert off chan tx context to wait roc req queue");
3278  				goto fail;
3279  			} else
3280  				return QDF_STATUS_SUCCESS;
3281  		} else if (curr_roc_ctx->roc_state == ROC_STATE_ON_CHAN) {
3282  			p2p_adjust_tx_wait(tx_ctx);
3283  			if (curr_roc_ctx->duration < tx_ctx->duration)
3284  				curr_roc_ctx->duration = tx_ctx->duration;
3285  			status = p2p_restart_roc_timer(curr_roc_ctx);
3286  			curr_roc_ctx->tx_ctx = tx_ctx;
3287  			if (status != QDF_STATUS_SUCCESS) {
3288  				p2p_err("restart roc timer fail");
3289  				goto fail;
3290  			}
3291  			status = p2p_execute_tx_action_frame(tx_ctx);
3292  			if (status != QDF_STATUS_SUCCESS) {
3293  				p2p_err("execute tx fail");
3294  				goto fail;
3295  			} else
3296  				return QDF_STATUS_SUCCESS;
3297  		}
3298  	}
3299  
3300  	curr_roc_ctx = p2p_find_roc_by_chan_freq(p2p_soc_obj,
3301  						 tx_ctx->chan_freq);
3302  	if (curr_roc_ctx && (curr_roc_ctx->roc_state == ROC_STATE_IDLE)) {
3303  		tx_ctx->roc_cookie = (uintptr_t)curr_roc_ctx;
3304  		status = qdf_list_insert_back(
3305  				&p2p_soc_obj->tx_q_roc,
3306  				&tx_ctx->node);
3307  		if (status != QDF_STATUS_SUCCESS) {
3308  			p2p_err("Failed to insert off chan tx context to wait roc req queue");
3309  			goto fail;
3310  		} else {
3311  			return QDF_STATUS_SUCCESS;
3312  		}
3313  	}
3314  
3315  	if (!tx_ctx->duration) {
3316  		tx_ctx->duration = P2P_ACTION_FRAME_DEFAULT_WAIT;
3317  		p2p_debug("use default wait %d",
3318  			  P2P_ACTION_FRAME_DEFAULT_WAIT);
3319  	}
3320  	status = p2p_roc_req_for_tx_action(tx_ctx);
3321  	if (status != QDF_STATUS_SUCCESS) {
3322  		p2p_err("Failed to request roc before off chan tx");
3323  		goto fail;
3324  	}
3325  
3326  	return QDF_STATUS_SUCCESS;
3327  
3328  fail:
3329  	p2p_send_tx_conf(tx_ctx, false);
3330  	qdf_idr_remove(&p2p_soc_obj->p2p_idr, tx_ctx->id);
3331  	qdf_mem_free(tx_ctx->buf);
3332  	qdf_mem_free(tx_ctx);
3333  
3334  	return status;
3335  }
3336  
p2p_process_mgmt_tx_cancel(struct cancel_roc_context * cancel_tx)3337  QDF_STATUS p2p_process_mgmt_tx_cancel(
3338  	struct cancel_roc_context *cancel_tx)
3339  {
3340  	bool is_roc_q = false;
3341  	bool is_ack_q = false;
3342  	struct tx_action_context *cur_tx_ctx;
3343  	struct p2p_roc_context *cur_roc_ctx;
3344  	struct cancel_roc_context cancel_roc;
3345  
3346  	if (!cancel_tx || !(cancel_tx->cookie)) {
3347  		p2p_info("invalid cancel info");
3348  		return QDF_STATUS_SUCCESS;
3349  	}
3350  
3351  	cur_tx_ctx = p2p_find_tx_ctx(cancel_tx->p2p_soc_obj,
3352  			cancel_tx->cookie, &is_roc_q, &is_ack_q);
3353  	if (cur_tx_ctx) {
3354  		if (is_roc_q) {
3355  			cancel_roc.p2p_soc_obj =
3356  					cancel_tx->p2p_soc_obj;
3357  			cancel_roc.cookie =
3358  					cur_tx_ctx->roc_cookie;
3359  			p2p_remove_tx_context(cur_tx_ctx);
3360  			return p2p_process_cancel_roc_req(&cancel_roc);
3361  		}
3362  		if (is_ack_q) {
3363  			/*Has tx action frame, waiting for ack*/
3364  			p2p_debug("Waiting for ack, cookie %llx",
3365  				cancel_tx->cookie);
3366  		}
3367  	} else {
3368  		p2p_debug("Failed to find tx ctx by cookie, cookie %llx",
3369  			cancel_tx->cookie);
3370  
3371  		cur_roc_ctx = p2p_find_roc_by_tx_ctx(cancel_tx->p2p_soc_obj,
3372  					cancel_tx->cookie);
3373  		if (cur_roc_ctx) {
3374  			p2p_debug("tx ctx:%llx, roc:%pK",
3375  				cancel_tx->cookie, cur_roc_ctx);
3376  			cancel_roc.p2p_soc_obj =
3377  					cancel_tx->p2p_soc_obj;
3378  			cancel_roc.cookie = (uintptr_t) cur_roc_ctx;
3379  			return p2p_process_cancel_roc_req(&cancel_roc);
3380  		}
3381  
3382  		p2p_debug("Failed to find roc by tx ctx");
3383  		return QDF_STATUS_E_INVAL;
3384  	}
3385  
3386  	return QDF_STATUS_SUCCESS;
3387  }
3388  
p2p_process_mgmt_tx_ack_cnf(struct p2p_tx_conf_event * tx_cnf_event)3389  QDF_STATUS p2p_process_mgmt_tx_ack_cnf(
3390  	struct p2p_tx_conf_event *tx_cnf_event)
3391  {
3392  	struct p2p_tx_cnf tx_cnf;
3393  	struct tx_action_context *tx_ctx;
3394  	struct p2p_soc_priv_obj *p2p_soc_obj;
3395  	struct p2p_start_param *start_param;
3396  
3397  	p2p_soc_obj = tx_cnf_event->p2p_soc_obj;
3398  
3399  	if (!p2p_soc_obj || !(p2p_soc_obj->start_param)) {
3400  		qdf_nbuf_free(tx_cnf_event->nbuf);
3401  		p2p_err("Invalid p2p soc object or start parameters");
3402  		return QDF_STATUS_E_INVAL;
3403  	}
3404  
3405  	tx_ctx = p2p_find_tx_ctx_by_nbuf(p2p_soc_obj, tx_cnf_event->nbuf);
3406  	qdf_nbuf_free(tx_cnf_event->nbuf);
3407  	if (!tx_ctx) {
3408  		p2p_err("can't find tx_ctx, tx ack comes late");
3409  		return QDF_STATUS_SUCCESS;
3410  	}
3411  
3412  	/* disable tx timer */
3413  	p2p_disable_tx_timer(tx_ctx);
3414  	tx_cnf.vdev_id = tx_ctx->vdev_id;
3415  	tx_cnf.action_cookie = (uint64_t)tx_ctx->id;
3416  	tx_cnf.buf = tx_ctx->buf;
3417  	tx_cnf.buf_len = tx_ctx->buf_len;
3418  	tx_cnf.status = tx_cnf_event->status;
3419  
3420  	p2p_debug("soc:%pK, vdev_id:%d, action_cookie:%llx, len:%d, status:%d, buf:%pK",
3421  		p2p_soc_obj->soc, tx_cnf.vdev_id,
3422  		tx_cnf.action_cookie, tx_cnf.buf_len,
3423  		tx_cnf.status, tx_cnf.buf);
3424  
3425  	p2p_rand_mac_tx_done(p2p_soc_obj->soc, tx_ctx);
3426  
3427  	start_param = p2p_soc_obj->start_param;
3428  	if (start_param->tx_cnf_cb)
3429  		start_param->tx_cnf_cb(start_param->tx_cnf_cb_data,
3430  					&tx_cnf);
3431  	else
3432  		p2p_debug("Got tx conf, but no valid up layer callback");
3433  
3434  	p2p_remove_tx_context(tx_ctx);
3435  
3436  	return QDF_STATUS_SUCCESS;
3437  }
3438  
3439  #define P2P_IS_SOCIAL_CHANNEL(center_freq)	\
3440  	(((center_freq) == 2412) || ((center_freq) == 2437) || \
3441  		((center_freq) == 2462))
3442  
p2p_process_rx_mgmt(struct p2p_rx_mgmt_event * rx_mgmt_event)3443  QDF_STATUS p2p_process_rx_mgmt(
3444  	struct p2p_rx_mgmt_event *rx_mgmt_event)
3445  {
3446  	struct p2p_rx_mgmt_frame *rx_mgmt;
3447  	struct p2p_soc_priv_obj *p2p_soc_obj;
3448  	struct p2p_start_param *start_param;
3449  	struct p2p_frame_info frame_info;
3450  	uint8_t *mac_from;
3451  
3452  	p2p_soc_obj = rx_mgmt_event->p2p_soc_obj;
3453  	rx_mgmt = rx_mgmt_event->rx_mgmt;
3454  
3455  	if (!p2p_soc_obj || !(p2p_soc_obj->start_param)) {
3456  		p2p_err("Invalid psoc object or start parameters");
3457  		qdf_mem_free(rx_mgmt);
3458  		return QDF_STATUS_E_INVAL;
3459  	}
3460  
3461  	p2p_debug("soc:%pK, frame_len:%d, rx_freq:%d, vdev_id:%d, frm_type:%d, rx_rssi:%d, buf:%pK",
3462  		  p2p_soc_obj->soc, rx_mgmt->frame_len,
3463  		  rx_mgmt->rx_freq, rx_mgmt->vdev_id, rx_mgmt->frm_type,
3464  		  rx_mgmt->rx_rssi, rx_mgmt->buf);
3465  
3466  	if (rx_mgmt->frm_type == MGMT_ACTION_VENDOR_SPECIFIC) {
3467  		p2p_get_frame_info(rx_mgmt->buf, rx_mgmt->frame_len,
3468  				&frame_info);
3469  
3470  		/* update P2P connection status with rx frame info */
3471  		mac_from = &(rx_mgmt->buf[SRC_MAC_ADDR_OFFSET]);
3472  		p2p_rx_update_connection_status(p2p_soc_obj,
3473  						&frame_info, mac_from);
3474  
3475  		p2p_debug("action_sub_type %u, action_type %d",
3476  				frame_info.public_action_type,
3477  				frame_info.action_type);
3478  
3479  		if ((frame_info.public_action_type ==
3480  			P2P_PUBLIC_ACTION_NOT_SUPPORT) &&
3481  		   (frame_info.action_type ==
3482  			P2P_ACTION_NOT_SUPPORT)) {
3483  			p2p_debug("non-p2p frame, drop it");
3484  			qdf_mem_free(rx_mgmt);
3485  			return QDF_STATUS_SUCCESS;
3486  		} else {
3487  			p2p_debug("p2p frame, extend roc accordingly");
3488  			p2p_extend_roc_timer(p2p_soc_obj, &frame_info);
3489  		}
3490  
3491  		if (frame_info.public_action_type ==
3492  		    P2P_PUBLIC_ACTION_NEG_REQ &&
3493  		    wlan_reg_is_24ghz_ch_freq(rx_mgmt->rx_freq) &&
3494  		    !P2P_IS_SOCIAL_CHANNEL(rx_mgmt->rx_freq)) {
3495  			p2p_debug("Drop P2P Negotiation Req due to non-Social channel: %d",
3496  				  rx_mgmt->rx_freq);
3497  			qdf_mem_free(rx_mgmt);
3498  			return QDF_STATUS_SUCCESS;
3499  		}
3500  	}
3501  
3502  	if (rx_mgmt->frm_type == MGMT_ACTION_CATEGORY_VENDOR_SPECIFIC)
3503  		p2p_get_frame_info(rx_mgmt->buf, rx_mgmt->frame_len,
3504  				&frame_info);
3505  
3506  	start_param = p2p_soc_obj->start_param;
3507  	if (start_param->rx_cb)
3508  		start_param->rx_cb(start_param->rx_cb_data, rx_mgmt);
3509  	else
3510  		p2p_debug("rx mgmt, but no valid up layer callback");
3511  
3512  	qdf_mem_free(rx_mgmt);
3513  
3514  	return QDF_STATUS_SUCCESS;
3515  }
3516