2005-04-16 15:20:36 -07:00
/*
* Cryptographic API .
*
2006-01-06 00:19:17 -08:00
* s390 implementation of the DES Cipher Algorithm .
2005-04-16 15:20:36 -07:00
*
2011-05-04 15:09:44 +10:00
* Copyright IBM Corp . 2003 , 2011
2007-02-05 21:18:14 +01:00
* Author ( s ) : Thomas Spatzier
* Jan Glauber ( jan . glauber @ de . ibm . com )
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 .
*
*/
2006-08-21 21:39:24 +10:00
2005-04-16 15:20:36 -07:00
# include <linux/init.h>
# include <linux/module.h>
2010-05-21 22:04:46 +10:00
# include <linux/crypto.h>
# include <crypto/algapi.h>
# include <crypto/des.h>
2006-01-14 13:20:53 -08:00
2006-01-06 00:19:17 -08:00
# include "crypt_s390.h"
2005-04-16 15:20:36 -07:00
2011-04-19 21:29:15 +02:00
# define DES3_KEY_SIZE (3 * DES_KEY_SIZE)
2005-04-16 15:20:36 -07:00
2011-05-04 15:09:44 +10:00
static u8 * ctrblk ;
2011-04-19 21:29:15 +02:00
struct s390_des_ctx {
2005-04-16 15:20:36 -07:00
u8 iv [ DES_BLOCK_SIZE ] ;
2011-04-19 21:29:15 +02:00
u8 key [ DES3_KEY_SIZE ] ;
2005-04-16 15:20:36 -07:00
} ;
2006-05-16 22:09:29 +10:00
static int des_setkey ( struct crypto_tfm * tfm , const u8 * key ,
2011-04-19 21:29:15 +02:00
unsigned int key_len )
2005-04-16 15:20:36 -07:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2006-08-13 14:16:39 +10:00
u32 * flags = & tfm - > crt_flags ;
2010-05-21 22:04:46 +10:00
u32 tmp [ DES_EXPKEY_WORDS ] ;
2005-04-16 15:20:36 -07:00
2010-05-21 22:04:46 +10:00
/* check for weak keys */
if ( ! des_ekey ( tmp , key ) & & ( * flags & CRYPTO_TFM_REQ_WEAK_KEY ) ) {
* flags | = CRYPTO_TFM_RES_WEAK_KEY ;
return - EINVAL ;
}
2011-04-19 21:29:15 +02:00
memcpy ( ctx - > key , key , key_len ) ;
2010-05-21 22:04:46 +10:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2006-05-16 22:09:29 +10:00
static void des_encrypt ( struct crypto_tfm * tfm , u8 * out , const u8 * in )
2005-04-16 15:20:36 -07:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2005-04-16 15:20:36 -07:00
2011-04-19 21:29:15 +02:00
crypt_s390_km ( KM_DEA_ENCRYPT , ctx - > key , out , in , DES_BLOCK_SIZE ) ;
2005-04-16 15:20:36 -07:00
}
2006-05-16 22:09:29 +10:00
static void des_decrypt ( struct crypto_tfm * tfm , u8 * out , const u8 * in )
2005-04-16 15:20:36 -07:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2005-04-16 15:20:36 -07:00
2011-04-19 21:29:15 +02:00
crypt_s390_km ( KM_DEA_DECRYPT , ctx - > key , out , in , DES_BLOCK_SIZE ) ;
2006-01-14 13:20:53 -08:00
}
2005-04-16 15:20:36 -07:00
static struct crypto_alg des_alg = {
. cra_name = " des " ,
2006-08-21 21:18:50 +10:00
. cra_driver_name = " des-s390 " ,
. cra_priority = CRYPT_S390_PRIORITY ,
2005-04-16 15:20:36 -07:00
. cra_flags = CRYPTO_ALG_TYPE_CIPHER ,
. cra_blocksize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
2005-04-16 15:20:36 -07:00
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT ( des_alg . cra_list ) ,
2006-01-14 13:20:53 -08:00
. cra_u = {
. cipher = {
. cia_min_keysize = DES_KEY_SIZE ,
. cia_max_keysize = DES_KEY_SIZE ,
. cia_setkey = des_setkey ,
. cia_encrypt = des_encrypt ,
2006-01-14 13:20:53 -08:00
. cia_decrypt = des_decrypt ,
2006-01-14 13:20:53 -08:00
}
}
2005-04-16 15:20:36 -07:00
} ;
2006-08-21 21:39:24 +10:00
static int ecb_desall_crypt ( struct blkcipher_desc * desc , long func ,
2011-04-19 21:29:15 +02:00
u8 * key , struct blkcipher_walk * walk )
2006-08-21 21:39:24 +10:00
{
int ret = blkcipher_walk_virt ( desc , walk ) ;
unsigned int nbytes ;
while ( ( nbytes = walk - > nbytes ) ) {
/* only use complete blocks */
unsigned int n = nbytes & ~ ( DES_BLOCK_SIZE - 1 ) ;
u8 * out = walk - > dst . virt . addr ;
u8 * in = walk - > src . virt . addr ;
2011-04-19 21:29:15 +02:00
ret = crypt_s390_km ( func , key , out , in , n ) ;
2006-08-21 21:39:24 +10:00
BUG_ON ( ( ret < 0 ) | | ( ret ! = n ) ) ;
nbytes & = DES_BLOCK_SIZE - 1 ;
ret = blkcipher_walk_done ( desc , walk , nbytes ) ;
}
return ret ;
}
static int cbc_desall_crypt ( struct blkcipher_desc * desc , long func ,
2011-04-19 21:29:15 +02:00
u8 * iv , struct blkcipher_walk * walk )
2006-08-21 21:39:24 +10:00
{
int ret = blkcipher_walk_virt ( desc , walk ) ;
unsigned int nbytes = walk - > nbytes ;
if ( ! nbytes )
goto out ;
2011-04-19 21:29:15 +02:00
memcpy ( iv , walk - > iv , DES_BLOCK_SIZE ) ;
2006-08-21 21:39:24 +10:00
do {
/* only use complete blocks */
unsigned int n = nbytes & ~ ( DES_BLOCK_SIZE - 1 ) ;
u8 * out = walk - > dst . virt . addr ;
u8 * in = walk - > src . virt . addr ;
2011-04-19 21:29:15 +02:00
ret = crypt_s390_kmc ( func , iv , out , in , n ) ;
2006-08-21 21:39:24 +10:00
BUG_ON ( ( ret < 0 ) | | ( ret ! = n ) ) ;
nbytes & = DES_BLOCK_SIZE - 1 ;
ret = blkcipher_walk_done ( desc , walk , nbytes ) ;
} while ( ( nbytes = walk - > nbytes ) ) ;
2011-04-19 21:29:15 +02:00
memcpy ( walk - > iv , iv , DES_BLOCK_SIZE ) ;
2006-08-21 21:39:24 +10:00
out :
return ret ;
}
static int ecb_des_encrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return ecb_desall_crypt ( desc , KM_DEA_ENCRYPT , ctx - > key , & walk ) ;
2006-08-21 21:39:24 +10:00
}
static int ecb_des_decrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return ecb_desall_crypt ( desc , KM_DEA_DECRYPT , ctx - > key , & walk ) ;
2006-08-21 21:39:24 +10:00
}
static struct crypto_alg ecb_des_alg = {
. cra_name = " ecb(des) " ,
. cra_driver_name = " ecb-des-s390 " ,
. cra_priority = CRYPT_S390_COMPOSITE_PRIORITY ,
. cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER ,
. cra_blocksize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
2006-08-21 21:39:24 +10:00
. cra_type = & crypto_blkcipher_type ,
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT ( ecb_des_alg . cra_list ) ,
. cra_u = {
. blkcipher = {
. min_keysize = DES_KEY_SIZE ,
. max_keysize = DES_KEY_SIZE ,
. setkey = des_setkey ,
. encrypt = ecb_des_encrypt ,
. decrypt = ecb_des_decrypt ,
}
}
} ;
static int cbc_des_encrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return cbc_desall_crypt ( desc , KMC_DEA_ENCRYPT , ctx - > iv , & walk ) ;
2006-08-21 21:39:24 +10:00
}
static int cbc_des_decrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return cbc_desall_crypt ( desc , KMC_DEA_DECRYPT , ctx - > iv , & walk ) ;
2006-08-21 21:39:24 +10:00
}
static struct crypto_alg cbc_des_alg = {
. cra_name = " cbc(des) " ,
. cra_driver_name = " cbc-des-s390 " ,
. cra_priority = CRYPT_S390_COMPOSITE_PRIORITY ,
. cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER ,
. cra_blocksize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
2006-08-21 21:39:24 +10:00
. cra_type = & crypto_blkcipher_type ,
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT ( cbc_des_alg . cra_list ) ,
. cra_u = {
. blkcipher = {
. min_keysize = DES_KEY_SIZE ,
. max_keysize = DES_KEY_SIZE ,
. ivsize = DES_BLOCK_SIZE ,
. setkey = des_setkey ,
. encrypt = cbc_des_encrypt ,
. decrypt = cbc_des_decrypt ,
}
}
} ;
2005-04-16 15:20:36 -07:00
/*
* RFC2451 :
*
* For DES - EDE3 , there is no known need to reject weak or
* complementation keys . Any weakness is obviated by the use of
* multiple keys .
*
* However , if the first two or last two independent 64 - bit keys are
* equal ( k1 = = k2 or k2 = = k3 ) , then the DES3 operation is simply the
* same as DES . Implementers MUST reject keys that exhibit this
* property .
*
*/
2011-04-19 21:29:15 +02:00
static int des3_setkey ( struct crypto_tfm * tfm , const u8 * key ,
unsigned int key_len )
2005-04-16 15:20:36 -07:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2006-08-13 14:16:39 +10:00
u32 * flags = & tfm - > crt_flags ;
2005-04-16 15:20:36 -07:00
if ( ! ( memcmp ( key , & key [ DES_KEY_SIZE ] , DES_KEY_SIZE ) & &
memcmp ( & key [ DES_KEY_SIZE ] , & key [ DES_KEY_SIZE * 2 ] ,
2009-06-18 19:52:59 +08:00
DES_KEY_SIZE ) ) & &
( * flags & CRYPTO_TFM_REQ_WEAK_KEY ) ) {
* flags | = CRYPTO_TFM_RES_WEAK_KEY ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
}
2011-04-19 21:29:15 +02:00
memcpy ( ctx - > key , key , key_len ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2011-04-19 21:29:15 +02:00
static void des3_encrypt ( struct crypto_tfm * tfm , u8 * dst , const u8 * src )
2005-04-16 15:20:36 -07:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2005-04-16 15:20:36 -07:00
2011-04-19 21:29:15 +02:00
crypt_s390_km ( KM_TDEA_192_ENCRYPT , ctx - > key , dst , src , DES_BLOCK_SIZE ) ;
2005-04-16 15:20:36 -07:00
}
2011-04-19 21:29:15 +02:00
static void des3_decrypt ( struct crypto_tfm * tfm , u8 * dst , const u8 * src )
2005-04-16 15:20:36 -07:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2005-04-16 15:20:36 -07:00
2011-04-19 21:29:15 +02:00
crypt_s390_km ( KM_TDEA_192_DECRYPT , ctx - > key , dst , src , DES_BLOCK_SIZE ) ;
2005-04-16 15:20:36 -07:00
}
2011-04-19 21:29:15 +02:00
static struct crypto_alg des3_alg = {
2005-04-16 15:20:36 -07:00
. cra_name = " des3_ede " ,
2006-08-21 21:18:50 +10:00
. cra_driver_name = " des3_ede-s390 " ,
. cra_priority = CRYPT_S390_PRIORITY ,
2005-04-16 15:20:36 -07:00
. cra_flags = CRYPTO_ALG_TYPE_CIPHER ,
2010-05-21 22:04:46 +10:00
. cra_blocksize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
2005-04-16 15:20:36 -07:00
. cra_module = THIS_MODULE ,
2011-04-19 21:29:15 +02:00
. cra_list = LIST_HEAD_INIT ( des3_alg . cra_list ) ,
2006-01-14 13:20:53 -08:00
. cra_u = {
. cipher = {
2011-04-19 21:29:15 +02:00
. cia_min_keysize = DES3_KEY_SIZE ,
. cia_max_keysize = DES3_KEY_SIZE ,
. cia_setkey = des3_setkey ,
. cia_encrypt = des3_encrypt ,
. cia_decrypt = des3_decrypt ,
2006-01-14 13:20:53 -08:00
}
}
2005-04-16 15:20:36 -07:00
} ;
2011-04-19 21:29:15 +02:00
static int ecb_des3_encrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
2006-08-21 21:39:24 +10:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return ecb_desall_crypt ( desc , KM_TDEA_192_ENCRYPT , ctx - > key , & walk ) ;
2006-08-21 21:39:24 +10:00
}
2011-04-19 21:29:15 +02:00
static int ecb_des3_decrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
2006-08-21 21:39:24 +10:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return ecb_desall_crypt ( desc , KM_TDEA_192_DECRYPT , ctx - > key , & walk ) ;
2006-08-21 21:39:24 +10:00
}
2011-04-19 21:29:15 +02:00
static struct crypto_alg ecb_des3_alg = {
2006-08-21 21:39:24 +10:00
. cra_name = " ecb(des3_ede) " ,
. cra_driver_name = " ecb-des3_ede-s390 " ,
. cra_priority = CRYPT_S390_COMPOSITE_PRIORITY ,
. cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER ,
2010-05-21 22:04:46 +10:00
. cra_blocksize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
2006-08-21 21:39:24 +10:00
. cra_type = & crypto_blkcipher_type ,
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT (
2011-04-19 21:29:15 +02:00
ecb_des3_alg . cra_list ) ,
2006-08-21 21:39:24 +10:00
. cra_u = {
. blkcipher = {
2011-04-19 21:29:15 +02:00
. min_keysize = DES3_KEY_SIZE ,
. max_keysize = DES3_KEY_SIZE ,
. setkey = des3_setkey ,
. encrypt = ecb_des3_encrypt ,
. decrypt = ecb_des3_decrypt ,
2006-08-21 21:39:24 +10:00
}
}
} ;
2011-04-19 21:29:15 +02:00
static int cbc_des3_encrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
2006-08-21 21:39:24 +10:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return cbc_desall_crypt ( desc , KMC_TDEA_192_ENCRYPT , ctx - > iv , & walk ) ;
2006-08-21 21:39:24 +10:00
}
2011-04-19 21:29:15 +02:00
static int cbc_des3_decrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
2006-08-21 21:39:24 +10:00
{
2011-04-19 21:29:15 +02:00
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
2006-08-21 21:39:24 +10:00
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
2011-04-19 21:29:15 +02:00
return cbc_desall_crypt ( desc , KMC_TDEA_192_DECRYPT , ctx - > iv , & walk ) ;
2006-08-21 21:39:24 +10:00
}
2011-04-19 21:29:15 +02:00
static struct crypto_alg cbc_des3_alg = {
2006-08-21 21:39:24 +10:00
. cra_name = " cbc(des3_ede) " ,
. cra_driver_name = " cbc-des3_ede-s390 " ,
. cra_priority = CRYPT_S390_COMPOSITE_PRIORITY ,
. cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER ,
2010-05-21 22:04:46 +10:00
. cra_blocksize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
2006-08-21 21:39:24 +10:00
. cra_type = & crypto_blkcipher_type ,
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT (
2011-04-19 21:29:15 +02:00
cbc_des3_alg . cra_list ) ,
2006-08-21 21:39:24 +10:00
. cra_u = {
. blkcipher = {
2011-04-19 21:29:15 +02:00
. min_keysize = DES3_KEY_SIZE ,
. max_keysize = DES3_KEY_SIZE ,
2010-05-21 22:04:46 +10:00
. ivsize = DES_BLOCK_SIZE ,
2011-04-19 21:29:15 +02:00
. setkey = des3_setkey ,
. encrypt = cbc_des3_encrypt ,
. decrypt = cbc_des3_decrypt ,
2006-08-21 21:39:24 +10:00
}
}
} ;
2011-05-04 15:09:44 +10:00
static int ctr_desall_crypt ( struct blkcipher_desc * desc , long func ,
struct s390_des_ctx * ctx , struct blkcipher_walk * walk )
{
int ret = blkcipher_walk_virt_block ( desc , walk , DES_BLOCK_SIZE ) ;
unsigned int i , n , nbytes ;
u8 buf [ DES_BLOCK_SIZE ] ;
u8 * out , * in ;
memcpy ( ctrblk , walk - > iv , DES_BLOCK_SIZE ) ;
while ( ( nbytes = walk - > nbytes ) > = DES_BLOCK_SIZE ) {
out = walk - > dst . virt . addr ;
in = walk - > src . virt . addr ;
while ( nbytes > = DES_BLOCK_SIZE ) {
/* align to block size, max. PAGE_SIZE */
n = ( nbytes > PAGE_SIZE ) ? PAGE_SIZE :
nbytes & ~ ( DES_BLOCK_SIZE - 1 ) ;
for ( i = DES_BLOCK_SIZE ; i < n ; i + = DES_BLOCK_SIZE ) {
memcpy ( ctrblk + i , ctrblk + i - DES_BLOCK_SIZE ,
DES_BLOCK_SIZE ) ;
crypto_inc ( ctrblk + i , DES_BLOCK_SIZE ) ;
}
ret = crypt_s390_kmctr ( func , ctx - > key , out , in , n , ctrblk ) ;
BUG_ON ( ( ret < 0 ) | | ( ret ! = n ) ) ;
if ( n > DES_BLOCK_SIZE )
memcpy ( ctrblk , ctrblk + n - DES_BLOCK_SIZE ,
DES_BLOCK_SIZE ) ;
crypto_inc ( ctrblk , DES_BLOCK_SIZE ) ;
out + = n ;
in + = n ;
nbytes - = n ;
}
ret = blkcipher_walk_done ( desc , walk , nbytes ) ;
}
/* final block may be < DES_BLOCK_SIZE, copy only nbytes */
if ( nbytes ) {
out = walk - > dst . virt . addr ;
in = walk - > src . virt . addr ;
ret = crypt_s390_kmctr ( func , ctx - > key , buf , in ,
DES_BLOCK_SIZE , ctrblk ) ;
BUG_ON ( ret < 0 | | ret ! = DES_BLOCK_SIZE ) ;
memcpy ( out , buf , nbytes ) ;
crypto_inc ( ctrblk , DES_BLOCK_SIZE ) ;
ret = blkcipher_walk_done ( desc , walk , 0 ) ;
}
memcpy ( walk - > iv , ctrblk , DES_BLOCK_SIZE ) ;
return ret ;
}
static int ctr_des_encrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
return ctr_desall_crypt ( desc , KMCTR_DEA_ENCRYPT , ctx , & walk ) ;
}
static int ctr_des_decrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
return ctr_desall_crypt ( desc , KMCTR_DEA_DECRYPT , ctx , & walk ) ;
}
static struct crypto_alg ctr_des_alg = {
. cra_name = " ctr(des) " ,
. cra_driver_name = " ctr-des-s390 " ,
. cra_priority = CRYPT_S390_COMPOSITE_PRIORITY ,
. cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER ,
. cra_blocksize = 1 ,
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
. cra_type = & crypto_blkcipher_type ,
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT ( ctr_des_alg . cra_list ) ,
. cra_u = {
. blkcipher = {
. min_keysize = DES_KEY_SIZE ,
. max_keysize = DES_KEY_SIZE ,
. ivsize = DES_BLOCK_SIZE ,
. setkey = des_setkey ,
. encrypt = ctr_des_encrypt ,
. decrypt = ctr_des_decrypt ,
}
}
} ;
static int ctr_des3_encrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
return ctr_desall_crypt ( desc , KMCTR_TDEA_192_ENCRYPT , ctx , & walk ) ;
}
static int ctr_des3_decrypt ( struct blkcipher_desc * desc ,
struct scatterlist * dst , struct scatterlist * src ,
unsigned int nbytes )
{
struct s390_des_ctx * ctx = crypto_blkcipher_ctx ( desc - > tfm ) ;
struct blkcipher_walk walk ;
blkcipher_walk_init ( & walk , dst , src , nbytes ) ;
return ctr_desall_crypt ( desc , KMCTR_TDEA_192_DECRYPT , ctx , & walk ) ;
}
static struct crypto_alg ctr_des3_alg = {
. cra_name = " ctr(des3_ede) " ,
. cra_driver_name = " ctr-des3_ede-s390 " ,
. cra_priority = CRYPT_S390_COMPOSITE_PRIORITY ,
. cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER ,
. cra_blocksize = 1 ,
. cra_ctxsize = sizeof ( struct s390_des_ctx ) ,
. cra_type = & crypto_blkcipher_type ,
. cra_module = THIS_MODULE ,
. cra_list = LIST_HEAD_INIT ( ctr_des3_alg . cra_list ) ,
. cra_u = {
. blkcipher = {
. min_keysize = DES3_KEY_SIZE ,
. max_keysize = DES3_KEY_SIZE ,
. ivsize = DES_BLOCK_SIZE ,
. setkey = des3_setkey ,
. encrypt = ctr_des3_encrypt ,
. decrypt = ctr_des3_decrypt ,
}
}
} ;
2011-04-19 21:29:15 +02:00
static int __init des_s390_init ( void )
2005-04-16 15:20:36 -07:00
{
2010-05-21 22:04:08 +10:00
int ret ;
2005-04-16 15:20:36 -07:00
2011-04-19 21:29:14 +02:00
if ( ! crypt_s390_func_available ( KM_DEA_ENCRYPT , CRYPT_S390_MSA ) | |
! crypt_s390_func_available ( KM_TDEA_192_ENCRYPT , CRYPT_S390_MSA ) )
2007-02-05 21:18:14 +01:00
return - EOPNOTSUPP ;
2005-04-16 15:20:36 -07:00
2006-08-21 21:39:24 +10:00
ret = crypto_register_alg ( & des_alg ) ;
if ( ret )
goto des_err ;
ret = crypto_register_alg ( & ecb_des_alg ) ;
if ( ret )
goto ecb_des_err ;
ret = crypto_register_alg ( & cbc_des_alg ) ;
if ( ret )
goto cbc_des_err ;
2011-04-19 21:29:15 +02:00
ret = crypto_register_alg ( & des3_alg ) ;
2006-08-21 21:39:24 +10:00
if ( ret )
2011-04-19 21:29:15 +02:00
goto des3_err ;
ret = crypto_register_alg ( & ecb_des3_alg ) ;
2006-08-21 21:39:24 +10:00
if ( ret )
2011-04-19 21:29:15 +02:00
goto ecb_des3_err ;
ret = crypto_register_alg ( & cbc_des3_alg ) ;
2006-08-21 21:39:24 +10:00
if ( ret )
2011-04-19 21:29:15 +02:00
goto cbc_des3_err ;
2011-05-04 15:09:44 +10:00
if ( crypt_s390_func_available ( KMCTR_DEA_ENCRYPT ,
CRYPT_S390_MSA | CRYPT_S390_MSA4 ) & &
crypt_s390_func_available ( KMCTR_TDEA_192_ENCRYPT ,
CRYPT_S390_MSA | CRYPT_S390_MSA4 ) ) {
ret = crypto_register_alg ( & ctr_des_alg ) ;
if ( ret )
goto ctr_des_err ;
ret = crypto_register_alg ( & ctr_des3_alg ) ;
if ( ret )
goto ctr_des3_err ;
ctrblk = ( u8 * ) __get_free_page ( GFP_KERNEL ) ;
if ( ! ctrblk ) {
ret = - ENOMEM ;
goto ctr_mem_err ;
}
}
2006-08-21 21:39:24 +10:00
out :
return ret ;
2011-05-04 15:09:44 +10:00
ctr_mem_err :
crypto_unregister_alg ( & ctr_des3_alg ) ;
ctr_des3_err :
crypto_unregister_alg ( & ctr_des_alg ) ;
ctr_des_err :
crypto_unregister_alg ( & cbc_des3_alg ) ;
2011-04-19 21:29:15 +02:00
cbc_des3_err :
crypto_unregister_alg ( & ecb_des3_alg ) ;
ecb_des3_err :
crypto_unregister_alg ( & des3_alg ) ;
des3_err :
2006-08-21 21:39:24 +10:00
crypto_unregister_alg ( & cbc_des_alg ) ;
cbc_des_err :
crypto_unregister_alg ( & ecb_des_alg ) ;
ecb_des_err :
crypto_unregister_alg ( & des_alg ) ;
des_err :
goto out ;
2005-04-16 15:20:36 -07:00
}
2010-05-21 22:04:46 +10:00
static void __exit des_s390_exit ( void )
2005-04-16 15:20:36 -07:00
{
2011-05-04 15:09:44 +10:00
if ( ctrblk ) {
crypto_unregister_alg ( & ctr_des_alg ) ;
crypto_unregister_alg ( & ctr_des3_alg ) ;
free_page ( ( unsigned long ) ctrblk ) ;
}
2011-04-19 21:29:15 +02:00
crypto_unregister_alg ( & cbc_des3_alg ) ;
crypto_unregister_alg ( & ecb_des3_alg ) ;
crypto_unregister_alg ( & des3_alg ) ;
2006-08-21 21:39:24 +10:00
crypto_unregister_alg ( & cbc_des_alg ) ;
crypto_unregister_alg ( & ecb_des_alg ) ;
2005-04-16 15:20:36 -07:00
crypto_unregister_alg ( & des_alg ) ;
}
2008-04-17 07:46:17 +02:00
module_init ( des_s390_init ) ;
2010-05-21 22:04:46 +10:00
module_exit ( des_s390_exit ) ;
2005-04-16 15:20:36 -07:00
MODULE_ALIAS ( " des " ) ;
MODULE_ALIAS ( " des3_ede " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " DES & Triple DES EDE Cipher Algorithms " ) ;