blob: 7e3cf012c4589169b303d8a847bef512f7d3758e [file] [log] [blame]
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001/*
Jeff Johnson4fca3b52017-01-12 08:44:18 -08002 * Copyright (c) 2012-2017 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>
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -080048#include <cdp_txrx_handle.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#ifdef IPA_OFFLOAD
52#include <wlan_hdd_ipa.h>
53#endif
54
55/* Preprocessor definitions and constants */
56#undef QCA_HDD_SAP_DUMP_SK_BUFF
57
58/* Type declarations */
59
60/* Function definitions and documenation */
61#ifdef QCA_HDD_SAP_DUMP_SK_BUFF
62/**
63 * hdd_softap_dump_sk_buff() - Dump an skb
64 * @skb: skb to dump
65 *
66 * Return: None
67 */
68static void hdd_softap_dump_sk_buff(struct sk_buff *skb)
69{
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053070 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Jeff Johnson36e74c42017-09-18 08:15:42 -070071 "%s: head = %pK ", __func__, skb->head);
Srinivas Girigowdae806a822017-03-25 11:25:30 -070072 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Jeff Johnson36e74c42017-09-18 08:15:42 -070073 "%s: tail = %pK ", __func__, skb->tail);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053074 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Jeff Johnson36e74c42017-09-18 08:15:42 -070075 "%s: end = %pK ", __func__, skb->end);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053076 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080077 "%s: len = %d ", __func__, skb->len);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053078 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080079 "%s: data_len = %d ", __func__, skb->data_len);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053080 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080081 "%s: mac_len = %d", __func__, skb->mac_len);
82
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053083 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080084 "0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x ", skb->data[0],
85 skb->data[1], skb->data[2], skb->data[3], skb->data[4],
86 skb->data[5], skb->data[6], skb->data[7]);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053087 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080088 "0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x", skb->data[8],
89 skb->data[9], skb->data[10], skb->data[11], skb->data[12],
90 skb->data[13], skb->data[14], skb->data[15]);
91}
92#else
93static void hdd_softap_dump_sk_buff(struct sk_buff *skb)
94{
95}
96#endif
97
98#ifdef QCA_LL_LEGACY_TX_FLOW_CONTROL
99/**
100 * hdd_softap_tx_resume_timer_expired_handler() - TX Q resume timer handler
101 * @adapter_context: pointer to vdev adapter
102 *
103 * TX Q resume timer handler for SAP and P2P GO interface. If Blocked
104 * OS Q is not resumed during timeout period, to prevent permanent
105 * stall, resume OS Q forcefully for SAP and P2P GO interface.
106 *
107 * Return: None
108 */
109void hdd_softap_tx_resume_timer_expired_handler(void *adapter_context)
110{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700111 struct hdd_adapter *adapter = (struct hdd_adapter *) adapter_context;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800112
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700113 if (!adapter) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800114 hdd_err("NULL adapter");
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800115 return;
116 }
117
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700118 hdd_debug("Enabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700119 wlan_hdd_netif_queue_control(adapter, WLAN_WAKE_ALL_NETIF_QUEUE,
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800120 WLAN_CONTROL_PATH);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800121}
122
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530123#if defined(CONFIG_PER_VDEV_TX_DESC_POOL)
124
125/**
126 * hdd_softap_tx_resume_false() - Resume OS TX Q false leads to queue disabling
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700127 * @adapter: pointer to hdd adapter
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530128 * @tx_resume: TX Q resume trigger
129 *
130 *
131 * Return: None
132 */
133static void
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700134hdd_softap_tx_resume_false(struct hdd_adapter *adapter, bool tx_resume)
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530135{
136 if (true == tx_resume)
137 return;
138
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700139 hdd_debug("Disabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700140 wlan_hdd_netif_queue_control(adapter, WLAN_STOP_ALL_NETIF_QUEUE,
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530141 WLAN_DATA_FLOW_CONTROL);
142
143 if (QDF_TIMER_STATE_STOPPED ==
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700144 qdf_mc_timer_get_current_state(&adapter->
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530145 tx_flow_control_timer)) {
146 QDF_STATUS status;
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700147
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700148 status = qdf_mc_timer_start(&adapter->tx_flow_control_timer,
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530149 WLAN_SAP_HDD_TX_FLOW_CONTROL_OS_Q_BLOCK_TIME);
150
151 if (!QDF_IS_STATUS_SUCCESS(status))
152 hdd_err("Failed to start tx_flow_control_timer");
153 else
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700154 adapter->hdd_stats.hddTxRxStats.txflow_timer_cnt++;
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530155 }
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530156}
157#else
158
159static inline void
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700160hdd_softap_tx_resume_false(struct hdd_adapter *adapter, bool tx_resume)
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530161{
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530162}
163#endif
164
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800165/**
166 * hdd_softap_tx_resume_cb() - Resume OS TX Q.
167 * @adapter_context: pointer to vdev apdapter
168 * @tx_resume: TX Q resume trigger
169 *
170 * Q was stopped due to WLAN TX path low resource condition
171 *
172 * Return: None
173 */
174void hdd_softap_tx_resume_cb(void *adapter_context, bool tx_resume)
175{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700176 struct hdd_adapter *adapter = (struct hdd_adapter *) adapter_context;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800177
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700178 if (!adapter) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800179 hdd_err("NULL adapter");
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800180 return;
181 }
182
183 /* Resume TX */
184 if (true == tx_resume) {
Anurag Chouhan210db072016-02-22 18:42:15 +0530185 if (QDF_TIMER_STATE_STOPPED !=
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700186 qdf_mc_timer_get_current_state(&adapter->
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800187 tx_flow_control_timer)) {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700188 qdf_mc_timer_stop(&adapter->tx_flow_control_timer);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800189 }
190
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -0700191 hdd_debug("Enabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700192 wlan_hdd_netif_queue_control(adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800193 WLAN_WAKE_ALL_NETIF_QUEUE,
194 WLAN_DATA_FLOW_CONTROL);
195 }
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700196 hdd_softap_tx_resume_false(adapter, tx_resume);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800197}
gbianec670c592016-11-24 11:21:30 +0800198
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700199static inline struct sk_buff *hdd_skb_orphan(struct hdd_adapter *adapter,
gbianec670c592016-11-24 11:21:30 +0800200 struct sk_buff *skb)
201{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700202 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
tfyubdf453e2017-09-27 13:34:30 +0800203 int need_orphan = 0;
204
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700205 if (adapter->tx_flow_low_watermark > 0) {
tfyubdf453e2017-09-27 13:34:30 +0800206#if (LINUX_VERSION_CODE > KERNEL_VERSION(3, 19, 0))
207 /*
208 * The TCP TX throttling logic is changed a little after
209 * 3.19-rc1 kernel, the TCP sending limit will be smaller,
210 * which will throttle the TCP packets to the host driver.
211 * The TCP UP LINK throughput will drop heavily. In order to
212 * fix this issue, need to orphan the socket buffer asap, which
213 * will call skb's destructor to notify the TCP stack that the
214 * SKB buffer is unowned. And then the TCP stack will pump more
215 * packets to host driver.
216 *
217 * The TX packets might be dropped for UDP case in the iperf
218 * testing. So need to be protected by follow control.
219 */
220 need_orphan = 1;
221#else
222 if (hdd_ctx->config->tx_orphan_enable)
223 need_orphan = 1;
224#endif
tfyu5f01db22017-10-11 13:51:04 +0800225 } else if (hdd_ctx->config->tx_orphan_enable) {
226 if (qdf_nbuf_is_ipv4_tcp_pkt(skb) ||
Tiger Yu438c6482017-10-13 11:07:00 +0800227 qdf_nbuf_is_ipv6_tcp_pkt(skb))
tfyu5f01db22017-10-11 13:51:04 +0800228 need_orphan = 1;
tfyubdf453e2017-09-27 13:34:30 +0800229 }
230
tfyu5f01db22017-10-11 13:51:04 +0800231 if (need_orphan) {
gbianec670c592016-11-24 11:21:30 +0800232 skb_orphan(skb);
tfyu5f01db22017-10-11 13:51:04 +0800233 ++adapter->hdd_stats.hddTxRxStats.txXmitOrphaned;
Tiger Yu438c6482017-10-13 11:07:00 +0800234 } else
gbianec670c592016-11-24 11:21:30 +0800235 skb = skb_unshare(skb, GFP_ATOMIC);
gbianec670c592016-11-24 11:21:30 +0800236
237 return skb;
238}
239
240#else
Mohit Khannad0b63f52017-02-18 18:05:52 -0800241/**
242 * hdd_skb_orphan() - skb_unshare a cloned packed else skb_orphan
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700243 * @adapter: pointer to HDD adapter
Mohit Khannad0b63f52017-02-18 18:05:52 -0800244 * @skb: pointer to skb data packet
245 *
246 * Return: pointer to skb structure
247 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700248static inline struct sk_buff *hdd_skb_orphan(struct hdd_adapter *adapter,
Mohit Khannad0b63f52017-02-18 18:05:52 -0800249 struct sk_buff *skb) {
gbianec670c592016-11-24 11:21:30 +0800250
Mohit Khannad0b63f52017-02-18 18:05:52 -0800251 struct sk_buff *nskb;
tfyubdf453e2017-09-27 13:34:30 +0800252#if (LINUX_VERSION_CODE > KERNEL_VERSION(3, 19, 0))
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700253 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
tfyubdf453e2017-09-27 13:34:30 +0800254#endif
Mohit Khannad0b63f52017-02-18 18:05:52 -0800255
Manjunathappa Prakashdab74fa2017-06-19 12:11:03 -0700256 nskb = skb_unshare(skb, GFP_ATOMIC);
tfyubdf453e2017-09-27 13:34:30 +0800257#if (LINUX_VERSION_CODE > KERNEL_VERSION(3, 19, 0))
Manjunathappa Prakashdab74fa2017-06-19 12:11:03 -0700258 if (unlikely(hdd_ctx->config->tx_orphan_enable) && (nskb == skb)) {
Mohit Khannad0b63f52017-02-18 18:05:52 -0800259 /*
260 * For UDP packets we want to orphan the packet to allow the app
261 * to send more packets. The flow would ultimately be controlled
262 * by the limited number of tx descriptors for the vdev.
263 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700264 ++adapter->hdd_stats.hddTxRxStats.txXmitOrphaned;
Mohit Khannad0b63f52017-02-18 18:05:52 -0800265 skb_orphan(skb);
266 }
tfyubdf453e2017-09-27 13:34:30 +0800267#endif
Mohit Khannad0b63f52017-02-18 18:05:52 -0800268 return nskb;
gbianec670c592016-11-24 11:21:30 +0800269}
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800270#endif /* QCA_LL_LEGACY_TX_FLOW_CONTROL */
271
272/**
Mukul Sharmac4de4ef2016-09-12 15:39:00 +0530273 * __hdd_softap_hard_start_xmit() - Transmit a frame
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800274 * @skb: pointer to OS packet (sk_buff)
275 * @dev: pointer to network device
276 *
277 * Function registered with the Linux OS for transmitting
278 * packets. This version of the function directly passes
279 * the packet to Transport Layer.
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530280 * In case of any packet drop or error, log the error with
281 * INFO HIGH/LOW/MEDIUM to avoid excessive logging in kmsg.
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800282 *
283 * Return: Always returns NETDEV_TX_OK
284 */
Jeff Johnson6376abe2016-10-05 16:24:56 -0700285static int __hdd_softap_hard_start_xmit(struct sk_buff *skb,
286 struct net_device *dev)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800287{
288 sme_ac_enum_type ac = SME_AC_BE;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700289 struct hdd_adapter *adapter = (struct hdd_adapter *) netdev_priv(dev);
Jeff Johnson9bf24972017-10-04 09:26:58 -0700290 struct hdd_ap_ctx *ap_ctx = WLAN_HDD_GET_AP_CTX_PTR(adapter);
Anurag Chouhan6d760662016-02-20 16:05:43 +0530291 struct qdf_mac_addr *pDestMacAddress;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800292 uint8_t STAId;
Will Huang496b36c2017-07-11 16:38:50 +0800293 uint32_t num_seg;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800294
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700295 ++adapter->hdd_stats.hddTxRxStats.txXmitCalled;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800296 /* Prevent this function from being called during SSR since TL
297 * context may not be reinitialized at this time which may
298 * lead to a crash.
299 */
Hanumanth Reddy Pothula2a8a7402017-07-03 14:06:11 +0530300 if (cds_is_driver_recovering() || cds_is_driver_in_bad_state()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530301 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800302 "%s: Recovery in Progress. Ignore!!!", __func__);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800303 goto drop_pkt;
304 }
305
306 /*
307 * If the device is operating on a DFS Channel
308 * then check if SAP is in CAC WAIT state and
309 * drop the packets. In CAC WAIT state device
310 * is expected not to transmit any frames.
311 * SAP starts Tx only after the BSS START is
312 * done.
313 */
Jeff Johnson9bf24972017-10-04 09:26:58 -0700314 if (ap_ctx->dfs_cac_block_tx)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800315 goto drop_pkt;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800316
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800317 /*
Jeff Johnson4fca3b52017-01-12 08:44:18 -0800318 * If a transmit function is not registered, drop packet
319 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700320 if (!adapter->tx_fn) {
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800321 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
322 "%s: TX function not registered by the data path",
323 __func__);
324 goto drop_pkt;
325 }
326
Nirav Shah5e74bb82016-07-20 16:01:27 +0530327 wlan_hdd_classify_pkt(skb);
328
Anurag Chouhan6d760662016-02-20 16:05:43 +0530329 pDestMacAddress = (struct qdf_mac_addr *) skb->data;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800330
Nirav Shah5e74bb82016-07-20 16:01:27 +0530331 if (QDF_NBUF_CB_GET_IS_BCAST(skb) ||
332 QDF_NBUF_CB_GET_IS_MCAST(skb)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800333 /* The BC/MC station ID is assigned during BSS
334 * starting phase. SAP will return the station ID
335 * used for BC/MC traffic.
336 */
Jeff Johnson9bf24972017-10-04 09:26:58 -0700337 STAId = ap_ctx->uBCStaId;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800338 } else {
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530339 if (QDF_STATUS_SUCCESS !=
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700340 hdd_softap_get_sta_id(adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800341 pDestMacAddress, &STAId)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530342 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530343 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800344 "%s: Failed to find right station", __func__);
345 goto drop_pkt;
346 }
347
Prakash Dhavali3107b752016-11-28 14:35:06 +0530348 if (STAId >= WLAN_MAX_STA_COUNT) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530349 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530350 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800351 "%s: Failed to find right station", __func__);
352 goto drop_pkt;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700353 } else if (false == adapter->aStaInfo[STAId].isUsed) {
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: STA %d is unregistered", __func__,
357 STAId);
358 goto drop_pkt;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700359 } else if (true == adapter->aStaInfo[STAId].
Poddar, Siddartha5075462017-03-16 19:20:09 +0530360 isDeauthInProgress) {
361 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530362 QDF_TRACE_LEVEL_INFO_HIGH,
Poddar, Siddartha5075462017-03-16 19:20:09 +0530363 "%s: STA %d deauth in progress", __func__,
364 STAId);
365 goto drop_pkt;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800366 }
367
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800368 if ((OL_TXRX_PEER_STATE_CONN !=
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700369 adapter->aStaInfo[STAId].tlSTAState)
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800370 && (OL_TXRX_PEER_STATE_AUTH !=
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700371 adapter->aStaInfo[STAId].tlSTAState)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530372 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530373 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800374 "%s: Station not connected yet", __func__);
375 goto drop_pkt;
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800376 } else if (OL_TXRX_PEER_STATE_CONN ==
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700377 adapter->aStaInfo[STAId].tlSTAState) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800378 if (ntohs(skb->protocol) != HDD_ETHERTYPE_802_1_X) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530379 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530380 QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800381 "%s: NON-EAPOL packet in non-Authenticated state",
382 __func__);
383 goto drop_pkt;
384 }
385 }
386 }
387
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700388 hdd_get_tx_resource(adapter, STAId,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800389 WLAN_SAP_HDD_TX_FLOW_CONTROL_OS_Q_BLOCK_TIME);
390
391 /* Get TL AC corresponding to Qdisc queue index/AC. */
392 ac = hdd_qdisc_ac_to_tl_ac[skb->queue_mapping];
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700393 ++adapter->hdd_stats.hddTxRxStats.txXmitClassifiedAC[ac];
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800394
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700395#if defined(IPA_OFFLOAD)
Nirav Shahcbc6d722016-03-01 16:24:53 +0530396 if (!qdf_nbuf_ipa_owned_get(skb)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800397#endif
gbianec670c592016-11-24 11:21:30 +0800398
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700399 skb = hdd_skb_orphan(adapter, skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800400 if (!skb)
Jeff Johnsonedeff232015-11-11 17:19:42 -0800401 goto drop_pkt_accounting;
gbianec670c592016-11-24 11:21:30 +0800402
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700403#if defined(IPA_OFFLOAD)
Yun Park01deb2c2017-06-14 15:21:44 -0700404 } else {
405 /*
406 * Clear the IPA ownership after check it to avoid ipa_free_skb
407 * is called when Tx completed for intra-BSS Tx packets
408 */
409 qdf_nbuf_ipa_owned_clear(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800410 }
411#endif
412
Himanshu Agarwal53298d12017-02-20 19:14:17 +0530413 /*
414 * Add SKB to internal tracking table before further processing
415 * in WLAN driver.
416 */
417 qdf_net_buf_debug_acquire_skb(skb, __FILE__, __LINE__);
418
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700419 adapter->stats.tx_bytes += skb->len;
420 adapter->aStaInfo[STAId].tx_bytes += skb->len;
Mohit Khannab1dd1e82017-02-04 15:14:38 -0800421
Will Huang496b36c2017-07-11 16:38:50 +0800422 if (qdf_nbuf_is_tso(skb)) {
423 num_seg = qdf_nbuf_get_tso_num_seg(skb);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700424 adapter->stats.tx_packets += num_seg;
425 adapter->aStaInfo[STAId].tx_packets += num_seg;
Will Huang496b36c2017-07-11 16:38:50 +0800426 } else {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700427 ++adapter->stats.tx_packets;
428 adapter->aStaInfo[STAId].tx_packets++;
Will Huang496b36c2017-07-11 16:38:50 +0800429 }
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700430 adapter->aStaInfo[STAId].last_tx_rx_ts = qdf_system_ticks();
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800431
Nirav Shah5e74bb82016-07-20 16:01:27 +0530432 hdd_event_eapol_log(skb, QDF_TX);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700433 qdf_dp_trace_log_pkt(adapter->sessionId, skb, QDF_TX,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700434 QDF_TRACE_DEFAULT_PDEV_ID);
Nirav Shahcbc6d722016-03-01 16:24:53 +0530435 QDF_NBUF_CB_TX_PACKET_TRACK(skb) = QDF_NBUF_TX_PKT_DATA_TRACK;
436 QDF_NBUF_UPDATE_TX_PKT_COUNT(skb, QDF_NBUF_TX_PKT_HDD);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800437
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530438 qdf_dp_trace_set_track(skb, QDF_TX);
439 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_PTR_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700440 QDF_TRACE_DEFAULT_PDEV_ID, qdf_nbuf_data_addr(skb),
441 sizeof(qdf_nbuf_data(skb)),
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530442 QDF_TX));
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530443 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700444 QDF_TRACE_DEFAULT_PDEV_ID, (uint8_t *)skb->data,
445 qdf_nbuf_len(skb), QDF_TX));
Nirav Shahcbc6d722016-03-01 16:24:53 +0530446 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530447 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700448 QDF_TRACE_DEFAULT_PDEV_ID,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530449 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
450 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE), QDF_TX));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800451
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700452 if (adapter->tx_fn(adapter->txrx_vdev,
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800453 (qdf_nbuf_t) skb) != NULL) {
Poddar, Siddarth31b9b8b2017-04-07 12:04:55 +0530454 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800455 "%s: Failed to send packet to txrx for staid:%d",
456 __func__, STAId);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700457 ++adapter->hdd_stats.hddTxRxStats.txXmitDroppedAC[ac];
Himanshu Agarwal53298d12017-02-20 19:14:17 +0530458 goto drop_pkt_and_release_skb;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800459 }
Dustin Browne0024fa2016-10-14 16:29:21 -0700460 netif_trans_update(dev);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800461
462 return NETDEV_TX_OK;
463
Himanshu Agarwal53298d12017-02-20 19:14:17 +0530464drop_pkt_and_release_skb:
465 qdf_net_buf_debug_release_skb(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800466drop_pkt:
467
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530468 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_DROP_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700469 QDF_TRACE_DEFAULT_PDEV_ID, (uint8_t *)skb->data,
470 qdf_nbuf_len(skb), QDF_TX));
Nirav Shahcbc6d722016-03-01 16:24:53 +0530471 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530472 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_DROP_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700473 QDF_TRACE_DEFAULT_PDEV_ID,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530474 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
475 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE), QDF_TX));
Jeff Johnsonedeff232015-11-11 17:19:42 -0800476 kfree_skb(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800477
Jeff Johnsonedeff232015-11-11 17:19:42 -0800478drop_pkt_accounting:
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700479 ++adapter->stats.tx_dropped;
480 ++adapter->hdd_stats.hddTxRxStats.txXmitDropped;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800481
482 return NETDEV_TX_OK;
483}
484
485/**
Mukul Sharmac4de4ef2016-09-12 15:39:00 +0530486 * hdd_softap_hard_start_xmit() - Wrapper function to protect
487 * __hdd_softap_hard_start_xmit from SSR
488 * @skb: pointer to OS packet
489 * @dev: pointer to net_device structure
490 *
491 * Function called by OS if any packet needs to transmit.
492 *
493 * Return: Always returns NETDEV_TX_OK
494 */
495int hdd_softap_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
496{
497 int ret;
498
499 cds_ssr_protect(__func__);
500 ret = __hdd_softap_hard_start_xmit(skb, dev);
501 cds_ssr_unprotect(__func__);
502
503 return ret;
504}
505
Sen, Devendra154b3c42017-02-13 20:44:15 +0530506#ifdef FEATURE_WLAN_DIAG_SUPPORT
507/**
508 * hdd_wlan_datastall_sap_event()- Send SAP datastall information
509 *
510 * This Function send send SAP datastall diag event
511 *
512 * Return: void.
513 */
514static void hdd_wlan_datastall_sap_event(void)
515{
516 WLAN_HOST_DIAG_EVENT_DEF(sap_data_stall,
517 struct host_event_wlan_datastall);
518 qdf_mem_zero(&sap_data_stall, sizeof(sap_data_stall));
519 sap_data_stall.reason = SOFTAP_TX_TIMEOUT;
520 WLAN_HOST_DIAG_EVENT_REPORT(&sap_data_stall,
521 EVENT_WLAN_SOFTAP_DATASTALL);
522}
523#else
524static inline void hdd_wlan_datastall_sap_event(void)
525{
526
527}
528#endif
529
Mukul Sharmac4de4ef2016-09-12 15:39:00 +0530530/**
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800531 * __hdd_softap_tx_timeout() - TX timeout handler
532 * @dev: pointer to network device
533 *
534 * This function is registered as a netdev ndo_tx_timeout method, and
535 * is invoked by the kernel if the driver takes too long to transmit a
536 * frame.
537 *
538 * Return: None
539 */
540static void __hdd_softap_tx_timeout(struct net_device *dev)
541{
Jeff Johnson18dd7e12017-08-29 14:22:28 -0700542 struct hdd_adapter *adapter = WLAN_HDD_GET_PRIV_PTR(dev);
Jeff Johnsonf7726872017-08-28 11:38:31 -0700543 struct hdd_context *hdd_ctx;
Nirav Shah89223f72016-03-01 18:10:38 +0530544 struct netdev_queue *txq;
545 int i;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800546
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530547 DPTRACE(qdf_dp_trace(NULL, QDF_DP_TRACE_HDD_SOFTAP_TX_TIMEOUT,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700548 QDF_TRACE_DEFAULT_PDEV_ID,
549 NULL, 0, QDF_TX));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800550 /* Getting here implies we disabled the TX queues for too
551 * long. Queues are disabled either because of disassociation
552 * or low resource scenarios. In case of disassociation it is
553 * ok to ignore this. But if associated, we have do possible
554 * recovery here
555 */
556 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Hanumanth Reddy Pothula2a8a7402017-07-03 14:06:11 +0530557 if (cds_is_driver_recovering() || cds_is_driver_in_bad_state()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530558 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800559 "%s: Recovery in Progress. Ignore!!!", __func__);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800560 return;
561 }
Nirav Shah89223f72016-03-01 18:10:38 +0530562
Dustin Browne0024fa2016-10-14 16:29:21 -0700563 TX_TIMEOUT_TRACE(dev, QDF_MODULE_ID_HDD_SAP_DATA);
Nirav Shah89223f72016-03-01 18:10:38 +0530564
565 for (i = 0; i < NUM_TX_QUEUES; i++) {
566 txq = netdev_get_tx_queue(dev, i);
567 QDF_TRACE(QDF_MODULE_ID_HDD_DATA,
Srinivas Girigowda028c4482017-03-09 18:52:02 -0800568 QDF_TRACE_LEVEL_DEBUG,
569 "Queue: %d status: %d txq->trans_start: %lu",
Nirav Shah89223f72016-03-01 18:10:38 +0530570 i, netif_tx_queue_stopped(txq), txq->trans_start);
571 }
572
573 wlan_hdd_display_netif_queue_history(hdd_ctx);
Leo Changfdb45c32016-10-28 11:09:23 -0700574 cdp_dump_flow_pool_info(cds_get_context(QDF_MODULE_ID_SOC));
Srinivas Girigowda028c4482017-03-09 18:52:02 -0800575 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_DEBUG,
Nirav Shah89223f72016-03-01 18:10:38 +0530576 "carrier state: %d", netif_carrier_ok(dev));
Sen, Devendra154b3c42017-02-13 20:44:15 +0530577 hdd_wlan_datastall_sap_event();
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800578}
579
580/**
581 * hdd_softap_tx_timeout() - SSR wrapper for __hdd_softap_tx_timeout
582 * @dev: pointer to net_device
583 *
584 * Return: none
585 */
586void hdd_softap_tx_timeout(struct net_device *dev)
587{
588 cds_ssr_protect(__func__);
589 __hdd_softap_tx_timeout(dev);
590 cds_ssr_unprotect(__func__);
591}
592
593/**
594 * @hdd_softap_init_tx_rx() - Initialize Tx/RX module
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700595 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800596 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530597 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
598 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800599 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700600QDF_STATUS hdd_softap_init_tx_rx(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800601{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530602 QDF_STATUS status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800603
604 uint8_t STAId = 0;
605
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700606 qdf_mem_zero(&adapter->stats, sizeof(struct net_device_stats));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800607
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700608 spin_lock_init(&adapter->staInfo_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800609
610 for (STAId = 0; STAId < WLAN_MAX_STA_COUNT; STAId++) {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700611 qdf_mem_zero(&adapter->aStaInfo[STAId],
Jeff Johnson82155922017-09-30 16:54:14 -0700612 sizeof(struct hdd_station_info));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800613 }
614
615 return status;
616}
617
618/**
619 * @hdd_softap_deinit_tx_rx() - Deinitialize Tx/RX module
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700620 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800621 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530622 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
623 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800624 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700625QDF_STATUS hdd_softap_deinit_tx_rx(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800626{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700627 if (adapter == NULL) {
628 hdd_err("Called with adapter = NULL.");
Dhanashri Atreed3bf512017-02-21 12:25:53 -0800629 return QDF_STATUS_E_FAILURE;
630 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800631
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700632 adapter->txrx_vdev = NULL;
633 adapter->tx_fn = NULL;
Dhanashri Atreed3bf512017-02-21 12:25:53 -0800634 hdd_info("Deregistering TX function hook !");
635 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800636}
637
638/**
639 * hdd_softap_init_tx_rx_sta() - Initialize tx/rx for a softap station
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700640 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800641 * @STAId: Station ID to initialize
642 * @pmacAddrSTA: pointer to the MAC address of the station
643 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530644 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
645 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800646 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700647QDF_STATUS hdd_softap_init_tx_rx_sta(struct hdd_adapter *adapter,
Jeff Johnson18dd7e12017-08-29 14:22:28 -0700648 uint8_t STAId,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530649 struct qdf_mac_addr *pmacAddrSTA)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800650{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700651 spin_lock_bh(&adapter->staInfo_lock);
652 if (adapter->aStaInfo[STAId].isUsed) {
653 spin_unlock_bh(&adapter->staInfo_lock);
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800654 hdd_err("Reinit of in use station %d", STAId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530655 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800656 }
657
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700658 qdf_mem_zero(&adapter->aStaInfo[STAId],
Jeff Johnson82155922017-09-30 16:54:14 -0700659 sizeof(struct hdd_station_info));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800660
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700661 adapter->aStaInfo[STAId].isUsed = true;
662 adapter->aStaInfo[STAId].isDeauthInProgress = false;
663 qdf_copy_macaddr(&adapter->aStaInfo[STAId].macAddrSTA, pmacAddrSTA);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800664
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700665 spin_unlock_bh(&adapter->staInfo_lock);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530666 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800667}
668
669/**
670 * hdd_softap_deinit_tx_rx_sta() - Deinitialize tx/rx for a softap station
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700671 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800672 * @STAId: Station ID to deinitialize
673 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530674 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
675 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800676 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700677QDF_STATUS hdd_softap_deinit_tx_rx_sta(struct hdd_adapter *adapter,
Jeff Johnson18dd7e12017-08-29 14:22:28 -0700678 uint8_t STAId)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800679{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530680 QDF_STATUS status = QDF_STATUS_SUCCESS;
Jeff Johnson5c19ade2017-10-04 09:52:12 -0700681 struct hdd_hostapd_state *hostapd_state;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800682
Jeff Johnson5c19ade2017-10-04 09:52:12 -0700683 hostapd_state = WLAN_HDD_GET_HOSTAP_STATE_PTR(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800684
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700685 spin_lock_bh(&adapter->staInfo_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800686
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700687 if (false == adapter->aStaInfo[STAId].isUsed) {
688 spin_unlock_bh(&adapter->staInfo_lock);
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800689 hdd_err("Deinit station not inited %d", STAId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530690 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800691 }
692
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700693 adapter->aStaInfo[STAId].isUsed = false;
694 adapter->aStaInfo[STAId].isDeauthInProgress = false;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800695
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700696 spin_unlock_bh(&adapter->staInfo_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800697 return status;
698}
699
700/**
701 * hdd_softap_rx_packet_cbk() - Receive packet handler
Dhanashri Atre182b0272016-02-17 15:35:07 -0800702 * @context: pointer to HDD context
Nirav Shahcbc6d722016-03-01 16:24:53 +0530703 * @rxBuf: pointer to rx qdf_nbuf
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800704 *
705 * Receive callback registered with TL. TL will call this to notify
706 * the HDD when one or more packets were received for a registered
707 * STA.
708 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530709 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
710 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800711 */
Dhanashri Atre182b0272016-02-17 15:35:07 -0800712QDF_STATUS hdd_softap_rx_packet_cbk(void *context, qdf_nbuf_t rxBuf)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800713{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700714 struct hdd_adapter *adapter = NULL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800715 int rxstat;
716 unsigned int cpu_index;
717 struct sk_buff *skb = NULL;
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800718 struct sk_buff *next = NULL;
Jeff Johnson92402872017-09-03 09:25:37 -0700719 struct hdd_context *hdd_ctx = NULL;
Will Huang496b36c2017-07-11 16:38:50 +0800720 struct qdf_mac_addr src_mac;
721 uint8_t staid;
Mohit Khannaf8f96822017-05-17 17:11:59 -0700722 bool proto_pkt_logged = false;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800723
724 /* Sanity check on inputs */
Dhanashri Atre182b0272016-02-17 15:35:07 -0800725 if (unlikely((NULL == context) || (NULL == rxBuf))) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530726 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800727 "%s: Null params being passed", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530728 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800729 }
730
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700731 adapter = (struct hdd_adapter *)context;
732 if (unlikely(WLAN_HDD_ADAPTER_MAGIC != adapter->magic)) {
Srinivas Girigowda028c4482017-03-09 18:52:02 -0800733 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_ERROR,
Dhanashri Atre182b0272016-02-17 15:35:07 -0800734 "Magic cookie(%x) for adapter sanity verification is invalid",
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700735 adapter->magic);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530736 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800737 }
738
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700739 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Jeff Johnson92402872017-09-03 09:25:37 -0700740 if (unlikely(NULL == hdd_ctx)) {
Dhanashri Atre182b0272016-02-17 15:35:07 -0800741 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
742 "%s: HDD context is Null", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530743 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800744 }
745
746 /* walk the chain until all are processed */
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800747 next = (struct sk_buff *)rxBuf;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800748
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800749 while (next) {
750 skb = next;
751 next = skb->next;
Dhanashri Atre63d98022017-01-24 18:22:09 -0800752 skb->next = NULL;
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800753
psimha884025c2017-08-01 15:07:32 -0700754#ifdef QCA_WIFI_QCA6290 /* Debug code, remove later */
Venkata Sharath Chandra Manchalacc789172017-07-25 23:28:45 -0700755 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_INFO,
Jeff Johnson36e74c42017-09-18 08:15:42 -0700756 "%s: skb %pK skb->len %d\n", __func__, skb, skb->len);
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800757#endif
758
Dhanashri Atre63d98022017-01-24 18:22:09 -0800759 hdd_softap_dump_sk_buff(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800760
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700761 skb->dev = adapter->dev;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800762
Dhanashri Atre63d98022017-01-24 18:22:09 -0800763 if (unlikely(skb->dev == NULL)) {
764
765 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
766 "%s: ERROR!!Invalid netdevice", __func__);
Dhanashri Atrecefa8802017-02-02 16:17:14 -0800767 continue;
Dhanashri Atre63d98022017-01-24 18:22:09 -0800768 }
769 cpu_index = wlan_hdd_get_cpu();
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700770 ++adapter->hdd_stats.hddTxRxStats.rxPackets[cpu_index];
771 ++adapter->stats.rx_packets;
772 adapter->stats.rx_bytes += skb->len;
Dhanashri Atre63d98022017-01-24 18:22:09 -0800773
Will Huang496b36c2017-07-11 16:38:50 +0800774 qdf_mem_copy(&src_mac, skb->data + QDF_NBUF_SRC_MAC_OFFSET,
775 sizeof(src_mac));
776 if (QDF_STATUS_SUCCESS ==
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700777 hdd_softap_get_sta_id(adapter, &src_mac, &staid)) {
Will Huang496b36c2017-07-11 16:38:50 +0800778 if (staid < WLAN_MAX_STA_COUNT) {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700779 adapter->aStaInfo[staid].rx_packets++;
780 adapter->aStaInfo[staid].rx_bytes += skb->len;
781 adapter->aStaInfo[staid].last_tx_rx_ts =
Will Huang496b36c2017-07-11 16:38:50 +0800782 qdf_system_ticks();
783 }
784 }
Dhanashri Atre63d98022017-01-24 18:22:09 -0800785 hdd_event_eapol_log(skb, QDF_RX);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700786 proto_pkt_logged = qdf_dp_trace_log_pkt(adapter->sessionId,
Mohit Khannaf8f96822017-05-17 17:11:59 -0700787 skb, QDF_RX,
788 QDF_TRACE_DEFAULT_PDEV_ID);
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530789 DPTRACE(qdf_dp_trace(skb,
Dhanashri Atre63d98022017-01-24 18:22:09 -0800790 QDF_DP_TRACE_RX_HDD_PACKET_PTR_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700791 QDF_TRACE_DEFAULT_PDEV_ID,
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530792 qdf_nbuf_data_addr(skb),
793 sizeof(qdf_nbuf_data(skb)), QDF_RX));
Mohit Khannaf8f96822017-05-17 17:11:59 -0700794 if (!proto_pkt_logged) {
795 DPTRACE(qdf_dp_trace(skb,
796 QDF_DP_TRACE_HDD_RX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700797 QDF_TRACE_DEFAULT_PDEV_ID,
798 (uint8_t *)skb->data, qdf_nbuf_len(skb),
799 QDF_RX));
Mohit Khannaf8f96822017-05-17 17:11:59 -0700800 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
801 DPTRACE(qdf_dp_trace(skb,
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530802 QDF_DP_TRACE_HDD_RX_PACKET_RECORD,
Venkata Sharath Chandra Manchala0b9fc632017-05-15 14:35:15 -0700803 QDF_TRACE_DEFAULT_PDEV_ID,
Himanshu Agarwalee3411a2017-01-31 12:56:47 +0530804 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
805 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE),
806 QDF_RX));
Mohit Khannaf8f96822017-05-17 17:11:59 -0700807 }
Dhanashri Atre63d98022017-01-24 18:22:09 -0800808
809 skb->protocol = eth_type_trans(skb, skb->dev);
810
811 /* hold configurable wakelock for unicast traffic */
Jeff Johnson92402872017-09-03 09:25:37 -0700812 if (hdd_ctx->config->rx_wakelock_timeout &&
Dhanashri Atre63d98022017-01-24 18:22:09 -0800813 skb->pkt_type != PACKET_BROADCAST &&
814 skb->pkt_type != PACKET_MULTICAST) {
Jeff Johnson92402872017-09-03 09:25:37 -0700815 cds_host_diag_log_work(&hdd_ctx->rx_wake_lock,
816 hdd_ctx->config->rx_wakelock_timeout,
Dhanashri Atre63d98022017-01-24 18:22:09 -0800817 WIFI_POWER_EVENT_WAKELOCK_HOLD_RX);
Jeff Johnson92402872017-09-03 09:25:37 -0700818 qdf_wake_lock_timeout_acquire(&hdd_ctx->rx_wake_lock,
819 hdd_ctx->config->
Dhanashri Atre63d98022017-01-24 18:22:09 -0800820 rx_wakelock_timeout);
821 }
822
823 /* Remove SKB from internal tracking table before submitting
824 * it to stack
825 */
Manjunathappa Prakash17c07bd2017-04-27 21:24:28 -0700826 qdf_net_buf_debug_release_skb(skb);
Dhanashri Atre63d98022017-01-24 18:22:09 -0800827 if (hdd_napi_enabled(HDD_NAPI_ANY) &&
Jeff Johnson92402872017-09-03 09:25:37 -0700828 !hdd_ctx->enableRxThread)
Dhanashri Atre63d98022017-01-24 18:22:09 -0800829 rxstat = netif_receive_skb(skb);
830 else
831 rxstat = netif_rx_ni(skb);
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700832 if (NET_RX_SUCCESS == rxstat)
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700833 ++adapter->hdd_stats.hddTxRxStats.rxDelivered[cpu_index];
Srinivas Girigowdae806a822017-03-25 11:25:30 -0700834 else
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700835 ++adapter->hdd_stats.hddTxRxStats.rxRefused[cpu_index];
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800836 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800837
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530838 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800839}
840
841/**
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700842 * hdd_softap_deregister_sta(struct hdd_adapter *adapter, uint8_t staId)
843 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800844 * @staId: Station ID to deregister
845 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530846 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800847 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700848QDF_STATUS hdd_softap_deregister_sta(struct hdd_adapter *adapter,
Jeff Johnson18dd7e12017-08-29 14:22:28 -0700849 uint8_t staId)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800850{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530851 QDF_STATUS qdf_status = QDF_STATUS_SUCCESS;
Jeff Johnson92402872017-09-03 09:25:37 -0700852 struct hdd_context *hdd_ctx;
Abhishek Singh07c627e2017-03-20 17:56:34 +0530853 int ret;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800854
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700855 if (NULL == adapter) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800856 hdd_err("NULL adapter");
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530857 return QDF_STATUS_E_INVAL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800858 }
859
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700860 if (WLAN_HDD_ADAPTER_MAGIC != adapter->magic) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800861 hdd_err("Invalid adapter magic");
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530862 return QDF_STATUS_E_INVAL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800863 }
864
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700865 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800866 /* Clear station in TL and then update HDD data
867 * structures. This helps to block RX frames from other
868 * station to this station.
869 */
Krishna Kumaar Natarajane58b4092017-01-25 15:47:35 -0800870 qdf_status = cdp_clear_peer(cds_get_context(QDF_MODULE_ID_SOC),
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800871 (struct cdp_pdev *)cds_get_context(QDF_MODULE_ID_TXRX),
872 staId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530873 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Krishna Kumaar Natarajane58b4092017-01-25 15:47:35 -0800874 hdd_err("cdp_clear_peer failed for staID %d, Status=%d [0x%08X]",
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800875 staId, qdf_status, qdf_status);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800876 }
877
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700878 ret = hdd_objmgr_remove_peer_object(adapter->hdd_vdev,
879 adapter->aStaInfo[staId].
Dustin Brown7d043f62017-03-27 12:07:36 -0700880 macAddrSTA.bytes);
Abhishek Singh07c627e2017-03-20 17:56:34 +0530881 if (ret)
882 hdd_err("Peer obj %pM delete fails",
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700883 adapter->aStaInfo[staId].macAddrSTA.bytes);
Abhishek Singh07c627e2017-03-20 17:56:34 +0530884
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700885 if (adapter->aStaInfo[staId].isUsed) {
886 spin_lock_bh(&adapter->staInfo_lock);
887 qdf_mem_zero(&adapter->aStaInfo[staId],
Jeff Johnson82155922017-09-30 16:54:14 -0700888 sizeof(struct hdd_station_info));
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700889 spin_unlock_bh(&adapter->staInfo_lock);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800890 }
Abhishek Singh07c627e2017-03-20 17:56:34 +0530891
Jeff Johnson92402872017-09-03 09:25:37 -0700892 hdd_ctx->sta_to_adapter[staId] = NULL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800893
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530894 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800895}
896
897/**
898 * hdd_softap_register_sta() - Register a SoftAP STA
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700899 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800900 * @fAuthRequired: is additional authentication required?
901 * @fPrivacyBit: should 802.11 privacy bit be set?
902 * @staId: station ID assigned to this station
903 * @ucastSig: unicast security signature
904 * @bcastSig: broadcast security signature
905 * @pPeerMacAddress: station MAC address
906 * @fWmmEnabled: is WMM enabled for this STA?
907 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530908 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800909 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700910QDF_STATUS hdd_softap_register_sta(struct hdd_adapter *adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800911 bool fAuthRequired,
912 bool fPrivacyBit,
913 uint8_t staId,
914 uint8_t ucastSig,
915 uint8_t bcastSig,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530916 struct qdf_mac_addr *pPeerMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800917 bool fWmmEnabled)
918{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530919 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800920 struct ol_txrx_desc_type staDesc = { 0 };
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700921 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Dhanashri Atre182b0272016-02-17 15:35:07 -0800922 struct ol_txrx_ops txrx_ops;
Leo Changfdb45c32016-10-28 11:09:23 -0700923 void *soc = cds_get_context(QDF_MODULE_ID_SOC);
924 void *pdev = cds_get_context(QDF_MODULE_ID_TXRX);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800925
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800926 hdd_info("STA:%u, Auth:%u, Priv:%u, WMM:%u",
927 staId, fAuthRequired, fPrivacyBit, fWmmEnabled);
928
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800929 /*
930 * Clean up old entry if it is not cleaned up properly
931 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700932 if (adapter->aStaInfo[staId].isUsed) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800933 hdd_info("clean up old entry for STA %d", staId);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700934 hdd_softap_deregister_sta(adapter, staId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800935 }
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800936
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800937 /* Get the Station ID from the one saved during the assocation. */
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800938 staDesc.sta_id = staId;
939
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700940 /* Save the adapter Pointer for this staId */
941 hdd_ctx->sta_to_adapter[staId] = adapter;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800942
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530943 qdf_status =
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700944 hdd_softap_init_tx_rx_sta(adapter, staId,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800945 pPeerMacAddress);
946
947 staDesc.is_qos_enabled = fWmmEnabled;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800948
Dhanashri Atre182b0272016-02-17 15:35:07 -0800949 /* Register the vdev transmit and receive functions */
950 qdf_mem_zero(&txrx_ops, sizeof(txrx_ops));
951 txrx_ops.rx.rx = hdd_softap_rx_packet_cbk;
Leo Changfdb45c32016-10-28 11:09:23 -0700952 cdp_vdev_register(soc,
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800953 (struct cdp_vdev *)cdp_get_vdev_from_vdev_id(soc,
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700954 (struct cdp_pdev *)pdev, adapter->sessionId),
955 adapter, &txrx_ops);
956 adapter->txrx_vdev = (void *)cdp_get_vdev_from_vdev_id(soc,
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800957 (struct cdp_pdev *)pdev,
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700958 adapter->sessionId);
959 adapter->tx_fn = txrx_ops.tx.tx;
Dhanashri Atre182b0272016-02-17 15:35:07 -0800960
Venkata Sharath Chandra Manchala0d44d452016-11-23 17:48:15 -0800961 qdf_status = cdp_peer_register(soc,
962 (struct cdp_pdev *)pdev, &staDesc);
Dhanashri Atre50141c52016-04-07 13:15:29 -0700963 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800964 hdd_err("cdp_peer_register() failed to register. Status = %d [0x%08X]",
965 qdf_status, qdf_status);
Dhanashri Atre50141c52016-04-07 13:15:29 -0700966 return qdf_status;
967 }
968
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800969 /* if ( WPA ), tell TL to go to 'connected' and after keys come to the
970 * driver then go to 'authenticated'. For all other authentication
971 * types (those that do not require upper layer authentication) we can
972 * put TL directly into 'authenticated' state
973 */
974
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700975 adapter->aStaInfo[staId].ucSTAId = staId;
976 adapter->aStaInfo[staId].isQosEnabled = fWmmEnabled;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800977
978 if (!fAuthRequired) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800979 hdd_info("open/shared auth StaId= %d. Changing TL state to AUTHENTICATED at Join time",
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700980 adapter->aStaInfo[staId].ucSTAId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800981
982 /* Connections that do not need Upper layer auth,
983 * transition TL directly to 'Authenticated' state.
984 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700985 qdf_status = hdd_change_peer_state(adapter, staDesc.sta_id,
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800986 OL_TXRX_PEER_STATE_AUTH, false);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800987
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700988 adapter->aStaInfo[staId].tlSTAState = OL_TXRX_PEER_STATE_AUTH;
989 adapter->sessionCtx.ap.uIsAuthenticated = true;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800990 } else {
991
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -0800992 hdd_info("ULA auth StaId= %d. Changing TL state to CONNECTED at Join time",
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700993 adapter->aStaInfo[staId].ucSTAId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800994
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700995 qdf_status = hdd_change_peer_state(adapter, staDesc.sta_id,
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800996 OL_TXRX_PEER_STATE_CONN, false);
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700997 adapter->aStaInfo[staId].tlSTAState = OL_TXRX_PEER_STATE_CONN;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800998
Jeff Johnsond31ab6a2017-10-02 13:23:51 -0700999 adapter->sessionCtx.ap.uIsAuthenticated = false;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001000
1001 }
1002
Varun Reddy Yeturu8a5d3d42017-08-02 13:03:27 -07001003 hdd_debug("Enabling queues");
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001004 wlan_hdd_netif_queue_control(adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001005 WLAN_START_ALL_NETIF_QUEUE_N_CARRIER,
1006 WLAN_CONTROL_PATH);
1007
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301008 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001009}
1010
1011/**
1012 * hdd_softap_register_bc_sta() - Register the SoftAP broadcast STA
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001013 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001014 * @fPrivacyBit: should 802.11 privacy bit be set?
1015 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301016 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001017 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001018QDF_STATUS hdd_softap_register_bc_sta(struct hdd_adapter *adapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001019 bool fPrivacyBit)
1020{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301021 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001022 struct hdd_context *hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Anurag Chouhan6d760662016-02-20 16:05:43 +05301023 struct qdf_mac_addr broadcastMacAddr =
1024 QDF_MAC_ADDR_BROADCAST_INITIALIZER;
Jeff Johnson9bf24972017-10-04 09:26:58 -07001025 struct hdd_ap_ctx *ap_ctx;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001026
Jeff Johnson9bf24972017-10-04 09:26:58 -07001027 ap_ctx = WLAN_HDD_GET_AP_CTX_PTR(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001028
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001029 hdd_ctx->sta_to_adapter[WLAN_RX_BCMC_STA_ID] = adapter;
Jeff Johnson9bf24972017-10-04 09:26:58 -07001030 hdd_ctx->sta_to_adapter[ap_ctx->uBCStaId] = adapter;
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301031 qdf_status =
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001032 hdd_softap_register_sta(adapter, false, fPrivacyBit,
1033 (WLAN_HDD_GET_AP_CTX_PTR(adapter))->
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001034 uBCStaId, 0, 1, &broadcastMacAddr, 0);
1035
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301036 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001037}
1038
1039/**
1040 * hdd_softap_deregister_bc_sta() - Deregister the SoftAP broadcast STA
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001041 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001042 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301043 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001044 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001045QDF_STATUS hdd_softap_deregister_bc_sta(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001046{
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001047 return hdd_softap_deregister_sta(adapter,
1048 (WLAN_HDD_GET_AP_CTX_PTR(adapter))->
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001049 uBCStaId);
1050}
1051
1052/**
1053 * hdd_softap_stop_bss() - Stop the BSS
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001054 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001055 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301056 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001057 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001058QDF_STATUS hdd_softap_stop_bss(struct hdd_adapter *adapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001059{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301060 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001061 uint8_t staId = 0;
Jeff Johnson92402872017-09-03 09:25:37 -07001062 struct hdd_context *hdd_ctx;
Srinivas Girigowdae806a822017-03-25 11:25:30 -07001063
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001064 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001065
1066 /* bss deregister is not allowed during wlan driver loading or
1067 * unloading
1068 */
Rajeev Kumarfec3dbe2016-01-19 15:23:52 -08001069 if (cds_is_load_or_unload_in_progress()) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001070 hdd_err("Loading_unloading in Progress, state: 0x%x. Ignore!!!",
1071 cds_get_driver_state());
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301072 return QDF_STATUS_E_PERM;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001073 }
1074
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001075 qdf_status = hdd_softap_deregister_bc_sta(adapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001076
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301077 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001078 hdd_err("Failed to deregister BC sta Id %d",
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001079 (WLAN_HDD_GET_AP_CTX_PTR(adapter))->uBCStaId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001080 }
1081
1082 for (staId = 0; staId < WLAN_MAX_STA_COUNT; staId++) {
1083 /* This excludes BC sta as it is already deregistered */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001084 if (adapter->aStaInfo[staId].isUsed) {
1085 qdf_status = hdd_softap_deregister_sta(adapter, staId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301086 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001087 hdd_err("Failed to deregister sta Id %d",
1088 staId);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001089 }
1090 }
1091 }
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301092 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001093}
1094
1095/**
1096 * hdd_softap_change_sta_state() - Change the state of a SoftAP station
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001097 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001098 * @pDestMacAddress: MAC address of the station
1099 * @state: new state of the station
1100 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +05301101 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001102 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001103QDF_STATUS hdd_softap_change_sta_state(struct hdd_adapter *adapter,
Anurag Chouhan6d760662016-02-20 16:05:43 +05301104 struct qdf_mac_addr *pDestMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001105 enum ol_txrx_peer_state state)
1106{
1107 uint8_t ucSTAId = WLAN_MAX_STA_COUNT;
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001108 QDF_STATUS qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001109
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001110 ENTER_DEV(adapter->dev);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001111
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001112 qdf_status = hdd_softap_get_sta_id(adapter, pDestMacAddress, &ucSTAId);
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001113 if (QDF_STATUS_SUCCESS != qdf_status) {
1114 hdd_err("Failed to find right station");
1115 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001116 }
1117
1118 if (false ==
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001119 qdf_is_macaddr_equal(&adapter->aStaInfo[ucSTAId].macAddrSTA,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001120 pDestMacAddress)) {
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001121 hdd_err("Station %u MAC address not matching", ucSTAId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301122 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001123 }
1124
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301125 qdf_status =
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001126 hdd_change_peer_state(adapter, ucSTAId, state, false);
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001127 hdd_info("Station %u changed to state %d", ucSTAId, state);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001128
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301129 if (QDF_STATUS_SUCCESS == qdf_status) {
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001130 adapter->aStaInfo[ucSTAId].tlSTAState =
Dhanashri Atreb08959a2016-03-01 17:28:03 -08001131 OL_TXRX_PEER_STATE_AUTH;
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001132 p2p_peer_authorized(adapter->hdd_vdev, pDestMacAddress->bytes);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001133 }
1134
Jeff Johnsond5bd6fd2016-12-05 12:29:21 -08001135 EXIT();
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301136 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001137}
1138
1139/*
1140 * hdd_softap_get_sta_id() - Find station ID from MAC address
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001141 * @adapter: pointer to adapter context
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001142 * @pDestMacAddress: MAC address of the destination
1143 * @staId: Station ID associated with the MAC address
1144 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301145 * Return: QDF_STATUS_SUCCESS if a match was found, in which case
1146 * staId is populated, QDF_STATUS_E_FAILURE if a match is
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001147 * not found
1148 */
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001149QDF_STATUS hdd_softap_get_sta_id(struct hdd_adapter *adapter,
Anurag Chouhan6d760662016-02-20 16:05:43 +05301150 struct qdf_mac_addr *pMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001151 uint8_t *staId)
1152{
1153 uint8_t i;
1154
1155 for (i = 0; i < WLAN_MAX_STA_COUNT; i++) {
Anurag Chouhan600c3a02016-03-01 10:33:54 +05301156 if (!qdf_mem_cmp
Jeff Johnsond31ab6a2017-10-02 13:23:51 -07001157 (&adapter->aStaInfo[i].macAddrSTA, pMacAddress,
1158 QDF_MAC_ADDR_SIZE) && adapter->aStaInfo[i].isUsed) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001159 *staId = i;
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301160 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001161 }
1162 }
1163
Anurag Chouhanfb54ab02016-02-18 18:00:46 +05301164 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001165}