2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/*
* Cryptographic API .
*
* Cipher operations .
*
* Copyright ( c ) 2002 James Morris < jmorris @ intercode . com . au >
2005-07-06 13:51:31 -07:00
* Copyright ( c ) 2005 Herbert Xu < herbert @ gondor . apana . org . au >
2005-04-16 15:20:36 -07:00
*/
2007-01-27 10:05:15 +11:00
2018-04-09 15:54:47 +02:00
# include <crypto/algapi.h>
2005-04-16 15:20:36 -07:00
# include <linux/kernel.h>
# include <linux/crypto.h>
# include <linux/errno.h>
2007-08-23 16:23:01 +08:00
# include <linux/slab.h>
2005-04-16 15:20:36 -07:00
# include <linux/string.h>
# include "internal.h"
2007-08-23 16:23:01 +08:00
static int setkey_unaligned ( struct crypto_tfm * tfm , const u8 * key ,
unsigned int keylen )
2007-05-19 19:51:21 +10:00
{
struct cipher_alg * cia = & tfm - > __crt_alg - > cra_cipher ;
unsigned long alignmask = crypto_tfm_alg_alignmask ( tfm ) ;
int ret ;
u8 * buffer , * alignbuffer ;
unsigned long absize ;
absize = keylen + alignmask ;
buffer = kmalloc ( absize , GFP_ATOMIC ) ;
if ( ! buffer )
return - ENOMEM ;
alignbuffer = ( u8 * ) ALIGN ( ( unsigned long ) buffer , alignmask + 1 ) ;
memcpy ( alignbuffer , key , keylen ) ;
ret = cia - > cia_setkey ( tfm , alignbuffer , keylen ) ;
2007-08-03 20:33:47 +08:00
memset ( alignbuffer , 0 , keylen ) ;
2007-05-19 19:51:21 +10:00
kfree ( buffer ) ;
return ret ;
}
2005-04-16 15:20:36 -07:00
static int setkey ( struct crypto_tfm * tfm , const u8 * key , unsigned int keylen )
{
struct cipher_alg * cia = & tfm - > __crt_alg - > cra_cipher ;
2007-05-19 19:51:21 +10:00
unsigned long alignmask = crypto_tfm_alg_alignmask ( tfm ) ;
2006-08-13 14:16:39 +10:00
tfm - > crt_flags & = ~ CRYPTO_TFM_RES_MASK ;
2005-04-16 15:20:36 -07:00
if ( keylen < cia - > cia_min_keysize | | keylen > cia - > cia_max_keysize ) {
tfm - > crt_flags | = CRYPTO_TFM_RES_BAD_KEY_LEN ;
return - EINVAL ;
2007-05-19 19:51:21 +10:00
}
if ( ( unsigned long ) key & alignmask )
return setkey_unaligned ( tfm , key , keylen ) ;
return cia - > cia_setkey ( tfm , key , keylen ) ;
2005-04-16 15:20:36 -07:00
}
2006-08-13 20:58:18 +10:00
static void cipher_crypt_unaligned ( void ( * fn ) ( struct crypto_tfm * , u8 * ,
const u8 * ) ,
struct crypto_tfm * tfm ,
u8 * dst , const u8 * src )
{
unsigned long alignmask = crypto_tfm_alg_alignmask ( tfm ) ;
unsigned int size = crypto_tfm_alg_blocksize ( tfm ) ;
2018-04-09 15:54:47 +02:00
u8 buffer [ MAX_CIPHER_BLOCKSIZE + MAX_CIPHER_ALIGNMASK ] ;
2006-08-13 20:58:18 +10:00
u8 * tmp = ( u8 * ) ALIGN ( ( unsigned long ) buffer , alignmask + 1 ) ;
memcpy ( tmp , src , size ) ;
fn ( tfm , tmp , tmp ) ;
memcpy ( dst , tmp , size ) ;
}
static void cipher_encrypt_unaligned ( struct crypto_tfm * tfm ,
u8 * dst , const u8 * src )
{
unsigned long alignmask = crypto_tfm_alg_alignmask ( tfm ) ;
struct cipher_alg * cipher = & tfm - > __crt_alg - > cra_cipher ;
if ( unlikely ( ( ( unsigned long ) dst | ( unsigned long ) src ) & alignmask ) ) {
cipher_crypt_unaligned ( cipher - > cia_encrypt , tfm , dst , src ) ;
return ;
}
cipher - > cia_encrypt ( tfm , dst , src ) ;
}
static void cipher_decrypt_unaligned ( struct crypto_tfm * tfm ,
u8 * dst , const u8 * src )
{
unsigned long alignmask = crypto_tfm_alg_alignmask ( tfm ) ;
struct cipher_alg * cipher = & tfm - > __crt_alg - > cra_cipher ;
if ( unlikely ( ( ( unsigned long ) dst | ( unsigned long ) src ) & alignmask ) ) {
cipher_crypt_unaligned ( cipher - > cia_decrypt , tfm , dst , src ) ;
return ;
}
cipher - > cia_decrypt ( tfm , dst , src ) ;
}
2005-04-16 15:20:36 -07:00
int crypto_init_cipher_ops ( struct crypto_tfm * tfm )
{
struct cipher_tfm * ops = & tfm - > crt_cipher ;
2006-08-13 20:58:18 +10:00
struct cipher_alg * cipher = & tfm - > __crt_alg - > cra_cipher ;
2005-04-16 15:20:36 -07:00
ops - > cit_setkey = setkey ;
2006-08-13 20:58:18 +10:00
ops - > cit_encrypt_one = crypto_tfm_alg_alignmask ( tfm ) ?
cipher_encrypt_unaligned : cipher - > cia_encrypt ;
ops - > cit_decrypt_one = crypto_tfm_alg_alignmask ( tfm ) ?
cipher_decrypt_unaligned : cipher - > cia_decrypt ;
2005-04-16 15:20:36 -07:00
2007-01-27 10:05:15 +11:00
return 0 ;
2005-04-16 15:20:36 -07:00
}