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