mirror of
https://github.com/curl/curl.git
synced 2024-12-21 06:50:10 +08:00
2100d9fde2
The goal of this patch is to avoid unnecessary feature detection work
when doing Windows builds with CMake. Do this by pre-filling well-known
detection results for Windows and specifically for mingw-w64 and MSVC
compilers. Also limit feature checks to platforms where the results are
actually used. Drop a few redundant ones. And some tidying up.
- pre-fill remaining detection values in Windows CMake builds.
Based on actual detection results observed in CI runs, preceding
similar work over libssh2 and matching up values with
`lib/config-win32.h`.
This brings down CMake configuration time from 58 to 14 seconds on the
same local machine.
On AppVeyor CI this translates to:
- 128 seconds -> 50 seconds VS2022 MSVC with OpenSSL (per CMake job):
https://ci.appveyor.com/project/curlorg/curl/builds/48208419/job/4gw66ecrjpy7necb#L296
https://ci.appveyor.com/project/curlorg/curl/builds/48217440/job/8m4fwrr2fe249uo8#L186
- 62 seconds -> 16 seconds VS2017 MINGW (per CMake job):
https://ci.appveyor.com/project/curlorg/curl/builds/48208419/job/s1y8q5ivlcs7ub29?fullLog=true#L290
https://ci.appveyor.com/project/curlorg/curl/builds/48217440/job/pchpxyjsyc9kl13a?fullLog=true#L194
The formula is about 1-3 seconds delay for each detection. Almost all
of these trigger a full compile-link cycle behind the scenes, slow
even today, both cross and native, mingw-w64 and apparently MSVC too.
Enabling .map files or other custom build features slows it down
further. (Similar is expected for autotools configure.)
- stop detecting `idn2.h` if idn2 was deselected.
autotools does this.
- stop detecting `idn2.h` if idn2 was not found.
This deviates from autotools. Source code requires both header and
lib, so this is still correct, but faster.
- limit `ADDRESS_FAMILY` detection to Windows.
- normalize `HAVE_WIN32_WINNT` value to lowercase `0x0a12` format.
- pre-fill `HAVE_WIN32_WINNT`-dependent detection results.
Saving 4 (slow) feature-detections in most builds: `getaddrinfo`,
`freeaddrinfo`, `inet_ntop`, `inet_pton`
- fix pre-filled `HAVE_SYS_TIME_H`, `HAVE_SYS_PARAM_H`,
`HAVE_GETTIMEOFDAY` for mingw-w64.
Luckily this do not change build results, as `WIN32` took
priority over `HAVE_GETTIMEOFDAY` with the current source
code.
- limit `HAVE_CLOCK_GETTIME_MONOTONIC_RAW` and
`HAVE_CLOCK_GETTIME_MONOTONIC` detections to non-Windows.
We're not using these in the source code for Windows.
- reduce compiler warning noise in CMake internal logs:
- fix to include `winsock2.h` before `windows.h`.
Apply it to autotools test snippets too.
- delete previous `-D_WINSOCKAPI_=` hack that aimed to fix the above.
- cleanup `CMake/CurlTests.c` to emit less warnings.
- delete redundant `HAVE_MACRO_SIGSETJMP` feature check.
It was the same check as `HAVE_SIGSETJMP`.
- delete 'experimental' marking from `CURL_USE_OPENSSL`.
- show CMake version via `CMakeLists.txt`.
Credit to the `zlib-ng` project for the idea:
61e181c8ae/CMakeLists.txt (L7)
- make `CMake/CurlTests.c` pass `checksrc`.
- `CMake/WindowsCache.cmake` tidy-ups.
- replace `WIN32` guard with `_WIN32` in `CMake/CurlTests.c`.
Closes #12044
498 lines
10 KiB
C
498 lines
10 KiB
C
/***************************************************************************
|
|
* _ _ ____ _
|
|
* Project ___| | | | _ \| |
|
|
* / __| | | | |_) | |
|
|
* | (__| |_| | _ <| |___
|
|
* \___|\___/|_| \_\_____|
|
|
*
|
|
* Copyright (C) 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.
|
|
*
|
|
* SPDX-License-Identifier: curl
|
|
*
|
|
***************************************************************************/
|
|
|
|
#ifdef HAVE_FCNTL_O_NONBLOCK
|
|
/* headers for FCNTL_O_NONBLOCK test */
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
/* */
|
|
#if defined(sun) || defined(__sun__) || \
|
|
defined(__SUNPRO_C) || defined(__SUNPRO_CC)
|
|
# if defined(__SVR4) || defined(__srv4__)
|
|
# define PLATFORM_SOLARIS
|
|
# else
|
|
# define PLATFORM_SUNOS4
|
|
# endif
|
|
#endif
|
|
#if (defined(_AIX) || defined(__xlC__)) && !defined(_AIX41)
|
|
# define PLATFORM_AIX_V3
|
|
#endif
|
|
/* */
|
|
#if defined(PLATFORM_SUNOS4) || defined(PLATFORM_AIX_V3)
|
|
#error "O_NONBLOCK does not work on this platform"
|
|
#endif
|
|
|
|
int main(void)
|
|
{
|
|
/* O_NONBLOCK source test */
|
|
int flags = 0;
|
|
if(0 != fcntl(0, F_SETFL, flags | O_NONBLOCK))
|
|
return 1;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
/* tests for gethostbyname_r */
|
|
#if defined(HAVE_GETHOSTBYNAME_R_3_REENTRANT) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_5_REENTRANT) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_6_REENTRANT)
|
|
# define _REENTRANT
|
|
/* no idea whether _REENTRANT is always set, just invent a new flag */
|
|
# define TEST_GETHOSTBYFOO_REENTRANT
|
|
#endif
|
|
#if defined(HAVE_GETHOSTBYNAME_R_3) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_5) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_6) || \
|
|
defined(TEST_GETHOSTBYFOO_REENTRANT)
|
|
#include <sys/types.h>
|
|
#include <netdb.h>
|
|
int main(void)
|
|
{
|
|
char *address = "example.com";
|
|
int length = 0;
|
|
int type = 0;
|
|
struct hostent h;
|
|
int rc = 0;
|
|
#if defined(HAVE_GETHOSTBYNAME_R_3) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_3_REENTRANT)
|
|
struct hostent_data hdata;
|
|
#elif defined(HAVE_GETHOSTBYNAME_R_5) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_5_REENTRANT) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_6) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_6_REENTRANT)
|
|
char buffer[8192];
|
|
int h_errnop;
|
|
struct hostent *hp;
|
|
#endif
|
|
|
|
#if defined(HAVE_GETHOSTBYNAME_R_3) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_3_REENTRANT)
|
|
rc = gethostbyname_r(address, &h, &hdata);
|
|
#elif defined(HAVE_GETHOSTBYNAME_R_5) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_5_REENTRANT)
|
|
rc = gethostbyname_r(address, &h, buffer, 8192, &h_errnop);
|
|
(void)hp; /* not used for test */
|
|
#elif defined(HAVE_GETHOSTBYNAME_R_6) || \
|
|
defined(HAVE_GETHOSTBYNAME_R_6_REENTRANT)
|
|
rc = gethostbyname_r(address, &h, buffer, 8192, &hp, &h_errnop);
|
|
#endif
|
|
|
|
(void)length;
|
|
(void)type;
|
|
(void)rc;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_IN_ADDR_T
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <arpa/inet.h>
|
|
int main(void)
|
|
{
|
|
if((in_addr_t *) 0)
|
|
return 0;
|
|
if(sizeof(in_addr_t))
|
|
return 0;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_BOOL_T
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
#include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_STDBOOL_H
|
|
#include <stdbool.h>
|
|
#endif
|
|
int main(void)
|
|
{
|
|
if(sizeof(bool *))
|
|
return 0;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef STDC_HEADERS
|
|
#include <stdlib.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#include <float.h>
|
|
int main(void) { return 0; }
|
|
#endif
|
|
|
|
#ifdef HAVE_FILE_OFFSET_BITS
|
|
#ifdef _FILE_OFFSET_BITS
|
|
#undef _FILE_OFFSET_BITS
|
|
#endif
|
|
#define _FILE_OFFSET_BITS 64
|
|
#include <sys/types.h>
|
|
/* Check that off_t can represent 2**63 - 1 correctly.
|
|
We can't simply define LARGE_OFF_T to be 9223372036854775807,
|
|
since some C++ compilers masquerading as C compilers
|
|
incorrectly reject 9223372036854775807. */
|
|
#define LARGE_OFF_T (((off_t) 1 << 62) - 1 + ((off_t) 1 << 62))
|
|
int off_t_is_large[(LARGE_OFF_T % 2147483629 == 721
|
|
&& LARGE_OFF_T % 2147483647 == 1)
|
|
? 1 : -1];
|
|
int main(void) { ; return 0; }
|
|
#endif
|
|
|
|
#ifdef HAVE_IOCTLSOCKET
|
|
/* includes start */
|
|
#ifdef HAVE_WINDOWS_H
|
|
# ifndef WIN32_LEAN_AND_MEAN
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# endif
|
|
# ifdef HAVE_WINSOCK2_H
|
|
# include <winsock2.h>
|
|
# endif
|
|
# include <windows.h>
|
|
#endif
|
|
int main(void)
|
|
{
|
|
/* ioctlsocket source code */
|
|
int socket;
|
|
unsigned long flags = ioctlsocket(socket, FIONBIO, &flags);
|
|
;
|
|
return 0;
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_IOCTLSOCKET_CAMEL
|
|
/* includes start */
|
|
#ifdef HAVE_WINDOWS_H
|
|
# ifndef WIN32_LEAN_AND_MEAN
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# endif
|
|
# ifdef HAVE_WINSOCK2_H
|
|
# include <winsock2.h>
|
|
# endif
|
|
# include <windows.h>
|
|
#endif
|
|
int main(void)
|
|
{
|
|
/* IoctlSocket source code */
|
|
if(0 != IoctlSocket(0, 0, 0))
|
|
return 1;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_IOCTLSOCKET_CAMEL_FIONBIO
|
|
/* includes start */
|
|
#ifdef HAVE_WINDOWS_H
|
|
# ifndef WIN32_LEAN_AND_MEAN
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# endif
|
|
# ifdef HAVE_WINSOCK2_H
|
|
# include <winsock2.h>
|
|
# endif
|
|
# include <windows.h>
|
|
#endif
|
|
int main(void)
|
|
{
|
|
/* IoctlSocket source code */
|
|
long flags = 0;
|
|
if(0 != IoctlSocket(0, FIONBIO, &flags))
|
|
return 1;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_IOCTLSOCKET_FIONBIO
|
|
/* includes start */
|
|
#ifdef HAVE_WINDOWS_H
|
|
# ifndef WIN32_LEAN_AND_MEAN
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# endif
|
|
# ifdef HAVE_WINSOCK2_H
|
|
# include <winsock2.h>
|
|
# endif
|
|
# include <windows.h>
|
|
#endif
|
|
int main(void)
|
|
{
|
|
int flags = 0;
|
|
if(0 != ioctlsocket(0, FIONBIO, &flags))
|
|
return 1;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_IOCTL_FIONBIO
|
|
/* headers for FIONBIO test */
|
|
/* includes start */
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
# include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
# include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_SOCKET_H
|
|
# include <sys/socket.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_IOCTL_H
|
|
# include <sys/ioctl.h>
|
|
#endif
|
|
#ifdef HAVE_STROPTS_H
|
|
# include <stropts.h>
|
|
#endif
|
|
int main(void)
|
|
{
|
|
int flags = 0;
|
|
if(0 != ioctl(0, FIONBIO, &flags))
|
|
return 1;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_IOCTL_SIOCGIFADDR
|
|
/* headers for FIONBIO test */
|
|
/* includes start */
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
# include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
# include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_SOCKET_H
|
|
# include <sys/socket.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_IOCTL_H
|
|
# include <sys/ioctl.h>
|
|
#endif
|
|
#ifdef HAVE_STROPTS_H
|
|
# include <stropts.h>
|
|
#endif
|
|
#include <net/if.h>
|
|
int main(void)
|
|
{
|
|
struct ifreq ifr;
|
|
if(0 != ioctl(0, SIOCGIFADDR, &ifr))
|
|
return 1;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_SETSOCKOPT_SO_NONBLOCK
|
|
/* includes start */
|
|
#ifdef HAVE_WINDOWS_H
|
|
# ifndef WIN32_LEAN_AND_MEAN
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# endif
|
|
# ifdef HAVE_WINSOCK2_H
|
|
# include <winsock2.h>
|
|
# endif
|
|
# include <windows.h>
|
|
#endif
|
|
/* includes start */
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
# include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_SOCKET_H
|
|
# include <sys/socket.h>
|
|
#endif
|
|
/* includes end */
|
|
int main(void)
|
|
{
|
|
if(0 != setsockopt(0, SOL_SOCKET, SO_NONBLOCK, 0, 0))
|
|
return 1;
|
|
;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_GLIBC_STRERROR_R
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
void check(char c) {}
|
|
|
|
int main(void)
|
|
{
|
|
char buffer[1024];
|
|
/* This will not compile if strerror_r does not return a char* */
|
|
check(strerror_r(EACCES, buffer, sizeof(buffer))[0]);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_POSIX_STRERROR_R
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
/* float, because a pointer can't be implicitly cast to float */
|
|
void check(float f) {}
|
|
|
|
int main(void)
|
|
{
|
|
char buffer[1024];
|
|
/* This will not compile if strerror_r does not return an int */
|
|
check(strerror_r(EACCES, buffer, sizeof(buffer)));
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_FSETXATTR_6
|
|
#include <sys/xattr.h> /* header from libc, not from libattr */
|
|
int main(void)
|
|
{
|
|
fsetxattr(0, 0, 0, 0, 0, 0);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_FSETXATTR_5
|
|
#include <sys/xattr.h> /* header from libc, not from libattr */
|
|
int main(void)
|
|
{
|
|
fsetxattr(0, 0, 0, 0, 0);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_CLOCK_GETTIME_MONOTONIC
|
|
#include <time.h>
|
|
int main(void)
|
|
{
|
|
struct timespec ts = {0, 0};
|
|
clock_gettime(CLOCK_MONOTONIC, &ts);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_BUILTIN_AVAILABLE
|
|
int main(void)
|
|
{
|
|
if(__builtin_available(macOS 10.12, *)) {}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_VARIADIC_MACROS_C99
|
|
#define c99_vmacro3(first, ...) fun3(first, __VA_ARGS__)
|
|
#define c99_vmacro2(first, ...) fun2(first, __VA_ARGS__)
|
|
|
|
int fun3(int arg1, int arg2, int arg3);
|
|
int fun2(int arg1, int arg2);
|
|
|
|
int fun3(int arg1, int arg2, int arg3)
|
|
{
|
|
return arg1 + arg2 + arg3;
|
|
}
|
|
int fun2(int arg1, int arg2)
|
|
{
|
|
return arg1 + arg2;
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
int res3 = c99_vmacro3(1, 2, 3);
|
|
int res2 = c99_vmacro2(1, 2);
|
|
(void)res3;
|
|
(void)res2;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_VARIADIC_MACROS_GCC
|
|
#define gcc_vmacro3(first, args...) fun3(first, args)
|
|
#define gcc_vmacro2(first, args...) fun2(first, args)
|
|
|
|
int fun3(int arg1, int arg2, int arg3);
|
|
int fun2(int arg1, int arg2);
|
|
|
|
int fun3(int arg1, int arg2, int arg3)
|
|
{
|
|
return arg1 + arg2 + arg3;
|
|
}
|
|
int fun2(int arg1, int arg2)
|
|
{
|
|
return arg1 + arg2;
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
int res3 = gcc_vmacro3(1, 2, 3);
|
|
int res2 = gcc_vmacro2(1, 2);
|
|
(void)res3;
|
|
(void)res2;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_ATOMIC
|
|
/* includes start */
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
# include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
# include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_STDATOMIC_H
|
|
# include <stdatomic.h>
|
|
#endif
|
|
/* includes end */
|
|
|
|
int main(void)
|
|
{
|
|
_Atomic int i = 1;
|
|
i = 0; /* Force an atomic-write operation. */
|
|
return i;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_WIN32_WINNT
|
|
/* includes start */
|
|
#ifdef _WIN32
|
|
# ifndef WIN32_LEAN_AND_MEAN
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# endif
|
|
# ifndef NOGDI
|
|
# define NOGDI
|
|
# endif
|
|
# include <windows.h>
|
|
#endif
|
|
/* includes end */
|
|
|
|
#define enquote(x) #x
|
|
#define expand(x) enquote(x)
|
|
#pragma message("_WIN32_WINNT=" expand(_WIN32_WINNT))
|
|
|
|
int main(void)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|