2008-08-31 15:47:27 +10:00
/*
* Synchronous Cryptographic Hash operations .
*
* Copyright ( c ) 2008 Herbert Xu < herbert @ gondor . apana . org . au >
*
* 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 .
*
*/
2008-08-31 18:52:18 +10:00
# include <crypto/scatterwalk.h>
2008-08-31 15:47:27 +10:00
# include <crypto/internal/hash.h>
# include <linux/err.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/seq_file.h>
2008-08-31 18:52:18 +10:00
# include "internal.h"
2009-02-18 16:56:59 +08:00
static const struct crypto_type crypto_shash_type ;
2009-07-11 22:17:39 +08:00
static int shash_no_setkey ( struct crypto_shash * tfm , const u8 * key ,
unsigned int keylen )
{
return - ENOSYS ;
}
2008-08-31 15:47:27 +10:00
static int shash_setkey_unaligned ( struct crypto_shash * tfm , const u8 * key ,
unsigned int keylen )
{
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
unsigned long absize ;
u8 * buffer , * alignbuffer ;
int err ;
2010-05-19 11:50:58 +10:00
absize = keylen + ( alignmask & ~ ( crypto_tfm_ctx_alignment ( ) - 1 ) ) ;
2008-08-31 15:47:27 +10:00
buffer = kmalloc ( absize , GFP_KERNEL ) ;
if ( ! buffer )
return - ENOMEM ;
alignbuffer = ( u8 * ) ALIGN ( ( unsigned long ) buffer , alignmask + 1 ) ;
memcpy ( alignbuffer , key , keylen ) ;
err = shash - > setkey ( tfm , alignbuffer , keylen ) ;
2009-07-14 21:35:36 +08:00
kzfree ( buffer ) ;
2008-08-31 15:47:27 +10:00
return err ;
}
int crypto_shash_setkey ( struct crypto_shash * tfm , const u8 * key ,
unsigned int keylen )
{
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
if ( ( unsigned long ) key & alignmask )
return shash_setkey_unaligned ( tfm , key , keylen ) ;
return shash - > setkey ( tfm , key , keylen ) ;
}
EXPORT_SYMBOL_GPL ( crypto_shash_setkey ) ;
static inline unsigned int shash_align_buffer_size ( unsigned len ,
unsigned long mask )
{
return len + ( mask & ~ ( __alignof__ ( u8 __attribute__ ( ( aligned ) ) ) - 1 ) ) ;
}
static int shash_update_unaligned ( struct shash_desc * desc , const u8 * data ,
unsigned int len )
{
struct crypto_shash * tfm = desc - > tfm ;
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
unsigned int unaligned_len = alignmask + 1 -
( ( unsigned long ) data & alignmask ) ;
2009-07-14 21:43:56 +08:00
u8 ubuf [ shash_align_buffer_size ( unaligned_len , alignmask ) ]
2008-08-31 15:47:27 +10:00
__attribute__ ( ( aligned ) ) ;
2009-07-14 21:43:56 +08:00
u8 * buf = PTR_ALIGN ( & ubuf [ 0 ] , alignmask + 1 ) ;
2009-07-14 21:35:36 +08:00
int err ;
2008-08-31 15:47:27 +10:00
2009-03-27 13:03:51 +08:00
if ( unaligned_len > len )
unaligned_len = len ;
2008-08-31 15:47:27 +10:00
memcpy ( buf , data , unaligned_len ) ;
2009-07-14 21:35:36 +08:00
err = shash - > update ( desc , buf , unaligned_len ) ;
memset ( buf , 0 , unaligned_len ) ;
2008-08-31 15:47:27 +10:00
2009-07-14 21:35:36 +08:00
return err ? :
2008-08-31 15:47:27 +10:00
shash - > update ( desc , data + unaligned_len , len - unaligned_len ) ;
}
int crypto_shash_update ( struct shash_desc * desc , const u8 * data ,
unsigned int len )
{
struct crypto_shash * tfm = desc - > tfm ;
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
if ( ( unsigned long ) data & alignmask )
return shash_update_unaligned ( desc , data , len ) ;
return shash - > update ( desc , data , len ) ;
}
EXPORT_SYMBOL_GPL ( crypto_shash_update ) ;
static int shash_final_unaligned ( struct shash_desc * desc , u8 * out )
{
struct crypto_shash * tfm = desc - > tfm ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned int ds = crypto_shash_digestsize ( tfm ) ;
2009-07-14 21:43:56 +08:00
u8 ubuf [ shash_align_buffer_size ( ds , alignmask ) ]
2008-08-31 15:47:27 +10:00
__attribute__ ( ( aligned ) ) ;
2009-07-14 21:43:56 +08:00
u8 * buf = PTR_ALIGN ( & ubuf [ 0 ] , alignmask + 1 ) ;
2008-08-31 15:47:27 +10:00
int err ;
err = shash - > final ( desc , buf ) ;
2009-07-14 21:35:36 +08:00
if ( err )
goto out ;
2008-08-31 15:47:27 +10:00
memcpy ( out , buf , ds ) ;
2009-07-14 21:35:36 +08:00
out :
memset ( buf , 0 , ds ) ;
2008-08-31 15:47:27 +10:00
return err ;
}
int crypto_shash_final ( struct shash_desc * desc , u8 * out )
{
struct crypto_shash * tfm = desc - > tfm ;
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
if ( ( unsigned long ) out & alignmask )
return shash_final_unaligned ( desc , out ) ;
return shash - > final ( desc , out ) ;
}
EXPORT_SYMBOL_GPL ( crypto_shash_final ) ;
static int shash_finup_unaligned ( struct shash_desc * desc , const u8 * data ,
unsigned int len , u8 * out )
{
return crypto_shash_update ( desc , data , len ) ? :
crypto_shash_final ( desc , out ) ;
}
int crypto_shash_finup ( struct shash_desc * desc , const u8 * data ,
unsigned int len , u8 * out )
{
struct crypto_shash * tfm = desc - > tfm ;
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
2009-07-09 20:36:44 +08:00
if ( ( ( unsigned long ) data | ( unsigned long ) out ) & alignmask )
2008-08-31 15:47:27 +10:00
return shash_finup_unaligned ( desc , data , len , out ) ;
return shash - > finup ( desc , data , len , out ) ;
}
EXPORT_SYMBOL_GPL ( crypto_shash_finup ) ;
static int shash_digest_unaligned ( struct shash_desc * desc , const u8 * data ,
unsigned int len , u8 * out )
{
return crypto_shash_init ( desc ) ? :
2009-07-08 23:32:08 +08:00
crypto_shash_finup ( desc , data , len , out ) ;
2008-08-31 15:47:27 +10:00
}
int crypto_shash_digest ( struct shash_desc * desc , const u8 * data ,
unsigned int len , u8 * out )
{
struct crypto_shash * tfm = desc - > tfm ;
struct shash_alg * shash = crypto_shash_alg ( tfm ) ;
unsigned long alignmask = crypto_shash_alignmask ( tfm ) ;
2009-07-09 20:36:44 +08:00
if ( ( ( unsigned long ) data | ( unsigned long ) out ) & alignmask )
2008-08-31 15:47:27 +10:00
return shash_digest_unaligned ( desc , data , len , out ) ;
return shash - > digest ( desc , data , len , out ) ;
}
EXPORT_SYMBOL_GPL ( crypto_shash_digest ) ;
2009-07-22 12:37:06 +08:00
static int shash_default_export ( struct shash_desc * desc , void * out )
2008-11-02 21:38:11 +08:00
{
2009-07-22 12:37:06 +08:00
memcpy ( out , shash_desc_ctx ( desc ) , crypto_shash_descsize ( desc - > tfm ) ) ;
return 0 ;
2009-07-09 20:30:57 +08:00
}
2008-11-02 21:38:11 +08:00
2009-07-22 12:37:06 +08:00
static int shash_default_import ( struct shash_desc * desc , const void * in )
2009-07-09 20:30:57 +08:00
{
2009-07-22 12:37:06 +08:00
memcpy ( shash_desc_ctx ( desc ) , in , crypto_shash_descsize ( desc - > tfm ) ) ;
return 0 ;
2008-11-02 21:38:11 +08:00
}
2008-08-31 18:52:18 +10:00
static int shash_async_setkey ( struct crypto_ahash * tfm , const u8 * key ,
unsigned int keylen )
{
struct crypto_shash * * ctx = crypto_ahash_ctx ( tfm ) ;
return crypto_shash_setkey ( * ctx , key , keylen ) ;
}
static int shash_async_init ( struct ahash_request * req )
{
struct crypto_shash * * ctx = crypto_ahash_ctx ( crypto_ahash_reqtfm ( req ) ) ;
struct shash_desc * desc = ahash_request_ctx ( req ) ;
desc - > tfm = * ctx ;
desc - > flags = req - > base . flags ;
return crypto_shash_init ( desc ) ;
}
2009-07-12 21:25:20 +08:00
int shash_ahash_update ( struct ahash_request * req , struct shash_desc * desc )
2008-08-31 18:52:18 +10:00
{
struct crypto_hash_walk walk ;
int nbytes ;
for ( nbytes = crypto_hash_walk_first ( req , & walk ) ; nbytes > 0 ;
nbytes = crypto_hash_walk_done ( & walk , nbytes ) )
nbytes = crypto_shash_update ( desc , walk . data , nbytes ) ;
return nbytes ;
}
2009-07-12 21:25:20 +08:00
EXPORT_SYMBOL_GPL ( shash_ahash_update ) ;
static int shash_async_update ( struct ahash_request * req )
{
return shash_ahash_update ( req , ahash_request_ctx ( req ) ) ;
}
2008-08-31 18:52:18 +10:00
static int shash_async_final ( struct ahash_request * req )
{
return crypto_shash_final ( ahash_request_ctx ( req ) , req - > result ) ;
}
2009-07-15 12:40:40 +08:00
int shash_ahash_finup ( struct ahash_request * req , struct shash_desc * desc )
{
struct crypto_hash_walk walk ;
int nbytes ;
2009-07-15 21:26:41 +08:00
nbytes = crypto_hash_walk_first ( req , & walk ) ;
if ( ! nbytes )
return crypto_shash_final ( desc , req - > result ) ;
do {
2009-07-15 12:40:40 +08:00
nbytes = crypto_hash_walk_last ( & walk ) ?
crypto_shash_finup ( desc , walk . data , nbytes ,
req - > result ) :
crypto_shash_update ( desc , walk . data , nbytes ) ;
2009-07-15 21:26:41 +08:00
nbytes = crypto_hash_walk_done ( & walk , nbytes ) ;
} while ( nbytes > 0 ) ;
2009-07-15 12:40:40 +08:00
return nbytes ;
}
EXPORT_SYMBOL_GPL ( shash_ahash_finup ) ;
static int shash_async_finup ( struct ahash_request * req )
{
struct crypto_shash * * ctx = crypto_ahash_ctx ( crypto_ahash_reqtfm ( req ) ) ;
struct shash_desc * desc = ahash_request_ctx ( req ) ;
desc - > tfm = * ctx ;
desc - > flags = req - > base . flags ;
return shash_ahash_finup ( req , desc ) ;
}
2009-07-12 21:25:20 +08:00
int shash_ahash_digest ( struct ahash_request * req , struct shash_desc * desc )
2008-08-31 18:52:18 +10:00
{
struct scatterlist * sg = req - > src ;
unsigned int offset = sg - > offset ;
unsigned int nbytes = req - > nbytes ;
int err ;
if ( nbytes < min ( sg - > length , ( ( unsigned int ) ( PAGE_SIZE ) ) - offset ) ) {
void * data ;
data = crypto_kmap ( sg_page ( sg ) , 0 ) ;
err = crypto_shash_digest ( desc , data + offset , nbytes ,
req - > result ) ;
crypto_kunmap ( data , 0 ) ;
crypto_yield ( desc - > flags ) ;
2009-07-12 21:25:20 +08:00
} else
err = crypto_shash_init ( desc ) ? :
2009-07-15 12:40:40 +08:00
shash_ahash_finup ( req , desc ) ;
2008-08-31 18:52:18 +10:00
2009-07-12 21:25:20 +08:00
return err ;
}
EXPORT_SYMBOL_GPL ( shash_ahash_digest ) ;
2008-08-31 18:52:18 +10:00
2009-07-12 21:25:20 +08:00
static int shash_async_digest ( struct ahash_request * req )
{
struct crypto_shash * * ctx = crypto_ahash_ctx ( crypto_ahash_reqtfm ( req ) ) ;
struct shash_desc * desc = ahash_request_ctx ( req ) ;
2008-08-31 18:52:18 +10:00
2009-07-12 21:25:20 +08:00
desc - > tfm = * ctx ;
desc - > flags = req - > base . flags ;
2008-08-31 18:52:18 +10:00
2009-07-12 21:25:20 +08:00
return shash_ahash_digest ( req , desc ) ;
2008-08-31 18:52:18 +10:00
}
2009-07-15 12:40:40 +08:00
static int shash_async_export ( struct ahash_request * req , void * out )
{
return crypto_shash_export ( ahash_request_ctx ( req ) , out ) ;
}
static int shash_async_import ( struct ahash_request * req , const void * in )
{
return crypto_shash_import ( ahash_request_ctx ( req ) , in ) ;
}
2008-08-31 18:52:18 +10:00
static void crypto_exit_shash_ops_async ( struct crypto_tfm * tfm )
{
struct crypto_shash * * ctx = crypto_tfm_ctx ( tfm ) ;
crypto_free_shash ( * ctx ) ;
}
2009-07-14 12:28:26 +08:00
int crypto_init_shash_ops_async ( struct crypto_tfm * tfm )
2008-08-31 18:52:18 +10:00
{
struct crypto_alg * calg = tfm - > __crt_alg ;
2009-07-15 12:40:40 +08:00
struct shash_alg * alg = __crypto_shash_alg ( calg ) ;
2009-07-14 12:28:26 +08:00
struct crypto_ahash * crt = __crypto_ahash_cast ( tfm ) ;
2008-08-31 18:52:18 +10:00
struct crypto_shash * * ctx = crypto_tfm_ctx ( tfm ) ;
struct crypto_shash * shash ;
if ( ! crypto_mod_get ( calg ) )
return - EAGAIN ;
2009-02-18 16:56:59 +08:00
shash = crypto_create_tfm ( calg , & crypto_shash_type ) ;
2008-08-31 18:52:18 +10:00
if ( IS_ERR ( shash ) ) {
crypto_mod_put ( calg ) ;
return PTR_ERR ( shash ) ;
}
* ctx = shash ;
tfm - > exit = crypto_exit_shash_ops_async ;
crt - > init = shash_async_init ;
crt - > update = shash_async_update ;
2009-07-15 12:40:40 +08:00
crt - > final = shash_async_final ;
crt - > finup = shash_async_finup ;
2008-08-31 18:52:18 +10:00
crt - > digest = shash_async_digest ;
2009-07-15 12:40:40 +08:00
if ( alg - > setkey )
crt - > setkey = shash_async_setkey ;
if ( alg - > export )
crt - > export = shash_async_export ;
2009-07-24 13:57:13 +08:00
if ( alg - > import )
2009-07-15 12:40:40 +08:00
crt - > import = shash_async_import ;
2008-08-31 18:52:18 +10:00
crt - > reqsize = sizeof ( struct shash_desc ) + crypto_shash_descsize ( shash ) ;
return 0 ;
}
2008-08-31 22:21:09 +10:00
static int shash_compat_setkey ( struct crypto_hash * tfm , const u8 * key ,
unsigned int keylen )
{
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_hash_ctx ( tfm ) ;
struct shash_desc * desc = * descp ;
2008-08-31 22:21:09 +10:00
return crypto_shash_setkey ( desc - > tfm , key , keylen ) ;
}
static int shash_compat_init ( struct hash_desc * hdesc )
{
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_hash_ctx ( hdesc - > tfm ) ;
struct shash_desc * desc = * descp ;
2008-08-31 22:21:09 +10:00
desc - > flags = hdesc - > flags ;
return crypto_shash_init ( desc ) ;
}
static int shash_compat_update ( struct hash_desc * hdesc , struct scatterlist * sg ,
unsigned int len )
{
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_hash_ctx ( hdesc - > tfm ) ;
struct shash_desc * desc = * descp ;
2008-08-31 22:21:09 +10:00
struct crypto_hash_walk walk ;
int nbytes ;
for ( nbytes = crypto_hash_walk_first_compat ( hdesc , & walk , sg , len ) ;
nbytes > 0 ; nbytes = crypto_hash_walk_done ( & walk , nbytes ) )
nbytes = crypto_shash_update ( desc , walk . data , nbytes ) ;
return nbytes ;
}
static int shash_compat_final ( struct hash_desc * hdesc , u8 * out )
{
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_hash_ctx ( hdesc - > tfm ) ;
return crypto_shash_final ( * descp , out ) ;
2008-08-31 22:21:09 +10:00
}
static int shash_compat_digest ( struct hash_desc * hdesc , struct scatterlist * sg ,
unsigned int nbytes , u8 * out )
{
unsigned int offset = sg - > offset ;
int err ;
if ( nbytes < min ( sg - > length , ( ( unsigned int ) ( PAGE_SIZE ) ) - offset ) ) {
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_hash_ctx ( hdesc - > tfm ) ;
struct shash_desc * desc = * descp ;
2008-08-31 22:21:09 +10:00
void * data ;
desc - > flags = hdesc - > flags ;
data = crypto_kmap ( sg_page ( sg ) , 0 ) ;
err = crypto_shash_digest ( desc , data + offset , nbytes , out ) ;
crypto_kunmap ( data , 0 ) ;
crypto_yield ( desc - > flags ) ;
goto out ;
}
err = shash_compat_init ( hdesc ) ;
if ( err )
goto out ;
err = shash_compat_update ( hdesc , sg , nbytes ) ;
if ( err )
goto out ;
err = shash_compat_final ( hdesc , out ) ;
out :
return err ;
}
static void crypto_exit_shash_ops_compat ( struct crypto_tfm * tfm )
{
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_tfm_ctx ( tfm ) ;
struct shash_desc * desc = * descp ;
2008-08-31 22:21:09 +10:00
crypto_free_shash ( desc - > tfm ) ;
2009-07-14 12:50:12 +08:00
kzfree ( desc ) ;
2008-08-31 22:21:09 +10:00
}
static int crypto_init_shash_ops_compat ( struct crypto_tfm * tfm )
{
struct hash_tfm * crt = & tfm - > crt_hash ;
struct crypto_alg * calg = tfm - > __crt_alg ;
struct shash_alg * alg = __crypto_shash_alg ( calg ) ;
2009-07-14 12:50:12 +08:00
struct shash_desc * * descp = crypto_tfm_ctx ( tfm ) ;
2008-08-31 22:21:09 +10:00
struct crypto_shash * shash ;
2009-07-14 12:50:12 +08:00
struct shash_desc * desc ;
2008-08-31 22:21:09 +10:00
2009-02-05 16:19:31 +11:00
if ( ! crypto_mod_get ( calg ) )
return - EAGAIN ;
2009-02-18 16:56:59 +08:00
shash = crypto_create_tfm ( calg , & crypto_shash_type ) ;
2009-02-05 16:19:31 +11:00
if ( IS_ERR ( shash ) ) {
crypto_mod_put ( calg ) ;
2008-08-31 22:21:09 +10:00
return PTR_ERR ( shash ) ;
2009-02-05 16:19:31 +11:00
}
2008-08-31 22:21:09 +10:00
2009-07-14 12:50:12 +08:00
desc = kmalloc ( sizeof ( * desc ) + crypto_shash_descsize ( shash ) ,
GFP_KERNEL ) ;
if ( ! desc ) {
crypto_free_shash ( shash ) ;
return - ENOMEM ;
}
* descp = desc ;
2008-08-31 22:21:09 +10:00
desc - > tfm = shash ;
tfm - > exit = crypto_exit_shash_ops_compat ;
crt - > init = shash_compat_init ;
crt - > update = shash_compat_update ;
crt - > final = shash_compat_final ;
crt - > digest = shash_compat_digest ;
crt - > setkey = shash_compat_setkey ;
crt - > digestsize = alg - > digestsize ;
return 0 ;
}
2008-08-31 18:52:18 +10:00
static int crypto_init_shash_ops ( struct crypto_tfm * tfm , u32 type , u32 mask )
{
switch ( mask & CRYPTO_ALG_TYPE_MASK ) {
2008-08-31 22:21:09 +10:00
case CRYPTO_ALG_TYPE_HASH_MASK :
return crypto_init_shash_ops_compat ( tfm ) ;
2008-08-31 18:52:18 +10:00
}
return - EINVAL ;
}
static unsigned int crypto_shash_ctxsize ( struct crypto_alg * alg , u32 type ,
u32 mask )
{
switch ( mask & CRYPTO_ALG_TYPE_MASK ) {
2008-08-31 22:21:09 +10:00
case CRYPTO_ALG_TYPE_HASH_MASK :
2009-07-14 12:50:12 +08:00
return sizeof ( struct shash_desc * ) ;
2008-08-31 18:52:18 +10:00
}
return 0 ;
}
2009-07-13 20:46:25 +08:00
static int crypto_shash_init_tfm ( struct crypto_tfm * tfm )
2008-08-31 15:47:27 +10:00
{
2009-07-14 12:50:12 +08:00
struct crypto_shash * hash = __crypto_shash_cast ( tfm ) ;
hash - > descsize = crypto_shash_alg ( hash ) - > descsize ;
2008-08-31 15:47:27 +10:00
return 0 ;
}
2009-07-13 20:46:25 +08:00
static unsigned int crypto_shash_extsize ( struct crypto_alg * alg )
2008-08-31 15:47:27 +10:00
{
return alg - > cra_ctxsize ;
}
static void crypto_shash_show ( struct seq_file * m , struct crypto_alg * alg )
__attribute__ ( ( unused ) ) ;
static void crypto_shash_show ( struct seq_file * m , struct crypto_alg * alg )
{
struct shash_alg * salg = __crypto_shash_alg ( alg ) ;
seq_printf ( m , " type : shash \n " ) ;
seq_printf ( m , " blocksize : %u \n " , alg - > cra_blocksize ) ;
seq_printf ( m , " digestsize : %u \n " , salg - > digestsize ) ;
}
static const struct crypto_type crypto_shash_type = {
2008-08-31 18:52:18 +10:00
. ctxsize = crypto_shash_ctxsize ,
2008-08-31 15:47:27 +10:00
. extsize = crypto_shash_extsize ,
2008-08-31 18:52:18 +10:00
. init = crypto_init_shash_ops ,
2008-08-31 15:47:27 +10:00
. init_tfm = crypto_shash_init_tfm ,
# ifdef CONFIG_PROC_FS
. show = crypto_shash_show ,
# endif
. maskclear = ~ CRYPTO_ALG_TYPE_MASK ,
. maskset = CRYPTO_ALG_TYPE_MASK ,
. type = CRYPTO_ALG_TYPE_SHASH ,
. tfmsize = offsetof ( struct crypto_shash , base ) ,
} ;
struct crypto_shash * crypto_alloc_shash ( const char * alg_name , u32 type ,
u32 mask )
{
2009-02-18 16:56:59 +08:00
return crypto_alloc_tfm ( alg_name , & crypto_shash_type , type , mask ) ;
2008-08-31 15:47:27 +10:00
}
EXPORT_SYMBOL_GPL ( crypto_alloc_shash ) ;
2009-07-08 18:46:23 +08:00
static int shash_prepare_alg ( struct shash_alg * alg )
2008-08-31 15:47:27 +10:00
{
struct crypto_alg * base = & alg - > base ;
if ( alg - > digestsize > PAGE_SIZE / 8 | |
2009-07-09 20:30:57 +08:00
alg - > descsize > PAGE_SIZE / 8 | |
alg - > statesize > PAGE_SIZE / 8 )
2008-08-31 15:47:27 +10:00
return - EINVAL ;
base - > cra_type = & crypto_shash_type ;
base - > cra_flags & = ~ CRYPTO_ALG_TYPE_MASK ;
base - > cra_flags | = CRYPTO_ALG_TYPE_SHASH ;
2009-07-09 20:30:57 +08:00
2009-07-09 20:36:44 +08:00
if ( ! alg - > finup )
alg - > finup = shash_finup_unaligned ;
if ( ! alg - > digest )
alg - > digest = shash_digest_unaligned ;
2009-07-22 12:37:06 +08:00
if ( ! alg - > export ) {
alg - > export = shash_default_export ;
alg - > import = shash_default_import ;
alg - > statesize = alg - > descsize ;
}
2009-07-11 22:17:39 +08:00
if ( ! alg - > setkey )
alg - > setkey = shash_no_setkey ;
2009-07-09 20:30:57 +08:00
2009-07-08 18:46:23 +08:00
return 0 ;
}
int crypto_register_shash ( struct shash_alg * alg )
{
struct crypto_alg * base = & alg - > base ;
int err ;
err = shash_prepare_alg ( alg ) ;
if ( err )
return err ;
2008-08-31 15:47:27 +10:00
return crypto_register_alg ( base ) ;
}
EXPORT_SYMBOL_GPL ( crypto_register_shash ) ;
int crypto_unregister_shash ( struct shash_alg * alg )
{
return crypto_unregister_alg ( & alg - > base ) ;
}
EXPORT_SYMBOL_GPL ( crypto_unregister_shash ) ;
2009-07-08 18:46:23 +08:00
int shash_register_instance ( struct crypto_template * tmpl ,
struct shash_instance * inst )
{
int err ;
err = shash_prepare_alg ( & inst - > alg ) ;
if ( err )
return err ;
return crypto_register_instance ( tmpl , shash_crypto_instance ( inst ) ) ;
}
EXPORT_SYMBOL_GPL ( shash_register_instance ) ;
2009-07-07 15:17:12 +08:00
void shash_free_instance ( struct crypto_instance * inst )
{
crypto_drop_spawn ( crypto_instance_ctx ( inst ) ) ;
kfree ( shash_instance ( inst ) ) ;
}
EXPORT_SYMBOL_GPL ( shash_free_instance ) ;
2009-07-08 17:21:37 +08:00
int crypto_init_shash_spawn ( struct crypto_shash_spawn * spawn ,
struct shash_alg * alg ,
struct crypto_instance * inst )
{
return crypto_init_spawn2 ( & spawn - > base , & alg - > base , inst ,
& crypto_shash_type ) ;
}
EXPORT_SYMBOL_GPL ( crypto_init_shash_spawn ) ;
2009-07-08 17:56:28 +08:00
struct shash_alg * shash_attr_alg ( struct rtattr * rta , u32 type , u32 mask )
{
struct crypto_alg * alg ;
alg = crypto_attr_alg2 ( rta , & crypto_shash_type , type , mask ) ;
return IS_ERR ( alg ) ? ERR_CAST ( alg ) :
container_of ( alg , struct shash_alg , base ) ;
}
EXPORT_SYMBOL_GPL ( shash_attr_alg ) ;
2008-08-31 15:47:27 +10:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Synchronous cryptographic hash type " ) ;