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