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