VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/init-win.cpp@ 63346

Last change on this file since 63346 was 62592, checked in by vboxsync, 8 years ago

IPRT: More unused parameters and undefined preprocessor macor warning (C4668) fixes/workarounds. The latter triggers in stdint.h from the compiler and in windows SDK/DDK headers.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.9 KB
Line 
1/* $Id: init-win.cpp 62592 2016-07-27 13:24:48Z vboxsync $ */
2/** @file
3 * IPRT - Init Ring-3, Windows Specific Code.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_DEFAULT
32#include <iprt/win/windows.h>
33#ifndef LOAD_LIBRARY_SEARCH_APPLICATION_DIR
34# define LOAD_LIBRARY_SEARCH_APPLICATION_DIR 0x200
35# define LOAD_LIBRARY_SEARCH_SYSTEM32 0x800
36#endif
37
38#include "internal-r3-win.h"
39#include <iprt/initterm.h>
40#include <iprt/assert.h>
41#include <iprt/err.h>
42#include <iprt/string.h>
43#include "../init.h"
44
45
46/*********************************************************************************************************************************
47* Global Variables *
48*********************************************************************************************************************************/
49/** Windows DLL loader protection level. */
50DECLHIDDEN(RTR3WINLDRPROT) g_enmWinLdrProt = RTR3WINLDRPROT_NONE;
51/** Our simplified windows version. */
52DECLHIDDEN(RTWINOSTYPE) g_enmWinVer = kRTWinOSType_UNKNOWN;
53/** Extended windows version information. */
54DECLHIDDEN(OSVERSIONINFOEXW) g_WinOsInfoEx;
55/** The native kernel32.dll handle. */
56DECLHIDDEN(HMODULE) g_hModKernel32 = NULL;
57/** The native ntdll.dll handle. */
58DECLHIDDEN(HMODULE) g_hModNtDll = NULL;
59/** GetSystemWindowsDirectoryW or GetWindowsDirectoryW (NT4). */
60DECLHIDDEN(PFNGETWINSYSDIR) g_pfnGetSystemWindowsDirectoryW = NULL;
61
62
63
64/**
65 * Translates OSVERSIONINOFEX into a Windows OS type.
66 *
67 * @returns The Windows OS type.
68 * @param pOSInfoEx The OS info returned by Windows.
69 *
70 * @remarks This table has been assembled from Usenet postings, personal
71 * observations, and reading other people's code. Please feel
72 * free to add to it or correct it.
73 * <pre>
74 dwPlatFormID dwMajorVersion dwMinorVersion dwBuildNumber
7595 1 4 0 950
7695 SP1 1 4 0 >950 && <=1080
7795 OSR2 1 4 <10 >1080
7898 1 4 10 1998
7998 SP1 1 4 10 >1998 && <2183
8098 SE 1 4 10 >=2183
81ME 1 4 90 3000
82
83NT 3.51 2 3 51 1057
84NT 4 2 4 0 1381
852000 2 5 0 2195
86XP 2 5 1 2600
872003 2 5 2 3790
88Vista 2 6 0
89
90CE 1.0 3 1 0
91CE 2.0 3 2 0
92CE 2.1 3 2 1
93CE 3.0 3 3 0
94</pre>
95 */
96static RTWINOSTYPE rtR3InitWinSimplifiedVersion(OSVERSIONINFOEXW const *pOSInfoEx)
97{
98 RTWINOSTYPE enmVer = kRTWinOSType_UNKNOWN;
99 BYTE const bProductType = pOSInfoEx->wProductType;
100 DWORD const dwPlatformId = pOSInfoEx->dwPlatformId;
101 DWORD const dwMinorVersion = pOSInfoEx->dwMinorVersion;
102 DWORD const dwMajorVersion = pOSInfoEx->dwMajorVersion;
103 DWORD const dwBuildNumber = pOSInfoEx->dwBuildNumber & 0xFFFF; /* Win 9x needs this. */
104
105 if ( dwPlatformId == VER_PLATFORM_WIN32_WINDOWS
106 && dwMajorVersion == 4)
107 {
108 if ( dwMinorVersion < 10
109 && dwBuildNumber == 950)
110 enmVer = kRTWinOSType_95;
111 else if ( dwMinorVersion < 10
112 && dwBuildNumber > 950
113 && dwBuildNumber <= 1080)
114 enmVer = kRTWinOSType_95SP1;
115 else if ( dwMinorVersion < 10
116 && dwBuildNumber > 1080)
117 enmVer = kRTWinOSType_95OSR2;
118 else if ( dwMinorVersion == 10
119 && dwBuildNumber == 1998)
120 enmVer = kRTWinOSType_98;
121 else if ( dwMinorVersion == 10
122 && dwBuildNumber > 1998
123 && dwBuildNumber < 2183)
124 enmVer = kRTWinOSType_98SP1;
125 else if ( dwMinorVersion == 10
126 && dwBuildNumber >= 2183)
127 enmVer = kRTWinOSType_98SE;
128 else if (dwMinorVersion == 90)
129 enmVer = kRTWinOSType_ME;
130 }
131 else if (dwPlatformId == VER_PLATFORM_WIN32_NT)
132 {
133 if ( dwMajorVersion == 3
134 && dwMinorVersion == 51)
135 enmVer = kRTWinOSType_NT351;
136 else if ( dwMajorVersion == 4
137 && dwMinorVersion == 0)
138 enmVer = kRTWinOSType_NT4;
139 else if ( dwMajorVersion == 5
140 && dwMinorVersion == 0)
141 enmVer = kRTWinOSType_2K;
142 else if ( dwMajorVersion == 5
143 && dwMinorVersion == 1)
144 enmVer = kRTWinOSType_XP;
145 else if ( dwMajorVersion == 5
146 && dwMinorVersion == 2)
147 enmVer = kRTWinOSType_2003;
148 else if ( dwMajorVersion == 6
149 && dwMinorVersion == 0)
150 {
151 if (bProductType != VER_NT_WORKSTATION)
152 enmVer = kRTWinOSType_2008;
153 else
154 enmVer = kRTWinOSType_VISTA;
155 }
156 else if ( dwMajorVersion == 6
157 && dwMinorVersion == 1)
158 {
159 if (bProductType != VER_NT_WORKSTATION)
160 enmVer = kRTWinOSType_2008R2;
161 else
162 enmVer = kRTWinOSType_7;
163 }
164 else if ( dwMajorVersion == 6
165 && dwMinorVersion == 2)
166 {
167 if (bProductType != VER_NT_WORKSTATION)
168 enmVer = kRTWinOSType_2012;
169 else
170 enmVer = kRTWinOSType_8;
171 }
172 else if ( dwMajorVersion == 6
173 && dwMinorVersion == 3)
174 {
175 if (bProductType != VER_NT_WORKSTATION)
176 enmVer = kRTWinOSType_2012R2;
177 else
178 enmVer = kRTWinOSType_81;
179 }
180 else if ( ( dwMajorVersion == 6
181 && dwMinorVersion == 4)
182 || ( dwMajorVersion == 10
183 && dwMinorVersion == 0))
184 {
185 if (bProductType != VER_NT_WORKSTATION)
186 enmVer = kRTWinOSType_2016;
187 else
188 enmVer = kRTWinOSType_10;
189 }
190 else
191 enmVer = kRTWinOSType_NT_UNKNOWN;
192 }
193
194 return enmVer;
195}
196
197
198/**
199 * Initializes the global variables related to windows version.
200 */
201static void rtR3InitWindowsVersion(void)
202{
203 Assert(g_hModNtDll != NULL);
204
205 /*
206 * ASSUMES OSVERSIONINFOEX starts with the exact same layout as OSVERSIONINFO (safe).
207 */
208 AssertCompileMembersSameSizeAndOffset(OSVERSIONINFOEX, szCSDVersion, OSVERSIONINFO, szCSDVersion);
209 AssertCompileMemberOffset(OSVERSIONINFOEX, wServicePackMajor, sizeof(OSVERSIONINFO));
210
211 /*
212 * Use the NT version of GetVersionExW so we don't get fooled by
213 * compatability shims.
214 */
215 RT_ZERO(g_WinOsInfoEx);
216 g_WinOsInfoEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEXW);
217
218 LONG (__stdcall *pfnRtlGetVersion)(OSVERSIONINFOEXW *);
219 *(FARPROC *)&pfnRtlGetVersion = GetProcAddress(g_hModNtDll, "RtlGetVersion");
220 LONG rcNt = -1;
221 if (pfnRtlGetVersion)
222 rcNt = pfnRtlGetVersion(&g_WinOsInfoEx);
223 if (rcNt != 0)
224 {
225 /*
226 * Couldn't find it or it failed, try the windows version of the API.
227 */
228 RT_ZERO(g_WinOsInfoEx);
229 g_WinOsInfoEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEXW);
230 if (!GetVersionExW((POSVERSIONINFOW)&g_WinOsInfoEx))
231 {
232 /*
233 * If that didn't work either, just get the basic version bits.
234 */
235 RT_ZERO(g_WinOsInfoEx);
236 g_WinOsInfoEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOW);
237 if (GetVersionExW((POSVERSIONINFOW)&g_WinOsInfoEx))
238 Assert(g_WinOsInfoEx.dwPlatformId != VER_PLATFORM_WIN32_NT || g_WinOsInfoEx.dwMajorVersion < 5);
239 else
240 {
241 AssertBreakpoint();
242 RT_ZERO(g_WinOsInfoEx);
243 }
244 }
245 }
246
247 if (g_WinOsInfoEx.dwOSVersionInfoSize)
248 g_enmWinVer = rtR3InitWinSimplifiedVersion(&g_WinOsInfoEx);
249}
250
251
252static int rtR3InitNativeObtrusiveWorker(uint32_t fFlags)
253{
254 /*
255 * Disable error popups.
256 */
257 UINT fOldErrMode = SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOOPENFILEERRORBOX);
258 SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOOPENFILEERRORBOX | fOldErrMode);
259
260 /*
261 * Restrict DLL searching for the process on windows versions which allow
262 * us to do so.
263 * - The first trick works on XP SP1+ and disables the searching of the
264 * current directory.
265 * - The second trick is W7 w/ KB2533623 and W8+, it restrict the DLL
266 * searching to the application directory (except when
267 * RTR3INIT_FLAGS_STANDALONE_APP is given) and the System32 directory.
268 */
269 int rc = VINF_SUCCESS;
270
271 typedef BOOL (WINAPI *PFNSETDLLDIRECTORY)(LPCWSTR);
272 PFNSETDLLDIRECTORY pfnSetDllDir = (PFNSETDLLDIRECTORY)GetProcAddress(g_hModKernel32, "SetDllDirectoryW");
273 if (pfnSetDllDir)
274 {
275 if (pfnSetDllDir(L""))
276 g_enmWinLdrProt = RTR3WINLDRPROT_NO_CWD;
277 else
278 rc = VERR_INTERNAL_ERROR_3;
279 }
280
281 /** @bugref{6861} Observed GUI issues on Vista (32-bit and 64-bit) when using
282 * SetDefaultDllDirectories.
283 * @bugref{8194} Try use SetDefaultDllDirectories on Vista for standalone apps
284 * despite potential GUI issues. */
285 if ( g_enmWinVer > kRTWinOSType_VISTA
286 || (fFlags & RTR3INIT_FLAGS_STANDALONE_APP))
287 {
288 typedef BOOL(WINAPI *PFNSETDEFAULTDLLDIRECTORIES)(DWORD);
289 PFNSETDEFAULTDLLDIRECTORIES pfnSetDefDllDirs;
290 pfnSetDefDllDirs = (PFNSETDEFAULTDLLDIRECTORIES)GetProcAddress(g_hModKernel32, "SetDefaultDllDirectories");
291 if (pfnSetDefDllDirs)
292 {
293 DWORD fDllDirs = LOAD_LIBRARY_SEARCH_SYSTEM32;
294 if (!(fFlags & RTR3INIT_FLAGS_STANDALONE_APP))
295 fDllDirs |= LOAD_LIBRARY_SEARCH_APPLICATION_DIR;
296 if (pfnSetDefDllDirs(fDllDirs))
297 g_enmWinLdrProt = fDllDirs & LOAD_LIBRARY_SEARCH_APPLICATION_DIR ? RTR3WINLDRPROT_SAFE : RTR3WINLDRPROT_SAFER;
298 else if (RT_SUCCESS(rc))
299 rc = VERR_INTERNAL_ERROR_4;
300 }
301 }
302
303 return rc;
304}
305
306
307DECLHIDDEN(int) rtR3InitNativeFirst(uint32_t fFlags)
308{
309 /*
310 * Make sure we've got the handles of the two main Windows NT dlls.
311 */
312 g_hModKernel32 = GetModuleHandleW(L"kernel32.dll");
313 if (g_hModKernel32 == NULL)
314 return VERR_INTERNAL_ERROR_2;
315 g_hModNtDll = GetModuleHandleW(L"ntdll.dll");
316 if (g_hModNtDll == NULL)
317 return VERR_INTERNAL_ERROR_2;
318
319 rtR3InitWindowsVersion();
320
321 int rc = VINF_SUCCESS;
322 if (!(fFlags & RTR3INIT_FLAGS_UNOBTRUSIVE))
323 rc = rtR3InitNativeObtrusiveWorker(fFlags);
324
325 /*
326 * Resolve some kernel32.dll APIs we may need but aren't necessarily
327 * present in older windows versions.
328 */
329 g_pfnGetSystemWindowsDirectoryW = (PFNGETWINSYSDIR)GetProcAddress(g_hModKernel32, "GetSystemWindowsDirectoryW");
330 if (g_pfnGetSystemWindowsDirectoryW)
331 g_pfnGetSystemWindowsDirectoryW = (PFNGETWINSYSDIR)GetProcAddress(g_hModKernel32, "GetWindowsDirectoryW");
332
333 return rc;
334}
335
336
337DECLHIDDEN(void) rtR3InitNativeObtrusive(uint32_t fFlags)
338{
339 rtR3InitNativeObtrusiveWorker(fFlags);
340}
341
342
343DECLHIDDEN(int) rtR3InitNativeFinal(uint32_t fFlags)
344{
345 /* Nothing to do here. */
346 RT_NOREF_PV(fFlags);
347 return VINF_SUCCESS;
348}
349
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette