diff options
author | Emmanuel Grumbach <emmanuel.grumbach@intel.com> | 2008-04-15 00:16:07 -0400 |
---|---|---|
committer | John W. Linville <linville@tuxdriver.com> | 2008-04-16 15:59:58 -0400 |
commit | 0211ddda9deb681a804572936cd49e466a1aa88b (patch) | |
tree | a9bb069458364ba40c8c7024352f9ac36a8bdfcf | |
parent | 6974e36356524fa856435cb1be40aaffbac9601a (diff) |
iwlwifi: add 1X HW WEP support
This patch adds support for HW encryption/decryption in 1X WEP.
Signed-off-by: Emmanuel Grumbach <emmanuel.grumbach@intel.com>
Signed-off-by: Tomas Winkler <tomas.winkler@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
-rw-r--r-- | drivers/net/wireless/iwlwifi/iwl-4965.h | 1 | ||||
-rw-r--r-- | drivers/net/wireless/iwlwifi/iwl-sta.c | 47 | ||||
-rw-r--r-- | drivers/net/wireless/iwlwifi/iwl-sta.h | 3 | ||||
-rw-r--r-- | drivers/net/wireless/iwlwifi/iwl4965-base.c | 9 |
4 files changed, 58 insertions, 2 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl-4965.h b/drivers/net/wireless/iwlwifi/iwl-4965.h index 93df29e4689a..23ab5236504b 100644 --- a/drivers/net/wireless/iwlwifi/iwl-4965.h +++ b/drivers/net/wireless/iwlwifi/iwl-4965.h | |||
@@ -461,6 +461,7 @@ struct iwl4965_tid_data { | |||
461 | struct iwl4965_hw_key { | 461 | struct iwl4965_hw_key { |
462 | enum ieee80211_key_alg alg; | 462 | enum ieee80211_key_alg alg; |
463 | int keylen; | 463 | int keylen; |
464 | u8 keyidx; | ||
464 | struct ieee80211_key_conf *conf; | 465 | struct ieee80211_key_conf *conf; |
465 | u8 key[32]; | 466 | u8 key[32]; |
466 | }; | 467 | }; |
diff --git a/drivers/net/wireless/iwlwifi/iwl-sta.c b/drivers/net/wireless/iwlwifi/iwl-sta.c index a48e228e2ffd..cb964196ad2a 100644 --- a/drivers/net/wireless/iwlwifi/iwl-sta.c +++ b/drivers/net/wireless/iwlwifi/iwl-sta.c | |||
@@ -35,6 +35,7 @@ | |||
35 | #include "iwl-sta.h" | 35 | #include "iwl-sta.h" |
36 | #include "iwl-io.h" | 36 | #include "iwl-io.h" |
37 | #include "iwl-helpers.h" | 37 | #include "iwl-helpers.h" |
38 | #include "iwl-4965.h" | ||
38 | 39 | ||
39 | int iwl_send_static_wepkey_cmd(struct iwl_priv *priv, u8 send_if_empty) | 40 | int iwl_send_static_wepkey_cmd(struct iwl_priv *priv, u8 send_if_empty) |
40 | { | 41 | { |
@@ -117,3 +118,49 @@ int iwl_set_default_wep_key(struct iwl_priv *priv, | |||
117 | return ret; | 118 | return ret; |
118 | } | 119 | } |
119 | 120 | ||
121 | int iwl_set_wep_dynamic_key_info(struct iwl_priv *priv, | ||
122 | struct ieee80211_key_conf *keyconf, | ||
123 | u8 sta_id) | ||
124 | { | ||
125 | unsigned long flags; | ||
126 | __le16 key_flags = 0; | ||
127 | int ret; | ||
128 | |||
129 | keyconf->flags &= ~IEEE80211_KEY_FLAG_GENERATE_IV; | ||
130 | keyconf->hw_key_idx = keyconf->keyidx; | ||
131 | |||
132 | key_flags |= (STA_KEY_FLG_WEP | STA_KEY_FLG_MAP_KEY_MSK); | ||
133 | key_flags |= cpu_to_le16(keyconf->keyidx << STA_KEY_FLG_KEYID_POS); | ||
134 | key_flags &= ~STA_KEY_FLG_INVALID; | ||
135 | |||
136 | if (keyconf->keylen == WEP_KEY_LEN_128) | ||
137 | key_flags |= STA_KEY_FLG_KEY_SIZE_MSK; | ||
138 | |||
139 | if (sta_id == priv->hw_setting.bcast_sta_id) | ||
140 | key_flags |= STA_KEY_MULTICAST_MSK; | ||
141 | |||
142 | spin_lock_irqsave(&priv->sta_lock, flags); | ||
143 | |||
144 | priv->stations[sta_id].keyinfo.alg = keyconf->alg; | ||
145 | priv->stations[sta_id].keyinfo.keylen = keyconf->keylen; | ||
146 | priv->stations[sta_id].keyinfo.keyidx = keyconf->keyidx; | ||
147 | |||
148 | memcpy(priv->stations[sta_id].keyinfo.key, | ||
149 | keyconf->key, keyconf->keylen); | ||
150 | |||
151 | memcpy(&priv->stations[sta_id].sta.key.key[3], | ||
152 | keyconf->key, keyconf->keylen); | ||
153 | |||
154 | priv->stations[sta_id].sta.key.key_offset = sta_id % 8; /* FIXME */ | ||
155 | priv->stations[sta_id].sta.key.key_flags = key_flags; | ||
156 | |||
157 | priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_KEY_MASK; | ||
158 | priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK; | ||
159 | |||
160 | ret = iwl4965_send_add_station(priv, | ||
161 | &priv->stations[sta_id].sta, CMD_ASYNC); | ||
162 | |||
163 | spin_unlock_irqrestore(&priv->sta_lock, flags); | ||
164 | |||
165 | return ret; | ||
166 | } | ||
diff --git a/drivers/net/wireless/iwlwifi/iwl-sta.h b/drivers/net/wireless/iwlwifi/iwl-sta.h index 50e9f1470d8d..78e0254c93b5 100644 --- a/drivers/net/wireless/iwlwifi/iwl-sta.h +++ b/drivers/net/wireless/iwlwifi/iwl-sta.h | |||
@@ -42,5 +42,8 @@ int iwl_remove_default_wep_key(struct iwl_priv *priv, | |||
42 | struct ieee80211_key_conf *key); | 42 | struct ieee80211_key_conf *key); |
43 | int iwl_set_default_wep_key(struct iwl_priv *priv, | 43 | int iwl_set_default_wep_key(struct iwl_priv *priv, |
44 | struct ieee80211_key_conf *key); | 44 | struct ieee80211_key_conf *key); |
45 | int iwl_set_wep_dynamic_key_info(struct iwl_priv *priv, | ||
46 | struct ieee80211_key_conf *keyconf, | ||
47 | u8 sta_id); | ||
45 | 48 | ||
46 | #endif /* __iwl_sta_h__ */ | 49 | #endif /* __iwl_sta_h__ */ |
diff --git a/drivers/net/wireless/iwlwifi/iwl4965-base.c b/drivers/net/wireless/iwlwifi/iwl4965-base.c index dfd2b75492e6..37ab1c565ce1 100644 --- a/drivers/net/wireless/iwlwifi/iwl4965-base.c +++ b/drivers/net/wireless/iwlwifi/iwl4965-base.c | |||
@@ -1214,7 +1214,7 @@ static int iwl4965_set_dynamic_key(struct iwl_priv *priv, | |||
1214 | ret = iwl4965_set_tkip_dynamic_key_info(priv, key, sta_id); | 1214 | ret = iwl4965_set_tkip_dynamic_key_info(priv, key, sta_id); |
1215 | break; | 1215 | break; |
1216 | case ALG_WEP: | 1216 | case ALG_WEP: |
1217 | ret = -EOPNOTSUPP; | 1217 | ret = iwl_set_wep_dynamic_key_info(priv, key, sta_id); |
1218 | break; | 1218 | break; |
1219 | default: | 1219 | default: |
1220 | IWL_ERROR("Unknown alg: %s alg = %d\n", __func__, key->alg); | 1220 | IWL_ERROR("Unknown alg: %s alg = %d\n", __func__, key->alg); |
@@ -2138,7 +2138,12 @@ static void iwl4965_build_tx_cmd_hwcrypto(struct iwl_priv *priv, | |||
2138 | if (wepkey->key_size == WEP_KEY_LEN_128) | 2138 | if (wepkey->key_size == WEP_KEY_LEN_128) |
2139 | cmd->cmd.tx.sec_ctl |= TX_CMD_SEC_KEY128; | 2139 | cmd->cmd.tx.sec_ctl |= TX_CMD_SEC_KEY128; |
2140 | } else { | 2140 | } else { |
2141 | IWL_ERROR("No support for WEP key mappings key\n"); | 2141 | /* the WEP key was sent as dynamic */ |
2142 | keyidx = keyinfo->keyidx; | ||
2143 | memcpy(&cmd->cmd.tx.key[3], keyinfo->key, | ||
2144 | keyinfo->keylen); | ||
2145 | if (keyinfo->keylen == WEP_KEY_LEN_128) | ||
2146 | cmd->cmd.tx.sec_ctl |= TX_CMD_SEC_KEY128; | ||
2142 | } | 2147 | } |
2143 | 2148 | ||
2144 | cmd->cmd.tx.sec_ctl |= (TX_CMD_SEC_WEP | | 2149 | cmd->cmd.tx.sec_ctl |= (TX_CMD_SEC_WEP | |