sl@0
|
1 |
/* crypto/engine/eng_dyn.c */
|
sl@0
|
2 |
/* Written by Geoff Thorpe (geoff@geoffthorpe.net) for the OpenSSL
|
sl@0
|
3 |
* project 2001.
|
sl@0
|
4 |
*/
|
sl@0
|
5 |
/* ====================================================================
|
sl@0
|
6 |
* Copyright (c) 1999-2001 The OpenSSL Project. All rights reserved.
|
sl@0
|
7 |
*
|
sl@0
|
8 |
* Redistribution and use in source and binary forms, with or without
|
sl@0
|
9 |
* modification, are permitted provided that the following conditions
|
sl@0
|
10 |
* are met:
|
sl@0
|
11 |
*
|
sl@0
|
12 |
* 1. Redistributions of source code must retain the above copyright
|
sl@0
|
13 |
* notice, this list of conditions and the following disclaimer.
|
sl@0
|
14 |
*
|
sl@0
|
15 |
* 2. Redistributions in binary form must reproduce the above copyright
|
sl@0
|
16 |
* notice, this list of conditions and the following disclaimer in
|
sl@0
|
17 |
* the documentation and/or other materials provided with the
|
sl@0
|
18 |
* distribution.
|
sl@0
|
19 |
*
|
sl@0
|
20 |
* 3. All advertising materials mentioning features or use of this
|
sl@0
|
21 |
* software must display the following acknowledgment:
|
sl@0
|
22 |
* "This product includes software developed by the OpenSSL Project
|
sl@0
|
23 |
* for use in the OpenSSL Toolkit. (http://www.OpenSSL.org/)"
|
sl@0
|
24 |
*
|
sl@0
|
25 |
* 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to
|
sl@0
|
26 |
* endorse or promote products derived from this software without
|
sl@0
|
27 |
* prior written permission. For written permission, please contact
|
sl@0
|
28 |
* licensing@OpenSSL.org.
|
sl@0
|
29 |
*
|
sl@0
|
30 |
* 5. Products derived from this software may not be called "OpenSSL"
|
sl@0
|
31 |
* nor may "OpenSSL" appear in their names without prior written
|
sl@0
|
32 |
* permission of the OpenSSL Project.
|
sl@0
|
33 |
*
|
sl@0
|
34 |
* 6. Redistributions of any form whatsoever must retain the following
|
sl@0
|
35 |
* acknowledgment:
|
sl@0
|
36 |
* "This product includes software developed by the OpenSSL Project
|
sl@0
|
37 |
* for use in the OpenSSL Toolkit (http://www.OpenSSL.org/)"
|
sl@0
|
38 |
*
|
sl@0
|
39 |
* THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY
|
sl@0
|
40 |
* EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
sl@0
|
41 |
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
sl@0
|
42 |
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE OpenSSL PROJECT OR
|
sl@0
|
43 |
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
sl@0
|
44 |
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
sl@0
|
45 |
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
sl@0
|
46 |
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
sl@0
|
47 |
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
sl@0
|
48 |
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
sl@0
|
49 |
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
sl@0
|
50 |
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
sl@0
|
51 |
* ====================================================================
|
sl@0
|
52 |
*
|
sl@0
|
53 |
* This product includes cryptographic software written by Eric Young
|
sl@0
|
54 |
* (eay@cryptsoft.com). This product includes software written by Tim
|
sl@0
|
55 |
* Hudson (tjh@cryptsoft.com).
|
sl@0
|
56 |
*
|
sl@0
|
57 |
*/
|
sl@0
|
58 |
|
sl@0
|
59 |
/*
|
sl@0
|
60 |
© Portions copyright (c) 2006 Nokia Corporation. All rights reserved.
|
sl@0
|
61 |
*/
|
sl@0
|
62 |
|
sl@0
|
63 |
|
sl@0
|
64 |
#include "eng_int.h"
|
sl@0
|
65 |
#include <openssl/dso.h>
|
sl@0
|
66 |
#if (defined(SYMBIAN) && (defined(__WINSCW__) || defined(__WINS__)))
|
sl@0
|
67 |
#include "libcrypto_wsd_macros.h"
|
sl@0
|
68 |
#include "libcrypto_wsd.h"
|
sl@0
|
69 |
#endif
|
sl@0
|
70 |
|
sl@0
|
71 |
/* Shared libraries implementing ENGINEs for use by the "dynamic" ENGINE loader
|
sl@0
|
72 |
* should implement the hook-up functions with the following prototypes. */
|
sl@0
|
73 |
|
sl@0
|
74 |
/* Our ENGINE handlers */
|
sl@0
|
75 |
static int dynamic_init(ENGINE *e);
|
sl@0
|
76 |
static int dynamic_finish(ENGINE *e);
|
sl@0
|
77 |
static int dynamic_ctrl(ENGINE *e, int cmd, long i, void *p, void (*f)(void));
|
sl@0
|
78 |
/* Predeclare our context type */
|
sl@0
|
79 |
typedef struct st_dynamic_data_ctx dynamic_data_ctx;
|
sl@0
|
80 |
/* The implementation for the important control command */
|
sl@0
|
81 |
static int dynamic_load(ENGINE *e, dynamic_data_ctx *ctx);
|
sl@0
|
82 |
|
sl@0
|
83 |
#define DYNAMIC_CMD_SO_PATH ENGINE_CMD_BASE
|
sl@0
|
84 |
#define DYNAMIC_CMD_NO_VCHECK (ENGINE_CMD_BASE + 1)
|
sl@0
|
85 |
#define DYNAMIC_CMD_ID (ENGINE_CMD_BASE + 2)
|
sl@0
|
86 |
#define DYNAMIC_CMD_LIST_ADD (ENGINE_CMD_BASE + 3)
|
sl@0
|
87 |
#define DYNAMIC_CMD_DIR_LOAD (ENGINE_CMD_BASE + 4)
|
sl@0
|
88 |
#define DYNAMIC_CMD_DIR_ADD (ENGINE_CMD_BASE + 5)
|
sl@0
|
89 |
#define DYNAMIC_CMD_LOAD (ENGINE_CMD_BASE + 6)
|
sl@0
|
90 |
|
sl@0
|
91 |
/* The constants used when creating the ENGINE */
|
sl@0
|
92 |
#ifndef EMULATOR
|
sl@0
|
93 |
static const char *engine_dynamic_id = "dynamic";
|
sl@0
|
94 |
static const char *engine_dynamic_name = "Dynamic engine loading support";
|
sl@0
|
95 |
#else
|
sl@0
|
96 |
static const char *const engine_dynamic_id = "dynamic";
|
sl@0
|
97 |
static const char *const engine_dynamic_name = "Dynamic engine loading support";
|
sl@0
|
98 |
#endif
|
sl@0
|
99 |
static const ENGINE_CMD_DEFN dynamic_cmd_defns[] = {
|
sl@0
|
100 |
{DYNAMIC_CMD_SO_PATH,
|
sl@0
|
101 |
"SO_PATH",
|
sl@0
|
102 |
"Specifies the path to the new ENGINE shared library",
|
sl@0
|
103 |
ENGINE_CMD_FLAG_STRING},
|
sl@0
|
104 |
{DYNAMIC_CMD_NO_VCHECK,
|
sl@0
|
105 |
"NO_VCHECK",
|
sl@0
|
106 |
"Specifies to continue even if version checking fails (boolean)",
|
sl@0
|
107 |
ENGINE_CMD_FLAG_NUMERIC},
|
sl@0
|
108 |
{DYNAMIC_CMD_ID,
|
sl@0
|
109 |
"ID",
|
sl@0
|
110 |
"Specifies an ENGINE id name for loading",
|
sl@0
|
111 |
ENGINE_CMD_FLAG_STRING},
|
sl@0
|
112 |
{DYNAMIC_CMD_LIST_ADD,
|
sl@0
|
113 |
"LIST_ADD",
|
sl@0
|
114 |
"Whether to add a loaded ENGINE to the internal list (0=no,1=yes,2=mandatory)",
|
sl@0
|
115 |
ENGINE_CMD_FLAG_NUMERIC},
|
sl@0
|
116 |
{DYNAMIC_CMD_DIR_LOAD,
|
sl@0
|
117 |
"DIR_LOAD",
|
sl@0
|
118 |
"Specifies whether to load from 'DIR_ADD' directories (0=no,1=yes,2=mandatory)",
|
sl@0
|
119 |
ENGINE_CMD_FLAG_NUMERIC},
|
sl@0
|
120 |
{DYNAMIC_CMD_DIR_ADD,
|
sl@0
|
121 |
"DIR_ADD",
|
sl@0
|
122 |
"Adds a directory from which ENGINEs can be loaded",
|
sl@0
|
123 |
ENGINE_CMD_FLAG_STRING},
|
sl@0
|
124 |
{DYNAMIC_CMD_LOAD,
|
sl@0
|
125 |
"LOAD",
|
sl@0
|
126 |
"Load up the ENGINE specified by other settings",
|
sl@0
|
127 |
ENGINE_CMD_FLAG_NO_INPUT},
|
sl@0
|
128 |
{0, NULL, NULL, 0}
|
sl@0
|
129 |
};
|
sl@0
|
130 |
static const ENGINE_CMD_DEFN dynamic_cmd_defns_empty[] = {
|
sl@0
|
131 |
{0, NULL, NULL, 0}
|
sl@0
|
132 |
};
|
sl@0
|
133 |
|
sl@0
|
134 |
/* Loading code stores state inside the ENGINE structure via the "ex_data"
|
sl@0
|
135 |
* element. We load all our state into a single structure and use that as a
|
sl@0
|
136 |
* single context in the "ex_data" stack. */
|
sl@0
|
137 |
struct st_dynamic_data_ctx
|
sl@0
|
138 |
{
|
sl@0
|
139 |
/* The DSO object we load that supplies the ENGINE code */
|
sl@0
|
140 |
DSO *dynamic_dso;
|
sl@0
|
141 |
/* The function pointer to the version checking shared library function */
|
sl@0
|
142 |
dynamic_v_check_fn v_check;
|
sl@0
|
143 |
/* The function pointer to the engine-binding shared library function */
|
sl@0
|
144 |
dynamic_bind_engine bind_engine;
|
sl@0
|
145 |
/* The default name/path for loading the shared library */
|
sl@0
|
146 |
const char *DYNAMIC_LIBNAME;
|
sl@0
|
147 |
/* Whether to continue loading on a version check failure */
|
sl@0
|
148 |
int no_vcheck;
|
sl@0
|
149 |
/* If non-NULL, stipulates the 'id' of the ENGINE to be loaded */
|
sl@0
|
150 |
const char *engine_id;
|
sl@0
|
151 |
/* If non-zero, a successfully loaded ENGINE should be added to the internal
|
sl@0
|
152 |
* ENGINE list. If 2, the add must succeed or the entire load should fail. */
|
sl@0
|
153 |
int list_add_value;
|
sl@0
|
154 |
/* The symbol name for the version checking function */
|
sl@0
|
155 |
const char *DYNAMIC_F1;
|
sl@0
|
156 |
/* The symbol name for the "initialise ENGINE structure" function */
|
sl@0
|
157 |
const char *DYNAMIC_F2;
|
sl@0
|
158 |
/* Whether to never use 'dirs', use 'dirs' as a fallback, or only use
|
sl@0
|
159 |
* 'dirs' for loading. Default is to use 'dirs' as a fallback. */
|
sl@0
|
160 |
int dir_load;
|
sl@0
|
161 |
/* A stack of directories from which ENGINEs could be loaded */
|
sl@0
|
162 |
STACK *dirs;
|
sl@0
|
163 |
};
|
sl@0
|
164 |
|
sl@0
|
165 |
/* This is the "ex_data" index we obtain and reserve for use with our context
|
sl@0
|
166 |
* structure. */
|
sl@0
|
167 |
#ifndef EMULATOR
|
sl@0
|
168 |
static int dynamic_ex_data_idx = -1;
|
sl@0
|
169 |
#else
|
sl@0
|
170 |
GET_STATIC_VAR_FROM_TLS(dynamic_ex_data_idx,eng_dyn,int)
|
sl@0
|
171 |
#define dynamic_ex_data_idx (*GET_WSD_VAR_NAME(dynamic_ex_data_idx,eng_dyn, s)())
|
sl@0
|
172 |
#endif
|
sl@0
|
173 |
|
sl@0
|
174 |
static void int_free_str(void *s) { OPENSSL_free(s); }
|
sl@0
|
175 |
/* Because our ex_data element may or may not get allocated depending on whether
|
sl@0
|
176 |
* a "first-use" occurs before the ENGINE is freed, we have a memory leak
|
sl@0
|
177 |
* problem to solve. We can't declare a "new" handler for the ex_data as we
|
sl@0
|
178 |
* don't want a dynamic_data_ctx in *all* ENGINE structures of all types (this
|
sl@0
|
179 |
* is a bug in the design of CRYPTO_EX_DATA). As such, we just declare a "free"
|
sl@0
|
180 |
* handler and that will get called if an ENGINE is being destroyed and there
|
sl@0
|
181 |
* was an ex_data element corresponding to our context type. */
|
sl@0
|
182 |
static void dynamic_data_ctx_free_func(void *parent, void *ptr,
|
sl@0
|
183 |
CRYPTO_EX_DATA *ad, int idx, long argl, void *argp)
|
sl@0
|
184 |
{
|
sl@0
|
185 |
if(ptr)
|
sl@0
|
186 |
{
|
sl@0
|
187 |
dynamic_data_ctx *ctx = (dynamic_data_ctx *)ptr;
|
sl@0
|
188 |
if(ctx->dynamic_dso)
|
sl@0
|
189 |
DSO_free(ctx->dynamic_dso);
|
sl@0
|
190 |
if(ctx->DYNAMIC_LIBNAME)
|
sl@0
|
191 |
OPENSSL_free((void*)ctx->DYNAMIC_LIBNAME);
|
sl@0
|
192 |
if(ctx->engine_id)
|
sl@0
|
193 |
OPENSSL_free((void*)ctx->engine_id);
|
sl@0
|
194 |
if(ctx->dirs)
|
sl@0
|
195 |
sk_pop_free(ctx->dirs, int_free_str);
|
sl@0
|
196 |
OPENSSL_free(ctx);
|
sl@0
|
197 |
}
|
sl@0
|
198 |
}
|
sl@0
|
199 |
|
sl@0
|
200 |
/* Construct the per-ENGINE context. We create it blindly and then use a lock to
|
sl@0
|
201 |
* check for a race - if so, all but one of the threads "racing" will have
|
sl@0
|
202 |
* wasted their time. The alternative involves creating everything inside the
|
sl@0
|
203 |
* lock which is far worse. */
|
sl@0
|
204 |
static int dynamic_set_data_ctx(ENGINE *e, dynamic_data_ctx **ctx)
|
sl@0
|
205 |
{
|
sl@0
|
206 |
dynamic_data_ctx *c;
|
sl@0
|
207 |
c = OPENSSL_malloc(sizeof(dynamic_data_ctx));
|
sl@0
|
208 |
if(!c)
|
sl@0
|
209 |
{
|
sl@0
|
210 |
ENGINEerr(ENGINE_F_DYNAMIC_SET_DATA_CTX,ERR_R_MALLOC_FAILURE);
|
sl@0
|
211 |
return 0;
|
sl@0
|
212 |
}
|
sl@0
|
213 |
memset(c, 0, sizeof(dynamic_data_ctx));
|
sl@0
|
214 |
c->dynamic_dso = NULL;
|
sl@0
|
215 |
c->v_check = NULL;
|
sl@0
|
216 |
c->bind_engine = NULL;
|
sl@0
|
217 |
c->DYNAMIC_LIBNAME = NULL;
|
sl@0
|
218 |
c->no_vcheck = 0;
|
sl@0
|
219 |
c->engine_id = NULL;
|
sl@0
|
220 |
c->list_add_value = 0;
|
sl@0
|
221 |
c->DYNAMIC_F1 = "v_check";
|
sl@0
|
222 |
c->DYNAMIC_F2 = "bind_engine";
|
sl@0
|
223 |
c->dir_load = 1;
|
sl@0
|
224 |
c->dirs = sk_new_null();
|
sl@0
|
225 |
if(!c->dirs)
|
sl@0
|
226 |
{
|
sl@0
|
227 |
ENGINEerr(ENGINE_F_DYNAMIC_SET_DATA_CTX,ERR_R_MALLOC_FAILURE);
|
sl@0
|
228 |
OPENSSL_free(c);
|
sl@0
|
229 |
return 0;
|
sl@0
|
230 |
}
|
sl@0
|
231 |
CRYPTO_w_lock(CRYPTO_LOCK_ENGINE);
|
sl@0
|
232 |
if((*ctx = (dynamic_data_ctx *)ENGINE_get_ex_data(e,
|
sl@0
|
233 |
dynamic_ex_data_idx)) == NULL)
|
sl@0
|
234 |
{
|
sl@0
|
235 |
/* Good, we're the first */
|
sl@0
|
236 |
ENGINE_set_ex_data(e, dynamic_ex_data_idx, c);
|
sl@0
|
237 |
*ctx = c;
|
sl@0
|
238 |
c = NULL;
|
sl@0
|
239 |
}
|
sl@0
|
240 |
CRYPTO_w_unlock(CRYPTO_LOCK_ENGINE);
|
sl@0
|
241 |
/* If we lost the race to set the context, c is non-NULL and *ctx is the
|
sl@0
|
242 |
* context of the thread that won. */
|
sl@0
|
243 |
if(c)
|
sl@0
|
244 |
OPENSSL_free(c);
|
sl@0
|
245 |
return 1;
|
sl@0
|
246 |
}
|
sl@0
|
247 |
|
sl@0
|
248 |
/* This function retrieves the context structure from an ENGINE's "ex_data", or
|
sl@0
|
249 |
* if it doesn't exist yet, sets it up. */
|
sl@0
|
250 |
static dynamic_data_ctx *dynamic_get_data_ctx(ENGINE *e)
|
sl@0
|
251 |
{
|
sl@0
|
252 |
dynamic_data_ctx *ctx;
|
sl@0
|
253 |
if(dynamic_ex_data_idx < 0)
|
sl@0
|
254 |
{
|
sl@0
|
255 |
/* Create and register the ENGINE ex_data, and associate our
|
sl@0
|
256 |
* "free" function with it to ensure any allocated contexts get
|
sl@0
|
257 |
* freed when an ENGINE goes underground. */
|
sl@0
|
258 |
int new_idx = ENGINE_get_ex_new_index(0, NULL, NULL, NULL,
|
sl@0
|
259 |
dynamic_data_ctx_free_func);
|
sl@0
|
260 |
if(new_idx == -1)
|
sl@0
|
261 |
{
|
sl@0
|
262 |
ENGINEerr(ENGINE_F_DYNAMIC_GET_DATA_CTX,ENGINE_R_NO_INDEX);
|
sl@0
|
263 |
return NULL;
|
sl@0
|
264 |
}
|
sl@0
|
265 |
CRYPTO_w_lock(CRYPTO_LOCK_ENGINE);
|
sl@0
|
266 |
/* Avoid a race by checking again inside this lock */
|
sl@0
|
267 |
if(dynamic_ex_data_idx < 0)
|
sl@0
|
268 |
{
|
sl@0
|
269 |
/* Good, someone didn't beat us to it */
|
sl@0
|
270 |
dynamic_ex_data_idx = new_idx;
|
sl@0
|
271 |
new_idx = -1;
|
sl@0
|
272 |
}
|
sl@0
|
273 |
CRYPTO_w_unlock(CRYPTO_LOCK_ENGINE);
|
sl@0
|
274 |
/* In theory we could "give back" the index here if
|
sl@0
|
275 |
* (new_idx>-1), but it's not possible and wouldn't gain us much
|
sl@0
|
276 |
* if it were. */
|
sl@0
|
277 |
}
|
sl@0
|
278 |
ctx = (dynamic_data_ctx *)ENGINE_get_ex_data(e, dynamic_ex_data_idx);
|
sl@0
|
279 |
/* Check if the context needs to be created */
|
sl@0
|
280 |
if((ctx == NULL) && !dynamic_set_data_ctx(e, &ctx))
|
sl@0
|
281 |
/* "set_data" will set errors if necessary */
|
sl@0
|
282 |
return NULL;
|
sl@0
|
283 |
return ctx;
|
sl@0
|
284 |
}
|
sl@0
|
285 |
|
sl@0
|
286 |
static ENGINE *engine_dynamic(void)
|
sl@0
|
287 |
{
|
sl@0
|
288 |
ENGINE *ret = ENGINE_new();
|
sl@0
|
289 |
if(!ret)
|
sl@0
|
290 |
return NULL;
|
sl@0
|
291 |
if(!ENGINE_set_id(ret, engine_dynamic_id) ||
|
sl@0
|
292 |
!ENGINE_set_name(ret, engine_dynamic_name) ||
|
sl@0
|
293 |
!ENGINE_set_init_function(ret, dynamic_init) ||
|
sl@0
|
294 |
!ENGINE_set_finish_function(ret, dynamic_finish) ||
|
sl@0
|
295 |
!ENGINE_set_ctrl_function(ret, dynamic_ctrl) ||
|
sl@0
|
296 |
!ENGINE_set_flags(ret, ENGINE_FLAGS_BY_ID_COPY) ||
|
sl@0
|
297 |
!ENGINE_set_cmd_defns(ret, dynamic_cmd_defns))
|
sl@0
|
298 |
{
|
sl@0
|
299 |
ENGINE_free(ret);
|
sl@0
|
300 |
return NULL;
|
sl@0
|
301 |
}
|
sl@0
|
302 |
return ret;
|
sl@0
|
303 |
}
|
sl@0
|
304 |
|
sl@0
|
305 |
EXPORT_C void ENGINE_load_dynamic(void)
|
sl@0
|
306 |
{
|
sl@0
|
307 |
ENGINE *toadd = engine_dynamic();
|
sl@0
|
308 |
if(!toadd) return;
|
sl@0
|
309 |
ENGINE_add(toadd);
|
sl@0
|
310 |
/* If the "add" worked, it gets a structural reference. So either way,
|
sl@0
|
311 |
* we release our just-created reference. */
|
sl@0
|
312 |
ENGINE_free(toadd);
|
sl@0
|
313 |
/* If the "add" didn't work, it was probably a conflict because it was
|
sl@0
|
314 |
* already added (eg. someone calling ENGINE_load_blah then calling
|
sl@0
|
315 |
* ENGINE_load_builtin_engines() perhaps). */
|
sl@0
|
316 |
ERR_clear_error();
|
sl@0
|
317 |
}
|
sl@0
|
318 |
|
sl@0
|
319 |
static int dynamic_init(ENGINE *e)
|
sl@0
|
320 |
{
|
sl@0
|
321 |
/* We always return failure - the "dyanamic" engine itself can't be used
|
sl@0
|
322 |
* for anything. */
|
sl@0
|
323 |
return 0;
|
sl@0
|
324 |
}
|
sl@0
|
325 |
|
sl@0
|
326 |
static int dynamic_finish(ENGINE *e)
|
sl@0
|
327 |
{
|
sl@0
|
328 |
/* This should never be called on account of "dynamic_init" always
|
sl@0
|
329 |
* failing. */
|
sl@0
|
330 |
return 0;
|
sl@0
|
331 |
}
|
sl@0
|
332 |
|
sl@0
|
333 |
static int dynamic_ctrl(ENGINE *e, int cmd, long i, void *p, void (*f)(void))
|
sl@0
|
334 |
{
|
sl@0
|
335 |
dynamic_data_ctx *ctx = dynamic_get_data_ctx(e);
|
sl@0
|
336 |
int initialised;
|
sl@0
|
337 |
|
sl@0
|
338 |
if(!ctx)
|
sl@0
|
339 |
{
|
sl@0
|
340 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,ENGINE_R_NOT_LOADED);
|
sl@0
|
341 |
return 0;
|
sl@0
|
342 |
}
|
sl@0
|
343 |
initialised = ((ctx->dynamic_dso == NULL) ? 0 : 1);
|
sl@0
|
344 |
/* All our control commands require the ENGINE to be uninitialised */
|
sl@0
|
345 |
if(initialised)
|
sl@0
|
346 |
{
|
sl@0
|
347 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,
|
sl@0
|
348 |
ENGINE_R_ALREADY_LOADED);
|
sl@0
|
349 |
return 0;
|
sl@0
|
350 |
}
|
sl@0
|
351 |
switch(cmd)
|
sl@0
|
352 |
{
|
sl@0
|
353 |
case DYNAMIC_CMD_SO_PATH:
|
sl@0
|
354 |
/* a NULL 'p' or a string of zero-length is the same thing */
|
sl@0
|
355 |
if(p && (strlen((const char *)p) < 1))
|
sl@0
|
356 |
p = NULL;
|
sl@0
|
357 |
if(ctx->DYNAMIC_LIBNAME)
|
sl@0
|
358 |
OPENSSL_free((void*)ctx->DYNAMIC_LIBNAME);
|
sl@0
|
359 |
if(p)
|
sl@0
|
360 |
ctx->DYNAMIC_LIBNAME = BUF_strdup(p);
|
sl@0
|
361 |
else
|
sl@0
|
362 |
ctx->DYNAMIC_LIBNAME = NULL;
|
sl@0
|
363 |
return (ctx->DYNAMIC_LIBNAME ? 1 : 0);
|
sl@0
|
364 |
case DYNAMIC_CMD_NO_VCHECK:
|
sl@0
|
365 |
ctx->no_vcheck = ((i == 0) ? 0 : 1);
|
sl@0
|
366 |
return 1;
|
sl@0
|
367 |
case DYNAMIC_CMD_ID:
|
sl@0
|
368 |
/* a NULL 'p' or a string of zero-length is the same thing */
|
sl@0
|
369 |
if(p && (strlen((const char *)p) < 1))
|
sl@0
|
370 |
p = NULL;
|
sl@0
|
371 |
if(ctx->engine_id)
|
sl@0
|
372 |
OPENSSL_free((void*)ctx->engine_id);
|
sl@0
|
373 |
if(p)
|
sl@0
|
374 |
ctx->engine_id = BUF_strdup(p);
|
sl@0
|
375 |
else
|
sl@0
|
376 |
ctx->engine_id = NULL;
|
sl@0
|
377 |
return (ctx->engine_id ? 1 : 0);
|
sl@0
|
378 |
case DYNAMIC_CMD_LIST_ADD:
|
sl@0
|
379 |
if((i < 0) || (i > 2))
|
sl@0
|
380 |
{
|
sl@0
|
381 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,
|
sl@0
|
382 |
ENGINE_R_INVALID_ARGUMENT);
|
sl@0
|
383 |
return 0;
|
sl@0
|
384 |
}
|
sl@0
|
385 |
ctx->list_add_value = (int)i;
|
sl@0
|
386 |
return 1;
|
sl@0
|
387 |
case DYNAMIC_CMD_LOAD:
|
sl@0
|
388 |
return dynamic_load(e, ctx);
|
sl@0
|
389 |
case DYNAMIC_CMD_DIR_LOAD:
|
sl@0
|
390 |
if((i < 0) || (i > 2))
|
sl@0
|
391 |
{
|
sl@0
|
392 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,
|
sl@0
|
393 |
ENGINE_R_INVALID_ARGUMENT);
|
sl@0
|
394 |
return 0;
|
sl@0
|
395 |
}
|
sl@0
|
396 |
ctx->dir_load = (int)i;
|
sl@0
|
397 |
return 1;
|
sl@0
|
398 |
case DYNAMIC_CMD_DIR_ADD:
|
sl@0
|
399 |
/* a NULL 'p' or a string of zero-length is the same thing */
|
sl@0
|
400 |
if(!p || (strlen((const char *)p) < 1))
|
sl@0
|
401 |
{
|
sl@0
|
402 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,
|
sl@0
|
403 |
ENGINE_R_INVALID_ARGUMENT);
|
sl@0
|
404 |
return 0;
|
sl@0
|
405 |
}
|
sl@0
|
406 |
{
|
sl@0
|
407 |
char *tmp_str = BUF_strdup(p);
|
sl@0
|
408 |
if(!tmp_str)
|
sl@0
|
409 |
{
|
sl@0
|
410 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,
|
sl@0
|
411 |
ERR_R_MALLOC_FAILURE);
|
sl@0
|
412 |
return 0;
|
sl@0
|
413 |
}
|
sl@0
|
414 |
sk_insert(ctx->dirs, tmp_str, -1);
|
sl@0
|
415 |
}
|
sl@0
|
416 |
return 1;
|
sl@0
|
417 |
default:
|
sl@0
|
418 |
break;
|
sl@0
|
419 |
}
|
sl@0
|
420 |
ENGINEerr(ENGINE_F_DYNAMIC_CTRL,ENGINE_R_CTRL_COMMAND_NOT_IMPLEMENTED);
|
sl@0
|
421 |
return 0;
|
sl@0
|
422 |
}
|
sl@0
|
423 |
|
sl@0
|
424 |
static int int_load(dynamic_data_ctx *ctx)
|
sl@0
|
425 |
{
|
sl@0
|
426 |
int num, loop;
|
sl@0
|
427 |
/* Unless told not to, try a direct load */
|
sl@0
|
428 |
if((ctx->dir_load != 2) && (DSO_load(ctx->dynamic_dso,
|
sl@0
|
429 |
ctx->DYNAMIC_LIBNAME, NULL, 0)) != NULL)
|
sl@0
|
430 |
return 1;
|
sl@0
|
431 |
/* If we're not allowed to use 'dirs' or we have none, fail */
|
sl@0
|
432 |
if(!ctx->dir_load || ((num = sk_num(ctx->dirs)) < 1))
|
sl@0
|
433 |
return 0;
|
sl@0
|
434 |
for(loop = 0; loop < num; loop++)
|
sl@0
|
435 |
{
|
sl@0
|
436 |
const char *s = sk_value(ctx->dirs, loop);
|
sl@0
|
437 |
char *merge = DSO_merge(ctx->dynamic_dso, ctx->DYNAMIC_LIBNAME, s);
|
sl@0
|
438 |
if(!merge)
|
sl@0
|
439 |
return 0;
|
sl@0
|
440 |
if(DSO_load(ctx->dynamic_dso, merge, NULL, 0))
|
sl@0
|
441 |
{
|
sl@0
|
442 |
/* Found what we're looking for */
|
sl@0
|
443 |
OPENSSL_free(merge);
|
sl@0
|
444 |
return 1;
|
sl@0
|
445 |
}
|
sl@0
|
446 |
OPENSSL_free(merge);
|
sl@0
|
447 |
}
|
sl@0
|
448 |
return 0;
|
sl@0
|
449 |
}
|
sl@0
|
450 |
|
sl@0
|
451 |
static int dynamic_load(ENGINE *e, dynamic_data_ctx *ctx)
|
sl@0
|
452 |
{
|
sl@0
|
453 |
ENGINE cpy;
|
sl@0
|
454 |
dynamic_fns fns;
|
sl@0
|
455 |
|
sl@0
|
456 |
if(!ctx->dynamic_dso)
|
sl@0
|
457 |
ctx->dynamic_dso = DSO_new();
|
sl@0
|
458 |
if(!ctx->DYNAMIC_LIBNAME)
|
sl@0
|
459 |
{
|
sl@0
|
460 |
if(!ctx->engine_id)
|
sl@0
|
461 |
return 0;
|
sl@0
|
462 |
ctx->DYNAMIC_LIBNAME =
|
sl@0
|
463 |
DSO_convert_filename(ctx->dynamic_dso, ctx->engine_id);
|
sl@0
|
464 |
}
|
sl@0
|
465 |
if(!int_load(ctx))
|
sl@0
|
466 |
{
|
sl@0
|
467 |
ENGINEerr(ENGINE_F_DYNAMIC_LOAD,
|
sl@0
|
468 |
ENGINE_R_DSO_NOT_FOUND);
|
sl@0
|
469 |
DSO_free(ctx->dynamic_dso);
|
sl@0
|
470 |
ctx->dynamic_dso = NULL;
|
sl@0
|
471 |
return 0;
|
sl@0
|
472 |
}
|
sl@0
|
473 |
/* We have to find a bind function otherwise it'll always end badly */
|
sl@0
|
474 |
if(!(ctx->bind_engine = (dynamic_bind_engine)DSO_bind_func(
|
sl@0
|
475 |
ctx->dynamic_dso, ctx->DYNAMIC_F2)))
|
sl@0
|
476 |
{
|
sl@0
|
477 |
ctx->bind_engine = NULL;
|
sl@0
|
478 |
DSO_free(ctx->dynamic_dso);
|
sl@0
|
479 |
ctx->dynamic_dso = NULL;
|
sl@0
|
480 |
ENGINEerr(ENGINE_F_DYNAMIC_LOAD,
|
sl@0
|
481 |
ENGINE_R_DSO_FAILURE);
|
sl@0
|
482 |
return 0;
|
sl@0
|
483 |
}
|
sl@0
|
484 |
/* Do we perform version checking? */
|
sl@0
|
485 |
if(!ctx->no_vcheck)
|
sl@0
|
486 |
{
|
sl@0
|
487 |
unsigned long vcheck_res = 0;
|
sl@0
|
488 |
/* Now we try to find a version checking function and decide how
|
sl@0
|
489 |
* to cope with failure if/when it fails. */
|
sl@0
|
490 |
ctx->v_check = (dynamic_v_check_fn)DSO_bind_func(
|
sl@0
|
491 |
ctx->dynamic_dso, ctx->DYNAMIC_F1);
|
sl@0
|
492 |
if(ctx->v_check)
|
sl@0
|
493 |
vcheck_res = ctx->v_check(OSSL_DYNAMIC_VERSION);
|
sl@0
|
494 |
/* We fail if the version checker veto'd the load *or* if it is
|
sl@0
|
495 |
* deferring to us (by returning its version) and we think it is
|
sl@0
|
496 |
* too old. */
|
sl@0
|
497 |
if(vcheck_res < OSSL_DYNAMIC_OLDEST)
|
sl@0
|
498 |
{
|
sl@0
|
499 |
/* Fail */
|
sl@0
|
500 |
ctx->bind_engine = NULL;
|
sl@0
|
501 |
ctx->v_check = NULL;
|
sl@0
|
502 |
DSO_free(ctx->dynamic_dso);
|
sl@0
|
503 |
ctx->dynamic_dso = NULL;
|
sl@0
|
504 |
ENGINEerr(ENGINE_F_DYNAMIC_LOAD,
|
sl@0
|
505 |
ENGINE_R_VERSION_INCOMPATIBILITY);
|
sl@0
|
506 |
return 0;
|
sl@0
|
507 |
}
|
sl@0
|
508 |
}
|
sl@0
|
509 |
/* First binary copy the ENGINE structure so that we can roll back if
|
sl@0
|
510 |
* the hand-over fails */
|
sl@0
|
511 |
memcpy(&cpy, e, sizeof(ENGINE));
|
sl@0
|
512 |
/* Provide the ERR, "ex_data", memory, and locking callbacks so the
|
sl@0
|
513 |
* loaded library uses our state rather than its own. FIXME: As noted in
|
sl@0
|
514 |
* engine.h, much of this would be simplified if each area of code
|
sl@0
|
515 |
* provided its own "summary" structure of all related callbacks. It
|
sl@0
|
516 |
* would also increase opaqueness. */
|
sl@0
|
517 |
fns.static_state = ENGINE_get_static_state();
|
sl@0
|
518 |
fns.err_fns = ERR_get_implementation();
|
sl@0
|
519 |
fns.ex_data_fns = CRYPTO_get_ex_data_implementation();
|
sl@0
|
520 |
CRYPTO_get_mem_functions(&fns.mem_fns.malloc_cb,
|
sl@0
|
521 |
&fns.mem_fns.realloc_cb,
|
sl@0
|
522 |
&fns.mem_fns.free_cb);
|
sl@0
|
523 |
fns.lock_fns.lock_locking_cb = CRYPTO_get_locking_callback();
|
sl@0
|
524 |
fns.lock_fns.lock_add_lock_cb = CRYPTO_get_add_lock_callback();
|
sl@0
|
525 |
fns.lock_fns.dynlock_create_cb = CRYPTO_get_dynlock_create_callback();
|
sl@0
|
526 |
fns.lock_fns.dynlock_lock_cb = CRYPTO_get_dynlock_lock_callback();
|
sl@0
|
527 |
fns.lock_fns.dynlock_destroy_cb = CRYPTO_get_dynlock_destroy_callback();
|
sl@0
|
528 |
/* Now that we've loaded the dynamic engine, make sure no "dynamic"
|
sl@0
|
529 |
* ENGINE elements will show through. */
|
sl@0
|
530 |
engine_set_all_null(e);
|
sl@0
|
531 |
|
sl@0
|
532 |
/* Try to bind the ENGINE onto our own ENGINE structure */
|
sl@0
|
533 |
if(!ctx->bind_engine(e, ctx->engine_id, &fns))
|
sl@0
|
534 |
{
|
sl@0
|
535 |
ctx->bind_engine = NULL;
|
sl@0
|
536 |
ctx->v_check = NULL;
|
sl@0
|
537 |
DSO_free(ctx->dynamic_dso);
|
sl@0
|
538 |
ctx->dynamic_dso = NULL;
|
sl@0
|
539 |
ENGINEerr(ENGINE_F_DYNAMIC_LOAD,ENGINE_R_INIT_FAILED);
|
sl@0
|
540 |
/* Copy the original ENGINE structure back */
|
sl@0
|
541 |
memcpy(e, &cpy, sizeof(ENGINE));
|
sl@0
|
542 |
return 0;
|
sl@0
|
543 |
}
|
sl@0
|
544 |
/* Do we try to add this ENGINE to the internal list too? */
|
sl@0
|
545 |
if(ctx->list_add_value > 0)
|
sl@0
|
546 |
{
|
sl@0
|
547 |
if(!ENGINE_add(e))
|
sl@0
|
548 |
{
|
sl@0
|
549 |
/* Do we tolerate this or fail? */
|
sl@0
|
550 |
if(ctx->list_add_value > 1)
|
sl@0
|
551 |
{
|
sl@0
|
552 |
/* Fail - NB: By this time, it's too late to
|
sl@0
|
553 |
* rollback, and trying to do so allows the
|
sl@0
|
554 |
* bind_engine() code to have created leaks. We
|
sl@0
|
555 |
* just have to fail where we are, after the
|
sl@0
|
556 |
* ENGINE has changed. */
|
sl@0
|
557 |
ENGINEerr(ENGINE_F_DYNAMIC_LOAD,
|
sl@0
|
558 |
ENGINE_R_CONFLICTING_ENGINE_ID);
|
sl@0
|
559 |
return 0;
|
sl@0
|
560 |
}
|
sl@0
|
561 |
/* Tolerate */
|
sl@0
|
562 |
ERR_clear_error();
|
sl@0
|
563 |
}
|
sl@0
|
564 |
}
|
sl@0
|
565 |
return 1;
|
sl@0
|
566 |
}
|