diff options
author | Felix Fietkau <nbd@openwrt.org> | 2011-04-17 20:56:14 +0000 |
---|---|---|
committer | Felix Fietkau <nbd@openwrt.org> | 2011-04-17 20:56:14 +0000 |
commit | 54397f378db83195248792978bf32b939c0e41eb (patch) | |
tree | 374f6b6b8fef2128a0ada39d3677d53404539cd7 | |
parent | dcb16d72ca497c99f5ea8d3972c96fd47261fafd (diff) | |
download | master-187ad058-54397f378db83195248792978bf32b939c0e41eb.tar.gz master-187ad058-54397f378db83195248792978bf32b939c0e41eb.tar.bz2 master-187ad058-54397f378db83195248792978bf32b939c0e41eb.zip |
ath9k: assign a keycache slot per station for unencrypted links to fix powersave frame filtering
git-svn-id: svn://svn.openwrt.org/openwrt/trunk@26712 3c298f89-4303-0410-b956-a3cf2f4a3e73
-rw-r--r-- | package/mac80211/patches/571-ath9k_assign_unencrypted_ps_keyidx.patch | 119 |
1 files changed, 119 insertions, 0 deletions
diff --git a/package/mac80211/patches/571-ath9k_assign_unencrypted_ps_keyidx.patch b/package/mac80211/patches/571-ath9k_assign_unencrypted_ps_keyidx.patch new file mode 100644 index 0000000000..7996f33f67 --- /dev/null +++ b/package/mac80211/patches/571-ath9k_assign_unencrypted_ps_keyidx.patch @@ -0,0 +1,119 @@ +--- a/drivers/net/wireless/ath/ath9k/ath9k.h ++++ b/drivers/net/wireless/ath/ath9k/ath9k.h +@@ -256,6 +256,8 @@ struct ath_node { + #endif + struct ath_atx_tid tid[WME_NUM_TID]; + struct ath_atx_ac ac[WME_NUM_AC]; ++ int ps_key; ++ + u16 maxampdu; + u8 mpdudensity; + +--- a/drivers/net/wireless/ath/ath9k/main.c ++++ b/drivers/net/wireless/ath/ath9k/main.c +@@ -1775,18 +1775,37 @@ static int ath9k_sta_add(struct ieee8021 + struct ieee80211_sta *sta) + { + struct ath_softc *sc = hw->priv; ++ struct ath_common *common = ath9k_hw_common(sc->sc_ah); ++ struct ath_node *an = (struct ath_node *) sta->drv_priv; ++ struct ieee80211_key_conf ps_key = { }; + + ath_node_attach(sc, sta); ++ an->ps_key = ath_key_config(common, vif, sta, &ps_key); + + return 0; + } + ++static void ath9k_del_ps_key(struct ath_softc *sc, ++ struct ieee80211_vif *vif, ++ struct ieee80211_sta *sta) ++{ ++ struct ath_common *common = ath9k_hw_common(sc->sc_ah); ++ struct ath_node *an = (struct ath_node *) sta->drv_priv; ++ struct ieee80211_key_conf ps_key = { .hw_key_idx = an->ps_key }; ++ ++ if (!an->ps_key) ++ return; ++ ++ ath_key_delete(common, &ps_key); ++} ++ + static int ath9k_sta_remove(struct ieee80211_hw *hw, + struct ieee80211_vif *vif, + struct ieee80211_sta *sta) + { + struct ath_softc *sc = hw->priv; + ++ ath9k_del_ps_key(sc, vif, sta); + ath_node_detach(sc, sta); + + return 0; +@@ -1889,6 +1908,9 @@ static int ath9k_set_key(struct ieee8021 + + switch (cmd) { + case SET_KEY: ++ if (sta) ++ ath9k_del_ps_key(sc, vif, sta); ++ + ret = ath_key_config(common, vif, sta, key); + if (ret >= 0) { + key->hw_key_idx = ret; +--- a/drivers/net/wireless/ath/key.c ++++ b/drivers/net/wireless/ath/key.c +@@ -483,6 +483,9 @@ int ath_key_config(struct ath_common *co + memset(&hk, 0, sizeof(hk)); + + switch (key->cipher) { ++ case 0: ++ hk.kv_type = ATH_CIPHER_CLR; ++ break; + case WLAN_CIPHER_SUITE_WEP40: + case WLAN_CIPHER_SUITE_WEP104: + hk.kv_type = ATH_CIPHER_WEP; +@@ -498,7 +501,8 @@ int ath_key_config(struct ath_common *co + } + + hk.kv_len = key->keylen; +- memcpy(hk.kv_val, key->key, key->keylen); ++ if (key->keylen) ++ memcpy(hk.kv_val, key->key, key->keylen); + + if (!(key->flags & IEEE80211_KEY_FLAG_PAIRWISE)) { + switch (vif->type) { +--- a/drivers/net/wireless/ath/ath9k/xmit.c ++++ b/drivers/net/wireless/ath/ath9k/xmit.c +@@ -1526,7 +1526,7 @@ static void setup_frame_info(struct ieee + struct ieee80211_key_conf *hw_key = tx_info->control.hw_key; + struct ieee80211_hdr *hdr; + struct ath_frame_info *fi = get_frame_info(skb); +- struct ath_node *an; ++ struct ath_node *an = NULL; + struct ath_atx_tid *tid; + enum ath9k_key_type keytype; + u16 seqno = 0; +@@ -1534,11 +1534,13 @@ static void setup_frame_info(struct ieee + + keytype = ath9k_cmn_get_hw_crypto_keytype(skb); + ++ if (sta) ++ an = (struct ath_node *) sta->drv_priv; ++ + hdr = (struct ieee80211_hdr *)skb->data; +- if (sta && ieee80211_is_data_qos(hdr->frame_control) && ++ if (an && ieee80211_is_data_qos(hdr->frame_control) && + conf_is_ht(&hw->conf) && (sc->sc_flags & SC_OP_TXAGGR)) { + +- an = (struct ath_node *) sta->drv_priv; + tidno = ieee80211_get_qos_ctl(hdr)[0] & IEEE80211_QOS_CTL_TID_MASK; + + /* +@@ -1554,6 +1556,8 @@ static void setup_frame_info(struct ieee + memset(fi, 0, sizeof(*fi)); + if (hw_key) + fi->keyix = hw_key->hw_key_idx; ++ else if (an && ieee80211_is_data(hdr->frame_control) && an->ps_key > 0) ++ fi->keyix = an->ps_key; + else + fi->keyix = ATH9K_TXKEYIX_INVALID; + fi->keytype = keytype; |