Minor changes and typo fixes.
[gnupg.git] / g10 / call-agent.c
1 /* call-agent.c - divert operations to the agent
2  * Copyright (C) 2001, 2002, 2003, 2006 Free Software Foundation, Inc.
3  *
4  * This file is part of GnuPG.
5  *
6  * GnuPG 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  * GnuPG 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
19  * USA.
20  */
21
22 #if 0  /* let Emacs display a red warning */
23 #error fixme: this shares a lot of code with the file in ../sm
24 #endif
25
26 #include <config.h>
27 #include <stdio.h>
28 #include <stdlib.h>
29 #include <string.h>
30 #include <errno.h>
31 #include <unistd.h> 
32 #include <time.h>
33 #include <assert.h>
34 #ifdef HAVE_LOCALE_H
35 #include <locale.h>
36 #endif
37 #include <assuan.h>
38
39 #include "gpg.h"
40 #include "util.h"
41 #include "membuf.h"
42 #include "options.h"
43 #include "i18n.h"
44 #include "call-agent.h"
45
46 #ifndef DBG_ASSUAN
47 # define DBG_ASSUAN 1
48 #endif
49
50 static assuan_context_t agent_ctx = NULL;
51 static int force_pipe_server; 
52
53 struct cipher_parm_s 
54 {
55   assuan_context_t ctx;
56   const char *ciphertext;
57   size_t ciphertextlen;
58 };
59
60 struct writekey_parm_s
61 {
62   assuan_context_t ctx;
63   const unsigned char *keydata;
64   size_t keydatalen;
65 };
66
67 struct genkey_parm_s 
68 {
69   assuan_context_t ctx;
70   const char *sexp;
71   size_t sexplen;
72 };
73
74
75 \f
76 /* Try to connect to the agent via socket or fork it off and work by
77    pipes.  Handle the server's initial greeting */
78 static int
79 start_agent (void)
80 {
81   int rc = 0;
82   char *infostr, *p;
83   ASSUAN_CONTEXT ctx;
84   char *dft_display = NULL;
85   char *dft_ttyname = NULL;
86   char *dft_ttytype = NULL;
87   char *old_lc = NULL;
88   char *dft_lc = NULL;
89
90   if (agent_ctx)
91     return 0; /* fixme: We need a context for each thread or serialize
92                  the access to the agent. */
93
94   infostr = force_pipe_server? NULL : getenv ("GPG_AGENT_INFO");
95   if (!infostr || !*infostr)
96     {
97       const char *pgmname;
98       const char *argv[3];
99       int no_close_list[3];
100       int i;
101
102       if (opt.verbose)
103         log_info (_("no running gpg-agent - starting one\n"));
104
105       if (fflush (NULL))
106         {
107           gpg_error_t tmperr = gpg_error_from_errno (errno);
108           log_error ("error flushing pending output: %s\n", strerror (errno));
109           return tmperr;
110         }
111
112       if (!opt.agent_program || !*opt.agent_program)
113         opt.agent_program = GNUPG_DEFAULT_AGENT;
114       if ( !(pgmname = strrchr (opt.agent_program, '/')))
115         pgmname = opt.agent_program;
116       else
117         pgmname++;
118
119       argv[0] = pgmname;
120       argv[1] = "--server";
121       argv[2] = NULL;
122
123       i=0;
124       if (log_get_fd () != -1)
125         no_close_list[i++] = log_get_fd ();
126       no_close_list[i++] = fileno (stderr);
127       no_close_list[i] = -1;
128
129       /* connect to the agent and perform initial handshaking */
130       rc = assuan_pipe_connect (&ctx, opt.agent_program, (char**)argv,
131                                 no_close_list);
132     }
133   else
134     {
135       int prot;
136       int pid;
137
138       infostr = xstrdup (infostr);
139       if ( !(p = strchr (infostr, ':')) || p == infostr)
140         {
141           log_error (_("malformed GPG_AGENT_INFO environment variable\n"));
142           xfree (infostr);
143           force_pipe_server = 1;
144           return start_agent ();
145         }
146       *p++ = 0;
147       pid = atoi (p);
148       while (*p && *p != ':')
149         p++;
150       prot = *p? atoi (p+1) : 0;
151       if (prot != 1)
152         {
153           log_error (_("gpg-agent protocol version %d is not supported\n"),
154                      prot);
155           xfree (infostr);
156           force_pipe_server = 1;
157           return start_agent ();
158         }
159
160       rc = assuan_socket_connect (&ctx, infostr, pid);
161       xfree (infostr);
162       if (rc == ASSUAN_Connect_Failed)
163         {
164           log_error (_("can't connect to the agent - trying fall back\n"));
165           force_pipe_server = 1;
166           return start_agent ();
167         }
168     }
169
170   if (rc)
171     {
172       log_error ("can't connect to the agent: %s\n", assuan_strerror (rc));
173       return gpg_error (GPG_ERR_NO_AGENT);
174     }
175   agent_ctx = ctx;
176
177   if (DBG_ASSUAN)
178     log_debug ("connection to agent established\n");
179
180   rc = assuan_transact (agent_ctx, "RESET", NULL, NULL, NULL, NULL, NULL, NULL);
181   if (rc)
182     return map_assuan_err (rc);
183
184 #ifdef __GNUC__
185 #warning put this code into common/asshelp.c
186 #endif
187
188   dft_display = getenv ("DISPLAY");
189   if (opt.display || dft_display)
190     {
191       char *optstr;
192       if (asprintf (&optstr, "OPTION display=%s",
193                     opt.display ? opt.display : dft_display) < 0)
194         return gpg_error_from_errno (errno);
195       rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
196                             NULL);
197       free (optstr);
198       if (rc)
199         return map_assuan_err (rc);
200     }
201   if (!opt.ttyname)
202     {
203       dft_ttyname = getenv ("GPG_TTY");
204       if ((!dft_ttyname || !*dft_ttyname) && ttyname (0))
205         dft_ttyname = ttyname (0);
206     }
207   if (opt.ttyname || dft_ttyname)
208     {
209       char *optstr;
210       if (asprintf (&optstr, "OPTION ttyname=%s",
211                     opt.ttyname ? opt.ttyname : dft_ttyname) < 0)
212         return gpg_error_from_errno (errno);
213       rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
214                             NULL);
215       free (optstr);
216       if (rc)
217         return map_assuan_err (rc);
218     }
219   dft_ttytype = getenv ("TERM");
220   if (opt.ttytype || (dft_ttyname && dft_ttytype))
221     {
222       char *optstr;
223       if (asprintf (&optstr, "OPTION ttytype=%s",
224                     opt.ttyname ? opt.ttytype : dft_ttytype) < 0)
225         return gpg_error_from_errno (errno);
226       rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
227                             NULL);
228       free (optstr);
229       if (rc)
230         return map_assuan_err (rc);
231     }
232 #if defined(HAVE_SETLOCALE) && defined(LC_CTYPE)
233   old_lc = setlocale (LC_CTYPE, NULL);
234   if (old_lc)
235     {
236       old_lc = strdup (old_lc);
237       if (!old_lc)
238         return gpg_error_from_errno (errno);
239
240     }
241   dft_lc = setlocale (LC_CTYPE, "");
242 #endif
243   if (opt.lc_ctype || (dft_ttyname && dft_lc))
244     {
245       char *optstr;
246       if (asprintf (&optstr, "OPTION lc-ctype=%s",
247                     opt.lc_ctype ? opt.lc_ctype : dft_lc) < 0)
248         rc = gpg_error_from_errno (errno);
249       else
250         {
251           rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
252                                 NULL);
253           free (optstr);
254           if (rc)
255             rc = map_assuan_err (rc);
256         }
257     }
258 #if defined(HAVE_SETLOCALE) && defined(LC_CTYPE)
259   if (old_lc)
260     {
261       setlocale (LC_CTYPE, old_lc);
262       free (old_lc);
263     }
264 #endif
265   if (rc)
266     return rc;
267 #if defined(HAVE_SETLOCALE) && defined(LC_MESSAGES)
268   old_lc = setlocale (LC_MESSAGES, NULL);
269   if (old_lc)
270     {
271       old_lc = strdup (old_lc);
272       if (!old_lc)
273         return gpg_error_from_errno (errno);
274     }
275   dft_lc = setlocale (LC_MESSAGES, "");
276 #endif
277   if (opt.lc_messages || (dft_ttyname && dft_lc))
278     {
279       char *optstr;
280       if (asprintf (&optstr, "OPTION lc-messages=%s",
281                     opt.lc_messages ? opt.lc_messages : dft_lc) < 0)
282         rc = gpg_error_from_errno (errno);
283       else
284         {
285           rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
286                                 NULL);
287           free (optstr);
288           if (rc)
289             rc = map_assuan_err (rc);
290         }
291     }
292 #if defined(HAVE_SETLOCALE) && defined(LC_MESSAGES)
293   if (old_lc)
294     {
295       setlocale (LC_MESSAGES, old_lc);
296       free (old_lc);
297     }
298 #endif
299
300   return rc;
301 }
302
303
304 /* Return a new malloced string by unescaping the string S.  Escaping
305    is percent escaping and '+'/space mapping.  A binary nul will
306    silently be replaced by a 0xFF.  Function returns NULL to indicate
307    an out of memory status. */
308 static char *
309 unescape_status_string (const unsigned char *s)
310 {
311   char *buffer, *d;
312
313   buffer = d = xtrymalloc (strlen (s)+1);
314   if (!buffer)
315     return NULL;
316   while (*s)
317     {
318       if (*s == '%' && s[1] && s[2])
319         { 
320           s++;
321           *d = xtoi_2 (s);
322           if (!*d)
323             *d = '\xff';
324           d++;
325           s += 2;
326         }
327       else if (*s == '+')
328         {
329           *d++ = ' ';
330           s++;
331         }
332       else
333         *d++ = *s++;
334     }
335   *d = 0; 
336   return buffer;
337 }
338
339 /* Take a 20 byte hexencoded string and put it into the the provided
340    20 byte buffer FPR in binary format. */
341 static int
342 unhexify_fpr (const char *hexstr, unsigned char *fpr)
343 {
344   const char *s;
345   int n;
346
347   for (s=hexstr, n=0; hexdigitp (s); s++, n++)
348     ;
349   if (*s || (n != 40))
350     return 0; /* no fingerprint (invalid or wrong length). */
351   n /= 2;
352   for (s=hexstr, n=0; *s; s += 2, n++)
353     fpr[n] = xtoi_2 (s);
354   return 1; /* okay */
355 }
356
357 /* Take the serial number from LINE and return it verbatim in a newly
358    allocated string.  We make sure that only hex characters are
359    returned. */
360 static char *
361 store_serialno (const char *line)
362 {
363   const char *s;
364   char *p;
365
366   for (s=line; hexdigitp (s); s++)
367     ;
368   p = xtrymalloc (s + 1 - line);
369   if (p)
370     {
371       memcpy (p, line, s-line);
372       p[s-line] = 0;
373     }
374   return p;
375 }
376
377
378 \f
379 #if 0
380 /* Handle a KEYPARMS inquiry.  Note, we only send the data,
381    assuan_transact takes care of flushing and writing the end */
382 static AssuanError
383 inq_genkey_parms (void *opaque, const char *keyword)
384 {
385   struct genkey_parm_s *parm = opaque; 
386   AssuanError rc;
387
388   rc = assuan_send_data (parm->ctx, parm->sexp, parm->sexplen);
389   return rc; 
390 }
391
392
393 \f
394 /* Call the agent to generate a new key */
395 int
396 agent_genkey (KsbaConstSexp keyparms, KsbaSexp *r_pubkey)
397 {
398   int rc;
399   struct genkey_parm_s gk_parm;
400   membuf_t data;
401   size_t len;
402   char *buf;
403
404   *r_pubkey = NULL;
405   rc = start_agent ();
406   if (rc)
407     return rc;
408
409   rc = assuan_transact (agent_ctx, "RESET", NULL, NULL,
410                         NULL, NULL, NULL, NULL);
411   if (rc)
412     return map_assuan_err (rc);
413
414   init_membuf (&data, 1024);
415   gk_parm.ctx = agent_ctx;
416   gk_parm.sexp = keyparms;
417   gk_parm.sexplen = gcry_sexp_canon_len (keyparms, 0, NULL, NULL);
418   if (!gk_parm.sexplen)
419     return gpg_error (GPG_ERR_INV_VALUE);
420   rc = assuan_transact (agent_ctx, "GENKEY",
421                         membuf_data_cb, &data, 
422                         inq_genkey_parms, &gk_parm, NULL, NULL);
423   if (rc)
424     {
425       xfree (get_membuf (&data, &len));
426       return map_assuan_err (rc);
427     }
428   buf = get_membuf (&data, &len);
429   if (!buf)
430     return gpg_error (GPG_ERR_ENOMEM);
431   if (!gcry_sexp_canon_len (buf, len, NULL, NULL))
432     {
433       xfree (buf);
434       return gpg_error (GPG_ERR_INV_SEXP);
435     }
436   *r_pubkey = buf;
437   return 0;
438 }
439 #endif /*0*/
440
441
442 \f
443 /* Ask the agent whether the corresponding secret key is available for
444    the given keygrip. */
445 int
446 agent_havekey (const char *hexkeygrip)
447 {
448   int rc;
449   char line[ASSUAN_LINELENGTH];
450
451   rc = start_agent ();
452   if (rc)
453     return rc;
454
455   if (!hexkeygrip || strlen (hexkeygrip) != 40)
456     return gpg_error (GPG_ERR_INV_VALUE);
457
458   snprintf (line, DIM(line)-1, "HAVEKEY %s", hexkeygrip);
459   line[DIM(line)-1] = 0;
460
461   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
462   return map_assuan_err (rc);
463 }
464
465 \f
466 /* Release the card info structure INFO. */
467 void
468 agent_release_card_info (struct agent_card_info_s *info)
469 {
470   if (!info)
471     return;
472
473   xfree (info->serialno); info->serialno = NULL;
474   xfree (info->disp_name); info->disp_name = NULL;
475   xfree (info->disp_lang); info->disp_lang = NULL;
476   xfree (info->pubkey_url); info->pubkey_url = NULL;
477   xfree (info->login_data); info->login_data = NULL;
478   info->cafpr1valid = info->cafpr2valid = info->cafpr3valid = 0;
479   info->fpr1valid = info->fpr2valid = info->fpr3valid = 0;
480 }
481
482 static AssuanError
483 learn_status_cb (void *opaque, const char *line)
484 {
485   struct agent_card_info_s *parm = opaque;
486   const char *keyword = line;
487   int keywordlen;
488   int i;
489
490   for (keywordlen=0; *line && !spacep (line); line++, keywordlen++)
491     ;
492   while (spacep (line))
493     line++;
494
495   if (keywordlen == 8 && !memcmp (keyword, "SERIALNO", keywordlen))
496     {
497       xfree (parm->serialno);
498       parm->serialno = store_serialno (line);
499     }
500   else if (keywordlen == 9 && !memcmp (keyword, "DISP-NAME", keywordlen))
501     {
502       xfree (parm->disp_name);
503       parm->disp_name = unescape_status_string (line);
504     }
505   else if (keywordlen == 9 && !memcmp (keyword, "DISP-LANG", keywordlen))
506     {
507       xfree (parm->disp_lang);
508       parm->disp_lang = unescape_status_string (line);
509     }
510   else if (keywordlen == 8 && !memcmp (keyword, "DISP-SEX", keywordlen))
511     {
512       parm->disp_sex = *line == '1'? 1 : *line == '2' ? 2: 0;
513     }
514   else if (keywordlen == 10 && !memcmp (keyword, "PUBKEY-URL", keywordlen))
515     {
516       xfree (parm->pubkey_url);
517       parm->pubkey_url = unescape_status_string (line);
518     }
519   else if (keywordlen == 10 && !memcmp (keyword, "LOGIN-DATA", keywordlen))
520     {
521       xfree (parm->login_data);
522       parm->login_data = unescape_status_string (line);
523     }
524   else if (keywordlen == 11 && !memcmp (keyword, "SIG-COUNTER", keywordlen))
525     {
526       parm->sig_counter = strtoul (line, NULL, 0);
527     }
528   else if (keywordlen == 10 && !memcmp (keyword, "CHV-STATUS", keywordlen))
529     {
530       char *p, *buf;
531
532       buf = p = unescape_status_string (line);
533       if (buf)
534         {
535           while (spacep (p))
536             p++;
537           parm->chv1_cached = atoi (p);
538           while (*p && !spacep (p))
539             p++;
540           while (spacep (p))
541             p++;
542           for (i=0; *p && i < 3; i++)
543             {
544               parm->chvmaxlen[i] = atoi (p);
545               while (*p && !spacep (p))
546                 p++;
547               while (spacep (p))
548                 p++;
549             }
550           for (i=0; *p && i < 3; i++)
551             {
552               parm->chvretry[i] = atoi (p);
553               while (*p && !spacep (p))
554                 p++;
555               while (spacep (p))
556                 p++;
557             }
558           xfree (buf);
559         }
560     }
561   else if (keywordlen == 7 && !memcmp (keyword, "KEY-FPR", keywordlen))
562     {
563       int no = atoi (line);
564       while (*line && !spacep (line))
565         line++;
566       while (spacep (line))
567         line++;
568       if (no == 1)
569         parm->fpr1valid = unhexify_fpr (line, parm->fpr1);
570       else if (no == 2)
571         parm->fpr2valid = unhexify_fpr (line, parm->fpr2);
572       else if (no == 3)
573         parm->fpr3valid = unhexify_fpr (line, parm->fpr3);
574     }
575   else if (keywordlen == 6 && !memcmp (keyword, "CA-FPR", keywordlen))
576     {
577       int no = atoi (line);
578       while (*line && !spacep (line))
579         line++;
580       while (spacep (line))
581         line++;
582       if (no == 1)
583         parm->cafpr1valid = unhexify_fpr (line, parm->cafpr1);
584       else if (no == 2)
585         parm->cafpr2valid = unhexify_fpr (line, parm->cafpr2);
586       else if (no == 3)
587         parm->cafpr3valid = unhexify_fpr (line, parm->cafpr3);
588     }
589   
590   return 0;
591 }
592
593 /* Call the agent to learn about a smartcard */
594 int
595 agent_learn (struct agent_card_info_s *info)
596 {
597   int rc;
598
599   rc = start_agent ();
600   if (rc)
601     return rc;
602
603   memset (info, 0, sizeof *info);
604   rc = assuan_transact (agent_ctx, "LEARN --send",
605                         NULL, NULL, NULL, NULL,
606                         learn_status_cb, info);
607   
608   return map_assuan_err (rc);
609 }
610
611 /* Call the agent to retrieve a data object.  This function returns
612    the data in the same structure as used by the learn command.  It is
613    allowed to update such a structure using this commmand. */
614 int
615 agent_scd_getattr (const char *name, struct agent_card_info_s *info)
616 {
617   int rc;
618   char line[ASSUAN_LINELENGTH];
619
620   if (!*name)
621     return gpg_error (GPG_ERR_INV_VALUE);
622
623   /* We assume that NAME does not need escaping. */
624   if (12 + strlen (name) > DIM(line)-1)
625     return gpg_error (GPG_ERR_TOO_LARGE);
626   stpcpy (stpcpy (line, "SCD GETATTR "), name); 
627
628   rc = start_agent ();
629   if (rc)
630     return rc;
631
632   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL,
633                         learn_status_cb, info);
634   
635   return map_assuan_err (rc);
636 }
637
638 \f
639 /* Send an setattr command to the SCdaemon.  SERIALNO is not actually
640    used here but required by gpg 1.4's implementation of this code in
641    cardglue.c. */
642 int
643 agent_scd_setattr (const char *name,
644                    const unsigned char *value, size_t valuelen,
645                    const char *serialno)
646 {
647   int rc;
648   char line[ASSUAN_LINELENGTH];
649   char *p;
650
651   if (!*name || !valuelen)
652     return gpg_error (GPG_ERR_INV_VALUE);
653
654   /* We assume that NAME does not need escaping. */
655   if (12 + strlen (name) > DIM(line)-1)
656     return gpg_error (GPG_ERR_TOO_LARGE);
657       
658   p = stpcpy (stpcpy (line, "SCD SETATTR "), name); 
659   *p++ = ' ';
660   for (; valuelen; value++, valuelen--)
661     {
662       if (p >= line + DIM(line)-5 )
663         return gpg_error (GPG_ERR_TOO_LARGE);
664       if (*value < ' ' || *value == '+' || *value == '%')
665         {
666           sprintf (p, "%%%02X", *value);
667           p += 3;
668         }
669       else if (*value == ' ')
670         *p++ = '+';
671       else
672         *p++ = *value;
673     }
674   *p = 0;
675
676   rc = start_agent ();
677   if (rc)
678     return rc;
679
680   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
681   return map_assuan_err (rc);
682 }
683
684
685 \f
686 /* Handle a KEYDATA inquiry.  Note, we only send the data,
687    assuan_transact takes care of flushing and writing the end */
688 static assuan_error_t
689 inq_writekey_parms (void *opaque, const char *keyword)
690 {
691   struct writekey_parm_s *parm = opaque; 
692
693   return assuan_send_data (parm->ctx, parm->keydata, parm->keydatalen);
694 }
695
696
697 /* Send a WRITEKEY command to the SCdaemon. */
698 int 
699 agent_scd_writekey (int keyno, const char *serialno,
700                     const unsigned char *keydata, size_t keydatalen)
701 {
702   int rc;
703   char line[ASSUAN_LINELENGTH];
704   struct writekey_parm_s parms;
705
706   rc = start_agent ();
707   if (rc)
708     return rc;
709
710   memset (&parms, 0, sizeof parms);
711
712   snprintf (line, DIM(line)-1, "SCD WRITEKEY --force OPENPGP.%d", keyno);
713   line[DIM(line)-1] = 0;
714   parms.ctx = agent_ctx;
715   parms.keydata = keydata;
716   parms.keydatalen = keydatalen;
717   
718   rc = assuan_transact (agent_ctx, line, NULL, NULL,
719                         inq_writekey_parms, &parms, NULL, NULL);
720
721   return map_assuan_err (rc);
722 }
723
724
725
726 \f
727 /* Status callback for the SCD GENKEY command. */
728 static AssuanError
729 scd_genkey_cb (void *opaque, const char *line)
730 {
731   struct agent_card_genkey_s *parm = opaque;
732   const char *keyword = line;
733   int keywordlen;
734   gpg_error_t rc;
735
736   log_debug ("got status line `%s'\n", line);
737   for (keywordlen=0; *line && !spacep (line); line++, keywordlen++)
738     ;
739   while (spacep (line))
740     line++;
741
742   if (keywordlen == 7 && !memcmp (keyword, "KEY-FPR", keywordlen))
743     {
744       parm->fprvalid = unhexify_fpr (line, parm->fpr);
745     }
746   if (keywordlen == 8 && !memcmp (keyword, "KEY-DATA", keywordlen))
747     {
748       gcry_mpi_t a;
749       const char *name = line;
750
751       while (*line && !spacep (line))
752         line++;
753       while (spacep (line))
754         line++;
755
756       rc = gcry_mpi_scan (&a, GCRYMPI_FMT_HEX, line, 0, NULL);
757       if (rc)
758         log_error ("error parsing received key data: %s\n", gpg_strerror (rc));
759       else if (*name == 'n' && spacep (name+1))
760         parm->n = a;
761       else if (*name == 'e' && spacep (name+1))
762         parm->e = a;
763       else
764         {
765           log_info ("unknown parameter name in received key data\n");
766           gcry_mpi_release (a);
767         }
768     }
769   else if (keywordlen == 14 && !memcmp (keyword,"KEY-CREATED-AT", keywordlen))
770     {
771       parm->created_at = (u32)strtoul (line, NULL, 10);
772     }
773
774   return 0;
775 }
776
777 /* Send a GENKEY command to the SCdaemon.  SERIALNO is not used in
778    this implementation. */
779 int
780 agent_scd_genkey (struct agent_card_genkey_s *info, int keyno, int force,
781                   const char *serialno)
782 {
783   int rc;
784   char line[ASSUAN_LINELENGTH];
785
786   rc = start_agent ();
787   if (rc)
788     return rc;
789
790   memset (info, 0, sizeof *info);
791   snprintf (line, DIM(line)-1, "SCD GENKEY %s%d",
792             force? "--force ":"", keyno);
793   line[DIM(line)-1] = 0;
794
795   memset (info, 0, sizeof *info);
796   rc = assuan_transact (agent_ctx, line,
797                         NULL, NULL, NULL, NULL,
798                         scd_genkey_cb, info);
799   
800   return map_assuan_err (rc);
801 }
802
803 \f
804 static AssuanError
805 membuf_data_cb (void *opaque, const void *buffer, size_t length)
806 {
807   membuf_t *data = opaque;
808
809   if (buffer)
810     put_membuf (data, buffer, length);
811   return 0;
812 }
813   
814 /* Send a sign command to the scdaemon via gpg-agent's pass thru
815    mechanism. */
816 int
817 agent_scd_pksign (const char *serialno, int hashalgo,
818                   const unsigned char *indata, size_t indatalen,
819                   unsigned char **r_buf, size_t *r_buflen)
820 {
821   int rc, i;
822   char *p, line[ASSUAN_LINELENGTH];
823   membuf_t data;
824   size_t len;
825
826   /* Note, hashalgo is not yet used but hardwired to SHA1 in SCdaemon. */
827
828   *r_buf = NULL;
829   *r_buflen = 0;
830
831   rc = start_agent ();
832   if (rc)
833     return rc;
834
835   if (indatalen*2 + 50 > DIM(line))
836     return gpg_error (GPG_ERR_GENERAL);
837
838   sprintf (line, "SCD SETDATA ");
839   p = line + strlen (line);
840   for (i=0; i < indatalen ; i++, p += 2 )
841     sprintf (p, "%02X", indata[i]);
842   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
843   if (rc)
844     return map_assuan_err (rc);
845
846   init_membuf (&data, 1024);
847 #if 0
848   if (!hashalgo) /* Temporary test hack. */
849     snprintf (line, DIM(line)-1, "SCD PKAUTH %s", serialno);
850   else
851 #endif
852    snprintf (line, DIM(line)-1, "SCD PKSIGN %s", serialno);
853   line[DIM(line)-1] = 0;
854   rc = assuan_transact (agent_ctx, line, membuf_data_cb, &data,
855                         NULL, NULL, NULL, NULL);
856   if (rc)
857     {
858       xfree (get_membuf (&data, &len));
859       return map_assuan_err (rc);
860     }
861   *r_buf = get_membuf (&data, r_buflen);
862
863   return 0;
864 }
865
866
867 /* Decrypt INDATA of length INDATALEN using the card identified by
868    SERIALNO.  Return the plaintext in a nwly allocated buffer stored
869    at the address of R_BUF. 
870
871    Note, we currently support only RSA or more exactly algorithms
872    taking one input data element. */
873 int
874 agent_scd_pkdecrypt (const char *serialno,
875                      const unsigned char *indata, size_t indatalen,
876                      unsigned char **r_buf, size_t *r_buflen)
877 {
878   int rc, i;
879   char *p, line[ASSUAN_LINELENGTH];
880   membuf_t data;
881   size_t len;
882
883   *r_buf = NULL;
884   rc = start_agent ();
885   if (rc)
886     return rc;
887
888   /* FIXME: use secure memory where appropriate */
889   if (indatalen*2 + 50 > DIM(line))
890     return gpg_error (GPG_ERR_GENERAL);
891
892   sprintf (line, "SCD SETDATA ");
893   p = line + strlen (line);
894   for (i=0; i < indatalen ; i++, p += 2 )
895     sprintf (p, "%02X", indata[i]);
896   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
897   if (rc)
898     return map_assuan_err (rc);
899
900   init_membuf (&data, 1024);
901   snprintf (line, DIM(line)-1, "SCD PKDECRYPT %s", serialno);
902   line[DIM(line)-1] = 0;
903   rc = assuan_transact (agent_ctx, line,
904                         membuf_data_cb, &data,
905                         NULL, NULL, NULL, NULL);
906   if (rc)
907     {
908       xfree (get_membuf (&data, &len));
909       return map_assuan_err (rc);
910     }
911   *r_buf = get_membuf (&data, r_buflen);
912   if (!*r_buf)
913     return gpg_error (GPG_ERR_ENOMEM);
914
915   return 0;
916 }
917
918
919 /* Change the PIN of an OpenPGP card or reset the retry counter.
920    CHVNO 1: Change the PIN
921          2: Same as 1
922          3: Change the admin PIN
923        101: Set a new PIN and reset the retry counter
924        102: Same as 101
925    SERIALNO is not used.
926  */
927 int
928 agent_scd_change_pin (int chvno, const char *serialno)
929 {
930   int rc;
931   char line[ASSUAN_LINELENGTH];
932   const char *reset = "";
933
934   if (chvno >= 100)
935     reset = "--reset";
936   chvno %= 100;
937
938   rc = start_agent ();
939   if (rc)
940     return rc;
941
942   snprintf (line, DIM(line)-1, "SCD PASSWD %s %d", reset, chvno);
943   line[DIM(line)-1] = 0;
944   rc = assuan_transact (agent_ctx, line, NULL, NULL,
945                         NULL, NULL, NULL, NULL);
946   return map_assuan_err (rc);
947 }
948
949
950 /* Perform a CHECKPIN operation.  SERIALNO should be the serial
951    number of the card - optionally followed by the fingerprint;
952    however the fingerprint is ignored here. */
953 int
954 agent_scd_checkpin  (const char *serialno)
955 {
956   int rc;
957   char line[ASSUAN_LINELENGTH];
958
959   rc = start_agent ();
960   if (rc)
961     return rc;
962
963   snprintf (line, DIM(line)-1, "SCD CHECKPIN %s", serialno);
964   line[DIM(line)-1] = 0;
965   return assuan_transact (agent_ctx, line,
966                           NULL, NULL,
967                           NULL, NULL, NULL, NULL);
968 }
969
970
971 /* Dummy function, only used by the gpg 1.4 implementation. */
972 void
973 agent_clear_pin_cache (const char *sn)
974 {
975
976 }