xref: /optee_os/lib/libmbedtls/mbedtls/library/md.c (revision 817466cb476de705a8e3dabe1ef165fe27a18c2f)
1*817466cbSJens Wiklander /**
2*817466cbSJens Wiklander  * \file mbedtls_md.c
3*817466cbSJens Wiklander  *
4*817466cbSJens Wiklander  * \brief Generic message digest wrapper for mbed TLS
5*817466cbSJens Wiklander  *
6*817466cbSJens Wiklander  * \author Adriaan de Jong <dejong@fox-it.com>
7*817466cbSJens Wiklander  *
8*817466cbSJens Wiklander  *  Copyright (C) 2006-2015, ARM Limited, All Rights Reserved
9*817466cbSJens Wiklander  *  SPDX-License-Identifier: Apache-2.0
10*817466cbSJens Wiklander  *
11*817466cbSJens Wiklander  *  Licensed under the Apache License, Version 2.0 (the "License"); you may
12*817466cbSJens Wiklander  *  not use this file except in compliance with the License.
13*817466cbSJens Wiklander  *  You may obtain a copy of the License at
14*817466cbSJens Wiklander  *
15*817466cbSJens Wiklander  *  http://www.apache.org/licenses/LICENSE-2.0
16*817466cbSJens Wiklander  *
17*817466cbSJens Wiklander  *  Unless required by applicable law or agreed to in writing, software
18*817466cbSJens Wiklander  *  distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
19*817466cbSJens Wiklander  *  WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20*817466cbSJens Wiklander  *  See the License for the specific language governing permissions and
21*817466cbSJens Wiklander  *  limitations under the License.
22*817466cbSJens Wiklander  *
23*817466cbSJens Wiklander  *  This file is part of mbed TLS (https://tls.mbed.org)
24*817466cbSJens Wiklander  */
25*817466cbSJens Wiklander 
26*817466cbSJens Wiklander #if !defined(MBEDTLS_CONFIG_FILE)
27*817466cbSJens Wiklander #include "mbedtls/config.h"
28*817466cbSJens Wiklander #else
29*817466cbSJens Wiklander #include MBEDTLS_CONFIG_FILE
30*817466cbSJens Wiklander #endif
31*817466cbSJens Wiklander 
32*817466cbSJens Wiklander #if defined(MBEDTLS_MD_C)
33*817466cbSJens Wiklander 
34*817466cbSJens Wiklander #include "mbedtls/md.h"
35*817466cbSJens Wiklander #include "mbedtls/md_internal.h"
36*817466cbSJens Wiklander 
37*817466cbSJens Wiklander #if defined(MBEDTLS_PLATFORM_C)
38*817466cbSJens Wiklander #include "mbedtls/platform.h"
39*817466cbSJens Wiklander #else
40*817466cbSJens Wiklander #include <stdlib.h>
41*817466cbSJens Wiklander #define mbedtls_calloc    calloc
42*817466cbSJens Wiklander #define mbedtls_free       free
43*817466cbSJens Wiklander #endif
44*817466cbSJens Wiklander 
45*817466cbSJens Wiklander #include <string.h>
46*817466cbSJens Wiklander 
47*817466cbSJens Wiklander #if defined(MBEDTLS_FS_IO)
48*817466cbSJens Wiklander #include <stdio.h>
49*817466cbSJens Wiklander #endif
50*817466cbSJens Wiklander 
51*817466cbSJens Wiklander /* Implementation that should never be optimized out by the compiler */
52*817466cbSJens Wiklander static void mbedtls_zeroize( void *v, size_t n ) {
53*817466cbSJens Wiklander     volatile unsigned char *p = v; while( n-- ) *p++ = 0;
54*817466cbSJens Wiklander }
55*817466cbSJens Wiklander 
56*817466cbSJens Wiklander /*
57*817466cbSJens Wiklander  * Reminder: update profiles in x509_crt.c when adding a new hash!
58*817466cbSJens Wiklander  */
59*817466cbSJens Wiklander static const int supported_digests[] = {
60*817466cbSJens Wiklander 
61*817466cbSJens Wiklander #if defined(MBEDTLS_SHA512_C)
62*817466cbSJens Wiklander         MBEDTLS_MD_SHA512,
63*817466cbSJens Wiklander         MBEDTLS_MD_SHA384,
64*817466cbSJens Wiklander #endif
65*817466cbSJens Wiklander 
66*817466cbSJens Wiklander #if defined(MBEDTLS_SHA256_C)
67*817466cbSJens Wiklander         MBEDTLS_MD_SHA256,
68*817466cbSJens Wiklander         MBEDTLS_MD_SHA224,
69*817466cbSJens Wiklander #endif
70*817466cbSJens Wiklander 
71*817466cbSJens Wiklander #if defined(MBEDTLS_SHA1_C)
72*817466cbSJens Wiklander         MBEDTLS_MD_SHA1,
73*817466cbSJens Wiklander #endif
74*817466cbSJens Wiklander 
75*817466cbSJens Wiklander #if defined(MBEDTLS_RIPEMD160_C)
76*817466cbSJens Wiklander         MBEDTLS_MD_RIPEMD160,
77*817466cbSJens Wiklander #endif
78*817466cbSJens Wiklander 
79*817466cbSJens Wiklander #if defined(MBEDTLS_MD5_C)
80*817466cbSJens Wiklander         MBEDTLS_MD_MD5,
81*817466cbSJens Wiklander #endif
82*817466cbSJens Wiklander 
83*817466cbSJens Wiklander #if defined(MBEDTLS_MD4_C)
84*817466cbSJens Wiklander         MBEDTLS_MD_MD4,
85*817466cbSJens Wiklander #endif
86*817466cbSJens Wiklander 
87*817466cbSJens Wiklander #if defined(MBEDTLS_MD2_C)
88*817466cbSJens Wiklander         MBEDTLS_MD_MD2,
89*817466cbSJens Wiklander #endif
90*817466cbSJens Wiklander 
91*817466cbSJens Wiklander         MBEDTLS_MD_NONE
92*817466cbSJens Wiklander };
93*817466cbSJens Wiklander 
94*817466cbSJens Wiklander const int *mbedtls_md_list( void )
95*817466cbSJens Wiklander {
96*817466cbSJens Wiklander     return( supported_digests );
97*817466cbSJens Wiklander }
98*817466cbSJens Wiklander 
99*817466cbSJens Wiklander const mbedtls_md_info_t *mbedtls_md_info_from_string( const char *md_name )
100*817466cbSJens Wiklander {
101*817466cbSJens Wiklander     if( NULL == md_name )
102*817466cbSJens Wiklander         return( NULL );
103*817466cbSJens Wiklander 
104*817466cbSJens Wiklander     /* Get the appropriate digest information */
105*817466cbSJens Wiklander #if defined(MBEDTLS_MD2_C)
106*817466cbSJens Wiklander     if( !strcmp( "MD2", md_name ) )
107*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_MD2 );
108*817466cbSJens Wiklander #endif
109*817466cbSJens Wiklander #if defined(MBEDTLS_MD4_C)
110*817466cbSJens Wiklander     if( !strcmp( "MD4", md_name ) )
111*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_MD4 );
112*817466cbSJens Wiklander #endif
113*817466cbSJens Wiklander #if defined(MBEDTLS_MD5_C)
114*817466cbSJens Wiklander     if( !strcmp( "MD5", md_name ) )
115*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_MD5 );
116*817466cbSJens Wiklander #endif
117*817466cbSJens Wiklander #if defined(MBEDTLS_RIPEMD160_C)
118*817466cbSJens Wiklander     if( !strcmp( "RIPEMD160", md_name ) )
119*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_RIPEMD160 );
120*817466cbSJens Wiklander #endif
121*817466cbSJens Wiklander #if defined(MBEDTLS_SHA1_C)
122*817466cbSJens Wiklander     if( !strcmp( "SHA1", md_name ) || !strcmp( "SHA", md_name ) )
123*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_SHA1 );
124*817466cbSJens Wiklander #endif
125*817466cbSJens Wiklander #if defined(MBEDTLS_SHA256_C)
126*817466cbSJens Wiklander     if( !strcmp( "SHA224", md_name ) )
127*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_SHA224 );
128*817466cbSJens Wiklander     if( !strcmp( "SHA256", md_name ) )
129*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_SHA256 );
130*817466cbSJens Wiklander #endif
131*817466cbSJens Wiklander #if defined(MBEDTLS_SHA512_C)
132*817466cbSJens Wiklander     if( !strcmp( "SHA384", md_name ) )
133*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_SHA384 );
134*817466cbSJens Wiklander     if( !strcmp( "SHA512", md_name ) )
135*817466cbSJens Wiklander         return mbedtls_md_info_from_type( MBEDTLS_MD_SHA512 );
136*817466cbSJens Wiklander #endif
137*817466cbSJens Wiklander     return( NULL );
138*817466cbSJens Wiklander }
139*817466cbSJens Wiklander 
140*817466cbSJens Wiklander const mbedtls_md_info_t *mbedtls_md_info_from_type( mbedtls_md_type_t md_type )
141*817466cbSJens Wiklander {
142*817466cbSJens Wiklander     switch( md_type )
143*817466cbSJens Wiklander     {
144*817466cbSJens Wiklander #if defined(MBEDTLS_MD2_C)
145*817466cbSJens Wiklander         case MBEDTLS_MD_MD2:
146*817466cbSJens Wiklander             return( &mbedtls_md2_info );
147*817466cbSJens Wiklander #endif
148*817466cbSJens Wiklander #if defined(MBEDTLS_MD4_C)
149*817466cbSJens Wiklander         case MBEDTLS_MD_MD4:
150*817466cbSJens Wiklander             return( &mbedtls_md4_info );
151*817466cbSJens Wiklander #endif
152*817466cbSJens Wiklander #if defined(MBEDTLS_MD5_C)
153*817466cbSJens Wiklander         case MBEDTLS_MD_MD5:
154*817466cbSJens Wiklander             return( &mbedtls_md5_info );
155*817466cbSJens Wiklander #endif
156*817466cbSJens Wiklander #if defined(MBEDTLS_RIPEMD160_C)
157*817466cbSJens Wiklander         case MBEDTLS_MD_RIPEMD160:
158*817466cbSJens Wiklander             return( &mbedtls_ripemd160_info );
159*817466cbSJens Wiklander #endif
160*817466cbSJens Wiklander #if defined(MBEDTLS_SHA1_C)
161*817466cbSJens Wiklander         case MBEDTLS_MD_SHA1:
162*817466cbSJens Wiklander             return( &mbedtls_sha1_info );
163*817466cbSJens Wiklander #endif
164*817466cbSJens Wiklander #if defined(MBEDTLS_SHA256_C)
165*817466cbSJens Wiklander         case MBEDTLS_MD_SHA224:
166*817466cbSJens Wiklander             return( &mbedtls_sha224_info );
167*817466cbSJens Wiklander         case MBEDTLS_MD_SHA256:
168*817466cbSJens Wiklander             return( &mbedtls_sha256_info );
169*817466cbSJens Wiklander #endif
170*817466cbSJens Wiklander #if defined(MBEDTLS_SHA512_C)
171*817466cbSJens Wiklander         case MBEDTLS_MD_SHA384:
172*817466cbSJens Wiklander             return( &mbedtls_sha384_info );
173*817466cbSJens Wiklander         case MBEDTLS_MD_SHA512:
174*817466cbSJens Wiklander             return( &mbedtls_sha512_info );
175*817466cbSJens Wiklander #endif
176*817466cbSJens Wiklander         default:
177*817466cbSJens Wiklander             return( NULL );
178*817466cbSJens Wiklander     }
179*817466cbSJens Wiklander }
180*817466cbSJens Wiklander 
181*817466cbSJens Wiklander void mbedtls_md_init( mbedtls_md_context_t *ctx )
182*817466cbSJens Wiklander {
183*817466cbSJens Wiklander     memset( ctx, 0, sizeof( mbedtls_md_context_t ) );
184*817466cbSJens Wiklander }
185*817466cbSJens Wiklander 
186*817466cbSJens Wiklander void mbedtls_md_free( mbedtls_md_context_t *ctx )
187*817466cbSJens Wiklander {
188*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL )
189*817466cbSJens Wiklander         return;
190*817466cbSJens Wiklander 
191*817466cbSJens Wiklander     if( ctx->md_ctx != NULL )
192*817466cbSJens Wiklander         ctx->md_info->ctx_free_func( ctx->md_ctx );
193*817466cbSJens Wiklander 
194*817466cbSJens Wiklander     if( ctx->hmac_ctx != NULL )
195*817466cbSJens Wiklander     {
196*817466cbSJens Wiklander         mbedtls_zeroize( ctx->hmac_ctx, 2 * ctx->md_info->block_size );
197*817466cbSJens Wiklander         mbedtls_free( ctx->hmac_ctx );
198*817466cbSJens Wiklander     }
199*817466cbSJens Wiklander 
200*817466cbSJens Wiklander     mbedtls_zeroize( ctx, sizeof( mbedtls_md_context_t ) );
201*817466cbSJens Wiklander }
202*817466cbSJens Wiklander 
203*817466cbSJens Wiklander int mbedtls_md_clone( mbedtls_md_context_t *dst,
204*817466cbSJens Wiklander                       const mbedtls_md_context_t *src )
205*817466cbSJens Wiklander {
206*817466cbSJens Wiklander     if( dst == NULL || dst->md_info == NULL ||
207*817466cbSJens Wiklander         src == NULL || src->md_info == NULL ||
208*817466cbSJens Wiklander         dst->md_info != src->md_info )
209*817466cbSJens Wiklander     {
210*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
211*817466cbSJens Wiklander     }
212*817466cbSJens Wiklander 
213*817466cbSJens Wiklander     dst->md_info->clone_func( dst->md_ctx, src->md_ctx );
214*817466cbSJens Wiklander 
215*817466cbSJens Wiklander     return( 0 );
216*817466cbSJens Wiklander }
217*817466cbSJens Wiklander 
218*817466cbSJens Wiklander #if ! defined(MBEDTLS_DEPRECATED_REMOVED)
219*817466cbSJens Wiklander int mbedtls_md_init_ctx( mbedtls_md_context_t *ctx, const mbedtls_md_info_t *md_info )
220*817466cbSJens Wiklander {
221*817466cbSJens Wiklander     return mbedtls_md_setup( ctx, md_info, 1 );
222*817466cbSJens Wiklander }
223*817466cbSJens Wiklander #endif
224*817466cbSJens Wiklander 
225*817466cbSJens Wiklander int mbedtls_md_setup( mbedtls_md_context_t *ctx, const mbedtls_md_info_t *md_info, int hmac )
226*817466cbSJens Wiklander {
227*817466cbSJens Wiklander     if( md_info == NULL || ctx == NULL )
228*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
229*817466cbSJens Wiklander 
230*817466cbSJens Wiklander     if( ( ctx->md_ctx = md_info->ctx_alloc_func() ) == NULL )
231*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_ALLOC_FAILED );
232*817466cbSJens Wiklander 
233*817466cbSJens Wiklander     if( hmac != 0 )
234*817466cbSJens Wiklander     {
235*817466cbSJens Wiklander         ctx->hmac_ctx = mbedtls_calloc( 2, md_info->block_size );
236*817466cbSJens Wiklander         if( ctx->hmac_ctx == NULL )
237*817466cbSJens Wiklander         {
238*817466cbSJens Wiklander             md_info->ctx_free_func( ctx->md_ctx );
239*817466cbSJens Wiklander             return( MBEDTLS_ERR_MD_ALLOC_FAILED );
240*817466cbSJens Wiklander         }
241*817466cbSJens Wiklander     }
242*817466cbSJens Wiklander 
243*817466cbSJens Wiklander     ctx->md_info = md_info;
244*817466cbSJens Wiklander 
245*817466cbSJens Wiklander     return( 0 );
246*817466cbSJens Wiklander }
247*817466cbSJens Wiklander 
248*817466cbSJens Wiklander int mbedtls_md_starts( mbedtls_md_context_t *ctx )
249*817466cbSJens Wiklander {
250*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL )
251*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
252*817466cbSJens Wiklander 
253*817466cbSJens Wiklander     ctx->md_info->starts_func( ctx->md_ctx );
254*817466cbSJens Wiklander 
255*817466cbSJens Wiklander     return( 0 );
256*817466cbSJens Wiklander }
257*817466cbSJens Wiklander 
258*817466cbSJens Wiklander int mbedtls_md_update( mbedtls_md_context_t *ctx, const unsigned char *input, size_t ilen )
259*817466cbSJens Wiklander {
260*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL )
261*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
262*817466cbSJens Wiklander 
263*817466cbSJens Wiklander     ctx->md_info->update_func( ctx->md_ctx, input, ilen );
264*817466cbSJens Wiklander 
265*817466cbSJens Wiklander     return( 0 );
266*817466cbSJens Wiklander }
267*817466cbSJens Wiklander 
268*817466cbSJens Wiklander int mbedtls_md_finish( mbedtls_md_context_t *ctx, unsigned char *output )
269*817466cbSJens Wiklander {
270*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL )
271*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
272*817466cbSJens Wiklander 
273*817466cbSJens Wiklander     ctx->md_info->finish_func( ctx->md_ctx, output );
274*817466cbSJens Wiklander 
275*817466cbSJens Wiklander     return( 0 );
276*817466cbSJens Wiklander }
277*817466cbSJens Wiklander 
278*817466cbSJens Wiklander int mbedtls_md( const mbedtls_md_info_t *md_info, const unsigned char *input, size_t ilen,
279*817466cbSJens Wiklander             unsigned char *output )
280*817466cbSJens Wiklander {
281*817466cbSJens Wiklander     if( md_info == NULL )
282*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
283*817466cbSJens Wiklander 
284*817466cbSJens Wiklander     md_info->digest_func( input, ilen, output );
285*817466cbSJens Wiklander 
286*817466cbSJens Wiklander     return( 0 );
287*817466cbSJens Wiklander }
288*817466cbSJens Wiklander 
289*817466cbSJens Wiklander #if defined(MBEDTLS_FS_IO)
290*817466cbSJens Wiklander int mbedtls_md_file( const mbedtls_md_info_t *md_info, const char *path, unsigned char *output )
291*817466cbSJens Wiklander {
292*817466cbSJens Wiklander     int ret;
293*817466cbSJens Wiklander     FILE *f;
294*817466cbSJens Wiklander     size_t n;
295*817466cbSJens Wiklander     mbedtls_md_context_t ctx;
296*817466cbSJens Wiklander     unsigned char buf[1024];
297*817466cbSJens Wiklander 
298*817466cbSJens Wiklander     if( md_info == NULL )
299*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
300*817466cbSJens Wiklander 
301*817466cbSJens Wiklander     if( ( f = fopen( path, "rb" ) ) == NULL )
302*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_FILE_IO_ERROR );
303*817466cbSJens Wiklander 
304*817466cbSJens Wiklander     mbedtls_md_init( &ctx );
305*817466cbSJens Wiklander 
306*817466cbSJens Wiklander     if( ( ret = mbedtls_md_setup( &ctx, md_info, 0 ) ) != 0 )
307*817466cbSJens Wiklander         goto cleanup;
308*817466cbSJens Wiklander 
309*817466cbSJens Wiklander     md_info->starts_func( ctx.md_ctx );
310*817466cbSJens Wiklander 
311*817466cbSJens Wiklander     while( ( n = fread( buf, 1, sizeof( buf ), f ) ) > 0 )
312*817466cbSJens Wiklander         md_info->update_func( ctx.md_ctx, buf, n );
313*817466cbSJens Wiklander 
314*817466cbSJens Wiklander     if( ferror( f ) != 0 )
315*817466cbSJens Wiklander     {
316*817466cbSJens Wiklander         ret = MBEDTLS_ERR_MD_FILE_IO_ERROR;
317*817466cbSJens Wiklander         goto cleanup;
318*817466cbSJens Wiklander     }
319*817466cbSJens Wiklander 
320*817466cbSJens Wiklander     md_info->finish_func( ctx.md_ctx, output );
321*817466cbSJens Wiklander 
322*817466cbSJens Wiklander cleanup:
323*817466cbSJens Wiklander     fclose( f );
324*817466cbSJens Wiklander     mbedtls_md_free( &ctx );
325*817466cbSJens Wiklander 
326*817466cbSJens Wiklander     return( ret );
327*817466cbSJens Wiklander }
328*817466cbSJens Wiklander #endif /* MBEDTLS_FS_IO */
329*817466cbSJens Wiklander 
330*817466cbSJens Wiklander int mbedtls_md_hmac_starts( mbedtls_md_context_t *ctx, const unsigned char *key, size_t keylen )
331*817466cbSJens Wiklander {
332*817466cbSJens Wiklander     unsigned char sum[MBEDTLS_MD_MAX_SIZE];
333*817466cbSJens Wiklander     unsigned char *ipad, *opad;
334*817466cbSJens Wiklander     size_t i;
335*817466cbSJens Wiklander 
336*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL || ctx->hmac_ctx == NULL )
337*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
338*817466cbSJens Wiklander 
339*817466cbSJens Wiklander     if( keylen > (size_t) ctx->md_info->block_size )
340*817466cbSJens Wiklander     {
341*817466cbSJens Wiklander         ctx->md_info->starts_func( ctx->md_ctx );
342*817466cbSJens Wiklander         ctx->md_info->update_func( ctx->md_ctx, key, keylen );
343*817466cbSJens Wiklander         ctx->md_info->finish_func( ctx->md_ctx, sum );
344*817466cbSJens Wiklander 
345*817466cbSJens Wiklander         keylen = ctx->md_info->size;
346*817466cbSJens Wiklander         key = sum;
347*817466cbSJens Wiklander     }
348*817466cbSJens Wiklander 
349*817466cbSJens Wiklander     ipad = (unsigned char *) ctx->hmac_ctx;
350*817466cbSJens Wiklander     opad = (unsigned char *) ctx->hmac_ctx + ctx->md_info->block_size;
351*817466cbSJens Wiklander 
352*817466cbSJens Wiklander     memset( ipad, 0x36, ctx->md_info->block_size );
353*817466cbSJens Wiklander     memset( opad, 0x5C, ctx->md_info->block_size );
354*817466cbSJens Wiklander 
355*817466cbSJens Wiklander     for( i = 0; i < keylen; i++ )
356*817466cbSJens Wiklander     {
357*817466cbSJens Wiklander         ipad[i] = (unsigned char)( ipad[i] ^ key[i] );
358*817466cbSJens Wiklander         opad[i] = (unsigned char)( opad[i] ^ key[i] );
359*817466cbSJens Wiklander     }
360*817466cbSJens Wiklander 
361*817466cbSJens Wiklander     mbedtls_zeroize( sum, sizeof( sum ) );
362*817466cbSJens Wiklander 
363*817466cbSJens Wiklander     ctx->md_info->starts_func( ctx->md_ctx );
364*817466cbSJens Wiklander     ctx->md_info->update_func( ctx->md_ctx, ipad, ctx->md_info->block_size );
365*817466cbSJens Wiklander 
366*817466cbSJens Wiklander     return( 0 );
367*817466cbSJens Wiklander }
368*817466cbSJens Wiklander 
369*817466cbSJens Wiklander int mbedtls_md_hmac_update( mbedtls_md_context_t *ctx, const unsigned char *input, size_t ilen )
370*817466cbSJens Wiklander {
371*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL || ctx->hmac_ctx == NULL )
372*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
373*817466cbSJens Wiklander 
374*817466cbSJens Wiklander     ctx->md_info->update_func( ctx->md_ctx, input, ilen );
375*817466cbSJens Wiklander 
376*817466cbSJens Wiklander     return( 0 );
377*817466cbSJens Wiklander }
378*817466cbSJens Wiklander 
379*817466cbSJens Wiklander int mbedtls_md_hmac_finish( mbedtls_md_context_t *ctx, unsigned char *output )
380*817466cbSJens Wiklander {
381*817466cbSJens Wiklander     unsigned char tmp[MBEDTLS_MD_MAX_SIZE];
382*817466cbSJens Wiklander     unsigned char *opad;
383*817466cbSJens Wiklander 
384*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL || ctx->hmac_ctx == NULL )
385*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
386*817466cbSJens Wiklander 
387*817466cbSJens Wiklander     opad = (unsigned char *) ctx->hmac_ctx + ctx->md_info->block_size;
388*817466cbSJens Wiklander 
389*817466cbSJens Wiklander     ctx->md_info->finish_func( ctx->md_ctx, tmp );
390*817466cbSJens Wiklander     ctx->md_info->starts_func( ctx->md_ctx );
391*817466cbSJens Wiklander     ctx->md_info->update_func( ctx->md_ctx, opad, ctx->md_info->block_size );
392*817466cbSJens Wiklander     ctx->md_info->update_func( ctx->md_ctx, tmp, ctx->md_info->size );
393*817466cbSJens Wiklander     ctx->md_info->finish_func( ctx->md_ctx, output );
394*817466cbSJens Wiklander 
395*817466cbSJens Wiklander     return( 0 );
396*817466cbSJens Wiklander }
397*817466cbSJens Wiklander 
398*817466cbSJens Wiklander int mbedtls_md_hmac_reset( mbedtls_md_context_t *ctx )
399*817466cbSJens Wiklander {
400*817466cbSJens Wiklander     unsigned char *ipad;
401*817466cbSJens Wiklander 
402*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL || ctx->hmac_ctx == NULL )
403*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
404*817466cbSJens Wiklander 
405*817466cbSJens Wiklander     ipad = (unsigned char *) ctx->hmac_ctx;
406*817466cbSJens Wiklander 
407*817466cbSJens Wiklander     ctx->md_info->starts_func( ctx->md_ctx );
408*817466cbSJens Wiklander     ctx->md_info->update_func( ctx->md_ctx, ipad, ctx->md_info->block_size );
409*817466cbSJens Wiklander 
410*817466cbSJens Wiklander     return( 0 );
411*817466cbSJens Wiklander }
412*817466cbSJens Wiklander 
413*817466cbSJens Wiklander int mbedtls_md_hmac( const mbedtls_md_info_t *md_info, const unsigned char *key, size_t keylen,
414*817466cbSJens Wiklander                 const unsigned char *input, size_t ilen,
415*817466cbSJens Wiklander                 unsigned char *output )
416*817466cbSJens Wiklander {
417*817466cbSJens Wiklander     mbedtls_md_context_t ctx;
418*817466cbSJens Wiklander     int ret;
419*817466cbSJens Wiklander 
420*817466cbSJens Wiklander     if( md_info == NULL )
421*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
422*817466cbSJens Wiklander 
423*817466cbSJens Wiklander     mbedtls_md_init( &ctx );
424*817466cbSJens Wiklander 
425*817466cbSJens Wiklander     if( ( ret = mbedtls_md_setup( &ctx, md_info, 1 ) ) != 0 )
426*817466cbSJens Wiklander         return( ret );
427*817466cbSJens Wiklander 
428*817466cbSJens Wiklander     mbedtls_md_hmac_starts( &ctx, key, keylen );
429*817466cbSJens Wiklander     mbedtls_md_hmac_update( &ctx, input, ilen );
430*817466cbSJens Wiklander     mbedtls_md_hmac_finish( &ctx, output );
431*817466cbSJens Wiklander 
432*817466cbSJens Wiklander     mbedtls_md_free( &ctx );
433*817466cbSJens Wiklander 
434*817466cbSJens Wiklander     return( 0 );
435*817466cbSJens Wiklander }
436*817466cbSJens Wiklander 
437*817466cbSJens Wiklander int mbedtls_md_process( mbedtls_md_context_t *ctx, const unsigned char *data )
438*817466cbSJens Wiklander {
439*817466cbSJens Wiklander     if( ctx == NULL || ctx->md_info == NULL )
440*817466cbSJens Wiklander         return( MBEDTLS_ERR_MD_BAD_INPUT_DATA );
441*817466cbSJens Wiklander 
442*817466cbSJens Wiklander     ctx->md_info->process_func( ctx->md_ctx, data );
443*817466cbSJens Wiklander 
444*817466cbSJens Wiklander     return( 0 );
445*817466cbSJens Wiklander }
446*817466cbSJens Wiklander 
447*817466cbSJens Wiklander unsigned char mbedtls_md_get_size( const mbedtls_md_info_t *md_info )
448*817466cbSJens Wiklander {
449*817466cbSJens Wiklander     if( md_info == NULL )
450*817466cbSJens Wiklander         return( 0 );
451*817466cbSJens Wiklander 
452*817466cbSJens Wiklander     return md_info->size;
453*817466cbSJens Wiklander }
454*817466cbSJens Wiklander 
455*817466cbSJens Wiklander mbedtls_md_type_t mbedtls_md_get_type( const mbedtls_md_info_t *md_info )
456*817466cbSJens Wiklander {
457*817466cbSJens Wiklander     if( md_info == NULL )
458*817466cbSJens Wiklander         return( MBEDTLS_MD_NONE );
459*817466cbSJens Wiklander 
460*817466cbSJens Wiklander     return md_info->type;
461*817466cbSJens Wiklander }
462*817466cbSJens Wiklander 
463*817466cbSJens Wiklander const char *mbedtls_md_get_name( const mbedtls_md_info_t *md_info )
464*817466cbSJens Wiklander {
465*817466cbSJens Wiklander     if( md_info == NULL )
466*817466cbSJens Wiklander         return( NULL );
467*817466cbSJens Wiklander 
468*817466cbSJens Wiklander     return md_info->name;
469*817466cbSJens Wiklander }
470*817466cbSJens Wiklander 
471*817466cbSJens Wiklander #endif /* MBEDTLS_MD_C */
472