blob: 8781ca73703fbbf152b04b00df4e0368f6bdadd6 [file] [log] [blame]
Prakash Dhavali7090c5f2015-11-02 17:55:19 -08001/*
Orhan K AKYILDIZ4f20db52015-12-30 12:35:44 -08002 * Copyright (c) 2012-2016 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
28/* Include files */
29#include <linux/semaphore.h>
30#include <wlan_hdd_tx_rx.h>
31#include <wlan_hdd_softap_tx_rx.h>
32#include <linux/netdevice.h>
33#include <linux/skbuff.h>
34#include <linux/etherdevice.h>
Anurag Chouhan6d760662016-02-20 16:05:43 +053035#include <qdf_types.h>
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080036#include <ani_global.h>
Anurag Chouhan6d760662016-02-20 16:05:43 +053037#include <qdf_types.h>
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080038#include <net/ieee80211_radiotap.h>
39#include <cds_sched.h>
40#include <wlan_hdd_napi.h>
Dhanashri Atre182b0272016-02-17 15:35:07 -080041#include <ol_txrx.h>
Dhanashri Atreb08959a2016-03-01 17:28:03 -080042#include <cdp_txrx_peer_ops.h>
Manjunathappa Prakash779e4862016-09-12 17:00:11 -070043#include <cds_utils.h>
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080044
45#ifdef IPA_OFFLOAD
46#include <wlan_hdd_ipa.h>
47#endif
48
49/* Preprocessor definitions and constants */
50#undef QCA_HDD_SAP_DUMP_SK_BUFF
51
52/* Type declarations */
53
54/* Function definitions and documenation */
55#ifdef QCA_HDD_SAP_DUMP_SK_BUFF
56/**
57 * hdd_softap_dump_sk_buff() - Dump an skb
58 * @skb: skb to dump
59 *
60 * Return: None
61 */
62static void hdd_softap_dump_sk_buff(struct sk_buff *skb)
63{
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053064 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080065 "%s: head = %p ", __func__, skb->head);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053066 /* QDF_TRACE( QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,"%s: data = %p ", __func__, skb->data); */
67 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080068 "%s: tail = %p ", __func__, skb->tail);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053069 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080070 "%s: end = %p ", __func__, skb->end);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053071 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080072 "%s: len = %d ", __func__, skb->len);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +053073 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -080074 "%s: data_len = %d ", __func__, skb->data_len);
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: mac_len = %d", __func__, skb->mac_len);
77
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 "0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x ", skb->data[0],
80 skb->data[1], skb->data[2], skb->data[3], skb->data[4],
81 skb->data[5], skb->data[6], skb->data[7]);
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[8],
84 skb->data[9], skb->data[10], skb->data[11], skb->data[12],
85 skb->data[13], skb->data[14], skb->data[15]);
86}
87#else
88static void hdd_softap_dump_sk_buff(struct sk_buff *skb)
89{
90}
91#endif
92
93#ifdef QCA_LL_LEGACY_TX_FLOW_CONTROL
94/**
95 * hdd_softap_tx_resume_timer_expired_handler() - TX Q resume timer handler
96 * @adapter_context: pointer to vdev adapter
97 *
98 * TX Q resume timer handler for SAP and P2P GO interface. If Blocked
99 * OS Q is not resumed during timeout period, to prevent permanent
100 * stall, resume OS Q forcefully for SAP and P2P GO interface.
101 *
102 * Return: None
103 */
104void hdd_softap_tx_resume_timer_expired_handler(void *adapter_context)
105{
106 hdd_adapter_t *pAdapter = (hdd_adapter_t *) adapter_context;
107
108 if (!pAdapter) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530109 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800110 "%s: INV ARG", __func__);
111 /* INVALID ARG */
112 return;
113 }
114
115 hddLog(LOG1, FL("Enabling queues"));
116 wlan_hdd_netif_queue_control(pAdapter, WLAN_WAKE_ALL_NETIF_QUEUE,
117 WLAN_CONTROL_PATH);
118 return;
119}
120
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530121#if defined(CONFIG_PER_VDEV_TX_DESC_POOL)
122
123/**
124 * hdd_softap_tx_resume_false() - Resume OS TX Q false leads to queue disabling
125 * @pAdapter: pointer to hdd adapter
126 * @tx_resume: TX Q resume trigger
127 *
128 *
129 * Return: None
130 */
131static void
132hdd_softap_tx_resume_false(hdd_adapter_t *pAdapter, bool tx_resume)
133{
134 if (true == tx_resume)
135 return;
136
137 hdd_notice("Disabling queues");
138 wlan_hdd_netif_queue_control(pAdapter, WLAN_STOP_ALL_NETIF_QUEUE,
139 WLAN_DATA_FLOW_CONTROL);
140
141 if (QDF_TIMER_STATE_STOPPED ==
142 qdf_mc_timer_get_current_state(&pAdapter->
143 tx_flow_control_timer)) {
144 QDF_STATUS status;
145 status = qdf_mc_timer_start(&pAdapter->tx_flow_control_timer,
146 WLAN_SAP_HDD_TX_FLOW_CONTROL_OS_Q_BLOCK_TIME);
147
148 if (!QDF_IS_STATUS_SUCCESS(status))
149 hdd_err("Failed to start tx_flow_control_timer");
150 else
151 pAdapter->hdd_stats.hddTxRxStats.txflow_timer_cnt++;
152 }
153 return;
154}
155#else
156
157static inline void
158hdd_softap_tx_resume_false(hdd_adapter_t *pAdapter, bool tx_resume)
159{
160 return;
161}
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{
175 hdd_adapter_t *pAdapter = (hdd_adapter_t *) adapter_context;
176
177 if (!pAdapter) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530178 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800179 "%s: INV ARG", __func__);
180 /* INVALID ARG */
181 return;
182 }
183
184 /* Resume TX */
185 if (true == tx_resume) {
Anurag Chouhan210db072016-02-22 18:42:15 +0530186 if (QDF_TIMER_STATE_STOPPED !=
187 qdf_mc_timer_get_current_state(&pAdapter->
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800188 tx_flow_control_timer)) {
Anurag Chouhan210db072016-02-22 18:42:15 +0530189 qdf_mc_timer_stop(&pAdapter->tx_flow_control_timer);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800190 }
191
192 hddLog(LOG1, FL("Enabling queues"));
193 wlan_hdd_netif_queue_control(pAdapter,
194 WLAN_WAKE_ALL_NETIF_QUEUE,
195 WLAN_DATA_FLOW_CONTROL);
196 }
Poddar, Siddarthb61cf642016-04-28 16:02:39 +0530197 hdd_softap_tx_resume_false(pAdapter, tx_resume);
198
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800199 return;
200}
201#endif /* QCA_LL_LEGACY_TX_FLOW_CONTROL */
202
203/**
204 * hdd_softap_hard_start_xmit() - Transmit a frame
205 * @skb: pointer to OS packet (sk_buff)
206 * @dev: pointer to network device
207 *
208 * Function registered with the Linux OS for transmitting
209 * packets. This version of the function directly passes
210 * the packet to Transport Layer.
211 *
212 * Return: Always returns NETDEV_TX_OK
213 */
214int hdd_softap_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
215{
216 sme_ac_enum_type ac = SME_AC_BE;
217 hdd_adapter_t *pAdapter = (hdd_adapter_t *) netdev_priv(dev);
218 hdd_ap_ctx_t *pHddApCtx = WLAN_HDD_GET_AP_CTX_PTR(pAdapter);
Anurag Chouhan6d760662016-02-20 16:05:43 +0530219 struct qdf_mac_addr *pDestMacAddress;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800220 uint8_t STAId;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800221
222 ++pAdapter->hdd_stats.hddTxRxStats.txXmitCalled;
223 /* Prevent this function from being called during SSR since TL
224 * context may not be reinitialized at this time which may
225 * lead to a crash.
226 */
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800227 if (cds_is_driver_recovering()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530228 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800229 "%s: Recovery in Progress. Ignore!!!", __func__);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800230 goto drop_pkt;
231 }
232
233 /*
234 * If the device is operating on a DFS Channel
235 * then check if SAP is in CAC WAIT state and
236 * drop the packets. In CAC WAIT state device
237 * is expected not to transmit any frames.
238 * SAP starts Tx only after the BSS START is
239 * done.
240 */
241 if (pHddApCtx->dfs_cac_block_tx) {
242 goto drop_pkt;
243 }
244
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800245 /*
246 * If a transmit function is not registered, drop packet
247 */
248 if (!pAdapter->tx_fn) {
249 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO_HIGH,
250 "%s: TX function not registered by the data path",
251 __func__);
252 goto drop_pkt;
253 }
254
Nirav Shah5e74bb82016-07-20 16:01:27 +0530255 wlan_hdd_classify_pkt(skb);
256
Anurag Chouhan6d760662016-02-20 16:05:43 +0530257 pDestMacAddress = (struct qdf_mac_addr *) skb->data;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800258
Nirav Shah5e74bb82016-07-20 16:01:27 +0530259 if (QDF_NBUF_CB_GET_IS_BCAST(skb) ||
260 QDF_NBUF_CB_GET_IS_MCAST(skb)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800261 /* The BC/MC station ID is assigned during BSS
262 * starting phase. SAP will return the station ID
263 * used for BC/MC traffic.
264 */
265 STAId = pHddApCtx->uBCStaId;
266 } else {
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530267 if (QDF_STATUS_SUCCESS !=
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800268 hdd_softap_get_sta_id(pAdapter,
269 pDestMacAddress, &STAId)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530270 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
271 QDF_TRACE_LEVEL_WARN,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800272 "%s: Failed to find right station", __func__);
273 goto drop_pkt;
274 }
275
276 if (STAId == HDD_WLAN_INVALID_STA_ID) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530277 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
278 QDF_TRACE_LEVEL_WARN,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800279 "%s: Failed to find right station", __func__);
280 goto drop_pkt;
281 } else if (false == pAdapter->aStaInfo[STAId].isUsed) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530282 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
283 QDF_TRACE_LEVEL_WARN,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800284 "%s: STA %d is unregistered", __func__,
285 STAId);
286 goto drop_pkt;
287 }
288
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800289 if ((OL_TXRX_PEER_STATE_CONN !=
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800290 pAdapter->aStaInfo[STAId].tlSTAState)
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800291 && (OL_TXRX_PEER_STATE_AUTH !=
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800292 pAdapter->aStaInfo[STAId].tlSTAState)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530293 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
294 QDF_TRACE_LEVEL_WARN,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800295 "%s: Station not connected yet", __func__);
296 goto drop_pkt;
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800297 } else if (OL_TXRX_PEER_STATE_CONN ==
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800298 pAdapter->aStaInfo[STAId].tlSTAState) {
299 if (ntohs(skb->protocol) != HDD_ETHERTYPE_802_1_X) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530300 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
301 QDF_TRACE_LEVEL_WARN,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800302 "%s: NON-EAPOL packet in non-Authenticated state",
303 __func__);
304 goto drop_pkt;
305 }
306 }
307 }
308
309 hdd_get_tx_resource(pAdapter, STAId,
310 WLAN_SAP_HDD_TX_FLOW_CONTROL_OS_Q_BLOCK_TIME);
311
312 /* Get TL AC corresponding to Qdisc queue index/AC. */
313 ac = hdd_qdisc_ac_to_tl_ac[skb->queue_mapping];
314 ++pAdapter->hdd_stats.hddTxRxStats.txXmitClassifiedAC[ac];
315
316#if defined (IPA_OFFLOAD)
Nirav Shahcbc6d722016-03-01 16:24:53 +0530317 if (!qdf_nbuf_ipa_owned_get(skb)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800318#endif
319 /* Check if the buffer has enough header room */
320 skb = skb_unshare(skb, GFP_ATOMIC);
321 if (!skb)
Jeff Johnsonedeff232015-11-11 17:19:42 -0800322 goto drop_pkt_accounting;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800323#if defined (IPA_OFFLOAD)
324 }
325#endif
326
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800327 pAdapter->stats.tx_bytes += skb->len;
328 ++pAdapter->stats.tx_packets;
329
Nirav Shah5e74bb82016-07-20 16:01:27 +0530330 hdd_event_eapol_log(skb, QDF_TX);
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530331 qdf_dp_trace_log_pkt(pAdapter->sessionId, skb, QDF_TX);
Nirav Shahcbc6d722016-03-01 16:24:53 +0530332 QDF_NBUF_CB_TX_PACKET_TRACK(skb) = QDF_NBUF_TX_PKT_DATA_TRACK;
333 QDF_NBUF_UPDATE_TX_PKT_COUNT(skb, QDF_NBUF_TX_PKT_HDD);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800334
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530335 qdf_dp_trace_set_track(skb, QDF_TX);
336 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_PTR_RECORD,
337 (uint8_t *)&skb->data, sizeof(skb->data), QDF_TX));
338 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_RECORD,
339 (uint8_t *)skb->data, qdf_nbuf_len(skb), QDF_TX));
Nirav Shahcbc6d722016-03-01 16:24:53 +0530340 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530341 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_HDD_TX_PACKET_RECORD,
342 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
343 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE), QDF_TX));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800344
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800345 if (pAdapter->tx_fn(ol_txrx_get_vdev_by_sta_id(STAId),
346 (qdf_nbuf_t) skb) != NULL) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530347 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_WARN,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800348 "%s: Failed to send packet to txrx for staid:%d",
349 __func__, STAId);
350 ++pAdapter->hdd_stats.hddTxRxStats.txXmitDroppedAC[ac];
351 goto drop_pkt;
352 }
353 dev->trans_start = jiffies;
354
355 return NETDEV_TX_OK;
356
357drop_pkt:
358
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530359 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_DROP_PACKET_RECORD,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530360 (uint8_t *)skb->data, qdf_nbuf_len(skb), QDF_TX));
Nirav Shahcbc6d722016-03-01 16:24:53 +0530361 if (qdf_nbuf_len(skb) > QDF_DP_TRACE_RECORD_SIZE)
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530362 DPTRACE(qdf_dp_trace(skb, QDF_DP_TRACE_DROP_PACKET_RECORD,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530363 (uint8_t *)&skb->data[QDF_DP_TRACE_RECORD_SIZE],
364 (qdf_nbuf_len(skb)-QDF_DP_TRACE_RECORD_SIZE), QDF_TX));
Jeff Johnsonedeff232015-11-11 17:19:42 -0800365 kfree_skb(skb);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800366
Jeff Johnsonedeff232015-11-11 17:19:42 -0800367drop_pkt_accounting:
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800368 ++pAdapter->stats.tx_dropped;
369 ++pAdapter->hdd_stats.hddTxRxStats.txXmitDropped;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800370
371 return NETDEV_TX_OK;
372}
373
374/**
375 * __hdd_softap_tx_timeout() - TX timeout handler
376 * @dev: pointer to network device
377 *
378 * This function is registered as a netdev ndo_tx_timeout method, and
379 * is invoked by the kernel if the driver takes too long to transmit a
380 * frame.
381 *
382 * Return: None
383 */
384static void __hdd_softap_tx_timeout(struct net_device *dev)
385{
386 hdd_adapter_t *adapter = WLAN_HDD_GET_PRIV_PTR(dev);
387 hdd_context_t *hdd_ctx;
Nirav Shah89223f72016-03-01 18:10:38 +0530388 struct netdev_queue *txq;
389 int i;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800390
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530391 DPTRACE(qdf_dp_trace(NULL, QDF_DP_TRACE_HDD_SOFTAP_TX_TIMEOUT,
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530392 NULL, 0, QDF_TX));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800393 /* Getting here implies we disabled the TX queues for too
394 * long. Queues are disabled either because of disassociation
395 * or low resource scenarios. In case of disassociation it is
396 * ok to ignore this. But if associated, we have do possible
397 * recovery here
398 */
399 hdd_ctx = WLAN_HDD_GET_CTX(adapter);
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800400 if (cds_is_driver_recovering()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530401 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800402 "%s: Recovery in Progress. Ignore!!!", __func__);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800403 return;
404 }
Nirav Shah89223f72016-03-01 18:10:38 +0530405
406 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
407 "%s: Transmission timeout occurred jiffies %lu trans_start %lu"
408 , __func__, jiffies, dev->trans_start);
409
410 for (i = 0; i < NUM_TX_QUEUES; i++) {
411 txq = netdev_get_tx_queue(dev, i);
412 QDF_TRACE(QDF_MODULE_ID_HDD_DATA,
413 QDF_TRACE_LEVEL_ERROR,
414 "Queue%d status: %d txq->trans_start %lu",
415 i, netif_tx_queue_stopped(txq), txq->trans_start);
416 }
417
418 wlan_hdd_display_netif_queue_history(hdd_ctx);
419 ol_tx_dump_flow_pool_info();
420 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_ERROR,
421 "carrier state: %d", netif_carrier_ok(dev));
422
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800423}
424
425/**
426 * hdd_softap_tx_timeout() - SSR wrapper for __hdd_softap_tx_timeout
427 * @dev: pointer to net_device
428 *
429 * Return: none
430 */
431void hdd_softap_tx_timeout(struct net_device *dev)
432{
433 cds_ssr_protect(__func__);
434 __hdd_softap_tx_timeout(dev);
435 cds_ssr_unprotect(__func__);
436}
437
438/**
439 * @hdd_softap_init_tx_rx() - Initialize Tx/RX module
440 * @pAdapter: pointer to adapter context
441 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530442 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
443 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800444 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530445QDF_STATUS hdd_softap_init_tx_rx(hdd_adapter_t *pAdapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800446{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530447 QDF_STATUS status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800448
449 uint8_t STAId = 0;
450
Anurag Chouhan600c3a02016-03-01 10:33:54 +0530451 qdf_mem_zero(&pAdapter->stats, sizeof(struct net_device_stats));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800452
453 spin_lock_init(&pAdapter->staInfo_lock);
454
455 for (STAId = 0; STAId < WLAN_MAX_STA_COUNT; STAId++) {
Anurag Chouhan600c3a02016-03-01 10:33:54 +0530456 qdf_mem_zero(&pAdapter->aStaInfo[STAId],
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800457 sizeof(hdd_station_info_t));
458 }
459
460 return status;
461}
462
463/**
464 * @hdd_softap_deinit_tx_rx() - Deinitialize Tx/RX module
465 * @pAdapter: pointer to adapter context
466 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530467 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
468 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800469 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530470QDF_STATUS hdd_softap_deinit_tx_rx(hdd_adapter_t *pAdapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800471{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530472 QDF_STATUS status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800473
474 return status;
475}
476
477/**
478 * hdd_softap_init_tx_rx_sta() - Initialize tx/rx for a softap station
479 * @pAdapter: pointer to adapter context
480 * @STAId: Station ID to initialize
481 * @pmacAddrSTA: pointer to the MAC address of the station
482 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530483 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
484 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800485 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530486QDF_STATUS hdd_softap_init_tx_rx_sta(hdd_adapter_t *pAdapter, uint8_t STAId,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530487 struct qdf_mac_addr *pmacAddrSTA)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800488{
489 spin_lock_bh(&pAdapter->staInfo_lock);
490 if (pAdapter->aStaInfo[STAId].isUsed) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530491 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800492 "%s: Reinit station %d", __func__, STAId);
493 spin_unlock_bh(&pAdapter->staInfo_lock);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530494 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800495 }
496
Anurag Chouhan600c3a02016-03-01 10:33:54 +0530497 qdf_mem_zero(&pAdapter->aStaInfo[STAId], sizeof(hdd_station_info_t));
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800498
499 pAdapter->aStaInfo[STAId].isUsed = true;
500 pAdapter->aStaInfo[STAId].isDeauthInProgress = false;
Anurag Chouhanc5548422016-02-24 18:33:27 +0530501 qdf_copy_macaddr(&pAdapter->aStaInfo[STAId].macAddrSTA, pmacAddrSTA);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800502
503 spin_unlock_bh(&pAdapter->staInfo_lock);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530504 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800505}
506
507/**
508 * hdd_softap_deinit_tx_rx_sta() - Deinitialize tx/rx for a softap station
509 * @pAdapter: pointer to adapter context
510 * @STAId: Station ID to deinitialize
511 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530512 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
513 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800514 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530515QDF_STATUS hdd_softap_deinit_tx_rx_sta(hdd_adapter_t *pAdapter, uint8_t STAId)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800516{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530517 QDF_STATUS status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800518 hdd_hostapd_state_t *pHostapdState;
519
520 pHostapdState = WLAN_HDD_GET_HOSTAP_STATE_PTR(pAdapter);
521
522 spin_lock_bh(&pAdapter->staInfo_lock);
523
524 if (false == pAdapter->aStaInfo[STAId].isUsed) {
525 spin_unlock_bh(&pAdapter->staInfo_lock);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530526 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800527 "%s: Deinit station not inited %d", __func__, STAId);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530528 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800529 }
530
531 pAdapter->aStaInfo[STAId].isUsed = false;
532 pAdapter->aStaInfo[STAId].isDeauthInProgress = false;
533
534 spin_unlock_bh(&pAdapter->staInfo_lock);
535 return status;
536}
537
538/**
539 * hdd_softap_rx_packet_cbk() - Receive packet handler
Dhanashri Atre182b0272016-02-17 15:35:07 -0800540 * @context: pointer to HDD context
Nirav Shahcbc6d722016-03-01 16:24:53 +0530541 * @rxBuf: pointer to rx qdf_nbuf
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800542 *
543 * Receive callback registered with TL. TL will call this to notify
544 * the HDD when one or more packets were received for a registered
545 * STA.
546 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530547 * Return: QDF_STATUS_E_FAILURE if any errors encountered,
548 * QDF_STATUS_SUCCESS otherwise
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800549 */
Dhanashri Atre182b0272016-02-17 15:35:07 -0800550QDF_STATUS hdd_softap_rx_packet_cbk(void *context, qdf_nbuf_t rxBuf)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800551{
552 hdd_adapter_t *pAdapter = NULL;
553 int rxstat;
554 unsigned int cpu_index;
555 struct sk_buff *skb = NULL;
556 hdd_context_t *pHddCtx = NULL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800557
558 /* Sanity check on inputs */
Dhanashri Atre182b0272016-02-17 15:35:07 -0800559 if (unlikely((NULL == context) || (NULL == rxBuf))) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530560 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800561 "%s: Null params being passed", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530562 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800563 }
564
Dhanashri Atre182b0272016-02-17 15:35:07 -0800565 pAdapter = (hdd_adapter_t *)context;
566 if (unlikely(WLAN_HDD_ADAPTER_MAGIC != pAdapter->magic)) {
567 QDF_TRACE(QDF_MODULE_ID_HDD_DATA, QDF_TRACE_LEVEL_FATAL,
568 "Magic cookie(%x) for adapter sanity verification is invalid",
569 pAdapter->magic);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530570 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800571 }
572
Dhanashri Atre182b0272016-02-17 15:35:07 -0800573 pHddCtx = pAdapter->pHddCtx;
574 if (unlikely(NULL == pHddCtx)) {
575 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
576 "%s: HDD context is Null", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530577 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800578 }
579
580 /* walk the chain until all are processed */
581 skb = (struct sk_buff *)rxBuf;
582
583 hdd_softap_dump_sk_buff(skb);
584
585 skb->dev = pAdapter->dev;
586
Dhanashri Atre182b0272016-02-17 15:35:07 -0800587 if (unlikely(skb->dev == NULL)) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800588
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530589 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800590 "%s: ERROR!!Invalid netdevice", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530591 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800592 }
593 cpu_index = wlan_hdd_get_cpu();
594 ++pAdapter->hdd_stats.hddTxRxStats.rxPackets[cpu_index];
595 ++pAdapter->stats.rx_packets;
596 pAdapter->stats.rx_bytes += skb->len;
597
Nirav Shah5e74bb82016-07-20 16:01:27 +0530598 hdd_event_eapol_log(skb, QDF_RX);
Nirav Shah0d58a7e2016-04-26 22:54:12 +0530599 DPTRACE(qdf_dp_trace(rxBuf,
600 QDF_DP_TRACE_RX_HDD_PACKET_PTR_RECORD,
601 qdf_nbuf_data_addr(rxBuf),
602 sizeof(qdf_nbuf_data(rxBuf)), QDF_RX));
603
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800604 skb->protocol = eth_type_trans(skb, skb->dev);
605#ifdef WLAN_FEATURE_HOLD_RX_WAKELOCK
Anurag Chouhan01cfa4e2016-09-04 15:10:49 +0530606 cds_host_diag_log_work(&pHddCtx->rx_wake_lock,
607 HDD_WAKE_LOCK_DURATION,
608 WIFI_POWER_EVENT_WAKELOCK_HOLD_RX);
Anurag Chouhana37b5b72016-02-21 14:53:42 +0530609 qdf_wake_lock_timeout_acquire(&pHddCtx->rx_wake_lock,
Anurag Chouhan01cfa4e2016-09-04 15:10:49 +0530610 HDD_WAKE_LOCK_DURATION);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800611#endif
612
613 /* Remove SKB from internal tracking table before submitting
614 * it to stack
615 */
Nirav Shahcbc6d722016-03-01 16:24:53 +0530616 qdf_net_buf_debug_release_skb(rxBuf);
Nirav Shahbd36b062016-07-18 11:12:59 +0530617 if (hdd_napi_enabled(HDD_NAPI_ANY) &&
618 !pHddCtx->enableRxThread)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800619 rxstat = netif_receive_skb(skb);
620 else
621 rxstat = netif_rx_ni(skb);
622 if (NET_RX_SUCCESS == rxstat) {
623 ++pAdapter->hdd_stats.hddTxRxStats.rxDelivered[cpu_index];
624 } else {
625 ++pAdapter->hdd_stats.hddTxRxStats.rxRefused[cpu_index];
626 }
627
628 pAdapter->dev->last_rx = jiffies;
629
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530630 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800631}
632
633/**
634 * hdd_softap_deregister_sta(hdd_adapter_t *pAdapter, uint8_t staId)
635 * @pAdapter: pointer to adapter context
636 * @staId: Station ID to deregister
637 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530638 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800639 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530640QDF_STATUS hdd_softap_deregister_sta(hdd_adapter_t *pAdapter, uint8_t staId)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800641{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530642 QDF_STATUS qdf_status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800643 hdd_context_t *pHddCtx;
644
645 if (NULL == pAdapter) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530646 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800647 "%s: pAdapter is NULL", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530648 return QDF_STATUS_E_INVAL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800649 }
650
651 if (WLAN_HDD_ADAPTER_MAGIC != pAdapter->magic) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530652 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800653 "%s: Invalid pAdapter magic", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530654 return QDF_STATUS_E_INVAL;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800655 }
656
657 pHddCtx = (hdd_context_t *) (pAdapter->pHddCtx);
658 /* Clear station in TL and then update HDD data
659 * structures. This helps to block RX frames from other
660 * station to this station.
661 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530662 qdf_status = ol_txrx_clear_peer(staId);
663 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530664 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800665 "ol_txrx_clear_peer() failed to for staID %d. "
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530666 "Status= %d [0x%08X]", staId, qdf_status, qdf_status);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800667 }
668
669 if (pAdapter->aStaInfo[staId].isUsed) {
670 spin_lock_bh(&pAdapter->staInfo_lock);
Anurag Chouhan600c3a02016-03-01 10:33:54 +0530671 qdf_mem_zero(&pAdapter->aStaInfo[staId],
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800672 sizeof(hdd_station_info_t));
673 spin_unlock_bh(&pAdapter->staInfo_lock);
674 }
675 pHddCtx->sta_to_adapter[staId] = NULL;
676
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530677 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800678}
679
680/**
681 * hdd_softap_register_sta() - Register a SoftAP STA
682 * @pAdapter: pointer to adapter context
683 * @fAuthRequired: is additional authentication required?
684 * @fPrivacyBit: should 802.11 privacy bit be set?
685 * @staId: station ID assigned to this station
686 * @ucastSig: unicast security signature
687 * @bcastSig: broadcast security signature
688 * @pPeerMacAddress: station MAC address
689 * @fWmmEnabled: is WMM enabled for this STA?
690 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530691 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800692 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530693QDF_STATUS hdd_softap_register_sta(hdd_adapter_t *pAdapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800694 bool fAuthRequired,
695 bool fPrivacyBit,
696 uint8_t staId,
697 uint8_t ucastSig,
698 uint8_t bcastSig,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530699 struct qdf_mac_addr *pPeerMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800700 bool fWmmEnabled)
701{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530702 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800703 struct ol_txrx_desc_type staDesc = { 0 };
704 hdd_context_t *pHddCtx = pAdapter->pHddCtx;
Dhanashri Atre182b0272016-02-17 15:35:07 -0800705 struct ol_txrx_ops txrx_ops;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800706
707 /*
708 * Clean up old entry if it is not cleaned up properly
709 */
710 if (pAdapter->aStaInfo[staId].isUsed) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530711 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800712 "clean up old entry for STA %d", staId);
713 hdd_softap_deregister_sta(pAdapter, staId);
714 }
715 /* Get the Station ID from the one saved during the assocation. */
716
717 staDesc.sta_id = staId;
718
719 /*Save the pAdapter Pointer for this staId */
720 pHddCtx->sta_to_adapter[staId] = pAdapter;
721
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530722 qdf_status =
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800723 hdd_softap_init_tx_rx_sta(pAdapter, staId,
724 pPeerMacAddress);
725
726 staDesc.is_qos_enabled = fWmmEnabled;
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530727 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800728 "HDD SOFTAP register TL QoS_enabled=%d",
729 staDesc.is_qos_enabled);
730
Dhanashri Atre182b0272016-02-17 15:35:07 -0800731 /* Register the vdev transmit and receive functions */
732 qdf_mem_zero(&txrx_ops, sizeof(txrx_ops));
733 txrx_ops.rx.rx = hdd_softap_rx_packet_cbk;
734 ol_txrx_vdev_register(
735 ol_txrx_get_vdev_from_vdev_id(pAdapter->sessionId),
736 pAdapter, &txrx_ops);
Dhanashri Atre168d2b42016-02-22 14:43:06 -0800737 pAdapter->tx_fn = txrx_ops.tx.tx;
Dhanashri Atre182b0272016-02-17 15:35:07 -0800738
Dhanashri Atre50141c52016-04-07 13:15:29 -0700739 qdf_status = ol_txrx_register_peer(&staDesc);
740 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
741 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
742 "SOFTAP ol_txrx_register_peer() failed to register. Status= %d [0x%08X]",
743 qdf_status, qdf_status);
744 return qdf_status;
745 }
746
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800747 /* if ( WPA ), tell TL to go to 'connected' and after keys come to the
748 * driver then go to 'authenticated'. For all other authentication
749 * types (those that do not require upper layer authentication) we can
750 * put TL directly into 'authenticated' state
751 */
752
753 pAdapter->aStaInfo[staId].ucSTAId = staId;
754 pAdapter->aStaInfo[staId].isQosEnabled = fWmmEnabled;
755
756 if (!fAuthRequired) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530757 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800758 "open/shared auth StaId= %d. Changing TL state to AUTHENTICATED at Join time",
759 pAdapter->aStaInfo[staId].ucSTAId);
760
761 /* Connections that do not need Upper layer auth,
762 * transition TL directly to 'Authenticated' state.
763 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530764 qdf_status = hdd_change_peer_state(pAdapter, staDesc.sta_id,
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800765 OL_TXRX_PEER_STATE_AUTH, false);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800766
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800767 pAdapter->aStaInfo[staId].tlSTAState = OL_TXRX_PEER_STATE_AUTH;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800768 pAdapter->sessionCtx.ap.uIsAuthenticated = true;
769 } else {
770
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530771 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800772 "ULA auth StaId= %d. Changing TL state to CONNECTED at Join time",
773 pAdapter->aStaInfo[staId].ucSTAId);
774
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530775 qdf_status = hdd_change_peer_state(pAdapter, staDesc.sta_id,
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800776 OL_TXRX_PEER_STATE_CONN, false);
777 pAdapter->aStaInfo[staId].tlSTAState = OL_TXRX_PEER_STATE_CONN;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800778
779 pAdapter->sessionCtx.ap.uIsAuthenticated = false;
780
781 }
782
783 /* Enable Tx queue */
784 hddLog(LOG1, FL("Enabling queues"));
785 wlan_hdd_netif_queue_control(pAdapter,
786 WLAN_START_ALL_NETIF_QUEUE_N_CARRIER,
787 WLAN_CONTROL_PATH);
788
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530789 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800790}
791
792/**
793 * hdd_softap_register_bc_sta() - Register the SoftAP broadcast STA
794 * @pAdapter: pointer to adapter context
795 * @fPrivacyBit: should 802.11 privacy bit be set?
796 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530797 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800798 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530799QDF_STATUS hdd_softap_register_bc_sta(hdd_adapter_t *pAdapter,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800800 bool fPrivacyBit)
801{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530802 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800803 hdd_context_t *pHddCtx = WLAN_HDD_GET_CTX(pAdapter);
Anurag Chouhan6d760662016-02-20 16:05:43 +0530804 struct qdf_mac_addr broadcastMacAddr =
805 QDF_MAC_ADDR_BROADCAST_INITIALIZER;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800806 hdd_ap_ctx_t *pHddApCtx;
807
808 pHddApCtx = WLAN_HDD_GET_AP_CTX_PTR(pAdapter);
809
810 pHddCtx->sta_to_adapter[WLAN_RX_BCMC_STA_ID] = pAdapter;
811#ifdef WLAN_FEATURE_MBSSID
812 pHddCtx->sta_to_adapter[pHddApCtx->uBCStaId] = pAdapter;
813#else
814 pHddCtx->sta_to_adapter[WLAN_RX_SAP_SELF_STA_ID] = pAdapter;
815#endif
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530816 qdf_status =
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800817 hdd_softap_register_sta(pAdapter, false, fPrivacyBit,
818 (WLAN_HDD_GET_AP_CTX_PTR(pAdapter))->
819 uBCStaId, 0, 1, &broadcastMacAddr, 0);
820
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530821 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800822}
823
824/**
825 * hdd_softap_deregister_bc_sta() - Deregister the SoftAP broadcast STA
826 * @pAdapter: pointer to adapter context
827 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530828 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800829 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530830QDF_STATUS hdd_softap_deregister_bc_sta(hdd_adapter_t *pAdapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800831{
832 return hdd_softap_deregister_sta(pAdapter,
833 (WLAN_HDD_GET_AP_CTX_PTR(pAdapter))->
834 uBCStaId);
835}
836
837/**
838 * hdd_softap_stop_bss() - Stop the BSS
839 * @pAdapter: pointer to adapter context
840 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530841 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800842 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530843QDF_STATUS hdd_softap_stop_bss(hdd_adapter_t *pAdapter)
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800844{
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530845 QDF_STATUS qdf_status = QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800846 uint8_t staId = 0;
847 hdd_context_t *pHddCtx;
848 pHddCtx = WLAN_HDD_GET_CTX(pAdapter);
849
850 /* bss deregister is not allowed during wlan driver loading or
851 * unloading
852 */
Rajeev Kumarfec3dbe2016-01-19 15:23:52 -0800853 if (cds_is_load_or_unload_in_progress()) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530854 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prashanth Bhatta9e143052015-12-04 11:56:47 -0800855 "%s: Loading_unloading in Progress, state: 0x%x. Ignore!!!",
856 __func__, cds_get_driver_state());
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530857 return QDF_STATUS_E_PERM;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800858 }
859
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530860 qdf_status = hdd_softap_deregister_bc_sta(pAdapter);
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800861
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530862 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530863 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800864 "%s: Failed to deregister BC sta Id %d", __func__,
865 (WLAN_HDD_GET_AP_CTX_PTR(pAdapter))->uBCStaId);
866 }
867
868 for (staId = 0; staId < WLAN_MAX_STA_COUNT; staId++) {
869 /* This excludes BC sta as it is already deregistered */
870 if (pAdapter->aStaInfo[staId].isUsed) {
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530871 qdf_status = hdd_softap_deregister_sta(pAdapter, staId);
872 if (!QDF_IS_STATUS_SUCCESS(qdf_status)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530873 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA,
874 QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800875 "%s: Failed to deregister sta Id %d",
876 __func__, staId);
877 }
878 }
879 }
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530880 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800881}
882
883/**
884 * hdd_softap_change_sta_state() - Change the state of a SoftAP station
885 * @pAdapter: pointer to adapter context
886 * @pDestMacAddress: MAC address of the station
887 * @state: new state of the station
888 *
Anurag Chouhanf04e84f2016-03-03 10:12:12 +0530889 * Return: QDF_STATUS_SUCCESS on success, QDF_STATUS_E_** on error
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800890 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530891QDF_STATUS hdd_softap_change_sta_state(hdd_adapter_t *pAdapter,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530892 struct qdf_mac_addr *pDestMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800893 enum ol_txrx_peer_state state)
894{
895 uint8_t ucSTAId = WLAN_MAX_STA_COUNT;
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530896 QDF_STATUS qdf_status = QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800897
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530898 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800899 "%s: enter", __func__);
900
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530901 if (QDF_STATUS_SUCCESS !=
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800902 hdd_softap_get_sta_id(pAdapter,
903 pDestMacAddress, &ucSTAId)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530904 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800905 "%s: Failed to find right station", __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530906 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800907 }
908
909 if (false ==
Anurag Chouhanc5548422016-02-24 18:33:27 +0530910 qdf_is_macaddr_equal(&pAdapter->aStaInfo[ucSTAId].macAddrSTA,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800911 pDestMacAddress)) {
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530912 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_ERROR,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800913 "%s: Station MAC address does not matching",
914 __func__);
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530915 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800916 }
917
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530918 qdf_status =
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800919 hdd_change_peer_state(pAdapter, ucSTAId, state, false);
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530920 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800921 "%s: change station to state %d succeed", __func__, state);
922
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530923 if (QDF_STATUS_SUCCESS == qdf_status) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800924 pAdapter->aStaInfo[ucSTAId].tlSTAState =
Dhanashri Atreb08959a2016-03-01 17:28:03 -0800925 OL_TXRX_PEER_STATE_AUTH;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800926 }
927
Anurag Chouhanb2dc16f2016-02-25 11:47:37 +0530928 QDF_TRACE(QDF_MODULE_ID_HDD_SAP_DATA, QDF_TRACE_LEVEL_INFO,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800929 "%s exit", __func__);
930
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530931 return qdf_status;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800932}
933
934/*
935 * hdd_softap_get_sta_id() - Find station ID from MAC address
936 * @pAdapter: pointer to adapter context
937 * @pDestMacAddress: MAC address of the destination
938 * @staId: Station ID associated with the MAC address
939 *
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530940 * Return: QDF_STATUS_SUCCESS if a match was found, in which case
941 * staId is populated, QDF_STATUS_E_FAILURE if a match is
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800942 * not found
943 */
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530944QDF_STATUS hdd_softap_get_sta_id(hdd_adapter_t *pAdapter,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530945 struct qdf_mac_addr *pMacAddress,
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800946 uint8_t *staId)
947{
948 uint8_t i;
949
950 for (i = 0; i < WLAN_MAX_STA_COUNT; i++) {
Anurag Chouhan600c3a02016-03-01 10:33:54 +0530951 if (!qdf_mem_cmp
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800952 (&pAdapter->aStaInfo[i].macAddrSTA, pMacAddress,
Anurag Chouhan6d760662016-02-20 16:05:43 +0530953 QDF_MAC_ADDR_SIZE) && pAdapter->aStaInfo[i].isUsed) {
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800954 *staId = i;
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530955 return QDF_STATUS_SUCCESS;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800956 }
957 }
958
Anurag Chouhanfb54ab02016-02-18 18:00:46 +0530959 return QDF_STATUS_E_FAILURE;
Prakash Dhavali7090c5f2015-11-02 17:55:19 -0800960}