mirror of
https://github.com/curl/curl.git
synced 2025-09-17 17:42:49 +03:00
curl_version_info: add CURL_VERSION_THREADSAFE_INIT
This flag can be used to make sure that curl_global_init() is thread-safe. This can be useful for libraries that can't control what other dependencies are doing with Curl. Closes #8680
This commit is contained in:
parent
23af112f55
commit
2ed1012564
17
configure.ac
17
configure.ac
|
@ -4258,6 +4258,23 @@ if test ${ac_cv_sizeof_curl_off_t} -gt 4; then
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if test "$tst_atomic" = "yes"; then
|
||||||
|
SUPPORT_FEATURES="$SUPPORT_FEATURES threadsafe-init"
|
||||||
|
else
|
||||||
|
AC_COMPILE_IFELSE([
|
||||||
|
AC_LANG_PROGRAM([[
|
||||||
|
#include <windows.h>
|
||||||
|
]],[[
|
||||||
|
#if (WINVER < 0x600) && (_WIN32_WINNT < 0x600)
|
||||||
|
#error
|
||||||
|
#endif
|
||||||
|
]])
|
||||||
|
],[
|
||||||
|
SUPPORT_FEATURES="$SUPPORT_FEATURES threadsafe-init"
|
||||||
|
],[
|
||||||
|
])
|
||||||
|
fi
|
||||||
|
|
||||||
dnl replace spaces with newlines
|
dnl replace spaces with newlines
|
||||||
dnl sort the lines
|
dnl sort the lines
|
||||||
dnl replace the newlines back to spaces
|
dnl replace the newlines back to spaces
|
||||||
|
|
|
@ -204,6 +204,9 @@ libcurl was built with support for SSPI. This is only available on Windows and
|
||||||
makes libcurl use Windows-provided functions for Kerberos, NTLM, SPNEGO and
|
makes libcurl use Windows-provided functions for Kerberos, NTLM, SPNEGO and
|
||||||
Digest authentication. It also allows libcurl to use the current user
|
Digest authentication. It also allows libcurl to use the current user
|
||||||
credentials without the app having to pass them on. (Added in 7.13.2)
|
credentials without the app having to pass them on. (Added in 7.13.2)
|
||||||
|
.IP CURL_VERSION_THREADSAFE_INIT
|
||||||
|
libcurl was built with thread-safety support (Atomic or SRWLOCK) to protect
|
||||||
|
curl initialisation. (Added in 7.84.0)
|
||||||
.IP CURL_VERSION_TLSAUTH_SRP
|
.IP CURL_VERSION_TLSAUTH_SRP
|
||||||
libcurl was built with support for TLS-SRP (in one or more of the built-in TLS
|
libcurl was built with support for TLS-SRP (in one or more of the built-in TLS
|
||||||
backends). (Added in 7.21.4)
|
backends). (Added in 7.21.4)
|
||||||
|
|
|
@ -172,6 +172,7 @@ CURL_VERSION_PSL 7.47.0
|
||||||
CURL_VERSION_SPNEGO 7.10.8
|
CURL_VERSION_SPNEGO 7.10.8
|
||||||
CURL_VERSION_SSL 7.10
|
CURL_VERSION_SSL 7.10
|
||||||
CURL_VERSION_SSPI 7.13.2
|
CURL_VERSION_SSPI 7.13.2
|
||||||
|
CURL_VERSION_THREADSAFE_INIT 7.84.0
|
||||||
CURL_VERSION_TLSAUTH_SRP 7.21.4
|
CURL_VERSION_TLSAUTH_SRP 7.21.4
|
||||||
CURL_VERSION_UNICODE 7.72.0
|
CURL_VERSION_UNICODE 7.72.0
|
||||||
CURL_VERSION_UNIX_SOCKETS 7.40.0
|
CURL_VERSION_UNIX_SOCKETS 7.40.0
|
||||||
|
|
|
@ -2607,8 +2607,10 @@ CURL_EXTERN void curl_free(void *p);
|
||||||
*
|
*
|
||||||
* curl_global_init() should be invoked exactly once for each application that
|
* curl_global_init() should be invoked exactly once for each application that
|
||||||
* uses libcurl and before any call of other libcurl functions.
|
* uses libcurl and before any call of other libcurl functions.
|
||||||
*
|
|
||||||
* This function is not thread-safe!
|
* This function is thread-safe if CURL_VERSION_THREADSAFE_INIT is set in the
|
||||||
|
* curl_version_info_data.features flag (fetch by curl_version_info()).
|
||||||
|
|
||||||
*/
|
*/
|
||||||
CURL_EXTERN CURLcode curl_global_init(long flags);
|
CURL_EXTERN CURLcode curl_global_init(long flags);
|
||||||
|
|
||||||
|
@ -3025,6 +3027,8 @@ typedef struct curl_version_info_data curl_version_info_data;
|
||||||
#define CURL_VERSION_UNICODE (1<<27) /* Unicode support on Windows */
|
#define CURL_VERSION_UNICODE (1<<27) /* Unicode support on Windows */
|
||||||
#define CURL_VERSION_HSTS (1<<28) /* HSTS is supported */
|
#define CURL_VERSION_HSTS (1<<28) /* HSTS is supported */
|
||||||
#define CURL_VERSION_GSASL (1<<29) /* libgsasl is supported */
|
#define CURL_VERSION_GSASL (1<<29) /* libgsasl is supported */
|
||||||
|
#define CURL_VERSION_THREADSAFE_INIT (1<<30) /* curl_global_init/cleanup() are
|
||||||
|
thread-safe */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* NAME curl_version_info()
|
* NAME curl_version_info()
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
#include "vssh/ssh.h"
|
#include "vssh/ssh.h"
|
||||||
#include "quic.h"
|
#include "quic.h"
|
||||||
#include "curl_printf.h"
|
#include "curl_printf.h"
|
||||||
|
#include "easy_lock.h"
|
||||||
|
|
||||||
#ifdef USE_ARES
|
#ifdef USE_ARES
|
||||||
# if defined(CURL_STATICLIB) && !defined(CARES_STATICLIB) && \
|
# if defined(CURL_STATICLIB) && !defined(CARES_STATICLIB) && \
|
||||||
|
@ -450,6 +451,9 @@ static curl_version_info_data version_info = {
|
||||||
#endif
|
#endif
|
||||||
#if defined(USE_GSASL)
|
#if defined(USE_GSASL)
|
||||||
| CURL_VERSION_GSASL
|
| CURL_VERSION_GSASL
|
||||||
|
#endif
|
||||||
|
#if defined(GLOBAL_INIT_IS_THREADSAFE)
|
||||||
|
| CURL_VERSION_THREADSAFE_INIT
|
||||||
#endif
|
#endif
|
||||||
,
|
,
|
||||||
NULL, /* ssl_version */
|
NULL, /* ssl_version */
|
||||||
|
|
|
@ -148,6 +148,8 @@
|
||||||
d c X'10000000'
|
d c X'10000000'
|
||||||
d CURL_VERSION_GSASL...
|
d CURL_VERSION_GSASL...
|
||||||
d c X'20000000'
|
d c X'20000000'
|
||||||
|
d CURL_VERSION_THREADSAFE_INIT...
|
||||||
|
d c X'40000000'
|
||||||
*
|
*
|
||||||
d CURL_HTTPPOST_FILENAME...
|
d CURL_HTTPPOST_FILENAME...
|
||||||
d c X'00000001'
|
d c X'00000001'
|
||||||
|
|
|
@ -110,6 +110,7 @@ static const struct feat feats[] = {
|
||||||
{"alt-svc", CURL_VERSION_ALTSVC},
|
{"alt-svc", CURL_VERSION_ALTSVC},
|
||||||
{"HSTS", CURL_VERSION_HSTS},
|
{"HSTS", CURL_VERSION_HSTS},
|
||||||
{"gsasl", CURL_VERSION_GSASL},
|
{"gsasl", CURL_VERSION_GSASL},
|
||||||
|
{"threadsafe-init",CURL_VERSION_THREADSAFE_INIT},
|
||||||
};
|
};
|
||||||
|
|
||||||
static void print_category(curlhelp_t category)
|
static void print_category(curlhelp_t category)
|
||||||
|
|
|
@ -245,4 +245,4 @@ test2200 test2201 test2202 test2203 test2204 test2205 \
|
||||||
test3000 test3001 test3002 test3003 test3004 test3005 test3006 test3007 \
|
test3000 test3001 test3002 test3003 test3004 test3005 test3006 test3007 \
|
||||||
test3008 test3009 test3010 test3011 test3012 test3013 test3014 test3015 \
|
test3008 test3009 test3010 test3011 test3012 test3013 test3014 test3015 \
|
||||||
test3016 test3017 test3018 test3019 test3020 test3021 test3022 test3023 \
|
test3016 test3017 test3018 test3019 test3020 test3021 test3022 test3023 \
|
||||||
test3024 test3025
|
test3024 test3025 test3026
|
||||||
|
|
|
@ -63,7 +63,7 @@ noinst_PROGRAMS = chkhostname libauthretry libntlmconnect \
|
||||||
lib1915 lib1916 lib1917 lib1918 lib1919 \
|
lib1915 lib1916 lib1917 lib1918 lib1919 \
|
||||||
lib1933 lib1934 lib1935 lib1936 lib1937 lib1938 lib1939 lib1940 \
|
lib1933 lib1934 lib1935 lib1936 lib1937 lib1938 lib1939 lib1940 \
|
||||||
lib1945 lib1946 \
|
lib1945 lib1946 \
|
||||||
lib3010 lib3025
|
lib3010 lib3025 lib3026
|
||||||
|
|
||||||
chkdecimalpoint_SOURCES = chkdecimalpoint.c ../../lib/mprintf.c \
|
chkdecimalpoint_SOURCES = chkdecimalpoint.c ../../lib/mprintf.c \
|
||||||
../../lib/curl_ctype.c ../../lib/dynbuf.c ../../lib/strdup.c
|
../../lib/curl_ctype.c ../../lib/dynbuf.c ../../lib/strdup.c
|
||||||
|
@ -747,3 +747,7 @@ lib3010_CPPFLAGS = $(AM_CPPFLAGS)
|
||||||
lib3025_SOURCES = lib3025.c $(SUPPORTFILES) $(TESTUTIL) $(WARNLESS)
|
lib3025_SOURCES = lib3025.c $(SUPPORTFILES) $(TESTUTIL) $(WARNLESS)
|
||||||
lib3025_LDADD = $(TESTUTIL_LIBS)
|
lib3025_LDADD = $(TESTUTIL_LIBS)
|
||||||
lib3025_CPPFLAGS = $(AM_CPPFLAGS)
|
lib3025_CPPFLAGS = $(AM_CPPFLAGS)
|
||||||
|
|
||||||
|
lib3026_SOURCES = lib3026.c $(SUPPORTFILES) $(TESTUTIL) $(WARNLESS)
|
||||||
|
lib3026_LDADD = $(TESTUTIL_LIBS)
|
||||||
|
lib3026_CPPFLAGS = $(AM_CPPFLAGS)
|
||||||
|
|
101
tests/libtest/lib3026.c
Normal file
101
tests/libtest/lib3026.c
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
/***************************************************************************
|
||||||
|
* _ _ ____ _
|
||||||
|
* Project ___| | | | _ \| |
|
||||||
|
* / __| | | | |_) | |
|
||||||
|
* | (__| |_| | _ <| |___
|
||||||
|
* \___|\___/|_| \_\_____|
|
||||||
|
*
|
||||||
|
* Copyright (C) 1998 - 2022, Daniel Stenberg, <daniel@haxx.se>, et al.
|
||||||
|
*
|
||||||
|
* This software is licensed as described in the file COPYING, which
|
||||||
|
* you should have received as part of this distribution. The terms
|
||||||
|
* are also available at https://curl.se/docs/copyright.html.
|
||||||
|
*
|
||||||
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
||||||
|
* copies of the Software, and permit persons to whom the Software is
|
||||||
|
* furnished to do so, under the terms of the COPYING file.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
***************************************************************************/
|
||||||
|
#include "test.h"
|
||||||
|
|
||||||
|
#include "testutil.h"
|
||||||
|
#include "warnless.h"
|
||||||
|
|
||||||
|
#ifdef HAVE_PTHREAD_H
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#define NUM_THREADS 1000
|
||||||
|
|
||||||
|
static void *run_thread(void *ptr)
|
||||||
|
{
|
||||||
|
CURLcode *result = ptr;
|
||||||
|
|
||||||
|
*result = curl_global_init(CURL_GLOBAL_ALL);
|
||||||
|
if(*result == CURLE_OK)
|
||||||
|
curl_global_cleanup();
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int test(char *URL)
|
||||||
|
{
|
||||||
|
CURLcode results[NUM_THREADS];
|
||||||
|
pthread_t tids[NUM_THREADS];
|
||||||
|
unsigned tid_count = NUM_THREADS, i;
|
||||||
|
int test_failure = 0;
|
||||||
|
curl_version_info_data *ver;
|
||||||
|
(void) URL;
|
||||||
|
|
||||||
|
ver = curl_version_info(CURLVERSION_NOW);
|
||||||
|
if((ver->features & CURL_VERSION_THREADSAFE_INIT) == 0) {
|
||||||
|
fprintf(stderr, "%s:%d Have pthread but the "
|
||||||
|
"CURL_VERSION_THREADSAFE_INIT feature flag is not set\n",
|
||||||
|
__FILE__, __LINE__);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(i = 0; i < tid_count; i++) {
|
||||||
|
int res = pthread_create(&tids[i], NULL, run_thread, &results[i]);
|
||||||
|
if(res) {
|
||||||
|
fprintf(stderr, "%s:%d Couldn't create thread, errno %d\n",
|
||||||
|
__FILE__, __LINE__, res);
|
||||||
|
tid_count = i;
|
||||||
|
test_failure = -1;
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cleanup:
|
||||||
|
for(i = 0; i < tid_count; i++) {
|
||||||
|
pthread_join(tids[i], NULL);
|
||||||
|
if(results[i] != CURLE_OK) {
|
||||||
|
fprintf(stderr, "%s:%d thread[%u]: curl_global_init() failed,"
|
||||||
|
"with code %d (%s)\n", __FILE__, __LINE__,
|
||||||
|
i, (int) results[i], curl_easy_strerror(results[i]));
|
||||||
|
test_failure = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return test_failure;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* without pthread, this test doesn't work */
|
||||||
|
int test(char *URL)
|
||||||
|
{
|
||||||
|
curl_version_info_data *ver;
|
||||||
|
(void)URL;
|
||||||
|
|
||||||
|
ver = curl_version_info(CURLVERSION_NOW);
|
||||||
|
if((ver->features & CURL_VERSION_THREADSAFE_INIT) != 0) {
|
||||||
|
fprintf(stderr, "%s:%d No pthread but the "
|
||||||
|
"CURL_VERSION_THREADSAFE_INIT feature flag is set\n",
|
||||||
|
__FILE__, __LINE__);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
Loading…
Reference in New Issue
Block a user