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
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the Free
* Software Foundation ; either version 2 of the License , or ( at your option )
* any later version .
*
*/
2007-01-27 10:05:15 +11:00
2005-04-16 15:20:36 -07:00
# include <linux/kernel.h>
# include <linux/crypto.h>
# include <linux/errno.h>
2007-01-27 10:05:15 +11:00
# include <linux/scatterlist.h>
2005-04-16 15:20:36 -07:00
# include <linux/string.h>
# include "internal.h"
static int setkey ( struct crypto_tfm * tfm , const u8 * key , unsigned int keylen )
{
struct cipher_alg * cia = & tfm - > __crt_alg - > cra_cipher ;
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 ;
} else
2006-08-13 14:16:39 +10:00
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 ) ;
u8 buffer [ size + alignmask ] ;
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
}
void crypto_exit_cipher_ops ( struct crypto_tfm * tfm )
{
}