/**************************************************************************** * * * cryptlib RIPEMD-160 Hash Routines * * Copyright Peter Gutmann 1996-2005 * * * ****************************************************************************/ #if defined( INC_ALL ) #include "crypt.h" #include "context.h" #include "ripemd.h" #else #include "crypt.h" #include "context/context.h" #include "crypt/ripemd.h" #endif /* Compiler-specific includes */ #ifdef USE_RIPEMD160 #define HASH_STATE_SIZE sizeof( RIPEMD160_CTX ) /**************************************************************************** * * * RIPEMD160 Self-test Routines * * * ****************************************************************************/ /* Test the RIPEMD160 output against the test vectors given in the RIPEMD-160 paper */ void ripemd160HashBuffer( HASHINFO hashInfo, BYTE *outBuffer, const int outBufMaxLength, const BYTE *inBuffer, const int inLength, const HASH_STATE hashState ); static const struct { const char FAR_BSS *data; /* Data to hash */ const int length; /* Length of data */ const BYTE digest[ RIPEMD160_DIGEST_LENGTH ]; /* Digest of data */ } FAR_BSS digestValues[] = { { NULL, 0, { 0x9C, 0x11, 0x85, 0xA5, 0xC5, 0xE9, 0xFC, 0x54, 0x61, 0x28, 0x08, 0x97, 0x7E, 0xE8, 0xF5, 0x48, 0xB2, 0x25, 0x8D, 0x31 } }, { "a", 1, { 0x0B, 0xDC, 0x9D, 0x2D, 0x25, 0x6B, 0x3E, 0xE9, 0xDA, 0xAE, 0x34, 0x7B, 0xE6, 0xF4, 0xDC, 0x83, 0x5A, 0x46, 0x7F, 0xFE } }, { "abc", 3, { 0x8E, 0xB2, 0x08, 0xF7, 0xE0, 0x5D, 0x98, 0x7A, 0x9B, 0x04, 0x4A, 0x8E, 0x98, 0xC6, 0xB0, 0x87, 0xF1, 0x5A, 0x0B, 0xFC } }, { "message digest", 14, { 0x5D, 0x06, 0x89, 0xEF, 0x49, 0xD2, 0xFA, 0xE5, 0x72, 0xB8, 0x81, 0xB1, 0x23, 0xA8, 0x5F, 0xFA, 0x21, 0x59, 0x5F, 0x36 } }, { "abcdefghijklmnopqrstuvwxyz", 26, { 0xF7, 0x1C, 0x27, 0x10, 0x9C, 0x69, 0x2C, 0x1B, 0x56, 0xBB, 0xDC, 0xEB, 0x5B, 0x9D, 0x28, 0x65, 0xB3, 0x70, 0x8D, 0xBC } }, { "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq", 56, { 0x12, 0xA0, 0x53, 0x38, 0x4A, 0x9C, 0x0C, 0x88, 0xE4, 0x05, 0xA0, 0x6C, 0x27, 0xDC, 0xF4, 0x9A, 0xDA, 0x62, 0xEB, 0x2B } }, { "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789", 62, { 0xB0, 0xE2, 0x0B, 0x6E, 0x31, 0x16, 0x64, 0x02, 0x86, 0xED, 0x3A, 0x87, 0xA5, 0x71, 0x30, 0x79, 0xB2, 0x1F, 0x51, 0x89 } }, { "12345678901234567890123456789012345678901234567890123456789012345678901234567890", 80, { 0x9B, 0x75, 0x2E, 0x45, 0x57, 0x3D, 0x4B, 0x39, 0xF4, 0xDB, 0xD3, 0x32, 0x3C, 0xAB, 0x82, 0xBF, 0x63, 0x32, 0x6B, 0xFB } }, { NULL, 0, { 0 } } }; static int selfTest( void ) { const CAPABILITY_INFO *capabilityInfo = getRipemd160Capability(); BYTE hashData[ HASH_STATE_SIZE + 8 ]; int i, status; /* Test RIPEMD160 against the test vectors from the RIPEMD-160 paper */ for( i = 0; digestValues[ i ].data != NULL; i++ ) { status = testHash( capabilityInfo, hashData, digestValues[ i ].data, digestValues[ i ].length, digestValues[ i ].digest ); if( cryptStatusError( status ) ) return( status ); } return( CRYPT_OK ); } /**************************************************************************** * * * Control Routines * * * ****************************************************************************/ /* Return context subtype-specific information */ static int getInfo( const CAPABILITY_INFO_TYPE type, void *varParam, const int constParam ) { if( type == CAPABILITY_INFO_STATESIZE ) return( HASH_STATE_SIZE ); return( getDefaultInfo( type, varParam, constParam ) ); } /**************************************************************************** * * * RIPEMD160 Hash Routines * * * ****************************************************************************/ /* Hash data using RIPEMD160 */ static int hash( CONTEXT_INFO *contextInfoPtr, BYTE *buffer, int noBytes ) { RIPEMD160_CTX *ripemd160Info = ( RIPEMD160_CTX * ) contextInfoPtr->ctxHash->hashInfo; /* If the hash state was reset to allow another round of hashing, reinitialise things */ if( !( contextInfoPtr->flags & CONTEXT_HASH_INITED ) ) RIPEMD160_Init( ripemd160Info ); if( noBytes > 0 ) RIPEMD160_Update( ripemd160Info, buffer, noBytes ); else RIPEMD160_Final( contextInfoPtr->ctxHash->hash, ripemd160Info ); return( CRYPT_OK ); } /* Internal API: Hash a single block of memory without the overhead of creating an encryption context */ void ripemd160HashBuffer( HASHINFO hashInfo, BYTE *outBuffer, const int outBufMaxLength, const BYTE *inBuffer, const int inLength, const HASH_STATE hashState ) { RIPEMD160_CTX *ripemd160Info = ( RIPEMD160_CTX * ) hashInfo; assert( ( hashState == HASH_ALL && hashInfo == NULL ) || \ ( hashState != HASH_ALL && \ isWritePtr( hashInfo, sizeof( HASHINFO ) ) ) ); assert( ( ( hashState != HASH_END && hashState != HASH_ALL ) && \ outBuffer == NULL && outBufMaxLength == 0 ) || \ ( ( hashState == HASH_END || hashState == HASH_ALL ) && \ isWritePtr( outBuffer, outBufMaxLength ) && \ outBufMaxLength >= 20 ) ); assert( inBuffer == NULL || isReadPtr( inBuffer, inLength ) ); switch( hashState ) { case HASH_START: RIPEMD160_Init( ripemd160Info ); /* Drop through */ case HASH_CONTINUE: RIPEMD160_Update( ripemd160Info, ( BYTE * ) inBuffer, inLength ); break; case HASH_END: if( inBuffer != NULL ) RIPEMD160_Update( ripemd160Info, ( BYTE * ) inBuffer, inLength ); RIPEMD160_Final( outBuffer, ripemd160Info ); break; case HASH_ALL: { RIPEMD160_CTX ripemd160InfoBuffer; RIPEMD160_Init( &ripemd160InfoBuffer ); RIPEMD160_Update( &ripemd160InfoBuffer, ( BYTE * ) inBuffer, inLength ); RIPEMD160_Final( outBuffer, &ripemd160InfoBuffer ); zeroise( &ripemd160InfoBuffer, sizeof( RIPEMD160_CTX ) ); break; } default: assert( NOTREACHED ); } } /**************************************************************************** * * * Capability Access Routines * * * ****************************************************************************/ static const CAPABILITY_INFO FAR_BSS capabilityInfo = { CRYPT_ALGO_RIPEMD160, bitsToBytes( 160 ), "RIPEMD-160", bitsToBytes( 0 ), bitsToBytes( 0 ), bitsToBytes( 0 ), selfTest, getInfo, NULL, NULL, NULL, NULL, hash, hash }; const CAPABILITY_INFO *getRipemd160Capability( void ) { return( &capabilityInfo ); } #endif /* USE_RIPEMD160 */