mac80211: restrict delayed tailroom needed decrement
authorManikanta Pubbisetty <mpubbise@codeaurora.org>
Tue, 10 Jul 2018 11:18:27 +0000 (16:48 +0530)
committerJohannes Berg <johannes.berg@intel.com>
Tue, 24 Jul 2018 07:21:12 +0000 (09:21 +0200)
As explained in ieee80211_delayed_tailroom_dec(), during roam,
keys of the old AP will be destroyed and new keys will be
installed. Deletion of the old key causes
crypto_tx_tailroom_needed_cnt to go from 1 to 0 and the new key
installation causes a transition from 0 to 1.

Whenever crypto_tx_tailroom_needed_cnt transitions from 0 to 1,
we invoke synchronize_net(); the reason for doing this is to avoid
a race in the TX path as explained in increment_tailroom_need_count().
This synchronize_net() operation can be slow and can affect the station
roam time. To avoid this, decrementing the crypto_tx_tailroom_needed_cnt
is delayed for a while so that upon installation of new key the
transition would be from 1 to 2 instead of 0 to 1 and thereby
improving the roam time.

This is all correct for a STA iftype, but deferring the tailroom_needed
decrement for other iftypes may be unnecessary.

For example, let's consider the case of a 4-addr client connecting to
an AP for which AP_VLAN interface is also created, let the initial
value for tailroom_needed on the AP be 1.

* 4-addr client connects to the AP (AP: tailroom_needed = 1)
* AP will clear old keys, delay decrement of tailroom_needed count
* AP_VLAN is created, it takes the tailroom count from master
  (AP_VLAN: tailroom_needed = 1, AP: tailroom_needed = 1)
* Install new key for the station, assume key is plumbed in the HW,
  there won't be any change in tailroom_needed count on AP iface
* Delayed decrement of tailroom_needed count on AP
  (AP: tailroom_needed = 0, AP_VLAN: tailroom_needed = 1)

Because of the delayed decrement on AP iface, tailroom_needed count goes
out of sync between AP(master iface) and AP_VLAN(slave iface) and
there would be unnecessary tailroom created for the packets going
through AP_VLAN iface.

Also, WARN_ONs were observed while trying to bring down the AP_VLAN
interface:
(warn_slowpath_common) (warn_slowpath_null+0x18/0x20)
(warn_slowpath_null) (ieee80211_free_keys+0x114/0x1e4)
(ieee80211_free_keys) (ieee80211_del_virtual_monitor+0x51c/0x850)
(ieee80211_del_virtual_monitor) (ieee80211_stop+0x30/0x3c)
(ieee80211_stop) (__dev_close_many+0x94/0xb8)
(__dev_close_many) (dev_close_many+0x5c/0xc8)

Restricting delayed decrement to station interface alone fixes the problem
and it makes sense to do so because delayed decrement is done to improve
roam time which is applicable only for client devices.

Signed-off-by: Manikanta Pubbisetty <mpubbise@codeaurora.org>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
net/mac80211/cfg.c
net/mac80211/key.c

index 02f3672e7b5e2ed89046db9b0a6b65069e3940c7..d25da0e66da16218c340e4c7f8a9aaf663985b9c 100644 (file)
@@ -495,7 +495,7 @@ static int ieee80211_del_key(struct wiphy *wiphy, struct net_device *dev,
                goto out_unlock;
        }
 
-       ieee80211_key_free(key, true);
+       ieee80211_key_free(key, sdata->vif.type == NL80211_IFTYPE_STATION);
 
        ret = 0;
  out_unlock:
index ee0d0cc8dc3bd686341d6c3e172bf1664c882bcb..c054ac85793c3f484a1425e0a052f6f420eb0df3 100644 (file)
@@ -656,11 +656,15 @@ int ieee80211_key_link(struct ieee80211_key *key,
 {
        struct ieee80211_local *local = sdata->local;
        struct ieee80211_key *old_key;
-       int idx, ret;
-       bool pairwise;
-
-       pairwise = key->conf.flags & IEEE80211_KEY_FLAG_PAIRWISE;
-       idx = key->conf.keyidx;
+       int idx = key->conf.keyidx;
+       bool pairwise = key->conf.flags & IEEE80211_KEY_FLAG_PAIRWISE;
+       /*
+        * We want to delay tailroom updates only for station - in that
+        * case it helps roaming speed, but in other cases it hurts and
+        * can cause warnings to appear.
+        */
+       bool delay_tailroom = sdata->vif.type == NL80211_IFTYPE_STATION;
+       int ret;
 
        mutex_lock(&sdata->local->key_mtx);
 
@@ -688,14 +692,14 @@ int ieee80211_key_link(struct ieee80211_key *key,
        increment_tailroom_need_count(sdata);
 
        ieee80211_key_replace(sdata, sta, pairwise, old_key, key);
-       ieee80211_key_destroy(old_key, true);
+       ieee80211_key_destroy(old_key, delay_tailroom);
 
        ieee80211_debugfs_key_add(key);
 
        if (!local->wowlan) {
                ret = ieee80211_key_enable_hw_accel(key);
                if (ret)
-                       ieee80211_key_free(key, true);
+                       ieee80211_key_free(key, delay_tailroom);
        } else {
                ret = 0;
        }
@@ -930,7 +934,8 @@ void ieee80211_free_sta_keys(struct ieee80211_local *local,
                ieee80211_key_replace(key->sdata, key->sta,
                                key->conf.flags & IEEE80211_KEY_FLAG_PAIRWISE,
                                key, NULL);
-               __ieee80211_key_destroy(key, true);
+               __ieee80211_key_destroy(key, key->sdata->vif.type ==
+                                       NL80211_IFTYPE_STATION);
        }
 
        for (i = 0; i < NUM_DEFAULT_KEYS; i++) {
@@ -940,7 +945,8 @@ void ieee80211_free_sta_keys(struct ieee80211_local *local,
                ieee80211_key_replace(key->sdata, key->sta,
                                key->conf.flags & IEEE80211_KEY_FLAG_PAIRWISE,
                                key, NULL);
-               __ieee80211_key_destroy(key, true);
+               __ieee80211_key_destroy(key, key->sdata->vif.type ==
+                                       NL80211_IFTYPE_STATION);
        }
 
        mutex_unlock(&local->key_mtx);