2005-04-16 15:20:36 -07:00
/*
* Cryptographic API .
*
* HMAC : Keyed - Hashing for Message Authentication ( RFC2104 ) .
*
* Copyright ( c ) 2002 James Morris < jmorris @ intercode . com . au >
2006-08-21 20:50:52 +10:00
* Copyright ( c ) 2006 Herbert Xu < herbert @ gondor . apana . org . au >
2005-04-16 15:20:36 -07:00
*
* The HMAC implementation is derived from USAGI .
* Copyright ( c ) 2002 Kazunori Miyazawa < miyazawa @ linux - ipv6 . org > / USAGI
*
* 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
2006-08-21 20:50:52 +10:00
* Software Foundation ; either version 2 of the License , or ( at your option )
2005-04-16 15:20:36 -07:00
* any later version .
*
*/
2006-08-21 20:50:52 +10:00
2008-08-31 22:21:09 +10:00
# include <crypto/internal/hash.h>
2007-12-05 20:59:25 +11:00
# include <crypto/scatterwalk.h>
2006-08-21 20:50:52 +10:00
# include <linux/err.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
2005-09-17 17:55:31 +10:00
# include <linux/scatterlist.h>
2006-08-21 20:50:52 +10:00
# include <linux/string.h>
struct hmac_ctx {
2009-07-24 15:18:41 +08:00
struct crypto_shash * hash ;
2006-08-21 20:50:52 +10:00
} ;
2005-04-16 15:20:36 -07:00
2006-08-21 20:50:52 +10:00
static inline void * align_ptr ( void * p , unsigned int align )
{
return ( void * ) ALIGN ( ( unsigned long ) p , align ) ;
}
2009-07-09 12:43:37 +08:00
static inline struct hmac_ctx * hmac_ctx ( struct crypto_shash * tfm )
2006-08-21 20:50:52 +10:00
{
2009-07-09 12:43:37 +08:00
return align_ptr ( crypto_shash_ctx_aligned ( tfm ) +
2009-07-24 15:18:41 +08:00
crypto_shash_statesize ( tfm ) * 2 ,
2009-07-09 12:43:37 +08:00
crypto_tfm_ctx_alignment ( ) ) ;
2006-08-21 20:50:52 +10:00
}
2009-07-09 12:43:37 +08:00
static int hmac_setkey ( struct crypto_shash * parent ,
2006-08-21 20:50:52 +10:00
const u8 * inkey , unsigned int keylen )
{
2009-07-09 12:43:37 +08:00
int bs = crypto_shash_blocksize ( parent ) ;
int ds = crypto_shash_digestsize ( parent ) ;
2009-07-24 15:18:41 +08:00
int ss = crypto_shash_statesize ( parent ) ;
2009-07-09 12:43:37 +08:00
char * ipad = crypto_shash_ctx_aligned ( parent ) ;
2009-07-24 15:18:41 +08:00
char * opad = ipad + ss ;
struct hmac_ctx * ctx = align_ptr ( opad + ss ,
2009-07-09 12:43:37 +08:00
crypto_tfm_ctx_alignment ( ) ) ;
2009-07-24 15:18:41 +08:00
struct crypto_shash * hash = ctx - > hash ;
struct {
struct shash_desc shash ;
char ctx [ crypto_shash_descsize ( hash ) ] ;
} desc ;
2006-08-21 20:50:52 +10:00
unsigned int i ;
2009-07-24 15:18:41 +08:00
desc . shash . tfm = hash ;
desc . shash . flags = crypto_shash_get_flags ( parent ) &
CRYPTO_TFM_REQ_MAY_SLEEP ;
2006-08-21 20:50:52 +10:00
if ( keylen > bs ) {
int err ;
2009-07-24 15:18:41 +08:00
err = crypto_shash_digest ( & desc . shash , inkey , keylen , ipad ) ;
2006-08-21 20:50:52 +10:00
if ( err )
return err ;
keylen = ds ;
2009-07-24 15:18:41 +08:00
} else
memcpy ( ipad , inkey , keylen ) ;
2006-08-21 20:50:52 +10:00
memset ( ipad + keylen , 0 , bs - keylen ) ;
memcpy ( opad , ipad , bs ) ;
for ( i = 0 ; i < bs ; i + + ) {
ipad [ i ] ^ = 0x36 ;
opad [ i ] ^ = 0x5c ;
}
2009-07-24 15:18:41 +08:00
return crypto_shash_init ( & desc . shash ) ? :
crypto_shash_update ( & desc . shash , ipad , bs ) ? :
crypto_shash_export ( & desc . shash , ipad ) ? :
crypto_shash_init ( & desc . shash ) ? :
crypto_shash_update ( & desc . shash , opad , bs ) ? :
crypto_shash_export ( & desc . shash , opad ) ;
2006-08-21 20:50:52 +10:00
}
2009-07-24 15:18:41 +08:00
static int hmac_export ( struct shash_desc * pdesc , void * out )
{
struct shash_desc * desc = shash_desc_ctx ( pdesc ) ;
desc - > flags = pdesc - > flags & CRYPTO_TFM_REQ_MAY_SLEEP ;
return crypto_shash_export ( desc , out ) ;
}
static int hmac_import ( struct shash_desc * pdesc , const void * in )
2006-08-21 20:50:52 +10:00
{
2009-07-09 12:43:37 +08:00
struct shash_desc * desc = shash_desc_ctx ( pdesc ) ;
2009-07-24 15:18:41 +08:00
struct hmac_ctx * ctx = hmac_ctx ( pdesc - > tfm ) ;
2009-07-09 12:43:37 +08:00
2009-07-24 15:18:41 +08:00
desc - > tfm = ctx - > hash ;
2009-07-09 12:43:37 +08:00
desc - > flags = pdesc - > flags & CRYPTO_TFM_REQ_MAY_SLEEP ;
2009-07-24 15:18:41 +08:00
return crypto_shash_import ( desc , in ) ;
}
static int hmac_init ( struct shash_desc * pdesc )
{
return hmac_import ( pdesc , crypto_shash_ctx_aligned ( pdesc - > tfm ) ) ;
2006-08-21 20:50:52 +10:00
}
2009-07-09 12:43:37 +08:00
static int hmac_update ( struct shash_desc * pdesc ,
const u8 * data , unsigned int nbytes )
2006-08-21 20:50:52 +10:00
{
2009-07-09 12:43:37 +08:00
struct shash_desc * desc = shash_desc_ctx ( pdesc ) ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
desc - > flags = pdesc - > flags & CRYPTO_TFM_REQ_MAY_SLEEP ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
return crypto_shash_update ( desc , data , nbytes ) ;
2006-08-21 20:50:52 +10:00
}
2009-07-09 12:43:37 +08:00
static int hmac_final ( struct shash_desc * pdesc , u8 * out )
2006-08-21 20:50:52 +10:00
{
2009-07-09 12:43:37 +08:00
struct crypto_shash * parent = pdesc - > tfm ;
int ds = crypto_shash_digestsize ( parent ) ;
2009-07-24 15:18:41 +08:00
int ss = crypto_shash_statesize ( parent ) ;
char * opad = crypto_shash_ctx_aligned ( parent ) + ss ;
2009-07-09 12:43:37 +08:00
struct shash_desc * desc = shash_desc_ctx ( pdesc ) ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
desc - > flags = pdesc - > flags & CRYPTO_TFM_REQ_MAY_SLEEP ;
2006-09-24 09:30:19 +10:00
2009-07-24 15:18:41 +08:00
return crypto_shash_final ( desc , out ) ? :
crypto_shash_import ( desc , opad ) ? :
crypto_shash_finup ( desc , out , ds , out ) ;
2006-08-21 20:50:52 +10:00
}
2009-07-09 12:43:37 +08:00
static int hmac_finup ( struct shash_desc * pdesc , const u8 * data ,
unsigned int nbytes , u8 * out )
2006-08-21 20:50:52 +10:00
{
2009-07-09 12:43:37 +08:00
struct crypto_shash * parent = pdesc - > tfm ;
int ds = crypto_shash_digestsize ( parent ) ;
2009-07-24 15:18:41 +08:00
int ss = crypto_shash_statesize ( parent ) ;
char * opad = crypto_shash_ctx_aligned ( parent ) + ss ;
2009-07-09 12:43:37 +08:00
struct shash_desc * desc = shash_desc_ctx ( pdesc ) ;
2007-10-22 19:40:16 +02:00
2009-07-09 12:43:37 +08:00
desc - > flags = pdesc - > flags & CRYPTO_TFM_REQ_MAY_SLEEP ;
2006-08-21 20:50:52 +10:00
2009-07-24 15:18:41 +08:00
return crypto_shash_finup ( desc , data , nbytes , out ) ? :
crypto_shash_import ( desc , opad ) ? :
crypto_shash_finup ( desc , out , ds , out ) ;
2006-08-21 20:50:52 +10:00
}
static int hmac_init_tfm ( struct crypto_tfm * tfm )
{
2009-07-09 12:43:37 +08:00
struct crypto_shash * parent = __crypto_shash_cast ( tfm ) ;
struct crypto_shash * hash ;
2006-08-21 20:50:52 +10:00
struct crypto_instance * inst = ( void * ) tfm - > __crt_alg ;
2009-07-09 12:43:37 +08:00
struct crypto_shash_spawn * spawn = crypto_instance_ctx ( inst ) ;
struct hmac_ctx * ctx = hmac_ctx ( parent ) ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
hash = crypto_spawn_shash ( spawn ) ;
2006-12-17 10:05:58 +11:00
if ( IS_ERR ( hash ) )
return PTR_ERR ( hash ) ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
parent - > descsize = sizeof ( struct shash_desc ) +
crypto_shash_descsize ( hash ) ;
2009-07-24 15:18:41 +08:00
ctx - > hash = hash ;
2006-08-21 20:50:52 +10:00
return 0 ;
}
static void hmac_exit_tfm ( struct crypto_tfm * tfm )
{
2009-07-09 12:43:37 +08:00
struct hmac_ctx * ctx = hmac_ctx ( __crypto_shash_cast ( tfm ) ) ;
2009-07-24 15:18:41 +08:00
crypto_free_shash ( ctx - > hash ) ;
2006-08-21 20:50:52 +10:00
}
2009-07-09 12:43:37 +08:00
static int hmac_create ( struct crypto_template * tmpl , struct rtattr * * tb )
2006-08-21 20:50:52 +10:00
{
2009-07-09 12:43:37 +08:00
struct shash_instance * inst ;
2006-08-21 20:50:52 +10:00
struct crypto_alg * alg ;
2009-07-09 12:43:37 +08:00
struct shash_alg * salg ;
2007-01-01 18:37:02 +11:00
int err ;
2008-07-07 20:23:56 +08:00
int ds ;
2009-07-24 15:18:41 +08:00
int ss ;
2007-01-01 18:37:02 +11:00
2009-07-09 12:43:37 +08:00
err = crypto_check_attr_type ( tb , CRYPTO_ALG_TYPE_SHASH ) ;
2007-01-01 18:37:02 +11:00
if ( err )
2009-07-09 12:43:37 +08:00
return err ;
salg = shash_attr_alg ( tb [ 1 ] , 0 , 0 ) ;
if ( IS_ERR ( salg ) )
return PTR_ERR ( salg ) ;
err = - EINVAL ;
ds = salg - > digestsize ;
2009-07-24 15:18:41 +08:00
ss = salg - > statesize ;
2009-07-09 12:43:37 +08:00
alg = & salg - > base ;
2009-07-24 15:18:41 +08:00
if ( ds > alg - > cra_blocksize | |
ss < alg - > cra_blocksize )
2008-07-07 20:23:56 +08:00
goto out_put_alg ;
2009-07-09 12:43:37 +08:00
inst = shash_alloc_instance ( " hmac " , alg ) ;
2009-07-15 16:52:55 +08:00
err = PTR_ERR ( inst ) ;
2006-08-21 20:50:52 +10:00
if ( IS_ERR ( inst ) )
goto out_put_alg ;
2009-07-09 12:43:37 +08:00
err = crypto_init_shash_spawn ( shash_instance_ctx ( inst ) , salg ,
shash_crypto_instance ( inst ) ) ;
if ( err )
goto out_free_inst ;
inst - > alg . base . cra_priority = alg - > cra_priority ;
inst - > alg . base . cra_blocksize = alg - > cra_blocksize ;
inst - > alg . base . cra_alignmask = alg - > cra_alignmask ;
2006-08-21 20:50:52 +10:00
2009-07-24 15:18:41 +08:00
ss = ALIGN ( ss , alg - > cra_alignmask + 1 ) ;
2009-07-09 12:43:37 +08:00
inst - > alg . digestsize = ds ;
2009-07-24 15:18:41 +08:00
inst - > alg . statesize = ss ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
inst - > alg . base . cra_ctxsize = sizeof ( struct hmac_ctx ) +
2009-07-24 15:18:41 +08:00
ALIGN ( ss * 2 , crypto_tfm_ctx_alignment ( ) ) ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
inst - > alg . base . cra_init = hmac_init_tfm ;
inst - > alg . base . cra_exit = hmac_exit_tfm ;
2006-08-21 20:50:52 +10:00
2009-07-09 12:43:37 +08:00
inst - > alg . init = hmac_init ;
inst - > alg . update = hmac_update ;
inst - > alg . final = hmac_final ;
inst - > alg . finup = hmac_finup ;
2009-07-24 15:18:41 +08:00
inst - > alg . export = hmac_export ;
inst - > alg . import = hmac_import ;
2009-07-09 12:43:37 +08:00
inst - > alg . setkey = hmac_setkey ;
err = shash_register_instance ( tmpl , inst ) ;
if ( err ) {
out_free_inst :
shash_free_instance ( shash_crypto_instance ( inst ) ) ;
}
2006-08-21 20:50:52 +10:00
out_put_alg :
crypto_mod_put ( alg ) ;
2009-07-09 12:43:37 +08:00
return err ;
2006-08-21 20:50:52 +10:00
}
static struct crypto_template hmac_tmpl = {
. name = " hmac " ,
2009-07-09 12:43:37 +08:00
. create = hmac_create ,
. free = shash_free_instance ,
2006-08-21 20:50:52 +10:00
. module = THIS_MODULE ,
} ;
static int __init hmac_module_init ( void )
{
return crypto_register_template ( & hmac_tmpl ) ;
}
static void __exit hmac_module_exit ( void )
{
crypto_unregister_template ( & hmac_tmpl ) ;
}
module_init ( hmac_module_init ) ;
module_exit ( hmac_module_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " HMAC hash algorithm " ) ;