VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/Installer/InstallHelper/VBoxGuestInstallHelper.cpp@ 71872

Last change on this file since 71872 was 68743, checked in by vboxsync, 7 years ago

WINNT/InstallHelper: Resolved a todo; rewritten FileGetArchitecture() to make use of IPRT's RTLdr API to retrieve the architecture and validate the PE signature.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.6 KB
Line 
1/* $Id: VBoxGuestInstallHelper.cpp 68743 2017-09-13 13:14:39Z vboxsync $ */
2/** @file
3 * VBoxGuestInstallHelper - Various helper routines for Windows guest installer.
4 */
5
6/*
7 * Copyright (C) 2011-2017 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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <iprt/win/windows.h>
23#include <stdlib.h>
24#include <tchar.h>
25#include <strsafe.h>
26#pragma warning(push)
27#pragma warning(disable: 4995) /* warning C4995: 'lstrcpyA': name was marked as #pragma deprecated */
28#include "exdll.h"
29#pragma warning(pop)
30
31#include <iprt/err.h>
32#include <iprt/initterm.h>
33#include <iprt/ldr.h>
34#include <iprt/localipc.h>
35#include <iprt/mem.h>
36#include <iprt/process.h>
37#include <iprt/string.h>
38
39/* Required structures/defines of VBoxTray. */
40#include "../../VBoxTray/VBoxTrayMsg.h"
41
42
43/*********************************************************************************************************************************
44* Defined Constants And Macros *
45*********************************************************************************************************************************/
46#define VBOXINSTALLHELPER_EXPORT extern "C" void __declspec(dllexport)
47
48
49/*********************************************************************************************************************************
50* Structures and Typedefs *
51*********************************************************************************************************************************/
52typedef DWORD (WINAPI *PFNSFCFILEEXCEPTION)(DWORD param1, PWCHAR param2, DWORD param3);
53
54
55/*********************************************************************************************************************************
56* Global Variables *
57*********************************************************************************************************************************/
58HINSTANCE g_hInstance;
59HWND g_hwndParent;
60PFNSFCFILEEXCEPTION g_pfnSfcFileException = NULL;
61
62/**
63 * @todo Clean up this DLL, use more IPRT in here!
64 */
65
66/**
67 * Pops (gets) a value from the internal NSIS stack.
68 * Since the supplied popstring() method easily can cause buffer
69 * overflows, use vboxPopString() instead!
70 *
71 * @return HRESULT
72 * @param pszDest Pointer to pre-allocated string to store result.
73 * @param cchDest Size (in characters) of pre-allocated string.
74 */
75static HRESULT vboxPopString(TCHAR *pszDest, size_t cchDest)
76{
77 HRESULT hr = S_OK;
78 if (!g_stacktop || !*g_stacktop)
79 hr = __HRESULT_FROM_WIN32(ERROR_INVALID_HANDLE);
80 else
81 {
82 stack_t *pStack = (*g_stacktop);
83 if (pStack)
84 {
85 hr = StringCchCopy(pszDest, cchDest, pStack->text);
86 if (SUCCEEDED(hr))
87 {
88 *g_stacktop = pStack->next;
89 GlobalFree((HGLOBAL)pStack);
90 }
91 }
92 else
93 hr = __HRESULT_FROM_WIN32(ERROR_INVALID_HANDLE);
94 }
95 return hr;
96}
97
98static HRESULT vboxPopULong(PULONG pulValue)
99{
100 HRESULT hr = S_OK;
101 if (!g_stacktop || !*g_stacktop)
102 hr = __HRESULT_FROM_WIN32(ERROR_EMPTY);
103 else
104 {
105 stack_t *pStack = (*g_stacktop);
106 if (pStack)
107 {
108 *pulValue = strtoul(pStack->text, NULL, 10 /* Base */);
109
110 *g_stacktop = pStack->next;
111 GlobalFree((HGLOBAL)pStack);
112 }
113 }
114 return hr;
115}
116
117static void vboxPushResultAsString(HRESULT hr)
118{
119 TCHAR szErr[MAX_PATH + 1];
120 if (FAILED(hr))
121 {
122 if (FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, hr, 0, szErr, MAX_PATH, NULL))
123 szErr[MAX_PATH] = '\0';
124 else
125 StringCchPrintf(szErr, sizeof(szErr),
126 "FormatMessage failed! Error = %ld", GetLastError());
127 }
128 else
129 StringCchPrintf(szErr, sizeof(szErr), "0");
130 pushstring(szErr);
131}
132
133/**
134 * Connects to VBoxTray IPC under the behalf of the user running
135 * in the current thread context.
136 *
137 * @return IPRT status code.
138 * @param phSession Where to store the IPC session.
139 */
140static int vboxConnectToVBoxTray(RTLOCALIPCSESSION *phSession)
141{
142 char szPipeName[512 + sizeof(VBOXTRAY_IPC_PIPE_PREFIX)];
143 memcpy(szPipeName, VBOXTRAY_IPC_PIPE_PREFIX, sizeof(VBOXTRAY_IPC_PIPE_PREFIX));
144 int rc = RTProcQueryUsername(NIL_RTPROCESS,
145 &szPipeName[sizeof(VBOXTRAY_IPC_PIPE_PREFIX) - 1],
146 sizeof(szPipeName) - sizeof(VBOXTRAY_IPC_PIPE_PREFIX) + 1,
147 NULL /*pcbUser*/);
148 if (RT_SUCCESS(rc))
149 rc = RTLocalIpcSessionConnect(phSession, szPipeName, RTLOCALIPC_FLAGS_NATIVE_NAME);
150 return rc;
151}
152
153static void vboxChar2WCharFree(PWCHAR pwString)
154{
155 if (pwString)
156 HeapFree(GetProcessHeap(), 0, pwString);
157}
158
159static HRESULT vboxChar2WCharAlloc(const char *pszString, PWCHAR *ppwString)
160{
161 HRESULT hr;
162 int iLen = strlen(pszString) + 2;
163 WCHAR *pwString = (WCHAR*)HeapAlloc(GetProcessHeap(), 0, iLen * sizeof(WCHAR));
164 if (!pwString)
165 hr = __HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
166 else
167 {
168 if (MultiByteToWideChar(CP_ACP, 0, pszString, -1, pwString, iLen) == 0)
169 {
170 hr = HRESULT_FROM_WIN32(GetLastError());
171 HeapFree(GetProcessHeap(), 0, pwString);
172 }
173 else
174 {
175 hr = S_OK;
176 *ppwString = pwString;
177 }
178 }
179 return hr;
180}
181
182/**
183 * Loads a system DLL.
184 *
185 * @returns Module handle or NULL
186 * @param pszName The DLL name.
187 */
188static HMODULE loadSystemDll(const char *pszName)
189{
190 char szPath[MAX_PATH];
191 UINT cchPath = GetSystemDirectoryA(szPath, sizeof(szPath));
192 size_t cbName = strlen(pszName) + 1;
193 if (cchPath + 1 + cbName > sizeof(szPath))
194 return NULL;
195 szPath[cchPath] = '\\';
196 memcpy(&szPath[cchPath + 1], pszName, cbName);
197 return LoadLibraryA(szPath);
198}
199
200/**
201 * Disables the Windows File Protection for a specified file
202 * using an undocumented SFC API call. Don't try this at home!
203 *
204 * @param hwndParent Window handle of parent.
205 * @param string_size Size of variable string.
206 * @param variables The actual variable string.
207 * @param stacktop Pointer to a pointer to the current stack.
208 */
209VBOXINSTALLHELPER_EXPORT DisableWFP(HWND hwndParent, int string_size, TCHAR *variables, stack_t **stacktop)
210{
211 NOREF(hwndParent);
212 EXDLL_INIT();
213
214 TCHAR szFile[MAX_PATH + 1];
215 HRESULT hr = vboxPopString(szFile, sizeof(szFile) / sizeof(TCHAR));
216 if (SUCCEEDED(hr))
217 {
218 HMODULE hSFC = loadSystemDll("sfc_os.dll"); /** @todo Replace this by RTLdr APIs. */
219 if (NULL != hSFC)
220 {
221 g_pfnSfcFileException = (PFNSFCFILEEXCEPTION)GetProcAddress(hSFC, "SfcFileException");
222 if (g_pfnSfcFileException == NULL)
223 {
224 /* If we didn't get the proc address with the call above, try it harder with
225 * the (zero based) index of the function list. */
226 g_pfnSfcFileException = (PFNSFCFILEEXCEPTION)GetProcAddress(hSFC, (LPCSTR)5);
227 if (g_pfnSfcFileException == NULL)
228 hr = HRESULT_FROM_WIN32(ERROR_PROC_NOT_FOUND);
229 }
230 }
231 else
232 hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
233
234 if (SUCCEEDED(hr))
235 {
236 WCHAR *pwszFile;
237 hr = vboxChar2WCharAlloc(szFile, &pwszFile);
238 if (SUCCEEDED(hr))
239 {
240 if (g_pfnSfcFileException(0, pwszFile, UINT32_MAX) != 0)
241 hr = HRESULT_FROM_WIN32(GetLastError());
242 vboxChar2WCharFree(pwszFile);
243 }
244 }
245
246 if (hSFC)
247 FreeLibrary(hSFC);
248 }
249
250 vboxPushResultAsString(hr);
251}
252
253/**
254 * Retrieves a file's architecture (x86 or amd64).
255 * Outputs "x86", "amd64" or an error message (if not found/invalid) on stack.
256 *
257 * @param hwndParent Window handle of parent.
258 * @param string_size Size of variable string.
259 * @param variables The actual variable string.
260 * @param stacktop Pointer to a pointer to the current stack.
261 */
262VBOXINSTALLHELPER_EXPORT FileGetArchitecture(HWND hwndParent, int string_size, TCHAR *variables, stack_t **stacktop)
263{
264 NOREF(hwndParent);
265 EXDLL_INIT();
266
267 TCHAR szFile[MAX_PATH + 1];
268 HRESULT hr = vboxPopString(szFile, sizeof(szFile) / sizeof(TCHAR));
269 if (SUCCEEDED(hr))
270 {
271 RTLDRMOD hLdrMod;
272 int rc = RTLdrOpen(szFile, RTLDR_O_FOR_VALIDATION, RTLDRARCH_WHATEVER, &hLdrMod);
273 if (RT_SUCCESS(rc))
274 {
275 if (RTLdrGetFormat(hLdrMod) == RTLDRFMT_PE)
276 {
277 RTLDRARCH enmLdrArch = RTLdrGetArch(hLdrMod);
278 switch (enmLdrArch)
279 {
280 case RTLDRARCH_X86_32:
281 pushstring("x86");
282 break;
283
284 case RTLDRARCH_AMD64:
285 pushstring("amd64");
286 break;
287
288 default:
289 pushstring("Error: Unknown / invalid architecture");
290 break;
291 }
292 }
293 else
294 pushstring("Error: Unknown / invalid PE signature");
295
296 RTLdrClose(hLdrMod);
297 }
298 else
299 {
300 char szMsg[64];
301 RTStrPrintf(szMsg, sizeof(szMsg), "Error: Could not open file: %Rrc", rc);
302
303 pushstring(szMsg);
304 }
305 }
306 else
307 vboxPushResultAsString(hr);
308}
309
310/**
311 * Retrieves a file's vendor.
312 * Outputs the vendor's name or an error message (if not found/invalid) on stack.
313 *
314 * @param hwndParent Window handle of parent.
315 * @param string_size Size of variable string.
316 * @param variables The actual variable string.
317 * @param stacktop Pointer to a pointer to the current stack.
318 */
319VBOXINSTALLHELPER_EXPORT FileGetVendor(HWND hwndParent, int string_size, TCHAR *variables, stack_t **stacktop)
320{
321 NOREF(hwndParent);
322 EXDLL_INIT();
323
324 TCHAR szFile[MAX_PATH + 1];
325 HRESULT hr = vboxPopString(szFile, sizeof(szFile) / sizeof(TCHAR));
326 if (SUCCEEDED(hr))
327 {
328 DWORD dwInfoSize = GetFileVersionInfoSize(szFile, NULL /* lpdwHandle */);
329 if (dwInfoSize)
330 {
331 void *pFileInfo = GlobalAlloc(GMEM_FIXED, dwInfoSize);
332 if (pFileInfo)
333 {
334 if (GetFileVersionInfo(szFile, 0, dwInfoSize, pFileInfo))
335 {
336 LPVOID pvInfo;
337 UINT puInfoLen;
338 if (VerQueryValue(pFileInfo, _T("\\VarFileInfo\\Translation"),
339 &pvInfo, &puInfoLen))
340 {
341 WORD wCodePage = LOWORD(*(DWORD*)pvInfo);
342 WORD wLanguageID = HIWORD(*(DWORD*)pvInfo);
343
344 TCHAR szQuery[MAX_PATH];
345#pragma warning(suppress:4995) /* warning C4995: '_sntprintf': name was marked as #pragma deprecated */
346 _sntprintf(szQuery, sizeof(szQuery), _T("StringFileInfo\\%04X%04X\\CompanyName"),
347 wCodePage,wLanguageID);
348
349 LPCTSTR pcData;
350 if (VerQueryValue(pFileInfo, szQuery,(void**)&pcData, &puInfoLen))
351 {
352 pushstring(pcData);
353 }
354 else
355 hr = __HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
356 }
357 else
358 hr = __HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
359 }
360 GlobalFree(pFileInfo);
361 }
362 else
363 hr = __HRESULT_FROM_WIN32(ERROR_OUTOFMEMORY);
364 }
365 else
366 hr = __HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
367 }
368
369 if (FAILED(hr))
370 vboxPushResultAsString(hr);
371}
372
373/**
374 * Shows a balloon message using VBoxTray's notification area in the
375 * Windows task bar.
376 *
377 * @param hwndParent Window handle of parent.
378 * @param string_size Size of variable string.
379 * @param variables The actual variable string.
380 * @param stacktop Pointer to a pointer to the current stack.
381 */
382VBOXINSTALLHELPER_EXPORT VBoxTrayShowBallonMsg(HWND hwndParent, int string_size, TCHAR *variables, stack_t **stacktop)
383{
384 NOREF(hwndParent);
385 EXDLL_INIT();
386
387 char szMsg[256];
388 char szTitle[128];
389 HRESULT hr = vboxPopString(szMsg, sizeof(szMsg) / sizeof(char));
390 if (SUCCEEDED(hr))
391 hr = vboxPopString(szTitle, sizeof(szTitle) / sizeof(char));
392
393 /** @todo Do we need to restore the stack on failure? */
394
395 if (SUCCEEDED(hr))
396 {
397 RTR3InitDll(0);
398
399 uint32_t cbMsg = sizeof(VBOXTRAYIPCMSG_SHOWBALLOONMSG)
400 + strlen(szMsg) + 1 /* Include terminating zero */
401 + strlen(szTitle) + 1; /* Dito. */
402 Assert(cbMsg);
403 PVBOXTRAYIPCMSG_SHOWBALLOONMSG pIpcMsg =
404 (PVBOXTRAYIPCMSG_SHOWBALLOONMSG)RTMemAlloc(cbMsg);
405 if (pIpcMsg)
406 {
407 /* Stuff in the strings. */
408 memcpy(pIpcMsg->szMsgContent, szMsg, strlen(szMsg) + 1);
409 memcpy(pIpcMsg->szMsgTitle, szTitle, strlen(szTitle) + 1);
410
411 /* Pop off the values in reverse order from the stack. */
412 if (SUCCEEDED(hr))
413 hr = vboxPopULong((ULONG*)&pIpcMsg->uType);
414 if (SUCCEEDED(hr))
415 hr = vboxPopULong((ULONG*)&pIpcMsg->uShowMS);
416
417 if (SUCCEEDED(hr))
418 {
419 RTLOCALIPCSESSION hSession = 0;
420 int rc = vboxConnectToVBoxTray(&hSession);
421 if (RT_SUCCESS(rc))
422 {
423 VBOXTRAYIPCHEADER ipcHdr = { VBOXTRAY_IPC_HDR_MAGIC, 0 /* Header version */,
424 VBOXTRAYIPCMSGTYPE_SHOWBALLOONMSG, cbMsg };
425
426 rc = RTLocalIpcSessionWrite(hSession, &ipcHdr, sizeof(ipcHdr));
427 if (RT_SUCCESS(rc))
428 rc = RTLocalIpcSessionWrite(hSession, pIpcMsg, cbMsg);
429
430 int rc2 = RTLocalIpcSessionClose(hSession);
431 if (RT_SUCCESS(rc))
432 rc = rc2;
433 }
434
435 if (RT_FAILURE(rc))
436 hr = __HRESULT_FROM_WIN32(ERROR_BROKEN_PIPE);
437 }
438
439 RTMemFree(pIpcMsg);
440 }
441 else
442 hr = __HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
443 }
444
445 /* Push simple return value on stack. */
446 SUCCEEDED(hr) ? pushstring("0") : pushstring("1");
447}
448
449BOOL WINAPI DllMain(HANDLE hInst, ULONG uReason, LPVOID pReserved)
450{
451 RT_NOREF(uReason, pReserved);
452 g_hInstance = (HINSTANCE)hInst;
453 return TRUE;
454}
455
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