1*274bfb8dSJohn W. Linville /* 2*274bfb8dSJohn W. Linville * lib80211 crypt: host-based CCMP encryption implementation for lib80211 3*274bfb8dSJohn W. Linville * 4*274bfb8dSJohn W. Linville * Copyright (c) 2003-2004, Jouni Malinen <j@w1.fi> 5*274bfb8dSJohn W. Linville * Copyright (c) 2008, John W. Linville <linville@tuxdriver.com> 6*274bfb8dSJohn W. Linville * 7*274bfb8dSJohn W. Linville * This program is free software; you can redistribute it and/or modify 8*274bfb8dSJohn W. Linville * it under the terms of the GNU General Public License version 2 as 9*274bfb8dSJohn W. Linville * published by the Free Software Foundation. See README and COPYING for 10*274bfb8dSJohn W. Linville * more details. 11*274bfb8dSJohn W. Linville */ 12*274bfb8dSJohn W. Linville 13*274bfb8dSJohn W. Linville #include <linux/kernel.h> 14*274bfb8dSJohn W. Linville #include <linux/err.h> 15*274bfb8dSJohn W. Linville #include <linux/module.h> 16*274bfb8dSJohn W. Linville #include <linux/init.h> 17*274bfb8dSJohn W. Linville #include <linux/slab.h> 18*274bfb8dSJohn W. Linville #include <linux/random.h> 19*274bfb8dSJohn W. Linville #include <linux/skbuff.h> 20*274bfb8dSJohn W. Linville #include <linux/netdevice.h> 21*274bfb8dSJohn W. Linville #include <linux/if_ether.h> 22*274bfb8dSJohn W. Linville #include <linux/if_arp.h> 23*274bfb8dSJohn W. Linville #include <asm/string.h> 24*274bfb8dSJohn W. Linville #include <linux/wireless.h> 25*274bfb8dSJohn W. Linville 26*274bfb8dSJohn W. Linville #include <linux/ieee80211.h> 27*274bfb8dSJohn W. Linville 28*274bfb8dSJohn W. Linville #include <linux/crypto.h> 29*274bfb8dSJohn W. Linville 30*274bfb8dSJohn W. Linville #include <net/lib80211.h> 31*274bfb8dSJohn W. Linville 32*274bfb8dSJohn W. Linville MODULE_AUTHOR("Jouni Malinen"); 33*274bfb8dSJohn W. Linville MODULE_DESCRIPTION("Host AP crypt: CCMP"); 34*274bfb8dSJohn W. Linville MODULE_LICENSE("GPL"); 35*274bfb8dSJohn W. Linville 36*274bfb8dSJohn W. Linville #define AES_BLOCK_LEN 16 37*274bfb8dSJohn W. Linville #define CCMP_HDR_LEN 8 38*274bfb8dSJohn W. Linville #define CCMP_MIC_LEN 8 39*274bfb8dSJohn W. Linville #define CCMP_TK_LEN 16 40*274bfb8dSJohn W. Linville #define CCMP_PN_LEN 6 41*274bfb8dSJohn W. Linville 42*274bfb8dSJohn W. Linville struct lib80211_ccmp_data { 43*274bfb8dSJohn W. Linville u8 key[CCMP_TK_LEN]; 44*274bfb8dSJohn W. Linville int key_set; 45*274bfb8dSJohn W. Linville 46*274bfb8dSJohn W. Linville u8 tx_pn[CCMP_PN_LEN]; 47*274bfb8dSJohn W. Linville u8 rx_pn[CCMP_PN_LEN]; 48*274bfb8dSJohn W. Linville 49*274bfb8dSJohn W. Linville u32 dot11RSNAStatsCCMPFormatErrors; 50*274bfb8dSJohn W. Linville u32 dot11RSNAStatsCCMPReplays; 51*274bfb8dSJohn W. Linville u32 dot11RSNAStatsCCMPDecryptErrors; 52*274bfb8dSJohn W. Linville 53*274bfb8dSJohn W. Linville int key_idx; 54*274bfb8dSJohn W. Linville 55*274bfb8dSJohn W. Linville struct crypto_cipher *tfm; 56*274bfb8dSJohn W. Linville 57*274bfb8dSJohn W. Linville /* scratch buffers for virt_to_page() (crypto API) */ 58*274bfb8dSJohn W. Linville u8 tx_b0[AES_BLOCK_LEN], tx_b[AES_BLOCK_LEN], 59*274bfb8dSJohn W. Linville tx_e[AES_BLOCK_LEN], tx_s0[AES_BLOCK_LEN]; 60*274bfb8dSJohn W. Linville u8 rx_b0[AES_BLOCK_LEN], rx_b[AES_BLOCK_LEN], rx_a[AES_BLOCK_LEN]; 61*274bfb8dSJohn W. Linville }; 62*274bfb8dSJohn W. Linville 63*274bfb8dSJohn W. Linville static inline void lib80211_ccmp_aes_encrypt(struct crypto_cipher *tfm, 64*274bfb8dSJohn W. Linville const u8 pt[16], u8 ct[16]) 65*274bfb8dSJohn W. Linville { 66*274bfb8dSJohn W. Linville crypto_cipher_encrypt_one(tfm, ct, pt); 67*274bfb8dSJohn W. Linville } 68*274bfb8dSJohn W. Linville 69*274bfb8dSJohn W. Linville static void *lib80211_ccmp_init(int key_idx) 70*274bfb8dSJohn W. Linville { 71*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *priv; 72*274bfb8dSJohn W. Linville 73*274bfb8dSJohn W. Linville priv = kzalloc(sizeof(*priv), GFP_ATOMIC); 74*274bfb8dSJohn W. Linville if (priv == NULL) 75*274bfb8dSJohn W. Linville goto fail; 76*274bfb8dSJohn W. Linville priv->key_idx = key_idx; 77*274bfb8dSJohn W. Linville 78*274bfb8dSJohn W. Linville priv->tfm = crypto_alloc_cipher("aes", 0, CRYPTO_ALG_ASYNC); 79*274bfb8dSJohn W. Linville if (IS_ERR(priv->tfm)) { 80*274bfb8dSJohn W. Linville printk(KERN_DEBUG "lib80211_crypt_ccmp: could not allocate " 81*274bfb8dSJohn W. Linville "crypto API aes\n"); 82*274bfb8dSJohn W. Linville priv->tfm = NULL; 83*274bfb8dSJohn W. Linville goto fail; 84*274bfb8dSJohn W. Linville } 85*274bfb8dSJohn W. Linville 86*274bfb8dSJohn W. Linville return priv; 87*274bfb8dSJohn W. Linville 88*274bfb8dSJohn W. Linville fail: 89*274bfb8dSJohn W. Linville if (priv) { 90*274bfb8dSJohn W. Linville if (priv->tfm) 91*274bfb8dSJohn W. Linville crypto_free_cipher(priv->tfm); 92*274bfb8dSJohn W. Linville kfree(priv); 93*274bfb8dSJohn W. Linville } 94*274bfb8dSJohn W. Linville 95*274bfb8dSJohn W. Linville return NULL; 96*274bfb8dSJohn W. Linville } 97*274bfb8dSJohn W. Linville 98*274bfb8dSJohn W. Linville static void lib80211_ccmp_deinit(void *priv) 99*274bfb8dSJohn W. Linville { 100*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *_priv = priv; 101*274bfb8dSJohn W. Linville if (_priv && _priv->tfm) 102*274bfb8dSJohn W. Linville crypto_free_cipher(_priv->tfm); 103*274bfb8dSJohn W. Linville kfree(priv); 104*274bfb8dSJohn W. Linville } 105*274bfb8dSJohn W. Linville 106*274bfb8dSJohn W. Linville static inline void xor_block(u8 * b, u8 * a, size_t len) 107*274bfb8dSJohn W. Linville { 108*274bfb8dSJohn W. Linville int i; 109*274bfb8dSJohn W. Linville for (i = 0; i < len; i++) 110*274bfb8dSJohn W. Linville b[i] ^= a[i]; 111*274bfb8dSJohn W. Linville } 112*274bfb8dSJohn W. Linville 113*274bfb8dSJohn W. Linville static void ccmp_init_blocks(struct crypto_cipher *tfm, 114*274bfb8dSJohn W. Linville struct ieee80211_hdr *hdr, 115*274bfb8dSJohn W. Linville u8 * pn, size_t dlen, u8 * b0, u8 * auth, u8 * s0) 116*274bfb8dSJohn W. Linville { 117*274bfb8dSJohn W. Linville u8 *pos, qc = 0; 118*274bfb8dSJohn W. Linville size_t aad_len; 119*274bfb8dSJohn W. Linville int a4_included, qc_included; 120*274bfb8dSJohn W. Linville u8 aad[2 * AES_BLOCK_LEN]; 121*274bfb8dSJohn W. Linville 122*274bfb8dSJohn W. Linville a4_included = ieee80211_has_a4(hdr->frame_control); 123*274bfb8dSJohn W. Linville qc_included = ieee80211_is_data_qos(hdr->frame_control); 124*274bfb8dSJohn W. Linville 125*274bfb8dSJohn W. Linville aad_len = 22; 126*274bfb8dSJohn W. Linville if (a4_included) 127*274bfb8dSJohn W. Linville aad_len += 6; 128*274bfb8dSJohn W. Linville if (qc_included) { 129*274bfb8dSJohn W. Linville pos = (u8 *) & hdr->addr4; 130*274bfb8dSJohn W. Linville if (a4_included) 131*274bfb8dSJohn W. Linville pos += 6; 132*274bfb8dSJohn W. Linville qc = *pos & 0x0f; 133*274bfb8dSJohn W. Linville aad_len += 2; 134*274bfb8dSJohn W. Linville } 135*274bfb8dSJohn W. Linville 136*274bfb8dSJohn W. Linville /* CCM Initial Block: 137*274bfb8dSJohn W. Linville * Flag (Include authentication header, M=3 (8-octet MIC), 138*274bfb8dSJohn W. Linville * L=1 (2-octet Dlen)) 139*274bfb8dSJohn W. Linville * Nonce: 0x00 | A2 | PN 140*274bfb8dSJohn W. Linville * Dlen */ 141*274bfb8dSJohn W. Linville b0[0] = 0x59; 142*274bfb8dSJohn W. Linville b0[1] = qc; 143*274bfb8dSJohn W. Linville memcpy(b0 + 2, hdr->addr2, ETH_ALEN); 144*274bfb8dSJohn W. Linville memcpy(b0 + 8, pn, CCMP_PN_LEN); 145*274bfb8dSJohn W. Linville b0[14] = (dlen >> 8) & 0xff; 146*274bfb8dSJohn W. Linville b0[15] = dlen & 0xff; 147*274bfb8dSJohn W. Linville 148*274bfb8dSJohn W. Linville /* AAD: 149*274bfb8dSJohn W. Linville * FC with bits 4..6 and 11..13 masked to zero; 14 is always one 150*274bfb8dSJohn W. Linville * A1 | A2 | A3 151*274bfb8dSJohn W. Linville * SC with bits 4..15 (seq#) masked to zero 152*274bfb8dSJohn W. Linville * A4 (if present) 153*274bfb8dSJohn W. Linville * QC (if present) 154*274bfb8dSJohn W. Linville */ 155*274bfb8dSJohn W. Linville pos = (u8 *) hdr; 156*274bfb8dSJohn W. Linville aad[0] = 0; /* aad_len >> 8 */ 157*274bfb8dSJohn W. Linville aad[1] = aad_len & 0xff; 158*274bfb8dSJohn W. Linville aad[2] = pos[0] & 0x8f; 159*274bfb8dSJohn W. Linville aad[3] = pos[1] & 0xc7; 160*274bfb8dSJohn W. Linville memcpy(aad + 4, hdr->addr1, 3 * ETH_ALEN); 161*274bfb8dSJohn W. Linville pos = (u8 *) & hdr->seq_ctrl; 162*274bfb8dSJohn W. Linville aad[22] = pos[0] & 0x0f; 163*274bfb8dSJohn W. Linville aad[23] = 0; /* all bits masked */ 164*274bfb8dSJohn W. Linville memset(aad + 24, 0, 8); 165*274bfb8dSJohn W. Linville if (a4_included) 166*274bfb8dSJohn W. Linville memcpy(aad + 24, hdr->addr4, ETH_ALEN); 167*274bfb8dSJohn W. Linville if (qc_included) { 168*274bfb8dSJohn W. Linville aad[a4_included ? 30 : 24] = qc; 169*274bfb8dSJohn W. Linville /* rest of QC masked */ 170*274bfb8dSJohn W. Linville } 171*274bfb8dSJohn W. Linville 172*274bfb8dSJohn W. Linville /* Start with the first block and AAD */ 173*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(tfm, b0, auth); 174*274bfb8dSJohn W. Linville xor_block(auth, aad, AES_BLOCK_LEN); 175*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(tfm, auth, auth); 176*274bfb8dSJohn W. Linville xor_block(auth, &aad[AES_BLOCK_LEN], AES_BLOCK_LEN); 177*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(tfm, auth, auth); 178*274bfb8dSJohn W. Linville b0[0] &= 0x07; 179*274bfb8dSJohn W. Linville b0[14] = b0[15] = 0; 180*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(tfm, b0, s0); 181*274bfb8dSJohn W. Linville } 182*274bfb8dSJohn W. Linville 183*274bfb8dSJohn W. Linville static int lib80211_ccmp_hdr(struct sk_buff *skb, int hdr_len, 184*274bfb8dSJohn W. Linville u8 *aeskey, int keylen, void *priv) 185*274bfb8dSJohn W. Linville { 186*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *key = priv; 187*274bfb8dSJohn W. Linville int i; 188*274bfb8dSJohn W. Linville u8 *pos; 189*274bfb8dSJohn W. Linville 190*274bfb8dSJohn W. Linville if (skb_headroom(skb) < CCMP_HDR_LEN || skb->len < hdr_len) 191*274bfb8dSJohn W. Linville return -1; 192*274bfb8dSJohn W. Linville 193*274bfb8dSJohn W. Linville if (aeskey != NULL && keylen >= CCMP_TK_LEN) 194*274bfb8dSJohn W. Linville memcpy(aeskey, key->key, CCMP_TK_LEN); 195*274bfb8dSJohn W. Linville 196*274bfb8dSJohn W. Linville pos = skb_push(skb, CCMP_HDR_LEN); 197*274bfb8dSJohn W. Linville memmove(pos, pos + CCMP_HDR_LEN, hdr_len); 198*274bfb8dSJohn W. Linville pos += hdr_len; 199*274bfb8dSJohn W. Linville 200*274bfb8dSJohn W. Linville i = CCMP_PN_LEN - 1; 201*274bfb8dSJohn W. Linville while (i >= 0) { 202*274bfb8dSJohn W. Linville key->tx_pn[i]++; 203*274bfb8dSJohn W. Linville if (key->tx_pn[i] != 0) 204*274bfb8dSJohn W. Linville break; 205*274bfb8dSJohn W. Linville i--; 206*274bfb8dSJohn W. Linville } 207*274bfb8dSJohn W. Linville 208*274bfb8dSJohn W. Linville *pos++ = key->tx_pn[5]; 209*274bfb8dSJohn W. Linville *pos++ = key->tx_pn[4]; 210*274bfb8dSJohn W. Linville *pos++ = 0; 211*274bfb8dSJohn W. Linville *pos++ = (key->key_idx << 6) | (1 << 5) /* Ext IV included */ ; 212*274bfb8dSJohn W. Linville *pos++ = key->tx_pn[3]; 213*274bfb8dSJohn W. Linville *pos++ = key->tx_pn[2]; 214*274bfb8dSJohn W. Linville *pos++ = key->tx_pn[1]; 215*274bfb8dSJohn W. Linville *pos++ = key->tx_pn[0]; 216*274bfb8dSJohn W. Linville 217*274bfb8dSJohn W. Linville return CCMP_HDR_LEN; 218*274bfb8dSJohn W. Linville } 219*274bfb8dSJohn W. Linville 220*274bfb8dSJohn W. Linville static int lib80211_ccmp_encrypt(struct sk_buff *skb, int hdr_len, void *priv) 221*274bfb8dSJohn W. Linville { 222*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *key = priv; 223*274bfb8dSJohn W. Linville int data_len, i, blocks, last, len; 224*274bfb8dSJohn W. Linville u8 *pos, *mic; 225*274bfb8dSJohn W. Linville struct ieee80211_hdr *hdr; 226*274bfb8dSJohn W. Linville u8 *b0 = key->tx_b0; 227*274bfb8dSJohn W. Linville u8 *b = key->tx_b; 228*274bfb8dSJohn W. Linville u8 *e = key->tx_e; 229*274bfb8dSJohn W. Linville u8 *s0 = key->tx_s0; 230*274bfb8dSJohn W. Linville 231*274bfb8dSJohn W. Linville if (skb_tailroom(skb) < CCMP_MIC_LEN || skb->len < hdr_len) 232*274bfb8dSJohn W. Linville return -1; 233*274bfb8dSJohn W. Linville 234*274bfb8dSJohn W. Linville data_len = skb->len - hdr_len; 235*274bfb8dSJohn W. Linville len = lib80211_ccmp_hdr(skb, hdr_len, NULL, 0, priv); 236*274bfb8dSJohn W. Linville if (len < 0) 237*274bfb8dSJohn W. Linville return -1; 238*274bfb8dSJohn W. Linville 239*274bfb8dSJohn W. Linville pos = skb->data + hdr_len + CCMP_HDR_LEN; 240*274bfb8dSJohn W. Linville mic = skb_put(skb, CCMP_MIC_LEN); 241*274bfb8dSJohn W. Linville hdr = (struct ieee80211_hdr *)skb->data; 242*274bfb8dSJohn W. Linville ccmp_init_blocks(key->tfm, hdr, key->tx_pn, data_len, b0, b, s0); 243*274bfb8dSJohn W. Linville 244*274bfb8dSJohn W. Linville blocks = DIV_ROUND_UP(data_len, AES_BLOCK_LEN); 245*274bfb8dSJohn W. Linville last = data_len % AES_BLOCK_LEN; 246*274bfb8dSJohn W. Linville 247*274bfb8dSJohn W. Linville for (i = 1; i <= blocks; i++) { 248*274bfb8dSJohn W. Linville len = (i == blocks && last) ? last : AES_BLOCK_LEN; 249*274bfb8dSJohn W. Linville /* Authentication */ 250*274bfb8dSJohn W. Linville xor_block(b, pos, len); 251*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(key->tfm, b, b); 252*274bfb8dSJohn W. Linville /* Encryption, with counter */ 253*274bfb8dSJohn W. Linville b0[14] = (i >> 8) & 0xff; 254*274bfb8dSJohn W. Linville b0[15] = i & 0xff; 255*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(key->tfm, b0, e); 256*274bfb8dSJohn W. Linville xor_block(pos, e, len); 257*274bfb8dSJohn W. Linville pos += len; 258*274bfb8dSJohn W. Linville } 259*274bfb8dSJohn W. Linville 260*274bfb8dSJohn W. Linville for (i = 0; i < CCMP_MIC_LEN; i++) 261*274bfb8dSJohn W. Linville mic[i] = b[i] ^ s0[i]; 262*274bfb8dSJohn W. Linville 263*274bfb8dSJohn W. Linville return 0; 264*274bfb8dSJohn W. Linville } 265*274bfb8dSJohn W. Linville 266*274bfb8dSJohn W. Linville /* 267*274bfb8dSJohn W. Linville * deal with seq counter wrapping correctly. 268*274bfb8dSJohn W. Linville * refer to timer_after() for jiffies wrapping handling 269*274bfb8dSJohn W. Linville */ 270*274bfb8dSJohn W. Linville static inline int ccmp_replay_check(u8 *pn_n, u8 *pn_o) 271*274bfb8dSJohn W. Linville { 272*274bfb8dSJohn W. Linville u32 iv32_n, iv16_n; 273*274bfb8dSJohn W. Linville u32 iv32_o, iv16_o; 274*274bfb8dSJohn W. Linville 275*274bfb8dSJohn W. Linville iv32_n = (pn_n[0] << 24) | (pn_n[1] << 16) | (pn_n[2] << 8) | pn_n[3]; 276*274bfb8dSJohn W. Linville iv16_n = (pn_n[4] << 8) | pn_n[5]; 277*274bfb8dSJohn W. Linville 278*274bfb8dSJohn W. Linville iv32_o = (pn_o[0] << 24) | (pn_o[1] << 16) | (pn_o[2] << 8) | pn_o[3]; 279*274bfb8dSJohn W. Linville iv16_o = (pn_o[4] << 8) | pn_o[5]; 280*274bfb8dSJohn W. Linville 281*274bfb8dSJohn W. Linville if ((s32)iv32_n - (s32)iv32_o < 0 || 282*274bfb8dSJohn W. Linville (iv32_n == iv32_o && iv16_n <= iv16_o)) 283*274bfb8dSJohn W. Linville return 1; 284*274bfb8dSJohn W. Linville return 0; 285*274bfb8dSJohn W. Linville } 286*274bfb8dSJohn W. Linville 287*274bfb8dSJohn W. Linville static int lib80211_ccmp_decrypt(struct sk_buff *skb, int hdr_len, void *priv) 288*274bfb8dSJohn W. Linville { 289*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *key = priv; 290*274bfb8dSJohn W. Linville u8 keyidx, *pos; 291*274bfb8dSJohn W. Linville struct ieee80211_hdr *hdr; 292*274bfb8dSJohn W. Linville u8 *b0 = key->rx_b0; 293*274bfb8dSJohn W. Linville u8 *b = key->rx_b; 294*274bfb8dSJohn W. Linville u8 *a = key->rx_a; 295*274bfb8dSJohn W. Linville u8 pn[6]; 296*274bfb8dSJohn W. Linville int i, blocks, last, len; 297*274bfb8dSJohn W. Linville size_t data_len = skb->len - hdr_len - CCMP_HDR_LEN - CCMP_MIC_LEN; 298*274bfb8dSJohn W. Linville u8 *mic = skb->data + skb->len - CCMP_MIC_LEN; 299*274bfb8dSJohn W. Linville 300*274bfb8dSJohn W. Linville if (skb->len < hdr_len + CCMP_HDR_LEN + CCMP_MIC_LEN) { 301*274bfb8dSJohn W. Linville key->dot11RSNAStatsCCMPFormatErrors++; 302*274bfb8dSJohn W. Linville return -1; 303*274bfb8dSJohn W. Linville } 304*274bfb8dSJohn W. Linville 305*274bfb8dSJohn W. Linville hdr = (struct ieee80211_hdr *)skb->data; 306*274bfb8dSJohn W. Linville pos = skb->data + hdr_len; 307*274bfb8dSJohn W. Linville keyidx = pos[3]; 308*274bfb8dSJohn W. Linville if (!(keyidx & (1 << 5))) { 309*274bfb8dSJohn W. Linville if (net_ratelimit()) { 310*274bfb8dSJohn W. Linville printk(KERN_DEBUG "CCMP: received packet without ExtIV" 311*274bfb8dSJohn W. Linville " flag from %pM\n", hdr->addr2); 312*274bfb8dSJohn W. Linville } 313*274bfb8dSJohn W. Linville key->dot11RSNAStatsCCMPFormatErrors++; 314*274bfb8dSJohn W. Linville return -2; 315*274bfb8dSJohn W. Linville } 316*274bfb8dSJohn W. Linville keyidx >>= 6; 317*274bfb8dSJohn W. Linville if (key->key_idx != keyidx) { 318*274bfb8dSJohn W. Linville printk(KERN_DEBUG "CCMP: RX tkey->key_idx=%d frame " 319*274bfb8dSJohn W. Linville "keyidx=%d priv=%p\n", key->key_idx, keyidx, priv); 320*274bfb8dSJohn W. Linville return -6; 321*274bfb8dSJohn W. Linville } 322*274bfb8dSJohn W. Linville if (!key->key_set) { 323*274bfb8dSJohn W. Linville if (net_ratelimit()) { 324*274bfb8dSJohn W. Linville printk(KERN_DEBUG "CCMP: received packet from %pM" 325*274bfb8dSJohn W. Linville " with keyid=%d that does not have a configured" 326*274bfb8dSJohn W. Linville " key\n", hdr->addr2, keyidx); 327*274bfb8dSJohn W. Linville } 328*274bfb8dSJohn W. Linville return -3; 329*274bfb8dSJohn W. Linville } 330*274bfb8dSJohn W. Linville 331*274bfb8dSJohn W. Linville pn[0] = pos[7]; 332*274bfb8dSJohn W. Linville pn[1] = pos[6]; 333*274bfb8dSJohn W. Linville pn[2] = pos[5]; 334*274bfb8dSJohn W. Linville pn[3] = pos[4]; 335*274bfb8dSJohn W. Linville pn[4] = pos[1]; 336*274bfb8dSJohn W. Linville pn[5] = pos[0]; 337*274bfb8dSJohn W. Linville pos += 8; 338*274bfb8dSJohn W. Linville 339*274bfb8dSJohn W. Linville if (ccmp_replay_check(pn, key->rx_pn)) { 340*274bfb8dSJohn W. Linville if (net_ratelimit()) { 341*274bfb8dSJohn W. Linville printk(KERN_DEBUG "CCMP: replay detected: STA=%pM " 342*274bfb8dSJohn W. Linville "previous PN %02x%02x%02x%02x%02x%02x " 343*274bfb8dSJohn W. Linville "received PN %02x%02x%02x%02x%02x%02x\n", 344*274bfb8dSJohn W. Linville hdr->addr2, 345*274bfb8dSJohn W. Linville key->rx_pn[0], key->rx_pn[1], key->rx_pn[2], 346*274bfb8dSJohn W. Linville key->rx_pn[3], key->rx_pn[4], key->rx_pn[5], 347*274bfb8dSJohn W. Linville pn[0], pn[1], pn[2], pn[3], pn[4], pn[5]); 348*274bfb8dSJohn W. Linville } 349*274bfb8dSJohn W. Linville key->dot11RSNAStatsCCMPReplays++; 350*274bfb8dSJohn W. Linville return -4; 351*274bfb8dSJohn W. Linville } 352*274bfb8dSJohn W. Linville 353*274bfb8dSJohn W. Linville ccmp_init_blocks(key->tfm, hdr, pn, data_len, b0, a, b); 354*274bfb8dSJohn W. Linville xor_block(mic, b, CCMP_MIC_LEN); 355*274bfb8dSJohn W. Linville 356*274bfb8dSJohn W. Linville blocks = DIV_ROUND_UP(data_len, AES_BLOCK_LEN); 357*274bfb8dSJohn W. Linville last = data_len % AES_BLOCK_LEN; 358*274bfb8dSJohn W. Linville 359*274bfb8dSJohn W. Linville for (i = 1; i <= blocks; i++) { 360*274bfb8dSJohn W. Linville len = (i == blocks && last) ? last : AES_BLOCK_LEN; 361*274bfb8dSJohn W. Linville /* Decrypt, with counter */ 362*274bfb8dSJohn W. Linville b0[14] = (i >> 8) & 0xff; 363*274bfb8dSJohn W. Linville b0[15] = i & 0xff; 364*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(key->tfm, b0, b); 365*274bfb8dSJohn W. Linville xor_block(pos, b, len); 366*274bfb8dSJohn W. Linville /* Authentication */ 367*274bfb8dSJohn W. Linville xor_block(a, pos, len); 368*274bfb8dSJohn W. Linville lib80211_ccmp_aes_encrypt(key->tfm, a, a); 369*274bfb8dSJohn W. Linville pos += len; 370*274bfb8dSJohn W. Linville } 371*274bfb8dSJohn W. Linville 372*274bfb8dSJohn W. Linville if (memcmp(mic, a, CCMP_MIC_LEN) != 0) { 373*274bfb8dSJohn W. Linville if (net_ratelimit()) { 374*274bfb8dSJohn W. Linville printk(KERN_DEBUG "CCMP: decrypt failed: STA=" 375*274bfb8dSJohn W. Linville "%pM\n", hdr->addr2); 376*274bfb8dSJohn W. Linville } 377*274bfb8dSJohn W. Linville key->dot11RSNAStatsCCMPDecryptErrors++; 378*274bfb8dSJohn W. Linville return -5; 379*274bfb8dSJohn W. Linville } 380*274bfb8dSJohn W. Linville 381*274bfb8dSJohn W. Linville memcpy(key->rx_pn, pn, CCMP_PN_LEN); 382*274bfb8dSJohn W. Linville 383*274bfb8dSJohn W. Linville /* Remove hdr and MIC */ 384*274bfb8dSJohn W. Linville memmove(skb->data + CCMP_HDR_LEN, skb->data, hdr_len); 385*274bfb8dSJohn W. Linville skb_pull(skb, CCMP_HDR_LEN); 386*274bfb8dSJohn W. Linville skb_trim(skb, skb->len - CCMP_MIC_LEN); 387*274bfb8dSJohn W. Linville 388*274bfb8dSJohn W. Linville return keyidx; 389*274bfb8dSJohn W. Linville } 390*274bfb8dSJohn W. Linville 391*274bfb8dSJohn W. Linville static int lib80211_ccmp_set_key(void *key, int len, u8 * seq, void *priv) 392*274bfb8dSJohn W. Linville { 393*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *data = priv; 394*274bfb8dSJohn W. Linville int keyidx; 395*274bfb8dSJohn W. Linville struct crypto_cipher *tfm = data->tfm; 396*274bfb8dSJohn W. Linville 397*274bfb8dSJohn W. Linville keyidx = data->key_idx; 398*274bfb8dSJohn W. Linville memset(data, 0, sizeof(*data)); 399*274bfb8dSJohn W. Linville data->key_idx = keyidx; 400*274bfb8dSJohn W. Linville data->tfm = tfm; 401*274bfb8dSJohn W. Linville if (len == CCMP_TK_LEN) { 402*274bfb8dSJohn W. Linville memcpy(data->key, key, CCMP_TK_LEN); 403*274bfb8dSJohn W. Linville data->key_set = 1; 404*274bfb8dSJohn W. Linville if (seq) { 405*274bfb8dSJohn W. Linville data->rx_pn[0] = seq[5]; 406*274bfb8dSJohn W. Linville data->rx_pn[1] = seq[4]; 407*274bfb8dSJohn W. Linville data->rx_pn[2] = seq[3]; 408*274bfb8dSJohn W. Linville data->rx_pn[3] = seq[2]; 409*274bfb8dSJohn W. Linville data->rx_pn[4] = seq[1]; 410*274bfb8dSJohn W. Linville data->rx_pn[5] = seq[0]; 411*274bfb8dSJohn W. Linville } 412*274bfb8dSJohn W. Linville crypto_cipher_setkey(data->tfm, data->key, CCMP_TK_LEN); 413*274bfb8dSJohn W. Linville } else if (len == 0) 414*274bfb8dSJohn W. Linville data->key_set = 0; 415*274bfb8dSJohn W. Linville else 416*274bfb8dSJohn W. Linville return -1; 417*274bfb8dSJohn W. Linville 418*274bfb8dSJohn W. Linville return 0; 419*274bfb8dSJohn W. Linville } 420*274bfb8dSJohn W. Linville 421*274bfb8dSJohn W. Linville static int lib80211_ccmp_get_key(void *key, int len, u8 * seq, void *priv) 422*274bfb8dSJohn W. Linville { 423*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *data = priv; 424*274bfb8dSJohn W. Linville 425*274bfb8dSJohn W. Linville if (len < CCMP_TK_LEN) 426*274bfb8dSJohn W. Linville return -1; 427*274bfb8dSJohn W. Linville 428*274bfb8dSJohn W. Linville if (!data->key_set) 429*274bfb8dSJohn W. Linville return 0; 430*274bfb8dSJohn W. Linville memcpy(key, data->key, CCMP_TK_LEN); 431*274bfb8dSJohn W. Linville 432*274bfb8dSJohn W. Linville if (seq) { 433*274bfb8dSJohn W. Linville seq[0] = data->tx_pn[5]; 434*274bfb8dSJohn W. Linville seq[1] = data->tx_pn[4]; 435*274bfb8dSJohn W. Linville seq[2] = data->tx_pn[3]; 436*274bfb8dSJohn W. Linville seq[3] = data->tx_pn[2]; 437*274bfb8dSJohn W. Linville seq[4] = data->tx_pn[1]; 438*274bfb8dSJohn W. Linville seq[5] = data->tx_pn[0]; 439*274bfb8dSJohn W. Linville } 440*274bfb8dSJohn W. Linville 441*274bfb8dSJohn W. Linville return CCMP_TK_LEN; 442*274bfb8dSJohn W. Linville } 443*274bfb8dSJohn W. Linville 444*274bfb8dSJohn W. Linville static char *lib80211_ccmp_print_stats(char *p, void *priv) 445*274bfb8dSJohn W. Linville { 446*274bfb8dSJohn W. Linville struct lib80211_ccmp_data *ccmp = priv; 447*274bfb8dSJohn W. Linville 448*274bfb8dSJohn W. Linville p += sprintf(p, "key[%d] alg=CCMP key_set=%d " 449*274bfb8dSJohn W. Linville "tx_pn=%02x%02x%02x%02x%02x%02x " 450*274bfb8dSJohn W. Linville "rx_pn=%02x%02x%02x%02x%02x%02x " 451*274bfb8dSJohn W. Linville "format_errors=%d replays=%d decrypt_errors=%d\n", 452*274bfb8dSJohn W. Linville ccmp->key_idx, ccmp->key_set, 453*274bfb8dSJohn W. Linville ccmp->tx_pn[0], ccmp->tx_pn[1], ccmp->tx_pn[2], 454*274bfb8dSJohn W. Linville ccmp->tx_pn[3], ccmp->tx_pn[4], ccmp->tx_pn[5], 455*274bfb8dSJohn W. Linville ccmp->rx_pn[0], ccmp->rx_pn[1], ccmp->rx_pn[2], 456*274bfb8dSJohn W. Linville ccmp->rx_pn[3], ccmp->rx_pn[4], ccmp->rx_pn[5], 457*274bfb8dSJohn W. Linville ccmp->dot11RSNAStatsCCMPFormatErrors, 458*274bfb8dSJohn W. Linville ccmp->dot11RSNAStatsCCMPReplays, 459*274bfb8dSJohn W. Linville ccmp->dot11RSNAStatsCCMPDecryptErrors); 460*274bfb8dSJohn W. Linville 461*274bfb8dSJohn W. Linville return p; 462*274bfb8dSJohn W. Linville } 463*274bfb8dSJohn W. Linville 464*274bfb8dSJohn W. Linville static struct lib80211_crypto_ops lib80211_crypt_ccmp = { 465*274bfb8dSJohn W. Linville .name = "CCMP", 466*274bfb8dSJohn W. Linville .init = lib80211_ccmp_init, 467*274bfb8dSJohn W. Linville .deinit = lib80211_ccmp_deinit, 468*274bfb8dSJohn W. Linville .build_iv = lib80211_ccmp_hdr, 469*274bfb8dSJohn W. Linville .encrypt_mpdu = lib80211_ccmp_encrypt, 470*274bfb8dSJohn W. Linville .decrypt_mpdu = lib80211_ccmp_decrypt, 471*274bfb8dSJohn W. Linville .encrypt_msdu = NULL, 472*274bfb8dSJohn W. Linville .decrypt_msdu = NULL, 473*274bfb8dSJohn W. Linville .set_key = lib80211_ccmp_set_key, 474*274bfb8dSJohn W. Linville .get_key = lib80211_ccmp_get_key, 475*274bfb8dSJohn W. Linville .print_stats = lib80211_ccmp_print_stats, 476*274bfb8dSJohn W. Linville .extra_mpdu_prefix_len = CCMP_HDR_LEN, 477*274bfb8dSJohn W. Linville .extra_mpdu_postfix_len = CCMP_MIC_LEN, 478*274bfb8dSJohn W. Linville .owner = THIS_MODULE, 479*274bfb8dSJohn W. Linville }; 480*274bfb8dSJohn W. Linville 481*274bfb8dSJohn W. Linville static int __init lib80211_crypto_ccmp_init(void) 482*274bfb8dSJohn W. Linville { 483*274bfb8dSJohn W. Linville return lib80211_register_crypto_ops(&lib80211_crypt_ccmp); 484*274bfb8dSJohn W. Linville } 485*274bfb8dSJohn W. Linville 486*274bfb8dSJohn W. Linville static void __exit lib80211_crypto_ccmp_exit(void) 487*274bfb8dSJohn W. Linville { 488*274bfb8dSJohn W. Linville lib80211_unregister_crypto_ops(&lib80211_crypt_ccmp); 489*274bfb8dSJohn W. Linville } 490*274bfb8dSJohn W. Linville 491*274bfb8dSJohn W. Linville module_init(lib80211_crypto_ccmp_init); 492*274bfb8dSJohn W. Linville module_exit(lib80211_crypto_ccmp_exit); 493