1*817466cbSJens Wiklander /* 2*817466cbSJens Wiklander * RIPE MD-160 implementation 3*817466cbSJens Wiklander * 4*817466cbSJens Wiklander * Copyright (C) 2006-2015, ARM Limited, All Rights Reserved 5*817466cbSJens Wiklander * SPDX-License-Identifier: Apache-2.0 6*817466cbSJens Wiklander * 7*817466cbSJens Wiklander * Licensed under the Apache License, Version 2.0 (the "License"); you may 8*817466cbSJens Wiklander * not use this file except in compliance with the License. 9*817466cbSJens Wiklander * You may obtain a copy of the License at 10*817466cbSJens Wiklander * 11*817466cbSJens Wiklander * http://www.apache.org/licenses/LICENSE-2.0 12*817466cbSJens Wiklander * 13*817466cbSJens Wiklander * Unless required by applicable law or agreed to in writing, software 14*817466cbSJens Wiklander * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT 15*817466cbSJens Wiklander * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 16*817466cbSJens Wiklander * See the License for the specific language governing permissions and 17*817466cbSJens Wiklander * limitations under the License. 18*817466cbSJens Wiklander * 19*817466cbSJens Wiklander * This file is part of mbed TLS (https://tls.mbed.org) 20*817466cbSJens Wiklander */ 21*817466cbSJens Wiklander 22*817466cbSJens Wiklander /* 23*817466cbSJens Wiklander * The RIPEMD-160 algorithm was designed by RIPE in 1996 24*817466cbSJens Wiklander * http://homes.esat.kuleuven.be/~bosselae/mbedtls_ripemd160.html 25*817466cbSJens Wiklander * http://ehash.iaik.tugraz.at/wiki/RIPEMD-160 26*817466cbSJens Wiklander */ 27*817466cbSJens Wiklander 28*817466cbSJens Wiklander #if !defined(MBEDTLS_CONFIG_FILE) 29*817466cbSJens Wiklander #include "mbedtls/config.h" 30*817466cbSJens Wiklander #else 31*817466cbSJens Wiklander #include MBEDTLS_CONFIG_FILE 32*817466cbSJens Wiklander #endif 33*817466cbSJens Wiklander 34*817466cbSJens Wiklander #if defined(MBEDTLS_RIPEMD160_C) 35*817466cbSJens Wiklander 36*817466cbSJens Wiklander #include "mbedtls/ripemd160.h" 37*817466cbSJens Wiklander 38*817466cbSJens Wiklander #include <string.h> 39*817466cbSJens Wiklander 40*817466cbSJens Wiklander #if defined(MBEDTLS_SELF_TEST) 41*817466cbSJens Wiklander #if defined(MBEDTLS_PLATFORM_C) 42*817466cbSJens Wiklander #include "mbedtls/platform.h" 43*817466cbSJens Wiklander #else 44*817466cbSJens Wiklander #include <stdio.h> 45*817466cbSJens Wiklander #define mbedtls_printf printf 46*817466cbSJens Wiklander #endif /* MBEDTLS_PLATFORM_C */ 47*817466cbSJens Wiklander #endif /* MBEDTLS_SELF_TEST */ 48*817466cbSJens Wiklander 49*817466cbSJens Wiklander /* 50*817466cbSJens Wiklander * 32-bit integer manipulation macros (little endian) 51*817466cbSJens Wiklander */ 52*817466cbSJens Wiklander #ifndef GET_UINT32_LE 53*817466cbSJens Wiklander #define GET_UINT32_LE(n,b,i) \ 54*817466cbSJens Wiklander { \ 55*817466cbSJens Wiklander (n) = ( (uint32_t) (b)[(i) ] ) \ 56*817466cbSJens Wiklander | ( (uint32_t) (b)[(i) + 1] << 8 ) \ 57*817466cbSJens Wiklander | ( (uint32_t) (b)[(i) + 2] << 16 ) \ 58*817466cbSJens Wiklander | ( (uint32_t) (b)[(i) + 3] << 24 ); \ 59*817466cbSJens Wiklander } 60*817466cbSJens Wiklander #endif 61*817466cbSJens Wiklander 62*817466cbSJens Wiklander #ifndef PUT_UINT32_LE 63*817466cbSJens Wiklander #define PUT_UINT32_LE(n,b,i) \ 64*817466cbSJens Wiklander { \ 65*817466cbSJens Wiklander (b)[(i) ] = (unsigned char) ( ( (n) ) & 0xFF ); \ 66*817466cbSJens Wiklander (b)[(i) + 1] = (unsigned char) ( ( (n) >> 8 ) & 0xFF ); \ 67*817466cbSJens Wiklander (b)[(i) + 2] = (unsigned char) ( ( (n) >> 16 ) & 0xFF ); \ 68*817466cbSJens Wiklander (b)[(i) + 3] = (unsigned char) ( ( (n) >> 24 ) & 0xFF ); \ 69*817466cbSJens Wiklander } 70*817466cbSJens Wiklander #endif 71*817466cbSJens Wiklander 72*817466cbSJens Wiklander /* Implementation that should never be optimized out by the compiler */ 73*817466cbSJens Wiklander static void mbedtls_zeroize( void *v, size_t n ) { 74*817466cbSJens Wiklander volatile unsigned char *p = v; while( n-- ) *p++ = 0; 75*817466cbSJens Wiklander } 76*817466cbSJens Wiklander 77*817466cbSJens Wiklander void mbedtls_ripemd160_init( mbedtls_ripemd160_context *ctx ) 78*817466cbSJens Wiklander { 79*817466cbSJens Wiklander memset( ctx, 0, sizeof( mbedtls_ripemd160_context ) ); 80*817466cbSJens Wiklander } 81*817466cbSJens Wiklander 82*817466cbSJens Wiklander void mbedtls_ripemd160_free( mbedtls_ripemd160_context *ctx ) 83*817466cbSJens Wiklander { 84*817466cbSJens Wiklander if( ctx == NULL ) 85*817466cbSJens Wiklander return; 86*817466cbSJens Wiklander 87*817466cbSJens Wiklander mbedtls_zeroize( ctx, sizeof( mbedtls_ripemd160_context ) ); 88*817466cbSJens Wiklander } 89*817466cbSJens Wiklander 90*817466cbSJens Wiklander void mbedtls_ripemd160_clone( mbedtls_ripemd160_context *dst, 91*817466cbSJens Wiklander const mbedtls_ripemd160_context *src ) 92*817466cbSJens Wiklander { 93*817466cbSJens Wiklander *dst = *src; 94*817466cbSJens Wiklander } 95*817466cbSJens Wiklander 96*817466cbSJens Wiklander /* 97*817466cbSJens Wiklander * RIPEMD-160 context setup 98*817466cbSJens Wiklander */ 99*817466cbSJens Wiklander void mbedtls_ripemd160_starts( mbedtls_ripemd160_context *ctx ) 100*817466cbSJens Wiklander { 101*817466cbSJens Wiklander ctx->total[0] = 0; 102*817466cbSJens Wiklander ctx->total[1] = 0; 103*817466cbSJens Wiklander 104*817466cbSJens Wiklander ctx->state[0] = 0x67452301; 105*817466cbSJens Wiklander ctx->state[1] = 0xEFCDAB89; 106*817466cbSJens Wiklander ctx->state[2] = 0x98BADCFE; 107*817466cbSJens Wiklander ctx->state[3] = 0x10325476; 108*817466cbSJens Wiklander ctx->state[4] = 0xC3D2E1F0; 109*817466cbSJens Wiklander } 110*817466cbSJens Wiklander 111*817466cbSJens Wiklander #if !defined(MBEDTLS_RIPEMD160_PROCESS_ALT) 112*817466cbSJens Wiklander /* 113*817466cbSJens Wiklander * Process one block 114*817466cbSJens Wiklander */ 115*817466cbSJens Wiklander void mbedtls_ripemd160_process( mbedtls_ripemd160_context *ctx, const unsigned char data[64] ) 116*817466cbSJens Wiklander { 117*817466cbSJens Wiklander uint32_t A, B, C, D, E, Ap, Bp, Cp, Dp, Ep, X[16]; 118*817466cbSJens Wiklander 119*817466cbSJens Wiklander GET_UINT32_LE( X[ 0], data, 0 ); 120*817466cbSJens Wiklander GET_UINT32_LE( X[ 1], data, 4 ); 121*817466cbSJens Wiklander GET_UINT32_LE( X[ 2], data, 8 ); 122*817466cbSJens Wiklander GET_UINT32_LE( X[ 3], data, 12 ); 123*817466cbSJens Wiklander GET_UINT32_LE( X[ 4], data, 16 ); 124*817466cbSJens Wiklander GET_UINT32_LE( X[ 5], data, 20 ); 125*817466cbSJens Wiklander GET_UINT32_LE( X[ 6], data, 24 ); 126*817466cbSJens Wiklander GET_UINT32_LE( X[ 7], data, 28 ); 127*817466cbSJens Wiklander GET_UINT32_LE( X[ 8], data, 32 ); 128*817466cbSJens Wiklander GET_UINT32_LE( X[ 9], data, 36 ); 129*817466cbSJens Wiklander GET_UINT32_LE( X[10], data, 40 ); 130*817466cbSJens Wiklander GET_UINT32_LE( X[11], data, 44 ); 131*817466cbSJens Wiklander GET_UINT32_LE( X[12], data, 48 ); 132*817466cbSJens Wiklander GET_UINT32_LE( X[13], data, 52 ); 133*817466cbSJens Wiklander GET_UINT32_LE( X[14], data, 56 ); 134*817466cbSJens Wiklander GET_UINT32_LE( X[15], data, 60 ); 135*817466cbSJens Wiklander 136*817466cbSJens Wiklander A = Ap = ctx->state[0]; 137*817466cbSJens Wiklander B = Bp = ctx->state[1]; 138*817466cbSJens Wiklander C = Cp = ctx->state[2]; 139*817466cbSJens Wiklander D = Dp = ctx->state[3]; 140*817466cbSJens Wiklander E = Ep = ctx->state[4]; 141*817466cbSJens Wiklander 142*817466cbSJens Wiklander #define F1( x, y, z ) ( x ^ y ^ z ) 143*817466cbSJens Wiklander #define F2( x, y, z ) ( ( x & y ) | ( ~x & z ) ) 144*817466cbSJens Wiklander #define F3( x, y, z ) ( ( x | ~y ) ^ z ) 145*817466cbSJens Wiklander #define F4( x, y, z ) ( ( x & z ) | ( y & ~z ) ) 146*817466cbSJens Wiklander #define F5( x, y, z ) ( x ^ ( y | ~z ) ) 147*817466cbSJens Wiklander 148*817466cbSJens Wiklander #define S( x, n ) ( ( x << n ) | ( x >> (32 - n) ) ) 149*817466cbSJens Wiklander 150*817466cbSJens Wiklander #define P( a, b, c, d, e, r, s, f, k ) \ 151*817466cbSJens Wiklander a += f( b, c, d ) + X[r] + k; \ 152*817466cbSJens Wiklander a = S( a, s ) + e; \ 153*817466cbSJens Wiklander c = S( c, 10 ); 154*817466cbSJens Wiklander 155*817466cbSJens Wiklander #define P2( a, b, c, d, e, r, s, rp, sp ) \ 156*817466cbSJens Wiklander P( a, b, c, d, e, r, s, F, K ); \ 157*817466cbSJens Wiklander P( a ## p, b ## p, c ## p, d ## p, e ## p, rp, sp, Fp, Kp ); 158*817466cbSJens Wiklander 159*817466cbSJens Wiklander #define F F1 160*817466cbSJens Wiklander #define K 0x00000000 161*817466cbSJens Wiklander #define Fp F5 162*817466cbSJens Wiklander #define Kp 0x50A28BE6 163*817466cbSJens Wiklander P2( A, B, C, D, E, 0, 11, 5, 8 ); 164*817466cbSJens Wiklander P2( E, A, B, C, D, 1, 14, 14, 9 ); 165*817466cbSJens Wiklander P2( D, E, A, B, C, 2, 15, 7, 9 ); 166*817466cbSJens Wiklander P2( C, D, E, A, B, 3, 12, 0, 11 ); 167*817466cbSJens Wiklander P2( B, C, D, E, A, 4, 5, 9, 13 ); 168*817466cbSJens Wiklander P2( A, B, C, D, E, 5, 8, 2, 15 ); 169*817466cbSJens Wiklander P2( E, A, B, C, D, 6, 7, 11, 15 ); 170*817466cbSJens Wiklander P2( D, E, A, B, C, 7, 9, 4, 5 ); 171*817466cbSJens Wiklander P2( C, D, E, A, B, 8, 11, 13, 7 ); 172*817466cbSJens Wiklander P2( B, C, D, E, A, 9, 13, 6, 7 ); 173*817466cbSJens Wiklander P2( A, B, C, D, E, 10, 14, 15, 8 ); 174*817466cbSJens Wiklander P2( E, A, B, C, D, 11, 15, 8, 11 ); 175*817466cbSJens Wiklander P2( D, E, A, B, C, 12, 6, 1, 14 ); 176*817466cbSJens Wiklander P2( C, D, E, A, B, 13, 7, 10, 14 ); 177*817466cbSJens Wiklander P2( B, C, D, E, A, 14, 9, 3, 12 ); 178*817466cbSJens Wiklander P2( A, B, C, D, E, 15, 8, 12, 6 ); 179*817466cbSJens Wiklander #undef F 180*817466cbSJens Wiklander #undef K 181*817466cbSJens Wiklander #undef Fp 182*817466cbSJens Wiklander #undef Kp 183*817466cbSJens Wiklander 184*817466cbSJens Wiklander #define F F2 185*817466cbSJens Wiklander #define K 0x5A827999 186*817466cbSJens Wiklander #define Fp F4 187*817466cbSJens Wiklander #define Kp 0x5C4DD124 188*817466cbSJens Wiklander P2( E, A, B, C, D, 7, 7, 6, 9 ); 189*817466cbSJens Wiklander P2( D, E, A, B, C, 4, 6, 11, 13 ); 190*817466cbSJens Wiklander P2( C, D, E, A, B, 13, 8, 3, 15 ); 191*817466cbSJens Wiklander P2( B, C, D, E, A, 1, 13, 7, 7 ); 192*817466cbSJens Wiklander P2( A, B, C, D, E, 10, 11, 0, 12 ); 193*817466cbSJens Wiklander P2( E, A, B, C, D, 6, 9, 13, 8 ); 194*817466cbSJens Wiklander P2( D, E, A, B, C, 15, 7, 5, 9 ); 195*817466cbSJens Wiklander P2( C, D, E, A, B, 3, 15, 10, 11 ); 196*817466cbSJens Wiklander P2( B, C, D, E, A, 12, 7, 14, 7 ); 197*817466cbSJens Wiklander P2( A, B, C, D, E, 0, 12, 15, 7 ); 198*817466cbSJens Wiklander P2( E, A, B, C, D, 9, 15, 8, 12 ); 199*817466cbSJens Wiklander P2( D, E, A, B, C, 5, 9, 12, 7 ); 200*817466cbSJens Wiklander P2( C, D, E, A, B, 2, 11, 4, 6 ); 201*817466cbSJens Wiklander P2( B, C, D, E, A, 14, 7, 9, 15 ); 202*817466cbSJens Wiklander P2( A, B, C, D, E, 11, 13, 1, 13 ); 203*817466cbSJens Wiklander P2( E, A, B, C, D, 8, 12, 2, 11 ); 204*817466cbSJens Wiklander #undef F 205*817466cbSJens Wiklander #undef K 206*817466cbSJens Wiklander #undef Fp 207*817466cbSJens Wiklander #undef Kp 208*817466cbSJens Wiklander 209*817466cbSJens Wiklander #define F F3 210*817466cbSJens Wiklander #define K 0x6ED9EBA1 211*817466cbSJens Wiklander #define Fp F3 212*817466cbSJens Wiklander #define Kp 0x6D703EF3 213*817466cbSJens Wiklander P2( D, E, A, B, C, 3, 11, 15, 9 ); 214*817466cbSJens Wiklander P2( C, D, E, A, B, 10, 13, 5, 7 ); 215*817466cbSJens Wiklander P2( B, C, D, E, A, 14, 6, 1, 15 ); 216*817466cbSJens Wiklander P2( A, B, C, D, E, 4, 7, 3, 11 ); 217*817466cbSJens Wiklander P2( E, A, B, C, D, 9, 14, 7, 8 ); 218*817466cbSJens Wiklander P2( D, E, A, B, C, 15, 9, 14, 6 ); 219*817466cbSJens Wiklander P2( C, D, E, A, B, 8, 13, 6, 6 ); 220*817466cbSJens Wiklander P2( B, C, D, E, A, 1, 15, 9, 14 ); 221*817466cbSJens Wiklander P2( A, B, C, D, E, 2, 14, 11, 12 ); 222*817466cbSJens Wiklander P2( E, A, B, C, D, 7, 8, 8, 13 ); 223*817466cbSJens Wiklander P2( D, E, A, B, C, 0, 13, 12, 5 ); 224*817466cbSJens Wiklander P2( C, D, E, A, B, 6, 6, 2, 14 ); 225*817466cbSJens Wiklander P2( B, C, D, E, A, 13, 5, 10, 13 ); 226*817466cbSJens Wiklander P2( A, B, C, D, E, 11, 12, 0, 13 ); 227*817466cbSJens Wiklander P2( E, A, B, C, D, 5, 7, 4, 7 ); 228*817466cbSJens Wiklander P2( D, E, A, B, C, 12, 5, 13, 5 ); 229*817466cbSJens Wiklander #undef F 230*817466cbSJens Wiklander #undef K 231*817466cbSJens Wiklander #undef Fp 232*817466cbSJens Wiklander #undef Kp 233*817466cbSJens Wiklander 234*817466cbSJens Wiklander #define F F4 235*817466cbSJens Wiklander #define K 0x8F1BBCDC 236*817466cbSJens Wiklander #define Fp F2 237*817466cbSJens Wiklander #define Kp 0x7A6D76E9 238*817466cbSJens Wiklander P2( C, D, E, A, B, 1, 11, 8, 15 ); 239*817466cbSJens Wiklander P2( B, C, D, E, A, 9, 12, 6, 5 ); 240*817466cbSJens Wiklander P2( A, B, C, D, E, 11, 14, 4, 8 ); 241*817466cbSJens Wiklander P2( E, A, B, C, D, 10, 15, 1, 11 ); 242*817466cbSJens Wiklander P2( D, E, A, B, C, 0, 14, 3, 14 ); 243*817466cbSJens Wiklander P2( C, D, E, A, B, 8, 15, 11, 14 ); 244*817466cbSJens Wiklander P2( B, C, D, E, A, 12, 9, 15, 6 ); 245*817466cbSJens Wiklander P2( A, B, C, D, E, 4, 8, 0, 14 ); 246*817466cbSJens Wiklander P2( E, A, B, C, D, 13, 9, 5, 6 ); 247*817466cbSJens Wiklander P2( D, E, A, B, C, 3, 14, 12, 9 ); 248*817466cbSJens Wiklander P2( C, D, E, A, B, 7, 5, 2, 12 ); 249*817466cbSJens Wiklander P2( B, C, D, E, A, 15, 6, 13, 9 ); 250*817466cbSJens Wiklander P2( A, B, C, D, E, 14, 8, 9, 12 ); 251*817466cbSJens Wiklander P2( E, A, B, C, D, 5, 6, 7, 5 ); 252*817466cbSJens Wiklander P2( D, E, A, B, C, 6, 5, 10, 15 ); 253*817466cbSJens Wiklander P2( C, D, E, A, B, 2, 12, 14, 8 ); 254*817466cbSJens Wiklander #undef F 255*817466cbSJens Wiklander #undef K 256*817466cbSJens Wiklander #undef Fp 257*817466cbSJens Wiklander #undef Kp 258*817466cbSJens Wiklander 259*817466cbSJens Wiklander #define F F5 260*817466cbSJens Wiklander #define K 0xA953FD4E 261*817466cbSJens Wiklander #define Fp F1 262*817466cbSJens Wiklander #define Kp 0x00000000 263*817466cbSJens Wiklander P2( B, C, D, E, A, 4, 9, 12, 8 ); 264*817466cbSJens Wiklander P2( A, B, C, D, E, 0, 15, 15, 5 ); 265*817466cbSJens Wiklander P2( E, A, B, C, D, 5, 5, 10, 12 ); 266*817466cbSJens Wiklander P2( D, E, A, B, C, 9, 11, 4, 9 ); 267*817466cbSJens Wiklander P2( C, D, E, A, B, 7, 6, 1, 12 ); 268*817466cbSJens Wiklander P2( B, C, D, E, A, 12, 8, 5, 5 ); 269*817466cbSJens Wiklander P2( A, B, C, D, E, 2, 13, 8, 14 ); 270*817466cbSJens Wiklander P2( E, A, B, C, D, 10, 12, 7, 6 ); 271*817466cbSJens Wiklander P2( D, E, A, B, C, 14, 5, 6, 8 ); 272*817466cbSJens Wiklander P2( C, D, E, A, B, 1, 12, 2, 13 ); 273*817466cbSJens Wiklander P2( B, C, D, E, A, 3, 13, 13, 6 ); 274*817466cbSJens Wiklander P2( A, B, C, D, E, 8, 14, 14, 5 ); 275*817466cbSJens Wiklander P2( E, A, B, C, D, 11, 11, 0, 15 ); 276*817466cbSJens Wiklander P2( D, E, A, B, C, 6, 8, 3, 13 ); 277*817466cbSJens Wiklander P2( C, D, E, A, B, 15, 5, 9, 11 ); 278*817466cbSJens Wiklander P2( B, C, D, E, A, 13, 6, 11, 11 ); 279*817466cbSJens Wiklander #undef F 280*817466cbSJens Wiklander #undef K 281*817466cbSJens Wiklander #undef Fp 282*817466cbSJens Wiklander #undef Kp 283*817466cbSJens Wiklander 284*817466cbSJens Wiklander C = ctx->state[1] + C + Dp; 285*817466cbSJens Wiklander ctx->state[1] = ctx->state[2] + D + Ep; 286*817466cbSJens Wiklander ctx->state[2] = ctx->state[3] + E + Ap; 287*817466cbSJens Wiklander ctx->state[3] = ctx->state[4] + A + Bp; 288*817466cbSJens Wiklander ctx->state[4] = ctx->state[0] + B + Cp; 289*817466cbSJens Wiklander ctx->state[0] = C; 290*817466cbSJens Wiklander } 291*817466cbSJens Wiklander #endif /* !MBEDTLS_RIPEMD160_PROCESS_ALT */ 292*817466cbSJens Wiklander 293*817466cbSJens Wiklander /* 294*817466cbSJens Wiklander * RIPEMD-160 process buffer 295*817466cbSJens Wiklander */ 296*817466cbSJens Wiklander void mbedtls_ripemd160_update( mbedtls_ripemd160_context *ctx, 297*817466cbSJens Wiklander const unsigned char *input, size_t ilen ) 298*817466cbSJens Wiklander { 299*817466cbSJens Wiklander size_t fill; 300*817466cbSJens Wiklander uint32_t left; 301*817466cbSJens Wiklander 302*817466cbSJens Wiklander if( ilen == 0 ) 303*817466cbSJens Wiklander return; 304*817466cbSJens Wiklander 305*817466cbSJens Wiklander left = ctx->total[0] & 0x3F; 306*817466cbSJens Wiklander fill = 64 - left; 307*817466cbSJens Wiklander 308*817466cbSJens Wiklander ctx->total[0] += (uint32_t) ilen; 309*817466cbSJens Wiklander ctx->total[0] &= 0xFFFFFFFF; 310*817466cbSJens Wiklander 311*817466cbSJens Wiklander if( ctx->total[0] < (uint32_t) ilen ) 312*817466cbSJens Wiklander ctx->total[1]++; 313*817466cbSJens Wiklander 314*817466cbSJens Wiklander if( left && ilen >= fill ) 315*817466cbSJens Wiklander { 316*817466cbSJens Wiklander memcpy( (void *) (ctx->buffer + left), input, fill ); 317*817466cbSJens Wiklander mbedtls_ripemd160_process( ctx, ctx->buffer ); 318*817466cbSJens Wiklander input += fill; 319*817466cbSJens Wiklander ilen -= fill; 320*817466cbSJens Wiklander left = 0; 321*817466cbSJens Wiklander } 322*817466cbSJens Wiklander 323*817466cbSJens Wiklander while( ilen >= 64 ) 324*817466cbSJens Wiklander { 325*817466cbSJens Wiklander mbedtls_ripemd160_process( ctx, input ); 326*817466cbSJens Wiklander input += 64; 327*817466cbSJens Wiklander ilen -= 64; 328*817466cbSJens Wiklander } 329*817466cbSJens Wiklander 330*817466cbSJens Wiklander if( ilen > 0 ) 331*817466cbSJens Wiklander { 332*817466cbSJens Wiklander memcpy( (void *) (ctx->buffer + left), input, ilen ); 333*817466cbSJens Wiklander } 334*817466cbSJens Wiklander } 335*817466cbSJens Wiklander 336*817466cbSJens Wiklander static const unsigned char ripemd160_padding[64] = 337*817466cbSJens Wiklander { 338*817466cbSJens Wiklander 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 339*817466cbSJens Wiklander 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 340*817466cbSJens Wiklander 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 341*817466cbSJens Wiklander 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 342*817466cbSJens Wiklander }; 343*817466cbSJens Wiklander 344*817466cbSJens Wiklander /* 345*817466cbSJens Wiklander * RIPEMD-160 final digest 346*817466cbSJens Wiklander */ 347*817466cbSJens Wiklander void mbedtls_ripemd160_finish( mbedtls_ripemd160_context *ctx, unsigned char output[20] ) 348*817466cbSJens Wiklander { 349*817466cbSJens Wiklander uint32_t last, padn; 350*817466cbSJens Wiklander uint32_t high, low; 351*817466cbSJens Wiklander unsigned char msglen[8]; 352*817466cbSJens Wiklander 353*817466cbSJens Wiklander high = ( ctx->total[0] >> 29 ) 354*817466cbSJens Wiklander | ( ctx->total[1] << 3 ); 355*817466cbSJens Wiklander low = ( ctx->total[0] << 3 ); 356*817466cbSJens Wiklander 357*817466cbSJens Wiklander PUT_UINT32_LE( low, msglen, 0 ); 358*817466cbSJens Wiklander PUT_UINT32_LE( high, msglen, 4 ); 359*817466cbSJens Wiklander 360*817466cbSJens Wiklander last = ctx->total[0] & 0x3F; 361*817466cbSJens Wiklander padn = ( last < 56 ) ? ( 56 - last ) : ( 120 - last ); 362*817466cbSJens Wiklander 363*817466cbSJens Wiklander mbedtls_ripemd160_update( ctx, ripemd160_padding, padn ); 364*817466cbSJens Wiklander mbedtls_ripemd160_update( ctx, msglen, 8 ); 365*817466cbSJens Wiklander 366*817466cbSJens Wiklander PUT_UINT32_LE( ctx->state[0], output, 0 ); 367*817466cbSJens Wiklander PUT_UINT32_LE( ctx->state[1], output, 4 ); 368*817466cbSJens Wiklander PUT_UINT32_LE( ctx->state[2], output, 8 ); 369*817466cbSJens Wiklander PUT_UINT32_LE( ctx->state[3], output, 12 ); 370*817466cbSJens Wiklander PUT_UINT32_LE( ctx->state[4], output, 16 ); 371*817466cbSJens Wiklander } 372*817466cbSJens Wiklander 373*817466cbSJens Wiklander /* 374*817466cbSJens Wiklander * output = RIPEMD-160( input buffer ) 375*817466cbSJens Wiklander */ 376*817466cbSJens Wiklander void mbedtls_ripemd160( const unsigned char *input, size_t ilen, 377*817466cbSJens Wiklander unsigned char output[20] ) 378*817466cbSJens Wiklander { 379*817466cbSJens Wiklander mbedtls_ripemd160_context ctx; 380*817466cbSJens Wiklander 381*817466cbSJens Wiklander mbedtls_ripemd160_init( &ctx ); 382*817466cbSJens Wiklander mbedtls_ripemd160_starts( &ctx ); 383*817466cbSJens Wiklander mbedtls_ripemd160_update( &ctx, input, ilen ); 384*817466cbSJens Wiklander mbedtls_ripemd160_finish( &ctx, output ); 385*817466cbSJens Wiklander mbedtls_ripemd160_free( &ctx ); 386*817466cbSJens Wiklander } 387*817466cbSJens Wiklander 388*817466cbSJens Wiklander #if defined(MBEDTLS_SELF_TEST) 389*817466cbSJens Wiklander /* 390*817466cbSJens Wiklander * Test vectors from the RIPEMD-160 paper and 391*817466cbSJens Wiklander * http://homes.esat.kuleuven.be/~bosselae/mbedtls_ripemd160.html#HMAC 392*817466cbSJens Wiklander */ 393*817466cbSJens Wiklander #define TESTS 8 394*817466cbSJens Wiklander #define KEYS 2 395*817466cbSJens Wiklander static const char *ripemd160_test_input[TESTS] = 396*817466cbSJens Wiklander { 397*817466cbSJens Wiklander "", 398*817466cbSJens Wiklander "a", 399*817466cbSJens Wiklander "abc", 400*817466cbSJens Wiklander "message digest", 401*817466cbSJens Wiklander "abcdefghijklmnopqrstuvwxyz", 402*817466cbSJens Wiklander "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq", 403*817466cbSJens Wiklander "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789", 404*817466cbSJens Wiklander "1234567890123456789012345678901234567890" 405*817466cbSJens Wiklander "1234567890123456789012345678901234567890", 406*817466cbSJens Wiklander }; 407*817466cbSJens Wiklander 408*817466cbSJens Wiklander static const unsigned char ripemd160_test_md[TESTS][20] = 409*817466cbSJens Wiklander { 410*817466cbSJens Wiklander { 0x9c, 0x11, 0x85, 0xa5, 0xc5, 0xe9, 0xfc, 0x54, 0x61, 0x28, 411*817466cbSJens Wiklander 0x08, 0x97, 0x7e, 0xe8, 0xf5, 0x48, 0xb2, 0x25, 0x8d, 0x31 }, 412*817466cbSJens Wiklander { 0x0b, 0xdc, 0x9d, 0x2d, 0x25, 0x6b, 0x3e, 0xe9, 0xda, 0xae, 413*817466cbSJens Wiklander 0x34, 0x7b, 0xe6, 0xf4, 0xdc, 0x83, 0x5a, 0x46, 0x7f, 0xfe }, 414*817466cbSJens Wiklander { 0x8e, 0xb2, 0x08, 0xf7, 0xe0, 0x5d, 0x98, 0x7a, 0x9b, 0x04, 415*817466cbSJens Wiklander 0x4a, 0x8e, 0x98, 0xc6, 0xb0, 0x87, 0xf1, 0x5a, 0x0b, 0xfc }, 416*817466cbSJens Wiklander { 0x5d, 0x06, 0x89, 0xef, 0x49, 0xd2, 0xfa, 0xe5, 0x72, 0xb8, 417*817466cbSJens Wiklander 0x81, 0xb1, 0x23, 0xa8, 0x5f, 0xfa, 0x21, 0x59, 0x5f, 0x36 }, 418*817466cbSJens Wiklander { 0xf7, 0x1c, 0x27, 0x10, 0x9c, 0x69, 0x2c, 0x1b, 0x56, 0xbb, 419*817466cbSJens Wiklander 0xdc, 0xeb, 0x5b, 0x9d, 0x28, 0x65, 0xb3, 0x70, 0x8d, 0xbc }, 420*817466cbSJens Wiklander { 0x12, 0xa0, 0x53, 0x38, 0x4a, 0x9c, 0x0c, 0x88, 0xe4, 0x05, 421*817466cbSJens Wiklander 0xa0, 0x6c, 0x27, 0xdc, 0xf4, 0x9a, 0xda, 0x62, 0xeb, 0x2b }, 422*817466cbSJens Wiklander { 0xb0, 0xe2, 0x0b, 0x6e, 0x31, 0x16, 0x64, 0x02, 0x86, 0xed, 423*817466cbSJens Wiklander 0x3a, 0x87, 0xa5, 0x71, 0x30, 0x79, 0xb2, 0x1f, 0x51, 0x89 }, 424*817466cbSJens Wiklander { 0x9b, 0x75, 0x2e, 0x45, 0x57, 0x3d, 0x4b, 0x39, 0xf4, 0xdb, 425*817466cbSJens Wiklander 0xd3, 0x32, 0x3c, 0xab, 0x82, 0xbf, 0x63, 0x32, 0x6b, 0xfb }, 426*817466cbSJens Wiklander }; 427*817466cbSJens Wiklander 428*817466cbSJens Wiklander /* 429*817466cbSJens Wiklander * Checkup routine 430*817466cbSJens Wiklander */ 431*817466cbSJens Wiklander int mbedtls_ripemd160_self_test( int verbose ) 432*817466cbSJens Wiklander { 433*817466cbSJens Wiklander int i; 434*817466cbSJens Wiklander unsigned char output[20]; 435*817466cbSJens Wiklander 436*817466cbSJens Wiklander memset( output, 0, sizeof output ); 437*817466cbSJens Wiklander 438*817466cbSJens Wiklander for( i = 0; i < TESTS; i++ ) 439*817466cbSJens Wiklander { 440*817466cbSJens Wiklander if( verbose != 0 ) 441*817466cbSJens Wiklander mbedtls_printf( " RIPEMD-160 test #%d: ", i + 1 ); 442*817466cbSJens Wiklander 443*817466cbSJens Wiklander mbedtls_ripemd160( (const unsigned char *) ripemd160_test_input[i], 444*817466cbSJens Wiklander strlen( ripemd160_test_input[i] ), 445*817466cbSJens Wiklander output ); 446*817466cbSJens Wiklander 447*817466cbSJens Wiklander if( memcmp( output, ripemd160_test_md[i], 20 ) != 0 ) 448*817466cbSJens Wiklander { 449*817466cbSJens Wiklander if( verbose != 0 ) 450*817466cbSJens Wiklander mbedtls_printf( "failed\n" ); 451*817466cbSJens Wiklander 452*817466cbSJens Wiklander return( 1 ); 453*817466cbSJens Wiklander } 454*817466cbSJens Wiklander 455*817466cbSJens Wiklander if( verbose != 0 ) 456*817466cbSJens Wiklander mbedtls_printf( "passed\n" ); 457*817466cbSJens Wiklander } 458*817466cbSJens Wiklander 459*817466cbSJens Wiklander if( verbose != 0 ) 460*817466cbSJens Wiklander mbedtls_printf( "\n" ); 461*817466cbSJens Wiklander 462*817466cbSJens Wiklander return( 0 ); 463*817466cbSJens Wiklander } 464*817466cbSJens Wiklander 465*817466cbSJens Wiklander #endif /* MBEDTLS_SELF_TEST */ 466*817466cbSJens Wiklander 467*817466cbSJens Wiklander #endif /* MBEDTLS_RIPEMD160_C */ 468