In the async code for MacOS/X define _XOPEN_SOURCE (if not already
defined) as early as possible. We must do this before including
any header files, because on MacOS/X <stlib.h> includes <signal.h>
which includes <ucontext.h>. If we delay defining _XOPEN_SOURCE
and include <ucontext.h> after various system headers are included,
we are very likely to end up with the wrong (truncated) definition
of ucontext_t.
Also, better error handling and some code cleanup in POSIX fibre
construction and destruction. We make sure that async_fibre_makecontext()
always initializes the fibre to a state that can be freed.
For all implementations, check for error returns from
async_fibre_makecontext().
Reviewed-by: Matt Caswell <matt@openssl.org>
* ====================================================================
*/
* ====================================================================
*/
+/* This must be the first #include file */
#include "../async_locl.h"
#include "../async_locl.h"
-#include <openssl/async.h>
* ====================================================================
*/
* ====================================================================
*/
+/* This must be the first #include file */
#include "../async_locl.h"
#include "../async_locl.h"
-#include <openssl/async.h>
-# include <openssl/crypto.h>
-# include <openssl/async.h>
pthread_key_t posixctx;
pthread_key_t posixpool;
pthread_key_t posixctx;
pthread_key_t posixpool;
-int async_fibre_init(async_fibre *fibre)
+int async_fibre_makecontext(async_fibre *fibre)
- void *stack = NULL;
-
- stack = OPENSSL_malloc(STACKSIZE);
- if (stack == NULL) {
- return 0;
- }
-
- fibre->fibre.uc_stack.ss_sp = stack;
- fibre->fibre.uc_stack.ss_size = STACKSIZE;
- fibre->fibre.uc_link = NULL;
+ if (getcontext(&fibre->fibre) == 0) {
+ fibre->fibre.uc_stack.ss_sp = OPENSSL_malloc(STACKSIZE);
+ if (fibre->fibre.uc_stack.ss_sp != NULL) {
+ fibre->fibre.uc_stack.ss_size = STACKSIZE;
+ fibre->fibre.uc_link = NULL;
+ makecontext(&fibre->fibre, async_start_func, 0);
+ return 1;
+ }
+ } else {
+ fibre->fibre.uc_stack.ss_sp = NULL;
+ }
+ return 0;
}
void async_fibre_free(async_fibre *fibre)
{
}
void async_fibre_free(async_fibre *fibre)
{
- if (fibre->fibre.uc_stack.ss_sp)
- OPENSSL_free(fibre->fibre.uc_stack.ss_sp);
+ OPENSSL_free(fibre->fibre.uc_stack.ss_sp);
+ fibre->fibre.uc_stack.ss_sp = NULL;
}
int async_pipe(OSSL_ASYNC_FD *pipefds)
}
int async_pipe(OSSL_ASYNC_FD *pipefds)
* OF THE POSSIBILITY OF SUCH DAMAGE.
* ====================================================================
*/
* OF THE POSSIBILITY OF SUCH DAMAGE.
* ====================================================================
*/
+#ifndef OPENSSL_ASYNC_ARCH_ASYNC_POSIX_H
+#define OPENSSL_ASYNC_ARCH_ASYNC_POSIX_H
#include <openssl/e_os2.h>
#if defined(OPENSSL_SYS_UNIX) && defined(OPENSSL_THREADS)
#include <openssl/e_os2.h>
#if defined(OPENSSL_SYS_UNIX) && defined(OPENSSL_THREADS)
# define ASYNC_POSIX
# define ASYNC_ARCH
# define ASYNC_POSIX
# define ASYNC_ARCH
-/*
- * Some platforms complain (e.g. OS-X) that setcontext/getcontext/makecontext
- * are deprecated without the following defined. We know its deprecated but
- * there is no alternative.
- */
-# define _XOPEN_SOURCE
-# pragma GCC diagnostic ignored "-Wdeprecated-declarations"
-
# include <ucontext.h>
# include <setjmp.h>
# include "e_os.h"
# include <ucontext.h>
# include <setjmp.h>
# include "e_os.h"
-# define async_fibre_makecontext(c) \
- (!getcontext(&(c)->fibre) \
- && async_fibre_init(c) \
- && (makecontext(&(c)->fibre, async_start_func, 0), 1))
# define async_fibre_init_dispatcher(d)
# define async_fibre_init_dispatcher(d)
-int async_fibre_init(async_fibre *fibre);
+int async_fibre_makecontext(async_fibre *fibre);
void async_fibre_free(async_fibre *fibre);
# endif
#endif
void async_fibre_free(async_fibre *fibre);
# endif
#endif
+#endif /* OPENSSL_ASYNC_ARCH_ASYNC_POSIX_H */
* ====================================================================
*/
* ====================================================================
*/
+/* This must be the first #include file */
#include "../async_locl.h"
#ifdef ASYNC_WIN
#include "../async_locl.h"
#ifdef ASYNC_WIN
* ====================================================================
*/
* ====================================================================
*/
-#include <openssl/async.h>
-
/*
* This is the same detection used in cryptlib to set up the thread local
* storage that we depend on, so just copy that
*/
#if defined(_WIN32) || defined(__CYGWIN__)
/*
* This is the same detection used in cryptlib to set up the thread local
* storage that we depend on, so just copy that
*/
#if defined(_WIN32) || defined(__CYGWIN__)
+#include <openssl/async.h>
# define ASYNC_WIN
# define ASYNC_ARCH
# define ASYNC_WIN
# define ASYNC_ARCH
*/
#undef _FORTIFY_SOURCE
*/
#undef _FORTIFY_SOURCE
+/* This must be the first #include file */
+#include "async_locl.h"
+
-#include <openssl/async.h>
#define ASYNC_JOB_RUNNING 0
#define ASYNC_JOB_PAUSING 1
#define ASYNC_JOB_RUNNING 0
#define ASYNC_JOB_PAUSING 1
return NULL;
job = async_job_new();
return NULL;
job = async_job_new();
- if (job) {
- async_fibre_makecontext(&job->fibrectx);
+ if (job != NULL) {
+ if (! async_fibre_makecontext(&job->fibrectx)) {
+ async_job_free(job);
+ return NULL;
+ }
pool->max_size = max_size;
/* Pre-create jobs as required */
pool->max_size = max_size;
/* Pre-create jobs as required */
ASYNC_JOB *job;
job = async_job_new();
ASYNC_JOB *job;
job = async_job_new();
- if (job) {
- async_fibre_makecontext(&job->fibrectx);
- job->funcargs = NULL;
- sk_ASYNC_JOB_push(pool->jobs, job);
- curr_size++;
- init_size--;
- } else {
+ if (job == NULL || !async_fibre_makecontext(&job->fibrectx)) {
- * Not actually fatal because we already created the pool, just skip
- * creation of any more jobs
+ * Not actually fatal because we already created the pool, just
+ * skip creation of any more jobs
+ async_job_free(job);
+ break;
+ job->funcargs = NULL;
+ sk_ASYNC_JOB_push(pool->jobs, job);
+ curr_size++;
}
pool->curr_size = curr_size;
if (!async_set_pool(pool)) {
}
pool->curr_size = curr_size;
if (!async_set_pool(pool)) {
* ====================================================================
*/
* ====================================================================
*/
+/*
+ * Must do this before including any header files, because on MacOS/X <stlib.h>
+ * includes <signal.h> which includes <ucontext.h>
+ */
+#if defined(__APPLE__) && defined(__MACH__) && !defined(_XOPEN_SOURCE)
+# define _XOPEN_SOURCE /* Otherwise incomplete ucontext_t structure */
+# pragma GCC diagnostic ignored "-Wdeprecated-declarations"
+#endif
+
#include <openssl/async.h>
#include <openssl/crypto.h>
#include <openssl/async.h>
#include <openssl/crypto.h>