blob: b6441bff43dc2d7023867d0f23e825c518dc102d [file] [log] [blame]
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001/*
Yun Parke4239802018-01-09 11:01:40 -08002 * Copyright (c) 2012-2018 The Linux Foundation. All rights reserved.
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08003 *
4 * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
5 *
6 *
7 * Permission to use, copy, modify, and/or distribute this software for
8 * any purpose with or without fee is hereby granted, provided that the
9 * above copyright notice and this permission notice appear in all
10 * copies.
11 *
12 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
13 * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
14 * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
15 * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
16 * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
17 * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
18 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
19 * PERFORMANCE OF THIS SOFTWARE.
20 */
21
22/*
23 * This file was originally distributed by Qualcomm Atheros, Inc.
24 * under proprietary terms before Copyright ownership was assigned
25 * to the Linux Foundation.
26 */
27
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -080028/* denote that this file does not allow legacy hddLog */
29#define HDD_DISALLOW_LEGACY_HDDLOG 1
30
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080031/* Include files */
32#include <linux/semaphore.h>
33#include <wlan_hdd_tx_rx.h>
34#include <wlan_hdd_softap_tx_rx.h>
35#include <linux/netdevice.h>
36#include <linux/skbuff.h>
37#include <linux/etherdevice.h>
Anurag Chouhan6d760662016-02-20 16:05:43 +053038#include <qdf_types.h>
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080039#include <ani_global.h>
Anurag Chouhan6d760662016-02-20 16:05:43 +053040#include <qdf_types.h>
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080041#include <net/ieee80211_radiotap.h>
42#include <cds_sched.h>
43#include <wlan_hdd_napi.h>
Leo Changfdb45c32016-10-28 11:09:23 -070044#include <cdp_txrx_cmn.h>
Dhanashri Atreb08959a2016-03-01 17:28:03 -080045#include <cdp_txrx_peer_ops.h>
Manjunathappa Prakash779e4862016-09-12 17:00:11 -070046#include <cds_utils.h>
Leo Changfdb45c32016-10-28 11:09:23 -070047#include <cdp_txrx_flow_ctrl_v2.h>
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +053048#include <cdp_txrx_misc.h>
Abhishek Singh07c627e2017-03-20 17:56:34 +053049#include <wlan_hdd_object_manager.h>
Rachit Kankane2487f8f2017-04-19 14:30:19 +053050#include "wlan_p2p_ucfg_api.h"
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080051#include <wlan_hdd_ipa.h>
Mukul Sharmaecf8e092017-12-19 22:36:31 +053052#include <wlan_hdd_regulatory.h>
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080053
54/* Preprocessor definitions and constants */
55#undef QCA_HDD_SAP_DUMP_SK_BUFF
56
57/* Type declarations */
58
59/* Function definitions and documenation */
60#ifdef QCA_HDD_SAP_DUMP_SK_BUFF
61/**
62 * hdd_softap_dump_sk_buff() - Dump an skb
63 * @skb: skb to dump
64 *
65 * Return: None
66 */
67static void hdd_softap_dump_sk_buff(struct sk_buff *skb)
68{
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053069 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Jeff Johnson36e74c42017-09-18 08:15:42 -070070 "%s: head = %pK ", __func__, skb->head);
Srinivas Girigowdae806a822017-03-25 11:25:30 -070071 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Jeff Johnson36e74c42017-09-18 08:15:42 -070072 "%s: tail = %pK ", __func__, skb->tail);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053073 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Jeff Johnson36e74c42017-09-18 08:15:42 -070074 "%s: end = %pK ", __func__, skb->end);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053075 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080076 "%s: len = %d ", __func__, skb->len);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053077 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080078 "%s: data_len = %d ", __func__, skb->data_len);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053079 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080080 "%s: mac_len = %d", __func__, skb->mac_len);
81
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053082 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080083 "0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x ", skb->data[0],
84 skb->data[1], skb->data[2], skb->data[3], skb->data[4],
85 skb->data[5], skb->data[6], skb->data[7]);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053086 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080087 "0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x", skb->data[8],
88 skb->data[9], skb->data[10], skb->data[11], skb->data[12],
89 skb->data[13], skb->data[14], skb->data[15]);
90}
91#else
92static void hdd_softap_dump_sk_buff(struct sk_buff *skb)
93{
94}
95#endif
96
97#ifdef QCA_LL_LEGACY_TX_FLOW_CONTROL
98/**
99 * hdd_softap_tx_resume_timer_expired_handler() - TX Q resume timer handler
100 * @adapter_context: pointer to vdev adapter
101 *
102 * TX Q resume timer handler for SAP and P2P GO interface. If Blocked
103 * OS Q is not resumed during timeout period, to prevent permanent
104 * stall, resume OS Q forcefully for SAP and P2P GO interface.
105 *
106 * Return: None
107 */
108void hdd_softap_tx_resume_timer_expired_handler(void *adapter_context)
109{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700110 struct hdd_adapter *adapter = (struct hdd_adapter *) adapter_context;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800111
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700112 if (!adapter) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800113 hdd_err("NULL adapter");
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800114 return;
115 }
116
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700117 hdd_debug("Enabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700118 wlan_hdd_netif_queue_control(adapter, WLAN_WAKE_ALL_NETIF_QUEUE,
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800119 WLAN_CONTROL_PATH);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800120}
121
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530122#if defined(CONFIG_PER_VDEV_TX_DESC_POOL)
123
124/**
125 * hdd_softap_tx_resume_false() - Resume OS TX Q false leads to queue disabling
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700126 * @adapter: pointer to hdd adapter
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530127 * @tx_resume: TX Q resume trigger
128 *
129 *
130 * Return: None
131 */
132static void
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700133hdd_softap_tx_resume_false(struct hdd_adapter *adapter, bool tx_resume)
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530134{
135 if (true == tx_resume)
136 return;
137
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700138 hdd_debug("Disabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700139 wlan_hdd_netif_queue_control(adapter, WLAN_STOP_ALL_NETIF_QUEUE,
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530140 WLAN_DATA_FLOW_CONTROL);
141
142 if (QDF_TIMER_STATE_STOPPED ==
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700143 qdf_mc_timer_get_current_state(&adapter->
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530144 tx_flow_control_timer)) {
145 QDF_STATUS status;
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700146
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700147 status = qdf_mc_timer_start(&adapter->tx_flow_control_timer,
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530148 WLAN_SAP_HDD_TX_FLOW_CONTROL_OS_Q_BLOCK_TIME);
149
150 if (!QDF_IS_STATUS_SUCCESS(status))
151 hdd_err("Failed to start tx_flow_control_timer");
152 else
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700153 adapter->hdd_stats.tx_rx_stats.txflow_timer_cnt++;
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530154 }
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530155}
156#else
157
158static inline void
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700159hdd_softap_tx_resume_false(struct hdd_adapter *adapter, bool tx_resume)
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530160{
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530161}
162#endif
163
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800164/**
165 * hdd_softap_tx_resume_cb() - Resume OS TX Q.
166 * @adapter_context: pointer to vdev apdapter
167 * @tx_resume: TX Q resume trigger
168 *
169 * Q was stopped due to WLAN TX path low resource condition
170 *
171 * Return: None
172 */
173void hdd_softap_tx_resume_cb(void *adapter_context, bool tx_resume)
174{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700175 struct hdd_adapter *adapter = (struct hdd_adapter *) adapter_context;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800176
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700177 if (!adapter) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800178 hdd_err("NULL adapter");
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800179 return;
180 }
181
182 /* Resume TX */
183 if (true == tx_resume) {
Anurag Chouhan210db072016-02-22 18:42:15 +0530184 if (QDF_TIMER_STATE_STOPPED !=
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700185 qdf_mc_timer_get_current_state(&adapter->
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800186 tx_flow_control_timer)) {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700187 qdf_mc_timer_stop(&adapter->tx_flow_control_timer);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800188 }
189
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700190 hdd_debug("Enabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700191 wlan_hdd_netif_queue_control(adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800192 WLAN_WAKE_ALL_NETIF_QUEUE,
193 WLAN_DATA_FLOW_CONTROL);
194 }
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700195 hdd_softap_tx_resume_false(adapter, tx_resume);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800196}
gbianec670c592016-11-24 11:21:30 +0800197
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700198static inline struct sk_buff *hdd_skb_orphan(struct hdd_adapter *adapter,
gbianec670c592016-11-24 11:21:30 +0800199 struct sk_buff *skb)
200{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700201 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
tfyubdf453e2017-09-27 13:34:30 +0800202 int need_orphan = 0;
203
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700204 if (adapter->tx_flow_low_watermark > 0) {
tfyubdf453e2017-09-27 13:34:30 +0800205#if (LINUX_VERSION_CODE > KERNEL_VERSION(3, 19, 0))
206 /*
207 * The TCP TX throttling logic is changed a little after
208 * 3.19-rc1 kernel, the TCP sending limit will be smaller,
209 * which will throttle the TCP packets to the host driver.
210 * The TCP UP LINK throughput will drop heavily. In order to
211 * fix this issue, need to orphan the socket buffer asap, which
212 * will call skb's destructor to notify the TCP stack that the
213 * SKB buffer is unowned. And then the TCP stack will pump more
214 * packets to host driver.
215 *
216 * The TX packets might be dropped for UDP case in the iperf
217 * testing. So need to be protected by follow control.
218 */
219 need_orphan = 1;
220#else
221 if (hdd_ctx->config->tx_orphan_enable)
222 need_orphan = 1;
223#endif
tfyu5f01db22017-10-11 13:51:04 +0800224 } else if (hdd_ctx->config->tx_orphan_enable) {
225 if (qdf_nbuf_is_ipv4_tcp_pkt(skb) ||
Tiger Yu438c6482017-10-13 11:07:00 +0800226 qdf_nbuf_is_ipv6_tcp_pkt(skb))
tfyu5f01db22017-10-11 13:51:04 +0800227 need_orphan = 1;
tfyubdf453e2017-09-27 13:34:30 +0800228 }
229
tfyu5f01db22017-10-11 13:51:04 +0800230 if (need_orphan) {
gbianec670c592016-11-24 11:21:30 +0800231 skb_orphan(skb);
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700232 ++adapter->hdd_stats.tx_rx_stats.tx_orphaned;
Tiger Yu438c6482017-10-13 11:07:00 +0800233 } else
gbianec670c592016-11-24 11:21:30 +0800234 skb = skb_unshare(skb, GFP_ATOMIC);
gbianec670c592016-11-24 11:21:30 +0800235
236 return skb;
237}
238
239#else
Mohit Khannad0b63f52017-02-18 18:05:52 -0800240/**
241 * hdd_skb_orphan() - skb_unshare a cloned packed else skb_orphan
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700242 * @adapter: pointer to HDD adapter
Mohit Khannad0b63f52017-02-18 18:05:52 -0800243 * @skb: pointer to skb data packet
244 *
245 * Return: pointer to skb structure
246 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700247static inline struct sk_buff *hdd_skb_orphan(struct hdd_adapter *adapter,
Mohit Khannad0b63f52017-02-18 18:05:52 -0800248 struct sk_buff *skb) {
gbianec670c592016-11-24 11:21:30 +0800249
Mohit Khannad0b63f52017-02-18 18:05:52 -0800250 struct sk_buff *nskb;
tfyubdf453e2017-09-27 13:34:30 +0800251#if (LINUX_VERSION_CODE > KERNEL_VERSION(3, 19, 0))
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700252 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
tfyubdf453e2017-09-27 13:34:30 +0800253#endif
Mohit Khannad0b63f52017-02-18 18:05:52 -0800254
Mohit Khanna87493732017-08-27 23:26:44 -0700255 hdd_skb_fill_gso_size(adapter->dev, skb);
256
Manjunathappa Prakashdab74fa2017-06-19 12:11:03 -0700257 nskb = skb_unshare(skb, GFP_ATOMIC);
tfyubdf453e2017-09-27 13:34:30 +0800258#if (LINUX_VERSION_CODE > KERNEL_VERSION(3, 19, 0))
Manjunathappa Prakashdab74fa2017-06-19 12:11:03 -0700259 if (unlikely(hdd_ctx->config->tx_orphan_enable) && (nskb == skb)) {
Mohit Khannad0b63f52017-02-18 18:05:52 -0800260 /*
261 * For UDP packets we want to orphan the packet to allow the app
262 * to send more packets. The flow would ultimately be controlled
263 * by the limited number of tx descriptors for the vdev.
264 */
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700265 ++adapter->hdd_stats.tx_rx_stats.tx_orphaned;
Mohit Khannad0b63f52017-02-18 18:05:52 -0800266 skb_orphan(skb);
267 }
tfyubdf453e2017-09-27 13:34:30 +0800268#endif
Mohit Khannad0b63f52017-02-18 18:05:52 -0800269 return nskb;
gbianec670c592016-11-24 11:21:30 +0800270}
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800271#endif /* QCA_LL_LEGACY_TX_FLOW_CONTROL */
272
273/**
Mukul Sharmac4de4ef2016-09-12 15:39:00 +0530274 * __hdd_softap_hard_start_xmit() - Transmit a frame
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800275 * @skb: pointer to OS packet (sk_buff)
276 * @dev: pointer to network device
277 *
278 * Function registered with the Linux OS for transmitting
279 * packets. This version of the function directly passes
280 * the packet to Transport Layer.
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530281 * In case of any packet drop or error, log the error with
282 * INFO HIGH/LOW/MEDIUM to avoid excessive logging in kmsg.
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800283 *
284 * Return: Always returns NETDEV_TX_OK
285 */
Jeff Johnson6376abe2016-10-05 16:24:56 -0700286static int __hdd_softap_hard_start_xmit(struct sk_buff *skb,
287 struct net_device *dev)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800288{
289 sme_ac_enum_type ac = SME_AC_BE;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700290 struct hdd_adapter *adapter = (struct hdd_adapter *) netdev_priv(dev);
Jeff Johnson9bf24972017-10-04 09:26:58 -0700291 struct hdd_ap_ctx *ap_ctx = WLAN_HDD_GET_AP_CTX_PTR(adapter);
Anurag Chouhan6d760662016-02-20 16:05:43 +0530292 struct qdf_mac_addr *pDestMacAddress;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800293 uint8_t STAId;
Will Huang496b36c2017-07-11 16:38:50 +0800294 uint32_t num_seg;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800295
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700296 ++adapter->hdd_stats.tx_rx_stats.tx_called;
297 adapter->hdd_stats.tx_rx_stats.cont_txtimeout_cnt = 0;
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +0530298
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800299 /* Prevent this function from being called during SSR since TL
300 * context may not be reinitialized at this time which may
301 * lead to a crash.
302 */
Will Huang20de9432018-02-06 17:01:03 +0800303 if (cds_is_driver_recovering() || cds_is_driver_in_bad_state() ||
304 cds_is_load_or_unload_in_progress()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530305 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
Will Huang20de9432018-02-06 17:01:03 +0800306 "%s: Recovery/(Un)load in Progress. Ignore!!!",
307 __func__);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800308 goto drop_pkt;
309 }
310
311 /*
312 * If the device is operating on a DFS Channel
313 * then check if SAP is in CAC WAIT state and
314 * drop the packets. In CAC WAIT state device
315 * is expected not to transmit any frames.
316 * SAP starts Tx only after the BSS START is
317 * done.
318 */
Jeff Johnson9bf24972017-10-04 09:26:58 -0700319 if (ap_ctx->dfs_cac_block_tx)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800320 goto drop_pkt;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800321
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800322 /*
Jeff Johnson4fca3b52017-01-12 08:44:18 -0800323 * If a transmit function is not registered, drop packet
324 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700325 if (!adapter->tx_fn) {
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800326 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
327 "%s: TX function not registered by the data path",
328 __func__);
329 goto drop_pkt;
330 }
331
Nirav Shah5e74bb82016-07-20 16:01:27 +0530332 wlan_hdd_classify_pkt(skb);
333
Anurag Chouhan6d760662016-02-20 16:05:43 +0530334 pDestMacAddress = (struct qdf_mac_addr *) skb->data;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800335
Nirav Shah5e74bb82016-07-20 16:01:27 +0530336 if (QDF_NBUF_CB_GET_IS_BCAST(skb) ||
337 QDF_NBUF_CB_GET_IS_MCAST(skb)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800338 /* The BC/MC station ID is assigned during BSS
339 * starting phase. SAP will return the station ID
340 * used for BC/MC traffic.
341 */
Jeff Johnson42518cf2017-10-26 13:33:29 -0700342 STAId = ap_ctx->broadcast_sta_id;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800343 } else {
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530344 if (QDF_STATUS_SUCCESS !=
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700345 hdd_softap_get_sta_id(adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800346 pDestMacAddress, &STAId)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530347 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530348 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800349 "%s: Failed to find right station", __func__);
350 goto drop_pkt;
351 }
352
Prakash Dhavali3107b752016-11-28 14:35:06 +0530353 if (STAId >= WLAN_MAX_STA_COUNT) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530354 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530355 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800356 "%s: Failed to find right station", __func__);
357 goto drop_pkt;
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700358 } else if (false == adapter->sta_info[STAId].in_use) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530359 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530360 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800361 "%s: STA %d is unregistered", __func__,
362 STAId);
363 goto drop_pkt;
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700364 } else if (true == adapter->sta_info[STAId].
Jeff Johnsone4f5d932017-10-21 13:21:15 -0700365 is_deauth_in_progress) {
Poddar, Siddartha5075462017-03-16 19:20:09 +0530366 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530367 QDF_TRACE_LEVEL_INFO_HIGH,
Poddar, Siddartha5075462017-03-16 19:20:09 +0530368 "%s: STA %d deauth in progress", __func__,
369 STAId);
370 goto drop_pkt;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800371 }
372
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800373 if ((OL_TXRX_PEER_STATE_CONN !=
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700374 adapter->sta_info[STAId].peer_state)
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800375 && (OL_TXRX_PEER_STATE_AUTH !=
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700376 adapter->sta_info[STAId].peer_state)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530377 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530378 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800379 "%s: Station not connected yet", __func__);
380 goto drop_pkt;
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800381 } else if (OL_TXRX_PEER_STATE_CONN ==
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700382 adapter->sta_info[STAId].peer_state) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800383 if (ntohs(skb->protocol) != HDD_ETHERTYPE_802_1_X) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530384 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530385 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800386 "%s: NON-EAPOL packet in non-Authenticated state",
387 __func__);
388 goto drop_pkt;
389 }
390 }
391 }
392
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700393 hdd_get_tx_resource(adapter, STAId,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800394 WLAN_SAP_HDD_TX_FLOW_CONTROL_OS_Q_BLOCK_TIME);
395
396 /* Get TL AC corresponding to Qdisc queue index/AC. */
397 ac = hdd_qdisc_ac_to_tl_ac[skb->queue_mapping];
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700398 ++adapter->hdd_stats.tx_rx_stats.tx_classified_ac[ac];
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800399
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700400#if defined(IPA_OFFLOAD)
Nirav Shahcbc6d722016-03-01 16:24:53 +0530401 if (!qdf_nbuf_ipa_owned_get(skb)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800402#endif
gbianec670c592016-11-24 11:21:30 +0800403
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700404 skb = hdd_skb_orphan(adapter, skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800405 if (!skb)
Jeff Johnsonedeff232015-11-11 17:19:42 -0800406 goto drop_pkt_accounting;
gbianec670c592016-11-24 11:21:30 +0800407
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700408#if defined(IPA_OFFLOAD)
Yun Park01deb2c2017-06-14 15:21:44 -0700409 } else {
410 /*
411 * Clear the IPA ownership after check it to avoid ipa_free_skb
412 * is called when Tx completed for intra-BSS Tx packets
413 */
414 qdf_nbuf_ipa_owned_clear(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800415 }
416#endif
417
Himanshu Agarwal53298d12017-02-20 19:14:17 +0530418 /*
419 * Add SKB to internal tracking table before further processing
420 * in WLAN driver.
421 */
422 qdf_net_buf_debug_acquire_skb(skb, __FILE__, __LINE__);
423
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700424 adapter->stats.tx_bytes += skb->len;
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700425 adapter->sta_info[STAId].tx_bytes += skb->len;
Mohit Khannab1dd1e82017-02-04 15:14:38 -0800426
Will Huang496b36c2017-07-11 16:38:50 +0800427 if (qdf_nbuf_is_tso(skb)) {
428 num_seg = qdf_nbuf_get_tso_num_seg(skb);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700429 adapter->stats.tx_packets += num_seg;
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700430 adapter->sta_info[STAId].tx_packets += num_seg;
Will Huang496b36c2017-07-11 16:38:50 +0800431 } else {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700432 ++adapter->stats.tx_packets;
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700433 adapter->sta_info[STAId].tx_packets++;
Will Huang496b36c2017-07-11 16:38:50 +0800434 }
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700435 adapter->sta_info[STAId].last_tx_rx_ts = qdf_system_ticks();
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800436
Nirav Shah5e74bb82016-07-20 16:01:27 +0530437 hdd_event_eapol_log(skb, QDF_TX);
Jeff Johnson1b780e42017-10-31 14:11:45 -0700438 qdf_dp_trace_log_pkt(adapter->session_id, skb, QDF_TX,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700439 QDF_TRACE_DEFAULT_PDEV_ID);
Nirav Shahcbc6d722016-03-01 16:24:53 +0530440 QDF_NBUF_CB_TX_PACKET_TRACK(skb) = QDF_NBUF_TX_PKT_DATA_TRACK;
441 QDF_NBUF_UPDATE_TX_PKT_COUNT(skb, QDF_NBUF_TX_PKT_HDD);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800442
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530443 qdf_dp_trace_set_track(skb, QDF_TX);
444 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_PTR_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700445 QDF_TRACE_DEFAULT_PDEV_ID, qdf_nbuf_data_addr(skb),
446 sizeof(qdf_nbuf_data(skb)),
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530447 QDF_TX));
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530448 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700449 QDF_TRACE_DEFAULT_PDEV_ID, (uint8_t *)skb->data,
450 qdf_nbuf_len(skb), QDF_TX));
Nirav Shahcbc6d722016-03-01 16:24:53 +0530451 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530452 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700453 QDF_TRACE_DEFAULT_PDEV_ID,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530454 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
455 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE), QDF_TX));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800456
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700457 if (adapter->tx_fn(adapter->txrx_vdev,
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800458 (qdf_nbuf_t) skb) != NULL) {
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530459 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800460 "%s: Failed to send packet to txrx for staid:%d",
461 __func__, STAId);
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700462 ++adapter->hdd_stats.tx_rx_stats.tx_dropped_ac[ac];
Himanshu Agarwal53298d12017-02-20 19:14:17 +0530463 goto drop_pkt_and_release_skb;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800464 }
Dustin Browne0024fa2016-10-14 16:29:21 -0700465 netif_trans_update(dev);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800466
467 return NETDEV_TX_OK;
468
Himanshu Agarwal53298d12017-02-20 19:14:17 +0530469drop_pkt_and_release_skb:
470 qdf_net_buf_debug_release_skb(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800471drop_pkt:
472
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530473 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_DROP_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700474 QDF_TRACE_DEFAULT_PDEV_ID, (uint8_t *)skb->data,
475 qdf_nbuf_len(skb), QDF_TX));
Nirav Shahcbc6d722016-03-01 16:24:53 +0530476 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530477 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_DROP_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700478 QDF_TRACE_DEFAULT_PDEV_ID,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530479 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
480 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE), QDF_TX));
Jeff Johnsonedeff232015-11-11 17:19:42 -0800481 kfree_skb(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800482
Jeff Johnsonedeff232015-11-11 17:19:42 -0800483drop_pkt_accounting:
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700484 ++adapter->stats.tx_dropped;
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700485 ++adapter->hdd_stats.tx_rx_stats.tx_dropped;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800486
487 return NETDEV_TX_OK;
488}
489
490/**
Mukul Sharmac4de4ef2016-09-12 15:39:00 +0530491 * hdd_softap_hard_start_xmit() - Wrapper function to protect
492 * __hdd_softap_hard_start_xmit from SSR
493 * @skb: pointer to OS packet
494 * @dev: pointer to net_device structure
495 *
496 * Function called by OS if any packet needs to transmit.
497 *
498 * Return: Always returns NETDEV_TX_OK
499 */
500int hdd_softap_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
501{
502 int ret;
503
504 cds_ssr_protect(__func__);
505 ret = __hdd_softap_hard_start_xmit(skb, dev);
506 cds_ssr_unprotect(__func__);
507
508 return ret;
509}
510
511/**
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800512 * __hdd_softap_tx_timeout() - TX timeout handler
513 * @dev: pointer to network device
514 *
515 * This function is registered as a netdev ndo_tx_timeout method, and
516 * is invoked by the kernel if the driver takes too long to transmit a
517 * frame.
518 *
519 * Return: None
520 */
521static void __hdd_softap_tx_timeout(struct net_device *dev)
522{
Jeff Johnson18dd7e12017-08-29 14:22:28 -0700523 struct hdd_adapter *adapter = WLAN_HDD_GET_PRIV_PTR(dev);
Jeff Johnsonf7726872017-08-28 11:38:31 -0700524 struct hdd_context *hdd_ctx;
Nirav Shah89223f72016-03-01 18:10:38 +0530525 struct netdev_queue *txq;
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +0530526 void *soc = cds_get_context(QDF_MODULE_ID_SOC);
Nirav Shah89223f72016-03-01 18:10:38 +0530527 int i;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800528
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530529 DPTRACE(qdf_dp_trace(NULL, QDF_DP_TRACE_HDD_SOFTAP_TX_TIMEOUT,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700530 QDF_TRACE_DEFAULT_PDEV_ID,
531 NULL, 0, QDF_TX));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800532 /* Getting here implies we disabled the TX queues for too
533 * long. Queues are disabled either because of disassociation
534 * or low resource scenarios. In case of disassociation it is
535 * ok to ignore this. But if associated, we have do possible
536 * recovery here
537 */
538 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Hanumanth Reddy Pothula2a8a7402017-07-03 14:06:11 +0530539 if (cds_is_driver_recovering() || cds_is_driver_in_bad_state()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530540 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800541 "%s: Recovery in Progress. Ignore!!!", __func__);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800542 return;
543 }
Nirav Shah89223f72016-03-01 18:10:38 +0530544
Dustin Browne0024fa2016-10-14 16:29:21 -0700545 TX_TIMEOUT_TRACE(dev, QDF_MODULE_ID_HDD_SAP_DATA);
Nirav Shah89223f72016-03-01 18:10:38 +0530546
547 for (i = 0; i < NUM_TX_QUEUES; i++) {
548 txq = netdev_get_tx_queue(dev, i);
549 QDF_TRACE(QDF_MODULE_ID_HDD_DATA,
Srinivas Girigowda028c4482017-03-09 18:52:02 -0800550 QDF_TRACE_LEVEL_DEBUG,
551 "Queue: %d status: %d txq->trans_start: %lu",
Nirav Shah89223f72016-03-01 18:10:38 +0530552 i, netif_tx_queue_stopped(txq), txq->trans_start);
553 }
554
Mohit Khannaca4173b2017-09-12 21:52:19 -0700555 wlan_hdd_display_netif_queue_history(hdd_ctx,
556 QDF_STATS_VERBOSITY_LEVEL_HIGH);
Leo Changfdb45c32016-10-28 11:09:23 -0700557 cdp_dump_flow_pool_info(cds_get_context(QDF_MODULE_ID_SOC));
Srinivas Girigowda028c4482017-03-09 18:52:02 -0800558 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_DEBUG,
Nirav Shah89223f72016-03-01 18:10:38 +0530559 "carrier state: %d", netif_carrier_ok(dev));
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +0530560
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700561 ++adapter->hdd_stats.tx_rx_stats.tx_timeout_cnt;
562 ++adapter->hdd_stats.tx_rx_stats.cont_txtimeout_cnt;
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +0530563
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700564 if (adapter->hdd_stats.tx_rx_stats.cont_txtimeout_cnt >
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +0530565 HDD_TX_STALL_THRESHOLD) {
566 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_ERROR,
567 "Detected data stall due to continuous TX timeouts");
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700568 adapter->hdd_stats.tx_rx_stats.cont_txtimeout_cnt = 0;
Poddar, Siddarth37033032017-10-11 15:47:40 +0530569 if (hdd_ctx->config->enable_data_stall_det)
570 cdp_post_data_stall_event(soc,
Sravan Kumar Kairam3a698312017-10-16 14:16:16 +0530571 DATA_STALL_LOG_INDICATOR_HOST_DRIVER,
572 DATA_STALL_LOG_HOST_SOFTAP_TX_TIMEOUT,
573 0xFF, 0xFF,
574 DATA_STALL_LOG_RECOVERY_TRIGGER_PDR);
575 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800576}
577
578/**
579 * hdd_softap_tx_timeout() - SSR wrapper for __hdd_softap_tx_timeout
580 * @dev: pointer to net_device
581 *
582 * Return: none
583 */
584void hdd_softap_tx_timeout(struct net_device *dev)
585{
586 cds_ssr_protect(__func__);
587 __hdd_softap_tx_timeout(dev);
588 cds_ssr_unprotect(__func__);
589}
590
591/**
592 * @hdd_softap_init_tx_rx() - Initialize Tx/RX module
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700593 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800594 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530595 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
596 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800597 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700598QDF_STATUS hdd_softap_init_tx_rx(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800599{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530600 QDF_STATUS status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800601
602 uint8_t STAId = 0;
603
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700604 qdf_mem_zero(&adapter->stats, sizeof(struct net_device_stats));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800605
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700606 spin_lock_init(&adapter->sta_info_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800607
608 for (STAId = 0; STAId < WLAN_MAX_STA_COUNT; STAId++) {
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700609 qdf_mem_zero(&adapter->sta_info[STAId],
Jeff Johnson82155922017-09-30 16:54:14 -0700610 sizeof(struct hdd_station_info));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800611 }
612
613 return status;
614}
615
616/**
617 * @hdd_softap_deinit_tx_rx() - Deinitialize Tx/RX module
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700618 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800619 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530620 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
621 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800622 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700623QDF_STATUS hdd_softap_deinit_tx_rx(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800624{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700625 if (adapter == NULL) {
626 hdd_err("Called with adapter = NULL.");
Dhanashri Atreed3bf512017-02-21 12:25:53 -0800627 return QDF_STATUS_E_FAILURE;
628 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800629
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700630 adapter->txrx_vdev = NULL;
631 adapter->tx_fn = NULL;
Dhanashri Atreed3bf512017-02-21 12:25:53 -0800632 hdd_info("Deregistering TX function hook !");
633 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800634}
635
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700636QDF_STATUS hdd_softap_init_tx_rx_sta(struct hdd_adapter *adapter,
Jeff Johnson349d9a92017-10-21 15:38:03 -0700637 uint8_t sta_id,
638 struct qdf_mac_addr *sta_mac)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800639{
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700640 spin_lock_bh(&adapter->sta_info_lock);
641 if (adapter->sta_info[sta_id].in_use) {
642 spin_unlock_bh(&adapter->sta_info_lock);
Jeff Johnson349d9a92017-10-21 15:38:03 -0700643 hdd_err("Reinit of in use station %d", sta_id);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530644 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800645 }
646
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700647 qdf_mem_zero(&adapter->sta_info[sta_id],
Jeff Johnson82155922017-09-30 16:54:14 -0700648 sizeof(struct hdd_station_info));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800649
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700650 adapter->sta_info[sta_id].in_use = true;
651 adapter->sta_info[sta_id].is_deauth_in_progress = false;
652 qdf_copy_macaddr(&adapter->sta_info[sta_id].sta_mac, sta_mac);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800653
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700654 spin_unlock_bh(&adapter->sta_info_lock);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530655 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800656}
657
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700658QDF_STATUS hdd_softap_deinit_tx_rx_sta(struct hdd_adapter *adapter,
Jeff Johnsonb157c722017-10-21 15:46:42 -0700659 uint8_t sta_id)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800660{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530661 QDF_STATUS status = QDF_STATUS_SUCCESS;
Jeff Johnson5c19ade2017-10-04 09:52:12 -0700662 struct hdd_hostapd_state *hostapd_state;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800663
Jeff Johnson5c19ade2017-10-04 09:52:12 -0700664 hostapd_state = WLAN_HDD_GET_HOSTAP_STATE_PTR(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800665
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700666 spin_lock_bh(&adapter->sta_info_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800667
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700668 if (false == adapter->sta_info[sta_id].in_use) {
669 spin_unlock_bh(&adapter->sta_info_lock);
Jeff Johnsonb157c722017-10-21 15:46:42 -0700670 hdd_err("Deinit station not inited %d", sta_id);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530671 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800672 }
673
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700674 adapter->sta_info[sta_id].in_use = false;
675 adapter->sta_info[sta_id].is_deauth_in_progress = false;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800676
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700677 spin_unlock_bh(&adapter->sta_info_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800678 return status;
679}
680
681/**
682 * hdd_softap_rx_packet_cbk() - Receive packet handler
Dhanashri Atre182b0272016-02-17 15:35:07 -0800683 * @context: pointer to HDD context
Nirav Shahcbc6d722016-03-01 16:24:53 +0530684 * @rxBuf: pointer to rx qdf_nbuf
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800685 *
686 * Receive callback registered with TL. TL will call this to notify
687 * the HDD when one or more packets were received for a registered
688 * STA.
689 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530690 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
691 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800692 */
Dhanashri Atre182b0272016-02-17 15:35:07 -0800693QDF_STATUS hdd_softap_rx_packet_cbk(void *context, qdf_nbuf_t rxBuf)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800694{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700695 struct hdd_adapter *adapter = NULL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800696 int rxstat;
697 unsigned int cpu_index;
698 struct sk_buff *skb = NULL;
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800699 struct sk_buff *next = NULL;
Jeff Johnson92402872017-09-03 09:25:37 -0700700 struct hdd_context *hdd_ctx = NULL;
Will Huang496b36c2017-07-11 16:38:50 +0800701 struct qdf_mac_addr src_mac;
702 uint8_t staid;
Mohit Khannaf8f96822017-05-17 17:11:59 -0700703 bool proto_pkt_logged = false;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800704
705 /* Sanity check on inputs */
Dhanashri Atre182b0272016-02-17 15:35:07 -0800706 if (unlikely((NULL == context) || (NULL == rxBuf))) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530707 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800708 "%s: Null params being passed", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530709 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800710 }
711
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700712 adapter = (struct hdd_adapter *)context;
713 if (unlikely(WLAN_HDD_ADAPTER_MAGIC != adapter->magic)) {
Srinivas Girigowda028c4482017-03-09 18:52:02 -0800714 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_ERROR,
Dhanashri Atre182b0272016-02-17 15:35:07 -0800715 "Magic cookie(%x) for adapter sanity verification is invalid",
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700716 adapter->magic);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530717 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800718 }
719
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700720 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Jeff Johnson92402872017-09-03 09:25:37 -0700721 if (unlikely(NULL == hdd_ctx)) {
Dhanashri Atre182b0272016-02-17 15:35:07 -0800722 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
723 "%s: HDD context is Null", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530724 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800725 }
726
727 /* walk the chain until all are processed */
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800728 next = (struct sk_buff *)rxBuf;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800729
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800730 while (next) {
731 skb = next;
732 next = skb->next;
Dhanashri Atre63d98022017-01-24 18:22:09 -0800733 skb->next = NULL;
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800734
psimha884025c2017-08-01 15:07:32 -0700735#ifdef QCA_WIFI_QCA6290 /* Debug code, remove later */
Yun Parke4239802018-01-09 11:01:40 -0800736 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_DEBUG,
Jeff Johnson36e74c42017-09-18 08:15:42 -0700737 "%s: skb %pK skb->len %d\n", __func__, skb, skb->len);
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800738#endif
739
Dhanashri Atre63d98022017-01-24 18:22:09 -0800740 hdd_softap_dump_sk_buff(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800741
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700742 skb->dev = adapter->dev;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800743
Dhanashri Atre63d98022017-01-24 18:22:09 -0800744 if (unlikely(skb->dev == NULL)) {
745
746 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
747 "%s: ERROR!!Invalid netdevice", __func__);
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800748 continue;
Dhanashri Atre63d98022017-01-24 18:22:09 -0800749 }
750 cpu_index = wlan_hdd_get_cpu();
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700751 ++adapter->hdd_stats.tx_rx_stats.rx_packets[cpu_index];
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700752 ++adapter->stats.rx_packets;
753 adapter->stats.rx_bytes += skb->len;
Dhanashri Atre63d98022017-01-24 18:22:09 -0800754
Will Huang496b36c2017-07-11 16:38:50 +0800755 qdf_mem_copy(&src_mac, skb->data + QDF_NBUF_SRC_MAC_OFFSET,
756 sizeof(src_mac));
757 if (QDF_STATUS_SUCCESS ==
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700758 hdd_softap_get_sta_id(adapter, &src_mac, &staid)) {
Will Huang496b36c2017-07-11 16:38:50 +0800759 if (staid < WLAN_MAX_STA_COUNT) {
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700760 adapter->sta_info[staid].rx_packets++;
761 adapter->sta_info[staid].rx_bytes += skb->len;
762 adapter->sta_info[staid].last_tx_rx_ts =
Will Huang496b36c2017-07-11 16:38:50 +0800763 qdf_system_ticks();
764 }
765 }
Dhanashri Atre63d98022017-01-24 18:22:09 -0800766 hdd_event_eapol_log(skb, QDF_RX);
Jeff Johnson1b780e42017-10-31 14:11:45 -0700767 proto_pkt_logged = qdf_dp_trace_log_pkt(adapter->session_id,
Mohit Khannaf8f96822017-05-17 17:11:59 -0700768 skb, QDF_RX,
769 QDF_TRACE_DEFAULT_PDEV_ID);
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530770 DPTRACE(qdf_dp_trace(skb,
Dhanashri Atre63d98022017-01-24 18:22:09 -0800771 QDF_DP_TRACE_RX_HDD_PACKET_PTR_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700772 QDF_TRACE_DEFAULT_PDEV_ID,
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530773 qdf_nbuf_data_addr(skb),
774 sizeof(qdf_nbuf_data(skb)), QDF_RX));
Mohit Khannaf8f96822017-05-17 17:11:59 -0700775 if (!proto_pkt_logged) {
776 DPTRACE(qdf_dp_trace(skb,
777 QDF_DP_TRACE_HDD_RX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700778 QDF_TRACE_DEFAULT_PDEV_ID,
779 (uint8_t *)skb->data, qdf_nbuf_len(skb),
780 QDF_RX));
Mohit Khannaf8f96822017-05-17 17:11:59 -0700781 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
782 DPTRACE(qdf_dp_trace(skb,
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530783 QDF_DP_TRACE_HDD_RX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700784 QDF_TRACE_DEFAULT_PDEV_ID,
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530785 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
786 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE),
787 QDF_RX));
Mohit Khannaf8f96822017-05-17 17:11:59 -0700788 }
Dhanashri Atre63d98022017-01-24 18:22:09 -0800789
790 skb->protocol = eth_type_trans(skb, skb->dev);
791
792 /* hold configurable wakelock for unicast traffic */
Jeff Johnson92402872017-09-03 09:25:37 -0700793 if (hdd_ctx->config->rx_wakelock_timeout &&
Dhanashri Atre63d98022017-01-24 18:22:09 -0800794 skb->pkt_type != PACKET_BROADCAST &&
795 skb->pkt_type != PACKET_MULTICAST) {
Jeff Johnson92402872017-09-03 09:25:37 -0700796 cds_host_diag_log_work(&hdd_ctx->rx_wake_lock,
797 hdd_ctx->config->rx_wakelock_timeout,
Dhanashri Atre63d98022017-01-24 18:22:09 -0800798 WIFI_POWER_EVENT_WAKELOCK_HOLD_RX);
Jeff Johnson92402872017-09-03 09:25:37 -0700799 qdf_wake_lock_timeout_acquire(&hdd_ctx->rx_wake_lock,
800 hdd_ctx->config->
Dhanashri Atre63d98022017-01-24 18:22:09 -0800801 rx_wakelock_timeout);
802 }
803
804 /* Remove SKB from internal tracking table before submitting
805 * it to stack
806 */
Manjunathappa Prakash17c07bd2017-04-27 21:24:28 -0700807 qdf_net_buf_debug_release_skb(skb);
Dhanashri Atre63d98022017-01-24 18:22:09 -0800808 if (hdd_napi_enabled(HDD_NAPI_ANY) &&
Jeff Johnsone2ba3cd2017-10-30 20:02:09 -0700809 !hdd_ctx->enable_rxthread)
Dhanashri Atre63d98022017-01-24 18:22:09 -0800810 rxstat = netif_receive_skb(skb);
811 else
812 rxstat = netif_rx_ni(skb);
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700813 if (NET_RX_SUCCESS == rxstat)
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700814 ++adapter->hdd_stats.tx_rx_stats.rx_delivered[cpu_index];
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700815 else
Jeff Johnson6ced42c2017-10-20 12:48:11 -0700816 ++adapter->hdd_stats.tx_rx_stats.rx_refused[cpu_index];
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800817 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800818
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530819 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800820}
821
822/**
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700823 * hdd_softap_deregister_sta(struct hdd_adapter *adapter, uint8_t staId)
824 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800825 * @staId: Station ID to deregister
826 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530827 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800828 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700829QDF_STATUS hdd_softap_deregister_sta(struct hdd_adapter *adapter,
Jeff Johnson18dd7e12017-08-29 14:22:28 -0700830 uint8_t staId)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800831{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530832 QDF_STATUS qdf_status = QDF_STATUS_SUCCESS;
Jeff Johnson92402872017-09-03 09:25:37 -0700833 struct hdd_context *hdd_ctx;
Abhishek Singh07c627e2017-03-20 17:56:34 +0530834 int ret;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800835
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700836 if (NULL == adapter) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800837 hdd_err("NULL adapter");
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530838 return QDF_STATUS_E_INVAL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800839 }
840
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700841 if (WLAN_HDD_ADAPTER_MAGIC != adapter->magic) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800842 hdd_err("Invalid adapter magic");
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530843 return QDF_STATUS_E_INVAL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800844 }
845
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700846 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800847 /* Clear station in TL and then update HDD data
848 * structures. This helps to block RX frames from other
849 * station to this station.
850 */
Krishna Kumaar Natarajane58b4092017-01-25 15:47:35 -0800851 qdf_status = cdp_clear_peer(cds_get_context(QDF_MODULE_ID_SOC),
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800852 (struct cdp_pdev *)cds_get_context(QDF_MODULE_ID_TXRX),
853 staId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530854 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Krishna Kumaar Natarajane58b4092017-01-25 15:47:35 -0800855 hdd_err("cdp_clear_peer failed for staID %d, Status=%d [0x%08X]",
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800856 staId, qdf_status, qdf_status);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800857 }
858
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700859 ret = hdd_objmgr_remove_peer_object(adapter->hdd_vdev,
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700860 adapter->sta_info[staId].
Jeff Johnsonf2356512017-10-21 16:04:12 -0700861 sta_mac.bytes);
Abhishek Singh07c627e2017-03-20 17:56:34 +0530862 if (ret)
863 hdd_err("Peer obj %pM delete fails",
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700864 adapter->sta_info[staId].sta_mac.bytes);
Abhishek Singh07c627e2017-03-20 17:56:34 +0530865
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700866 if (adapter->sta_info[staId].in_use) {
Will Huangac3fd9a2017-11-01 16:18:12 +0800867 if (hdd_ipa_uc_is_enabled(hdd_ctx)) {
868 hdd_ipa_wlan_evt(adapter,
869 adapter->sta_info[staId].sta_id,
870 HDD_IPA_CLIENT_DISCONNECT,
871 adapter->sta_info[staId].sta_mac.
872 bytes);
873 }
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700874 spin_lock_bh(&adapter->sta_info_lock);
875 qdf_mem_zero(&adapter->sta_info[staId],
Jeff Johnson82155922017-09-30 16:54:14 -0700876 sizeof(struct hdd_station_info));
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700877 spin_unlock_bh(&adapter->sta_info_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800878 }
Abhishek Singh07c627e2017-03-20 17:56:34 +0530879
Jeff Johnson92402872017-09-03 09:25:37 -0700880 hdd_ctx->sta_to_adapter[staId] = NULL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800881
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530882 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800883}
884
885/**
886 * hdd_softap_register_sta() - Register a SoftAP STA
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700887 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800888 * @fAuthRequired: is additional authentication required?
889 * @fPrivacyBit: should 802.11 privacy bit be set?
890 * @staId: station ID assigned to this station
891 * @ucastSig: unicast security signature
892 * @bcastSig: broadcast security signature
893 * @pPeerMacAddress: station MAC address
894 * @fWmmEnabled: is WMM enabled for this STA?
895 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530896 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800897 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700898QDF_STATUS hdd_softap_register_sta(struct hdd_adapter *adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800899 bool fAuthRequired,
900 bool fPrivacyBit,
901 uint8_t staId,
902 uint8_t ucastSig,
903 uint8_t bcastSig,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530904 struct qdf_mac_addr *pPeerMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800905 bool fWmmEnabled)
906{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530907 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800908 struct ol_txrx_desc_type staDesc = { 0 };
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700909 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Dhanashri Atre182b0272016-02-17 15:35:07 -0800910 struct ol_txrx_ops txrx_ops;
Leo Changfdb45c32016-10-28 11:09:23 -0700911 void *soc = cds_get_context(QDF_MODULE_ID_SOC);
912 void *pdev = cds_get_context(QDF_MODULE_ID_TXRX);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800913
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800914 hdd_info("STA:%u, Auth:%u, Priv:%u, WMM:%u",
915 staId, fAuthRequired, fPrivacyBit, fWmmEnabled);
916
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800917 /*
918 * Clean up old entry if it is not cleaned up properly
919 */
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700920 if (adapter->sta_info[staId].in_use) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800921 hdd_info("clean up old entry for STA %d", staId);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700922 hdd_softap_deregister_sta(adapter, staId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800923 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800924
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800925 /* Get the Station ID from the one saved during the assocation. */
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800926 staDesc.sta_id = staId;
927
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700928 /* Save the adapter Pointer for this staId */
929 hdd_ctx->sta_to_adapter[staId] = adapter;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800930
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530931 qdf_status =
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700932 hdd_softap_init_tx_rx_sta(adapter, staId,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800933 pPeerMacAddress);
934
935 staDesc.is_qos_enabled = fWmmEnabled;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800936
Dhanashri Atre182b0272016-02-17 15:35:07 -0800937 /* Register the vdev transmit and receive functions */
938 qdf_mem_zero(&txrx_ops, sizeof(txrx_ops));
939 txrx_ops.rx.rx = hdd_softap_rx_packet_cbk;
Leo Changfdb45c32016-10-28 11:09:23 -0700940 cdp_vdev_register(soc,
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800941 (struct cdp_vdev *)cdp_get_vdev_from_vdev_id(soc,
Jeff Johnson1b780e42017-10-31 14:11:45 -0700942 (struct cdp_pdev *)pdev, adapter->session_id),
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700943 adapter, &txrx_ops);
944 adapter->txrx_vdev = (void *)cdp_get_vdev_from_vdev_id(soc,
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800945 (struct cdp_pdev *)pdev,
Jeff Johnson1b780e42017-10-31 14:11:45 -0700946 adapter->session_id);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700947 adapter->tx_fn = txrx_ops.tx.tx;
Dhanashri Atre182b0272016-02-17 15:35:07 -0800948
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800949 qdf_status = cdp_peer_register(soc,
950 (struct cdp_pdev *)pdev, &staDesc);
Dhanashri Atre50141c52016-04-07 13:15:29 -0700951 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800952 hdd_err("cdp_peer_register() failed to register. Status = %d [0x%08X]",
953 qdf_status, qdf_status);
Dhanashri Atre50141c52016-04-07 13:15:29 -0700954 return qdf_status;
955 }
956
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800957 /* if ( WPA ), tell TL to go to 'connected' and after keys come to the
958 * driver then go to 'authenticated'. For all other authentication
959 * types (those that do not require upper layer authentication) we can
960 * put TL directly into 'authenticated' state
961 */
962
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700963 adapter->sta_info[staId].sta_id = staId;
964 adapter->sta_info[staId].is_qos_enabled = fWmmEnabled;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800965
966 if (!fAuthRequired) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800967 hdd_info("open/shared auth StaId= %d. Changing TL state to AUTHENTICATED at Join time",
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700968 adapter->sta_info[staId].sta_id);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800969
970 /* Connections that do not need Upper layer auth,
971 * transition TL directly to 'Authenticated' state.
972 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700973 qdf_status = hdd_change_peer_state(adapter, staDesc.sta_id,
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800974 OL_TXRX_PEER_STATE_AUTH, false);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800975
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700976 adapter->sta_info[staId].peer_state = OL_TXRX_PEER_STATE_AUTH;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800977 } else {
978
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800979 hdd_info("ULA auth StaId= %d. Changing TL state to CONNECTED at Join time",
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700980 adapter->sta_info[staId].sta_id);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800981
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700982 qdf_status = hdd_change_peer_state(adapter, staDesc.sta_id,
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800983 OL_TXRX_PEER_STATE_CONN, false);
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -0700984 adapter->sta_info[staId].peer_state = OL_TXRX_PEER_STATE_CONN;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800985 }
986
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700987 hdd_debug("Enabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700988 wlan_hdd_netif_queue_control(adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800989 WLAN_START_ALL_NETIF_QUEUE_N_CARRIER,
990 WLAN_CONTROL_PATH);
991
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530992 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800993}
994
995/**
996 * hdd_softap_register_bc_sta() - Register the SoftAP broadcast STA
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700997 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800998 * @fPrivacyBit: should 802.11 privacy bit be set?
999 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301000 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001001 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001002QDF_STATUS hdd_softap_register_bc_sta(struct hdd_adapter *adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001003 bool fPrivacyBit)
1004{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301005 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001006 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Dustin Brownce5b3d32018-01-17 15:07:38 -08001007 struct qdf_mac_addr broadcastMacAddr = QDF_MAC_ADDR_BCAST_INIT;
Jeff Johnson9bf24972017-10-04 09:26:58 -07001008 struct hdd_ap_ctx *ap_ctx;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001009
Jeff Johnson9bf24972017-10-04 09:26:58 -07001010 ap_ctx = WLAN_HDD_GET_AP_CTX_PTR(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001011
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001012 hdd_ctx->sta_to_adapter[WLAN_RX_BCMC_STA_ID] = adapter;
Jeff Johnson42518cf2017-10-26 13:33:29 -07001013 hdd_ctx->sta_to_adapter[ap_ctx->broadcast_sta_id] = adapter;
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301014 qdf_status =
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001015 hdd_softap_register_sta(adapter, false, fPrivacyBit,
Jeff Johnson42518cf2017-10-26 13:33:29 -07001016 ap_ctx->broadcast_sta_id,
1017 0, 1, &broadcastMacAddr, 0);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001018
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301019 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001020}
1021
1022/**
1023 * hdd_softap_deregister_bc_sta() - Deregister the SoftAP broadcast STA
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001024 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001025 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301026 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001027 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001028QDF_STATUS hdd_softap_deregister_bc_sta(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001029{
Jeff Johnson42518cf2017-10-26 13:33:29 -07001030 struct hdd_ap_ctx *ap_ctx;
1031
1032 ap_ctx = WLAN_HDD_GET_AP_CTX_PTR(adapter);
1033 return hdd_softap_deregister_sta(adapter, ap_ctx->broadcast_sta_id);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001034}
1035
1036/**
1037 * hdd_softap_stop_bss() - Stop the BSS
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001038 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001039 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301040 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001041 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001042QDF_STATUS hdd_softap_stop_bss(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001043{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301044 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001045 uint8_t staId = 0;
Jeff Johnson92402872017-09-03 09:25:37 -07001046 struct hdd_context *hdd_ctx;
Srinivas Girigowdae806a822017-03-25 11:25:30 -07001047
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001048 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001049
Rajeev Kumar07bbe122017-10-24 13:06:05 -07001050 /* This is stop bss callback running in scheduler thread so do not
1051 * driver unload in progress check otherwise it can lead to peer
1052 * object leak
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001053 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001054 qdf_status = hdd_softap_deregister_bc_sta(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001055
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301056 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Jeff Johnson42518cf2017-10-26 13:33:29 -07001057 struct hdd_ap_ctx *ap_ctx;
1058
1059 ap_ctx = WLAN_HDD_GET_AP_CTX_PTR(adapter);
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001060 hdd_err("Failed to deregister BC sta Id %d",
Jeff Johnson42518cf2017-10-26 13:33:29 -07001061 ap_ctx->broadcast_sta_id);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001062 }
1063
1064 for (staId = 0; staId < WLAN_MAX_STA_COUNT; staId++) {
1065 /* This excludes BC sta as it is already deregistered */
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -07001066 if (adapter->sta_info[staId].in_use) {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001067 qdf_status = hdd_softap_deregister_sta(adapter, staId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301068 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001069 hdd_err("Failed to deregister sta Id %d",
1070 staId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001071 }
1072 }
1073 }
Mukul Sharmaecf8e092017-12-19 22:36:31 +05301074
1075 /* Mark the indoor channel (passive) to enable */
1076 if (hdd_ctx->config->force_ssc_disable_indoor_channel) {
1077 hdd_update_indoor_channel(hdd_ctx, false);
1078 sme_update_channel_list(hdd_ctx->hHal);
1079 }
1080
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301081 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001082}
1083
1084/**
1085 * hdd_softap_change_sta_state() - Change the state of a SoftAP station
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001086 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001087 * @pDestMacAddress: MAC address of the station
1088 * @state: new state of the station
1089 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301090 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001091 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001092QDF_STATUS hdd_softap_change_sta_state(struct hdd_adapter *adapter,
Anurag Chouhan6d760662016-02-20 16:05:43 +05301093 struct qdf_mac_addr *pDestMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001094 enum ol_txrx_peer_state state)
1095{
Jeff Johnson4c0ab7b2017-10-21 16:13:09 -07001096 uint8_t sta_id = WLAN_MAX_STA_COUNT;
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001097 QDF_STATUS qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001098
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001099 ENTER_DEV(adapter->dev);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001100
Jeff Johnson4c0ab7b2017-10-21 16:13:09 -07001101 qdf_status = hdd_softap_get_sta_id(adapter, pDestMacAddress, &sta_id);
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001102 if (QDF_STATUS_SUCCESS != qdf_status) {
1103 hdd_err("Failed to find right station");
1104 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001105 }
1106
1107 if (false ==
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -07001108 qdf_is_macaddr_equal(&adapter->sta_info[sta_id].sta_mac,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001109 pDestMacAddress)) {
Jeff Johnson4c0ab7b2017-10-21 16:13:09 -07001110 hdd_err("Station %u MAC address not matching", sta_id);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301111 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001112 }
1113
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301114 qdf_status =
Jeff Johnson4c0ab7b2017-10-21 16:13:09 -07001115 hdd_change_peer_state(adapter, sta_id, state, false);
1116 hdd_info("Station %u changed to state %d", sta_id, state);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001117
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301118 if (QDF_STATUS_SUCCESS == qdf_status) {
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -07001119 adapter->sta_info[sta_id].peer_state =
Dhanashri Atreb08959a2016-03-01 17:28:03 -08001120 OL_TXRX_PEER_STATE_AUTH;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001121 p2p_peer_authorized(adapter->hdd_vdev, pDestMacAddress->bytes);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001122 }
1123
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001124 EXIT();
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301125 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001126}
1127
1128/*
1129 * hdd_softap_get_sta_id() - Find station ID from MAC address
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001130 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001131 * @pDestMacAddress: MAC address of the destination
1132 * @staId: Station ID associated with the MAC address
1133 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301134 * Return: QDF_STATUS_SUCCESS if a match was found, in which case
1135 * staId is populated, QDF_STATUS_E_FAILURE if a match is
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001136 * not found
1137 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001138QDF_STATUS hdd_softap_get_sta_id(struct hdd_adapter *adapter,
Anurag Chouhan6d760662016-02-20 16:05:43 +05301139 struct qdf_mac_addr *pMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001140 uint8_t *staId)
1141{
1142 uint8_t i;
1143
1144 for (i = 0; i < WLAN_MAX_STA_COUNT; i++) {
Anurag Chouhan600c3a02016-03-01 10:33:54 +05301145 if (!qdf_mem_cmp
Jeff Johnsonbb8b56a2017-10-23 07:02:36 -07001146 (&adapter->sta_info[i].sta_mac, pMacAddress,
1147 QDF_MAC_ADDR_SIZE) && adapter->sta_info[i].in_use) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001148 *staId = i;
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301149 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001150 }
1151 }
1152
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301153 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001154}