aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/ath5k/pcu.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/wireless/ath5k/pcu.c')
-rw-r--r--drivers/net/wireless/ath5k/pcu.c64
1 files changed, 59 insertions, 5 deletions
diff --git a/drivers/net/wireless/ath5k/pcu.c b/drivers/net/wireless/ath5k/pcu.c
index 9c6698520edc..d7f0c1017bda 100644
--- a/drivers/net/wireless/ath5k/pcu.c
+++ b/drivers/net/wireless/ath5k/pcu.c
@@ -1021,17 +1021,29 @@ int ath5k_hw_set_key(struct ath5k_hw *ah, u16 entry,
1021 const struct ieee80211_key_conf *key, const u8 *mac) 1021 const struct ieee80211_key_conf *key, const u8 *mac)
1022{ 1022{
1023 unsigned int i; 1023 unsigned int i;
1024 int keylen;
1024 __le32 key_v[5] = {}; 1025 __le32 key_v[5] = {};
1026 __le32 key0 = 0, key1 = 0;
1027 __le32 *rxmic, *txmic;
1025 u32 keytype; 1028 u32 keytype;
1029 u16 micentry = entry + AR5K_KEYTABLE_MIC_OFFSET;
1030 bool is_tkip;
1026 1031
1027 ATH5K_TRACE(ah->ah_sc); 1032 ATH5K_TRACE(ah->ah_sc);
1028 1033
1029 /* key->keylen comes in from mac80211 in bytes */ 1034 is_tkip = (key->alg == ALG_TKIP);
1035
1036 /*
1037 * key->keylen comes in from mac80211 in bytes.
1038 * TKIP is 128 bit + 128 bit mic
1039 */
1040 keylen = (is_tkip) ? (128 / 8) : key->keylen;
1030 1041
1031 if (key->keylen > AR5K_KEYTABLE_SIZE / 8) 1042 if (entry > AR5K_KEYTABLE_SIZE ||
1043 (is_tkip && micentry > AR5K_KEYTABLE_SIZE))
1032 return -EOPNOTSUPP; 1044 return -EOPNOTSUPP;
1033 1045
1034 switch (key->keylen) { 1046 switch (keylen) {
1035 /* WEP 40-bit = 40-bit entered key + 24 bit IV = 64-bit */ 1047 /* WEP 40-bit = 40-bit entered key + 24 bit IV = 64-bit */
1036 case 40 / 8: 1048 case 40 / 8:
1037 memcpy(&key_v[0], key->key, 5); 1049 memcpy(&key_v[0], key->key, 5);
@@ -1045,24 +1057,66 @@ int ath5k_hw_set_key(struct ath5k_hw *ah, u16 entry,
1045 memcpy(&key_v[4], &key->key[12], 1); 1057 memcpy(&key_v[4], &key->key[12], 1);
1046 keytype = AR5K_KEYTABLE_TYPE_104; 1058 keytype = AR5K_KEYTABLE_TYPE_104;
1047 break; 1059 break;
1048 /* WEP 128-bit = 128-bit entered key + 24 bit IV = 152-bit */ 1060 /* WEP/TKIP 128-bit = 128-bit entered key + 24 bit IV = 152-bit */
1049 case 128 / 8: 1061 case 128 / 8:
1050 memcpy(&key_v[0], &key->key[0], 6); 1062 memcpy(&key_v[0], &key->key[0], 6);
1051 memcpy(&key_v[2], &key->key[6], 6); 1063 memcpy(&key_v[2], &key->key[6], 6);
1052 memcpy(&key_v[4], &key->key[12], 4); 1064 memcpy(&key_v[4], &key->key[12], 4);
1053 keytype = AR5K_KEYTABLE_TYPE_128; 1065 keytype = is_tkip ?
1066 AR5K_KEYTABLE_TYPE_TKIP :
1067 AR5K_KEYTABLE_TYPE_128;
1054 break; 1068 break;
1055 1069
1056 default: 1070 default:
1057 return -EINVAL; /* shouldn't happen */ 1071 return -EINVAL; /* shouldn't happen */
1058 } 1072 }
1059 1073
1074 /* intentionally corrupt key until mic is installed */
1075 if (is_tkip) {
1076 key0 = key_v[0] = ~key_v[0];
1077 key1 = key_v[1] = ~key_v[1];
1078 }
1079
1060 for (i = 0; i < ARRAY_SIZE(key_v); i++) 1080 for (i = 0; i < ARRAY_SIZE(key_v); i++)
1061 ath5k_hw_reg_write(ah, le32_to_cpu(key_v[i]), 1081 ath5k_hw_reg_write(ah, le32_to_cpu(key_v[i]),
1062 AR5K_KEYTABLE_OFF(entry, i)); 1082 AR5K_KEYTABLE_OFF(entry, i));
1063 1083
1064 ath5k_hw_reg_write(ah, keytype, AR5K_KEYTABLE_TYPE(entry)); 1084 ath5k_hw_reg_write(ah, keytype, AR5K_KEYTABLE_TYPE(entry));
1065 1085
1086 if (is_tkip) {
1087 /* Install rx/tx MIC */
1088 rxmic = (__le32 *) &key->key[16];
1089 txmic = (__le32 *) &key->key[24];
1090#if 0
1091 /* MISC_MODE register & 0x04 - for mac srev >= griffin */
1092 key_v[0] = rxmic[0];
1093 key_v[1] = (txmic[0] >> 16) & 0xffff;
1094 key_v[2] = rxmic[1];
1095 key_v[3] = txmic[0] & 0xffff;
1096 key_v[4] = txmic[1];
1097#else
1098 key_v[0] = rxmic[0];
1099 key_v[1] = 0;
1100 key_v[2] = rxmic[1];
1101 key_v[3] = 0;
1102 key_v[4] = 0;
1103#endif
1104 for (i = 0; i < ARRAY_SIZE(key_v); i++)
1105 ath5k_hw_reg_write(ah, le32_to_cpu(key_v[i]),
1106 AR5K_KEYTABLE_OFF(micentry, i));
1107
1108 ath5k_hw_reg_write(ah, AR5K_KEYTABLE_TYPE_NULL,
1109 AR5K_KEYTABLE_TYPE(micentry));
1110 ath5k_hw_reg_write(ah, 0, AR5K_KEYTABLE_MAC0(micentry));
1111 ath5k_hw_reg_write(ah, 0, AR5K_KEYTABLE_MAC1(micentry));
1112
1113 /* restore first 2 words of key */
1114 ath5k_hw_reg_write(ah, le32_to_cpu(~key0),
1115 AR5K_KEYTABLE_OFF(entry, 0));
1116 ath5k_hw_reg_write(ah, le32_to_cpu(~key1),
1117 AR5K_KEYTABLE_OFF(entry, 1));
1118 }
1119
1066 return ath5k_hw_set_key_lladdr(ah, entry, mac); 1120 return ath5k_hw_set_key_lladdr(ah, entry, mac);
1067} 1121}
1068 1122