* md.c: Added another oddball OIW OID (sha-1WithRSAEncryption).
[libgcrypt.git] / cipher / md.c
1 /* md.c  -  message digest dispatcher
2  * Copyright (C) 1998, 1999, 2002 Free Software Foundation, Inc.
3  *
4  * This file is part of Libgcrypt.
5  *
6  * Libgcrypt is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU Lesser general Public License as
8  * published by the Free Software Foundation; either version 2.1 of
9  * the License, or (at your option) any later version.
10  *
11  * Libgcrypt is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
19  */
20
21 #include <config.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <errno.h>
26 #include <assert.h>
27
28 #include "g10lib.h"
29 #include "cipher.h"
30 #include "dynload.h"
31 #include "rmd.h"
32
33
34 static struct {
35   const char *oidstring;
36   int algo;
37 } oid_table[] = {
38   /* iso.member-body.us.rsadsi.pkcs.pkcs-1.5 (sha1WithRSAEncryption) */
39   { "1.2.840.113549.1.1.5",  GCRY_MD_SHA1 },
40   /* iso.member-body.us.rsadsi.pkcs.pkcs-1.4 (md5WithRSAEncryption) */
41   { "1.2.840.113549.1.1.4",  GCRY_MD_MD5 },
42   /* iso.member-body.us.x9-57.x9cm.3 (dsaWithSha1)*/
43   { "1.2.840.10040.4.3",     GCRY_MD_SHA1 },
44   /* from NIST's OIW  (sha1) */
45   { "1.3.14.3.2.26",         GCRY_MD_SHA1 },
46   /* rsaSignatureWithripemd160 */
47   { "1.3.36.3.3.1.2",       GCRY_MD_RMD160 },
48   /* RSADSI digestAlgorithm MD5 */
49   { "1.2.840.113549.2.5",   GCRY_MD_MD5 },
50   /* GNU.digestAlgorithm TIGER */
51   { "1.3.6.1.4.1.11591.12.2", GCRY_MD_TIGER },
52   /* iso.member-body.us.rsadsi.digestAlgorithm.md4 */
53   { "1.2.840.113549.2.4", GCRY_MD_MD4 },
54   /* from NIST OIW (sha-1WithRSAEncryption) */
55   { "1.3.14.3.2.29", GCRY_MD_SHA1 },
56   {NULL}
57 };
58
59
60
61
62 struct md_digest_list_s;
63
64 /* this structure is put right after the GCRY_MD_HD buffer, so that
65  * only one memory block is needed. */
66 struct gcry_md_context {
67     int  magic;
68     int  secure;
69     FILE  *debug;
70     int finalized;
71     struct md_digest_list_s *list;
72     byte *macpads;
73 };
74 #define CTX_MAGIC_NORMAL 0x11071961
75 #define CTX_MAGIC_SECURE 0x16917011
76
77 static const char * digest_algo_to_string( int algo );
78 static int check_digest_algo( int algo );
79 static GCRY_MD_HD md_open( int algo, int secure, int hmac );
80 static int  md_enable( GCRY_MD_HD hd, int algo );
81 static GCRY_MD_HD md_copy( GCRY_MD_HD a );
82 static void md_close(GCRY_MD_HD a);
83 static void md_write( GCRY_MD_HD a, byte *inbuf, size_t inlen);
84 static void md_final(GCRY_MD_HD a);
85 static byte *md_read( GCRY_MD_HD a, int algo );
86 static int md_get_algo( GCRY_MD_HD a );
87 static int md_digest_length( int algo );
88 static const byte *md_asn_oid( int algo, size_t *asnlen, size_t *mdlen );
89 static void md_start_debug( GCRY_MD_HD a, const char *suffix );
90 static void md_stop_debug( GCRY_MD_HD a );
91
92 /****************
93  * This structure is used for the list of available algorithms
94  * and for the list of algorithms in GCRY_MD_HD.
95  */
96 struct md_digest_list_s {
97     struct md_digest_list_s *next;
98     const char *name;
99     int algo;
100     byte *asnoid;
101     int asnlen;
102     int mdlen;
103     void (*init)( void *c );
104     void (*write)( void *c, byte *buf, size_t nbytes );
105     void (*final)( void *c );
106     byte *(*read)( void *c );
107     size_t contextsize; /* allocate this amount of context */
108     PROPERLY_ALIGNED_TYPE context;
109 };
110
111 static struct md_digest_list_s *digest_list;
112
113 #define digitp(p)   (*(p) >= 0 && *(p) <= '9')
114
115
116
117 \f
118 static struct md_digest_list_s *
119 new_list_item( int algo,
120                const char *(*get_info)( int, size_t*,byte**, int*, int*,
121                                        void (**)(void*),
122                                        void (**)(void*,byte*,size_t),
123                                        void (**)(void*),byte *(**)(void*)) )
124 {
125     struct md_digest_list_s *r;
126
127     r = gcry_xcalloc( 1, sizeof *r );
128     r->algo = algo,
129     r->name = (*get_info)( algo, &r->contextsize,
130                            &r->asnoid, &r->asnlen, &r->mdlen,
131                            &r->init, &r->write, &r->final, &r->read );
132     if( !r->name ) {
133         gcry_free(r);
134         r = NULL;
135     }
136     return r;
137 }
138
139
140
141 /****************
142  * Try to load the modules with the requested algorithm
143  * and return true if new modules are available
144  * If req_alog is -1 try to load all digest algorithms.
145  */
146 static int
147 load_digest_module( int req_algo )
148 {
149     static int initialized = 0;
150     static u32 checked_algos[512/32];
151     static int checked_all = 0;
152     struct md_digest_list_s *r;
153     void *context = NULL;
154     int algo;
155     int any = 0;
156     const char *(*get_info)( int, size_t*,byte**, int*, int*,
157                             void (**)(void*),
158                             void (**)(void*,byte*,size_t),
159                             void (**)(void*),byte *(**)(void*));
160
161     if( !initialized ) {
162         _gcry_cipher_modules_constructor();
163         initialized = 1;
164     }
165     algo = req_algo;
166     if( algo > 511 || !algo )
167         return 0; /* algorithm number too high (does not fit into out bitmap)*/
168     if( checked_all )
169         return 0; /* already called with -1 */
170     if( algo < 0 )
171         checked_all = 1;
172     else if( (checked_algos[algo/32] & (1 << (algo%32))) )
173         return 0; /* already checked and not found */
174     else
175         checked_algos[algo/32] |= (1 << (algo%32));
176
177     while( _gcry_enum_gnupgext_digests( &context, &algo, &get_info ) ) {
178         if( req_algo != -1 && algo != req_algo )
179             continue;
180         for(r=digest_list; r; r = r->next )
181             if( r->algo == algo )
182                 break;
183         if( r ) {
184             log_info("skipping digest %d: already loaded\n", algo );
185             continue;
186         }
187         r = new_list_item( algo, get_info );
188         if( ! r ) {
189             log_info("skipping digest %d: no name\n", algo );
190             continue;
191         }
192         /* put it into the list */
193         if( _gcry_log_verbosity( 2 ) )
194             log_info("loaded digest %d\n", algo);
195         r->next = digest_list;
196         digest_list = r;
197         any = 1;
198         if( req_algo != -1 )
199             break;
200     }
201     _gcry_enum_gnupgext_digests( &context, NULL, NULL );
202     return any;
203 }
204
205
206
207 /****************
208  * Map a string to the digest algo
209  */
210 int
211 gcry_md_map_name( const char *string )
212 {
213     struct md_digest_list_s *r;
214     
215     if (!string)
216       return 0;
217
218     /* If the string starts with a digit (optionally prefixed with
219        either "OID." or "oid."), we first look into our table of ASN.1
220        object identifiers to figure out the algorithm */
221     if (digitp (string)
222         || !strncmp (string, "oid.", 4) 
223         || !strncmp (string, "OID.", 4) )
224       {
225         int i;
226         const char *s =  digitp(string)? string : (string+4);
227
228         for (i=0; oid_table[i].oidstring; i++)
229           {
230             if (!strcmp (s, oid_table[i].oidstring))
231               return oid_table[i].algo;
232           }
233       }
234
235     do {
236         for(r = digest_list; r; r = r->next )
237             if( !stricmp( r->name, string ) )
238                 return r->algo;
239     } while( !r && load_digest_module(-1) );
240     return 0;
241 }
242
243
244 /****************
245  * Map a digest algo to a string
246  */
247 static const char *
248 digest_algo_to_string( int algo )
249 {
250     struct md_digest_list_s *r;
251
252     do {
253         for(r = digest_list; r; r = r->next )
254             if( r->algo == algo )
255                 return r->name;
256     } while( !r && load_digest_module( algo ) );
257     return NULL;
258 }
259
260 /****************
261  * This function simply returns the name of the algorithm or some constant
262  * string when there is no algo.  It will never return NULL.
263  * Use  the macro gcry_md_test_algo() to check whether the algorithm
264  * is valid.
265  */
266 const char *
267 gcry_md_algo_name( int algo )
268 {
269     const char *s = digest_algo_to_string( algo );
270     return s? s: "?";
271 }
272
273
274 static int
275 check_digest_algo( int algo )
276 {
277     struct md_digest_list_s *r;
278
279     do {
280         for(r = digest_list; r; r = r->next )
281             if( r->algo == algo )
282                 return 0;
283     } while( !r && load_digest_module(algo) );
284     return GCRYERR_INV_MD_ALGO;
285 }
286
287
288
289 /****************
290  * Open a message digest handle for use with algorithm ALGO.
291  * More algorithms may be added by md_enable(). The initial algorithm
292  * may be 0.
293  */
294 static GCRY_MD_HD
295 md_open( int algo, int secure, int hmac )
296 {
297     GCRY_MD_HD hd;
298     struct gcry_md_context *ctx;
299     int bufsize = secure? 512 : 1024;
300     size_t n;
301
302     /* Allocate a memory area to hold the caller visible buffer with it's
303      * control information and the data required by this module. Set the
304      * context pointer at the beginning to this area.
305      * We have to use this strange scheme because we want to hide the
306      * internal data but have a variable sized buffer.
307      *
308      *  +---+------+---........------+-------------+
309      *  !ctx! bctl !  buffer         ! private     !
310      *  +---+------+---........------+-------------+
311      *    !                           ^
312      *    !---------------------------!
313      *
314      * We have to make sture that private is well aligned.
315      */
316     n = sizeof( struct gcry_md_handle ) + bufsize;
317     n = ((n + sizeof(PROPERLY_ALIGNED_TYPE)-1)
318          / sizeof(PROPERLY_ALIGNED_TYPE) ) * sizeof(PROPERLY_ALIGNED_TYPE);
319
320     /* allocate and set the Context pointer to the private data */
321     hd = secure ? gcry_malloc_secure( n + sizeof( struct gcry_md_context ) )
322                 : gcry_malloc(       n + sizeof( struct gcry_md_context ) );
323     if( !hd ) {
324         set_lasterr( GCRYERR_NO_MEM );
325         return NULL;
326     }
327
328     hd->ctx = ctx = (struct gcry_md_context*)( (char*)hd + n );
329     /* setup the globally visible data (bctl in the diagram)*/
330     hd->bufsize = n - sizeof( struct gcry_md_handle ) + 1;
331     hd->bufpos = 0;
332     /* initialize the private data */
333     memset( hd->ctx, 0, sizeof *hd->ctx );
334     ctx->magic = secure ? CTX_MAGIC_SECURE : CTX_MAGIC_NORMAL;
335     ctx->secure = secure;
336     if( hmac ) {
337         ctx->macpads = gcry_malloc_secure( 128 );
338         if( !ctx->macpads ) {
339             md_close( hd );
340             set_lasterr( GCRYERR_NO_MEM );
341             return NULL;
342         }
343     }
344     fast_random_poll(); /* FIXME: should we really do that? */
345     if( algo && md_enable( hd, algo ) ) {
346         md_close( hd );
347         return NULL;
348     }
349     return hd;
350 }
351
352
353 GCRY_MD_HD
354 gcry_md_open (int algo, unsigned int flags)
355 {
356   GCRY_MD_HD hd;
357
358   if (check_digest_algo (algo))
359     {
360       set_lasterr (GCRYERR_INV_MD_ALGO);
361       return NULL;
362     }
363   if ((flags &~ GCRY_MD_FLAG_SECURE) > GCRY_MD_FLAG_SECURE
364       && (flags &~ GCRY_MD_FLAG_HMAC) > GCRY_MD_FLAG_HMAC)
365     {
366       set_lasterr (GCRYERR_INV_ARG);
367       return NULL;
368     }
369   hd = md_open (algo, (flags & GCRY_MD_FLAG_SECURE),
370                 (flags & GCRY_MD_FLAG_HMAC));
371   return hd;
372 }
373
374
375
376 static int
377 md_enable( GCRY_MD_HD hd, int algo )
378 {
379     struct gcry_md_context *h = hd->ctx;
380     struct md_digest_list_s *r, *ac;
381
382     for( ac=h->list; ac; ac = ac->next )
383         if( ac->algo == algo )
384             return 0; /* already enabled */
385     /* find the algorithm */
386     do {
387         for(r = digest_list; r; r = r->next )
388             if( r->algo == algo )
389                 break;
390     } while( !r && load_digest_module( algo ) );
391     if( !r ) {
392         log_debug("md_enable: algorithm %d not available\n", algo );
393         return set_lasterr( GCRYERR_INV_MD_ALGO );
394     }
395     /* and allocate a new list entry */
396     ac = h->secure? gcry_malloc_secure( sizeof *ac + r->contextsize
397                                                - sizeof(r->context) )
398                   : gcry_malloc( sizeof *ac + r->contextsize
399                                                - sizeof(r->context) );
400     if( !ac )
401         return set_lasterr( GCRYERR_NO_MEM );
402
403     *ac = *r;
404     ac->next = h->list;
405     h->list = ac;
406     /* and init this instance */
407     (*ac->init)( &ac->context.c );
408     return 0;
409 }
410
411
412 int
413 gcry_md_enable( GCRY_MD_HD hd, int algo )
414 {
415     return md_enable( hd, algo );
416 }
417
418 static GCRY_MD_HD
419 md_copy( GCRY_MD_HD ahd )
420 {
421     struct gcry_md_context *a = ahd->ctx;
422     struct gcry_md_context *b;
423     GCRY_MD_HD bhd;
424     struct md_digest_list_s *ar, *br;
425     size_t n;
426
427     if( ahd->bufpos )
428         md_write( ahd, NULL, 0 );
429
430     n = (char*)ahd->ctx - (char*)ahd;
431     bhd = a->secure ? gcry_malloc_secure( n + sizeof( struct gcry_md_context ) )
432                     : gcry_malloc(       n + sizeof( struct gcry_md_context ) );
433     if( !bhd ) {
434         set_lasterr( GCRYERR_NO_MEM );
435         return NULL;
436     }
437
438     bhd->ctx = b = (struct gcry_md_context*)( (char*)bhd + n );
439     /* no need to copy the buffer due to the write above */
440     assert( ahd->bufsize == (n - sizeof( struct gcry_md_handle ) + 1) );
441     bhd->bufsize = ahd->bufsize;
442     bhd->bufpos = 0;  assert( !ahd->bufpos );
443     memcpy( b, a, sizeof *a );
444     b->list = NULL;
445     b->debug = NULL;
446     if( a->macpads ) {
447         b->macpads = gcry_malloc_secure( 128 );
448         memcpy( b->macpads, a->macpads, 128 );
449     }
450     /* and now copy the complete list of algorithms */
451     /* I know that the copied list is reversed, but that doesn't matter */
452     for( ar=a->list; ar; ar = ar->next ) {
453         br = a->secure ? gcry_xmalloc_secure( sizeof *br + ar->contextsize
454                                                - sizeof(ar->context) )
455                        : gcry_xmalloc( sizeof *br + ar->contextsize
456                                                - sizeof(ar->context) );
457         memcpy( br, ar, sizeof(*br) + ar->contextsize
458                                     - sizeof(ar->context) );
459         br->next = b->list;
460         b->list = br;
461     }
462
463     if( a->debug )
464         md_start_debug( bhd, "unknown" );
465     return bhd;
466 }
467
468 GCRY_MD_HD
469 gcry_md_copy( GCRY_MD_HD hd )
470 {
471     return md_copy( hd );
472 }
473
474 /****************
475  * Reset all contexts and discard any buffered stuff.  This may be used
476  * instead of a md_close(); md_open().
477  */
478 void
479 gcry_md_reset( GCRY_MD_HD a )
480 {
481     struct md_digest_list_s *r;
482
483     a->bufpos = a->ctx->finalized = 0;
484     for( r=a->ctx->list; r; r = r->next ) {
485         memset( r->context.c, 0, r->contextsize );
486         (*r->init)( &r->context.c );
487     }
488     if( a->ctx->macpads ) {
489         md_write( a, a->ctx->macpads, 64 ); /* inner pad */
490     }
491 }
492
493
494 static void
495 md_close(GCRY_MD_HD a)
496 {
497     struct md_digest_list_s *r, *r2;
498
499     if( !a )
500         return;
501     if( a->ctx->debug )
502         md_stop_debug(a);
503     for(r=a->ctx->list; r; r = r2 ) {
504         r2 = r->next;
505         gcry_free(r);
506     }
507     gcry_free(a->ctx->macpads);
508     gcry_free(a);
509 }
510
511
512 void
513 gcry_md_close( GCRY_MD_HD hd )
514 {
515     md_close( hd );
516 }
517
518
519 static void
520 md_write( GCRY_MD_HD a, byte *inbuf, size_t inlen)
521 {
522     struct md_digest_list_s *r;
523
524     if( a->ctx->debug ) {
525         if( a->bufpos && fwrite(a->buf, a->bufpos, 1, a->ctx->debug ) != 1 )
526             BUG();
527         if( inlen && fwrite(inbuf, inlen, 1, a->ctx->debug ) != 1 )
528             BUG();
529     }
530     for(r=a->ctx->list; r; r = r->next ) {
531         if( a->bufpos )
532             (*r->write)( &r->context.c, a->buf, a->bufpos );
533         (*r->write)( &r->context.c, inbuf, inlen );
534     }
535     a->bufpos = 0;
536 }
537
538
539 void
540 gcry_md_write( GCRY_MD_HD hd, const byte *inbuf, size_t inlen)
541 {
542     md_write( hd, (byte*)inbuf, inlen );
543 }
544
545
546
547 static void
548 md_final(GCRY_MD_HD a)
549 {
550     struct md_digest_list_s *r;
551
552     if( a->ctx->finalized )
553         return;
554
555     if( a->bufpos )
556         md_write( a, NULL, 0 );
557
558     for(r=a->ctx->list; r; r = r->next ) {
559         (*r->final)( &r->context.c );
560     }
561     a->ctx->finalized = 1;
562     if( a->ctx->macpads ) {  /* finish the hmac */
563         int algo = md_get_algo( a );
564         byte *p = md_read( a, algo );
565         size_t dlen = md_digest_length(algo);
566
567         GCRY_MD_HD om = md_open( algo, a->ctx->secure, 0 );
568         if( !om )
569             _gcry_fatal_error( gcry_errno(), NULL );
570         md_write( om, a->ctx->macpads+64, 64 );
571         md_write( om, p, dlen );
572         md_final( om );
573         /* replace our digest with the mac (they have the same size) */
574         memcpy( p, md_read( om, algo ), dlen );
575         md_close( om );
576     }
577 }
578
579
580
581 static int
582 prepare_macpads( GCRY_MD_HD hd, const byte *key, size_t keylen)
583 {
584     int i;
585     int algo = md_get_algo( hd );
586     byte *helpkey = NULL;
587     byte *ipad, *opad;
588
589     if( !algo )
590         return GCRYERR_INV_MD_ALGO; /* i.e. no algo enabled */
591
592     if( keylen > 64 ) {
593         helpkey = gcry_malloc_secure( md_digest_length( algo ) );
594         if( !helpkey )
595             return GCRYERR_NO_MEM;
596         gcry_md_hash_buffer( algo, helpkey, key, keylen );
597         key = helpkey;
598         keylen = md_digest_length( algo );
599         assert( keylen <= 64 );
600     }
601
602     memset( hd->ctx->macpads, 0, 128 );
603     ipad = hd->ctx->macpads;
604     opad = hd->ctx->macpads+64;
605     memcpy( ipad, key, keylen );
606     memcpy( opad, key, keylen );
607     for(i=0; i < 64; i++ ) {
608         ipad[i] ^= 0x36;
609         opad[i] ^= 0x5c;
610     }
611     gcry_free( helpkey );
612     return 0;
613 }
614
615 int
616 gcry_md_ctl( GCRY_MD_HD hd, int cmd, byte *buffer, size_t buflen)
617 {
618     int rc = 0;
619     if( cmd == GCRYCTL_FINALIZE )
620         md_final( hd );
621     else if( cmd == GCRYCTL_SET_KEY ) {
622         rc = gcry_md_setkey ( hd, buffer, buflen );
623     }
624     else if( cmd == GCRYCTL_START_DUMP ) {
625         md_start_debug( hd, buffer );
626     }
627     else if( cmd == GCRYCTL_STOP_DUMP ) {
628         md_stop_debug( hd );
629     }
630     else
631         rc = GCRYERR_INV_OP;
632     return set_lasterr( rc );
633 }
634
635
636 int
637 gcry_md_setkey( GCRY_MD_HD hd, const char *key, size_t keylen )
638 {
639     int rc = 0;
640
641     if( !(hd->ctx->macpads ) )
642         rc = GCRYERR_CONFLICT;
643     else if ( !(rc = prepare_macpads( hd, key, keylen )) )
644         gcry_md_reset( hd );
645
646     return rc;
647 }
648
649
650 /****************
651  * if ALGO is null get the digest for the used algo (which should be only one)
652  */
653 static byte *
654 md_read( GCRY_MD_HD a, int algo )
655 {
656     struct md_digest_list_s *r;
657
658     if( !algo ) {  /* return the first algorithm */
659         if( (r=a->ctx->list) ) {
660             if( r->next )
661                 log_debug("more than algorithm in md_read(0)\n");
662             return (*r->read)( &r->context.c );
663         }
664     }
665     else {
666         for(r=a->ctx->list; r; r = r->next )
667             if( r->algo == algo )
668                 return (*r->read)( &r->context.c );
669     }
670     BUG();
671     return NULL;
672 }
673
674 /****************
675  * Read out the complete digest, this function implictly finalizes
676  * the hash.
677  */
678 byte *
679 gcry_md_read( GCRY_MD_HD hd, int algo )
680 {
681     gcry_md_ctl( hd, GCRYCTL_FINALIZE, NULL, 0 );
682     return md_read( hd, algo);
683 }
684
685
686 /****************
687  * This function combines md_final and md_read but keeps the context
688  * intact.  This function can be used to calculate intermediate
689  * digests.  The digest is copied into buffer and the digestlength is
690  * returned.  If buffer is NULL only the needed size for buffer is returned.
691  * buflen gives the max size of buffer. If the buffer is too shourt to
692  * hold the complete digest, the buffer is filled with as many bytes are
693  * possible and this value is returned.
694  */
695 #if 0
696 static int
697 md_digest( GCRY_MD_HD a, int algo, byte *buffer, int buflen )
698 {
699     struct md_digest_list_s *r = NULL;
700     char *context;
701     char *digest;
702
703     if( a->bufpos )
704         md_write( a, NULL, 0 );
705
706     if( !algo ) {  /* return digest for the first algorithm */
707         if( (r=a->ctx->list) && r->next )
708             log_debug("more than algorithm in md_digest(0)\n");
709     }
710     else {
711         for(r=a->ctx->list; r; r = r->next )
712             if( r->algo == algo )
713                 break;
714     }
715     if( !r )
716         BUG();
717
718     if( !buffer )
719         return r->mdlen;
720
721     /* I don't want to change the interface, so I simply work on a copy
722      * of the context (extra overhead - should be fixed)*/
723     context = a->ctx->secure ? gcry_xmalloc_secure( r->contextsize )
724                              : gcry_xmalloc( r->contextsize );
725     memcpy( context, r->context.c, r->contextsize );
726     (*r->final)( context );
727     digest = (*r->read)( context );
728
729     if( buflen > r->mdlen )
730         buflen = r->mdlen;
731     memcpy( buffer, digest, buflen );
732
733     gcry_free(context);
734     return buflen;
735 }
736 #endif
737
738 /****************
739  * Read out an intermediate digest.
740  */
741 int
742 gcry_md_get( GCRY_MD_HD hd, int algo, byte *buffer, int buflen )
743 {
744     /*md_digest ... */
745     return GCRYERR_INTERNAL;
746 }
747
748
749 /****************
750  * Shortcut function to hash a buffer with a given algo. The only supported
751  * algorithm is RIPE-MD. The supplied digest buffer must be large enough
752  * to store the resulting hash.  No error is returned, the function will
753  * abort on an invalid algo.  DISABLED_ALGOS are ignored here.
754  */
755 void
756 gcry_md_hash_buffer( int algo, char *digest, const char *buffer, size_t length)
757 {
758     if( algo == GCRY_MD_RMD160 )
759         _gcry_rmd160_hash_buffer( digest, buffer, length );
760     else { /* for the others we do not have a fast function, so
761             * we use the normal functions to do it */
762         GCRY_MD_HD h = md_open( algo, 0, 0 );
763         if( !h )
764             BUG(); /* algo not available */
765         md_write( h, (byte*)buffer, length );
766         md_final( h );
767         memcpy( digest, md_read( h, algo ), md_digest_length( algo ) );
768         md_close (h);
769     }
770 }
771
772 static int
773 md_get_algo( GCRY_MD_HD a )
774 {
775     struct md_digest_list_s *r;
776
777     if( (r=a->ctx->list) ) {
778         if( r->next )
779             log_error("WARNING: more than algorithm in md_get_algo()\n");
780         return r->algo;
781     }
782     return 0;
783 }
784
785
786 int
787 gcry_md_get_algo (GCRY_MD_HD hd)
788 {
789   int algo = md_get_algo (hd);
790   if (!algo)
791     {
792       set_lasterr (GCRYERR_INV_MD_ALGO);
793       return 0;
794     }
795   return algo;
796 }
797
798
799 /****************
800  * Return the length of the digest
801  */
802 static int
803 md_digest_length( int algo )
804 {
805     struct md_digest_list_s *r;
806
807     do {
808         for(r = digest_list; r; r = r->next ) {
809             if( r->algo == algo )
810                 return r->mdlen;
811         }
812     } while( !r && load_digest_module( algo ) );
813     return 0;
814 }
815
816 /****************
817  * Return the length of the digest in bytes.
818  * This function will return 0 in case of errors.
819  */
820 unsigned int
821 gcry_md_get_algo_dlen( int algo )
822 {
823     /* we do some very quick checks here */
824     switch( algo )
825     {
826       case GCRY_MD_MD4:
827       case GCRY_MD_MD5: return 16;
828       case GCRY_MD_SHA1:
829       case GCRY_MD_RMD160: return 20;
830       default: {
831             int len = md_digest_length( algo );
832             if( !len )
833                 set_lasterr( GCRYERR_INV_MD_ALGO );
834             return 0;
835         }
836     }
837 }
838
839
840 /* Hmmm: add a mode to enumerate the OIDs
841  *      to make g10/sig-check.c more portable */
842 static const byte *
843 md_asn_oid( int algo, size_t *asnlen, size_t *mdlen )
844 {
845     struct md_digest_list_s *r;
846
847     do {
848         for(r = digest_list; r; r = r->next ) {
849             if( r->algo == algo ) {
850                 if( asnlen )
851                     *asnlen = r->asnlen;
852                 if( mdlen )
853                     *mdlen = r->mdlen;
854                 return r->asnoid;
855             }
856         }
857     } while( !r && load_digest_module( algo ) );
858     log_bug("no asn for md algo %d\n", algo);
859     return NULL;
860 }
861
862
863
864 /****************
865  * Return information about the given cipher algorithm
866  * WHAT select the kind of information returned:
867  *  GCRYCTL_TEST_ALGO:
868  *      Returns 0 when the specified algorithm is available for use.
869  *      buffer and nbytes must be zero.
870  *  GCRYCTL_GET_ASNOID:
871  *      Return the ASNOID of the algorithm in buffer. if buffer is NULL, only
872  *      the required length is returned.
873  *
874  * On error the value -1 is returned and the error reason may be
875  * retrieved by gcry_errno().
876  * Note:  Because this function is in most cases used to return an
877  * integer value, we can make it easier for the caller to just look at
878  * the return value.  The caller will in all cases consult the value
879  * and thereby detecting whether a error occured or not (i.e. while checking
880  * the block size)
881  */
882 int
883 gcry_md_algo_info( int algo, int what, void *buffer, size_t *nbytes)
884 {
885     switch( what ) {
886       case GCRYCTL_TEST_ALGO:
887         if( buffer || nbytes ) {
888             set_lasterr( GCRYERR_INV_ARG );
889             return -1;
890         }
891         if( check_digest_algo( algo ) ) {
892             set_lasterr( GCRYERR_INV_MD_ALGO );
893             return -1;
894         }
895         break;
896
897       case GCRYCTL_GET_ASNOID: {
898             size_t asnlen;
899             const char *asn = md_asn_oid( algo, &asnlen, NULL );
900             if( buffer && *nbytes >= asnlen ) {
901                 memcpy( buffer, asn, asnlen );
902                 *nbytes = asnlen;
903                 return 0;
904             }
905             if( !buffer && nbytes ) {
906                 *nbytes = asnlen;
907                 return 0;
908             }
909             set_lasterr( buffer ? GCRYERR_TOO_SHORT : GCRYERR_INV_ARG );
910             return -1;
911         }
912         break;
913
914       default:
915         set_lasterr( GCRYERR_INV_OP );
916         return -1;
917     }
918     return 0;
919 }
920
921
922
923
924 static void
925 md_start_debug( GCRY_MD_HD md, const char *suffix )
926 {
927     static int idx=0;
928     char buf[25];
929
930     if( md->ctx->debug ) {
931         log_debug("Oops: md debug already started\n");
932         return;
933     }
934     idx++;
935     sprintf(buf, "dbgmd-%05d.%.10s", idx, suffix );
936     md->ctx->debug = fopen(buf, "w");
937     if( !md->ctx->debug )
938         log_debug("md debug: can't open %s\n", buf );
939 }
940
941 static void
942 md_stop_debug( GCRY_MD_HD md )
943 {
944     if( md->ctx->debug ) {
945         if( md->bufpos )
946             md_write( md, NULL, 0 );
947         fclose(md->ctx->debug);
948         md->ctx->debug = NULL;
949     }
950   #ifdef HAVE_U64_TYPEDEF
951     {  /* a kludge to pull in the __muldi3 for Solaris */
952        volatile u32 a = (u32)(ulong)md;
953        volatile u64 b = 42;
954        volatile u64 c;
955        c = a * b;
956     }
957   #endif
958 }
959
960
961
962 /****************
963  * Return information about the digest handle.
964  *  GCRYCTL_IS_SECURE:
965  *      Returns 1 when the handle works on secured memory
966  *      otherwise 0 is returned.  There is no error return.
967  *  GCRYCTL_IS_ALGO_ENABLED:
968  *     Returns 1 if the algo is enanled for that handle.
969  *     The algo must be passed as the address of an int.
970  */
971 int
972 gcry_md_info( GCRY_MD_HD h, int cmd, void *buffer, size_t *nbytes)
973 {
974
975     switch( cmd ) {
976       case GCRYCTL_IS_SECURE:
977         return h->ctx->secure;
978
979       case GCRYCTL_IS_ALGO_ENABLED:
980         {
981             int algo;
982             struct md_digest_list_s *r;
983
984             if (!buffer || (nbytes && *nbytes != sizeof (int))) {
985                 set_lasterr (GCRYERR_INV_ARG);
986                 return -1;
987             }
988             algo = *(int*)buffer;        
989             for(r=h->ctx->list; r; r = r->next ) {
990                 if( r->algo == algo )
991                     return 1;
992             }
993         }
994         break;
995
996       default:
997         set_lasterr( GCRYERR_INV_OP );
998         return -1;
999     }
1000     return 0;
1001 }
1002