224 lines
5.5 KiB
C
Executable File
224 lines
5.5 KiB
C
Executable File
/*
|
|
|
|
This program is part of the TACLeBench benchmark suite.
|
|
Version V 1.x
|
|
|
|
Name: sha.c
|
|
|
|
Author: Peter C. Gutmann's (heavily modified by Uwe Hollerbach)
|
|
|
|
NIST Secure Hash Algorithm
|
|
|
|
Source: Peter C. Gutmann's implementation as found in Applied Cryptography by Bruce Schneier
|
|
|
|
Changes: no major functional changes
|
|
|
|
License: GNU Lesser General Public License
|
|
|
|
*/
|
|
|
|
#include "memcpy.h"
|
|
#include "memset.h"
|
|
#include "sha.h"
|
|
|
|
/*
|
|
Declaration of global variables
|
|
*/
|
|
struct SHA_INFO sha_info;
|
|
|
|
/* SHA f()-functions */
|
|
#define f1(x,y,z) ((x & y) | (~x & z))
|
|
#define f2(x,y,z) (x ^ y ^ z)
|
|
#define f3(x,y,z) ((x & y) | (x & z) | (y & z))
|
|
#define f4(x,y,z) (x ^ y ^ z)
|
|
|
|
/* SHA constants */
|
|
#define CONST1 0x5a827999L
|
|
#define CONST2 0x6ed9eba1L
|
|
#define CONST3 0x8f1bbcdcL
|
|
#define CONST4 0xca62c1d6L
|
|
|
|
#define BLOCK_SIZE 8192
|
|
|
|
/* 32-bit rotate */
|
|
|
|
#define ROT32(x,n) ((x << n) | (x >> (32 - n)))
|
|
|
|
#define FUNC(n,i) \
|
|
temp = ROT32(A,5) + f##n(B,C,D) + E + W[ i ] + CONST##n; \
|
|
E = D; D = C; C = ROT32(B,30); B = A; A = temp
|
|
|
|
/* do SHA transformation */
|
|
void sha_transform( struct SHA_INFO *sha_info )
|
|
{
|
|
int i;
|
|
LONG temp, A, B, C, D, E, W[ 80 ];
|
|
|
|
_Pragma( "loopbound min 16 max 16" )
|
|
for ( i = 0; i < 16; ++i )
|
|
W[ i ] = sha_info->data[ i ];
|
|
_Pragma( "loopbound min 64 max 64" )
|
|
for ( i = 16; i < 80; ++i )
|
|
W[ i ] = W[ i - 3 ] ^ W[ i - 8 ] ^ W[ i - 14 ] ^ W[ i - 16 ];
|
|
|
|
A = sha_info->digest[ 0 ];
|
|
B = sha_info->digest[ 1 ];
|
|
C = sha_info->digest[ 2 ];
|
|
D = sha_info->digest[ 3 ];
|
|
E = sha_info->digest[ 4 ];
|
|
|
|
|
|
_Pragma( "loopbound min 20 max 20" )
|
|
for ( i = 0; i < 20; ++i ) {
|
|
FUNC( 1, i );
|
|
}
|
|
_Pragma( "loopbound min 20 max 20" )
|
|
for ( i = 20; i < 40; ++i ) {
|
|
FUNC( 2, i );
|
|
}
|
|
_Pragma( "loopbound min 20 max 20" )
|
|
for ( i = 40; i < 60; ++i ) {
|
|
FUNC( 3, i );
|
|
}
|
|
_Pragma( "loopbound min 20 max 20" )
|
|
for ( i = 60; i < 80; ++i ) {
|
|
FUNC( 4, i );
|
|
}
|
|
sha_info->digest[ 0 ] += A;
|
|
sha_info->digest[ 1 ] += B;
|
|
sha_info->digest[ 2 ] += C;
|
|
sha_info->digest[ 3 ] += D;
|
|
sha_info->digest[ 4 ] += E;
|
|
}
|
|
|
|
|
|
/* change endianness of data */
|
|
void sha_byte_reverse( LONG *buffer, int count )
|
|
{
|
|
int i;
|
|
BYTE ct[ 4 ], *cp;
|
|
|
|
count /= sizeof( LONG );
|
|
cp = ( BYTE * ) buffer;
|
|
_Pragma( "loopbound min 16 max 16" )
|
|
for ( i = 0; i < count; ++i ) {
|
|
ct[ 0 ] = cp[ 0 ];
|
|
ct[ 1 ] = cp[ 1 ];
|
|
ct[ 2 ] = cp[ 2 ];
|
|
ct[ 3 ] = cp[ 3 ];
|
|
cp[ 0 ] = ct[ 3 ];
|
|
cp[ 1 ] = ct[ 2 ];
|
|
cp[ 2 ] = ct[ 1 ];
|
|
cp[ 3 ] = ct[ 0 ];
|
|
cp += sizeof( LONG );
|
|
}
|
|
}
|
|
|
|
/* initialize the SHA digest */
|
|
void sha_init( void )
|
|
{
|
|
int i;
|
|
sha_info.digest[ 0 ] = 0x67452301L;
|
|
sha_info.digest[ 1 ] = 0xefcdab89L;
|
|
sha_info.digest[ 2 ] = 0x98badcfeL;
|
|
sha_info.digest[ 3 ] = 0x10325476L;
|
|
sha_info.digest[ 4 ] = 0xc3d2e1f0L;
|
|
sha_info.count_lo = 0L;
|
|
sha_info.count_hi = 0L;
|
|
for ( i = 0; i < 16; i++ )
|
|
sha_info.data[ i ] = 0;
|
|
}
|
|
|
|
size_t sha_fread( void *ptr, size_t size, size_t count,
|
|
struct SHA_MY_FILE *stream )
|
|
{
|
|
unsigned i = stream->cur_pos, i2 = 0;
|
|
size_t number_of_chars_to_read =
|
|
stream->size - stream->cur_pos >= size * count ?
|
|
size * count : stream->size - stream->cur_pos;
|
|
_Pragma( "loopbound min 0 max 8192" )
|
|
while ( i < stream->cur_pos + number_of_chars_to_read )
|
|
( ( unsigned char * )ptr )[ i2++ ] = stream->data[ i++ ];
|
|
stream->cur_pos += number_of_chars_to_read;
|
|
return ( number_of_chars_to_read );
|
|
}
|
|
|
|
/* update the SHA digest */
|
|
void sha_update( struct SHA_INFO *sha_info, BYTE *buffer, int count )
|
|
{
|
|
if ( ( sha_info->count_lo + ( ( LONG ) count << 3 ) ) < sha_info->count_lo )
|
|
++sha_info->count_hi;
|
|
sha_info->count_lo += ( LONG ) count << 3;
|
|
sha_info->count_hi += ( LONG ) count >> 29;
|
|
_Pragma( "loopbound min 8 max 128" )
|
|
while ( count >= SHA_BLOCKSIZE ) {
|
|
sha_glibc_memcpy( sha_info->data, buffer, SHA_BLOCKSIZE );
|
|
sha_byte_reverse( sha_info->data, SHA_BLOCKSIZE );
|
|
sha_transform( sha_info );
|
|
buffer += SHA_BLOCKSIZE;
|
|
count -= SHA_BLOCKSIZE;
|
|
}
|
|
|
|
sha_glibc_memcpy( sha_info->data, buffer, count );
|
|
}
|
|
|
|
/* finish computing the SHA digest */
|
|
void sha_final( struct SHA_INFO *sha_info )
|
|
{
|
|
int count;
|
|
LONG lo_bit_count, hi_bit_count;
|
|
|
|
lo_bit_count = sha_info->count_lo;
|
|
hi_bit_count = sha_info->count_hi;
|
|
count = ( int ) ( ( lo_bit_count >> 3 ) & 0x3f );
|
|
( ( BYTE * ) sha_info->data )[ count++ ] = 0x80;
|
|
if ( count > 56 ) {
|
|
sha_glibc_memset( ( BYTE * ) &sha_info->data + count, 0, 64 - count );
|
|
sha_byte_reverse( sha_info->data, SHA_BLOCKSIZE );
|
|
sha_transform( sha_info );
|
|
sha_glibc_memset( &sha_info->data, 0, 56 );
|
|
} else
|
|
sha_glibc_memset( ( BYTE * ) &sha_info->data + count, 0, 56 - count );
|
|
|
|
sha_byte_reverse( sha_info->data, SHA_BLOCKSIZE );
|
|
sha_info->data[ 14 ] = hi_bit_count;
|
|
sha_info->data[ 15 ] = lo_bit_count;
|
|
sha_transform( sha_info );
|
|
}
|
|
|
|
/* compute the SHA digest of a FILE stream */
|
|
|
|
void sha_stream( struct SHA_INFO *sha_info, struct SHA_MY_FILE *fin )
|
|
{
|
|
int i;
|
|
BYTE data[ BLOCK_SIZE ];
|
|
_Pragma( "loopbound min 5 max 5" )
|
|
while ( ( i = sha_fread( data, 1, BLOCK_SIZE, fin ) ) > 0 )
|
|
sha_update( sha_info, data, i );
|
|
|
|
sha_final( sha_info );
|
|
}
|
|
|
|
void _Pragma( "entrypoint" ) sha_main( void )
|
|
{
|
|
struct SHA_MY_FILE fin;
|
|
fin.data = sha_data;
|
|
fin.size = 32743; // set size = 3247552 for input_large
|
|
fin.cur_pos = 0;
|
|
sha_stream( &sha_info, &fin );
|
|
}
|
|
|
|
int sha_return( void )
|
|
{
|
|
int sum = 0;
|
|
sum = sha_info.data[ 14 ] + sha_info.data[ 15 ];
|
|
return ( sum - 261944 != 0 );
|
|
}
|
|
|
|
int main ( void )
|
|
{
|
|
sha_init();
|
|
sha_main();
|
|
return ( sha_return() );
|
|
}
|