aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/iwlwifi/iwl4965-base.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/wireless/iwlwifi/iwl4965-base.c')
-rw-r--r--drivers/net/wireless/iwlwifi/iwl4965-base.c50
1 files changed, 36 insertions, 14 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl4965-base.c b/drivers/net/wireless/iwlwifi/iwl4965-base.c
index 6a8c629ce0bb..8bc2634bb407 100644
--- a/drivers/net/wireless/iwlwifi/iwl4965-base.c
+++ b/drivers/net/wireless/iwlwifi/iwl4965-base.c
@@ -7403,6 +7403,8 @@ static void iwl4965_bg_abort_scan(struct work_struct *work)
7403 mutex_unlock(&priv->mutex); 7403 mutex_unlock(&priv->mutex);
7404} 7404}
7405 7405
7406static int iwl4965_mac_config(struct ieee80211_hw *hw, struct ieee80211_conf *conf);
7407
7406static void iwl4965_bg_scan_completed(struct work_struct *work) 7408static void iwl4965_bg_scan_completed(struct work_struct *work)
7407{ 7409{
7408 struct iwl4965_priv *priv = 7410 struct iwl4965_priv *priv =
@@ -7413,6 +7415,9 @@ static void iwl4965_bg_scan_completed(struct work_struct *work)
7413 if (test_bit(STATUS_EXIT_PENDING, &priv->status)) 7415 if (test_bit(STATUS_EXIT_PENDING, &priv->status))
7414 return; 7416 return;
7415 7417
7418 if (priv->cache_conf)
7419 iwl4965_mac_config(priv->hw, priv->cache_conf);
7420
7416 ieee80211_scan_completed(priv->hw); 7421 ieee80211_scan_completed(priv->hw);
7417 7422
7418 /* Since setting the TXPOWER may have been deferred while 7423 /* Since setting the TXPOWER may have been deferred while
@@ -7541,23 +7546,38 @@ static int iwl4965_mac_config(struct ieee80211_hw *hw, struct ieee80211_conf *co
7541 struct iwl4965_priv *priv = hw->priv; 7546 struct iwl4965_priv *priv = hw->priv;
7542 const struct iwl4965_channel_info *ch_info; 7547 const struct iwl4965_channel_info *ch_info;
7543 unsigned long flags; 7548 unsigned long flags;
7549 int ret = 0;
7544 7550
7545 mutex_lock(&priv->mutex); 7551 mutex_lock(&priv->mutex);
7546 IWL_DEBUG_MAC80211("enter to channel %d\n", conf->channel); 7552 IWL_DEBUG_MAC80211("enter to channel %d\n", conf->channel);
7547 7553
7548 if (!iwl4965_is_ready(priv)) { 7554 if (!iwl4965_is_ready(priv)) {
7549 IWL_DEBUG_MAC80211("leave - not ready\n"); 7555 IWL_DEBUG_MAC80211("leave - not ready\n");
7550 mutex_unlock(&priv->mutex); 7556 ret = -EIO;
7551 return -EIO; 7557 goto out;
7552 } 7558 }
7553 7559
7554 /* TODO: Figure out how to get ieee80211_local->sta_scanning w/ only 7560 /* TODO: Figure out how to get ieee80211_local->sta_scanning w/ only
7555 * what is exposed through include/ declarations */ 7561 * what is exposed through include/ declarations */
7556 if (unlikely(!iwl4965_param_disable_hw_scan && 7562 if (unlikely(!iwl4965_param_disable_hw_scan &&
7557 test_bit(STATUS_SCANNING, &priv->status))) { 7563 test_bit(STATUS_SCANNING, &priv->status))) {
7558 IWL_DEBUG_MAC80211("leave - scanning\n"); 7564
7565 if (unlikely(priv->cache_conf))
7566 IWL_DEBUG_MAC80211("leave - still scanning\n");
7567 else {
7568 /* Cache the configuration now so that we can
7569 * replay it after the hardware scan is finished. */
7570 priv->cache_conf = kmalloc(sizeof(*conf), GFP_KERNEL);
7571 if (priv->cache_conf) {
7572 memcpy(priv->cache_conf, conf, sizeof(*conf));
7573 IWL_DEBUG_MAC80211("leave - scanning\n");
7574 } else {
7575 IWL_DEBUG_MAC80211("leave - no memory\n");
7576 ret = -ENOMEM;
7577 }
7578 }
7559 mutex_unlock(&priv->mutex); 7579 mutex_unlock(&priv->mutex);
7560 return 0; 7580 return ret;
7561 } 7581 }
7562 7582
7563 spin_lock_irqsave(&priv->lock, flags); 7583 spin_lock_irqsave(&priv->lock, flags);
@@ -7568,8 +7588,8 @@ static int iwl4965_mac_config(struct ieee80211_hw *hw, struct ieee80211_conf *co
7568 conf->channel, conf->phymode); 7588 conf->channel, conf->phymode);
7569 IWL_DEBUG_MAC80211("leave - invalid channel\n"); 7589 IWL_DEBUG_MAC80211("leave - invalid channel\n");
7570 spin_unlock_irqrestore(&priv->lock, flags); 7590 spin_unlock_irqrestore(&priv->lock, flags);
7571 mutex_unlock(&priv->mutex); 7591 ret = -EINVAL;
7572 return -EINVAL; 7592 goto out;
7573 } 7593 }
7574 7594
7575#ifdef CONFIG_IWL4965_HT 7595#ifdef CONFIG_IWL4965_HT
@@ -7598,8 +7618,7 @@ static int iwl4965_mac_config(struct ieee80211_hw *hw, struct ieee80211_conf *co
7598#ifdef IEEE80211_CONF_CHANNEL_SWITCH 7618#ifdef IEEE80211_CONF_CHANNEL_SWITCH
7599 if (conf->flags & IEEE80211_CONF_CHANNEL_SWITCH) { 7619 if (conf->flags & IEEE80211_CONF_CHANNEL_SWITCH) {
7600 iwl4965_hw_channel_switch(priv, conf->channel); 7620 iwl4965_hw_channel_switch(priv, conf->channel);
7601 mutex_unlock(&priv->mutex); 7621 goto out;
7602 return 0;
7603 } 7622 }
7604#endif 7623#endif
7605 7624
@@ -7607,14 +7626,13 @@ static int iwl4965_mac_config(struct ieee80211_hw *hw, struct ieee80211_conf *co
7607 7626
7608 if (!conf->radio_enabled) { 7627 if (!conf->radio_enabled) {
7609 IWL_DEBUG_MAC80211("leave - radio disabled\n"); 7628 IWL_DEBUG_MAC80211("leave - radio disabled\n");
7610 mutex_unlock(&priv->mutex); 7629 goto out;
7611 return 0;
7612 } 7630 }
7613 7631
7614 if (iwl4965_is_rfkill(priv)) { 7632 if (iwl4965_is_rfkill(priv)) {
7615 IWL_DEBUG_MAC80211("leave - RF kill\n"); 7633 IWL_DEBUG_MAC80211("leave - RF kill\n");
7616 mutex_unlock(&priv->mutex); 7634 ret = -EIO;
7617 return -EIO; 7635 goto out;
7618 } 7636 }
7619 7637
7620 iwl4965_set_rate(priv); 7638 iwl4965_set_rate(priv);
@@ -7627,9 +7645,13 @@ static int iwl4965_mac_config(struct ieee80211_hw *hw, struct ieee80211_conf *co
7627 7645
7628 IWL_DEBUG_MAC80211("leave\n"); 7646 IWL_DEBUG_MAC80211("leave\n");
7629 7647
7648out:
7649 if (priv->cache_conf) {
7650 kfree(priv->cache_conf);
7651 priv->cache_conf = NULL;
7652 }
7630 mutex_unlock(&priv->mutex); 7653 mutex_unlock(&priv->mutex);
7631 7654 return ret;
7632 return 0;
7633} 7655}
7634 7656
7635static void iwl4965_config_ap(struct iwl4965_priv *priv) 7657static void iwl4965_config_ap(struct iwl4965_priv *priv)