doc/
[gpgme.git] / gpgme / op-support.c
1 /* op-support.c 
2    Copyright (C) 2002, 2003 g10 Code GmbH
3
4    This file is part of GPGME.
5  
6    GPGME is free software; you can redistribute it and/or modify it
7    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    GPGME is distributed in the hope that it will be useful, but
12    WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14    General Public License for more details.
15  
16    You should have received a copy of the GNU General Public License
17    along with GPGME; if not, write to the Free Software Foundation,
18    Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
19
20 #if HAVE_CONFIG_H
21 #include <config.h>
22 #endif
23 #include <stdlib.h>
24 #include <errno.h>
25 #include <string.h>
26
27 #include "gpgme.h"
28 #include "context.h"
29 #include "ops.h"
30
31 \f
32 gpgme_error_t
33 _gpgme_op_data_lookup (gpgme_ctx_t ctx, ctx_op_data_id_t type, void **hook,
34                        int size, void (*cleanup) (void *))
35 {
36   struct ctx_op_data *data = ctx->op_data;
37   while (data && data->type != type)
38     data = data->next;
39   if (!data)
40     {
41       if (size < 0)
42         {
43           *hook = NULL;
44           return 0;
45         }
46
47       data = calloc (1, sizeof (struct ctx_op_data) + size);
48       if (!data)
49         return gpg_error_from_errno (errno);
50       data->next = ctx->op_data;
51       data->type = type;
52       data->cleanup = cleanup;
53       data->hook = ((void *) data) + sizeof (struct ctx_op_data);
54       ctx->op_data = data;
55     }
56   *hook = data->hook;
57   return 0;
58 }
59
60
61 /* type is: 0: asynchronous operation (use global or user event loop).
62             1: synchronous operation (always use private event loop).
63             2: asynchronous private operation (use private or user
64             event loop).  */
65 gpgme_error_t
66 _gpgme_op_reset (gpgme_ctx_t ctx, int type)
67 {
68   gpgme_error_t err = 0;
69   struct gpgme_io_cbs io_cbs;
70
71   _gpgme_release_result (ctx);
72
73   /* Create an engine object.  */
74   if  (ctx->engine)
75     {
76       _gpgme_engine_release (ctx->engine);
77       ctx->engine = NULL;
78     }
79   err = _gpgme_engine_new (ctx->protocol, &ctx->engine,
80                            ctx->lc_ctype, ctx->lc_messages);
81   if (err)
82     return err;
83
84   if (type == 1 || (type == 2 && !ctx->io_cbs.add))
85     {
86       /* Use private event loop.  */
87       io_cbs.add = _gpgme_add_io_cb;
88       io_cbs.add_priv = ctx;
89       io_cbs.remove = _gpgme_remove_io_cb;
90       io_cbs.event = _gpgme_wait_private_event_cb;
91       io_cbs.event_priv = ctx;
92     }
93   else if (! ctx->io_cbs.add)
94     {
95       /* Use global event loop.  */
96       io_cbs.add = _gpgme_add_io_cb;
97       io_cbs.add_priv = ctx;
98       io_cbs.remove = _gpgme_remove_io_cb;
99       io_cbs.event = _gpgme_wait_global_event_cb;
100       io_cbs.event_priv = ctx;
101     }
102   else
103     {
104       /* Use user event loop.  */
105       io_cbs.add = _gpgme_wait_user_add_io_cb;
106       io_cbs.add_priv = ctx;
107       io_cbs.remove = _gpgme_wait_user_remove_io_cb;
108       io_cbs.event = _gpgme_wait_user_event_cb;
109       io_cbs.event_priv = ctx;
110     }
111   _gpgme_engine_set_io_cbs (ctx->engine, &io_cbs);
112   return err;
113 }
114
115 \f
116 gpgme_error_t
117 _gpgme_parse_inv_recp (char *args, gpgme_invalid_key_t *key)
118 {
119   gpgme_invalid_key_t inv_key;
120   char *tail;
121   long int reason;
122
123   inv_key = malloc (sizeof (*inv_key));
124   if (!inv_key)
125     return gpg_error_from_errno (errno);
126   inv_key->next = NULL;
127   errno = 0;
128   reason = strtol (args, &tail, 0);
129   if (errno || args == tail || *tail != ' ')
130     {
131       /* The crypto backend does not behave.  */
132       free (inv_key);
133       return gpg_error (GPG_ERR_INV_ENGINE);
134     }
135
136   switch (reason)
137     {
138     default:
139     case 0:
140       inv_key->reason = gpg_error (GPG_ERR_GENERAL);
141       break;
142
143     case 1:
144       inv_key->reason = gpg_error (GPG_ERR_NO_PUBKEY);
145       break;
146
147     case 2:
148       inv_key->reason = gpg_error (GPG_ERR_AMBIGUOUS_NAME);
149       break;
150
151     case 3:
152       inv_key->reason = gpg_error (GPG_ERR_WRONG_KEY_USAGE);
153       break;
154
155     case 4:
156       inv_key->reason = gpg_error (GPG_ERR_CERT_REVOKED);
157       break;
158
159     case 5:
160       inv_key->reason = gpg_error (GPG_ERR_CERT_EXPIRED);
161       break;
162
163     case 6:
164       inv_key->reason = gpg_error (GPG_ERR_NO_CRL_KNOWN);
165       break;
166
167     case 7:
168       inv_key->reason = gpg_error (GPG_ERR_CRL_TOO_OLD);
169       break;
170
171     case 8:
172       inv_key->reason = gpg_error (GPG_ERR_NO_POLICY_MATCH);
173       break;
174
175     case 9:
176       inv_key->reason = gpg_error (GPG_ERR_NO_SECKEY);
177       break;
178
179     case 10:
180       inv_key->reason = gpg_error (GPG_ERR_PUBKEY_NOT_TRUSTED);
181       break;
182     }
183
184   while (*tail == ' ')
185     tail++;
186   if (*tail)
187     {
188       inv_key->fpr = strdup (tail);
189       if (!inv_key->fpr)
190         {
191           int saved_errno = errno;
192           free (inv_key);
193           return gpg_error_from_errno (saved_errno);
194         }
195     }
196   else
197     inv_key->fpr = NULL;
198
199   *key = inv_key;
200   return 0;
201 }