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   if (err)
81     return err;
82
83   if (type == 1 || (type == 2 && !ctx->io_cbs.add))
84     {
85       /* Use private event loop.  */
86       io_cbs.add = _gpgme_add_io_cb;
87       io_cbs.add_priv = ctx;
88       io_cbs.remove = _gpgme_remove_io_cb;
89       io_cbs.event = _gpgme_wait_private_event_cb;
90       io_cbs.event_priv = ctx;
91     }
92   else if (! ctx->io_cbs.add)
93     {
94       /* Use global event loop.  */
95       io_cbs.add = _gpgme_add_io_cb;
96       io_cbs.add_priv = ctx;
97       io_cbs.remove = _gpgme_remove_io_cb;
98       io_cbs.event = _gpgme_wait_global_event_cb;
99       io_cbs.event_priv = ctx;
100     }
101   else
102     {
103       /* Use user event loop.  */
104       io_cbs.add = _gpgme_wait_user_add_io_cb;
105       io_cbs.add_priv = ctx;
106       io_cbs.remove = _gpgme_wait_user_remove_io_cb;
107       io_cbs.event = _gpgme_wait_user_event_cb;
108       io_cbs.event_priv = ctx;
109     }
110   _gpgme_engine_set_io_cbs (ctx->engine, &io_cbs);
111   return err;
112 }
113
114 \f
115 gpgme_error_t
116 _gpgme_parse_inv_recp (char *args, gpgme_invalid_key_t *key)
117 {
118   gpgme_invalid_key_t inv_key;
119   char *tail;
120   long int reason;
121
122   inv_key = malloc (sizeof (*inv_key));
123   if (!inv_key)
124     return gpg_error_from_errno (errno);
125   inv_key->next = NULL;
126   errno = 0;
127   reason = strtol (args, &tail, 0);
128   if (errno || args == tail || *tail != ' ')
129     {
130       /* The crypto backend does not behave.  */
131       free (inv_key);
132       return gpg_error (GPG_ERR_INV_ENGINE);
133     }
134
135   switch (reason)
136     {
137     default:
138     case 0:
139       inv_key->reason = gpg_error (GPG_ERR_GENERAL);
140       break;
141
142     case 1:
143       inv_key->reason = gpg_error (GPG_ERR_NO_PUBKEY);
144       break;
145
146     case 2:
147       inv_key->reason = gpg_error (GPG_ERR_AMBIGUOUS_NAME);
148       break;
149
150     case 3:
151       inv_key->reason = gpg_error (GPG_ERR_WRONG_KEY_USAGE);
152       break;
153
154     case 4:
155       inv_key->reason = gpg_error (GPG_ERR_CERT_REVOKED);
156       break;
157
158     case 5:
159       inv_key->reason = gpg_error (GPG_ERR_CERT_EXPIRED);
160       break;
161
162     case 6:
163       inv_key->reason = gpg_error (GPG_ERR_NO_CRL_KNOWN);
164       break;
165
166     case 7:
167       inv_key->reason = gpg_error (GPG_ERR_CRL_TOO_OLD);
168       break;
169
170     case 8:
171       inv_key->reason = gpg_error (GPG_ERR_NO_POLICY_MATCH);
172       break;
173
174     case 9:
175       inv_key->reason = gpg_error (GPG_ERR_NO_SECKEY);
176       break;
177
178     case 10:
179       inv_key->reason = gpg_error (GPG_ERR_PUBKEY_NOT_TRUSTED);
180       break;
181     }
182
183   while (*tail == ' ')
184     tail++;
185   if (*tail)
186     {
187       inv_key->fpr = strdup (tail);
188       if (!inv_key->fpr)
189         {
190           int saved_errno = errno;
191           free (inv_key);
192           return gpg_error_from_errno (saved_errno);
193         }
194     }
195   else
196     inv_key->fpr = NULL;
197
198   *key = inv_key;
199   return 0;
200 }