xref: /aosp_15_r20/external/curl/lib/system_win32.c (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker  *                                  _   _ ____  _
3*6236dae4SAndroid Build Coastguard Worker  *  Project                     ___| | | |  _ \| |
4*6236dae4SAndroid Build Coastguard Worker  *                             / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker  *                            | (__| |_| |  _ <| |___
6*6236dae4SAndroid Build Coastguard Worker  *                             \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker  *
8*6236dae4SAndroid Build Coastguard Worker  * Copyright (C) Steve Holme, <[email protected]>.
9*6236dae4SAndroid Build Coastguard Worker  *
10*6236dae4SAndroid Build Coastguard Worker  * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker  * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker  * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker  *
14*6236dae4SAndroid Build Coastguard Worker  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker  * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker  * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker  *
18*6236dae4SAndroid Build Coastguard Worker  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker  * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker  *
21*6236dae4SAndroid Build Coastguard Worker  * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker  *
23*6236dae4SAndroid Build Coastguard Worker  ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker 
25*6236dae4SAndroid Build Coastguard Worker #include "curl_setup.h"
26*6236dae4SAndroid Build Coastguard Worker 
27*6236dae4SAndroid Build Coastguard Worker #if defined(_WIN32)
28*6236dae4SAndroid Build Coastguard Worker 
29*6236dae4SAndroid Build Coastguard Worker #include <curl/curl.h>
30*6236dae4SAndroid Build Coastguard Worker #include "system_win32.h"
31*6236dae4SAndroid Build Coastguard Worker #include "version_win32.h"
32*6236dae4SAndroid Build Coastguard Worker #include "curl_sspi.h"
33*6236dae4SAndroid Build Coastguard Worker #include "warnless.h"
34*6236dae4SAndroid Build Coastguard Worker 
35*6236dae4SAndroid Build Coastguard Worker /* The last #include files should be: */
36*6236dae4SAndroid Build Coastguard Worker #include "curl_memory.h"
37*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h"
38*6236dae4SAndroid Build Coastguard Worker 
39*6236dae4SAndroid Build Coastguard Worker LARGE_INTEGER Curl_freq;
40*6236dae4SAndroid Build Coastguard Worker bool Curl_isVistaOrGreater;
41*6236dae4SAndroid Build Coastguard Worker 
42*6236dae4SAndroid Build Coastguard Worker /* Handle of iphlpapp.dll */
43*6236dae4SAndroid Build Coastguard Worker static HMODULE s_hIpHlpApiDll = NULL;
44*6236dae4SAndroid Build Coastguard Worker 
45*6236dae4SAndroid Build Coastguard Worker /* Pointer to the if_nametoindex function */
46*6236dae4SAndroid Build Coastguard Worker IF_NAMETOINDEX_FN Curl_if_nametoindex = NULL;
47*6236dae4SAndroid Build Coastguard Worker 
48*6236dae4SAndroid Build Coastguard Worker /* Curl_win32_init() performs Win32 global initialization */
Curl_win32_init(long flags)49*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_win32_init(long flags)
50*6236dae4SAndroid Build Coastguard Worker {
51*6236dae4SAndroid Build Coastguard Worker   /* CURL_GLOBAL_WIN32 controls the *optional* part of the initialization which
52*6236dae4SAndroid Build Coastguard Worker      is just for Winsock at the moment. Any required Win32 initialization
53*6236dae4SAndroid Build Coastguard Worker      should take place after this block. */
54*6236dae4SAndroid Build Coastguard Worker   if(flags & CURL_GLOBAL_WIN32) {
55*6236dae4SAndroid Build Coastguard Worker #ifdef USE_WINSOCK
56*6236dae4SAndroid Build Coastguard Worker     WORD wVersionRequested;
57*6236dae4SAndroid Build Coastguard Worker     WSADATA wsaData;
58*6236dae4SAndroid Build Coastguard Worker     int res;
59*6236dae4SAndroid Build Coastguard Worker 
60*6236dae4SAndroid Build Coastguard Worker     wVersionRequested = MAKEWORD(2, 2);
61*6236dae4SAndroid Build Coastguard Worker     res = WSAStartup(wVersionRequested, &wsaData);
62*6236dae4SAndroid Build Coastguard Worker 
63*6236dae4SAndroid Build Coastguard Worker     if(res)
64*6236dae4SAndroid Build Coastguard Worker       /* Tell the user that we could not find a usable */
65*6236dae4SAndroid Build Coastguard Worker       /* winsock.dll.     */
66*6236dae4SAndroid Build Coastguard Worker       return CURLE_FAILED_INIT;
67*6236dae4SAndroid Build Coastguard Worker 
68*6236dae4SAndroid Build Coastguard Worker     /* Confirm that the Windows Sockets DLL supports what we need.*/
69*6236dae4SAndroid Build Coastguard Worker     /* Note that if the DLL supports versions greater */
70*6236dae4SAndroid Build Coastguard Worker     /* than wVersionRequested, it will still return */
71*6236dae4SAndroid Build Coastguard Worker     /* wVersionRequested in wVersion. wHighVersion contains the */
72*6236dae4SAndroid Build Coastguard Worker     /* highest supported version. */
73*6236dae4SAndroid Build Coastguard Worker 
74*6236dae4SAndroid Build Coastguard Worker     if(LOBYTE(wsaData.wVersion) != LOBYTE(wVersionRequested) ||
75*6236dae4SAndroid Build Coastguard Worker        HIBYTE(wsaData.wVersion) != HIBYTE(wVersionRequested) ) {
76*6236dae4SAndroid Build Coastguard Worker       /* Tell the user that we could not find a usable */
77*6236dae4SAndroid Build Coastguard Worker 
78*6236dae4SAndroid Build Coastguard Worker       /* winsock.dll. */
79*6236dae4SAndroid Build Coastguard Worker       WSACleanup();
80*6236dae4SAndroid Build Coastguard Worker       return CURLE_FAILED_INIT;
81*6236dae4SAndroid Build Coastguard Worker     }
82*6236dae4SAndroid Build Coastguard Worker     /* The Windows Sockets DLL is acceptable. Proceed. */
83*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_LWIPSOCK)
84*6236dae4SAndroid Build Coastguard Worker     lwip_init();
85*6236dae4SAndroid Build Coastguard Worker #endif
86*6236dae4SAndroid Build Coastguard Worker   } /* CURL_GLOBAL_WIN32 */
87*6236dae4SAndroid Build Coastguard Worker 
88*6236dae4SAndroid Build Coastguard Worker #ifdef USE_WINDOWS_SSPI
89*6236dae4SAndroid Build Coastguard Worker   {
90*6236dae4SAndroid Build Coastguard Worker     CURLcode result = Curl_sspi_global_init();
91*6236dae4SAndroid Build Coastguard Worker     if(result)
92*6236dae4SAndroid Build Coastguard Worker       return result;
93*6236dae4SAndroid Build Coastguard Worker   }
94*6236dae4SAndroid Build Coastguard Worker #endif
95*6236dae4SAndroid Build Coastguard Worker 
96*6236dae4SAndroid Build Coastguard Worker   s_hIpHlpApiDll = Curl_load_library(TEXT("iphlpapi.dll"));
97*6236dae4SAndroid Build Coastguard Worker   if(s_hIpHlpApiDll) {
98*6236dae4SAndroid Build Coastguard Worker     /* Get the address of the if_nametoindex function */
99*6236dae4SAndroid Build Coastguard Worker     IF_NAMETOINDEX_FN pIfNameToIndex =
100*6236dae4SAndroid Build Coastguard Worker       CURLX_FUNCTION_CAST(IF_NAMETOINDEX_FN,
101*6236dae4SAndroid Build Coastguard Worker                           (GetProcAddress(s_hIpHlpApiDll, "if_nametoindex")));
102*6236dae4SAndroid Build Coastguard Worker 
103*6236dae4SAndroid Build Coastguard Worker     if(pIfNameToIndex)
104*6236dae4SAndroid Build Coastguard Worker       Curl_if_nametoindex = pIfNameToIndex;
105*6236dae4SAndroid Build Coastguard Worker   }
106*6236dae4SAndroid Build Coastguard Worker 
107*6236dae4SAndroid Build Coastguard Worker   /* curlx_verify_windows_version must be called during init at least once
108*6236dae4SAndroid Build Coastguard Worker      because it has its own initialization routine. */
109*6236dae4SAndroid Build Coastguard Worker   if(curlx_verify_windows_version(6, 0, 0, PLATFORM_WINNT,
110*6236dae4SAndroid Build Coastguard Worker                                   VERSION_GREATER_THAN_EQUAL)) {
111*6236dae4SAndroid Build Coastguard Worker     Curl_isVistaOrGreater = TRUE;
112*6236dae4SAndroid Build Coastguard Worker   }
113*6236dae4SAndroid Build Coastguard Worker   else
114*6236dae4SAndroid Build Coastguard Worker     Curl_isVistaOrGreater = FALSE;
115*6236dae4SAndroid Build Coastguard Worker 
116*6236dae4SAndroid Build Coastguard Worker   QueryPerformanceFrequency(&Curl_freq);
117*6236dae4SAndroid Build Coastguard Worker   return CURLE_OK;
118*6236dae4SAndroid Build Coastguard Worker }
119*6236dae4SAndroid Build Coastguard Worker 
120*6236dae4SAndroid Build Coastguard Worker /* Curl_win32_cleanup() is the opposite of Curl_win32_init() */
Curl_win32_cleanup(long init_flags)121*6236dae4SAndroid Build Coastguard Worker void Curl_win32_cleanup(long init_flags)
122*6236dae4SAndroid Build Coastguard Worker {
123*6236dae4SAndroid Build Coastguard Worker   if(s_hIpHlpApiDll) {
124*6236dae4SAndroid Build Coastguard Worker     FreeLibrary(s_hIpHlpApiDll);
125*6236dae4SAndroid Build Coastguard Worker     s_hIpHlpApiDll = NULL;
126*6236dae4SAndroid Build Coastguard Worker     Curl_if_nametoindex = NULL;
127*6236dae4SAndroid Build Coastguard Worker   }
128*6236dae4SAndroid Build Coastguard Worker 
129*6236dae4SAndroid Build Coastguard Worker #ifdef USE_WINDOWS_SSPI
130*6236dae4SAndroid Build Coastguard Worker   Curl_sspi_global_cleanup();
131*6236dae4SAndroid Build Coastguard Worker #endif
132*6236dae4SAndroid Build Coastguard Worker 
133*6236dae4SAndroid Build Coastguard Worker   if(init_flags & CURL_GLOBAL_WIN32) {
134*6236dae4SAndroid Build Coastguard Worker #ifdef USE_WINSOCK
135*6236dae4SAndroid Build Coastguard Worker     WSACleanup();
136*6236dae4SAndroid Build Coastguard Worker #endif
137*6236dae4SAndroid Build Coastguard Worker   }
138*6236dae4SAndroid Build Coastguard Worker }
139*6236dae4SAndroid Build Coastguard Worker 
140*6236dae4SAndroid Build Coastguard Worker #if !defined(LOAD_WITH_ALTERED_SEARCH_PATH)
141*6236dae4SAndroid Build Coastguard Worker #define LOAD_WITH_ALTERED_SEARCH_PATH  0x00000008
142*6236dae4SAndroid Build Coastguard Worker #endif
143*6236dae4SAndroid Build Coastguard Worker 
144*6236dae4SAndroid Build Coastguard Worker #if !defined(LOAD_LIBRARY_SEARCH_SYSTEM32)
145*6236dae4SAndroid Build Coastguard Worker #define LOAD_LIBRARY_SEARCH_SYSTEM32   0x00000800
146*6236dae4SAndroid Build Coastguard Worker #endif
147*6236dae4SAndroid Build Coastguard Worker 
148*6236dae4SAndroid Build Coastguard Worker /* We use our own typedef here since some headers might lack these */
149*6236dae4SAndroid Build Coastguard Worker typedef HMODULE (APIENTRY *LOADLIBRARYEX_FN)(LPCTSTR, HANDLE, DWORD);
150*6236dae4SAndroid Build Coastguard Worker 
151*6236dae4SAndroid Build Coastguard Worker /* See function definitions in winbase.h */
152*6236dae4SAndroid Build Coastguard Worker #ifdef UNICODE
153*6236dae4SAndroid Build Coastguard Worker #  ifdef _WIN32_WCE
154*6236dae4SAndroid Build Coastguard Worker #    define LOADLIBARYEX  L"LoadLibraryExW"
155*6236dae4SAndroid Build Coastguard Worker #  else
156*6236dae4SAndroid Build Coastguard Worker #    define LOADLIBARYEX  "LoadLibraryExW"
157*6236dae4SAndroid Build Coastguard Worker #  endif
158*6236dae4SAndroid Build Coastguard Worker #else
159*6236dae4SAndroid Build Coastguard Worker #  define LOADLIBARYEX    "LoadLibraryExA"
160*6236dae4SAndroid Build Coastguard Worker #endif
161*6236dae4SAndroid Build Coastguard Worker 
162*6236dae4SAndroid Build Coastguard Worker /*
163*6236dae4SAndroid Build Coastguard Worker  * Curl_load_library()
164*6236dae4SAndroid Build Coastguard Worker  *
165*6236dae4SAndroid Build Coastguard Worker  * This is used to dynamically load DLLs using the most secure method available
166*6236dae4SAndroid Build Coastguard Worker  * for the version of Windows that we are running on.
167*6236dae4SAndroid Build Coastguard Worker  *
168*6236dae4SAndroid Build Coastguard Worker  * Parameters:
169*6236dae4SAndroid Build Coastguard Worker  *
170*6236dae4SAndroid Build Coastguard Worker  * filename  [in] - The filename or full path of the DLL to load. If only the
171*6236dae4SAndroid Build Coastguard Worker  *                  filename is passed then the DLL will be loaded from the
172*6236dae4SAndroid Build Coastguard Worker  *                  Windows system directory.
173*6236dae4SAndroid Build Coastguard Worker  *
174*6236dae4SAndroid Build Coastguard Worker  * Returns the handle of the module on success; otherwise NULL.
175*6236dae4SAndroid Build Coastguard Worker  */
Curl_load_library(LPCTSTR filename)176*6236dae4SAndroid Build Coastguard Worker HMODULE Curl_load_library(LPCTSTR filename)
177*6236dae4SAndroid Build Coastguard Worker {
178*6236dae4SAndroid Build Coastguard Worker #ifndef CURL_WINDOWS_UWP
179*6236dae4SAndroid Build Coastguard Worker   HMODULE hModule = NULL;
180*6236dae4SAndroid Build Coastguard Worker   LOADLIBRARYEX_FN pLoadLibraryEx = NULL;
181*6236dae4SAndroid Build Coastguard Worker 
182*6236dae4SAndroid Build Coastguard Worker   /* Get a handle to kernel32 so we can access its functions at runtime */
183*6236dae4SAndroid Build Coastguard Worker   HMODULE hKernel32 = GetModuleHandle(TEXT("kernel32"));
184*6236dae4SAndroid Build Coastguard Worker   if(!hKernel32)
185*6236dae4SAndroid Build Coastguard Worker     return NULL;
186*6236dae4SAndroid Build Coastguard Worker 
187*6236dae4SAndroid Build Coastguard Worker   /* Attempt to find LoadLibraryEx() which is only available on Windows 2000
188*6236dae4SAndroid Build Coastguard Worker      and above */
189*6236dae4SAndroid Build Coastguard Worker   pLoadLibraryEx =
190*6236dae4SAndroid Build Coastguard Worker     CURLX_FUNCTION_CAST(LOADLIBRARYEX_FN,
191*6236dae4SAndroid Build Coastguard Worker                         (GetProcAddress(hKernel32, LOADLIBARYEX)));
192*6236dae4SAndroid Build Coastguard Worker 
193*6236dae4SAndroid Build Coastguard Worker   /* Detect if there is already a path in the filename and load the library if
194*6236dae4SAndroid Build Coastguard Worker      there is. Note: Both back slashes and forward slashes have been supported
195*6236dae4SAndroid Build Coastguard Worker      since the earlier days of DOS at an API level although they are not
196*6236dae4SAndroid Build Coastguard Worker      supported by command prompt */
197*6236dae4SAndroid Build Coastguard Worker   if(_tcspbrk(filename, TEXT("\\/"))) {
198*6236dae4SAndroid Build Coastguard Worker     /** !checksrc! disable BANNEDFUNC 1 **/
199*6236dae4SAndroid Build Coastguard Worker     hModule = pLoadLibraryEx ?
200*6236dae4SAndroid Build Coastguard Worker       pLoadLibraryEx(filename, NULL, LOAD_WITH_ALTERED_SEARCH_PATH) :
201*6236dae4SAndroid Build Coastguard Worker       LoadLibrary(filename);
202*6236dae4SAndroid Build Coastguard Worker   }
203*6236dae4SAndroid Build Coastguard Worker   /* Detect if KB2533623 is installed, as LOAD_LIBRARY_SEARCH_SYSTEM32 is only
204*6236dae4SAndroid Build Coastguard Worker      supported on Windows Vista, Windows Server 2008, Windows 7 and Windows
205*6236dae4SAndroid Build Coastguard Worker      Server 2008 R2 with this patch or natively on Windows 8 and above */
206*6236dae4SAndroid Build Coastguard Worker   else if(pLoadLibraryEx && GetProcAddress(hKernel32, "AddDllDirectory")) {
207*6236dae4SAndroid Build Coastguard Worker     /* Load the DLL from the Windows system directory */
208*6236dae4SAndroid Build Coastguard Worker     hModule = pLoadLibraryEx(filename, NULL, LOAD_LIBRARY_SEARCH_SYSTEM32);
209*6236dae4SAndroid Build Coastguard Worker   }
210*6236dae4SAndroid Build Coastguard Worker   else {
211*6236dae4SAndroid Build Coastguard Worker     /* Attempt to get the Windows system path */
212*6236dae4SAndroid Build Coastguard Worker     UINT systemdirlen = GetSystemDirectory(NULL, 0);
213*6236dae4SAndroid Build Coastguard Worker     if(systemdirlen) {
214*6236dae4SAndroid Build Coastguard Worker       /* Allocate space for the full DLL path (Room for the null terminator
215*6236dae4SAndroid Build Coastguard Worker          is included in systemdirlen) */
216*6236dae4SAndroid Build Coastguard Worker       size_t filenamelen = _tcslen(filename);
217*6236dae4SAndroid Build Coastguard Worker       TCHAR *path = malloc(sizeof(TCHAR) * (systemdirlen + 1 + filenamelen));
218*6236dae4SAndroid Build Coastguard Worker       if(path && GetSystemDirectory(path, systemdirlen)) {
219*6236dae4SAndroid Build Coastguard Worker         /* Calculate the full DLL path */
220*6236dae4SAndroid Build Coastguard Worker         _tcscpy(path + _tcslen(path), TEXT("\\"));
221*6236dae4SAndroid Build Coastguard Worker         _tcscpy(path + _tcslen(path), filename);
222*6236dae4SAndroid Build Coastguard Worker 
223*6236dae4SAndroid Build Coastguard Worker         /* Load the DLL from the Windows system directory */
224*6236dae4SAndroid Build Coastguard Worker         /** !checksrc! disable BANNEDFUNC 1 **/
225*6236dae4SAndroid Build Coastguard Worker         hModule = pLoadLibraryEx ?
226*6236dae4SAndroid Build Coastguard Worker           pLoadLibraryEx(path, NULL, LOAD_WITH_ALTERED_SEARCH_PATH) :
227*6236dae4SAndroid Build Coastguard Worker           LoadLibrary(path);
228*6236dae4SAndroid Build Coastguard Worker 
229*6236dae4SAndroid Build Coastguard Worker       }
230*6236dae4SAndroid Build Coastguard Worker       free(path);
231*6236dae4SAndroid Build Coastguard Worker     }
232*6236dae4SAndroid Build Coastguard Worker   }
233*6236dae4SAndroid Build Coastguard Worker   return hModule;
234*6236dae4SAndroid Build Coastguard Worker #else
235*6236dae4SAndroid Build Coastguard Worker   /* the Universal Windows Platform (UWP) cannot do this */
236*6236dae4SAndroid Build Coastguard Worker   (void)filename;
237*6236dae4SAndroid Build Coastguard Worker   return NULL;
238*6236dae4SAndroid Build Coastguard Worker #endif
239*6236dae4SAndroid Build Coastguard Worker }
240*6236dae4SAndroid Build Coastguard Worker 
241*6236dae4SAndroid Build Coastguard Worker #endif /* _WIN32 */
242