VirtualBox

source: vbox/trunk/src/VBox/Installer/win/InstallHelper/VBoxInstallHelper.cpp@ 66280

Last change on this file since 66280 was 63311, checked in by vboxsync, 8 years ago

Use shlobj.h wrapper.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 55.3 KB
Line 
1/* $Id: VBoxInstallHelper.cpp 63311 2016-08-11 00:02:38Z vboxsync $ */
2/** @file
3 * VBoxInstallHelper - Various helper routines for Windows host installer.
4 */
5
6/*
7 * Copyright (C) 2008-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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#ifdef VBOX_WITH_NETFLT
23# include "VBox/VBoxNetCfg-win.h"
24# include "VBox/VBoxDrvCfg-win.h"
25#endif /* VBOX_WITH_NETFLT */
26
27#include <VBox/version.h>
28
29#include <iprt/win/windows.h>
30#include <wchar.h>
31#include <stdio.h>
32
33#include <msi.h>
34#include <msiquery.h>
35
36#define _WIN32_DCOM
37#include <iprt/win/windows.h>
38#include <assert.h>
39#include <shellapi.h>
40#define INITGUID
41#include <guiddef.h>
42#include <devguid.h>
43#include <iprt/win/objbase.h>
44#include <iprt/win/setupapi.h>
45#include <iprt/win/shlobj.h>
46#include <cfgmgr32.h>
47
48#include "VBoxCommon.h"
49
50#ifndef VBOX_OSE
51# include "internal/VBoxSerial.h"
52#endif
53
54
55/*********************************************************************************************************************************
56* Header Files *
57*********************************************************************************************************************************/
58#ifdef DEBUG
59# define NonStandardAssert(_expr) assert(_expr)
60#else
61# define NonStandardAssert(_expr) do{ }while(0)
62#endif
63
64#define MY_WTEXT_HLP(a_str) L##a_str
65#define MY_WTEXT(a_str) MY_WTEXT_HLP(a_str)
66
67
68BOOL APIENTRY DllMain(HANDLE hModule, DWORD uReason, LPVOID lpReserved)
69{
70 RT_NOREF3(hModule, uReason, lpReserved);
71 return TRUE;
72}
73
74
75static void logString(MSIHANDLE hInstall, LPCSTR szString, ...)
76{
77 PMSIHANDLE newHandle = MsiCreateRecord(2);
78
79 char szBuffer[1024];
80 va_list va;
81 va_start(va, szString);
82 _vsnprintf(szBuffer, sizeof(szBuffer), szString, va);
83 va_end(va);
84
85 MsiRecordSetStringA(newHandle, 0, szBuffer);
86 MsiProcessMessage(hInstall, INSTALLMESSAGE(INSTALLMESSAGE_INFO), newHandle);
87 MsiCloseHandle(newHandle);
88
89#if 0/*def DEBUG - wrong? What does '%s' expect, wchar or char? */
90 _tprintf(_T("Debug: %s\n"), szBuffer);
91#endif
92}
93
94static void logStringW(MSIHANDLE hInstall, LPCWSTR pwszString, ...)
95{
96 PMSIHANDLE newHandle = MsiCreateRecord(2);
97
98 WCHAR szBuffer[1024];
99 va_list va;
100 va_start(va, pwszString);
101 _vsnwprintf(szBuffer, RT_ELEMENTS(szBuffer), pwszString, va);
102 va_end(va);
103
104 MsiRecordSetStringW(newHandle, 0, szBuffer);
105 MsiProcessMessage(hInstall, INSTALLMESSAGE(INSTALLMESSAGE_INFO), newHandle);
106 MsiCloseHandle(newHandle);
107}
108
109UINT __stdcall IsSerialCheckNeeded(MSIHANDLE hModule)
110{
111#ifndef VBOX_OSE
112 /*BOOL bRet =*/ serialCheckNeeded(hModule);
113#else
114 RT_NOREF1(hModule);
115#endif
116 return ERROR_SUCCESS;
117}
118
119UINT __stdcall CheckSerial(MSIHANDLE hModule)
120{
121#ifndef VBOX_OSE
122 /*BOOL bRet =*/ serialIsValid(hModule);
123#else
124 RT_NOREF1(hModule);
125#endif
126 return ERROR_SUCCESS;
127}
128
129DWORD Exec(MSIHANDLE hModule,
130 const WCHAR *pwszAppName, WCHAR *pwszCmdLine, const WCHAR *pwszWorkDir,
131 DWORD *pdwExitCode)
132{
133 STARTUPINFOW si;
134 PROCESS_INFORMATION pi;
135 DWORD rc = ERROR_SUCCESS;
136
137 ZeroMemory(&si, sizeof(si));
138 si.dwFlags = STARTF_USESHOWWINDOW;
139#ifdef UNICODE
140 si.dwFlags |= CREATE_UNICODE_ENVIRONMENT;
141#endif
142 si.wShowWindow = SW_HIDE; /* For debugging: SW_SHOW; */
143 si.cb = sizeof(si);
144 ZeroMemory(&pi, sizeof(pi));
145
146 logStringW(hModule, L"Executing command line: %s %s (Working Dir: %s)",
147 pwszAppName, pwszCmdLine, pwszWorkDir == NULL ? L"Current" : pwszWorkDir);
148
149 SetLastError(0);
150 if (!CreateProcessW(pwszAppName, /* Module name. */
151 pwszCmdLine, /* Command line. */
152 NULL, /* Process handle not inheritable. */
153 NULL, /* Thread handle not inheritable. */
154 FALSE, /* Set handle inheritance to FALSE .*/
155 0, /* No creation flags. */
156 NULL, /* Use parent's environment block. */
157 pwszWorkDir, /* Use parent's starting directory. */
158 &si, /* Pointer to STARTUPINFO structure. */
159 &pi)) /* Pointer to PROCESS_INFORMATION structure. */
160 {
161 rc = GetLastError();
162 logStringW(hModule, L"Executing command line: CreateProcess() failed! Error: %ld", rc);
163 return rc;
164 }
165
166 /* Wait until child process exits. */
167 if (WAIT_FAILED == WaitForSingleObject(pi.hProcess, 30 * 1000 /* Wait 30 secs max. */))
168 {
169 rc = GetLastError();
170 logStringW(hModule, L"Executing command line: WaitForSingleObject() failed! Error: %u", rc);
171 }
172 else
173 {
174 if (!GetExitCodeProcess(pi.hProcess, pdwExitCode))
175 {
176 rc = GetLastError();
177 logStringW(hModule, L"Executing command line: GetExitCodeProcess() failed! Error: %u", rc);
178 }
179 }
180
181 /* Close process and thread handles. */
182 CloseHandle(pi.hProcess);
183 CloseHandle(pi.hThread);
184
185 logStringW(hModule, L"Executing command returned: %u (exit code %u)", rc, *pdwExitCode);
186 return rc;
187}
188
189UINT __stdcall InstallPythonAPI(MSIHANDLE hModule)
190{
191 logStringW(hModule, L"InstallPythonAPI: Checking for installed Python environment ...");
192
193 HKEY hkPythonCore = NULL;
194 BOOL bFound = FALSE;
195 LONG rc = RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Python\\PythonCore", 0, KEY_READ, &hkPythonCore);
196 if (rc != ERROR_SUCCESS)
197 {
198 logStringW(hModule, L"InstallPythonAPI: Python seems not to be installed.");
199 return ERROR_SUCCESS;
200 }
201
202 WCHAR wszPath[MAX_PATH] = { 0 };
203 WCHAR wszVal[MAX_PATH] = { 0 };
204
205 for (int i = 0;; ++i)
206 {
207 WCHAR wszRoot[MAX_PATH] = { 0 };
208 DWORD dwLen = sizeof(wszPath);
209 DWORD dwKeyType = REG_SZ;
210
211 rc = RegEnumKeyExW(hkPythonCore, i, wszRoot, &dwLen, NULL, NULL, NULL, NULL);
212 if (rc != ERROR_SUCCESS || dwLen <= 0)
213 break;
214
215 swprintf_s(wszPath, RT_ELEMENTS(wszPath), L"%s\\InstallPath", wszRoot);
216 dwLen = sizeof(wszVal);
217
218 HKEY hkPythonInstPath = NULL;
219 rc = RegOpenKeyExW(hkPythonCore, wszPath, 0, KEY_READ, &hkPythonInstPath);
220 if (rc != ERROR_SUCCESS)
221 continue;
222
223 rc = RegQueryValueExW(hkPythonInstPath, L"", NULL, &dwKeyType, (LPBYTE)wszVal, &dwLen);
224 if (rc == ERROR_SUCCESS)
225 logStringW(hModule, L"InstallPythonAPI: Path \"%s\" detected.", wszVal);
226
227 RegCloseKey(hkPythonInstPath);
228 }
229 RegCloseKey(hkPythonCore);
230
231 /* Python path found? */
232 WCHAR wszExec[MAX_PATH] = { 0 };
233 WCHAR wszCmdLine[MAX_PATH] = { 0 };
234 DWORD dwExitCode = 0;
235 if (wcslen(wszVal) > 0)
236 {
237 /* Cool, check for installed Win32 extensions. */
238 logStringW(hModule, L"InstallPythonAPI: Python installed. Checking for Win32 extensions ...");
239 swprintf_s(wszExec, RT_ELEMENTS(wszExec), L"%s\\python.exe", wszVal);
240 swprintf_s(wszCmdLine, RT_ELEMENTS(wszCmdLine), L"%s\\python.exe -c \"import win32api\"", wszVal);
241
242 DWORD dwRetExec = Exec(hModule, wszExec, wszCmdLine, NULL, &dwExitCode);
243 if ( (ERROR_SUCCESS == dwRetExec)
244 && ( 0 == dwExitCode))
245 {
246 /* Did we get the correct error level (=0)? */
247 logStringW(hModule, L"InstallPythonAPI: Win32 extensions installed.");
248 bFound = TRUE;
249 }
250 else
251 logStringW(hModule, L"InstallPythonAPI: Win32 extensions not found.");
252 }
253
254 BOOL bInstalled = FALSE;
255 if (bFound) /* Is Python and all required stuff installed? */
256 {
257 /* Get the VBoxAPI setup string. */
258 WCHAR wszPathTargetDir[MAX_PATH] = {0};
259 VBoxGetProperty(hModule, L"CustomActionData", wszPathTargetDir, sizeof(wszPathTargetDir));
260 if (wcslen(wszPathTargetDir))
261 {
262 /* Set final path. */
263 swprintf_s(wszPath, RT_ELEMENTS(wszPath), L"%s", wszPathTargetDir);
264
265 /* Install our API module. */
266 swprintf_s(wszCmdLine, RT_ELEMENTS(wszCmdLine), L"%s\\python.exe vboxapisetup.py install", wszVal);
267
268 /* Set required environment variables. */
269 if (!SetEnvironmentVariableW(L"VBOX_INSTALL_PATH", wszPathTargetDir))
270 logStringW(hModule, L"InstallPythonAPI: Could set environment variable VBOX_INSTALL_PATH!");
271 else
272 {
273 DWORD dwRetExec = Exec(hModule, wszExec, wszCmdLine, wszPath, &dwExitCode);
274 if ( (ERROR_SUCCESS == dwRetExec)
275 && ( 0 == dwExitCode))
276 {
277 /* All done! */
278 logStringW(hModule, L"InstallPythonAPI: VBoxAPI for Python successfully installed.");
279 bInstalled = TRUE;
280 }
281 else
282 {
283 if (dwRetExec)
284 logStringW(hModule, L"InstallPythonAPI: Error while executing installation of VBox API: %ld", dwRetExec);
285 else
286 logStringW(hModule, L"InstallPythonAPI: Python reported an error while installing VBox API: %ld", dwExitCode);
287 }
288 }
289 }
290 else
291 logStringW(hModule, L"InstallPythonAPI: Unable to retrieve VBox installation path!");
292 }
293
294 VBoxSetProperty(hModule, L"VBOX_PYTHON_IS_INSTALLED", bInstalled ? L"1" : L"0");
295
296 if (!bInstalled)
297 logStringW(hModule, L"InstallPythonAPI: VBox API not installed.");
298 return ERROR_SUCCESS; /* Do not fail here. */
299}
300
301static LONG installBrandingValue(MSIHANDLE hModule,
302 const WCHAR *pwszFileName,
303 const WCHAR *pwszSection,
304 const WCHAR *pwszValue)
305{
306 LONG rc;
307 WCHAR wszValue[_MAX_PATH];
308 if (GetPrivateProfileStringW(pwszSection, pwszValue, NULL,
309 wszValue, sizeof(wszValue), pwszFileName) > 0)
310 {
311 HKEY hkBranding;
312 WCHAR wszKey[_MAX_PATH];
313
314 if (wcsicmp(L"General", pwszSection) != 0)
315 swprintf_s(wszKey, RT_ELEMENTS(wszKey), L"SOFTWARE\\%s\\VirtualBox\\Branding\\%s", VBOX_VENDOR_SHORT, pwszSection);
316 else
317 swprintf_s(wszKey, RT_ELEMENTS(wszKey), L"SOFTWARE\\%s\\VirtualBox\\Branding", VBOX_VENDOR_SHORT);
318
319 rc = RegOpenKeyExW(HKEY_LOCAL_MACHINE, wszKey, 0, KEY_WRITE, &hkBranding);
320 if (rc == ERROR_SUCCESS)
321 {
322 rc = RegSetValueExW(hkBranding,
323 pwszValue,
324 NULL,
325 REG_SZ,
326 (BYTE *)wszValue,
327 (DWORD)wcslen(wszValue));
328 if (rc != ERROR_SUCCESS)
329 logStringW(hModule, L"InstallBranding: Could not write value %s! Error %ld", pwszValue, rc);
330 RegCloseKey (hkBranding);
331 }
332 }
333 else
334 rc = ERROR_NOT_FOUND;
335 return rc;
336}
337
338UINT CopyDir(MSIHANDLE hModule, const WCHAR *pwszDestDir, const WCHAR *pwszSourceDir)
339{
340 UINT rc;
341 WCHAR wszDest[_MAX_PATH + 1];
342 WCHAR wszSource[_MAX_PATH + 1];
343
344 swprintf_s(wszDest, RT_ELEMENTS(wszDest), L"%s%c", pwszDestDir, L'\0');
345 swprintf_s(wszSource, RT_ELEMENTS(wszSource), L"%s%c", pwszSourceDir, L'\0');
346
347 SHFILEOPSTRUCTW s = {0};
348 s.hwnd = NULL;
349 s.wFunc = FO_COPY;
350 s.pTo = wszDest;
351 s.pFrom = wszSource;
352 s.fFlags = FOF_SILENT |
353 FOF_NOCONFIRMATION |
354 FOF_NOCONFIRMMKDIR |
355 FOF_NOERRORUI;
356
357 logStringW(hModule, L"CopyDir: DestDir=%s, SourceDir=%s", wszDest, wszSource);
358 int r = SHFileOperationW(&s);
359 if (r != 0)
360 {
361 logStringW(hModule, L"CopyDir: Copy operation returned status 0x%x", r);
362 rc = ERROR_GEN_FAILURE;
363 }
364 else
365 rc = ERROR_SUCCESS;
366 return rc;
367}
368
369UINT RemoveDir(MSIHANDLE hModule, const WCHAR *pwszDestDir)
370{
371 UINT rc;
372 WCHAR wszDest[_MAX_PATH + 1];
373
374 swprintf_s(wszDest, RT_ELEMENTS(wszDest), L"%s%c", pwszDestDir, L'\0');
375
376 SHFILEOPSTRUCTW s = {0};
377 s.hwnd = NULL;
378 s.wFunc = FO_DELETE;
379 s.pFrom = wszDest;
380 s.fFlags = FOF_SILENT
381 | FOF_NOCONFIRMATION
382 | FOF_NOCONFIRMMKDIR
383 | FOF_NOERRORUI;
384
385 logStringW(hModule, L"RemoveDir: DestDir=%s", wszDest);
386 int r = SHFileOperationW(&s);
387 if (r != 0)
388 {
389 logStringW(hModule, L"RemoveDir: Remove operation returned status 0x%x", r);
390 rc = ERROR_GEN_FAILURE;
391 }
392 else
393 rc = ERROR_SUCCESS;
394 return rc;
395}
396
397UINT RenameDir(MSIHANDLE hModule, const WCHAR *pwszDestDir, const WCHAR *pwszSourceDir)
398{
399 UINT rc;
400 WCHAR wszDest[_MAX_PATH + 1];
401 WCHAR wszSource[_MAX_PATH + 1];
402
403 swprintf_s(wszDest, RT_ELEMENTS(wszDest), L"%s%c", pwszDestDir, L'\0');
404 swprintf_s(wszSource, RT_ELEMENTS(wszSource), L"%s%c", pwszSourceDir, L'\0');
405
406 SHFILEOPSTRUCTW s = {0};
407 s.hwnd = NULL;
408 s.wFunc = FO_RENAME;
409 s.pTo = wszDest;
410 s.pFrom = wszSource;
411 s.fFlags = FOF_SILENT
412 | FOF_NOCONFIRMATION
413 | FOF_NOCONFIRMMKDIR
414 | FOF_NOERRORUI;
415
416 logStringW(hModule, L"RenameDir: DestDir=%s, SourceDir=%s", wszDest, wszSource);
417 int r = SHFileOperationW(&s);
418 if (r != 0)
419 {
420 logStringW(hModule, L"RenameDir: Rename operation returned status 0x%x", r);
421 rc = ERROR_GEN_FAILURE;
422 }
423 else
424 rc = ERROR_SUCCESS;
425 return rc;
426}
427
428UINT __stdcall UninstallBranding(MSIHANDLE hModule)
429{
430 UINT rc;
431 logStringW(hModule, L"UninstallBranding: Handling branding file ...");
432
433 WCHAR wszPathTargetDir[_MAX_PATH];
434 WCHAR wszPathDest[_MAX_PATH];
435
436 rc = VBoxGetProperty(hModule, L"CustomActionData", wszPathTargetDir, sizeof(wszPathTargetDir));
437 if (rc == ERROR_SUCCESS)
438 {
439 /** @todo Check trailing slash after %s. */
440/** @todo r=bird: using swprintf_s for formatting paths without checking for
441 * overflow not good. You're dodging the buffer overflow issue only to end up
442 * with incorrect behavior! (Truncated file/dir names)
443 *
444 * This applies almost to all swprintf_s calls in this file!!
445 */
446 swprintf_s(wszPathDest, RT_ELEMENTS(wszPathDest), L"%scustom", wszPathTargetDir);
447 rc = RemoveDir(hModule, wszPathDest);
448 if (rc != ERROR_SUCCESS)
449 {
450 /* Check for hidden .custom directory and remove it. */
451 swprintf_s(wszPathDest, RT_ELEMENTS(wszPathDest), L"%s.custom", wszPathTargetDir);
452 rc = RemoveDir(hModule, wszPathDest);
453 }
454 }
455
456 logStringW(hModule, L"UninstallBranding: Handling done. (rc=%u (ignored))", rc);
457 return ERROR_SUCCESS; /* Do not fail here. */
458}
459
460UINT __stdcall InstallBranding(MSIHANDLE hModule)
461{
462 UINT rc;
463 logStringW(hModule, L"InstallBranding: Handling branding file ...");
464
465 WCHAR wszPathMSI[_MAX_PATH];
466 rc = VBoxGetProperty(hModule, L"SOURCEDIR", wszPathMSI, sizeof(wszPathMSI));
467 if (rc == ERROR_SUCCESS)
468 {
469 WCHAR wszPathTargetDir[_MAX_PATH];
470 rc = VBoxGetProperty(hModule, L"CustomActionData", wszPathTargetDir, sizeof(wszPathTargetDir));
471 if (rc == ERROR_SUCCESS)
472 {
473 /** @todo Check for trailing slash after %s. */
474 WCHAR wszPathDest[_MAX_PATH];
475 swprintf_s(wszPathDest, RT_ELEMENTS(wszPathDest), L"%s", wszPathTargetDir);
476
477 WCHAR wszPathSource[_MAX_PATH];
478 swprintf_s(wszPathSource, RT_ELEMENTS(wszPathSource), L"%s.custom", wszPathMSI);
479 rc = CopyDir(hModule, wszPathDest, wszPathSource);
480 if (rc == ERROR_SUCCESS)
481 {
482 swprintf_s(wszPathDest, RT_ELEMENTS(wszPathDest), L"%scustom", wszPathTargetDir);
483 swprintf_s(wszPathSource, RT_ELEMENTS(wszPathSource), L"%s.custom", wszPathTargetDir);
484 rc = RenameDir(hModule, wszPathDest, wszPathSource);
485 }
486 }
487 }
488
489 logStringW(hModule, L"InstallBranding: Handling done. (rc=%u (ignored))", rc);
490 return ERROR_SUCCESS; /* Do not fail here. */
491}
492
493#ifdef VBOX_WITH_NETFLT
494
495/** @todo should use some real VBox app name */
496#define VBOX_NETCFG_APP_NAME L"VirtualBox Installer"
497#define VBOX_NETCFG_MAX_RETRIES 10
498#define NETFLT_PT_INF_REL_PATH L"VBoxNetFlt.inf"
499#define NETFLT_MP_INF_REL_PATH L"VBoxNetFltM.inf"
500#define NETFLT_ID L"sun_VBoxNetFlt" /** @todo Needs to be changed (?). */
501#define NETADP_ID L"sun_VBoxNetAdp" /** @todo Needs to be changed (?). */
502
503#define NETLWF_INF_NAME L"VBoxNetLwf.inf"
504
505static MSIHANDLE g_hCurrentModule = NULL;
506
507static VOID vboxDrvLoggerCallback(VBOXDRVCFG_LOG_SEVERITY enmSeverity, char *pszMsg, void *pvContext)
508{
509 RT_NOREF1(pvContext);
510 switch (enmSeverity)
511 {
512 case VBOXDRVCFG_LOG_SEVERITY_FLOW:
513 case VBOXDRVCFG_LOG_SEVERITY_REGULAR:
514 break;
515 case VBOXDRVCFG_LOG_SEVERITY_REL:
516 if (g_hCurrentModule)
517 logString(g_hCurrentModule, pszMsg);
518 break;
519 default:
520 break;
521 }
522}
523
524static VOID netCfgLoggerCallback(LPCSTR szString)
525{
526 if (g_hCurrentModule)
527 logString(g_hCurrentModule, szString);
528}
529
530static VOID netCfgLoggerDisable()
531{
532 if (g_hCurrentModule)
533 {
534 VBoxNetCfgWinSetLogging((LOG_ROUTINE)NULL);
535 g_hCurrentModule = NULL;
536 }
537}
538
539static VOID netCfgLoggerEnable(MSIHANDLE hModule)
540{
541 NonStandardAssert(hModule);
542
543 if (g_hCurrentModule)
544 netCfgLoggerDisable();
545
546 g_hCurrentModule = hModule;
547
548 VBoxNetCfgWinSetLogging((LOG_ROUTINE)netCfgLoggerCallback);
549 /* uncomment next line if you want to add logging information from VBoxDrvCfg.cpp */
550// VBoxDrvCfgLoggerSet(vboxDrvLoggerCallback, NULL);
551}
552
553static UINT errorConvertFromHResult(MSIHANDLE hModule, HRESULT hr)
554{
555 UINT uRet;
556 switch (hr)
557 {
558 case S_OK:
559 uRet = ERROR_SUCCESS;
560 break;
561
562 case NETCFG_S_REBOOT:
563 {
564 logStringW(hModule, L"Reboot required, setting REBOOT property to \"force\"");
565 HRESULT hr2 = MsiSetPropertyW(hModule, L"REBOOT", L"Force");
566 if (hr2 != ERROR_SUCCESS)
567 logStringW(hModule, L"Failed to set REBOOT property, error = 0x%x", hr2);
568 uRet = ERROR_SUCCESS; /* Never fail here. */
569 break;
570 }
571
572 default:
573 logStringW(hModule, L"Converting unhandled HRESULT (0x%x) to ERROR_GEN_FAILURE", hr);
574 uRet = ERROR_GEN_FAILURE;
575 }
576 return uRet;
577}
578
579static MSIHANDLE createNetCfgLockedMsgRecord(MSIHANDLE hModule)
580{
581 MSIHANDLE hRecord = MsiCreateRecord(2);
582 if (hRecord)
583 {
584 UINT uErr = MsiRecordSetInteger(hRecord, 1, 25001);
585 if (uErr != ERROR_SUCCESS)
586 {
587 logStringW(hModule, L"createNetCfgLockedMsgRecord: MsiRecordSetInteger failed, error = 0x%x", uErr);
588 MsiCloseHandle(hRecord);
589 hRecord = NULL;
590 }
591 }
592 else
593 logStringW(hModule, L"createNetCfgLockedMsgRecord: Failed to create a record");
594
595 return hRecord;
596}
597
598static UINT doNetCfgInit(MSIHANDLE hModule, INetCfg **ppnc, BOOL bWrite)
599{
600 MSIHANDLE hMsg = NULL;
601 UINT uErr = ERROR_GEN_FAILURE;
602 int MsgResult;
603 int cRetries = 0;
604
605 do
606 {
607 LPWSTR lpszLockedBy;
608 HRESULT hr = VBoxNetCfgWinQueryINetCfg(ppnc, bWrite, VBOX_NETCFG_APP_NAME, 10000, &lpszLockedBy);
609 if (hr != NETCFG_E_NO_WRITE_LOCK)
610 {
611 if (FAILED(hr))
612 logStringW(hModule, L"doNetCfgInit: VBoxNetCfgWinQueryINetCfg failed, error = 0x%x", hr);
613 uErr = errorConvertFromHResult(hModule, hr);
614 break;
615 }
616
617 /* hr == NETCFG_E_NO_WRITE_LOCK */
618
619 if (!lpszLockedBy)
620 {
621 logStringW(hModule, L"doNetCfgInit: lpszLockedBy == NULL, breaking");
622 break;
623 }
624
625 /* on vista the 6to4svc.dll periodically maintains the lock for some reason,
626 * if this is the case, increase the wait period by retrying multiple times
627 * NOTE: we could alternatively increase the wait timeout,
628 * however it seems unneeded for most cases, e.g. in case some network connection property
629 * dialog is opened, it would be better to post a notification to the user as soon as possible
630 * rather than waiting for a longer period of time before displaying it */
631 if ( cRetries < VBOX_NETCFG_MAX_RETRIES
632 && !wcscmp(lpszLockedBy, L"6to4svc.dll"))
633 {
634 cRetries++;
635 logStringW(hModule, L"doNetCfgInit: lpszLockedBy is 6to4svc.dll, retrying %d out of %d", cRetries, VBOX_NETCFG_MAX_RETRIES);
636 MsgResult = IDRETRY;
637 }
638 else
639 {
640 if (!hMsg)
641 {
642 hMsg = createNetCfgLockedMsgRecord(hModule);
643 if (!hMsg)
644 {
645 logStringW(hModule, L"doNetCfgInit: Failed to create a message record, breaking");
646 CoTaskMemFree(lpszLockedBy);
647 break;
648 }
649 }
650
651 UINT rTmp = MsiRecordSetStringW(hMsg, 2, lpszLockedBy);
652 NonStandardAssert(rTmp == ERROR_SUCCESS);
653 if (rTmp != ERROR_SUCCESS)
654 {
655 logStringW(hModule, L"doNetCfgInit: MsiRecordSetStringW failed, error = 0x%x", rTmp);
656 CoTaskMemFree(lpszLockedBy);
657 break;
658 }
659
660 MsgResult = MsiProcessMessage(hModule, (INSTALLMESSAGE)(INSTALLMESSAGE_USER | MB_RETRYCANCEL), hMsg);
661 NonStandardAssert(MsgResult == IDRETRY || MsgResult == IDCANCEL);
662 logStringW(hModule, L"doNetCfgInit: MsiProcessMessage returned (0x%x)", MsgResult);
663 }
664 CoTaskMemFree(lpszLockedBy);
665 } while(MsgResult == IDRETRY);
666
667 if (hMsg)
668 MsiCloseHandle(hMsg);
669
670 return uErr;
671}
672
673static UINT vboxNetFltQueryInfArray(MSIHANDLE hModule, OUT LPWSTR pwszPtInf, OUT LPWSTR pwszMpInf, DWORD dwSize)
674{
675 DWORD dwBuf = dwSize - RT_MAX(sizeof(NETFLT_PT_INF_REL_PATH), sizeof(NETFLT_MP_INF_REL_PATH));
676 UINT uErr = MsiGetPropertyW(hModule, L"CustomActionData", pwszPtInf, &dwBuf);
677 if ( uErr == ERROR_SUCCESS
678 && dwBuf)
679 {
680 wcscpy(pwszMpInf, pwszPtInf);
681
682 wcsncat(pwszPtInf, NETFLT_PT_INF_REL_PATH, sizeof(NETFLT_PT_INF_REL_PATH));
683 logStringW(hModule, L"vboxNetFltQueryInfArray: INF 1: %s", pwszPtInf);
684
685 wcsncat(pwszMpInf, NETFLT_MP_INF_REL_PATH, sizeof(NETFLT_MP_INF_REL_PATH));
686 logStringW(hModule, L"vboxNetFltQueryInfArray: INF 2: %s", pwszMpInf);
687 }
688 else if (uErr != ERROR_SUCCESS)
689 logStringW(hModule, L"vboxNetFltQueryInfArray: MsiGetPropertyW failed, error = 0x%x", uErr);
690 else
691 {
692 logStringW(hModule, L"vboxNetFltQueryInfArray: Empty installation directory");
693 uErr = ERROR_GEN_FAILURE;
694 }
695
696 return uErr;
697}
698
699#endif /*VBOX_WITH_NETFLT*/
700
701UINT __stdcall UninstallNetFlt(MSIHANDLE hModule)
702{
703#ifdef VBOX_WITH_NETFLT
704 INetCfg *pNetCfg;
705 UINT uErr;
706
707 netCfgLoggerEnable(hModule);
708
709 BOOL bOldIntMode = SetupSetNonInteractiveMode(FALSE);
710
711 __try
712 {
713 logStringW(hModule, L"Uninstalling NetFlt");
714
715 uErr = doNetCfgInit(hModule, &pNetCfg, TRUE);
716 if (uErr == ERROR_SUCCESS)
717 {
718 HRESULT hr = VBoxNetCfgWinNetFltUninstall(pNetCfg);
719 if (hr != S_OK)
720 logStringW(hModule, L"UninstallNetFlt: VBoxNetCfgWinUninstallComponent failed, error = 0x%x", hr);
721
722 uErr = errorConvertFromHResult(hModule, hr);
723
724 VBoxNetCfgWinReleaseINetCfg(pNetCfg, TRUE);
725
726 logStringW(hModule, L"Uninstalling NetFlt done, error = 0x%x", uErr);
727 }
728 else
729 logStringW(hModule, L"UninstallNetFlt: doNetCfgInit failed, error = 0x%x", uErr);
730 }
731 __finally
732 {
733 if (bOldIntMode)
734 {
735 /* The prev mode != FALSE, i.e. non-interactive. */
736 SetupSetNonInteractiveMode(bOldIntMode);
737 }
738 netCfgLoggerDisable();
739 }
740#endif /* VBOX_WITH_NETFLT */
741
742 /* Never fail the install even if we did not succeed. */
743 return ERROR_SUCCESS;
744}
745
746UINT __stdcall InstallNetFlt(MSIHANDLE hModule)
747{
748#ifdef VBOX_WITH_NETFLT
749 UINT uErr;
750 INetCfg *pNetCfg;
751
752 netCfgLoggerEnable(hModule);
753
754 BOOL bOldIntMode = SetupSetNonInteractiveMode(FALSE);
755
756 __try
757 {
758
759 logStringW(hModule, L"InstallNetFlt: Installing NetFlt");
760
761 uErr = doNetCfgInit(hModule, &pNetCfg, TRUE);
762 if (uErr == ERROR_SUCCESS)
763 {
764 WCHAR wszPtInf[MAX_PATH];
765 WCHAR wszMpInf[MAX_PATH];
766 uErr = vboxNetFltQueryInfArray(hModule, wszPtInf, wszMpInf, sizeof(wszMpInf));
767 if (uErr == ERROR_SUCCESS)
768 {
769 LPCWSTR const apwszInfs[] = { wszPtInf, wszMpInf };
770 HRESULT hr = VBoxNetCfgWinNetFltInstall(pNetCfg, &apwszInfs[0], RT_ELEMENTS(apwszInfs));
771 if (FAILED(hr))
772 logStringW(hModule, L"InstallNetFlt: VBoxNetCfgWinNetFltInstall failed, error = 0x%x", hr);
773
774 uErr = errorConvertFromHResult(hModule, hr);
775 }
776 else
777 logStringW(hModule, L"InstallNetFlt: vboxNetFltQueryInfArray failed, error = 0x%x", uErr);
778
779 VBoxNetCfgWinReleaseINetCfg(pNetCfg, TRUE);
780
781 logStringW(hModule, L"InstallNetFlt: Done");
782 }
783 else
784 logStringW(hModule, L"InstallNetFlt: doNetCfgInit failed, error = 0x%x", uErr);
785 }
786 __finally
787 {
788 if (bOldIntMode)
789 {
790 /* The prev mode != FALSE, i.e. non-interactive. */
791 SetupSetNonInteractiveMode(bOldIntMode);
792 }
793 netCfgLoggerDisable();
794 }
795#endif /* VBOX_WITH_NETFLT */
796
797 /* Never fail the install even if we did not succeed. */
798 return ERROR_SUCCESS;
799}
800
801
802UINT __stdcall UninstallNetLwf(MSIHANDLE hModule)
803{
804#ifdef VBOX_WITH_NETFLT
805 INetCfg *pNetCfg;
806 UINT uErr;
807
808 netCfgLoggerEnable(hModule);
809
810 BOOL bOldIntMode = SetupSetNonInteractiveMode(FALSE);
811
812 __try
813 {
814 logStringW(hModule, L"Uninstalling NetLwf");
815
816 uErr = doNetCfgInit(hModule, &pNetCfg, TRUE);
817 if (uErr == ERROR_SUCCESS)
818 {
819 HRESULT hr = VBoxNetCfgWinNetLwfUninstall(pNetCfg);
820 if (hr != S_OK)
821 logStringW(hModule, L"UninstallNetLwf: VBoxNetCfgWinUninstallComponent failed, error = 0x%x", hr);
822
823 uErr = errorConvertFromHResult(hModule, hr);
824
825 VBoxNetCfgWinReleaseINetCfg(pNetCfg, TRUE);
826
827 logStringW(hModule, L"Uninstalling NetLwf done, error = 0x%x", uErr);
828 }
829 else
830 logStringW(hModule, L"UninstallNetLwf: doNetCfgInit failed, error = 0x%x", uErr);
831 }
832 __finally
833 {
834 if (bOldIntMode)
835 {
836 /* The prev mode != FALSE, i.e. non-interactive. */
837 SetupSetNonInteractiveMode(bOldIntMode);
838 }
839 netCfgLoggerDisable();
840 }
841#endif /* VBOX_WITH_NETFLT */
842
843 /* Never fail the install even if we did not succeed. */
844 return ERROR_SUCCESS;
845}
846
847UINT __stdcall InstallNetLwf(MSIHANDLE hModule)
848{
849#ifdef VBOX_WITH_NETFLT
850 UINT uErr;
851 INetCfg *pNetCfg;
852
853 netCfgLoggerEnable(hModule);
854
855 BOOL bOldIntMode = SetupSetNonInteractiveMode(FALSE);
856
857 __try
858 {
859
860 logStringW(hModule, L"InstallNetLwf: Installing NetLwf");
861
862 uErr = doNetCfgInit(hModule, &pNetCfg, TRUE);
863 if (uErr == ERROR_SUCCESS)
864 {
865 WCHAR wszInf[MAX_PATH];
866 DWORD cchInf = RT_ELEMENTS(wszInf) - sizeof(NETLWF_INF_NAME) - 1;
867 UINT uErr = MsiGetPropertyW(hModule, L"CustomActionData", wszInf, &cchInf);
868 if (uErr == ERROR_SUCCESS)
869 {
870 if (cchInf)
871 {
872 if (wszInf[cchInf - 1] != L'\\')
873 {
874 wszInf[cchInf++] = L'\\';
875 wszInf[cchInf] = L'\0';
876 }
877
878 wcscat(wszInf, NETLWF_INF_NAME);
879
880 HRESULT hr = VBoxNetCfgWinNetLwfInstall(pNetCfg, wszInf);
881 if (FAILED(hr))
882 logStringW(hModule, L"InstallNetLwf: VBoxNetCfgWinNetLwfInstall failed, error = 0x%x", hr);
883
884 uErr = errorConvertFromHResult(hModule, hr);
885 }
886 else
887 {
888 logStringW(hModule, L"vboxNetFltQueryInfArray: Empty installation directory");
889 uErr = ERROR_GEN_FAILURE;
890 }
891 }
892 else
893 logStringW(hModule, L"vboxNetFltQueryInfArray: MsiGetPropertyW failed, error = 0x%x", uErr);
894
895 VBoxNetCfgWinReleaseINetCfg(pNetCfg, TRUE);
896
897 logStringW(hModule, L"InstallNetLwf: Done");
898 }
899 else
900 logStringW(hModule, L"InstallNetLwf: doNetCfgInit failed, error = 0x%x", uErr);
901 }
902 __finally
903 {
904 if (bOldIntMode)
905 {
906 /* The prev mode != FALSE, i.e. non-interactive. */
907 SetupSetNonInteractiveMode(bOldIntMode);
908 }
909 netCfgLoggerDisable();
910 }
911#endif /* VBOX_WITH_NETFLT */
912
913 /* Never fail the install even if we did not succeed. */
914 return ERROR_SUCCESS;
915}
916
917
918#if 0
919static BOOL RenameHostOnlyConnectionsCallback(HDEVINFO hDevInfo, PSP_DEVINFO_DATA pDev, PVOID pContext)
920{
921 WCHAR DevName[256];
922 DWORD winEr;
923
924 if (SetupDiGetDeviceRegistryPropertyW(hDevInfo, pDev,
925 SPDRP_FRIENDLYNAME , /* IN DWORD Property,*/
926 NULL, /*OUT PDWORD PropertyRegDataType, OPTIONAL*/
927 (PBYTE)DevName, /*OUT PBYTE PropertyBuffer,*/
928 sizeof(DevName), /* IN DWORD PropertyBufferSize,*/
929 NULL /*OUT PDWORD RequiredSize OPTIONAL*/
930 ))
931 {
932 HKEY hKey = SetupDiOpenDevRegKey(hDevInfo, pDev,
933 DICS_FLAG_GLOBAL, /* IN DWORD Scope,*/
934 0, /*IN DWORD HwProfile, */
935 DIREG_DRV, /* IN DWORD KeyType, */
936 KEY_READ /*IN REGSAM samDesired*/
937 );
938 NonStandardAssert(hKey != INVALID_HANDLE_VALUE);
939 if (hKey != INVALID_HANDLE_VALUE)
940 {
941 WCHAR guid[50];
942 DWORD cbGuid=sizeof(guid);
943 winEr = RegQueryValueExW(hKey,
944 L"NetCfgInstanceId", /*__in_opt LPCTSTR lpValueName,*/
945 NULL, /*__reserved LPDWORD lpReserved,*/
946 NULL, /*__out_opt LPDWORD lpType,*/
947 (LPBYTE)guid, /*__out_opt LPBYTE lpData,*/
948 &cbGuid /*guid__inout_opt LPDWORD lpcbData*/
949 );
950 NonStandardAssert(winEr == ERROR_SUCCESS);
951 if (winEr == ERROR_SUCCESS)
952 {
953 WCHAR ConnectoinName[128];
954 ULONG cbName = sizeof(ConnectoinName);
955
956 HRESULT hr = VBoxNetCfgWinGenHostonlyConnectionName (DevName, ConnectoinName, &cbName);
957 NonStandardAssert(hr == S_OK);
958 if (SUCCEEDED(hr))
959 {
960 hr = VBoxNetCfgWinRenameConnection(guid, ConnectoinName);
961 NonStandardAssert(hr == S_OK);
962 }
963 }
964 }
965 RegCloseKey(hKey);
966 }
967 else
968 {
969 NonStandardAssert(0);
970 }
971
972 return TRUE;
973}
974#endif
975
976static UINT _createHostOnlyInterface(MSIHANDLE hModule, LPCWSTR pwszId, LPCWSTR pwszInfName)
977{
978#ifdef VBOX_WITH_NETFLT
979 netCfgLoggerEnable(hModule);
980
981 BOOL fSetupModeInteractive = SetupSetNonInteractiveMode(FALSE);
982
983 logStringW(hModule, L"CreateHostOnlyInterface: Creating host-only interface");
984
985 HRESULT hr = E_FAIL;
986 GUID guid;
987 WCHAR wszMpInf[MAX_PATH];
988 DWORD cchMpInf = RT_ELEMENTS(wszMpInf) - (DWORD)wcslen(pwszInfName) - 1 - 1;
989 LPCWSTR pwszInfPath = NULL;
990 bool fIsFile = false;
991 UINT uErr = MsiGetPropertyW(hModule, L"CustomActionData", wszMpInf, &cchMpInf);
992 if (uErr == ERROR_SUCCESS)
993 {
994 if (cchMpInf)
995 {
996 logStringW(hModule, L"CreateHostOnlyInterface: NetAdpDir property = %s", wszMpInf);
997 if (wszMpInf[cchMpInf - 1] != L'\\')
998 {
999 wszMpInf[cchMpInf++] = L'\\';
1000 wszMpInf[cchMpInf] = L'\0';
1001 }
1002
1003 wcscat(wszMpInf, pwszInfName);
1004 pwszInfPath = wszMpInf;
1005 fIsFile = true;
1006
1007 logStringW(hModule, L"CreateHostOnlyInterface: Resulting INF path = %s", pwszInfPath);
1008 }
1009 else
1010 logStringW(hModule, L"CreateHostOnlyInterface: VBox installation path is empty");
1011 }
1012 else
1013 logStringW(hModule, L"CreateHostOnlyInterface: Unable to retrieve VBox installation path, error = 0x%x", uErr);
1014
1015 /* Make sure the inf file is installed. */
1016 if (pwszInfPath != NULL && fIsFile)
1017 {
1018 logStringW(hModule, L"CreateHostOnlyInterface: Calling VBoxDrvCfgInfInstall(%s)", pwszInfPath);
1019 hr = VBoxDrvCfgInfInstall(pwszInfPath);
1020 logStringW(hModule, L"CreateHostOnlyInterface: VBoxDrvCfgInfInstall returns 0x%x", hr);
1021 if (FAILED(hr))
1022 logStringW(hModule, L"CreateHostOnlyInterface: Failed to install INF file, error = 0x%x", hr);
1023 }
1024
1025 if (SUCCEEDED(hr))
1026 {
1027 //first, try to update Host Only Network Interface
1028 BOOL fRebootRequired = FALSE;
1029 hr = VBoxNetCfgWinUpdateHostOnlyNetworkInterface(pwszInfPath, &fRebootRequired, pwszId);
1030 if (SUCCEEDED(hr))
1031 {
1032 if (fRebootRequired)
1033 {
1034 logStringW(hModule, L"UpdateHostOnlyInterfaces: Reboot required, setting REBOOT property to force");
1035 HRESULT hr2 = MsiSetPropertyW(hModule, L"REBOOT", L"Force");
1036 if (hr2 != ERROR_SUCCESS)
1037 logStringW(hModule, L"UpdateHostOnlyInterfaces: Failed to set REBOOT property, error = 0x%x", hr2);
1038 }
1039 }
1040 else
1041 logStringW(hModule, L"UpdateHostOnlyInterfaces: VBoxNetCfgWinUpdateHostOnlyNetworkInterface failed, hr = 0x%x", hr);
1042 //in fail case call CreateHostOnlyInterface
1043 if (FAILED(hr))
1044 {
1045 logStringW(hModule, L"CreateHostOnlyInterface: calling VBoxNetCfgWinCreateHostOnlyNetworkInterface");
1046#ifdef VBOXNETCFG_DELAYEDRENAME
1047 BSTR devId;
1048 hr = VBoxNetCfgWinCreateHostOnlyNetworkInterface(pwszInfPath, fIsFile, &guid, &devId, NULL);
1049#else /* !VBOXNETCFG_DELAYEDRENAME */
1050 hr = VBoxNetCfgWinCreateHostOnlyNetworkInterface(pwszInfPath, fIsFile, &guid, NULL, NULL);
1051#endif /* !VBOXNETCFG_DELAYEDRENAME */
1052 logStringW(hModule, L"CreateHostOnlyInterface: VBoxNetCfgWinCreateHostOnlyNetworkInterface returns 0x%x", hr);
1053 if (SUCCEEDED(hr))
1054 {
1055 ULONG ip = inet_addr("192.168.56.1");
1056 ULONG mask = inet_addr("255.255.255.0");
1057 logStringW(hModule, L"CreateHostOnlyInterface: calling VBoxNetCfgWinEnableStaticIpConfig");
1058 hr = VBoxNetCfgWinEnableStaticIpConfig(&guid, ip, mask);
1059 logStringW(hModule, L"CreateHostOnlyInterface: VBoxNetCfgWinEnableStaticIpConfig returns 0x%x", hr);
1060 if (FAILED(hr))
1061 logStringW(hModule, L"CreateHostOnlyInterface: VBoxNetCfgWinEnableStaticIpConfig failed, error = 0x%x", hr);
1062#ifdef VBOXNETCFG_DELAYEDRENAME
1063 hr = VBoxNetCfgWinRenameHostOnlyConnection(&guid, devId, NULL);
1064 if (FAILED(hr))
1065 logStringW(hModule, L"CreateHostOnlyInterface: VBoxNetCfgWinRenameHostOnlyConnection failed, error = 0x%x", hr);
1066 SysFreeString(devId);
1067#endif /* VBOXNETCFG_DELAYEDRENAME */
1068 }
1069 else
1070 logStringW(hModule, L"CreateHostOnlyInterface: VBoxNetCfgWinCreateHostOnlyNetworkInterface failed, error = 0x%x", hr);
1071 }
1072 }
1073
1074 if (SUCCEEDED(hr))
1075 logStringW(hModule, L"CreateHostOnlyInterface: Creating host-only interface done");
1076
1077 /* Restore original setup mode. */
1078 logStringW(hModule, L"CreateHostOnlyInterface: Almost done...");
1079 if (fSetupModeInteractive)
1080 SetupSetNonInteractiveMode(fSetupModeInteractive);
1081
1082 netCfgLoggerDisable();
1083
1084#endif /* VBOX_WITH_NETFLT */
1085
1086 logStringW(hModule, L"CreateHostOnlyInterface: Returns success (ignoring all failures)");
1087 /* Never fail the install even if we did not succeed. */
1088 return ERROR_SUCCESS;
1089}
1090
1091UINT __stdcall CreateHostOnlyInterface(MSIHANDLE hModule)
1092{
1093 return _createHostOnlyInterface(hModule, NETADP_ID, L"VBoxNetAdp.inf");
1094}
1095
1096UINT __stdcall Ndis6CreateHostOnlyInterface(MSIHANDLE hModule)
1097{
1098 return _createHostOnlyInterface(hModule, NETADP_ID, L"VBoxNetAdp6.inf");
1099}
1100
1101static UINT _removeHostOnlyInterfaces(MSIHANDLE hModule, LPCWSTR pwszId)
1102{
1103#ifdef VBOX_WITH_NETFLT
1104 netCfgLoggerEnable(hModule);
1105
1106 logStringW(hModule, L"RemoveHostOnlyInterfaces: Removing all host-only interfaces");
1107
1108 BOOL fSetupModeInteractive = SetupSetNonInteractiveMode(FALSE);
1109
1110 HRESULT hr = VBoxNetCfgWinRemoveAllNetDevicesOfId(pwszId);
1111 if (SUCCEEDED(hr))
1112 {
1113 hr = VBoxDrvCfgInfUninstallAllSetupDi(&GUID_DEVCLASS_NET, L"Net", pwszId, SUOI_FORCEDELETE/* could be SUOI_FORCEDELETE */);
1114 if (FAILED(hr))
1115 {
1116 logStringW(hModule, L"RemoveHostOnlyInterfaces: NetAdp uninstalled successfully, but failed to remove INF files");
1117 }
1118 else
1119 logStringW(hModule, L"RemoveHostOnlyInterfaces: NetAdp uninstalled successfully");
1120
1121 }
1122 else
1123 logStringW(hModule, L"RemoveHostOnlyInterfaces: NetAdp uninstall failed, hr = 0x%x", hr);
1124
1125 /* Restore original setup mode. */
1126 if (fSetupModeInteractive)
1127 SetupSetNonInteractiveMode(fSetupModeInteractive);
1128
1129 netCfgLoggerDisable();
1130#endif /* VBOX_WITH_NETFLT */
1131
1132 /* Never fail the install even if we did not succeed. */
1133 return ERROR_SUCCESS;
1134}
1135
1136UINT __stdcall RemoveHostOnlyInterfaces(MSIHANDLE hModule)
1137{
1138 return _removeHostOnlyInterfaces(hModule, NETADP_ID);
1139}
1140
1141static UINT _stopHostOnlyInterfaces(MSIHANDLE hModule, LPCWSTR pwszId)
1142{
1143#ifdef VBOX_WITH_NETFLT
1144 netCfgLoggerEnable(hModule);
1145
1146 logStringW(hModule, L"StopHostOnlyInterfaces: Stopping all host-only interfaces");
1147
1148 BOOL fSetupModeInteractive = SetupSetNonInteractiveMode(FALSE);
1149
1150 HRESULT hr = VBoxNetCfgWinPropChangeAllNetDevicesOfId(pwszId, VBOXNECTFGWINPROPCHANGE_TYPE_DISABLE);
1151 if (SUCCEEDED(hr))
1152 {
1153 logStringW(hModule, L"StopHostOnlyInterfaces: Disabling host interfaces was successful, hr = 0x%x", hr);
1154 }
1155 else
1156 logStringW(hModule, L"StopHostOnlyInterfaces: Disabling host interfaces failed, hr = 0x%x", hr);
1157
1158 /* Restore original setup mode. */
1159 if (fSetupModeInteractive)
1160 SetupSetNonInteractiveMode(fSetupModeInteractive);
1161
1162 netCfgLoggerDisable();
1163#endif /* VBOX_WITH_NETFLT */
1164
1165 /* Never fail the install even if we did not succeed. */
1166 return ERROR_SUCCESS;
1167}
1168
1169UINT __stdcall StopHostOnlyInterfaces(MSIHANDLE hModule)
1170{
1171 return _stopHostOnlyInterfaces(hModule, NETADP_ID);
1172}
1173
1174static UINT _updateHostOnlyInterfaces(MSIHANDLE hModule, LPCWSTR pwszInfName, LPCWSTR pwszId)
1175{
1176#ifdef VBOX_WITH_NETFLT
1177 netCfgLoggerEnable(hModule);
1178
1179 logStringW(hModule, L"UpdateHostOnlyInterfaces: Updating all host-only interfaces");
1180
1181 BOOL fSetupModeInteractive = SetupSetNonInteractiveMode(FALSE);
1182
1183 WCHAR wszMpInf[MAX_PATH];
1184 DWORD cchMpInf = RT_ELEMENTS(wszMpInf) - (DWORD)wcslen(pwszInfName) - 1 - 1;
1185 LPCWSTR pwszInfPath = NULL;
1186 bool fIsFile = false;
1187 UINT uErr = MsiGetPropertyW(hModule, L"CustomActionData", wszMpInf, &cchMpInf);
1188 if (uErr == ERROR_SUCCESS)
1189 {
1190 if (cchMpInf)
1191 {
1192 logStringW(hModule, L"UpdateHostOnlyInterfaces: NetAdpDir property = %s", wszMpInf);
1193 if (wszMpInf[cchMpInf - 1] != L'\\')
1194 {
1195 wszMpInf[cchMpInf++] = L'\\';
1196 wszMpInf[cchMpInf] = L'\0';
1197 }
1198
1199 wcscat(wszMpInf, pwszInfName);
1200 pwszInfPath = wszMpInf;
1201 fIsFile = true;
1202
1203 logStringW(hModule, L"UpdateHostOnlyInterfaces: Resulting INF path = %s", pwszInfPath);
1204
1205 DWORD attrFile = GetFileAttributesW(pwszInfPath);
1206 if (attrFile == INVALID_FILE_ATTRIBUTES)
1207 {
1208 DWORD dwErr = GetLastError();
1209 logStringW(hModule, L"UpdateHostOnlyInterfaces: File \"%s\" not found, dwErr=%ld",
1210 pwszInfPath, dwErr);
1211 }
1212 else
1213 {
1214 logStringW(hModule, L"UpdateHostOnlyInterfaces: File \"%s\" exists",
1215 pwszInfPath);
1216
1217 BOOL fRebootRequired = FALSE;
1218 HRESULT hr = VBoxNetCfgWinUpdateHostOnlyNetworkInterface(pwszInfPath, &fRebootRequired, pwszId);
1219 if (SUCCEEDED(hr))
1220 {
1221 if (fRebootRequired)
1222 {
1223 logStringW(hModule, L"UpdateHostOnlyInterfaces: Reboot required, setting REBOOT property to force");
1224 HRESULT hr2 = MsiSetPropertyW(hModule, L"REBOOT", L"Force");
1225 if (hr2 != ERROR_SUCCESS)
1226 logStringW(hModule, L"UpdateHostOnlyInterfaces: Failed to set REBOOT property, error = 0x%x", hr2);
1227 }
1228 }
1229 else
1230 logStringW(hModule, L"UpdateHostOnlyInterfaces: VBoxNetCfgWinUpdateHostOnlyNetworkInterface failed, hr = 0x%x", hr);
1231 }
1232 }
1233 else
1234 logStringW(hModule, L"UpdateHostOnlyInterfaces: VBox installation path is empty");
1235 }
1236 else
1237 logStringW(hModule, L"UpdateHostOnlyInterfaces: Unable to retrieve VBox installation path, error = 0x%x", uErr);
1238
1239 /* Restore original setup mode. */
1240 if (fSetupModeInteractive)
1241 SetupSetNonInteractiveMode(fSetupModeInteractive);
1242
1243 netCfgLoggerDisable();
1244#endif /* VBOX_WITH_NETFLT */
1245
1246 /* Never fail the install even if we did not succeed. */
1247 return ERROR_SUCCESS;
1248}
1249
1250UINT __stdcall UpdateHostOnlyInterfaces(MSIHANDLE hModule)
1251{
1252 return _updateHostOnlyInterfaces(hModule, L"VBoxNetAdp.inf", NETADP_ID);
1253}
1254
1255UINT __stdcall Ndis6UpdateHostOnlyInterfaces(MSIHANDLE hModule)
1256{
1257 return _updateHostOnlyInterfaces(hModule, L"VBoxNetAdp6.inf", NETADP_ID);
1258}
1259
1260static UINT _uninstallNetAdp(MSIHANDLE hModule, LPCWSTR pwszId)
1261{
1262#ifdef VBOX_WITH_NETFLT
1263 INetCfg *pNetCfg;
1264 UINT uErr;
1265
1266 netCfgLoggerEnable(hModule);
1267
1268 BOOL bOldIntMode = SetupSetNonInteractiveMode(FALSE);
1269
1270 __try
1271 {
1272 logStringW(hModule, L"Uninstalling NetAdp");
1273
1274 uErr = doNetCfgInit(hModule, &pNetCfg, TRUE);
1275 if (uErr == ERROR_SUCCESS)
1276 {
1277 HRESULT hr = VBoxNetCfgWinNetAdpUninstall(pNetCfg, pwszId);
1278 if (hr != S_OK)
1279 logStringW(hModule, L"UninstallNetAdp: VBoxNetCfgWinUninstallComponent failed, error = 0x%x", hr);
1280
1281 uErr = errorConvertFromHResult(hModule, hr);
1282
1283 VBoxNetCfgWinReleaseINetCfg(pNetCfg, TRUE);
1284
1285 logStringW(hModule, L"Uninstalling NetAdp done, error = 0x%x", uErr);
1286 }
1287 else
1288 logStringW(hModule, L"UninstallNetAdp: doNetCfgInit failed, error = 0x%x", uErr);
1289 }
1290 __finally
1291 {
1292 if (bOldIntMode)
1293 {
1294 /* The prev mode != FALSE, i.e. non-interactive. */
1295 SetupSetNonInteractiveMode(bOldIntMode);
1296 }
1297 netCfgLoggerDisable();
1298 }
1299#endif /* VBOX_WITH_NETFLT */
1300
1301 /* Never fail the install even if we did not succeed. */
1302 return ERROR_SUCCESS;
1303}
1304
1305UINT __stdcall UninstallNetAdp(MSIHANDLE hModule)
1306{
1307 return _uninstallNetAdp(hModule, NETADP_ID);
1308}
1309
1310static bool isTAPDevice(const WCHAR *pwszGUID)
1311{
1312 HKEY hNetcard;
1313 bool bIsTapDevice = false;
1314 LONG lStatus = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
1315 L"SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002BE10318}",
1316 0, KEY_READ, &hNetcard);
1317 if (lStatus != ERROR_SUCCESS)
1318 return false;
1319
1320 int i = 0;
1321 for (;;)
1322 {
1323 WCHAR wszEnumName[256];
1324 WCHAR wszNetCfgInstanceId[256];
1325 DWORD dwKeyType;
1326 HKEY hNetCardGUID;
1327
1328 DWORD dwLen = sizeof(wszEnumName);
1329 lStatus = RegEnumKeyExW(hNetcard, i, wszEnumName, &dwLen, NULL, NULL, NULL, NULL);
1330 if (lStatus != ERROR_SUCCESS)
1331 break;
1332
1333 lStatus = RegOpenKeyExW(hNetcard, wszEnumName, 0, KEY_READ, &hNetCardGUID);
1334 if (lStatus == ERROR_SUCCESS)
1335 {
1336 dwLen = sizeof(wszNetCfgInstanceId);
1337 lStatus = RegQueryValueExW(hNetCardGUID, L"NetCfgInstanceId", NULL, &dwKeyType, (LPBYTE)wszNetCfgInstanceId, &dwLen);
1338 if ( lStatus == ERROR_SUCCESS
1339 && dwKeyType == REG_SZ)
1340 {
1341 WCHAR wszNetProductName[256];
1342 WCHAR wszNetProviderName[256];
1343
1344 wszNetProductName[0] = 0;
1345 dwLen = sizeof(wszNetProductName);
1346 lStatus = RegQueryValueExW(hNetCardGUID, L"ProductName", NULL, &dwKeyType, (LPBYTE)wszNetProductName, &dwLen);
1347
1348 wszNetProviderName[0] = 0;
1349 dwLen = sizeof(wszNetProviderName);
1350 lStatus = RegQueryValueExW(hNetCardGUID, L"ProviderName", NULL, &dwKeyType, (LPBYTE)wszNetProviderName, &dwLen);
1351
1352 if ( !wcscmp(wszNetCfgInstanceId, pwszGUID)
1353 && !wcscmp(wszNetProductName, L"VirtualBox TAP Adapter")
1354 && ( (!wcscmp(wszNetProviderName, L"innotek GmbH")) /* Legacy stuff. */
1355 || (!wcscmp(wszNetProviderName, L"Sun Microsystems, Inc.")) /* Legacy stuff. */
1356 || (!wcscmp(wszNetProviderName, MY_WTEXT(VBOX_VENDOR))) /* Reflects current vendor string. */
1357 )
1358 )
1359 {
1360 bIsTapDevice = true;
1361 RegCloseKey(hNetCardGUID);
1362 break;
1363 }
1364 }
1365 RegCloseKey(hNetCardGUID);
1366 }
1367 ++i;
1368 }
1369
1370 RegCloseKey(hNetcard);
1371 return bIsTapDevice;
1372}
1373
1374#define SetErrBreak(args) \
1375 if (1) { \
1376 rc = 0; \
1377 logStringW args; \
1378 break; \
1379 } else do {} while (0)
1380
1381int removeNetworkInterface(MSIHANDLE hModule, const WCHAR *pwszGUID)
1382{
1383 int rc = 1;
1384 do
1385 {
1386 WCHAR wszPnPInstanceId[512] = {0};
1387
1388 /* We have to find the device instance ID through a registry search */
1389
1390 HKEY hkeyNetwork = 0;
1391 HKEY hkeyConnection = 0;
1392
1393 do /* break-loop */
1394 {
1395 WCHAR wszRegLocation[256];
1396 swprintf(wszRegLocation,
1397 L"SYSTEM\\CurrentControlSet\\Control\\Network\\{4D36E972-E325-11CE-BFC1-08002BE10318}\\%s",
1398 pwszGUID);
1399 LONG lStatus = RegOpenKeyExW(HKEY_LOCAL_MACHINE, wszRegLocation, 0, KEY_READ, &hkeyNetwork);
1400 if ((lStatus != ERROR_SUCCESS) || !hkeyNetwork)
1401 SetErrBreak((hModule, L"VBox HostInterfaces: Host interface network was not found in registry (%s)! [1]",
1402 wszRegLocation));
1403
1404 lStatus = RegOpenKeyExW(hkeyNetwork, L"Connection", 0, KEY_READ, &hkeyConnection);
1405 if ((lStatus != ERROR_SUCCESS) || !hkeyConnection)
1406 SetErrBreak((hModule, L"VBox HostInterfaces: Host interface network was not found in registry (%s)! [2]",
1407 wszRegLocation));
1408
1409 DWORD len = sizeof(wszPnPInstanceId);
1410 DWORD dwKeyType;
1411 lStatus = RegQueryValueExW(hkeyConnection, L"PnPInstanceID", NULL,
1412 &dwKeyType, (LPBYTE)&wszPnPInstanceId[0], &len);
1413 if ((lStatus != ERROR_SUCCESS) || (dwKeyType != REG_SZ))
1414 SetErrBreak((hModule, L"VBox HostInterfaces: Host interface network was not found in registry (%s)! [3]",
1415 wszRegLocation));
1416 }
1417 while (0);
1418
1419 if (hkeyConnection)
1420 RegCloseKey(hkeyConnection);
1421 if (hkeyNetwork)
1422 RegCloseKey(hkeyNetwork);
1423
1424 /*
1425 * Now we are going to enumerate all network devices and
1426 * wait until we encounter the right device instance ID
1427 */
1428
1429 HDEVINFO hDeviceInfo = INVALID_HANDLE_VALUE;
1430 BOOL fResult;
1431
1432 do
1433 {
1434 GUID netGuid;
1435 SP_DEVINFO_DATA DeviceInfoData;
1436 DWORD index = 0;
1437 DWORD size = 0;
1438
1439 /* initialize the structure size */
1440 DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
1441
1442 /* copy the net class GUID */
1443 memcpy(&netGuid, &GUID_DEVCLASS_NET, sizeof (GUID_DEVCLASS_NET));
1444
1445 /* return a device info set contains all installed devices of the Net class */
1446 hDeviceInfo = SetupDiGetClassDevs(&netGuid, NULL, NULL, DIGCF_PRESENT);
1447 if (hDeviceInfo == INVALID_HANDLE_VALUE)
1448 {
1449 logStringW(hModule, L"VBox HostInterfaces: SetupDiGetClassDevs failed (0x%08X)!", GetLastError());
1450 SetErrBreak((hModule, L"VBox HostInterfaces: Uninstallation failed!"));
1451 }
1452
1453 BOOL fFoundDevice = FALSE;
1454
1455 /* enumerate the driver info list */
1456 while (TRUE)
1457 {
1458 WCHAR *pwszDeviceHwid;
1459
1460 fResult = SetupDiEnumDeviceInfo(hDeviceInfo, index, &DeviceInfoData);
1461 if (!fResult)
1462 {
1463 if (GetLastError() == ERROR_NO_MORE_ITEMS)
1464 break;
1465 else
1466 {
1467 index++;
1468 continue;
1469 }
1470 }
1471
1472 /* try to get the hardware ID registry property */
1473 fResult = SetupDiGetDeviceRegistryProperty(hDeviceInfo,
1474 &DeviceInfoData,
1475 SPDRP_HARDWAREID,
1476 NULL,
1477 NULL,
1478 0,
1479 &size);
1480 if (!fResult)
1481 {
1482 if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
1483 {
1484 index++;
1485 continue;
1486 }
1487
1488 pwszDeviceHwid = (WCHAR *)malloc(size);
1489 if (pwszDeviceHwid)
1490 {
1491 fResult = SetupDiGetDeviceRegistryProperty(hDeviceInfo,
1492 &DeviceInfoData,
1493 SPDRP_HARDWAREID,
1494 NULL,
1495 (PBYTE)pwszDeviceHwid,
1496 size,
1497 NULL);
1498 if (!fResult)
1499 {
1500 free(pwszDeviceHwid);
1501 pwszDeviceHwid = NULL;
1502 index++;
1503 continue;
1504 }
1505 }
1506 }
1507 else
1508 {
1509 /* something is wrong. This shouldn't have worked with a NULL buffer */
1510 index++;
1511 continue;
1512 }
1513
1514 for (WCHAR *t = pwszDeviceHwid;
1515 t && *t && t < &pwszDeviceHwid[size / sizeof(WCHAR)];
1516 t += wcslen(t) + 1)
1517 {
1518 if (!_wcsicmp(L"vboxtap", t))
1519 {
1520 /* get the device instance ID */
1521 WCHAR wszDevID[MAX_DEVICE_ID_LEN];
1522 if (CM_Get_Device_IDW(DeviceInfoData.DevInst,
1523 wszDevID, MAX_DEVICE_ID_LEN, 0) == CR_SUCCESS)
1524 {
1525 /* compare to what we determined before */
1526 if (!wcscmp(wszDevID, wszPnPInstanceId))
1527 {
1528 fFoundDevice = TRUE;
1529 break;
1530 }
1531 }
1532 }
1533 }
1534
1535 if (pwszDeviceHwid)
1536 {
1537 free(pwszDeviceHwid);
1538 pwszDeviceHwid = NULL;
1539 }
1540
1541 if (fFoundDevice)
1542 break;
1543
1544 index++;
1545 }
1546
1547 if (fFoundDevice)
1548 {
1549 fResult = SetupDiSetSelectedDevice(hDeviceInfo, &DeviceInfoData);
1550 if (!fResult)
1551 {
1552 logStringW(hModule, L"VBox HostInterfaces: SetupDiSetSelectedDevice failed (0x%08X)!", GetLastError());
1553 SetErrBreak((hModule, L"VBox HostInterfaces: Uninstallation failed!"));
1554 }
1555
1556 fResult = SetupDiCallClassInstaller(DIF_REMOVE, hDeviceInfo, &DeviceInfoData);
1557 if (!fResult)
1558 {
1559 logStringW(hModule, L"VBox HostInterfaces: SetupDiCallClassInstaller (DIF_REMOVE) failed (0x%08X)!", GetLastError());
1560 SetErrBreak((hModule, L"VBox HostInterfaces: Uninstallation failed!"));
1561 }
1562 }
1563 else
1564 SetErrBreak((hModule, L"VBox HostInterfaces: Host interface network device not found!"));
1565 } while (0);
1566
1567 /* clean up the device info set */
1568 if (hDeviceInfo != INVALID_HANDLE_VALUE)
1569 SetupDiDestroyDeviceInfoList(hDeviceInfo);
1570 } while (0);
1571 return rc;
1572}
1573
1574UINT __stdcall UninstallTAPInstances(MSIHANDLE hModule)
1575{
1576 static const WCHAR *s_wszNetworkKey = L"SYSTEM\\CurrentControlSet\\Control\\Network\\{4D36E972-E325-11CE-BFC1-08002BE10318}";
1577 HKEY hCtrlNet;
1578
1579 LONG lStatus = RegOpenKeyExW(HKEY_LOCAL_MACHINE, s_wszNetworkKey, 0, KEY_READ, &hCtrlNet);
1580 if (lStatus == ERROR_SUCCESS)
1581 {
1582 logStringW(hModule, L"VBox HostInterfaces: Enumerating interfaces ...");
1583 for (int i = 0; ; ++i)
1584 {
1585 WCHAR wszNetworkGUID[256] = { 0 };
1586 DWORD dwLen = (DWORD)sizeof(wszNetworkGUID);
1587 lStatus = RegEnumKeyExW(hCtrlNet, i, wszNetworkGUID, &dwLen, NULL, NULL, NULL, NULL);
1588 if (lStatus != ERROR_SUCCESS)
1589 {
1590 switch (lStatus)
1591 {
1592 case ERROR_NO_MORE_ITEMS:
1593 logStringW(hModule, L"VBox HostInterfaces: No interfaces found.");
1594 break;
1595 default:
1596 logStringW(hModule, L"VBox HostInterfaces: Enumeration failed: %ld", lStatus);
1597 break;
1598 }
1599 break;
1600 }
1601
1602 if (isTAPDevice(wszNetworkGUID))
1603 {
1604 logStringW(hModule, L"VBox HostInterfaces: Removing interface \"%s\" ...", wszNetworkGUID);
1605 removeNetworkInterface(hModule, wszNetworkGUID);
1606 lStatus = RegDeleteKeyW(hCtrlNet, wszNetworkGUID);
1607 }
1608 }
1609 RegCloseKey(hCtrlNet);
1610 logStringW(hModule, L"VBox HostInterfaces: Removing interfaces done.");
1611 }
1612 return ERROR_SUCCESS;
1613}
1614
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