2 * Marvell Wireless LAN device driver: major functions
4 * Copyright (C) 2011, Marvell International Ltd.
6 * This software file (the "File") is distributed by Marvell International
7 * Ltd. 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.
27 const char driver_version[] = "mwifiex " VERSION " (%s) ";
30 * This function registers the device and performs all the necessary
33 * The following initialization operations are performed -
34 * - Allocate adapter structure
35 * - Save interface specific operations table in adapter
36 * - Call interface specific initialization routine
37 * - Allocate private structures
38 * - Set default adapter structure parameters
41 * In case of any errors during inittialization, this function also ensures
42 * proper cleanup before exiting.
44 static int mwifiex_register(void *card, struct mwifiex_if_ops *if_ops,
47 struct mwifiex_adapter *adapter;
50 adapter = kzalloc(sizeof(struct mwifiex_adapter), GFP_KERNEL);
57 /* Save interface specific operations in adapter */
58 memmove(&adapter->if_ops, if_ops, sizeof(struct mwifiex_if_ops));
60 /* card specific initialization has been deferred until now .. */
61 if (adapter->if_ops.init_if(adapter))
64 adapter->priv_num = 0;
66 /* Allocate memory for private structure */
67 adapter->priv[0] = kzalloc(sizeof(struct mwifiex_private), GFP_KERNEL);
68 if (!adapter->priv[0]) {
70 "%s: failed to alloc priv[0]\n", __func__);
76 adapter->priv[0]->adapter = adapter;
77 mwifiex_init_lock_list(adapter);
79 init_timer(&adapter->cmd_timer);
80 adapter->cmd_timer.function = mwifiex_cmd_timeout_func;
81 adapter->cmd_timer.data = (unsigned long) adapter;
86 dev_dbg(adapter->dev, "info: leave mwifiex_register with error\n");
88 for (i = 0; i < adapter->priv_num; i++)
89 kfree(adapter->priv[i]);
97 * This function unregisters the device and performs all the necessary
100 * The following cleanup operations are performed -
102 * - Free beacon buffers
103 * - Free private structures
104 * - Free adapter structure
106 static int mwifiex_unregister(struct mwifiex_adapter *adapter)
110 del_timer(&adapter->cmd_timer);
112 /* Free private structures */
113 for (i = 0; i < adapter->priv_num; i++) {
114 if (adapter->priv[i]) {
115 mwifiex_free_curr_bcn(adapter->priv[i]);
116 kfree(adapter->priv[i]);
127 * This function is the main procedure of the driver and handles various driver
128 * operations. It runs in a loop and provides the core functionalities.
130 * The main responsibilities of this function are -
131 * - Ensure concurrency control
132 * - Handle pending interrupts and call interrupt handlers
133 * - Wake up the card if required
134 * - Handle command responses and call response handlers
135 * - Handle events and call event handlers
136 * - Execute pending commands
137 * - Transmit pending data packets
139 int mwifiex_main_process(struct mwifiex_adapter *adapter)
144 spin_lock_irqsave(&adapter->main_proc_lock, flags);
146 /* Check if already processing */
147 if (adapter->mwifiex_processing) {
148 spin_unlock_irqrestore(&adapter->main_proc_lock, flags);
151 adapter->mwifiex_processing = true;
152 spin_unlock_irqrestore(&adapter->main_proc_lock, flags);
156 if ((adapter->hw_status == MWIFIEX_HW_STATUS_CLOSING) ||
157 (adapter->hw_status == MWIFIEX_HW_STATUS_NOT_READY))
160 /* Handle pending interrupt if any */
161 if (adapter->int_status) {
162 if (adapter->hs_activated)
163 mwifiex_process_hs_config(adapter);
164 adapter->if_ops.process_int_status(adapter);
167 /* Need to wake up the card ? */
168 if ((adapter->ps_state == PS_STATE_SLEEP) &&
169 (adapter->pm_wakeup_card_req &&
170 !adapter->pm_wakeup_fw_try) &&
171 (is_command_pending(adapter) ||
172 !mwifiex_wmm_lists_empty(adapter))) {
173 adapter->pm_wakeup_fw_try = true;
174 adapter->if_ops.wakeup(adapter);
177 if (IS_CARD_RX_RCVD(adapter)) {
178 adapter->pm_wakeup_fw_try = false;
179 if (adapter->ps_state == PS_STATE_SLEEP)
180 adapter->ps_state = PS_STATE_AWAKE;
182 /* We have tried to wakeup the card already */
183 if (adapter->pm_wakeup_fw_try)
185 if (adapter->ps_state != PS_STATE_AWAKE ||
186 adapter->tx_lock_flag)
189 if ((adapter->scan_processing &&
190 !adapter->scan_delay_cnt) || adapter->data_sent ||
191 mwifiex_wmm_lists_empty(adapter)) {
192 if (adapter->cmd_sent || adapter->curr_cmd ||
193 (!is_command_pending(adapter)))
198 /* Check for Cmd Resp */
199 if (adapter->cmd_resp_received) {
200 adapter->cmd_resp_received = false;
201 mwifiex_process_cmdresp(adapter);
203 /* call mwifiex back when init_fw is done */
204 if (adapter->hw_status == MWIFIEX_HW_STATUS_INIT_DONE) {
205 adapter->hw_status = MWIFIEX_HW_STATUS_READY;
206 mwifiex_init_fw_complete(adapter);
210 /* Check for event */
211 if (adapter->event_received) {
212 adapter->event_received = false;
213 mwifiex_process_event(adapter);
216 /* Check if we need to confirm Sleep Request
217 received previously */
218 if (adapter->ps_state == PS_STATE_PRE_SLEEP) {
219 if (!adapter->cmd_sent && !adapter->curr_cmd)
220 mwifiex_check_ps_cond(adapter);
223 /* * The ps_state may have been changed during processing of
224 * Sleep Request event.
226 if ((adapter->ps_state == PS_STATE_SLEEP) ||
227 (adapter->ps_state == PS_STATE_PRE_SLEEP) ||
228 (adapter->ps_state == PS_STATE_SLEEP_CFM) ||
229 adapter->tx_lock_flag)
232 if (!adapter->cmd_sent && !adapter->curr_cmd) {
233 if (mwifiex_exec_next_cmd(adapter) == -1) {
239 if ((!adapter->scan_processing || adapter->scan_delay_cnt) &&
240 !adapter->data_sent && !mwifiex_wmm_lists_empty(adapter)) {
241 mwifiex_wmm_process_tx(adapter);
242 if (adapter->hs_activated) {
243 adapter->is_hs_configured = false;
244 mwifiex_hs_activated_event
246 (adapter, MWIFIEX_BSS_ROLE_ANY),
251 if (adapter->delay_null_pkt && !adapter->cmd_sent &&
252 !adapter->curr_cmd && !is_command_pending(adapter) &&
253 mwifiex_wmm_lists_empty(adapter)) {
254 if (!mwifiex_send_null_packet
255 (mwifiex_get_priv(adapter, MWIFIEX_BSS_ROLE_STA),
256 MWIFIEX_TxPD_POWER_MGMT_NULL_PACKET |
257 MWIFIEX_TxPD_POWER_MGMT_LAST_PACKET)) {
258 adapter->delay_null_pkt = false;
259 adapter->ps_state = PS_STATE_SLEEP;
265 if ((adapter->int_status) || IS_CARD_RX_RCVD(adapter))
268 spin_lock_irqsave(&adapter->main_proc_lock, flags);
269 adapter->mwifiex_processing = false;
270 spin_unlock_irqrestore(&adapter->main_proc_lock, flags);
273 if (adapter->hw_status == MWIFIEX_HW_STATUS_CLOSING)
274 mwifiex_shutdown_drv(adapter);
277 EXPORT_SYMBOL_GPL(mwifiex_main_process);
280 * This function frees the adapter structure.
282 * Additionally, this closes the netlink socket, frees the timers
283 * and private structures.
285 static void mwifiex_free_adapter(struct mwifiex_adapter *adapter)
288 pr_err("%s: adapter is NULL\n", __func__);
292 mwifiex_unregister(adapter);
293 pr_debug("info: %s: free adapter\n", __func__);
297 * This function gets firmware and initializes it.
299 * The main initialization steps followed are -
300 * - Download the correct firmware to card
301 * - Issue the init commands to firmware
303 static void mwifiex_fw_dpc(const struct firmware *firmware, void *context)
307 struct mwifiex_private *priv;
308 struct mwifiex_adapter *adapter = context;
309 struct mwifiex_fw_image fw;
312 dev_err(adapter->dev,
313 "Failed to get firmware %s\n", adapter->fw_name);
317 memset(&fw, 0, sizeof(struct mwifiex_fw_image));
318 adapter->firmware = firmware;
319 fw.fw_buf = (u8 *) adapter->firmware->data;
320 fw.fw_len = adapter->firmware->size;
322 ret = mwifiex_dnld_fw(adapter, &fw);
326 dev_notice(adapter->dev, "WLAN FW is active\n");
328 adapter->init_wait_q_woken = false;
329 ret = mwifiex_init_fw(adapter);
333 adapter->hw_status = MWIFIEX_HW_STATUS_READY;
336 /* Wait for mwifiex_init to complete */
337 wait_event_interruptible(adapter->init_wait_q,
338 adapter->init_wait_q_woken);
339 if (adapter->hw_status != MWIFIEX_HW_STATUS_READY)
342 priv = adapter->priv[0];
343 if (mwifiex_register_cfg80211(priv) != 0) {
344 dev_err(adapter->dev, "cannot register with cfg80211\n");
349 /* Create station interface by default */
350 if (!mwifiex_add_virtual_intf(priv->wdev->wiphy, "mlan%d",
351 NL80211_IFTYPE_STATION, NULL, NULL)) {
352 dev_err(adapter->dev, "cannot create default STA interface\n");
357 mwifiex_drv_get_driver_version(adapter, fmt, sizeof(fmt) - 1);
358 dev_notice(adapter->dev, "driver_version = %s\n", fmt);
362 mwifiex_del_virtual_intf(priv->wdev->wiphy, priv->netdev);
365 pr_debug("info: %s: unregister device\n", __func__);
366 adapter->if_ops.unregister_dev(adapter);
368 release_firmware(adapter->firmware);
369 complete(&adapter->fw_load);
374 * This function initializes the hardware and gets firmware.
376 static int mwifiex_init_hw_fw(struct mwifiex_adapter *adapter)
380 const struct firmware *fw;
382 init_completion(&adapter->fw_load);
383 ret = request_firmware(&fw, adapter->fw_name, adapter->dev);
385 dev_err(adapter->dev,
386 "request_firmware() returned error %d\n", ret);
388 mwifiex_fw_dpc(fw, adapter);
393 * CFG802.11 network device handler for open.
395 * Starts the data queue.
398 mwifiex_open(struct net_device *dev)
400 netif_tx_start_all_queues(dev);
405 * CFG802.11 network device handler for close.
408 mwifiex_close(struct net_device *dev)
410 struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev);
412 if (priv->scan_request) {
413 dev_dbg(priv->adapter->dev, "aborting scan on ndo_stop\n");
414 cfg80211_scan_done(priv->scan_request, 1);
415 priv->scan_request = NULL;
416 priv->scan_aborting = true;
423 * Add buffer into wmm tx queue and queue work to transmit it.
425 int mwifiex_queue_tx_pkt(struct mwifiex_private *priv, struct sk_buff *skb)
427 struct netdev_queue *txq;
428 int index = mwifiex_1d_to_wmm_queue[skb->priority];
430 if (atomic_inc_return(&priv->wmm_tx_pending[index]) >= MAX_TX_PENDING) {
431 txq = netdev_get_tx_queue(priv->netdev, index);
432 if (!netif_tx_queue_stopped(txq)) {
433 netif_tx_stop_queue(txq);
434 dev_dbg(priv->adapter->dev, "stop queue: %d\n", index);
438 atomic_inc(&priv->adapter->tx_pending);
439 mwifiex_wmm_add_buf_txqueue(priv, skb);
441 if (priv->adapter->scan_delay_cnt)
442 atomic_set(&priv->adapter->is_tx_received, true);
444 queue_work(priv->adapter->workqueue, &priv->adapter->main_work);
450 * CFG802.11 network device handler for data transmission.
453 mwifiex_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
455 struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev);
456 struct sk_buff *new_skb;
457 struct mwifiex_txinfo *tx_info;
460 dev_dbg(priv->adapter->dev, "data: %lu BSS(%d-%d): Data <= kernel\n",
461 jiffies, priv->bss_type, priv->bss_num);
463 if (priv->adapter->surprise_removed) {
465 priv->stats.tx_dropped++;
468 if (!skb->len || (skb->len > ETH_FRAME_LEN)) {
469 dev_err(priv->adapter->dev, "Tx: bad skb len %d\n", skb->len);
471 priv->stats.tx_dropped++;
474 if (skb_headroom(skb) < MWIFIEX_MIN_DATA_HEADER_LEN) {
475 dev_dbg(priv->adapter->dev,
476 "data: Tx: insufficient skb headroom %d\n",
478 /* Insufficient skb headroom - allocate a new skb */
480 skb_realloc_headroom(skb, MWIFIEX_MIN_DATA_HEADER_LEN);
481 if (unlikely(!new_skb)) {
482 dev_err(priv->adapter->dev, "Tx: cannot alloca new_skb\n");
484 priv->stats.tx_dropped++;
489 dev_dbg(priv->adapter->dev, "info: new skb headroomd %d\n",
493 tx_info = MWIFIEX_SKB_TXCB(skb);
494 tx_info->bss_num = priv->bss_num;
495 tx_info->bss_type = priv->bss_type;
497 /* Record the current time the packet was queued; used to
498 * determine the amount of time the packet was queued in
499 * the driver before it was sent to the firmware.
500 * The delay is then sent along with the packet to the
501 * firmware for aggregate delay calculation for stats and
502 * MSDU lifetime expiry.
504 do_gettimeofday(&tv);
505 skb->tstamp = timeval_to_ktime(tv);
507 mwifiex_queue_tx_pkt(priv, skb);
513 * CFG802.11 network device handler for setting MAC address.
516 mwifiex_set_mac_address(struct net_device *dev, void *addr)
518 struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev);
519 struct sockaddr *hw_addr = addr;
522 memcpy(priv->curr_addr, hw_addr->sa_data, ETH_ALEN);
524 /* Send request to firmware */
525 ret = mwifiex_send_cmd_sync(priv, HostCmd_CMD_802_11_MAC_ADDRESS,
526 HostCmd_ACT_GEN_SET, 0, NULL);
529 memcpy(priv->netdev->dev_addr, priv->curr_addr, ETH_ALEN);
531 dev_err(priv->adapter->dev,
532 "set mac address failed: ret=%d\n", ret);
534 memcpy(dev->dev_addr, priv->curr_addr, ETH_ALEN);
540 * CFG802.11 network device handler for setting multicast list.
542 static void mwifiex_set_multicast_list(struct net_device *dev)
544 struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev);
545 struct mwifiex_multicast_list mcast_list;
547 if (dev->flags & IFF_PROMISC) {
548 mcast_list.mode = MWIFIEX_PROMISC_MODE;
549 } else if (dev->flags & IFF_ALLMULTI ||
550 netdev_mc_count(dev) > MWIFIEX_MAX_MULTICAST_LIST_SIZE) {
551 mcast_list.mode = MWIFIEX_ALL_MULTI_MODE;
553 mcast_list.mode = MWIFIEX_MULTICAST_MODE;
554 if (netdev_mc_count(dev))
555 mcast_list.num_multicast_addr =
556 mwifiex_copy_mcast_addr(&mcast_list, dev);
558 mwifiex_request_set_multicast_list(priv, &mcast_list);
562 * CFG802.11 network device handler for transmission timeout.
565 mwifiex_tx_timeout(struct net_device *dev)
567 struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev);
569 priv->num_tx_timeout++;
570 priv->tx_timeout_cnt++;
571 dev_err(priv->adapter->dev,
572 "%lu : Tx timeout(#%d), bss_type-num = %d-%d\n",
573 jiffies, priv->tx_timeout_cnt, priv->bss_type, priv->bss_num);
574 mwifiex_set_trans_start(dev);
576 if (priv->adapter->if_ops.reg_dbg)
577 priv->adapter->if_ops.reg_dbg(priv->adapter);
579 if (priv->tx_timeout_cnt > TX_TIMEOUT_THRESHOLD &&
580 priv->adapter->if_ops.card_reset) {
581 dev_err(priv->adapter->dev, "tx_timeout_cnt exceeds threshold. "
582 "Triggering card reset!\n");
583 priv->adapter->if_ops.card_reset(priv->adapter);
588 * CFG802.11 network device handler for statistics retrieval.
590 static struct net_device_stats *mwifiex_get_stats(struct net_device *dev)
592 struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev);
598 mwifiex_netdev_select_wmm_queue(struct net_device *dev, struct sk_buff *skb)
600 skb->priority = cfg80211_classify8021d(skb);
601 return mwifiex_1d_to_wmm_queue[skb->priority];
604 /* Network device handlers */
605 static const struct net_device_ops mwifiex_netdev_ops = {
606 .ndo_open = mwifiex_open,
607 .ndo_stop = mwifiex_close,
608 .ndo_start_xmit = mwifiex_hard_start_xmit,
609 .ndo_set_mac_address = mwifiex_set_mac_address,
610 .ndo_tx_timeout = mwifiex_tx_timeout,
611 .ndo_get_stats = mwifiex_get_stats,
612 .ndo_set_rx_mode = mwifiex_set_multicast_list,
613 .ndo_select_queue = mwifiex_netdev_select_wmm_queue,
617 * This function initializes the private structure parameters.
619 * The following wait queues are initialized -
621 * - Command wait queue
622 * - Statistics wait queue
624 * ...and the following default parameters are set -
625 * - Current key index : Set to 0
626 * - Rate index : Set to auto
627 * - Media connected : Set to disconnected
628 * - Adhoc link sensed : Set to false
629 * - Nick name : Set to null
630 * - Number of Tx timeout : Set to 0
631 * - Device address : Set to current address
633 * In addition, the CFG80211 work queue is also created.
635 void mwifiex_init_priv_params(struct mwifiex_private *priv,
636 struct net_device *dev)
638 dev->netdev_ops = &mwifiex_netdev_ops;
639 dev->destructor = free_netdev;
640 /* Initialize private structure */
641 priv->current_key_index = 0;
642 priv->media_connected = false;
643 memset(&priv->nick_name, 0, sizeof(priv->nick_name));
644 priv->num_tx_timeout = 0;
645 memcpy(dev->dev_addr, priv->curr_addr, ETH_ALEN);
649 * This function check if command is pending.
651 int is_command_pending(struct mwifiex_adapter *adapter)
654 int is_cmd_pend_q_empty;
656 spin_lock_irqsave(&adapter->cmd_pending_q_lock, flags);
657 is_cmd_pend_q_empty = list_empty(&adapter->cmd_pending_q);
658 spin_unlock_irqrestore(&adapter->cmd_pending_q_lock, flags);
660 return !is_cmd_pend_q_empty;
664 * This is the main work queue function.
666 * It handles the main process, which in turn handles the complete
669 static void mwifiex_main_work_queue(struct work_struct *work)
671 struct mwifiex_adapter *adapter =
672 container_of(work, struct mwifiex_adapter, main_work);
674 if (adapter->surprise_removed)
676 mwifiex_main_process(adapter);
680 * This function cancels all works in the queue and destroys
681 * the main workqueue.
684 mwifiex_terminate_workqueue(struct mwifiex_adapter *adapter)
686 flush_workqueue(adapter->workqueue);
687 destroy_workqueue(adapter->workqueue);
688 adapter->workqueue = NULL;
692 * This function adds the card.
694 * This function follows the following major steps to set up the device -
695 * - Initialize software. This includes probing the card, registering
696 * the interface operations table, and allocating/initializing the
698 * - Set up the netlink socket
699 * - Create and start the main work queue
700 * - Register the device
701 * - Initialize firmware and hardware
702 * - Add logical interfaces
705 mwifiex_add_card(void *card, struct semaphore *sem,
706 struct mwifiex_if_ops *if_ops, u8 iface_type)
708 struct mwifiex_adapter *adapter;
710 if (down_interruptible(sem))
713 if (mwifiex_register(card, if_ops, (void **)&adapter)) {
714 pr_err("%s: software init failed\n", __func__);
718 adapter->iface_type = iface_type;
720 adapter->hw_status = MWIFIEX_HW_STATUS_INITIALIZING;
721 adapter->surprise_removed = false;
722 init_waitqueue_head(&adapter->init_wait_q);
723 adapter->is_suspended = false;
724 adapter->hs_activated = false;
725 init_waitqueue_head(&adapter->hs_activate_wait_q);
726 adapter->cmd_wait_q_required = false;
727 init_waitqueue_head(&adapter->cmd_wait_q.wait);
728 adapter->cmd_wait_q.status = 0;
729 adapter->scan_wait_q_woken = false;
731 adapter->workqueue = create_workqueue("MWIFIEX_WORK_QUEUE");
732 if (!adapter->workqueue)
735 INIT_WORK(&adapter->main_work, mwifiex_main_work_queue);
737 /* Register the device. Fill up the private data structure with relevant
738 information from the card and request for the required IRQ. */
739 if (adapter->if_ops.register_dev(adapter)) {
740 pr_err("%s: failed to register mwifiex device\n", __func__);
741 goto err_registerdev;
744 if (mwifiex_init_hw_fw(adapter)) {
745 pr_err("%s: firmware init failed\n", __func__);
753 pr_debug("info: %s: unregister device\n", __func__);
754 adapter->if_ops.unregister_dev(adapter);
756 adapter->surprise_removed = true;
757 mwifiex_terminate_workqueue(adapter);
759 if ((adapter->hw_status == MWIFIEX_HW_STATUS_FW_READY) ||
760 (adapter->hw_status == MWIFIEX_HW_STATUS_READY)) {
761 pr_debug("info: %s: shutdown mwifiex\n", __func__);
762 adapter->init_wait_q_woken = false;
764 if (mwifiex_shutdown_drv(adapter) == -EINPROGRESS)
765 wait_event_interruptible(adapter->init_wait_q,
766 adapter->init_wait_q_woken);
769 mwifiex_free_adapter(adapter);
777 EXPORT_SYMBOL_GPL(mwifiex_add_card);
780 * This function removes the card.
782 * This function follows the following major steps to remove the device -
783 * - Stop data traffic
784 * - Shutdown firmware
785 * - Remove the logical interfaces
786 * - Terminate the work queue
787 * - Unregister the device
788 * - Free the adapter structure
790 int mwifiex_remove_card(struct mwifiex_adapter *adapter, struct semaphore *sem)
792 struct mwifiex_private *priv = NULL;
795 if (down_interruptible(sem))
801 adapter->surprise_removed = true;
804 for (i = 0; i < adapter->priv_num; i++) {
805 priv = adapter->priv[i];
806 if (priv && priv->netdev) {
807 mwifiex_stop_net_dev_queue(priv->netdev, adapter);
808 if (netif_carrier_ok(priv->netdev))
809 netif_carrier_off(priv->netdev);
813 dev_dbg(adapter->dev, "cmd: calling mwifiex_shutdown_drv...\n");
814 adapter->init_wait_q_woken = false;
816 if (mwifiex_shutdown_drv(adapter) == -EINPROGRESS)
817 wait_event_interruptible(adapter->init_wait_q,
818 adapter->init_wait_q_woken);
819 dev_dbg(adapter->dev, "cmd: mwifiex_shutdown_drv done\n");
820 if (atomic_read(&adapter->rx_pending) ||
821 atomic_read(&adapter->tx_pending) ||
822 atomic_read(&adapter->cmd_pending)) {
823 dev_err(adapter->dev, "rx_pending=%d, tx_pending=%d, "
825 atomic_read(&adapter->rx_pending),
826 atomic_read(&adapter->tx_pending),
827 atomic_read(&adapter->cmd_pending));
830 for (i = 0; i < adapter->priv_num; i++) {
831 priv = adapter->priv[i];
837 if (priv->wdev && priv->netdev)
838 mwifiex_del_virtual_intf(priv->wdev->wiphy,
843 priv = adapter->priv[0];
848 wiphy_unregister(priv->wdev->wiphy);
849 wiphy_free(priv->wdev->wiphy);
853 mwifiex_terminate_workqueue(adapter);
855 /* Unregister device */
856 dev_dbg(adapter->dev, "info: unregister device\n");
857 adapter->if_ops.unregister_dev(adapter);
858 /* Free adapter structure */
859 dev_dbg(adapter->dev, "info: free adapter\n");
860 mwifiex_free_adapter(adapter);
867 EXPORT_SYMBOL_GPL(mwifiex_remove_card);
870 * This function initializes the module.
872 * The debug FS is also initialized if configured.
875 mwifiex_init_module(void)
877 #ifdef CONFIG_DEBUG_FS
878 mwifiex_debugfs_init();
884 * This function cleans up the module.
886 * The debug FS is removed if available.
889 mwifiex_cleanup_module(void)
891 #ifdef CONFIG_DEBUG_FS
892 mwifiex_debugfs_remove();
896 module_init(mwifiex_init_module);
897 module_exit(mwifiex_cleanup_module);
899 MODULE_AUTHOR("Marvell International Ltd.");
900 MODULE_DESCRIPTION("Marvell WiFi-Ex Driver version " VERSION);
901 MODULE_VERSION(VERSION);
902 MODULE_LICENSE("GPL v2");