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