|
@@ -28,6 +28,84 @@ const char driver_version[] = "mwifiex " VERSION " (%s) ";
|
|
|
static char *cal_data_cfg;
|
|
|
module_param(cal_data_cfg, charp, 0);
|
|
|
|
|
|
+static void scan_delay_timer_fn(unsigned long data)
|
|
|
+{
|
|
|
+ struct mwifiex_private *priv = (struct mwifiex_private *)data;
|
|
|
+ struct mwifiex_adapter *adapter = priv->adapter;
|
|
|
+ struct cmd_ctrl_node *cmd_node, *tmp_node;
|
|
|
+ unsigned long flags;
|
|
|
+
|
|
|
+ if (adapter->surprise_removed)
|
|
|
+ return;
|
|
|
+
|
|
|
+ if (adapter->scan_delay_cnt == MWIFIEX_MAX_SCAN_DELAY_CNT) {
|
|
|
+ /*
|
|
|
+ * Abort scan operation by cancelling all pending scan
|
|
|
+ * commands
|
|
|
+ */
|
|
|
+ spin_lock_irqsave(&adapter->scan_pending_q_lock, flags);
|
|
|
+ list_for_each_entry_safe(cmd_node, tmp_node,
|
|
|
+ &adapter->scan_pending_q, list) {
|
|
|
+ list_del(&cmd_node->list);
|
|
|
+ mwifiex_insert_cmd_to_free_q(adapter, cmd_node);
|
|
|
+ }
|
|
|
+ spin_unlock_irqrestore(&adapter->scan_pending_q_lock, flags);
|
|
|
+
|
|
|
+ spin_lock_irqsave(&adapter->mwifiex_cmd_lock, flags);
|
|
|
+ adapter->scan_processing = false;
|
|
|
+ adapter->scan_delay_cnt = 0;
|
|
|
+ adapter->empty_tx_q_cnt = 0;
|
|
|
+ spin_unlock_irqrestore(&adapter->mwifiex_cmd_lock, flags);
|
|
|
+
|
|
|
+ if (priv->scan_request) {
|
|
|
+ dev_dbg(adapter->dev, "info: aborting scan\n");
|
|
|
+ cfg80211_scan_done(priv->scan_request, 1);
|
|
|
+ priv->scan_request = NULL;
|
|
|
+ } else {
|
|
|
+ priv->scan_aborting = false;
|
|
|
+ dev_dbg(adapter->dev, "info: scan already aborted\n");
|
|
|
+ }
|
|
|
+ goto done;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!atomic_read(&priv->adapter->is_tx_received)) {
|
|
|
+ adapter->empty_tx_q_cnt++;
|
|
|
+ if (adapter->empty_tx_q_cnt == MWIFIEX_MAX_EMPTY_TX_Q_CNT) {
|
|
|
+ /*
|
|
|
+ * No Tx traffic for 200msec. Get scan command from
|
|
|
+ * scan pending queue and put to cmd pending queue to
|
|
|
+ * resume scan operation
|
|
|
+ */
|
|
|
+ adapter->scan_delay_cnt = 0;
|
|
|
+ adapter->empty_tx_q_cnt = 0;
|
|
|
+ spin_lock_irqsave(&adapter->scan_pending_q_lock, flags);
|
|
|
+ cmd_node = list_first_entry(&adapter->scan_pending_q,
|
|
|
+ struct cmd_ctrl_node, list);
|
|
|
+ list_del(&cmd_node->list);
|
|
|
+ spin_unlock_irqrestore(&adapter->scan_pending_q_lock,
|
|
|
+ flags);
|
|
|
+
|
|
|
+ mwifiex_insert_cmd_to_pending_q(adapter, cmd_node,
|
|
|
+ true);
|
|
|
+ queue_work(adapter->workqueue, &adapter->main_work);
|
|
|
+ goto done;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ adapter->empty_tx_q_cnt = 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Delay scan operation further by 20msec */
|
|
|
+ mod_timer(&priv->scan_delay_timer, jiffies +
|
|
|
+ msecs_to_jiffies(MWIFIEX_SCAN_DELAY_MSEC));
|
|
|
+ adapter->scan_delay_cnt++;
|
|
|
+
|
|
|
+done:
|
|
|
+ if (atomic_read(&priv->adapter->is_tx_received))
|
|
|
+ atomic_set(&priv->adapter->is_tx_received, false);
|
|
|
+
|
|
|
+ return;
|
|
|
+}
|
|
|
+
|
|
|
/*
|
|
|
* This function registers the device and performs all the necessary
|
|
|
* initializations.
|
|
@@ -75,6 +153,10 @@ static int mwifiex_register(void *card, struct mwifiex_if_ops *if_ops,
|
|
|
|
|
|
adapter->priv[i]->adapter = adapter;
|
|
|
adapter->priv_num++;
|
|
|
+
|
|
|
+ setup_timer(&adapter->priv[i]->scan_delay_timer,
|
|
|
+ scan_delay_timer_fn,
|
|
|
+ (unsigned long)adapter->priv[i]);
|
|
|
}
|
|
|
mwifiex_init_lock_list(adapter);
|
|
|
|