aboutsummaryrefslogtreecommitdiffstats
path: root/package/mac80211/patches/522-ath9k_remove_duplicate_code.patch
diff options
context:
space:
mode:
Diffstat (limited to 'package/mac80211/patches/522-ath9k_remove_duplicate_code.patch')
-rw-r--r--package/mac80211/patches/522-ath9k_remove_duplicate_code.patch255
1 files changed, 255 insertions, 0 deletions
diff --git a/package/mac80211/patches/522-ath9k_remove_duplicate_code.patch b/package/mac80211/patches/522-ath9k_remove_duplicate_code.patch
new file mode 100644
index 0000000000..53d86781f7
--- /dev/null
+++ b/package/mac80211/patches/522-ath9k_remove_duplicate_code.patch
@@ -0,0 +1,255 @@
+--- a/drivers/net/wireless/ath/ath9k/main.c
++++ b/drivers/net/wireless/ath/ath9k/main.c
+@@ -622,234 +622,6 @@ static u32 ath_get_extchanmode(struct at
+ return chanmode;
+ }
+
+-static int ath_setkey_tkip(struct ath_common *common, u16 keyix, const u8 *key,
+- struct ath9k_keyval *hk, const u8 *addr,
+- bool authenticator)
+-{
+- struct ath_hw *ah = common->ah;
+- const u8 *key_rxmic;
+- const u8 *key_txmic;
+-
+- key_txmic = key + NL80211_TKIP_DATA_OFFSET_TX_MIC_KEY;
+- key_rxmic = key + NL80211_TKIP_DATA_OFFSET_RX_MIC_KEY;
+-
+- if (addr == NULL) {
+- /*
+- * Group key installation - only two key cache entries are used
+- * regardless of splitmic capability since group key is only
+- * used either for TX or RX.
+- */
+- if (authenticator) {
+- memcpy(hk->kv_mic, key_txmic, sizeof(hk->kv_mic));
+- memcpy(hk->kv_txmic, key_txmic, sizeof(hk->kv_mic));
+- } else {
+- memcpy(hk->kv_mic, key_rxmic, sizeof(hk->kv_mic));
+- memcpy(hk->kv_txmic, key_rxmic, sizeof(hk->kv_mic));
+- }
+- return ath9k_hw_set_keycache_entry(ah, keyix, hk, addr);
+- }
+- if (!common->splitmic) {
+- /* TX and RX keys share the same key cache entry. */
+- memcpy(hk->kv_mic, key_rxmic, sizeof(hk->kv_mic));
+- memcpy(hk->kv_txmic, key_txmic, sizeof(hk->kv_txmic));
+- return ath9k_hw_set_keycache_entry(ah, keyix, hk, addr);
+- }
+-
+- /* Separate key cache entries for TX and RX */
+-
+- /* TX key goes at first index, RX key at +32. */
+- memcpy(hk->kv_mic, key_txmic, sizeof(hk->kv_mic));
+- if (!ath9k_hw_set_keycache_entry(ah, keyix, hk, NULL)) {
+- /* TX MIC entry failed. No need to proceed further */
+- ath_print(common, ATH_DBG_FATAL,
+- "Setting TX MIC Key Failed\n");
+- return 0;
+- }
+-
+- memcpy(hk->kv_mic, key_rxmic, sizeof(hk->kv_mic));
+- /* XXX delete tx key on failure? */
+- return ath9k_hw_set_keycache_entry(ah, keyix + 32, hk, addr);
+-}
+-
+-static int ath_reserve_key_cache_slot_tkip(struct ath_common *common)
+-{
+- int i;
+-
+- for (i = IEEE80211_WEP_NKID; i < common->keymax / 2; i++) {
+- if (test_bit(i, common->keymap) ||
+- test_bit(i + 64, common->keymap))
+- continue; /* At least one part of TKIP key allocated */
+- if (common->splitmic &&
+- (test_bit(i + 32, common->keymap) ||
+- test_bit(i + 64 + 32, common->keymap)))
+- continue; /* At least one part of TKIP key allocated */
+-
+- /* Found a free slot for a TKIP key */
+- return i;
+- }
+- return -1;
+-}
+-
+-static int ath_reserve_key_cache_slot(struct ath_common *common)
+-{
+- int i;
+-
+- /* First, try to find slots that would not be available for TKIP. */
+- if (common->splitmic) {
+- for (i = IEEE80211_WEP_NKID; i < common->keymax / 4; i++) {
+- if (!test_bit(i, common->keymap) &&
+- (test_bit(i + 32, common->keymap) ||
+- test_bit(i + 64, common->keymap) ||
+- test_bit(i + 64 + 32, common->keymap)))
+- return i;
+- if (!test_bit(i + 32, common->keymap) &&
+- (test_bit(i, common->keymap) ||
+- test_bit(i + 64, common->keymap) ||
+- test_bit(i + 64 + 32, common->keymap)))
+- return i + 32;
+- if (!test_bit(i + 64, common->keymap) &&
+- (test_bit(i , common->keymap) ||
+- test_bit(i + 32, common->keymap) ||
+- test_bit(i + 64 + 32, common->keymap)))
+- return i + 64;
+- if (!test_bit(i + 64 + 32, common->keymap) &&
+- (test_bit(i, common->keymap) ||
+- test_bit(i + 32, common->keymap) ||
+- test_bit(i + 64, common->keymap)))
+- return i + 64 + 32;
+- }
+- } else {
+- for (i = IEEE80211_WEP_NKID; i < common->keymax / 2; i++) {
+- if (!test_bit(i, common->keymap) &&
+- test_bit(i + 64, common->keymap))
+- return i;
+- if (test_bit(i, common->keymap) &&
+- !test_bit(i + 64, common->keymap))
+- return i + 64;
+- }
+- }
+-
+- /* No partially used TKIP slots, pick any available slot */
+- for (i = IEEE80211_WEP_NKID; i < common->keymax; i++) {
+- /* Do not allow slots that could be needed for TKIP group keys
+- * to be used. This limitation could be removed if we know that
+- * TKIP will not be used. */
+- if (i >= 64 && i < 64 + IEEE80211_WEP_NKID)
+- continue;
+- if (common->splitmic) {
+- if (i >= 32 && i < 32 + IEEE80211_WEP_NKID)
+- continue;
+- if (i >= 64 + 32 && i < 64 + 32 + IEEE80211_WEP_NKID)
+- continue;
+- }
+-
+- if (!test_bit(i, common->keymap))
+- return i; /* Found a free slot for a key */
+- }
+-
+- /* No free slot found */
+- return -1;
+-}
+-
+-static int ath_key_config(struct ath_common *common,
+- struct ieee80211_vif *vif,
+- struct ieee80211_sta *sta,
+- struct ieee80211_key_conf *key)
+-{
+- struct ath_hw *ah = common->ah;
+- struct ath9k_keyval hk;
+- const u8 *mac = NULL;
+- int ret = 0;
+- int idx;
+-
+- memset(&hk, 0, sizeof(hk));
+-
+- switch (key->alg) {
+- case ALG_WEP:
+- hk.kv_type = ATH9K_CIPHER_WEP;
+- break;
+- case ALG_TKIP:
+- hk.kv_type = ATH9K_CIPHER_TKIP;
+- break;
+- case ALG_CCMP:
+- hk.kv_type = ATH9K_CIPHER_AES_CCM;
+- break;
+- default:
+- return -EOPNOTSUPP;
+- }
+-
+- hk.kv_len = key->keylen;
+- memcpy(hk.kv_val, key->key, key->keylen);
+-
+- if (!(key->flags & IEEE80211_KEY_FLAG_PAIRWISE)) {
+- /* For now, use the default keys for broadcast keys. This may
+- * need to change with virtual interfaces. */
+- idx = key->keyidx;
+- } else if (key->keyidx) {
+- if (WARN_ON(!sta))
+- return -EOPNOTSUPP;
+- mac = sta->addr;
+-
+- if (vif->type != NL80211_IFTYPE_AP) {
+- /* Only keyidx 0 should be used with unicast key, but
+- * allow this for client mode for now. */
+- idx = key->keyidx;
+- } else
+- return -EIO;
+- } else {
+- if (WARN_ON(!sta))
+- return -EOPNOTSUPP;
+- mac = sta->addr;
+-
+- if (key->alg == ALG_TKIP)
+- idx = ath_reserve_key_cache_slot_tkip(common);
+- else
+- idx = ath_reserve_key_cache_slot(common);
+- if (idx < 0)
+- return -ENOSPC; /* no free key cache entries */
+- }
+-
+- if (key->alg == ALG_TKIP)
+- ret = ath_setkey_tkip(common, idx, key->key, &hk, mac,
+- vif->type == NL80211_IFTYPE_AP);
+- else
+- ret = ath9k_hw_set_keycache_entry(ah, idx, &hk, mac);
+-
+- if (!ret)
+- return -EIO;
+-
+- set_bit(idx, common->keymap);
+- if (key->alg == ALG_TKIP) {
+- set_bit(idx + 64, common->keymap);
+- if (common->splitmic) {
+- set_bit(idx + 32, common->keymap);
+- set_bit(idx + 64 + 32, common->keymap);
+- }
+- }
+-
+- return idx;
+-}
+-
+-static void ath_key_delete(struct ath_common *common, struct ieee80211_key_conf *key)
+-{
+- struct ath_hw *ah = common->ah;
+-
+- ath9k_hw_keyreset(ah, key->hw_key_idx);
+- if (key->hw_key_idx < IEEE80211_WEP_NKID)
+- return;
+-
+- clear_bit(key->hw_key_idx, common->keymap);
+- if (key->alg != ALG_TKIP)
+- return;
+-
+- clear_bit(key->hw_key_idx + 64, common->keymap);
+- if (common->splitmic) {
+- ath9k_hw_keyreset(ah, key->hw_key_idx + 32);
+- clear_bit(key->hw_key_idx + 32, common->keymap);
+- clear_bit(key->hw_key_idx + 64 + 32, common->keymap);
+- }
+-}
+-
+ static void ath9k_bss_assoc_info(struct ath_softc *sc,
+ struct ieee80211_vif *vif,
+ struct ieee80211_bss_conf *bss_conf)
+@@ -1814,7 +1586,7 @@ static int ath9k_set_key(struct ieee8021
+
+ switch (cmd) {
+ case SET_KEY:
+- ret = ath_key_config(common, vif, sta, key);
++ ret = ath9k_cmn_key_config(common, vif, sta, key);
+ if (ret >= 0) {
+ key->hw_key_idx = ret;
+ /* push IV and Michael MIC generation to stack */
+@@ -1827,7 +1599,7 @@ static int ath9k_set_key(struct ieee8021
+ }
+ break;
+ case DISABLE_KEY:
+- ath_key_delete(common, key);
++ ath9k_cmn_key_delete(common, key);
+ break;
+ default:
+ ret = -EINVAL;