ChangeLog:
[libgcrypt.git] / tests / keygen.c
1 /* keygen.c  -  key generation regression tests
2  *      Copyright (C) 2003, 2005 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 #ifdef HAVE_CONFIG_H
22 #include <config.h>
23 #endif
24 #include <stdio.h>
25 #include <stdlib.h>
26 #include <string.h>
27 #include <stdarg.h>
28 #include "../src/gcrypt.h"
29
30
31
32 static int verbose;
33 static int error_count;
34
35 static void
36 fail ( const char *format, ... )
37 {
38     va_list arg_ptr ;
39
40     va_start( arg_ptr, format ) ;
41     vfprintf (stderr, format, arg_ptr );
42     va_end(arg_ptr);
43     error_count++;
44 }
45
46 static void
47 die ( const char *format, ... )
48 {
49     va_list arg_ptr ;
50
51     va_start( arg_ptr, format ) ;
52     vfprintf (stderr, format, arg_ptr );
53     va_end(arg_ptr);
54     exit (1);
55 }
56
57
58 static void
59 print_mpi (const char *text, gcry_mpi_t a)
60 {
61   char *buf;
62   void *bufaddr = &buf;
63   gcry_error_t rc;
64
65   rc = gcry_mpi_aprint (GCRYMPI_FMT_HEX, bufaddr, NULL, a);
66   if (rc)
67     fprintf (stderr, "%s=[error printing number: %s]\n",
68              text, gpg_strerror (rc));
69   else
70     {
71       fprintf (stderr, "%s=0x%s\n", text, buf);
72       gcry_free (buf);
73     }
74 }
75
76
77 static void
78 check_generated_rsa_key (gcry_sexp_t key, unsigned long expected_e)
79 {
80   gcry_sexp_t skey, pkey, list;
81
82   pkey = gcry_sexp_find_token (key, "public-key", 0);
83   if (!pkey)
84     fail ("public part missing in return value\n");
85   else
86     {
87       gcry_mpi_t e = NULL;
88
89       list = gcry_sexp_find_token (pkey, "e", 0);
90       if (!list || !(e=gcry_sexp_nth_mpi (list, 1, 0)) )
91         fail ("public exponent not found\n");
92       else if (!expected_e)
93         {
94           if (verbose)
95             print_mpi ("e", e);
96         }
97       else if ( gcry_mpi_cmp_ui (e, expected_e)) 
98         {
99           print_mpi ("e", e);
100           fail ("public exponent is not %lu\n", expected_e);
101         }
102       gcry_sexp_release (list);
103       gcry_mpi_release (e);
104       gcry_sexp_release (pkey);
105     }
106  
107   skey = gcry_sexp_find_token (key, "private-key", 0);
108   if (!skey)
109     fail ("private part missing in return value\n");
110   else
111     {
112       int rc = gcry_pk_testkey (skey);
113       if (rc)
114         fail ("gcry_pk_testkey failed: %s\n", gpg_strerror (rc));
115       gcry_sexp_release (skey);
116     }
117
118  }
119
120 static void
121 check_rsa_keys (void)
122 {
123   gcry_sexp_t keyparm, key;
124   int rc;
125
126   if (verbose)
127     fprintf (stderr, "creating 1024 bit RSA key using old interface\n");
128   rc = gcry_sexp_new (&keyparm, 
129                       "(genkey\n"
130                       " (rsa\n"
131                       "  (nbits 4:1024)\n"
132                       " ))", 0, 1);
133   if (rc)
134     die ("error creating S-expression: %s\n", gpg_strerror (rc));
135   rc = gcry_pk_genkey (&key, keyparm);
136   gcry_sexp_release (keyparm);
137   if (rc)
138     die ("error generating RSA key: %s\n", gpg_strerror (rc));
139
140   check_generated_rsa_key (key, 65537);
141   gcry_sexp_release (key);
142
143   if (verbose)
144     fprintf (stderr, "creating 512 bit RSA key with e=257\n");
145   rc = gcry_sexp_new (&keyparm, 
146                       "(genkey\n"
147                       " (rsa\n"
148                       "  (nbits 3:512)\n"
149                       "  (rsa-use-e 3:257)\n"
150                       " ))", 0, 1);
151   if (rc)
152     die ("error creating S-expression: %s\n", gpg_strerror (rc));
153   rc = gcry_pk_genkey (&key, keyparm);
154   gcry_sexp_release (keyparm);
155   if (rc)
156     die ("error generating RSA key: %s\n", gpg_strerror (rc));
157
158   check_generated_rsa_key (key, 257);
159   gcry_sexp_release (key);
160
161   if (verbose)
162     fprintf (stderr, "creating 512 bit RSA key with default e\n");
163   rc = gcry_sexp_new (&keyparm, 
164                       "(genkey\n"
165                       " (rsa\n"
166                       "  (nbits 3:512)\n"
167                       "  (rsa-use-e 1:0)\n"
168                       " ))", 0, 1);
169   if (rc)
170     die ("error creating S-expression: %s\n", gpg_strerror (rc));
171   rc = gcry_pk_genkey (&key, keyparm);
172   gcry_sexp_release (keyparm);
173   if (rc)
174     die ("error generating RSA key: %s\n", gpg_strerror (rc));
175
176   check_generated_rsa_key (key, 0); /* We don't expect a constant exponent. */
177   gcry_sexp_release (key);
178
179 }
180
181
182 static void
183 check_nonce (void)
184 {
185   char a[32], b[32];
186   int i,j;
187   int oops=0;
188
189   if (verbose)
190     fprintf (stderr, "checking gcry_create_nonce\n");
191
192   gcry_create_nonce (a, sizeof a);
193   for (i=0; i < 10; i++)
194     {
195       gcry_create_nonce (b, sizeof b);
196       if (!memcmp (a, b, sizeof a))
197         die ("identical nounce found\n");
198     }
199   for (i=0; i < 10; i++)
200     {
201       gcry_create_nonce (a, sizeof a);
202       if (!memcmp (a, b, sizeof a))
203         die ("identical nounce found\n");
204     }
205
206  again:
207   for (i=1,j=0; i < sizeof a; i++)
208     if (a[0] == a[i])
209       j++;
210   if (j+1 == sizeof (a))
211     {
212       if (oops)
213         die ("impossible nonce found\n");
214       oops++;
215       gcry_create_nonce (a, sizeof a);
216       goto again;
217     }
218 }
219
220 int
221 main (int argc, char **argv)
222 {
223   int debug = 0;
224
225   if (argc > 1 && !strcmp (argv[1], "--verbose"))
226     verbose = 1;
227   else if (argc > 1 && !strcmp (argv[1], "--debug"))
228     verbose = debug = 1;
229
230   if (!gcry_check_version (GCRYPT_VERSION))
231     die ("version mismatch\n");
232   gcry_control (GCRYCTL_DISABLE_SECMEM, 0);
233   gcry_control (GCRYCTL_INITIALIZATION_FINISHED, 0);
234   if (debug)
235     gcry_control (GCRYCTL_SET_DEBUG_FLAGS, 1u , 0);
236   /* No valuable keys are create, so we can speed up our RNG. */
237   gcry_control (GCRYCTL_ENABLE_QUICK_RANDOM, 0);
238
239   check_rsa_keys ();
240   check_nonce ();
241   
242   return error_count? 1:0;
243 }
244