2016-05-17 19:38:09 +00:00
|
|
|
/*
|
2018-02-13 12:51:29 +00:00
|
|
|
* Copyright 1995-2018 The OpenSSL Project Authors. All Rights Reserved.
|
2001-09-04 11:02:23 +00:00
|
|
|
*
|
2016-05-17 19:38:09 +00:00
|
|
|
* Licensed under the OpenSSL license (the "License"). You may not use
|
|
|
|
* this file except in compliance with the License. You can obtain a copy
|
|
|
|
* in the file LICENSE in the source distribution or at
|
|
|
|
* https://www.openssl.org/source/license.html
|
2001-09-04 11:02:23 +00:00
|
|
|
*/
|
1998-12-21 10:56:39 +00:00
|
|
|
|
2016-03-11 21:53:18 +00:00
|
|
|
#include "internal/cryptlib_int.h"
|
2016-07-19 17:42:11 +00:00
|
|
|
#include "internal/thread_once.h"
|
1998-12-21 10:56:39 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
/*
|
|
|
|
* Each structure type (sometimes called a class), that supports
|
|
|
|
* exdata has a stack of callbacks for each instance.
|
|
|
|
*/
|
2016-01-06 02:54:18 +00:00
|
|
|
struct ex_callback_st {
|
2016-02-05 20:23:54 +00:00
|
|
|
long argl; /* Arbitrary long */
|
|
|
|
void *argp; /* Arbitrary void * */
|
2015-04-24 20:33:34 +00:00
|
|
|
CRYPTO_EX_new *new_func;
|
|
|
|
CRYPTO_EX_free *free_func;
|
|
|
|
CRYPTO_EX_dup *dup_func;
|
2016-01-06 02:54:18 +00:00
|
|
|
};
|
2015-01-22 03:40:55 +00:00
|
|
|
|
|
|
|
/*
|
2015-07-21 14:06:03 +00:00
|
|
|
* The state for each class. This could just be a typedef, but
|
|
|
|
* a structure allows future changes.
|
2015-01-22 03:40:55 +00:00
|
|
|
*/
|
2015-07-21 14:06:03 +00:00
|
|
|
typedef struct ex_callbacks_st {
|
|
|
|
STACK_OF(EX_CALLBACK) *meth;
|
|
|
|
} EX_CALLBACKS;
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
static EX_CALLBACKS ex_data[CRYPTO_EX_INDEX__COUNT];
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2016-04-14 20:28:54 +00:00
|
|
|
static CRYPTO_RWLOCK *ex_data_lock = NULL;
|
2016-03-05 21:54:02 +00:00
|
|
|
static CRYPTO_ONCE ex_data_init = CRYPTO_ONCE_STATIC_INIT;
|
|
|
|
|
2016-07-19 17:42:11 +00:00
|
|
|
DEFINE_RUN_ONCE_STATIC(do_ex_data_init)
|
2016-03-05 21:54:02 +00:00
|
|
|
{
|
2018-04-20 13:45:06 +00:00
|
|
|
if (!OPENSSL_init_crypto(0, NULL))
|
|
|
|
return 0;
|
Revert the crypto "global lock" implementation
Conceptually, this is a squashed version of:
Revert "Address feedback"
This reverts commit 75551e07bd2339dfea06ef1d31d69929e13a4495.
and
Revert "Add CRYPTO_thread_glock_new"
This reverts commit ed6b2c7938ec6f07b15745d4183afc276e74c6dd.
But there were some intervening commits that made neither revert apply
cleanly, so instead do it all as one shot.
The crypto global locks were an attempt to cope with the awkward
POSIX semantics for pthread_atfork(); its documentation (the "RATIONALE"
section) indicates that the expected usage is to have the prefork handler
lock all "global" locks, and the parent and child handlers release those
locks, to ensure that forking happens with a consistent (lock) state.
However, the set of functions available in the child process is limited
to async-signal-safe functions, and pthread_mutex_unlock() is not on
the list of async-signal-safe functions! The only synchronization
primitives that are async-signal-safe are the semaphore primitives,
which are not really appropriate for general-purpose usage.
However, the state consistency problem that the global locks were
attempting to solve is not actually a serious problem, particularly for
OpenSSL. That is, we can consider four cases of forking application
that might use OpenSSL:
(1) Single-threaded, does not call into OpenSSL in the child (e.g.,
the child calls exec() immediately)
For this class of process, no locking is needed at all, since there is
only ever a single thread of execution and the only reentrancy is due to
signal handlers (which are themselves limited to async-signal-safe
operation and should not be doing much work at all).
(2) Single-threaded, calls into OpenSSL after fork()
The application must ensure that it does not fork() with an unexpected
lock held (that is, one that would get unlocked in the parent but
accidentally remain locked in the child and cause deadlock). Since
OpenSSL does not expose any of its internal locks to the application
and the application is single-threaded, the OpenSSL internal locks
will be unlocked for the fork(), and the state will be consistent.
(OpenSSL will need to reseed its PRNG in the child, but that is
an orthogonal issue.) If the application makes use of locks from
libcrypto, proper handling for those locks is the responsibility of
the application, as for any other locking primitive that is available
for application programming.
(3) Multi-threaded, does not call into OpenSSL after fork()
As for (1), the OpenSSL state is only relevant in the parent, so
no particular fork()-related handling is needed. The internal locks
are relevant, but there is no interaction with the child to consider.
(4) Multi-threaded, calls into OpenSSL after fork()
This is the case where the pthread_atfork() hooks to ensure that all
global locks are in a known state across fork() would come into play,
per the above discussion. However, these "calls into OpenSSL after
fork()" are still subject to the restriction to async-signal-safe
functions. Since OpenSSL uses all sorts of locking and libc functions
that are not on the list of safe functions (e.g., malloc()), this
case is not currently usable and is unlikely to ever be usable,
independently of the locking situation. So, there is no need to
go through contortions to attempt to support this case in the one small
area of locking interaction with fork().
In light of the above analysis (thanks @davidben and @achernya), go
back to the simpler implementation that does not need to distinguish
"library-global" locks or to have complicated atfork handling for locks.
Reviewed-by: Kurt Roeckx <kurt@roeckx.be>
Reviewed-by: Matthias St. Pierre <Matthias.St.Pierre@ncp-e.com>
(Merged from https://github.com/openssl/openssl/pull/5089)
2018-01-16 15:49:54 +00:00
|
|
|
ex_data_lock = CRYPTO_THREAD_lock_new();
|
2016-07-19 17:42:11 +00:00
|
|
|
return ex_data_lock != NULL;
|
2016-03-11 21:53:18 +00:00
|
|
|
}
|
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
/*
|
2015-07-21 14:06:03 +00:00
|
|
|
* Return the EX_CALLBACKS from the |ex_data| array that corresponds to
|
2015-04-24 20:33:34 +00:00
|
|
|
* a given class. On success, *holds the lock.*
|
2015-01-22 03:40:55 +00:00
|
|
|
*/
|
2015-07-21 14:06:03 +00:00
|
|
|
static EX_CALLBACKS *get_and_lock(int class_index)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACKS *ip;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
|
|
|
if (class_index < 0 || class_index >= CRYPTO_EX_INDEX__COUNT) {
|
2016-03-14 20:46:54 +00:00
|
|
|
CRYPTOerr(CRYPTO_F_GET_AND_LOCK, ERR_R_PASSED_INVALID_ARGUMENT);
|
2015-04-24 20:33:34 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-07-19 17:42:11 +00:00
|
|
|
if (!RUN_ONCE(&ex_data_init, do_ex_data_init)) {
|
|
|
|
CRYPTOerr(CRYPTO_F_GET_AND_LOCK, ERR_R_MALLOC_FAILURE);
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-03-05 21:54:02 +00:00
|
|
|
|
2016-04-14 20:28:54 +00:00
|
|
|
if (ex_data_lock == NULL) {
|
|
|
|
/*
|
|
|
|
* This can happen in normal operation when using CRYPTO_mem_leaks().
|
|
|
|
* The CRYPTO_mem_leaks() function calls OPENSSL_cleanup() which cleans
|
|
|
|
* up the locks. Subsequently the BIO that CRYPTO_mem_leaks() uses gets
|
|
|
|
* freed, which also attempts to free the ex_data. However
|
|
|
|
* CRYPTO_mem_leaks() ensures that the ex_data is freed early (i.e.
|
|
|
|
* before OPENSSL_cleanup() is called), so if we get here we can safely
|
|
|
|
* ignore this operation. We just treat it as an error.
|
|
|
|
*/
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-04-24 20:33:34 +00:00
|
|
|
ip = &ex_data[class_index];
|
2016-03-05 21:54:02 +00:00
|
|
|
CRYPTO_THREAD_write_lock(ex_data_lock);
|
2015-04-24 20:33:34 +00:00
|
|
|
return ip;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
static void cleanup_cb(EX_CALLBACK *funcs)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
|
|
|
OPENSSL_free(funcs);
|
|
|
|
}
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
/*
|
2015-04-24 20:33:34 +00:00
|
|
|
* Release all "ex_data" state to prevent memory leaks. This can't be made
|
|
|
|
* thread-safe without overhauling a lot of stuff, and shouldn't really be
|
|
|
|
* called under potential race-conditions anyway (it's for program shutdown
|
|
|
|
* after all).
|
2015-01-22 03:40:55 +00:00
|
|
|
*/
|
2016-04-12 11:20:16 +00:00
|
|
|
void crypto_cleanup_all_ex_data_int(void)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
2015-04-24 20:33:34 +00:00
|
|
|
int i;
|
2015-01-22 03:40:55 +00:00
|
|
|
|
2015-04-24 20:33:34 +00:00
|
|
|
for (i = 0; i < CRYPTO_EX_INDEX__COUNT; ++i) {
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACKS *ip = &ex_data[i];
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
sk_EX_CALLBACK_pop_free(ip->meth, cleanup_cb);
|
2015-04-24 20:33:34 +00:00
|
|
|
ip->meth = NULL;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
2016-04-14 20:28:54 +00:00
|
|
|
|
|
|
|
CRYPTO_THREAD_lock_free(ex_data_lock);
|
|
|
|
ex_data_lock = NULL;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Unregister a new index by replacing the callbacks with no-ops.
|
|
|
|
* Any in-use instances are leaked.
|
|
|
|
*/
|
|
|
|
static void dummy_new(void *parent, void *ptr, CRYPTO_EX_DATA *ad, int idx,
|
|
|
|
long argl, void *argp)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dummy_free(void *parent, void *ptr, CRYPTO_EX_DATA *ad, int idx,
|
|
|
|
long argl, void *argp)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-07-31 10:04:48 +00:00
|
|
|
static int dummy_dup(CRYPTO_EX_DATA *to, const CRYPTO_EX_DATA *from,
|
2015-07-21 14:06:03 +00:00
|
|
|
void *from_d, int idx,
|
|
|
|
long argl, void *argp)
|
|
|
|
{
|
2017-03-19 15:14:33 +00:00
|
|
|
return 1;
|
2015-07-21 14:06:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CRYPTO_free_ex_index(int class_index, int idx)
|
|
|
|
{
|
|
|
|
EX_CALLBACKS *ip = get_and_lock(class_index);
|
|
|
|
EX_CALLBACK *a;
|
|
|
|
int toret = 0;
|
|
|
|
|
|
|
|
if (ip == NULL)
|
|
|
|
return 0;
|
|
|
|
if (idx < 0 || idx >= sk_EX_CALLBACK_num(ip->meth))
|
|
|
|
goto err;
|
|
|
|
a = sk_EX_CALLBACK_value(ip->meth, idx);
|
|
|
|
if (a == NULL)
|
|
|
|
goto err;
|
|
|
|
a->new_func = dummy_new;
|
|
|
|
a->dup_func = dummy_dup;
|
|
|
|
a->free_func = dummy_free;
|
|
|
|
toret = 1;
|
|
|
|
err:
|
2016-03-05 21:54:02 +00:00
|
|
|
CRYPTO_THREAD_unlock(ex_data_lock);
|
2015-07-21 14:06:03 +00:00
|
|
|
return toret;
|
|
|
|
}
|
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
/*
|
2015-07-21 14:06:03 +00:00
|
|
|
* Register a new index.
|
2015-01-22 03:40:55 +00:00
|
|
|
*/
|
2015-04-24 20:33:34 +00:00
|
|
|
int CRYPTO_get_ex_new_index(int class_index, long argl, void *argp,
|
|
|
|
CRYPTO_EX_new *new_func, CRYPTO_EX_dup *dup_func,
|
|
|
|
CRYPTO_EX_free *free_func)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
|
|
|
int toret = -1;
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACK *a;
|
|
|
|
EX_CALLBACKS *ip = get_and_lock(class_index);
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
if (ip == NULL)
|
2015-01-22 03:40:55 +00:00
|
|
|
return -1;
|
2016-03-12 19:46:13 +00:00
|
|
|
|
|
|
|
if (ip->meth == NULL) {
|
|
|
|
ip->meth = sk_EX_CALLBACK_new_null();
|
|
|
|
/* We push an initial value on the stack because the SSL
|
|
|
|
* "app_data" routines use ex_data index zero. See RT 3710. */
|
|
|
|
if (ip->meth == NULL
|
|
|
|
|| !sk_EX_CALLBACK_push(ip->meth, NULL)) {
|
2016-03-13 13:54:51 +00:00
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_GET_EX_NEW_INDEX, ERR_R_MALLOC_FAILURE);
|
2016-03-12 19:46:13 +00:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
a = (EX_CALLBACK *)OPENSSL_malloc(sizeof(*a));
|
2015-10-30 11:12:26 +00:00
|
|
|
if (a == NULL) {
|
2015-04-24 20:33:34 +00:00
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_GET_EX_NEW_INDEX, ERR_R_MALLOC_FAILURE);
|
|
|
|
goto err;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
a->argl = argl;
|
|
|
|
a->argp = argp;
|
|
|
|
a->new_func = new_func;
|
|
|
|
a->dup_func = dup_func;
|
|
|
|
a->free_func = free_func;
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
if (!sk_EX_CALLBACK_push(ip->meth, NULL)) {
|
2015-04-24 20:33:34 +00:00
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_GET_EX_NEW_INDEX, ERR_R_MALLOC_FAILURE);
|
|
|
|
OPENSSL_free(a);
|
|
|
|
goto err;
|
|
|
|
}
|
2015-07-21 14:06:03 +00:00
|
|
|
toret = sk_EX_CALLBACK_num(ip->meth) - 1;
|
|
|
|
(void)sk_EX_CALLBACK_set(ip->meth, toret, a);
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-04-24 20:33:34 +00:00
|
|
|
err:
|
2016-03-05 21:54:02 +00:00
|
|
|
CRYPTO_THREAD_unlock(ex_data_lock);
|
2015-01-22 03:40:55 +00:00
|
|
|
return toret;
|
|
|
|
}
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
/*
|
2015-04-24 20:33:34 +00:00
|
|
|
* Initialise a new CRYPTO_EX_DATA for use in a particular class - including
|
|
|
|
* calling new() callbacks for each index in the class used by this variable
|
2015-07-21 14:06:03 +00:00
|
|
|
* Thread-safe by copying a class's array of "EX_CALLBACK" entries
|
2015-04-24 20:33:34 +00:00
|
|
|
* in the lock, then using them outside the lock. Note this only applies
|
|
|
|
* to the global "ex_data" state (ie. class definitions), not 'ad' itself.
|
2015-01-22 03:40:55 +00:00
|
|
|
*/
|
2015-04-24 20:33:34 +00:00
|
|
|
int CRYPTO_new_ex_data(int class_index, void *obj, CRYPTO_EX_DATA *ad)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
|
|
|
int mx, i;
|
|
|
|
void *ptr;
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACK **storage = NULL;
|
|
|
|
EX_CALLBACK *stack[10];
|
|
|
|
EX_CALLBACKS *ip = get_and_lock(class_index);
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
if (ip == NULL)
|
2015-01-22 03:40:55 +00:00
|
|
|
return 0;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
ad->sk = NULL;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
mx = sk_EX_CALLBACK_num(ip->meth);
|
2015-01-22 03:40:55 +00:00
|
|
|
if (mx > 0) {
|
2015-04-24 20:33:34 +00:00
|
|
|
if (mx < (int)OSSL_NELEM(stack))
|
|
|
|
storage = stack;
|
|
|
|
else
|
|
|
|
storage = OPENSSL_malloc(sizeof(*storage) * mx);
|
2015-10-30 11:12:26 +00:00
|
|
|
if (storage != NULL)
|
2015-04-24 20:33:34 +00:00
|
|
|
for (i = 0; i < mx; i++)
|
2015-07-21 14:06:03 +00:00
|
|
|
storage[i] = sk_EX_CALLBACK_value(ip->meth, i);
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
2016-03-05 21:54:02 +00:00
|
|
|
CRYPTO_THREAD_unlock(ex_data_lock);
|
2015-04-24 20:33:34 +00:00
|
|
|
|
|
|
|
if (mx > 0 && storage == NULL) {
|
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_NEW_EX_DATA, ERR_R_MALLOC_FAILURE);
|
2015-01-22 03:40:55 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
for (i = 0; i < mx; i++) {
|
|
|
|
if (storage[i] && storage[i]->new_func) {
|
|
|
|
ptr = CRYPTO_get_ex_data(ad, i);
|
|
|
|
storage[i]->new_func(obj, ptr, ad, i,
|
|
|
|
storage[i]->argl, storage[i]->argp);
|
|
|
|
}
|
|
|
|
}
|
2015-04-24 20:33:34 +00:00
|
|
|
if (storage != stack)
|
|
|
|
OPENSSL_free(storage);
|
2015-01-22 03:40:55 +00:00
|
|
|
return 1;
|
|
|
|
}
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-04-24 20:33:34 +00:00
|
|
|
/*
|
|
|
|
* Duplicate a CRYPTO_EX_DATA variable - including calling dup() callbacks
|
|
|
|
* for each index in the class used by this variable
|
|
|
|
*/
|
|
|
|
int CRYPTO_dup_ex_data(int class_index, CRYPTO_EX_DATA *to,
|
2016-07-31 10:04:48 +00:00
|
|
|
const CRYPTO_EX_DATA *from)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
|
|
|
int mx, j, i;
|
2017-03-19 15:14:33 +00:00
|
|
|
void *ptr;
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACK *stack[10];
|
|
|
|
EX_CALLBACK **storage = NULL;
|
|
|
|
EX_CALLBACKS *ip;
|
2017-03-19 15:14:33 +00:00
|
|
|
int toret = 0;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
|
|
|
if (from->sk == NULL)
|
|
|
|
/* Nothing to copy over */
|
2015-01-22 03:40:55 +00:00
|
|
|
return 1;
|
2015-07-21 14:06:03 +00:00
|
|
|
if ((ip = get_and_lock(class_index)) == NULL)
|
2015-01-22 03:40:55 +00:00
|
|
|
return 0;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
mx = sk_EX_CALLBACK_num(ip->meth);
|
2015-01-22 03:40:55 +00:00
|
|
|
j = sk_void_num(from->sk);
|
|
|
|
if (j < mx)
|
|
|
|
mx = j;
|
|
|
|
if (mx > 0) {
|
2015-04-24 20:33:34 +00:00
|
|
|
if (mx < (int)OSSL_NELEM(stack))
|
|
|
|
storage = stack;
|
|
|
|
else
|
|
|
|
storage = OPENSSL_malloc(sizeof(*storage) * mx);
|
2015-10-30 11:12:26 +00:00
|
|
|
if (storage != NULL)
|
2015-04-24 20:33:34 +00:00
|
|
|
for (i = 0; i < mx; i++)
|
2015-07-21 14:06:03 +00:00
|
|
|
storage[i] = sk_EX_CALLBACK_value(ip->meth, i);
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
2016-03-05 21:54:02 +00:00
|
|
|
CRYPTO_THREAD_unlock(ex_data_lock);
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2017-03-19 15:14:33 +00:00
|
|
|
if (mx == 0)
|
|
|
|
return 1;
|
|
|
|
if (storage == NULL) {
|
2015-04-24 20:33:34 +00:00
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_DUP_EX_DATA, ERR_R_MALLOC_FAILURE);
|
2015-01-22 03:40:55 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2017-04-26 18:05:49 +00:00
|
|
|
/*
|
|
|
|
* Make sure the ex_data stack is at least |mx| elements long to avoid
|
|
|
|
* issues in the for loop that follows; so go get the |mx|'th element
|
|
|
|
* (if it does not exist CRYPTO_get_ex_data() returns NULL), and assign
|
|
|
|
* to itself. This is normally a no-op; but ensures the stack is the
|
|
|
|
* proper size
|
|
|
|
*/
|
|
|
|
if (!CRYPTO_set_ex_data(to, mx - 1, CRYPTO_get_ex_data(to, mx - 1)))
|
2017-03-19 15:14:33 +00:00
|
|
|
goto err;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
for (i = 0; i < mx; i++) {
|
|
|
|
ptr = CRYPTO_get_ex_data(from, i);
|
|
|
|
if (storage[i] && storage[i]->dup_func)
|
2017-03-19 15:14:33 +00:00
|
|
|
if (!storage[i]->dup_func(to, from, &ptr, i,
|
|
|
|
storage[i]->argl, storage[i]->argp))
|
|
|
|
goto err;
|
2015-01-22 03:40:55 +00:00
|
|
|
CRYPTO_set_ex_data(to, i, ptr);
|
|
|
|
}
|
2017-03-19 15:14:33 +00:00
|
|
|
toret = 1;
|
|
|
|
err:
|
2015-04-24 20:33:34 +00:00
|
|
|
if (storage != stack)
|
|
|
|
OPENSSL_free(storage);
|
2017-03-19 15:14:33 +00:00
|
|
|
return toret;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-04-24 20:33:34 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Cleanup a CRYPTO_EX_DATA variable - including calling free() callbacks for
|
|
|
|
* each index in the class used by this variable
|
|
|
|
*/
|
|
|
|
void CRYPTO_free_ex_data(int class_index, void *obj, CRYPTO_EX_DATA *ad)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
|
|
|
int mx, i;
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACKS *ip;
|
2015-01-22 03:40:55 +00:00
|
|
|
void *ptr;
|
2017-02-01 17:29:47 +00:00
|
|
|
EX_CALLBACK *f;
|
2015-07-21 14:06:03 +00:00
|
|
|
EX_CALLBACK *stack[10];
|
|
|
|
EX_CALLBACK **storage = NULL;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
if ((ip = get_and_lock(class_index)) == NULL)
|
2017-02-01 17:29:47 +00:00
|
|
|
goto err;
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-07-21 14:06:03 +00:00
|
|
|
mx = sk_EX_CALLBACK_num(ip->meth);
|
2015-01-22 03:40:55 +00:00
|
|
|
if (mx > 0) {
|
2015-04-24 20:33:34 +00:00
|
|
|
if (mx < (int)OSSL_NELEM(stack))
|
|
|
|
storage = stack;
|
|
|
|
else
|
|
|
|
storage = OPENSSL_malloc(sizeof(*storage) * mx);
|
2015-10-30 11:12:26 +00:00
|
|
|
if (storage != NULL)
|
2015-04-24 20:33:34 +00:00
|
|
|
for (i = 0; i < mx; i++)
|
2015-07-21 14:06:03 +00:00
|
|
|
storage[i] = sk_EX_CALLBACK_value(ip->meth, i);
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
2016-03-05 21:54:02 +00:00
|
|
|
CRYPTO_THREAD_unlock(ex_data_lock);
|
2015-04-24 20:33:34 +00:00
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
for (i = 0; i < mx; i++) {
|
2017-02-01 17:29:47 +00:00
|
|
|
if (storage != NULL)
|
|
|
|
f = storage[i];
|
|
|
|
else {
|
|
|
|
CRYPTO_THREAD_write_lock(ex_data_lock);
|
|
|
|
f = sk_EX_CALLBACK_value(ip->meth, i);
|
|
|
|
CRYPTO_THREAD_unlock(ex_data_lock);
|
|
|
|
}
|
|
|
|
if (f != NULL && f->free_func != NULL) {
|
2015-01-22 03:40:55 +00:00
|
|
|
ptr = CRYPTO_get_ex_data(ad, i);
|
2017-02-01 17:29:47 +00:00
|
|
|
f->free_func(obj, ptr, ad, i, f->argl, f->argp);
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
}
|
2015-04-24 20:33:34 +00:00
|
|
|
|
|
|
|
if (storage != stack)
|
|
|
|
OPENSSL_free(storage);
|
2017-02-01 17:29:47 +00:00
|
|
|
err:
|
2015-05-01 18:37:16 +00:00
|
|
|
sk_void_free(ad->sk);
|
|
|
|
ad->sk = NULL;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
First step in fixing "ex_data" support. Warning: big commit log ...
Currently, this change merely addresses where ex_data indexes are stored
and managed, and thus fixes the thread-safety issues that existed at that
level. "Class" code (eg. RSA, DSA, etc) no longer store their own STACKS
and per-class index counters - all such data is stored inside ex_data.c. So
rather than passing both STACK+counter to index-management ex_data
functions, a 'class_index' is instead passed to indicate the class (eg.
CRYPTO_EX_INDEX_RSA). New classes can be dynamically registered on-the-fly
and this is also thread-safe inside ex_data.c (though whether the caller
manages the return value in a thread-safe way is not addressed).
This does not change the "get/set" functions on individual "ex_data"
structures, and so thread-safety at that level isn't (yet) assured.
Likewise, the method of getting and storing per-class indexes has not
changed, so locking may still be required at the "caller" end but is
nonetheless thread-safe inside "ex_data"'s internal implementation.
Typically this occurs when code implements a new method of some kind and
stores its own per-class index in a global variable without locking the
setting and usage of that variable. If the code in question is likely to be
used in multiple threads, locking the setting and use of that index is
still up to the code in question. Possible fixes to this are being
sketched, but definitely require more major changes to the API itself than
this change undertakes.
The underlying implementation in ex_data.c has also been modularised so
that alternative "ex_data" implementations (that control all access to
state) can be plugged in. Eg. a loaded module can have its implementation
set to that of the application loaded it - the result being that
thread-safety and consistency of "ex_data" classes and indexes can be
maintained in the same place rather than the loaded module using its own
copy of ex_data support code and state.
Due to the centralisation of "state" with this change, cleanup of all
"ex_data" state can now be performed properly. Previously all allocation of
ex_data state was guaranteed to leak - and MemCheck_off() had been used to
avoid it flagging up the memory debugging. A new function has been added to
perfrom all this cleanup, CRYPTO_cleanup_all_ex_data(). The "openssl"
command(s) have been changed to use this cleanup, as have the relevant test
programs. External application code may want to do so too - failure to
cleanup will not induce more memory leaking than was the case before, but
the memory debugging is not tricked into hiding it any more so it may
"appear" where it previously did not.
2001-09-01 19:56:46 +00:00
|
|
|
|
2015-01-22 03:40:55 +00:00
|
|
|
/*
|
|
|
|
* For a given CRYPTO_EX_DATA variable, set the value corresponding to a
|
|
|
|
* particular index in the class used by this variable
|
|
|
|
*/
|
2000-01-23 23:41:49 +00:00
|
|
|
int CRYPTO_set_ex_data(CRYPTO_EX_DATA *ad, int idx, void *val)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (ad->sk == NULL) {
|
|
|
|
if ((ad->sk = sk_void_new_null()) == NULL) {
|
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_SET_EX_DATA, ERR_R_MALLOC_FAILURE);
|
2015-04-24 20:33:34 +00:00
|
|
|
return 0;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-24 20:33:34 +00:00
|
|
|
for (i = sk_void_num(ad->sk); i <= idx; ++i) {
|
2015-01-22 03:40:55 +00:00
|
|
|
if (!sk_void_push(ad->sk, NULL)) {
|
|
|
|
CRYPTOerr(CRYPTO_F_CRYPTO_SET_EX_DATA, ERR_R_MALLOC_FAILURE);
|
2015-04-24 20:33:34 +00:00
|
|
|
return 0;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
sk_void_set(ad->sk, idx, val);
|
2015-04-24 20:33:34 +00:00
|
|
|
return 1;
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For a given CRYPTO_EX_DATA_ variable, get the value corresponding to a
|
|
|
|
* particular index in the class used by this variable
|
|
|
|
*/
|
2000-11-06 23:24:59 +00:00
|
|
|
void *CRYPTO_get_ex_data(const CRYPTO_EX_DATA *ad, int idx)
|
2015-01-22 03:40:55 +00:00
|
|
|
{
|
2015-04-24 20:33:34 +00:00
|
|
|
if (ad->sk == NULL || idx >= sk_void_num(ad->sk))
|
|
|
|
return NULL;
|
|
|
|
return sk_void_value(ad->sk, idx);
|
2015-01-22 03:40:55 +00:00
|
|
|
}
|