2008-06-03 14:29:27 +10:00
/*
Unix SMB / CIFS implementation .
Interface header : HMAC SHA - 256 code
Copyright ( C ) Andrew Tridgell 2008
based in hmacsha1 . c which is :
Copyright ( C ) Stefan Metzmacher
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 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
/*
taken direct from rfc2202 implementation and modified for suitable use
*/
2010-06-15 12:01:09 +02:00
# include "replace.h"
2008-09-24 16:10:53 +02:00
# include "../lib/crypto/crypto.h"
2008-06-03 14:29:27 +10:00
/***********************************************************************
the rfc 2104 / 2202 version of hmac_sha256 initialisation .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
_PUBLIC_ void hmac_sha256_init ( const uint8_t * key , size_t key_len , struct HMACSHA256Context * ctx )
{
int i ;
uint8_t tk [ SHA256_DIGEST_LENGTH ] ;
/* if key is longer than 64 bytes reset it to key=HASH(key) */
if ( key_len > 64 )
{
SHA256_CTX tctx ;
2011-04-06 14:36:21 +10:00
samba_SHA256_Init ( & tctx ) ;
samba_SHA256_Update ( & tctx , key , key_len ) ;
samba_SHA256_Final ( tk , & tctx ) ;
2008-06-03 14:29:27 +10:00
key = tk ;
key_len = SHA256_DIGEST_LENGTH ;
}
/* start out by storing key in pads */
ZERO_STRUCT ( ctx - > k_ipad ) ;
ZERO_STRUCT ( ctx - > k_opad ) ;
memcpy ( ctx - > k_ipad , key , key_len ) ;
memcpy ( ctx - > k_opad , key , key_len ) ;
/* XOR key with ipad and opad values */
for ( i = 0 ; i < 64 ; i + + )
{
ctx - > k_ipad [ i ] ^ = 0x36 ;
ctx - > k_opad [ i ] ^ = 0x5c ;
}
2011-04-06 14:36:21 +10:00
samba_SHA256_Init ( & ctx - > ctx ) ;
samba_SHA256_Update ( & ctx - > ctx , ctx - > k_ipad , 64 ) ;
2008-06-03 14:29:27 +10:00
}
/***********************************************************************
update hmac_sha256 " inner " buffer
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
_PUBLIC_ void hmac_sha256_update ( const uint8_t * data , size_t data_len , struct HMACSHA256Context * ctx )
{
2011-04-06 14:36:21 +10:00
samba_SHA256_Update ( & ctx - > ctx , data , data_len ) ; /* then text of datagram */
2008-06-03 14:29:27 +10:00
}
/***********************************************************************
finish off hmac_sha256 " inner " buffer and generate outer one .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
_PUBLIC_ void hmac_sha256_final ( uint8_t digest [ SHA256_DIGEST_LENGTH ] , struct HMACSHA256Context * ctx )
{
SHA256_CTX ctx_o ;
2011-04-06 14:36:21 +10:00
samba_SHA256_Final ( digest , & ctx - > ctx ) ;
2008-06-03 14:29:27 +10:00
2011-04-06 14:36:21 +10:00
samba_SHA256_Init ( & ctx_o ) ;
samba_SHA256_Update ( & ctx_o , ctx - > k_opad , 64 ) ;
samba_SHA256_Update ( & ctx_o , digest , SHA256_DIGEST_LENGTH ) ;
samba_SHA256_Final ( digest , & ctx_o ) ;
2008-06-03 14:29:27 +10:00
}