diff options
-rw-r--r-- | include/net/mac80211.h | 30 | ||||
-rw-r--r-- | net/mac80211/tkip.c | 53 |
2 files changed, 83 insertions, 0 deletions
diff --git a/include/net/mac80211.h b/include/net/mac80211.h index 5ab6a350ee6d..2a134582fc16 100644 --- a/include/net/mac80211.h +++ b/include/net/mac80211.h | |||
@@ -644,6 +644,21 @@ enum sta_notify_cmd { | |||
644 | }; | 644 | }; |
645 | 645 | ||
646 | /** | 646 | /** |
647 | * enum ieee80211_tkip_key_type - get tkip key | ||
648 | * | ||
649 | * Used by drivers which need to get a tkip key for skb. Some drivers need a | ||
650 | * phase 1 key, others need a phase 2 key. A single function allows the driver | ||
651 | * to get the key, this enum indicates what type of key is required. | ||
652 | * | ||
653 | * @IEEE80211_TKIP_P1_KEY: the driver needs a phase 1 key | ||
654 | * @IEEE80211_TKIP_P2_KEY: the driver needs a phase 2 key | ||
655 | */ | ||
656 | enum ieee80211_tkip_key_type { | ||
657 | IEEE80211_TKIP_P1_KEY, | ||
658 | IEEE80211_TKIP_P2_KEY, | ||
659 | }; | ||
660 | |||
661 | /** | ||
647 | * enum ieee80211_hw_flags - hardware flags | 662 | * enum ieee80211_hw_flags - hardware flags |
648 | * | 663 | * |
649 | * These flags are used to indicate hardware capabilities to | 664 | * These flags are used to indicate hardware capabilities to |
@@ -1472,6 +1487,21 @@ int ieee80211_get_hdrlen_from_skb(const struct sk_buff *skb); | |||
1472 | int ieee80211_get_hdrlen(u16 fc); | 1487 | int ieee80211_get_hdrlen(u16 fc); |
1473 | 1488 | ||
1474 | /** | 1489 | /** |
1490 | * ieee80211_get_tkip_key - get a TKIP rc4 for skb | ||
1491 | * | ||
1492 | * This function computes a TKIP rc4 key for an skb. It computes | ||
1493 | * a phase 1 key if needed (iv16 wraps around). This function is to | ||
1494 | * be used by drivers which can do HW encryption but need to compute | ||
1495 | * to phase 1/2 key in SW. | ||
1496 | * | ||
1497 | * @keyconf: the parameter passed with the set key | ||
1498 | * @skb: the skb for which the key is needed | ||
1499 | * @rc4key: a buffer to which the key will be written | ||
1500 | */ | ||
1501 | void ieee80211_get_tkip_key(struct ieee80211_key_conf *keyconf, | ||
1502 | struct sk_buff *skb, | ||
1503 | enum ieee80211_tkip_key_type type, u8 *key); | ||
1504 | /** | ||
1475 | * ieee80211_wake_queue - wake specific queue | 1505 | * ieee80211_wake_queue - wake specific queue |
1476 | * @hw: pointer as obtained from ieee80211_alloc_hw(). | 1506 | * @hw: pointer as obtained from ieee80211_alloc_hw(). |
1477 | * @queue: queue number (counted from zero). | 1507 | * @queue: queue number (counted from zero). |
diff --git a/net/mac80211/tkip.c b/net/mac80211/tkip.c index 3abe194e4d55..5c36b2df3faf 100644 --- a/net/mac80211/tkip.c +++ b/net/mac80211/tkip.c | |||
@@ -214,6 +214,59 @@ void ieee80211_tkip_gen_rc4key(struct ieee80211_key *key, u8 *ta, | |||
214 | key->u.tkip.iv16, rc4key); | 214 | key->u.tkip.iv16, rc4key); |
215 | } | 215 | } |
216 | 216 | ||
217 | void ieee80211_get_tkip_key(struct ieee80211_key_conf *keyconf, | ||
218 | struct sk_buff *skb, enum ieee80211_tkip_key_type type, | ||
219 | u8 *outkey) | ||
220 | { | ||
221 | struct ieee80211_key *key = (struct ieee80211_key *) | ||
222 | container_of(keyconf, struct ieee80211_key, conf); | ||
223 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) skb->data; | ||
224 | u8 *data = (u8 *) hdr; | ||
225 | u16 fc = le16_to_cpu(hdr->frame_control); | ||
226 | int hdr_len = ieee80211_get_hdrlen(fc); | ||
227 | u8 *ta = hdr->addr2; | ||
228 | u16 iv16; | ||
229 | u32 iv32; | ||
230 | |||
231 | iv16 = data[hdr_len] << 8; | ||
232 | iv16 += data[hdr_len + 2]; | ||
233 | iv32 = data[hdr_len + 4] + | ||
234 | (data[hdr_len + 5] >> 8) + | ||
235 | (data[hdr_len + 6] >> 16) + | ||
236 | (data[hdr_len + 7] >> 24); | ||
237 | |||
238 | #ifdef CONFIG_TKIP_DEBUG | ||
239 | printk(KERN_DEBUG "TKIP encrypt: iv16 = 0x%04x, iv32 = 0x%08x\n", | ||
240 | iv16, iv32); | ||
241 | |||
242 | if (iv32 != key->u.tkip.iv32) { | ||
243 | printk(KERN_DEBUG "skb: iv32 = 0x%08x key: iv32 = 0x%08x\n", | ||
244 | iv32, key->u.tkip.iv32); | ||
245 | printk(KERN_DEBUG "Wrap around of iv16 in the middle of a " | ||
246 | "fragmented packet\n"); | ||
247 | } | ||
248 | #endif /* CONFIG_TKIP_DEBUG */ | ||
249 | |||
250 | /* Update the p1k only when the iv16 in the packet wraps around, this | ||
251 | * might occur after the wrap around of iv16 in the key in case of | ||
252 | * fragmented packets. */ | ||
253 | if (iv16 == 0 || !key->u.tkip.tx_initialized) { | ||
254 | /* IV16 wrapped around - perform TKIP phase 1 */ | ||
255 | tkip_mixing_phase1(ta, &key->conf.key[ALG_TKIP_TEMP_ENCR_KEY], | ||
256 | iv32, key->u.tkip.p1k); | ||
257 | key->u.tkip.tx_initialized = 1; | ||
258 | } | ||
259 | |||
260 | if (type == IEEE80211_TKIP_P1_KEY) { | ||
261 | memcpy(outkey, key->u.tkip.p1k, sizeof(u16) * 5); | ||
262 | return; | ||
263 | } | ||
264 | |||
265 | tkip_mixing_phase2(key->u.tkip.p1k, | ||
266 | &key->conf.key[ALG_TKIP_TEMP_ENCR_KEY], iv16, outkey); | ||
267 | } | ||
268 | EXPORT_SYMBOL(ieee80211_get_tkip_key); | ||
269 | |||
217 | /* Encrypt packet payload with TKIP using @key. @pos is a pointer to the | 270 | /* Encrypt packet payload with TKIP using @key. @pos is a pointer to the |
218 | * beginning of the buffer containing payload. This payload must include | 271 | * beginning of the buffer containing payload. This payload must include |
219 | * headroom of eight octets for IV and Ext. IV and taildroom of four octets | 272 | * headroom of eight octets for IV and Ext. IV and taildroom of four octets |