| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- --- 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;
|