2 * NXP Wireless LAN device driver: utility functions
4 * Copyright 2011-2020 NXP
6 * This software file (the "File") is distributed by NXP
7 * under the terms of the GNU General Public License Version 2, June 1991
8 * (the "License"). You may use, redistribute and/or modify this File in
9 * accordance with the terms and conditions of the License, a copy of which
10 * is available by writing to the Free Software Foundation, Inc.,
11 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA or on the
12 * worldwide web at http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt.
14 * THE FILE IS DISTRIBUTED AS-IS, WITHOUT WARRANTY OF ANY KIND, AND THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE
16 * ARE EXPRESSLY DISCLAIMED. The License provides additional details about
17 * this warranty disclaimer.
28 static struct mwifiex_debug_data items
[] = {
29 {"debug_mask", item_size(debug_mask
),
30 item_addr(debug_mask
), 1},
31 {"int_counter", item_size(int_counter
),
32 item_addr(int_counter
), 1},
33 {"wmm_ac_vo", item_size(packets_out
[WMM_AC_VO
]),
34 item_addr(packets_out
[WMM_AC_VO
]), 1},
35 {"wmm_ac_vi", item_size(packets_out
[WMM_AC_VI
]),
36 item_addr(packets_out
[WMM_AC_VI
]), 1},
37 {"wmm_ac_be", item_size(packets_out
[WMM_AC_BE
]),
38 item_addr(packets_out
[WMM_AC_BE
]), 1},
39 {"wmm_ac_bk", item_size(packets_out
[WMM_AC_BK
]),
40 item_addr(packets_out
[WMM_AC_BK
]), 1},
41 {"tx_buf_size", item_size(tx_buf_size
),
42 item_addr(tx_buf_size
), 1},
43 {"curr_tx_buf_size", item_size(curr_tx_buf_size
),
44 item_addr(curr_tx_buf_size
), 1},
45 {"ps_mode", item_size(ps_mode
),
46 item_addr(ps_mode
), 1},
47 {"ps_state", item_size(ps_state
),
48 item_addr(ps_state
), 1},
49 {"is_deep_sleep", item_size(is_deep_sleep
),
50 item_addr(is_deep_sleep
), 1},
51 {"wakeup_dev_req", item_size(pm_wakeup_card_req
),
52 item_addr(pm_wakeup_card_req
), 1},
53 {"wakeup_tries", item_size(pm_wakeup_fw_try
),
54 item_addr(pm_wakeup_fw_try
), 1},
55 {"hs_configured", item_size(is_hs_configured
),
56 item_addr(is_hs_configured
), 1},
57 {"hs_activated", item_size(hs_activated
),
58 item_addr(hs_activated
), 1},
59 {"num_tx_timeout", item_size(num_tx_timeout
),
60 item_addr(num_tx_timeout
), 1},
61 {"is_cmd_timedout", item_size(is_cmd_timedout
),
62 item_addr(is_cmd_timedout
), 1},
63 {"timeout_cmd_id", item_size(timeout_cmd_id
),
64 item_addr(timeout_cmd_id
), 1},
65 {"timeout_cmd_act", item_size(timeout_cmd_act
),
66 item_addr(timeout_cmd_act
), 1},
67 {"last_cmd_id", item_size(last_cmd_id
),
68 item_addr(last_cmd_id
), DBG_CMD_NUM
},
69 {"last_cmd_act", item_size(last_cmd_act
),
70 item_addr(last_cmd_act
), DBG_CMD_NUM
},
71 {"last_cmd_index", item_size(last_cmd_index
),
72 item_addr(last_cmd_index
), 1},
73 {"last_cmd_resp_id", item_size(last_cmd_resp_id
),
74 item_addr(last_cmd_resp_id
), DBG_CMD_NUM
},
75 {"last_cmd_resp_index", item_size(last_cmd_resp_index
),
76 item_addr(last_cmd_resp_index
), 1},
77 {"last_event", item_size(last_event
),
78 item_addr(last_event
), DBG_CMD_NUM
},
79 {"last_event_index", item_size(last_event_index
),
80 item_addr(last_event_index
), 1},
81 {"last_mp_wr_bitmap", item_size(last_mp_wr_bitmap
),
82 item_addr(last_mp_wr_bitmap
), MWIFIEX_DBG_SDIO_MP_NUM
},
83 {"last_mp_wr_ports", item_size(last_mp_wr_ports
),
84 item_addr(last_mp_wr_ports
), MWIFIEX_DBG_SDIO_MP_NUM
},
85 {"last_mp_wr_len", item_size(last_mp_wr_len
),
86 item_addr(last_mp_wr_len
), MWIFIEX_DBG_SDIO_MP_NUM
},
87 {"last_mp_curr_wr_port", item_size(last_mp_curr_wr_port
),
88 item_addr(last_mp_curr_wr_port
), MWIFIEX_DBG_SDIO_MP_NUM
},
89 {"last_sdio_mp_index", item_size(last_sdio_mp_index
),
90 item_addr(last_sdio_mp_index
), 1},
91 {"num_cmd_h2c_fail", item_size(num_cmd_host_to_card_failure
),
92 item_addr(num_cmd_host_to_card_failure
), 1},
93 {"num_cmd_sleep_cfm_fail",
94 item_size(num_cmd_sleep_cfm_host_to_card_failure
),
95 item_addr(num_cmd_sleep_cfm_host_to_card_failure
), 1},
96 {"num_tx_h2c_fail", item_size(num_tx_host_to_card_failure
),
97 item_addr(num_tx_host_to_card_failure
), 1},
98 {"num_evt_deauth", item_size(num_event_deauth
),
99 item_addr(num_event_deauth
), 1},
100 {"num_evt_disassoc", item_size(num_event_disassoc
),
101 item_addr(num_event_disassoc
), 1},
102 {"num_evt_link_lost", item_size(num_event_link_lost
),
103 item_addr(num_event_link_lost
), 1},
104 {"num_cmd_deauth", item_size(num_cmd_deauth
),
105 item_addr(num_cmd_deauth
), 1},
106 {"num_cmd_assoc_ok", item_size(num_cmd_assoc_success
),
107 item_addr(num_cmd_assoc_success
), 1},
108 {"num_cmd_assoc_fail", item_size(num_cmd_assoc_failure
),
109 item_addr(num_cmd_assoc_failure
), 1},
110 {"cmd_sent", item_size(cmd_sent
),
111 item_addr(cmd_sent
), 1},
112 {"data_sent", item_size(data_sent
),
113 item_addr(data_sent
), 1},
114 {"cmd_resp_received", item_size(cmd_resp_received
),
115 item_addr(cmd_resp_received
), 1},
116 {"event_received", item_size(event_received
),
117 item_addr(event_received
), 1},
119 /* variables defined in struct mwifiex_adapter */
120 {"cmd_pending", adapter_item_size(cmd_pending
),
121 adapter_item_addr(cmd_pending
), 1},
122 {"tx_pending", adapter_item_size(tx_pending
),
123 adapter_item_addr(tx_pending
), 1},
124 {"rx_pending", adapter_item_size(rx_pending
),
125 adapter_item_addr(rx_pending
), 1},
128 static int num_of_items
= ARRAY_SIZE(items
);
131 * Firmware initialization complete callback handler.
133 * This function wakes up the function waiting on the init
134 * wait queue for the firmware initialization to complete.
136 int mwifiex_init_fw_complete(struct mwifiex_adapter
*adapter
)
139 if (adapter
->hw_status
== MWIFIEX_HW_STATUS_READY
)
140 if (adapter
->if_ops
.init_fw_port
)
141 adapter
->if_ops
.init_fw_port(adapter
);
143 adapter
->init_wait_q_woken
= true;
144 wake_up_interruptible(&adapter
->init_wait_q
);
149 * This function sends init/shutdown command
152 int mwifiex_init_shutdown_fw(struct mwifiex_private
*priv
,
153 u32 func_init_shutdown
)
157 if (func_init_shutdown
== MWIFIEX_FUNC_INIT
) {
158 cmd
= HostCmd_CMD_FUNC_INIT
;
159 } else if (func_init_shutdown
== MWIFIEX_FUNC_SHUTDOWN
) {
160 cmd
= HostCmd_CMD_FUNC_SHUTDOWN
;
162 mwifiex_dbg(priv
->adapter
, ERROR
,
163 "unsupported parameter\n");
167 return mwifiex_send_cmd(priv
, cmd
, HostCmd_ACT_GEN_SET
, 0, NULL
, true);
169 EXPORT_SYMBOL_GPL(mwifiex_init_shutdown_fw
);
172 * IOCTL request handler to set/get debug information.
174 * This function collates/sets the information from/to different driver
177 int mwifiex_get_debug_info(struct mwifiex_private
*priv
,
178 struct mwifiex_debug_info
*info
)
180 struct mwifiex_adapter
*adapter
= priv
->adapter
;
183 info
->debug_mask
= adapter
->debug_mask
;
184 memcpy(info
->packets_out
,
185 priv
->wmm
.packets_out
,
186 sizeof(priv
->wmm
.packets_out
));
187 info
->curr_tx_buf_size
= (u32
) adapter
->curr_tx_buf_size
;
188 info
->tx_buf_size
= (u32
) adapter
->tx_buf_size
;
189 info
->rx_tbl_num
= mwifiex_get_rx_reorder_tbl(priv
,
191 info
->tx_tbl_num
= mwifiex_get_tx_ba_stream_tbl(priv
,
193 info
->tdls_peer_num
= mwifiex_get_tdls_list(priv
,
195 info
->ps_mode
= adapter
->ps_mode
;
196 info
->ps_state
= adapter
->ps_state
;
197 info
->is_deep_sleep
= adapter
->is_deep_sleep
;
198 info
->pm_wakeup_card_req
= adapter
->pm_wakeup_card_req
;
199 info
->pm_wakeup_fw_try
= adapter
->pm_wakeup_fw_try
;
200 info
->is_hs_configured
= test_bit(MWIFIEX_IS_HS_CONFIGURED
,
201 &adapter
->work_flags
);
202 info
->hs_activated
= adapter
->hs_activated
;
203 info
->is_cmd_timedout
= test_bit(MWIFIEX_IS_CMD_TIMEDOUT
,
204 &adapter
->work_flags
);
205 info
->num_cmd_host_to_card_failure
206 = adapter
->dbg
.num_cmd_host_to_card_failure
;
207 info
->num_cmd_sleep_cfm_host_to_card_failure
208 = adapter
->dbg
.num_cmd_sleep_cfm_host_to_card_failure
;
209 info
->num_tx_host_to_card_failure
210 = adapter
->dbg
.num_tx_host_to_card_failure
;
211 info
->num_event_deauth
= adapter
->dbg
.num_event_deauth
;
212 info
->num_event_disassoc
= adapter
->dbg
.num_event_disassoc
;
213 info
->num_event_link_lost
= adapter
->dbg
.num_event_link_lost
;
214 info
->num_cmd_deauth
= adapter
->dbg
.num_cmd_deauth
;
215 info
->num_cmd_assoc_success
=
216 adapter
->dbg
.num_cmd_assoc_success
;
217 info
->num_cmd_assoc_failure
=
218 adapter
->dbg
.num_cmd_assoc_failure
;
219 info
->num_tx_timeout
= adapter
->dbg
.num_tx_timeout
;
220 info
->timeout_cmd_id
= adapter
->dbg
.timeout_cmd_id
;
221 info
->timeout_cmd_act
= adapter
->dbg
.timeout_cmd_act
;
222 memcpy(info
->last_cmd_id
, adapter
->dbg
.last_cmd_id
,
223 sizeof(adapter
->dbg
.last_cmd_id
));
224 memcpy(info
->last_cmd_act
, adapter
->dbg
.last_cmd_act
,
225 sizeof(adapter
->dbg
.last_cmd_act
));
226 info
->last_cmd_index
= adapter
->dbg
.last_cmd_index
;
227 memcpy(info
->last_cmd_resp_id
, adapter
->dbg
.last_cmd_resp_id
,
228 sizeof(adapter
->dbg
.last_cmd_resp_id
));
229 info
->last_cmd_resp_index
= adapter
->dbg
.last_cmd_resp_index
;
230 memcpy(info
->last_event
, adapter
->dbg
.last_event
,
231 sizeof(adapter
->dbg
.last_event
));
232 info
->last_event_index
= adapter
->dbg
.last_event_index
;
233 memcpy(info
->last_mp_wr_bitmap
, adapter
->dbg
.last_mp_wr_bitmap
,
234 sizeof(adapter
->dbg
.last_mp_wr_bitmap
));
235 memcpy(info
->last_mp_wr_ports
, adapter
->dbg
.last_mp_wr_ports
,
236 sizeof(adapter
->dbg
.last_mp_wr_ports
));
237 memcpy(info
->last_mp_curr_wr_port
,
238 adapter
->dbg
.last_mp_curr_wr_port
,
239 sizeof(adapter
->dbg
.last_mp_curr_wr_port
));
240 memcpy(info
->last_mp_wr_len
, adapter
->dbg
.last_mp_wr_len
,
241 sizeof(adapter
->dbg
.last_mp_wr_len
));
242 info
->last_sdio_mp_index
= adapter
->dbg
.last_sdio_mp_index
;
243 info
->data_sent
= adapter
->data_sent
;
244 info
->cmd_sent
= adapter
->cmd_sent
;
245 info
->cmd_resp_received
= adapter
->cmd_resp_received
;
251 int mwifiex_debug_info_to_buffer(struct mwifiex_private
*priv
, char *buf
,
252 struct mwifiex_debug_info
*info
)
255 struct mwifiex_debug_data
*d
= &items
[0];
263 for (i
= 0; i
< num_of_items
; i
++) {
264 p
+= sprintf(p
, "%s=", d
[i
].name
);
266 size
= d
[i
].size
/ d
[i
].num
;
268 if (i
< (num_of_items
- 3))
269 addr
= d
[i
].addr
+ (size_t)info
;
270 else /* The last 3 items are struct mwifiex_adapter variables */
271 addr
= d
[i
].addr
+ (size_t)priv
->adapter
;
273 for (j
= 0; j
< d
[i
].num
; j
++) {
279 val
= get_unaligned((u16
*)addr
);
282 val
= get_unaligned((u32
*)addr
);
285 val
= get_unaligned((long long *)addr
);
292 p
+= sprintf(p
, "%#lx ", val
);
296 p
+= sprintf(p
, "\n");
299 if (info
->tx_tbl_num
) {
300 p
+= sprintf(p
, "Tx BA stream table:\n");
301 for (i
= 0; i
< info
->tx_tbl_num
; i
++)
302 p
+= sprintf(p
, "tid = %d, ra = %pM\n",
303 info
->tx_tbl
[i
].tid
, info
->tx_tbl
[i
].ra
);
306 if (info
->rx_tbl_num
) {
307 p
+= sprintf(p
, "Rx reorder table:\n");
308 for (i
= 0; i
< info
->rx_tbl_num
; i
++) {
309 p
+= sprintf(p
, "tid = %d, ta = %pM, ",
312 p
+= sprintf(p
, "start_win = %d, ",
313 info
->rx_tbl
[i
].start_win
);
314 p
+= sprintf(p
, "win_size = %d, buffer: ",
315 info
->rx_tbl
[i
].win_size
);
317 for (j
= 0; j
< info
->rx_tbl
[i
].win_size
; j
++)
318 p
+= sprintf(p
, "%c ",
319 info
->rx_tbl
[i
].buffer
[j
] ?
322 p
+= sprintf(p
, "\n");
326 if (info
->tdls_peer_num
) {
327 p
+= sprintf(p
, "TDLS peer table:\n");
328 for (i
= 0; i
< info
->tdls_peer_num
; i
++) {
329 p
+= sprintf(p
, "peer = %pM",
330 info
->tdls_list
[i
].peer_addr
);
331 p
+= sprintf(p
, "\n");
339 mwifiex_parse_mgmt_packet(struct mwifiex_private
*priv
, u8
*payload
, u16 len
,
343 u8 category
, action_code
, *addr2
;
344 struct ieee80211_hdr
*ieee_hdr
= (void *)payload
;
346 stype
= (le16_to_cpu(ieee_hdr
->frame_control
) & IEEE80211_FCTL_STYPE
);
349 case IEEE80211_STYPE_ACTION
:
350 category
= *(payload
+ sizeof(struct ieee80211_hdr
));
352 case WLAN_CATEGORY_PUBLIC
:
353 action_code
= *(payload
+ sizeof(struct ieee80211_hdr
)
355 if (action_code
== WLAN_PUB_ACTION_TDLS_DISCOVER_RES
) {
356 addr2
= ieee_hdr
->addr2
;
357 mwifiex_dbg(priv
->adapter
, INFO
,
358 "TDLS discovery response %pM nf=%d, snr=%d\n",
359 addr2
, rx_pd
->nf
, rx_pd
->snr
);
360 mwifiex_auto_tdls_update_peer_signal(priv
,
366 case WLAN_CATEGORY_BACK
:
367 /*we dont indicate BACK action frames to cfg80211*/
368 mwifiex_dbg(priv
->adapter
, INFO
,
369 "drop BACK action frames");
372 mwifiex_dbg(priv
->adapter
, INFO
,
373 "unknown public action frame category %d\n",
378 mwifiex_dbg(priv
->adapter
, INFO
,
379 "unknown mgmt frame subtype %#x\n", stype
);
386 * This function processes the received management packet and send it
390 mwifiex_process_mgmt_packet(struct mwifiex_private
*priv
,
395 struct ieee80211_hdr
*ieee_hdr
;
400 if (!priv
->mgmt_frame_mask
||
401 priv
->wdev
.iftype
== NL80211_IFTYPE_UNSPECIFIED
) {
402 mwifiex_dbg(priv
->adapter
, ERROR
,
403 "do not receive mgmt frames on uninitialized intf");
407 rx_pd
= (struct rxpd
*)skb
->data
;
409 skb_pull(skb
, le16_to_cpu(rx_pd
->rx_pkt_offset
));
410 skb_pull(skb
, sizeof(pkt_len
));
412 pkt_len
= le16_to_cpu(rx_pd
->rx_pkt_length
);
414 ieee_hdr
= (void *)skb
->data
;
415 if (ieee80211_is_mgmt(ieee_hdr
->frame_control
)) {
416 if (mwifiex_parse_mgmt_packet(priv
, (u8
*)ieee_hdr
,
420 /* Remove address4 */
421 memmove(skb
->data
+ sizeof(struct ieee80211_hdr_3addr
),
422 skb
->data
+ sizeof(struct ieee80211_hdr
),
423 pkt_len
- sizeof(struct ieee80211_hdr
));
425 pkt_len
-= ETH_ALEN
+ sizeof(pkt_len
);
426 rx_pd
->rx_pkt_length
= cpu_to_le16(pkt_len
);
428 cfg80211_rx_mgmt(&priv
->wdev
, priv
->roc_cfg
.chan
.center_freq
,
429 CAL_RSSI(rx_pd
->snr
, rx_pd
->nf
), skb
->data
, pkt_len
,
436 * This function processes the received packet before sending it to the
439 * It extracts the SKB from the received buffer and sends it to kernel.
440 * In case the received buffer does not contain the data in SKB format,
441 * the function creates a blank SKB, fills it with the data from the
442 * received buffer and then sends this new SKB to the kernel.
444 int mwifiex_recv_packet(struct mwifiex_private
*priv
, struct sk_buff
*skb
)
446 struct mwifiex_sta_node
*src_node
;
447 struct ethhdr
*p_ethhdr
;
452 priv
->stats
.rx_bytes
+= skb
->len
;
453 priv
->stats
.rx_packets
++;
455 if (GET_BSS_ROLE(priv
) == MWIFIEX_BSS_ROLE_UAP
) {
456 p_ethhdr
= (void *)skb
->data
;
457 src_node
= mwifiex_get_sta_entry(priv
, p_ethhdr
->h_source
);
459 src_node
->stats
.last_rx
= jiffies
;
460 src_node
->stats
.rx_bytes
+= skb
->len
;
461 src_node
->stats
.rx_packets
++;
465 skb
->dev
= priv
->netdev
;
466 skb
->protocol
= eth_type_trans(skb
, priv
->netdev
);
467 skb
->ip_summed
= CHECKSUM_NONE
;
469 /* This is required only in case of 11n and USB/PCIE as we alloc
470 * a buffer of 4K only if its 11N (to be able to receive 4K
471 * AMSDU packets). In case of SD we allocate buffers based
472 * on the size of packet and hence this is not needed.
474 * Modifying the truesize here as our allocation for each
475 * skb is 4K but we only receive 2K packets and this cause
476 * the kernel to start dropping packets in case where
477 * application has allocated buffer based on 2K size i.e.
478 * if there a 64K packet received (in IP fragments and
479 * application allocates 64K to receive this packet but
480 * this packet would almost double up because we allocate
481 * each 1.5K fragment in 4K and pass it up. As soon as the
482 * 64K limit hits kernel will start to drop rest of the
483 * fragments. Currently we fail the Filesndl-ht.scr script
484 * for UDP, hence this fix
486 if ((priv
->adapter
->iface_type
== MWIFIEX_USB
||
487 priv
->adapter
->iface_type
== MWIFIEX_PCIE
) &&
488 (skb
->truesize
> MWIFIEX_RX_DATA_BUF_SIZE
))
489 skb
->truesize
+= (skb
->len
- MWIFIEX_RX_DATA_BUF_SIZE
);
500 * IOCTL completion callback handler.
502 * This function is called when a pending IOCTL is completed.
504 * If work queue support is enabled, the function wakes up the
505 * corresponding waiting function. Otherwise, it processes the
506 * IOCTL response and frees the response buffer.
508 int mwifiex_complete_cmd(struct mwifiex_adapter
*adapter
,
509 struct cmd_ctrl_node
*cmd_node
)
511 WARN_ON(!cmd_node
->wait_q_enabled
);
512 mwifiex_dbg(adapter
, CMD
, "cmd completed: status=%d\n",
513 adapter
->cmd_wait_q
.status
);
515 *cmd_node
->condition
= true;
516 wake_up_interruptible(&adapter
->cmd_wait_q
.wait
);
521 /* This function will return the pointer to station entry in station list
522 * table which matches specified mac address.
523 * This function should be called after acquiring RA list spinlock.
524 * NULL is returned if station entry is not found in associated STA list.
526 struct mwifiex_sta_node
*
527 mwifiex_get_sta_entry(struct mwifiex_private
*priv
, const u8
*mac
)
529 struct mwifiex_sta_node
*node
;
534 list_for_each_entry(node
, &priv
->sta_list
, list
) {
535 if (!memcmp(node
->mac_addr
, mac
, ETH_ALEN
))
542 static struct mwifiex_sta_node
*
543 mwifiex_get_tdls_sta_entry(struct mwifiex_private
*priv
, u8 status
)
545 struct mwifiex_sta_node
*node
;
547 list_for_each_entry(node
, &priv
->sta_list
, list
) {
548 if (node
->tdls_status
== status
)
555 /* If tdls channel switching is on-going, tx data traffic should be
556 * blocked until the switching stage completed.
558 u8
mwifiex_is_tdls_chan_switching(struct mwifiex_private
*priv
)
560 struct mwifiex_sta_node
*sta_ptr
;
562 if (!priv
|| !ISSUPP_TDLS_ENABLED(priv
->adapter
->fw_cap_info
))
565 sta_ptr
= mwifiex_get_tdls_sta_entry(priv
, TDLS_CHAN_SWITCHING
);
572 u8
mwifiex_is_tdls_off_chan(struct mwifiex_private
*priv
)
574 struct mwifiex_sta_node
*sta_ptr
;
576 if (!priv
|| !ISSUPP_TDLS_ENABLED(priv
->adapter
->fw_cap_info
))
579 sta_ptr
= mwifiex_get_tdls_sta_entry(priv
, TDLS_IN_OFF_CHAN
);
586 /* If tdls channel switching is on-going or tdls operate on off-channel,
587 * cmd path should be blocked until tdls switched to base-channel.
589 u8
mwifiex_is_send_cmd_allowed(struct mwifiex_private
*priv
)
591 if (!priv
|| !ISSUPP_TDLS_ENABLED(priv
->adapter
->fw_cap_info
))
594 if (mwifiex_is_tdls_chan_switching(priv
) ||
595 mwifiex_is_tdls_off_chan(priv
))
601 /* This function will add a sta_node entry to associated station list
602 * table with the given mac address.
603 * If entry exist already, existing entry is returned.
604 * If received mac address is NULL, NULL is returned.
606 struct mwifiex_sta_node
*
607 mwifiex_add_sta_entry(struct mwifiex_private
*priv
, const u8
*mac
)
609 struct mwifiex_sta_node
*node
;
614 spin_lock_bh(&priv
->sta_list_spinlock
);
615 node
= mwifiex_get_sta_entry(priv
, mac
);
619 node
= kzalloc(sizeof(*node
), GFP_ATOMIC
);
623 memcpy(node
->mac_addr
, mac
, ETH_ALEN
);
624 list_add_tail(&node
->list
, &priv
->sta_list
);
627 spin_unlock_bh(&priv
->sta_list_spinlock
);
631 /* This function will search for HT IE in association request IEs
632 * and set station HT parameters accordingly.
635 mwifiex_set_sta_ht_cap(struct mwifiex_private
*priv
, const u8
*ies
,
636 int ies_len
, struct mwifiex_sta_node
*node
)
638 struct ieee_types_header
*ht_cap_ie
;
639 const struct ieee80211_ht_cap
*ht_cap
;
644 ht_cap_ie
= (void *)cfg80211_find_ie(WLAN_EID_HT_CAPABILITY
, ies
,
647 ht_cap
= (void *)(ht_cap_ie
+ 1);
648 node
->is_11n_enabled
= 1;
649 node
->max_amsdu
= le16_to_cpu(ht_cap
->cap_info
) &
650 IEEE80211_HT_CAP_MAX_AMSDU
?
651 MWIFIEX_TX_DATA_BUF_SIZE_8K
:
652 MWIFIEX_TX_DATA_BUF_SIZE_4K
;
654 node
->is_11n_enabled
= 0;
660 /* This function will delete a station entry from station list */
661 void mwifiex_del_sta_entry(struct mwifiex_private
*priv
, const u8
*mac
)
663 struct mwifiex_sta_node
*node
;
665 spin_lock_bh(&priv
->sta_list_spinlock
);
667 node
= mwifiex_get_sta_entry(priv
, mac
);
669 list_del(&node
->list
);
673 spin_unlock_bh(&priv
->sta_list_spinlock
);
677 /* This function will delete all stations from associated station list. */
678 void mwifiex_del_all_sta_list(struct mwifiex_private
*priv
)
680 struct mwifiex_sta_node
*node
, *tmp
;
682 spin_lock_bh(&priv
->sta_list_spinlock
);
684 list_for_each_entry_safe(node
, tmp
, &priv
->sta_list
, list
) {
685 list_del(&node
->list
);
689 INIT_LIST_HEAD(&priv
->sta_list
);
690 spin_unlock_bh(&priv
->sta_list_spinlock
);
694 /* This function adds histogram data to histogram array*/
695 void mwifiex_hist_data_add(struct mwifiex_private
*priv
,
696 u8 rx_rate
, s8 snr
, s8 nflr
)
698 struct mwifiex_histogram_data
*phist_data
= priv
->hist_data
;
700 if (atomic_read(&phist_data
->num_samples
) > MWIFIEX_HIST_MAX_SAMPLES
)
701 mwifiex_hist_data_reset(priv
);
702 mwifiex_hist_data_set(priv
, rx_rate
, snr
, nflr
);
705 /* function to add histogram record */
706 void mwifiex_hist_data_set(struct mwifiex_private
*priv
, u8 rx_rate
, s8 snr
,
709 struct mwifiex_histogram_data
*phist_data
= priv
->hist_data
;
711 s8 rssi
= snr
- nflr
;
713 atomic_inc(&phist_data
->num_samples
);
714 atomic_inc(&phist_data
->rx_rate
[rx_rate
]);
715 atomic_inc(&phist_data
->snr
[snr
+ 128]);
716 atomic_inc(&phist_data
->noise_flr
[nf
+ 128]);
717 atomic_inc(&phist_data
->sig_str
[rssi
+ 128]);
720 /* function to reset histogram data during init/reset */
721 void mwifiex_hist_data_reset(struct mwifiex_private
*priv
)
724 struct mwifiex_histogram_data
*phist_data
= priv
->hist_data
;
726 atomic_set(&phist_data
->num_samples
, 0);
727 for (ix
= 0; ix
< MWIFIEX_MAX_AC_RX_RATES
; ix
++)
728 atomic_set(&phist_data
->rx_rate
[ix
], 0);
729 for (ix
= 0; ix
< MWIFIEX_MAX_SNR
; ix
++)
730 atomic_set(&phist_data
->snr
[ix
], 0);
731 for (ix
= 0; ix
< MWIFIEX_MAX_NOISE_FLR
; ix
++)
732 atomic_set(&phist_data
->noise_flr
[ix
], 0);
733 for (ix
= 0; ix
< MWIFIEX_MAX_SIG_STRENGTH
; ix
++)
734 atomic_set(&phist_data
->sig_str
[ix
], 0);
737 void *mwifiex_alloc_dma_align_buf(int rx_len
, gfp_t flags
)
742 buf_len
= rx_len
+ MWIFIEX_RX_HEADROOM
+ MWIFIEX_DMA_ALIGN_SZ
;
744 skb
= __dev_alloc_skb(buf_len
, flags
);
749 skb_reserve(skb
, MWIFIEX_RX_HEADROOM
);
751 pad
= MWIFIEX_ALIGN_ADDR(skb
->data
, MWIFIEX_DMA_ALIGN_SZ
) -
754 skb_reserve(skb
, pad
);
758 EXPORT_SYMBOL_GPL(mwifiex_alloc_dma_align_buf
);
760 void mwifiex_fw_dump_event(struct mwifiex_private
*priv
)
762 mwifiex_send_cmd(priv
, HostCmd_CMD_FW_DUMP_EVENT
, HostCmd_ACT_GEN_SET
,
765 EXPORT_SYMBOL_GPL(mwifiex_fw_dump_event
);