aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/iwlwifi/iwl-agn.c
diff options
context:
space:
mode:
authorReinette Chatre <reinette.chatre@intel.com>2010-02-22 19:24:47 -0500
committerReinette Chatre <reinette.chatre@intel.com>2010-03-19 16:40:58 -0400
commitfe6b23dd361199bfbc50b0cbce6bed37c5797c75 (patch)
tree96acb9c114342d7dc731310159dc4195248d115e /drivers/net/wireless/iwlwifi/iwl-agn.c
parent7e2461910e9115c9964975f77584baf8c2f76bfe (diff)
iwlwifi: implement new mac80211 station add/remove calls
mac80211 recently implemented two new callbacks that are used to request station add/remove from the driver. The benefot from these new callbacks are that they enable the driver to sleep while performing this work. This is a big patch since a few things need to be coordinated in this move. First we need to decouple station management from rate scaling, which caused a lot of code to be moved and/or deleted. Next we needed to tie in with mac80211's station management callback and let it direct our station management as well as trigger the rate scaling initialization. Signed-off-by: Reinette Chatre <reinette.chatre@intel.com>
Diffstat (limited to 'drivers/net/wireless/iwlwifi/iwl-agn.c')
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-agn.c95
1 files changed, 55 insertions, 40 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl-agn.c b/drivers/net/wireless/iwlwifi/iwl-agn.c
index 85ff1552481f..4b4b47676ad8 100644
--- a/drivers/net/wireless/iwlwifi/iwl-agn.c
+++ b/drivers/net/wireless/iwlwifi/iwl-agn.c
@@ -144,9 +144,6 @@ int iwl_commit_rxon(struct iwl_priv *priv)
144 return 0; 144 return 0;
145 } 145 }
146 146
147 /* station table will be cleared */
148 priv->assoc_station_added = 0;
149
150 /* If we are currently associated and the new config requires 147 /* If we are currently associated and the new config requires
151 * an RXON_ASSOC and the new config wants the associated mask enabled, 148 * an RXON_ASSOC and the new config wants the associated mask enabled,
152 * we must clear the associated from the active configuration 149 * we must clear the associated from the active configuration
@@ -198,25 +195,7 @@ int iwl_commit_rxon(struct iwl_priv *priv)
198 } 195 }
199 196
200 priv->start_calib = 0; 197 priv->start_calib = 0;
201
202 /* If we have set the ASSOC_MSK and we are in BSS mode then
203 * add the IWL_AP_ID to the station rate table */
204 if (new_assoc) { 198 if (new_assoc) {
205 if (priv->iw_mode == NL80211_IFTYPE_STATION) {
206 ret = iwl_rxon_add_station(priv,
207 priv->active_rxon.bssid_addr, 1);
208 if (ret == IWL_INVALID_STATION) {
209 IWL_ERR(priv,
210 "Error adding AP address for TX.\n");
211 return -EIO;
212 }
213 priv->assoc_station_added = 1;
214 if (priv->default_wep_key &&
215 iwl_send_static_wepkey_cmd(priv, 0))
216 IWL_ERR(priv,
217 "Could not send WEP static key.\n");
218 }
219
220 /* 199 /*
221 * allow CTS-to-self if possible for new association. 200 * allow CTS-to-self if possible for new association.
222 * this is relevant only for 5000 series and up, 201 * this is relevant only for 5000 series and up,
@@ -2498,10 +2477,6 @@ void iwl_post_associate(struct iwl_priv *priv)
2498 return; 2477 return;
2499 } 2478 }
2500 2479
2501 IWL_DEBUG_ASSOC(priv, "Associated as %d to: %pM\n",
2502 priv->assoc_id, priv->active_rxon.bssid_addr);
2503
2504
2505 if (test_bit(STATUS_EXIT_PENDING, &priv->status)) 2480 if (test_bit(STATUS_EXIT_PENDING, &priv->status))
2506 return; 2481 return;
2507 2482
@@ -2553,6 +2528,9 @@ void iwl_post_associate(struct iwl_priv *priv)
2553 2528
2554 iwlcore_commit_rxon(priv); 2529 iwlcore_commit_rxon(priv);
2555 2530
2531 IWL_DEBUG_ASSOC(priv, "Associated as %d to: %pM\n",
2532 priv->assoc_id, priv->active_rxon.bssid_addr);
2533
2556 switch (priv->iw_mode) { 2534 switch (priv->iw_mode) {
2557 case NL80211_IFTYPE_STATION: 2535 case NL80211_IFTYPE_STATION:
2558 break; 2536 break;
@@ -2562,7 +2540,7 @@ void iwl_post_associate(struct iwl_priv *priv)
2562 /* assume default assoc id */ 2540 /* assume default assoc id */
2563 priv->assoc_id = 1; 2541 priv->assoc_id = 1;
2564 2542
2565 iwl_rxon_add_station(priv, priv->bssid, 0); 2543 iwl_add_local_station(priv, priv->bssid, true);
2566 iwl_send_beacon_cmd(priv); 2544 iwl_send_beacon_cmd(priv);
2567 2545
2568 break; 2546 break;
@@ -2573,9 +2551,6 @@ void iwl_post_associate(struct iwl_priv *priv)
2573 break; 2551 break;
2574 } 2552 }
2575 2553
2576 if (priv->iw_mode == NL80211_IFTYPE_ADHOC)
2577 priv->assoc_station_added = 1;
2578
2579 spin_lock_irqsave(&priv->lock, flags); 2554 spin_lock_irqsave(&priv->lock, flags);
2580 iwl_activate_qos(priv, 0); 2555 iwl_activate_qos(priv, 0);
2581 spin_unlock_irqrestore(&priv->lock, flags); 2556 spin_unlock_irqrestore(&priv->lock, flags);
@@ -2974,18 +2949,7 @@ static void iwl_mac_sta_notify(struct ieee80211_hw *hw,
2974 struct iwl_station_priv *sta_priv = (void *)sta->drv_priv; 2949 struct iwl_station_priv *sta_priv = (void *)sta->drv_priv;
2975 int sta_id; 2950 int sta_id;
2976 2951
2977 /*
2978 * TODO: We really should use this callback to
2979 * actually maintain the station table in
2980 * the device.
2981 */
2982
2983 switch (cmd) { 2952 switch (cmd) {
2984 case STA_NOTIFY_ADD:
2985 atomic_set(&sta_priv->pending_frames, 0);
2986 if (vif->type == NL80211_IFTYPE_AP)
2987 sta_priv->client = true;
2988 break;
2989 case STA_NOTIFY_SLEEP: 2953 case STA_NOTIFY_SLEEP:
2990 WARN_ON(!sta_priv->client); 2954 WARN_ON(!sta_priv->client);
2991 sta_priv->asleep = true; 2955 sta_priv->asleep = true;
@@ -3006,6 +2970,55 @@ static void iwl_mac_sta_notify(struct ieee80211_hw *hw,
3006 } 2970 }
3007} 2971}
3008 2972
2973/**
2974 * iwl_restore_wepkeys - Restore WEP keys to device
2975 */
2976static void iwl_restore_wepkeys(struct iwl_priv *priv)
2977{
2978 mutex_lock(&priv->mutex);
2979 if (priv->iw_mode == NL80211_IFTYPE_STATION &&
2980 priv->default_wep_key &&
2981 iwl_send_static_wepkey_cmd(priv, 0))
2982 IWL_ERR(priv, "Could not send WEP static key\n");
2983 mutex_unlock(&priv->mutex);
2984}
2985
2986static int iwlagn_mac_sta_add(struct ieee80211_hw *hw,
2987 struct ieee80211_vif *vif,
2988 struct ieee80211_sta *sta)
2989{
2990 struct iwl_priv *priv = hw->priv;
2991 struct iwl_station_priv *sta_priv = (void *)sta->drv_priv;
2992 bool is_ap = priv->iw_mode == NL80211_IFTYPE_STATION;
2993 int ret;
2994 u8 sta_id;
2995
2996 IWL_DEBUG_INFO(priv, "received request to add station %pM\n",
2997 sta->addr);
2998
2999 atomic_set(&sta_priv->pending_frames, 0);
3000 if (vif->type == NL80211_IFTYPE_AP)
3001 sta_priv->client = true;
3002
3003 ret = iwl_add_station_common(priv, sta->addr, is_ap, &sta->ht_cap,
3004 &sta_id);
3005 if (ret) {
3006 IWL_ERR(priv, "Unable to add station %pM (%d)\n",
3007 sta->addr, ret);
3008 /* Should we return success if return code is EEXIST ? */
3009 return ret;
3010 }
3011
3012 iwl_restore_wepkeys(priv);
3013
3014 /* Initialize rate scaling */
3015 IWL_DEBUG_INFO(priv, "Initializing rate scaling for station %pM \n",
3016 sta->addr);
3017 iwl_rs_rate_init(priv, sta, sta_id);
3018
3019 return ret;
3020}
3021
3009/***************************************************************************** 3022/*****************************************************************************
3010 * 3023 *
3011 * sysfs attributes 3024 * sysfs attributes
@@ -3359,6 +3372,8 @@ static struct ieee80211_ops iwl_hw_ops = {
3359 .ampdu_action = iwl_mac_ampdu_action, 3372 .ampdu_action = iwl_mac_ampdu_action,
3360 .hw_scan = iwl_mac_hw_scan, 3373 .hw_scan = iwl_mac_hw_scan,
3361 .sta_notify = iwl_mac_sta_notify, 3374 .sta_notify = iwl_mac_sta_notify,
3375 .sta_add = iwlagn_mac_sta_add,
3376 .sta_remove = iwl_mac_sta_remove,
3362}; 3377};
3363 3378
3364static int iwl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent) 3379static int iwl_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent)