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