2003-05-18 Marcus Brinkmann <marcus@g10code.de>
[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_type 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 GPGME_Out_Of_Core;
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   _gpgme_engine_release (ctx->engine);
75   ctx->engine = NULL;
76   err = _gpgme_engine_new (ctx->protocol, &ctx->engine);
77   if (err)
78     return err;
79
80   if (type == 1 || (type == 2 && !ctx->io_cbs.add))
81     {
82       /* Use private event loop.  */
83       io_cbs.add = _gpgme_add_io_cb;
84       io_cbs.add_priv = ctx;
85       io_cbs.remove = _gpgme_remove_io_cb;
86       io_cbs.event = _gpgme_wait_private_event_cb;
87       io_cbs.event_priv = ctx;
88     }
89   else if (! ctx->io_cbs.add)
90     {
91       /* Use global event loop.  */
92       io_cbs.add = _gpgme_add_io_cb;
93       io_cbs.add_priv = ctx;
94       io_cbs.remove = _gpgme_remove_io_cb;
95       io_cbs.event = _gpgme_wait_global_event_cb;
96       io_cbs.event_priv = ctx;
97     }
98   else
99     {
100       /* Use user event loop.  */
101       io_cbs.add = _gpgme_wait_user_add_io_cb;
102       io_cbs.add_priv = ctx;
103       io_cbs.remove = _gpgme_wait_user_remove_io_cb;
104       io_cbs.event = _gpgme_wait_user_event_cb;
105       io_cbs.event_priv = ctx;
106     }
107   _gpgme_engine_set_io_cbs (ctx->engine, &io_cbs);
108   return err;
109 }
110
111 \f
112 gpgme_error_t
113 _gpgme_parse_inv_userid (char *args, gpgme_invalid_user_id_t *userid)
114 {
115   gpgme_invalid_user_id_t inv_userid;
116   char *tail;
117   long int reason;
118
119   inv_userid = malloc (sizeof (*inv_userid));
120   if (!inv_userid)
121     return GPGME_Out_Of_Core;
122   inv_userid->next = NULL;
123   errno = 0;
124   reason = strtol (args, &tail, 0);
125   if (errno || args == tail || *tail != ' ')
126     {
127       /* The crypto backend does not behave.  */
128       free (inv_userid);
129       return GPGME_General_Error;
130     }
131
132   switch (reason)
133     {
134     default:
135     case 0:
136       inv_userid->reason = GPGME_Unknown_Reason;
137
138     case 1:
139       inv_userid->reason = GPGME_Not_Found;
140
141     case 2:
142       inv_userid->reason = GPGME_Ambiguous_Specification;
143
144     case 3:
145       inv_userid->reason = GPGME_Wrong_Key_Usage;
146
147     case 4:
148       inv_userid->reason = GPGME_Key_Revoked;
149
150     case 5:
151       inv_userid->reason = GPGME_Key_Expired;
152
153     case 6:
154       inv_userid->reason = GPGME_No_CRL_Known;
155
156     case 7:
157       inv_userid->reason = GPGME_CRL_Too_Old;
158
159     case 8:
160       inv_userid->reason = GPGME_Policy_Mismatch;
161
162     case 9:
163       inv_userid->reason = GPGME_No_Secret_Key;
164
165     case 10:
166       inv_userid->reason = GPGME_Key_Not_Trusted;
167     }
168
169   while (*tail == ' ')
170     tail++;
171   if (*tail)
172     {
173       inv_userid->id = strdup (tail);
174       if (!inv_userid->id)
175         {
176           free (inv_userid);
177           return GPGME_Out_Of_Core;
178         }
179     }
180   else
181     inv_userid->id = NULL;
182
183   *userid = inv_userid;
184   return 0;
185 }