2007-05-05 11:45:53 -07:00
/*
* Copyright 2003 - 2004 , Instant802 Networks , Inc .
* Copyright 2005 - 2006 , Devicescape Software , Inc .
*
2013-10-10 09:55:20 +02:00
* Rewrite : Copyright ( C ) 2013 Linaro Ltd < ard . biesheuvel @ linaro . org >
*
2007-05-05 11:45:53 -07:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
2007-08-28 15:50:33 -07:00
# include <linux/kernel.h>
2007-05-05 11:45:53 -07:00
# include <linux/types.h>
# include <linux/err.h>
2015-04-22 15:06:32 +08:00
# include <crypto/aead.h>
2007-05-05 11:45:53 -07:00
# include <net/mac80211.h>
2008-04-08 15:14:40 -04:00
# include "key.h"
2007-05-05 11:45:53 -07:00
# include "aes_ccm.h"
2016-10-17 15:05:33 +01:00
int ieee80211_aes_ccm_encrypt ( struct crypto_aead * tfm , u8 * b_0 , u8 * aad ,
u8 * data , size_t data_len , u8 * mic ,
size_t mic_len )
2007-05-05 11:45:53 -07:00
{
2015-05-27 16:03:50 +08:00
struct scatterlist sg [ 3 ] ;
2016-10-17 15:05:33 +01:00
struct aead_request * aead_req ;
int reqsize = sizeof ( * aead_req ) + crypto_aead_reqsize ( tfm ) ;
u8 * __aad ;
2007-05-05 11:45:53 -07:00
2016-10-17 15:05:33 +01:00
aead_req = kzalloc ( reqsize + CCM_AAD_LEN , GFP_ATOMIC ) ;
if ( ! aead_req )
return - ENOMEM ;
2014-03-20 23:39:32 -07:00
2016-10-17 15:05:33 +01:00
__aad = ( u8 * ) aead_req + reqsize ;
memcpy ( __aad , aad , CCM_AAD_LEN ) ;
2007-05-05 11:45:53 -07:00
2015-05-27 16:03:50 +08:00
sg_init_table ( sg , 3 ) ;
2016-10-17 15:05:33 +01:00
sg_set_buf ( & sg [ 0 ] , & __aad [ 2 ] , be16_to_cpup ( ( __be16 * ) __aad ) ) ;
2015-05-27 16:03:50 +08:00
sg_set_buf ( & sg [ 1 ] , data , data_len ) ;
sg_set_buf ( & sg [ 2 ] , mic , mic_len ) ;
2007-05-05 11:45:53 -07:00
2014-03-20 23:39:32 -07:00
aead_request_set_tfm ( aead_req , tfm ) ;
2015-05-27 16:03:50 +08:00
aead_request_set_crypt ( aead_req , sg , sg , data_len , b_0 ) ;
aead_request_set_ad ( aead_req , sg [ 0 ] . length ) ;
2007-05-05 11:45:53 -07:00
2014-03-20 23:39:32 -07:00
crypto_aead_encrypt ( aead_req ) ;
2016-10-17 15:05:33 +01:00
kzfree ( aead_req ) ;
return 0 ;
2007-05-05 11:45:53 -07:00
}
2013-10-10 09:55:20 +02:00
int ieee80211_aes_ccm_decrypt ( struct crypto_aead * tfm , u8 * b_0 , u8 * aad ,
2015-01-24 19:52:07 +02:00
u8 * data , size_t data_len , u8 * mic ,
size_t mic_len )
2007-05-05 11:45:53 -07:00
{
2015-05-27 16:03:50 +08:00
struct scatterlist sg [ 3 ] ;
2016-10-17 15:05:33 +01:00
struct aead_request * aead_req ;
int reqsize = sizeof ( * aead_req ) + crypto_aead_reqsize ( tfm ) ;
u8 * __aad ;
int err ;
2013-10-10 09:55:20 +02:00
2014-11-06 11:52:13 +01:00
if ( data_len = = 0 )
return - EINVAL ;
2016-10-17 15:05:33 +01:00
aead_req = kzalloc ( reqsize + CCM_AAD_LEN , GFP_ATOMIC ) ;
if ( ! aead_req )
return - ENOMEM ;
__aad = ( u8 * ) aead_req + reqsize ;
memcpy ( __aad , aad , CCM_AAD_LEN ) ;
2013-10-10 09:55:20 +02:00
2015-05-27 16:03:50 +08:00
sg_init_table ( sg , 3 ) ;
2016-10-17 15:05:33 +01:00
sg_set_buf ( & sg [ 0 ] , & __aad [ 2 ] , be16_to_cpup ( ( __be16 * ) __aad ) ) ;
2015-05-27 16:03:50 +08:00
sg_set_buf ( & sg [ 1 ] , data , data_len ) ;
sg_set_buf ( & sg [ 2 ] , mic , mic_len ) ;
2013-10-10 09:55:20 +02:00
2014-03-20 23:39:32 -07:00
aead_request_set_tfm ( aead_req , tfm ) ;
2015-05-27 16:03:50 +08:00
aead_request_set_crypt ( aead_req , sg , sg , data_len + mic_len , b_0 ) ;
aead_request_set_ad ( aead_req , sg [ 0 ] . length ) ;
2013-10-10 09:55:20 +02:00
2016-10-17 15:05:33 +01:00
err = crypto_aead_decrypt ( aead_req ) ;
kzfree ( aead_req ) ;
return err ;
2007-05-05 11:45:53 -07:00
}
2015-01-24 19:52:07 +02:00
struct crypto_aead * ieee80211_aes_key_setup_encrypt ( const u8 key [ ] ,
size_t key_len ,
size_t mic_len )
2007-05-05 11:45:53 -07:00
{
2013-10-10 09:55:20 +02:00
struct crypto_aead * tfm ;
int err ;
2007-05-05 11:45:53 -07:00
2013-10-10 09:55:20 +02:00
tfm = crypto_alloc_aead ( " ccm(aes) " , 0 , CRYPTO_ALG_ASYNC ) ;
if ( IS_ERR ( tfm ) )
return tfm ;
2007-05-05 11:45:53 -07:00
2015-01-24 19:52:07 +02:00
err = crypto_aead_setkey ( tfm , key , key_len ) ;
2015-03-23 17:08:14 +03:00
if ( err )
goto free_aead ;
err = crypto_aead_setauthsize ( tfm , mic_len ) ;
if ( err )
goto free_aead ;
return tfm ;
2007-05-05 11:45:53 -07:00
2015-03-23 17:08:14 +03:00
free_aead :
2013-10-10 09:55:20 +02:00
crypto_free_aead ( tfm ) ;
return ERR_PTR ( err ) ;
2007-05-05 11:45:53 -07:00
}
2013-10-10 09:55:20 +02:00
void ieee80211_aes_key_free ( struct crypto_aead * tfm )
2007-05-05 11:45:53 -07:00
{
2013-10-10 09:55:20 +02:00
crypto_free_aead ( tfm ) ;
2007-05-05 11:45:53 -07:00
}