VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/VBoxNetFlt/win/cfg/VBoxNetCfg.cpp@ 57192

Last change on this file since 57192 was 57192, checked in by vboxsync, 9 years ago

pr7933. Temporary decision: added new function VBoxNetCfgWinNetAdpInstall. Only for testing and using inside NetAdpInstall.exe

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 106.5 KB
Line 
1/* $Id: VBoxNetCfg.cpp 57192 2015-08-05 12:51:17Z vboxsync $ */
2/** @file
3 * VBoxNetCfg.cpp - Network Configuration API.
4 */
5/*
6 * Copyright (C) 2011-2015 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16#include "VBox/VBoxNetCfg-win.h"
17#include "VBox/VBoxDrvCfg-win.h"
18
19#define _WIN32_DCOM
20
21#include <iphlpapi.h>
22
23#include <devguid.h>
24#include <stdio.h>
25#include <regstr.h>
26#include <shlobj.h>
27#include <cfgmgr32.h>
28#include <tchar.h>
29#include <objbase.h>
30
31#include <crtdbg.h>
32#include <stdlib.h>
33#include <string.h>
34
35#include <Wbemidl.h>
36#include <comdef.h>
37
38
39#ifndef Assert /** @todo r=bird: where would this be defined? */
40//# ifdef DEBUG
41//# define Assert(_expr) assert(_expr)
42//# else
43//# define Assert(_expr) do{ }while (0)
44//# endif
45# define Assert _ASSERT
46# define AssertMsg(expr, msg) do{}while (0)
47#endif
48static LOG_ROUTINE g_Logger = NULL;
49
50static VOID DoLogging(LPCSTR szString, ...);
51#define NonStandardLog DoLogging
52#define NonStandardLogFlow(x) DoLogging x
53
54#define DbgLog /** @todo r=bird: What does this do? */
55
56#define VBOX_NETCFG_LOCK_TIME_OUT 5000 /** @todo r=bird: What does this do? */
57
58
59static HRESULT vboxNetCfgWinINetCfgLock(IN INetCfg *pNetCfg,
60 IN LPCWSTR pszwClientDescription,
61 IN DWORD cmsTimeout,
62 OUT LPWSTR *ppszwClientDescription)
63{
64 INetCfgLock *pLock;
65 HRESULT hr = pNetCfg->QueryInterface(IID_INetCfgLock, (PVOID*)&pLock);
66 if (FAILED(hr))
67 {
68 NonStandardLogFlow(("QueryInterface failed, hr (0x%x)\n", hr));
69 return hr;
70 }
71
72 hr = pLock->AcquireWriteLock(cmsTimeout, pszwClientDescription, ppszwClientDescription);
73 if (hr == S_FALSE)
74 {
75 NonStandardLogFlow(("Write lock busy\n"));
76 }
77 else if (FAILED(hr))
78 {
79 NonStandardLogFlow(("AcquireWriteLock failed, hr (0x%x)\n", hr));
80 }
81
82 pLock->Release();
83 return hr;
84}
85
86static HRESULT vboxNetCfgWinINetCfgUnlock(IN INetCfg *pNetCfg)
87{
88 INetCfgLock *pLock;
89 HRESULT hr = pNetCfg->QueryInterface(IID_INetCfgLock, (PVOID*)&pLock);
90 if (FAILED(hr))
91 {
92 NonStandardLogFlow(("QueryInterface failed, hr (0x%x)\n", hr));
93 return hr;
94 }
95
96 hr = pLock->ReleaseWriteLock();
97 if (FAILED(hr))
98 NonStandardLogFlow(("ReleaseWriteLock failed, hr (0x%x)\n", hr));
99
100 pLock->Release();
101 return hr;
102}
103
104VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinQueryINetCfg(OUT INetCfg **ppNetCfg,
105 IN BOOL fGetWriteLock,
106 IN LPCWSTR pszwClientDescription,
107 IN DWORD cmsTimeout,
108 OUT LPWSTR *ppszwClientDescription)
109{
110 INetCfg *pNetCfg;
111 HRESULT hr = CoCreateInstance(CLSID_CNetCfg, NULL, CLSCTX_INPROC_SERVER, IID_INetCfg, (PVOID*)&pNetCfg);
112 if (FAILED(hr))
113 {
114 NonStandardLogFlow(("CoCreateInstance failed, hr (0x%x)\n", hr));
115 return hr;
116 }
117
118 if (fGetWriteLock)
119 {
120 hr = vboxNetCfgWinINetCfgLock(pNetCfg, pszwClientDescription, cmsTimeout, ppszwClientDescription);
121 if (hr == S_FALSE)
122 {
123 NonStandardLogFlow(("Write lock is busy\n", hr));
124 hr = NETCFG_E_NO_WRITE_LOCK;
125 }
126 }
127
128 if (SUCCEEDED(hr))
129 {
130 hr = pNetCfg->Initialize(NULL);
131 if (SUCCEEDED(hr))
132 {
133 *ppNetCfg = pNetCfg;
134 return S_OK;
135 }
136 else
137 NonStandardLogFlow(("Initialize failed, hr (0x%x)\n", hr));
138 }
139
140 pNetCfg->Release();
141 return hr;
142}
143
144VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinReleaseINetCfg(IN INetCfg *pNetCfg, IN BOOL fHasWriteLock)
145{
146 if (!pNetCfg) /* If network config has been released already, just bail out. */
147 {
148 NonStandardLogFlow(("Warning: No network config given but write lock is set to TRUE\n"));
149 return S_OK;
150 }
151
152 HRESULT hr = pNetCfg->Uninitialize();
153 if (FAILED(hr))
154 {
155 NonStandardLogFlow(("Uninitialize failed, hr (0x%x)\n", hr));
156 /* Try to release the write lock below. */
157 }
158
159 if (fHasWriteLock)
160 {
161 HRESULT hr2 = vboxNetCfgWinINetCfgUnlock(pNetCfg);
162 if (FAILED(hr2))
163 NonStandardLogFlow(("vboxNetCfgWinINetCfgUnlock failed, hr (0x%x)\n", hr2));
164 if (SUCCEEDED(hr))
165 hr = hr2;
166 }
167
168 pNetCfg->Release();
169 return hr;
170}
171
172static HRESULT vboxNetCfgWinGetComponentByGuidEnum(IEnumNetCfgComponent *pEnumNcc,
173 IN const GUID *pGuid,
174 OUT INetCfgComponent **ppNcc)
175{
176 HRESULT hr = pEnumNcc->Reset();
177 if (FAILED(hr))
178 {
179 NonStandardLogFlow(("Reset failed, hr (0x%x)\n", hr));
180 return hr;
181 }
182
183 INetCfgComponent *pNcc;
184 while ((hr = pEnumNcc->Next(1, &pNcc, NULL)) == S_OK)
185 {
186 ULONG uComponentStatus;
187 hr = pNcc->GetDeviceStatus(&uComponentStatus);
188 if (SUCCEEDED(hr))
189 {
190 if (uComponentStatus == 0)
191 {
192 GUID NccGuid;
193 hr = pNcc->GetInstanceGuid(&NccGuid);
194
195 if (SUCCEEDED(hr))
196 {
197 if (NccGuid == *pGuid)
198 {
199 /* found the needed device */
200 *ppNcc = pNcc;
201 break;
202 }
203 }
204 else
205 NonStandardLogFlow(("GetInstanceGuid failed, hr (0x%x)\n", hr));
206 }
207 }
208
209 pNcc->Release();
210 }
211 return hr;
212}
213
214VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinGetComponentByGuid(IN INetCfg *pNc,
215 IN const GUID *pguidClass,
216 IN const GUID * pComponentGuid,
217 OUT INetCfgComponent **ppncc)
218{
219 IEnumNetCfgComponent *pEnumNcc;
220 HRESULT hr = pNc->EnumComponents(pguidClass, &pEnumNcc);
221
222 if (SUCCEEDED(hr))
223 {
224 hr = vboxNetCfgWinGetComponentByGuidEnum(pEnumNcc, pComponentGuid, ppncc);
225 if (hr == S_FALSE)
226 {
227 NonStandardLogFlow(("Component not found\n"));
228 }
229 else if (FAILED(hr))
230 {
231 NonStandardLogFlow(("vboxNetCfgWinGetComponentByGuidEnum failed, hr (0x%x)\n", hr));
232 }
233 pEnumNcc->Release();
234 }
235 else
236 NonStandardLogFlow(("EnumComponents failed, hr (0x%x)\n", hr));
237 return hr;
238}
239
240static HRESULT vboxNetCfgWinQueryInstaller(IN INetCfg *pNetCfg, IN const GUID *pguidClass, INetCfgClassSetup **ppSetup)
241{
242 HRESULT hr = pNetCfg->QueryNetCfgClass(pguidClass, IID_INetCfgClassSetup, (void**)ppSetup);
243 if (FAILED(hr))
244 NonStandardLogFlow(("QueryNetCfgClass failed, hr (0x%x)\n", hr));
245 return hr;
246}
247
248VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinInstallComponent(IN INetCfg *pNetCfg, IN LPCWSTR pszwComponentId, IN const GUID *pguidClass,
249 OUT INetCfgComponent **ppComponent)
250{
251 INetCfgClassSetup *pSetup;
252 HRESULT hr = vboxNetCfgWinQueryInstaller(pNetCfg, pguidClass, &pSetup);
253 if (FAILED(hr))
254 {
255 NonStandardLogFlow(("vboxNetCfgWinQueryInstaller failed, hr (0x%x)\n", hr));
256 return hr;
257 }
258
259 OBO_TOKEN Token;
260 ZeroMemory(&Token, sizeof (Token));
261 Token.Type = OBO_USER;
262
263 hr = pSetup->Install(pszwComponentId, &Token,
264 0, /* IN DWORD dwSetupFlags */
265 0, /* IN DWORD dwUpgradeFromBuildNo */
266 NULL, /* IN LPCWSTR pszwAnswerFile */
267 NULL, /* IN LPCWSTR pszwAnswerSections */
268 ppComponent);
269 if (SUCCEEDED(hr))
270 {
271 /* ignore the apply failure */
272 HRESULT tmpHr = pNetCfg->Apply();
273 Assert(tmpHr == S_OK);
274 if (tmpHr != S_OK)
275 NonStandardLogFlow(("Apply failed, hr (0x%x)\n", tmpHr));
276 }
277 else
278 NonStandardLogFlow(("Install failed, hr (0x%x)\n", hr));
279
280 pSetup->Release();
281 return hr;
282}
283
284static HRESULT vboxNetCfgWinInstallInfAndComponent(IN INetCfg *pNetCfg, IN LPCWSTR pszwComponentId, IN const GUID *pguidClass,
285 IN LPCWSTR const *apInfPaths, IN UINT cInfPaths,
286 OUT INetCfgComponent **ppComponent)
287{
288 HRESULT hr = S_OK;
289 UINT cFilesProcessed = 0;
290
291 NonStandardLogFlow(("Installing %u INF files ...\n", cInfPaths));
292
293 for (; cFilesProcessed < cInfPaths; cFilesProcessed++)
294 {
295 NonStandardLogFlow(("Installing INF file \"%ws\" ...\n", apInfPaths[cFilesProcessed]));
296 hr = VBoxDrvCfgInfInstall(apInfPaths[cFilesProcessed]);
297 if (FAILED(hr))
298 {
299 NonStandardLogFlow(("VBoxNetCfgWinInfInstall failed, hr (0x%x)\n", hr));
300 break;
301 }
302 }
303
304 if (SUCCEEDED(hr))
305 {
306 hr = VBoxNetCfgWinInstallComponent(pNetCfg, pszwComponentId, pguidClass, ppComponent);
307 if (FAILED(hr))
308 NonStandardLogFlow(("VBoxNetCfgWinInstallComponent failed, hr (0x%x)\n", hr));
309 }
310
311 if (FAILED(hr))
312 {
313 NonStandardLogFlow(("Installation failed, rolling back installation set ...\n"));
314
315 do
316 {
317 HRESULT hr2 = VBoxDrvCfgInfUninstall(apInfPaths[cFilesProcessed], 0);
318 if (FAILED(hr2))
319 NonStandardLogFlow(("VBoxDrvCfgInfUninstall failed, hr (0x%x)\n", hr2));
320 /* Keep going. */
321 if (!cFilesProcessed)
322 break;
323 } while (cFilesProcessed--);
324
325 NonStandardLogFlow(("Rollback complete\n"));
326 }
327
328 return hr;
329}
330
331VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinUninstallComponent(IN INetCfg *pNetCfg, IN INetCfgComponent *pComponent)
332{
333 GUID GuidClass;
334 HRESULT hr = pComponent->GetClassGuid(&GuidClass);
335 if (FAILED(hr))
336 {
337 NonStandardLogFlow(("GetClassGuid failed, hr (0x%x)\n", hr));
338 return hr;
339 }
340
341 INetCfgClassSetup *pSetup = NULL;
342 hr = vboxNetCfgWinQueryInstaller(pNetCfg, &GuidClass, &pSetup);
343 if (FAILED(hr))
344 {
345 NonStandardLogFlow(("vboxNetCfgWinQueryInstaller failed, hr (0x%x)\n", hr));
346 return hr;
347 }
348
349 OBO_TOKEN Token;
350 ZeroMemory(&Token, sizeof(Token));
351 Token.Type = OBO_USER;
352
353 hr = pSetup->DeInstall(pComponent, &Token, NULL /* OUT LPWSTR *pmszwRefs */);
354 if (SUCCEEDED(hr))
355 {
356 hr = pNetCfg->Apply();
357 if (FAILED(hr))
358 NonStandardLogFlow(("Apply failed, hr (0x%x)\n", hr));
359 }
360 else
361 NonStandardLogFlow(("DeInstall failed, hr (0x%x)\n", hr));
362
363 if (pSetup)
364 pSetup->Release();
365 return hr;
366}
367
368typedef BOOL (*VBOXNETCFGWIN_NETCFGENUM_CALLBACK) (IN INetCfg *pNetCfg, IN INetCfgComponent *pNetCfgComponent, PVOID pContext);
369
370static HRESULT vboxNetCfgWinEnumNetCfgComponents(IN INetCfg *pNetCfg,
371 IN const GUID *pguidClass,
372 VBOXNETCFGWIN_NETCFGENUM_CALLBACK callback,
373 PVOID pContext)
374{
375 IEnumNetCfgComponent *pEnumComponent;
376 HRESULT hr = pNetCfg->EnumComponents(pguidClass, &pEnumComponent);
377 if (SUCCEEDED(hr))
378 {
379 INetCfgComponent *pNetCfgComponent;
380 hr = pEnumComponent->Reset();
381 do
382 {
383 hr = pEnumComponent->Next(1, &pNetCfgComponent, NULL);
384 if (hr == S_OK)
385 {
386// ULONG uComponentStatus;
387// hr = pNcc->GetDeviceStatus(&uComponentStatus);
388// if (SUCCEEDED(hr))
389 BOOL fResult = FALSE;
390 if (pNetCfgComponent)
391 {
392 if (pContext)
393 fResult = callback(pNetCfg, pNetCfgComponent, pContext);
394 pNetCfgComponent->Release();
395 }
396
397 if (!fResult)
398 break;
399 }
400 else
401 {
402 if (hr == S_FALSE)
403 {
404 hr = S_OK;
405 }
406 else
407 NonStandardLogFlow(("Next failed, hr (0x%x)\n", hr));
408 break;
409 }
410 } while (true);
411 pEnumComponent->Release();
412 }
413 return hr;
414}
415
416/*
417 * Forward declarations of functions used in vboxNetCfgWinRemoveAllNetDevicesOfIdCallback.
418 */
419VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinGenHostonlyConnectionName(PCWSTR DevName, WCHAR *pBuf, PULONG pcbBuf);
420VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinRenameConnection(LPWSTR pGuid, PCWSTR NewName);
421
422static BOOL vboxNetCfgWinRemoveAllNetDevicesOfIdCallback(HDEVINFO hDevInfo, PSP_DEVINFO_DATA pDev, PVOID pContext)
423{
424 SP_REMOVEDEVICE_PARAMS rmdParams;
425 memset(&rmdParams, 0, sizeof(SP_REMOVEDEVICE_PARAMS));
426 rmdParams.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);
427 rmdParams.ClassInstallHeader.InstallFunction = DIF_REMOVE;
428 rmdParams.Scope = DI_REMOVEDEVICE_GLOBAL;
429
430 if (SetupDiSetClassInstallParams(hDevInfo,pDev,
431 &rmdParams.ClassInstallHeader, sizeof(rmdParams)))
432 {
433 if (SetupDiSetSelectedDevice(hDevInfo, pDev))
434 {
435 /* Figure out NetCfgInstanceId. */
436 HKEY hKey = SetupDiOpenDevRegKey(hDevInfo,
437 pDev,
438 DICS_FLAG_GLOBAL,
439 0,
440 DIREG_DRV,
441 KEY_READ);
442 if (hKey == INVALID_HANDLE_VALUE)
443 {
444 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: SetupDiOpenDevRegKey failed with error %ld\n",
445 GetLastError()));
446 }
447 else
448 {
449 WCHAR wszCfgGuidString[50] = { L'' };
450 DWORD cbSize = sizeof(wszCfgGuidString);
451 DWORD dwValueType;
452 DWORD ret = RegQueryValueExW(hKey, L"NetCfgInstanceId", NULL,
453 &dwValueType, (LPBYTE)wszCfgGuidString, &cbSize);
454 if (ret == ERROR_SUCCESS)
455 {
456 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: Processing device ID \"%S\"\n",
457 wszCfgGuidString));
458
459 /* Figure out device name. */
460 WCHAR wszDevName[256], wszTempName[256];
461 ULONG cbName = sizeof(wszTempName);
462
463 if (SetupDiGetDeviceRegistryPropertyW(hDevInfo, pDev,
464 SPDRP_FRIENDLYNAME, /* IN DWORD Property,*/
465 NULL, /* OUT PDWORD PropertyRegDataType, OPTIONAL*/
466 (PBYTE)wszDevName, /* OUT PBYTE PropertyBuffer,*/
467 sizeof(wszDevName), /* IN DWORD PropertyBufferSize,*/
468 NULL /* OUT PDWORD RequiredSize OPTIONAL*/))
469 {
470 /*
471 * Rename the connection before removing the device. This will
472 * hopefully prevent an error when we will be attempting
473 * to rename a newly created connection (see @bugref{6740}).
474 */
475 HRESULT hr = VBoxNetCfgWinGenHostonlyConnectionName(wszDevName, wszTempName, &cbName);
476 wcscat_s(wszTempName, sizeof(wszTempName), L" removed");
477 if (SUCCEEDED(hr))
478 hr = VBoxNetCfgWinRenameConnection(wszCfgGuidString, wszTempName);
479 //NonStandardLogFlow(("VBoxNetCfgWinRenameConnection(%S,%S) => 0x%x\n", pWCfgGuidString, TempName, hr_tmp));
480 }
481 else
482 {
483 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: Failed to get friendly name for device \"%S\"\n",
484 wszCfgGuidString));
485 }
486 }
487 else
488 {
489 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: Querying instance ID failed with %d\n",
490 ret));
491 }
492
493 RegCloseKey(hKey);
494 }
495
496 if (SetupDiCallClassInstaller(DIF_REMOVE, hDevInfo, pDev))
497 {
498 SP_DEVINSTALL_PARAMS devParams;
499 memset(&devParams, 0, sizeof(SP_DEVINSTALL_PARAMS));
500 devParams.cbSize = sizeof(devParams);
501
502 if (SetupDiGetDeviceInstallParams(hDevInfo, pDev, &devParams))
503 {
504 if ( (devParams.Flags & DI_NEEDRESTART)
505 || (devParams.Flags & DI_NEEDREBOOT))
506 {
507 NonStandardLog(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: A reboot is required\n"));
508 }
509 }
510 else
511 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: SetupDiGetDeviceInstallParams failed with %ld\n",
512 GetLastError()));
513 }
514 else
515 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: SetupDiCallClassInstaller failed with %ld\n",
516 GetLastError()));
517 }
518 else
519 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: SetupDiSetSelectedDevice failed with %ld\n",
520 GetLastError()));
521 }
522 else
523 NonStandardLogFlow(("vboxNetCfgWinRemoveAllNetDevicesOfIdCallback: SetupDiSetClassInstallParams failed with %ld\n",
524 GetLastError()));
525
526 /* Continue enumeration. */
527 return TRUE;
528}
529
530typedef struct VBOXNECTFGWINPROPCHANGE
531{
532 VBOXNECTFGWINPROPCHANGE_TYPE enmPcType;
533 HRESULT hr;
534} VBOXNECTFGWINPROPCHANGE ,*PVBOXNECTFGWINPROPCHANGE;
535
536static BOOL vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback(HDEVINFO hDevInfo, PSP_DEVINFO_DATA pDev, PVOID pContext)
537{
538 PVBOXNECTFGWINPROPCHANGE pPc = (PVBOXNECTFGWINPROPCHANGE)pContext;
539
540 SP_PROPCHANGE_PARAMS PcParams;
541 memset (&PcParams, 0, sizeof (PcParams));
542 PcParams.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);
543 PcParams.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;
544 PcParams.Scope = DICS_FLAG_GLOBAL;
545
546 switch(pPc->enmPcType)
547 {
548 case VBOXNECTFGWINPROPCHANGE_TYPE_DISABLE:
549 PcParams.StateChange = DICS_DISABLE;
550 NonStandardLogFlow(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback: Change type (DICS_DISABLE): %d\n", pPc->enmPcType));
551 break;
552 case VBOXNECTFGWINPROPCHANGE_TYPE_ENABLE:
553 PcParams.StateChange = DICS_ENABLE;
554 NonStandardLogFlow(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback: Change type (DICS_ENABLE): %d\n", pPc->enmPcType));
555 break;
556 default:
557 NonStandardLogFlow(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback: Unexpected prop change type: %d\n", pPc->enmPcType));
558 pPc->hr = E_INVALIDARG;
559 return FALSE;
560 }
561
562 if (SetupDiSetClassInstallParams(hDevInfo, pDev, &PcParams.ClassInstallHeader, sizeof(PcParams)))
563 {
564 if (SetupDiSetSelectedDevice(hDevInfo, pDev))
565 {
566 if (SetupDiCallClassInstaller(DIF_PROPERTYCHANGE, hDevInfo, pDev))
567 {
568 SP_DEVINSTALL_PARAMS devParams;
569 devParams.cbSize = sizeof(devParams);
570 if (SetupDiGetDeviceInstallParams(hDevInfo,pDev,&devParams))
571 {
572 if ( (devParams.Flags & DI_NEEDRESTART)
573 || (devParams.Flags & DI_NEEDREBOOT))
574 {
575 NonStandardLog(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback: A reboot is required\n"));
576 }
577 }
578 else
579 NonStandardLogFlow(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback: SetupDiGetDeviceInstallParams failed with %ld\n",
580 GetLastError()));
581 }
582 else
583 NonStandardLogFlow(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback: SetupDiCallClassInstaller failed with %ld\n",
584 GetLastError()));
585 }
586 else
587 NonStandardLogFlow(("SetupDiSetSelectedDevice failed with %ld\n", GetLastError()));
588 }
589 else
590 NonStandardLogFlow(("SetupDiSetClassInstallParams failed with %ld\n", GetLastError()));
591
592 /* Continue enumeration. */
593 return TRUE;
594}
595
596typedef BOOL (*VBOXNETCFGWIN_NETENUM_CALLBACK) (HDEVINFO hDevInfo, PSP_DEVINFO_DATA pDev, PVOID pContext);
597VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinEnumNetDevices(LPCWSTR pwszPnPId,
598 VBOXNETCFGWIN_NETENUM_CALLBACK callback, PVOID pContext)
599{
600 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: Searching for: %S\n", pwszPnPId));
601
602 HRESULT hr;
603 HDEVINFO hDevInfo = SetupDiGetClassDevsExW(&GUID_DEVCLASS_NET,
604 NULL, /* IN PCTSTR Enumerator, OPTIONAL */
605 NULL, /* IN HWND hwndParent, OPTIONAL */
606 DIGCF_PRESENT, /* IN DWORD Flags,*/
607 NULL, /* IN HDEVINFO DeviceInfoSet, OPTIONAL */
608 NULL, /* IN PCTSTR MachineName, OPTIONAL */
609 NULL /* IN PVOID Reserved */);
610 if (hDevInfo != INVALID_HANDLE_VALUE)
611 {
612 DWORD winEr;
613
614 DWORD dwDevId = 0;
615 size_t cPnPId = wcslen(pwszPnPId);
616
617 PBYTE pBuffer = NULL;
618
619 for (;;)
620 {
621 SP_DEVINFO_DATA Dev;
622 memset(&Dev, 0, sizeof(SP_DEVINFO_DATA));
623 Dev.cbSize = sizeof(SP_DEVINFO_DATA);
624
625 if (!SetupDiEnumDeviceInfo(hDevInfo, dwDevId, &Dev))
626 {
627 winEr = GetLastError();
628 if (winEr == ERROR_NO_MORE_ITEMS)
629 winEr = ERROR_SUCCESS;
630 break;
631 }
632
633 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: Enumerating device %ld ... \n", dwDevId));
634 dwDevId++;
635
636 if (pBuffer)
637 free(pBuffer);
638 pBuffer = NULL;
639 DWORD cbBuffer = 0;
640 DWORD cbRequired = 0;
641
642 if (!SetupDiGetDeviceRegistryPropertyW(hDevInfo, &Dev,
643 SPDRP_HARDWAREID, /* IN DWORD Property */
644 NULL, /* OUT PDWORD PropertyRegDataType OPTIONAL */
645 pBuffer, /* OUT PBYTE PropertyBuffer */
646 cbBuffer, /* IN DWORD PropertyBufferSize */
647 &cbRequired /* OUT PDWORD RequiredSize OPTIONAL */))
648 {
649 winEr = GetLastError();
650 if (winEr != ERROR_INSUFFICIENT_BUFFER)
651 {
652 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: SetupDiGetDeviceRegistryPropertyW (1) failed with %ld\n", winEr));
653 break;
654 }
655
656 pBuffer = (PBYTE)malloc(cbRequired);
657 if (!pBuffer)
658 {
659 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: Out of memory allocating %ld bytes\n",
660 cbRequired));
661 winEr = ERROR_OUTOFMEMORY;
662 break;
663 }
664
665 cbBuffer = cbRequired;
666
667 if (!SetupDiGetDeviceRegistryPropertyW(hDevInfo,&Dev,
668 SPDRP_HARDWAREID, /* IN DWORD Property */
669 NULL, /* OUT PDWORD PropertyRegDataType, OPTIONAL */
670 pBuffer, /* OUT PBYTE PropertyBuffer */
671 cbBuffer, /* IN DWORD PropertyBufferSize */
672 &cbRequired /* OUT PDWORD RequiredSize OPTIONAL */))
673 {
674 winEr = GetLastError();
675 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: SetupDiGetDeviceRegistryPropertyW (2) failed with %ld\n",
676 winEr));
677 break;
678 }
679 }
680
681 PWSTR pCurId = (PWSTR)pBuffer;
682 size_t cCurId = wcslen(pCurId);
683
684 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: Device %ld: %S\n", dwDevId, pCurId));
685
686 if (cCurId >= cPnPId)
687 {
688 NonStandardLogFlow(("!wcsnicmp(pCurId = (%S), pwszPnPId = (%S), cPnPId = (%d))", pCurId, pwszPnPId, cPnPId));
689
690 pCurId += cCurId - cPnPId;
691 if (!wcsnicmp(pCurId, pwszPnPId, cPnPId))
692 {
693 if (!callback(hDevInfo, &Dev, pContext))
694 break;
695 }
696 }
697 }
698
699 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: Found %ld devices total\n", dwDevId));
700
701 if (pBuffer)
702 free(pBuffer);
703
704 hr = HRESULT_FROM_WIN32(winEr);
705
706 SetupDiDestroyDeviceInfoList(hDevInfo);
707 }
708 else
709 {
710 DWORD winEr = GetLastError();
711 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: SetupDiGetClassDevsExW failed with %ld\n", winEr));
712 hr = HRESULT_FROM_WIN32(winEr);
713 }
714
715 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices: Ended with hr (0x%x)\n", hr));
716 return hr;
717}
718
719VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinRemoveAllNetDevicesOfId(IN LPCWSTR lpszPnPId)
720{
721 return VBoxNetCfgWinEnumNetDevices(lpszPnPId, vboxNetCfgWinRemoveAllNetDevicesOfIdCallback, NULL);
722}
723
724VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinPropChangeAllNetDevicesOfId(IN LPCWSTR lpszPnPId, VBOXNECTFGWINPROPCHANGE_TYPE enmPcType)
725{
726 VBOXNECTFGWINPROPCHANGE Pc;
727 Pc.enmPcType = enmPcType;
728 Pc.hr = S_OK;
729 NonStandardLogFlow(("Calling VBoxNetCfgWinEnumNetDevices with lpszPnPId =(%S) and vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback", lpszPnPId));
730
731 HRESULT hr = VBoxNetCfgWinEnumNetDevices(lpszPnPId, vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback, &Pc);
732 if (!SUCCEEDED(hr))
733 {
734 NonStandardLogFlow(("VBoxNetCfgWinEnumNetDevices failed 0x%x\n", hr));
735 return hr;
736 }
737
738 if (!SUCCEEDED(Pc.hr))
739 {
740 NonStandardLogFlow(("vboxNetCfgWinPropChangeAllNetDevicesOfIdCallback failed 0x%x\n", Pc.hr));
741 return Pc.hr;
742 }
743
744 return S_OK;
745}
746
747/*
748 * logging
749 */
750static VOID DoLogging(LPCSTR szString, ...)
751{
752 LOG_ROUTINE pfnRoutine = (LOG_ROUTINE)(*((void * volatile *)&g_Logger));
753 if (pfnRoutine)
754 {
755 char szBuffer[4096] = {0};
756 va_list va;
757 va_start(va, szString);
758 _vsnprintf(szBuffer, RT_ELEMENTS(szBuffer), szString, va);
759 va_end(va);
760
761 pfnRoutine(szBuffer);
762 }
763}
764
765VBOXNETCFGWIN_DECL(VOID) VBoxNetCfgWinSetLogging(IN LOG_ROUTINE pfnLog)
766{
767 *((void * volatile *)&g_Logger) = pfnLog;
768}
769
770/*
771 * IP configuration API
772 */
773/* network settings config */
774/**
775 * Strong referencing operators. Used as a second argument to ComPtr<>/ComObjPtr<>.
776 */
777template <class C>
778class ComStrongRef
779{
780protected:
781
782 static void addref (C *p) { p->AddRef(); }
783 static void release (C *p) { p->Release(); }
784};
785
786
787/**
788 * Base template for smart COM pointers. Not intended to be used directly.
789 */
790template <class C, template <class> class RefOps = ComStrongRef>
791class ComPtrBase : protected RefOps <C>
792{
793public:
794
795 /* special template to disable AddRef()/Release() */
796 template <class I>
797 class NoAddRefRelease : public I
798 {
799 private:
800#if !defined (VBOX_WITH_XPCOM)
801 STDMETHOD_(ULONG, AddRef)() = 0;
802 STDMETHOD_(ULONG, Release)() = 0;
803#else /* !defined (VBOX_WITH_XPCOM) */
804 NS_IMETHOD_(nsrefcnt) AddRef(void) = 0;
805 NS_IMETHOD_(nsrefcnt) Release(void) = 0;
806#endif /* !defined (VBOX_WITH_XPCOM) */
807 };
808
809protected:
810
811 ComPtrBase () : p (NULL) {}
812 ComPtrBase (const ComPtrBase &that) : p (that.p) { addref(); }
813 ComPtrBase (C *that_p) : p (that_p) { addref(); }
814
815 ~ComPtrBase() { release(); }
816
817 ComPtrBase &operator= (const ComPtrBase &that)
818 {
819 safe_assign (that.p);
820 return *this;
821 }
822
823 ComPtrBase &operator= (C *that_p)
824 {
825 safe_assign (that_p);
826 return *this;
827 }
828
829public:
830
831 void setNull()
832 {
833 release();
834 p = NULL;
835 }
836
837 bool isNull() const
838 {
839 return (p == NULL);
840 }
841
842 bool operator! () const { return isNull(); }
843
844 bool operator< (C* that_p) const { return p < that_p; }
845 bool operator== (C* that_p) const { return p == that_p; }
846
847 template <class I>
848 bool equalsTo (I *aThat) const
849 {
850 return ComPtrEquals (p, aThat);
851 }
852
853 template <class OC>
854 bool equalsTo (const ComPtrBase <OC> &oc) const
855 {
856 return equalsTo ((OC *) oc);
857 }
858
859 /** Intended to pass instances as in parameters to interface methods */
860 operator C* () const { return p; }
861
862 /**
863 * Dereferences the instance (redirects the -> operator to the managed
864 * pointer).
865 */
866 NoAddRefRelease <C> *operator-> () const
867 {
868 AssertMsg (p, ("Managed pointer must not be null\n"));
869 return (NoAddRefRelease <C> *) p;
870 }
871
872 template <class I>
873 HRESULT queryInterfaceTo (I **pp) const
874 {
875 if (pp)
876 {
877 if (p)
878 {
879 return p->QueryInterface (COM_IIDOF (I), (void **) pp);
880 }
881 else
882 {
883 *pp = NULL;
884 return S_OK;
885 }
886 }
887
888 return E_INVALIDARG;
889 }
890
891 /** Intended to pass instances as out parameters to interface methods */
892 C **asOutParam()
893 {
894 setNull();
895 return &p;
896 }
897
898private:
899
900 void addref()
901 {
902 if (p)
903 RefOps <C>::addref (p);
904 }
905
906 void release()
907 {
908 if (p)
909 RefOps <C>::release (p);
910 }
911
912 void safe_assign (C *that_p)
913 {
914 /* be aware of self-assignment */
915 if (that_p)
916 RefOps <C>::addref (that_p);
917 release();
918 p = that_p;
919 }
920
921 C *p;
922};
923
924/**
925 * Smart COM pointer wrapper that automatically manages refcounting of
926 * interface pointers.
927 *
928 * @param I COM interface class
929 */
930template <class I, template <class> class RefOps = ComStrongRef>
931class ComPtr : public ComPtrBase <I, RefOps>
932{
933 typedef ComPtrBase <I, RefOps> Base;
934
935public:
936
937 ComPtr () : Base() {}
938 ComPtr (const ComPtr &that) : Base(that) {}
939 ComPtr &operator= (const ComPtr &that)
940 {
941 Base::operator= (that);
942 return *this;
943 }
944
945 template <class OI>
946 ComPtr (OI *that_p) : Base () { operator= (that_p); }
947
948 /* specialization for I */
949 ComPtr (I *that_p) : Base (that_p) {}
950
951 template <class OC>
952 ComPtr (const ComPtr <OC, RefOps> &oc) : Base () { operator= ((OC *) oc); }
953
954 template <class OI>
955 ComPtr &operator= (OI *that_p)
956 {
957 if (that_p)
958 that_p->QueryInterface (COM_IIDOF (I), (void **) Base::asOutParam());
959 else
960 Base::setNull();
961 return *this;
962 }
963
964 /* specialization for I */
965 ComPtr &operator=(I *that_p)
966 {
967 Base::operator= (that_p);
968 return *this;
969 }
970
971 template <class OC>
972 ComPtr &operator= (const ComPtr <OC, RefOps> &oc)
973 {
974 return operator= ((OC *) oc);
975 }
976};
977
978static HRESULT netIfWinFindAdapterClassById(IWbemServices * pSvc, const GUID * pGuid, IWbemClassObject **pAdapterConfig)
979{
980 HRESULT hr;
981 WCHAR wszQuery[256];
982 WCHAR wszGuid[50];
983
984 int length = StringFromGUID2(*pGuid, wszGuid, RT_ELEMENTS(wszGuid));
985 if (length)
986 {
987 swprintf(wszQuery, L"SELECT * FROM Win32_NetworkAdapterConfiguration WHERE SettingID = \"%s\"", wszGuid);
988 IEnumWbemClassObject* pEnumerator = NULL;
989 hr = pSvc->ExecQuery(bstr_t("WQL"), bstr_t(wszQuery), WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
990 NULL, &pEnumerator);
991 if (SUCCEEDED(hr))
992 {
993 if (pEnumerator)
994 {
995 IWbemClassObject *pclsObj;
996 ULONG uReturn = 0;
997 hr = pEnumerator->Next(WBEM_INFINITE, 1, &pclsObj, &uReturn);
998 NonStandardLogFlow(("netIfWinFindAdapterClassById: IEnumWbemClassObject::Next -> hr=0x%x pclsObj=%p uReturn=%u 42=%u\n",
999 hr, (void *)pclsObj, uReturn, 42));
1000 if (SUCCEEDED(hr))
1001 {
1002 if (uReturn && pclsObj != NULL)
1003 {
1004 *pAdapterConfig = pclsObj;
1005 pEnumerator->Release();
1006 NonStandardLogFlow(("netIfWinFindAdapterClassById: S_OK and %p\n", *pAdapterConfig));
1007 return S_OK;
1008 }
1009
1010 hr = E_FAIL;
1011 }
1012
1013 pEnumerator->Release();
1014 }
1015 else
1016 {
1017 NonStandardLogFlow(("ExecQuery returned no enumerator\n"));
1018 hr = E_FAIL;
1019 }
1020 }
1021 else
1022 NonStandardLogFlow(("ExecQuery failed (0x%x)\n", hr));
1023 }
1024 else
1025 {
1026 DWORD winEr = GetLastError();
1027 hr = HRESULT_FROM_WIN32( winEr );
1028 if (SUCCEEDED(hr))
1029 hr = E_FAIL;
1030 NonStandardLogFlow(("StringFromGUID2 failed winEr=%u, hr=0x%x\n", winEr, hr));
1031 }
1032
1033 NonStandardLogFlow(("netIfWinFindAdapterClassById: 0x%x and %p\n", hr, *pAdapterConfig));
1034 return hr;
1035}
1036
1037static HRESULT netIfWinIsHostOnly(IWbemClassObject * pAdapterConfig, BOOL * pbIsHostOnly)
1038{
1039 VARIANT vtServiceName;
1040 BOOL bIsHostOnly = FALSE;
1041 VariantInit(&vtServiceName);
1042
1043 HRESULT hr = pAdapterConfig->Get(L"ServiceName", 0 /*lFlags*/, &vtServiceName, NULL /*pvtType*/, NULL /*plFlavor*/);
1044 if (SUCCEEDED(hr))
1045 {
1046 *pbIsHostOnly = bstr_t(vtServiceName.bstrVal) == bstr_t("VBoxNetAdp");
1047
1048 VariantClear(&vtServiceName);
1049 }
1050
1051 return hr;
1052}
1053
1054static HRESULT netIfWinGetIpSettings(IWbemClassObject * pAdapterConfig, ULONG *pIpv4, ULONG *pMaskv4)
1055{
1056 VARIANT vtIp;
1057 HRESULT hr;
1058 VariantInit(&vtIp);
1059
1060 *pIpv4 = 0;
1061 *pMaskv4 = 0;
1062
1063 hr = pAdapterConfig->Get(L"IPAddress", 0, &vtIp, 0, 0);
1064 if (SUCCEEDED(hr))
1065 {
1066 if (vtIp.vt == (VT_ARRAY | VT_BSTR))
1067 {
1068 VARIANT vtMask;
1069 VariantInit(&vtMask);
1070 hr = pAdapterConfig->Get(L"IPSubnet", 0, &vtMask, 0, 0);
1071 if (SUCCEEDED(hr))
1072 {
1073 if (vtMask.vt == (VT_ARRAY | VT_BSTR))
1074 {
1075 SAFEARRAY * pIpArray = vtIp.parray;
1076 SAFEARRAY * pMaskArray = vtMask.parray;
1077 if (pIpArray && pMaskArray)
1078 {
1079 BSTR pCurIp;
1080 BSTR pCurMask;
1081 for (LONG i = 0;
1082 SafeArrayGetElement(pIpArray, &i, (PVOID)&pCurIp) == S_OK
1083 && SafeArrayGetElement(pMaskArray, &i, (PVOID)&pCurMask) == S_OK;
1084 i++)
1085 {
1086 bstr_t ip(pCurIp);
1087
1088 ULONG Ipv4 = inet_addr((char*)(ip));
1089 if (Ipv4 != INADDR_NONE)
1090 {
1091 *pIpv4 = Ipv4;
1092 bstr_t mask(pCurMask);
1093 *pMaskv4 = inet_addr((char*)(mask));
1094 break;
1095 }
1096 }
1097 }
1098 }
1099 else
1100 {
1101 *pIpv4 = 0;
1102 *pMaskv4 = 0;
1103 }
1104
1105 VariantClear(&vtMask);
1106 }
1107 }
1108 else
1109 {
1110 *pIpv4 = 0;
1111 *pMaskv4 = 0;
1112 }
1113
1114 VariantClear(&vtIp);
1115 }
1116
1117 return hr;
1118}
1119
1120
1121static HRESULT netIfWinHasIpSettings(IWbemClassObject * pAdapterConfig, SAFEARRAY * pCheckIp, SAFEARRAY * pCheckMask, bool *pFound)
1122{
1123 VARIANT vtIp;
1124 HRESULT hr;
1125 VariantInit(&vtIp);
1126
1127 *pFound = false;
1128
1129 hr = pAdapterConfig->Get(L"IPAddress", 0, &vtIp, 0, 0);
1130 if (SUCCEEDED(hr))
1131 {
1132 VARIANT vtMask;
1133 VariantInit(&vtMask);
1134 hr = pAdapterConfig->Get(L"IPSubnet", 0, &vtMask, 0, 0);
1135 if (SUCCEEDED(hr))
1136 {
1137 SAFEARRAY * pIpArray = vtIp.parray;
1138 SAFEARRAY * pMaskArray = vtMask.parray;
1139 if (pIpArray && pMaskArray)
1140 {
1141 BSTR pIp, pMask;
1142 for (LONG k = 0;
1143 SafeArrayGetElement(pCheckIp, &k, (PVOID)&pIp) == S_OK
1144 && SafeArrayGetElement(pCheckMask, &k, (PVOID)&pMask) == S_OK;
1145 k++)
1146 {
1147 BSTR pCurIp;
1148 BSTR pCurMask;
1149 for (LONG i = 0;
1150 SafeArrayGetElement(pIpArray, &i, (PVOID)&pCurIp) == S_OK
1151 && SafeArrayGetElement(pMaskArray, &i, (PVOID)&pCurMask) == S_OK;
1152 i++)
1153 {
1154 if (!wcsicmp(pCurIp, pIp))
1155 {
1156 if (!wcsicmp(pCurMask, pMask))
1157 *pFound = true;
1158 break;
1159 }
1160 }
1161 }
1162 }
1163
1164
1165 VariantClear(&vtMask);
1166 }
1167
1168 VariantClear(&vtIp);
1169 }
1170
1171 return hr;
1172}
1173
1174static HRESULT netIfWinWaitIpSettings(IWbemServices *pSvc, const GUID * pGuid, SAFEARRAY * pCheckIp, SAFEARRAY * pCheckMask, ULONG sec2Wait, bool *pFound)
1175{
1176 /* on Vista we need to wait for the address to get applied */
1177 /* wait for the address to appear in the list */
1178 HRESULT hr = S_OK;
1179 ULONG i;
1180 *pFound = false;
1181 ComPtr <IWbemClassObject> pAdapterConfig;
1182 for (i = 0;
1183 (hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam())) == S_OK
1184 && (hr = netIfWinHasIpSettings(pAdapterConfig, pCheckIp, pCheckMask, pFound)) == S_OK
1185 && !(*pFound)
1186 && i < sec2Wait/6;
1187 i++)
1188 {
1189 Sleep(6000);
1190 }
1191
1192 return hr;
1193}
1194
1195static HRESULT netIfWinCreateIWbemServices(IWbemServices ** ppSvc)
1196{
1197 IWbemLocator *pLoc = NULL;
1198 HRESULT hr = CoCreateInstance(CLSID_WbemLocator, 0, CLSCTX_INPROC_SERVER, IID_IWbemLocator, (LPVOID *) &pLoc);
1199 if (SUCCEEDED(hr))
1200 {
1201 IWbemServices *pSvc = NULL;
1202 hr = pLoc->ConnectServer(bstr_t(L"ROOT\\CIMV2"), /* [in] const BSTR strNetworkResource */
1203 NULL, /* [in] const BSTR strUser */
1204 NULL, /* [in] const BSTR strPassword */
1205 0, /* [in] const BSTR strLocale */
1206 NULL, /* [in] LONG lSecurityFlags */
1207 0, /* [in] const BSTR strAuthority */
1208 0, /* [in] IWbemContext* pCtx */
1209 &pSvc /* [out] IWbemServices** ppNamespace */);
1210 if (SUCCEEDED(hr))
1211 {
1212 hr = CoSetProxyBlanket(pSvc, /* IUnknown * pProxy */
1213 RPC_C_AUTHN_WINNT, /* DWORD dwAuthnSvc */
1214 RPC_C_AUTHZ_NONE, /* DWORD dwAuthzSvc */
1215 NULL, /* WCHAR * pServerPrincName */
1216 RPC_C_AUTHN_LEVEL_CALL, /* DWORD dwAuthnLevel */
1217 RPC_C_IMP_LEVEL_IMPERSONATE, /* DWORD dwImpLevel */
1218 NULL, /* RPC_AUTH_IDENTITY_HANDLE pAuthInfo */
1219 EOAC_NONE /* DWORD dwCapabilities */
1220 );
1221 if (SUCCEEDED(hr))
1222 {
1223 *ppSvc = pSvc;
1224 /* do not need it any more */
1225 pLoc->Release();
1226 return hr;
1227 }
1228 else
1229 NonStandardLogFlow(("CoSetProxyBlanket failed, hr (0x%x)\n", hr));
1230
1231 pSvc->Release();
1232 }
1233 else
1234 NonStandardLogFlow(("ConnectServer failed, hr (0x%x)\n", hr));
1235 pLoc->Release();
1236 }
1237 else
1238 NonStandardLogFlow(("CoCreateInstance failed, hr (0x%x)\n", hr));
1239 return hr;
1240}
1241
1242static HRESULT netIfWinAdapterConfigPath(IWbemClassObject *pObj, BSTR * pStr)
1243{
1244 VARIANT index;
1245 HRESULT hr = pObj->Get(L"Index", 0, &index, 0, 0);
1246 if (SUCCEEDED(hr))
1247 {
1248 WCHAR strIndex[8];
1249 swprintf(strIndex, L"%u", index.uintVal);
1250 *pStr = (bstr_t(L"Win32_NetworkAdapterConfiguration.Index='") + strIndex + "'").copy();
1251 }
1252 else
1253 NonStandardLogFlow(("Get failed, hr (0x%x)\n", hr));
1254 return hr;
1255}
1256
1257static HRESULT netIfExecMethod(IWbemServices * pSvc, IWbemClassObject *pClass, BSTR ObjPath,
1258 BSTR MethodName, LPWSTR *pArgNames, LPVARIANT *pArgs, UINT cArgs,
1259 IWbemClassObject** ppOutParams
1260 )
1261{
1262 HRESULT hr = S_OK;
1263 ComPtr<IWbemClassObject> pInParamsDefinition;
1264 ComPtr<IWbemClassObject> pClassInstance;
1265
1266 if (cArgs)
1267 {
1268 hr = pClass->GetMethod(MethodName, 0, pInParamsDefinition.asOutParam(), NULL);
1269 if (SUCCEEDED(hr))
1270 {
1271 hr = pInParamsDefinition->SpawnInstance(0, pClassInstance.asOutParam());
1272 if (SUCCEEDED(hr))
1273 {
1274 for (UINT i = 0; i < cArgs; i++)
1275 {
1276 hr = pClassInstance->Put(pArgNames[i], 0,
1277 pArgs[i], 0);
1278 if (FAILED(hr))
1279 break;
1280 }
1281 }
1282 }
1283 }
1284
1285 if (SUCCEEDED(hr))
1286 {
1287 IWbemClassObject* pOutParams = NULL;
1288 hr = pSvc->ExecMethod(ObjPath, MethodName, 0, NULL, pClassInstance, &pOutParams, NULL);
1289 if (SUCCEEDED(hr))
1290 {
1291 *ppOutParams = pOutParams;
1292 }
1293 }
1294
1295 return hr;
1296}
1297
1298static HRESULT netIfWinCreateIpArray(SAFEARRAY **ppArray, in_addr* aIp, UINT cIp)
1299{
1300 HRESULT hr;
1301 SAFEARRAY * pIpArray = SafeArrayCreateVector(VT_BSTR, 0, cIp);
1302 if (pIpArray)
1303 {
1304 for (UINT i = 0; i < cIp; i++)
1305 {
1306 char* addr = inet_ntoa(aIp[i]);
1307 BSTR val = bstr_t(addr).copy();
1308 long aIndex[1];
1309 aIndex[0] = i;
1310 hr = SafeArrayPutElement(pIpArray, aIndex, val);
1311 if (FAILED(hr))
1312 {
1313 SysFreeString(val);
1314 SafeArrayDestroy(pIpArray);
1315 break;
1316 }
1317 }
1318
1319 if (SUCCEEDED(hr))
1320 {
1321 *ppArray = pIpArray;
1322 }
1323 }
1324 else
1325 hr = HRESULT_FROM_WIN32(GetLastError());
1326
1327 return hr;
1328}
1329
1330static HRESULT netIfWinCreateIpArrayV4V6(SAFEARRAY **ppArray, BSTR Ip)
1331{
1332 HRESULT hr;
1333 SAFEARRAY *pIpArray = SafeArrayCreateVector(VT_BSTR, 0, 1);
1334 if (pIpArray)
1335 {
1336 BSTR val = bstr_t(Ip, false).copy();
1337 long aIndex[1];
1338 aIndex[0] = 0;
1339 hr = SafeArrayPutElement(pIpArray, aIndex, val);
1340 if (FAILED(hr))
1341 {
1342 SysFreeString(val);
1343 SafeArrayDestroy(pIpArray);
1344 }
1345
1346 if (SUCCEEDED(hr))
1347 {
1348 *ppArray = pIpArray;
1349 }
1350 }
1351 else
1352 hr = HRESULT_FROM_WIN32(GetLastError());
1353
1354 return hr;
1355}
1356
1357
1358static HRESULT netIfWinCreateIpArrayVariantV4(VARIANT * pIpAddresses, in_addr* aIp, UINT cIp)
1359{
1360 HRESULT hr;
1361 VariantInit(pIpAddresses);
1362 pIpAddresses->vt = VT_ARRAY | VT_BSTR;
1363 SAFEARRAY *pIpArray;
1364 hr = netIfWinCreateIpArray(&pIpArray, aIp, cIp);
1365 if (SUCCEEDED(hr))
1366 {
1367 pIpAddresses->parray = pIpArray;
1368 }
1369 return hr;
1370}
1371
1372static HRESULT netIfWinCreateIpArrayVariantV4V6(VARIANT * pIpAddresses, BSTR Ip)
1373{
1374 HRESULT hr;
1375 VariantInit(pIpAddresses);
1376 pIpAddresses->vt = VT_ARRAY | VT_BSTR;
1377 SAFEARRAY *pIpArray;
1378 hr = netIfWinCreateIpArrayV4V6(&pIpArray, Ip);
1379 if (SUCCEEDED(hr))
1380 {
1381 pIpAddresses->parray = pIpArray;
1382 }
1383 return hr;
1384}
1385
1386static HRESULT netIfWinEnableStatic(IWbemServices * pSvc, const GUID * pGuid, BSTR ObjPath, VARIANT * pIp, VARIANT * pMask)
1387{
1388 ComPtr<IWbemClassObject> pClass;
1389 BSTR ClassName = SysAllocString(L"Win32_NetworkAdapterConfiguration");
1390 HRESULT hr;
1391 if (ClassName)
1392 {
1393 hr = pSvc->GetObject(ClassName, 0, NULL, pClass.asOutParam(), NULL);
1394 if (SUCCEEDED(hr))
1395 {
1396 LPWSTR argNames[] = {L"IPAddress", L"SubnetMask"};
1397 LPVARIANT args[] = {pIp, pMask};
1398 ComPtr<IWbemClassObject> pOutParams;
1399
1400 hr = netIfExecMethod(pSvc, pClass, ObjPath, bstr_t(L"EnableStatic"), argNames, args, 2, pOutParams.asOutParam());
1401 if (SUCCEEDED(hr))
1402 {
1403 VARIANT varReturnValue;
1404 hr = pOutParams->Get(bstr_t(L"ReturnValue"), 0,
1405 &varReturnValue, NULL, 0);
1406 Assert(SUCCEEDED(hr));
1407 if (SUCCEEDED(hr))
1408 {
1409// Assert(varReturnValue.vt == VT_UINT);
1410 int winEr = varReturnValue.uintVal;
1411 switch (winEr)
1412 {
1413 case 0:
1414 {
1415 hr = S_OK;
1416// bool bFound;
1417// HRESULT tmpHr = netIfWinWaitIpSettings(pSvc, pGuid, pIp->parray, pMask->parray, 180, &bFound);
1418 }
1419 break;
1420 default:
1421 hr = HRESULT_FROM_WIN32( winEr );
1422 break;
1423 }
1424 }
1425 }
1426 }
1427 SysFreeString(ClassName);
1428 }
1429 else
1430 hr = HRESULT_FROM_WIN32(GetLastError());
1431
1432 return hr;
1433}
1434
1435
1436static HRESULT netIfWinEnableStaticV4(IWbemServices * pSvc, const GUID * pGuid, BSTR ObjPath, in_addr* aIp, in_addr * aMask, UINT cIp)
1437{
1438 VARIANT ipAddresses;
1439 HRESULT hr = netIfWinCreateIpArrayVariantV4(&ipAddresses, aIp, cIp);
1440 if (SUCCEEDED(hr))
1441 {
1442 VARIANT ipMasks;
1443 hr = netIfWinCreateIpArrayVariantV4(&ipMasks, aMask, cIp);
1444 if (SUCCEEDED(hr))
1445 {
1446 hr = netIfWinEnableStatic(pSvc, pGuid, ObjPath, &ipAddresses, &ipMasks);
1447 VariantClear(&ipMasks);
1448 }
1449 VariantClear(&ipAddresses);
1450 }
1451 return hr;
1452}
1453
1454static HRESULT netIfWinEnableStaticV4V6(IWbemServices * pSvc, const GUID * pGuid, BSTR ObjPath, BSTR Ip, BSTR Mask)
1455{
1456 VARIANT ipAddresses;
1457 HRESULT hr = netIfWinCreateIpArrayVariantV4V6(&ipAddresses, Ip);
1458 if (SUCCEEDED(hr))
1459 {
1460 VARIANT ipMasks;
1461 hr = netIfWinCreateIpArrayVariantV4V6(&ipMasks, Mask);
1462 if (SUCCEEDED(hr))
1463 {
1464 hr = netIfWinEnableStatic(pSvc, pGuid, ObjPath, &ipAddresses, &ipMasks);
1465 VariantClear(&ipMasks);
1466 }
1467 VariantClear(&ipAddresses);
1468 }
1469 return hr;
1470}
1471
1472/* win API allows to set gw metrics as well, we are not setting them */
1473static HRESULT netIfWinSetGateways(IWbemServices * pSvc, BSTR ObjPath, VARIANT * pGw)
1474{
1475 ComPtr<IWbemClassObject> pClass;
1476 BSTR ClassName = SysAllocString(L"Win32_NetworkAdapterConfiguration");
1477 HRESULT hr;
1478 if (ClassName)
1479 {
1480 hr = pSvc->GetObject(ClassName, 0, NULL, pClass.asOutParam(), NULL);
1481 if (SUCCEEDED(hr))
1482 {
1483 LPWSTR argNames[] = {L"DefaultIPGateway"};
1484 LPVARIANT args[] = {pGw};
1485 ComPtr<IWbemClassObject> pOutParams;
1486
1487 hr = netIfExecMethod(pSvc, pClass, ObjPath, bstr_t(L"SetGateways"), argNames, args, 1, pOutParams.asOutParam());
1488 if (SUCCEEDED(hr))
1489 {
1490 VARIANT varReturnValue;
1491 hr = pOutParams->Get(bstr_t(L"ReturnValue"), 0, &varReturnValue, NULL, 0);
1492 Assert(SUCCEEDED(hr));
1493 if (SUCCEEDED(hr))
1494 {
1495// Assert(varReturnValue.vt == VT_UINT);
1496 int winEr = varReturnValue.uintVal;
1497 switch (winEr)
1498 {
1499 case 0:
1500 hr = S_OK;
1501 break;
1502 default:
1503 hr = HRESULT_FROM_WIN32( winEr );
1504 break;
1505 }
1506 }
1507 }
1508 }
1509 SysFreeString(ClassName);
1510 }
1511 else
1512 hr = HRESULT_FROM_WIN32(GetLastError());
1513
1514 return hr;
1515}
1516
1517/* win API allows to set gw metrics as well, we are not setting them */
1518static HRESULT netIfWinSetGatewaysV4(IWbemServices * pSvc, BSTR ObjPath, in_addr* aGw, UINT cGw)
1519{
1520 VARIANT gwais;
1521 HRESULT hr = netIfWinCreateIpArrayVariantV4(&gwais, aGw, cGw);
1522 if (SUCCEEDED(hr))
1523 {
1524 netIfWinSetGateways(pSvc, ObjPath, &gwais);
1525 VariantClear(&gwais);
1526 }
1527 return hr;
1528}
1529
1530/* win API allows to set gw metrics as well, we are not setting them */
1531static HRESULT netIfWinSetGatewaysV4V6(IWbemServices * pSvc, BSTR ObjPath, BSTR Gw)
1532{
1533 VARIANT vGw;
1534 HRESULT hr = netIfWinCreateIpArrayVariantV4V6(&vGw, Gw);
1535 if (SUCCEEDED(hr))
1536 {
1537 netIfWinSetGateways(pSvc, ObjPath, &vGw);
1538 VariantClear(&vGw);
1539 }
1540 return hr;
1541}
1542
1543static HRESULT netIfWinEnableDHCP(IWbemServices * pSvc, BSTR ObjPath)
1544{
1545 ComPtr<IWbemClassObject> pClass;
1546 BSTR ClassName = SysAllocString(L"Win32_NetworkAdapterConfiguration");
1547 HRESULT hr;
1548 if (ClassName)
1549 {
1550 hr = pSvc->GetObject(ClassName, 0, NULL, pClass.asOutParam(), NULL);
1551 if (SUCCEEDED(hr))
1552 {
1553 ComPtr<IWbemClassObject> pOutParams;
1554
1555 hr = netIfExecMethod(pSvc, pClass, ObjPath, bstr_t(L"EnableDHCP"), NULL, NULL, 0, pOutParams.asOutParam());
1556 if (SUCCEEDED(hr))
1557 {
1558 VARIANT varReturnValue;
1559 hr = pOutParams->Get(bstr_t(L"ReturnValue"), 0,
1560 &varReturnValue, NULL, 0);
1561 Assert(SUCCEEDED(hr));
1562 if (SUCCEEDED(hr))
1563 {
1564// Assert(varReturnValue.vt == VT_UINT);
1565 int winEr = varReturnValue.uintVal;
1566 switch (winEr)
1567 {
1568 case 0:
1569 hr = S_OK;
1570 break;
1571 default:
1572 hr = HRESULT_FROM_WIN32( winEr );
1573 break;
1574 }
1575 }
1576 }
1577 }
1578 SysFreeString(ClassName);
1579 }
1580 else
1581 hr = HRESULT_FROM_WIN32(GetLastError());
1582
1583 return hr;
1584}
1585
1586static HRESULT netIfWinDhcpRediscover(IWbemServices * pSvc, BSTR ObjPath)
1587{
1588 ComPtr<IWbemClassObject> pClass;
1589 BSTR ClassName = SysAllocString(L"Win32_NetworkAdapterConfiguration");
1590 HRESULT hr;
1591 if (ClassName)
1592 {
1593 hr = pSvc->GetObject(ClassName, 0, NULL, pClass.asOutParam(), NULL);
1594 if (SUCCEEDED(hr))
1595 {
1596 ComPtr<IWbemClassObject> pOutParams;
1597
1598 hr = netIfExecMethod(pSvc, pClass, ObjPath, bstr_t(L"ReleaseDHCPLease"), NULL, NULL, 0, pOutParams.asOutParam());
1599 if (SUCCEEDED(hr))
1600 {
1601 VARIANT varReturnValue;
1602 hr = pOutParams->Get(bstr_t(L"ReturnValue"), 0, &varReturnValue, NULL, 0);
1603 Assert(SUCCEEDED(hr));
1604 if (SUCCEEDED(hr))
1605 {
1606// Assert(varReturnValue.vt == VT_UINT);
1607 int winEr = varReturnValue.uintVal;
1608 if (winEr == 0)
1609 {
1610 hr = netIfExecMethod(pSvc, pClass, ObjPath, bstr_t(L"RenewDHCPLease"), NULL, NULL, 0, pOutParams.asOutParam());
1611 if (SUCCEEDED(hr))
1612 {
1613 VARIANT varReturnValue;
1614 hr = pOutParams->Get(bstr_t(L"ReturnValue"), 0, &varReturnValue, NULL, 0);
1615 Assert(SUCCEEDED(hr));
1616 if (SUCCEEDED(hr))
1617 {
1618 // Assert(varReturnValue.vt == VT_UINT);
1619 int winEr = varReturnValue.uintVal;
1620 if (winEr == 0)
1621 hr = S_OK;
1622 else
1623 hr = HRESULT_FROM_WIN32( winEr );
1624 }
1625 }
1626 }
1627 else
1628 hr = HRESULT_FROM_WIN32( winEr );
1629 }
1630 }
1631 }
1632 SysFreeString(ClassName);
1633 }
1634 else
1635 hr = HRESULT_FROM_WIN32(GetLastError());
1636
1637 return hr;
1638}
1639
1640static HRESULT vboxNetCfgWinIsDhcpEnabled(IWbemClassObject * pAdapterConfig, BOOL *pEnabled)
1641{
1642 VARIANT vtEnabled;
1643 HRESULT hr = pAdapterConfig->Get(L"DHCPEnabled", 0, &vtEnabled, 0, 0);
1644 if (SUCCEEDED(hr))
1645 *pEnabled = vtEnabled.boolVal;
1646 return hr;
1647}
1648
1649VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinGetAdapterSettings(IN const GUID * pGuid, OUT PADAPTER_SETTINGS pSettings)
1650{
1651 HRESULT hr;
1652 ComPtr <IWbemServices> pSvc;
1653 hr = netIfWinCreateIWbemServices(pSvc.asOutParam());
1654 if (SUCCEEDED(hr))
1655 {
1656 ComPtr<IWbemClassObject> pAdapterConfig;
1657 hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam());
1658 if (SUCCEEDED(hr))
1659 {
1660 hr = vboxNetCfgWinIsDhcpEnabled(pAdapterConfig, &pSettings->bDhcp);
1661 if (SUCCEEDED(hr))
1662 hr = netIfWinGetIpSettings(pAdapterConfig, &pSettings->ip, &pSettings->mask);
1663 }
1664 }
1665
1666 return hr;
1667}
1668
1669VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinIsDhcpEnabled(const GUID * pGuid, BOOL *pEnabled)
1670{
1671 HRESULT hr;
1672 ComPtr <IWbemServices> pSvc;
1673 hr = netIfWinCreateIWbemServices(pSvc.asOutParam());
1674 if (SUCCEEDED(hr))
1675 {
1676 ComPtr<IWbemClassObject> pAdapterConfig;
1677 hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam());
1678 if (SUCCEEDED(hr))
1679 {
1680 VARIANT vtEnabled;
1681 hr = pAdapterConfig->Get(L"DHCPEnabled", 0, &vtEnabled, 0, 0);
1682 if (SUCCEEDED(hr))
1683 *pEnabled = vtEnabled.boolVal;
1684 }
1685 }
1686
1687 return hr;
1688}
1689
1690VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinEnableStaticIpConfig(IN const GUID *pGuid, IN ULONG ip, IN ULONG mask)
1691{
1692 NonStandardLogFlow(("VBoxNetCfgWinEnableStaticIpConfig: ip=0x%x mask=0x%x", ip, mask));
1693 ComPtr<IWbemServices> pSvc;
1694 HRESULT hr = netIfWinCreateIWbemServices(pSvc.asOutParam());
1695 if (SUCCEEDED(hr))
1696 {
1697 ComPtr<IWbemClassObject> pAdapterConfig;
1698 hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam());
1699 if (SUCCEEDED(hr))
1700 {
1701 BOOL bIsHostOnly;
1702 hr = netIfWinIsHostOnly(pAdapterConfig, &bIsHostOnly);
1703 if (SUCCEEDED(hr))
1704 {
1705 if (bIsHostOnly)
1706 {
1707 in_addr aIp[1];
1708 in_addr aMask[1];
1709 aIp[0].S_un.S_addr = ip;
1710 aMask[0].S_un.S_addr = mask;
1711
1712 BSTR ObjPath;
1713 hr = netIfWinAdapterConfigPath(pAdapterConfig, &ObjPath);
1714 if (SUCCEEDED(hr))
1715 {
1716 hr = netIfWinEnableStaticV4(pSvc, pGuid, ObjPath, aIp, aMask, ip != 0 ? 1 : 0);
1717 if (SUCCEEDED(hr))
1718 {
1719#if 0
1720 in_addr aGw[1];
1721 aGw[0].S_un.S_addr = gw;
1722 hr = netIfWinSetGatewaysV4(pSvc, ObjPath, aGw, 1);
1723 if (SUCCEEDED(hr))
1724#endif
1725 {
1726 }
1727 }
1728 SysFreeString(ObjPath);
1729 }
1730 }
1731 else
1732 {
1733 hr = E_FAIL;
1734 }
1735 }
1736 }
1737 }
1738
1739 NonStandardLogFlow(("VBoxNetCfgWinEnableStaticIpConfig: returns 0x%x", hr));
1740 return hr;
1741}
1742
1743#if 0
1744static HRESULT netIfEnableStaticIpConfigV6(const GUID *pGuid, IN_BSTR aIPV6Address, IN_BSTR aIPV6Mask, IN_BSTR aIPV6DefaultGateway)
1745{
1746 HRESULT hr;
1747 ComPtr <IWbemServices> pSvc;
1748 hr = netIfWinCreateIWbemServices(pSvc.asOutParam());
1749 if (SUCCEEDED(hr))
1750 {
1751 ComPtr<IWbemClassObject> pAdapterConfig;
1752 hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam());
1753 if (SUCCEEDED(hr))
1754 {
1755 BSTR ObjPath;
1756 hr = netIfWinAdapterConfigPath(pAdapterConfig, &ObjPath);
1757 if (SUCCEEDED(hr))
1758 {
1759 hr = netIfWinEnableStaticV4V6(pSvc, pAdapterConfig, ObjPath, aIPV6Address, aIPV6Mask);
1760 if (SUCCEEDED(hr))
1761 {
1762 if (aIPV6DefaultGateway)
1763 {
1764 hr = netIfWinSetGatewaysV4V6(pSvc, ObjPath, aIPV6DefaultGateway);
1765 }
1766 if (SUCCEEDED(hr))
1767 {
1768// hr = netIfWinUpdateConfig(pIf);
1769 }
1770 }
1771 SysFreeString(ObjPath);
1772 }
1773 }
1774 }
1775
1776 return SUCCEEDED(hr) ? VINF_SUCCESS : VERR_GENERAL_FAILURE;
1777}
1778
1779static HRESULT netIfEnableStaticIpConfigV6(const GUID *pGuid, IN_BSTR aIPV6Address, ULONG aIPV6MaskPrefixLength)
1780{
1781 RTNETADDRIPV6 Mask;
1782 int rc = prefixLength2IPv6Address(aIPV6MaskPrefixLength, &Mask);
1783 if (RT_SUCCESS(rc))
1784 {
1785 Bstr maskStr = composeIPv6Address(&Mask);
1786 rc = netIfEnableStaticIpConfigV6(pGuid, aIPV6Address, maskStr, NULL);
1787 }
1788 return rc;
1789}
1790#endif
1791
1792VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinEnableDynamicIpConfig(IN const GUID *pGuid)
1793{
1794 HRESULT hr;
1795 ComPtr <IWbemServices> pSvc;
1796 hr = netIfWinCreateIWbemServices(pSvc.asOutParam());
1797 if (SUCCEEDED(hr))
1798 {
1799 ComPtr<IWbemClassObject> pAdapterConfig;
1800 hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam());
1801 if (SUCCEEDED(hr))
1802 {
1803 BOOL bIsHostOnly;
1804 hr = netIfWinIsHostOnly(pAdapterConfig, &bIsHostOnly);
1805 if (SUCCEEDED(hr))
1806 {
1807 if (bIsHostOnly)
1808 {
1809 BSTR ObjPath;
1810 hr = netIfWinAdapterConfigPath(pAdapterConfig, &ObjPath);
1811 if (SUCCEEDED(hr))
1812 {
1813 hr = netIfWinEnableDHCP(pSvc, ObjPath);
1814 if (SUCCEEDED(hr))
1815 {
1816// hr = netIfWinUpdateConfig(pIf);
1817 }
1818 SysFreeString(ObjPath);
1819 }
1820 }
1821 else
1822 {
1823 hr = E_FAIL;
1824 }
1825 }
1826 }
1827 }
1828
1829
1830 return hr;
1831}
1832
1833VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinDhcpRediscover(IN const GUID *pGuid)
1834{
1835 HRESULT hr;
1836 ComPtr <IWbemServices> pSvc;
1837 hr = netIfWinCreateIWbemServices(pSvc.asOutParam());
1838 if (SUCCEEDED(hr))
1839 {
1840 ComPtr<IWbemClassObject> pAdapterConfig;
1841 hr = netIfWinFindAdapterClassById(pSvc, pGuid, pAdapterConfig.asOutParam());
1842 if (SUCCEEDED(hr))
1843 {
1844 BOOL bIsHostOnly;
1845 hr = netIfWinIsHostOnly(pAdapterConfig, &bIsHostOnly);
1846 if (SUCCEEDED(hr))
1847 {
1848 if (bIsHostOnly)
1849 {
1850 BSTR ObjPath;
1851 hr = netIfWinAdapterConfigPath(pAdapterConfig, &ObjPath);
1852 if (SUCCEEDED(hr))
1853 {
1854 hr = netIfWinDhcpRediscover(pSvc, ObjPath);
1855 if (SUCCEEDED(hr))
1856 {
1857 //hr = netIfWinUpdateConfig(pIf);
1858 }
1859 SysFreeString(ObjPath);
1860 }
1861 }
1862 else
1863 {
1864 hr = E_FAIL;
1865 }
1866 }
1867 }
1868 }
1869
1870
1871 return hr;
1872}
1873
1874typedef bool (*PFNVBOXNETCFG_IPSETTINGS_CALLBACK) (ULONG ip, ULONG mask, PVOID pContext);
1875
1876static void vboxNetCfgWinEnumIpConfig(PIP_ADAPTER_ADDRESSES pAddresses, PFNVBOXNETCFG_IPSETTINGS_CALLBACK pfnCallback, PVOID pContext)
1877{
1878 PIP_ADAPTER_ADDRESSES pAdapter;
1879 for (pAdapter = pAddresses; pAdapter; pAdapter = pAdapter->Next)
1880 {
1881 PIP_ADAPTER_UNICAST_ADDRESS pAddr = pAdapter->FirstUnicastAddress;
1882 PIP_ADAPTER_PREFIX pPrefix = pAdapter->FirstPrefix;
1883
1884 if (pAddr && pPrefix)
1885 {
1886 do
1887 {
1888 bool fIPFound, fMaskFound;
1889 fIPFound = fMaskFound = false;
1890 ULONG ip, mask;
1891 for (; pAddr && !fIPFound; pAddr = pAddr->Next)
1892 {
1893 switch (pAddr->Address.lpSockaddr->sa_family)
1894 {
1895 case AF_INET:
1896 fIPFound = true;
1897 memcpy(&ip,
1898 &((struct sockaddr_in *)pAddr->Address.lpSockaddr)->sin_addr.s_addr,
1899 sizeof(ip));
1900 break;
1901// case AF_INET6:
1902// break;
1903 }
1904 }
1905
1906 for (; pPrefix && !fMaskFound; pPrefix = pPrefix->Next)
1907 {
1908 switch (pPrefix->Address.lpSockaddr->sa_family)
1909 {
1910 case AF_INET:
1911 if (!pPrefix->PrefixLength || pPrefix->PrefixLength > 31) /* in case the ip helper API is queried while NetCfg write lock is held */
1912 break; /* the address values can contain illegal values */
1913 fMaskFound = true;
1914 mask = (~(((ULONG)~0) >> pPrefix->PrefixLength));
1915 mask = htonl(mask);
1916 break;
1917// case AF_INET6:
1918// break;
1919 }
1920 }
1921
1922 if (!fIPFound || !fMaskFound)
1923 break;
1924
1925 if (!pfnCallback(ip, mask, pContext))
1926 return;
1927 } while (true);
1928 }
1929 }
1930}
1931
1932typedef struct _IPPROBE_CONTEXT
1933{
1934 ULONG Prefix;
1935 bool bConflict;
1936}IPPROBE_CONTEXT, *PIPPROBE_CONTEXT;
1937
1938#define IPPROBE_INIT(_pContext, _addr) \
1939 ((_pContext)->bConflict = false, \
1940 (_pContext)->Prefix = _addr)
1941
1942#define IPPROBE_INIT_STR(_pContext, _straddr) \
1943 IPROBE_INIT(_pContext, inet_addr(_straddr))
1944
1945static bool vboxNetCfgWinIpProbeCallback (ULONG ip, ULONG mask, PVOID pContext)
1946{
1947 PIPPROBE_CONTEXT pProbe = (PIPPROBE_CONTEXT)pContext;
1948
1949 if ((ip & mask) == (pProbe->Prefix & mask))
1950 {
1951 pProbe->bConflict = true;
1952 return false;
1953 }
1954
1955 return true;
1956}
1957
1958VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinGenHostOnlyNetworkNetworkIp(OUT PULONG pNetIp, OUT PULONG pNetMask)
1959{
1960 DWORD dwRc;
1961 HRESULT hr = S_OK;
1962 /*
1963 * Most of the hosts probably have less than 10 adapters,
1964 * so we'll mostly succeed from the first attempt.
1965 */
1966 ULONG uBufLen = sizeof(IP_ADAPTER_ADDRESSES) * 10;
1967 PIP_ADAPTER_ADDRESSES pAddresses = (PIP_ADAPTER_ADDRESSES)malloc(uBufLen);
1968 if (!pAddresses)
1969 return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
1970 dwRc = GetAdaptersAddresses(AF_UNSPEC, GAA_FLAG_INCLUDE_PREFIX, NULL, pAddresses, &uBufLen);
1971 if (dwRc == ERROR_BUFFER_OVERFLOW)
1972 {
1973 /* Impressive! More than 10 adapters! Get more memory and try again. */
1974 free(pAddresses);
1975 pAddresses = (PIP_ADAPTER_ADDRESSES)malloc(uBufLen);
1976 if (!pAddresses)
1977 return HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
1978 dwRc = GetAdaptersAddresses(AF_UNSPEC, GAA_FLAG_INCLUDE_PREFIX, NULL, pAddresses, &uBufLen);
1979 }
1980 if (dwRc == NO_ERROR)
1981 {
1982 IPPROBE_CONTEXT Context;
1983 const ULONG ip192168 = inet_addr("192.168.0.0");
1984 srand(GetTickCount());
1985
1986 *pNetIp = 0;
1987 *pNetMask = 0;
1988
1989 for (int i = 0; i < 255; i++)
1990 {
1991 ULONG ipProbe = rand()*255/RAND_MAX;
1992 ipProbe = ip192168 | (ipProbe << 16);
1993 IPPROBE_INIT(&Context, ipProbe);
1994 vboxNetCfgWinEnumIpConfig(pAddresses, vboxNetCfgWinIpProbeCallback, &Context);
1995 if (!Context.bConflict)
1996 {
1997 *pNetIp = ipProbe;
1998 *pNetMask = inet_addr("255.255.255.0");
1999 break;
2000 }
2001 }
2002 if (*pNetIp == 0)
2003 dwRc = ERROR_DHCP_ADDRESS_CONFLICT;
2004 }
2005 else
2006 NonStandardLogFlow(("GetAdaptersAddresses err (%d)\n", dwRc));
2007
2008 if (pAddresses)
2009 free(pAddresses);
2010
2011 if (dwRc != NO_ERROR)
2012 {
2013 hr = HRESULT_FROM_WIN32(dwRc);
2014 }
2015
2016 return hr;
2017}
2018
2019/*
2020 * convenience functions to perform netflt/adp manipulations
2021 */
2022#define VBOXNETCFGWIN_NETFLT_ID L"sun_VBoxNetFlt"
2023#define VBOXNETCFGWIN_NETFLT_MP_ID L"sun_VBoxNetFltmp"
2024
2025static HRESULT vboxNetCfgWinNetFltUninstall(IN INetCfg *pNc, DWORD InfRmFlags)
2026{
2027 INetCfgComponent *pNcc = NULL;
2028 HRESULT hr = pNc->FindComponent(VBOXNETCFGWIN_NETFLT_ID, &pNcc);
2029 if (hr == S_OK)
2030 {
2031 NonStandardLog("NetFlt is installed currently, uninstalling ...\n");
2032
2033 hr = VBoxNetCfgWinUninstallComponent(pNc, pNcc);
2034 NonStandardLogFlow(("NetFlt component uninstallation ended with hr (0x%x)\n", hr));
2035
2036 pNcc->Release();
2037 }
2038 else if (hr == S_FALSE)
2039 {
2040 NonStandardLog("NetFlt is not installed currently\n");
2041 }
2042 else
2043 {
2044 NonStandardLogFlow(("FindComponent failed, hr (0x%x)\n", hr));
2045 }
2046
2047 VBoxDrvCfgInfUninstallAllF(L"NetService", VBOXNETCFGWIN_NETFLT_ID, InfRmFlags);
2048 VBoxDrvCfgInfUninstallAllF(L"Net", VBOXNETCFGWIN_NETFLT_MP_ID, InfRmFlags);
2049
2050 return hr;
2051}
2052
2053VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinNetFltUninstall(IN INetCfg *pNc)
2054{
2055 return vboxNetCfgWinNetFltUninstall(pNc, 0);
2056}
2057
2058VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinNetFltInstall(IN INetCfg *pNc,
2059 IN LPCWSTR const *apInfFullPaths, IN UINT cInfFullPaths)
2060{
2061 HRESULT hr = vboxNetCfgWinNetFltUninstall(pNc, SUOI_FORCEDELETE);
2062 if (SUCCEEDED(hr))
2063 {
2064 NonStandardLog("NetFlt will be installed ...\n");
2065 hr = vboxNetCfgWinInstallInfAndComponent(pNc, VBOXNETCFGWIN_NETFLT_ID,
2066 &GUID_DEVCLASS_NETSERVICE,
2067 apInfFullPaths,
2068 cInfFullPaths,
2069 NULL);
2070 }
2071 return hr;
2072}
2073
2074#define VBOXNETCFGWIN_NETADP_ID L"sun_VBoxNetAdp"
2075static HRESULT vboxNetCfgWinNetAdpUninstall(IN INetCfg *pNc, LPCWSTR pwszId, DWORD InfRmFlags)
2076{
2077 HRESULT hr = S_OK;
2078 NonStandardLog("Finding NetAdp driver package and trying to uninstall it ...\n");
2079
2080 VBoxDrvCfgInfUninstallAllF(L"Net", pwszId, InfRmFlags);
2081 NonStandardLog("NetAdp is not installed currently\n");
2082 return hr;
2083}
2084
2085VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinNetAdpUninstall(IN INetCfg *pNc, IN LPCWSTR pwszId)
2086{
2087 return vboxNetCfgWinNetAdpUninstall(pNc, pwszId, SUOI_FORCEDELETE);
2088}
2089
2090VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinNetAdpInstall(IN INetCfg *pNc,
2091 IN LPCWSTR const pInfFullPath)
2092{
2093 NonStandardLog("NetAdp will be installed ...\n");
2094 HRESULT hr = vboxNetCfgWinInstallInfAndComponent(pNc, VBOXNETCFGWIN_NETADP_ID,
2095 &GUID_DEVCLASS_NET,
2096 &pInfFullPath,
2097 1,
2098 NULL);
2099 return hr;
2100}
2101
2102#define VBOXNETCFGWIN_NETLWF_ID L"oracle_VBoxNetLwf"
2103
2104static HRESULT vboxNetCfgWinNetLwfUninstall(IN INetCfg *pNc, DWORD InfRmFlags)
2105{
2106 INetCfgComponent * pNcc = NULL;
2107 HRESULT hr = pNc->FindComponent(VBOXNETCFGWIN_NETLWF_ID, &pNcc);
2108 if (hr == S_OK)
2109 {
2110 NonStandardLog("NetLwf is installed currently, uninstalling ...\n");
2111
2112 hr = VBoxNetCfgWinUninstallComponent(pNc, pNcc);
2113
2114 pNcc->Release();
2115 }
2116 else if (hr == S_FALSE)
2117 {
2118 NonStandardLog("NetLwf is not installed currently\n");
2119 hr = S_OK;
2120 }
2121 else
2122 {
2123 NonStandardLogFlow(("FindComponent failed, hr (0x%x)\n", hr));
2124 hr = S_OK;
2125 }
2126
2127 VBoxDrvCfgInfUninstallAllF(L"NetService", VBOXNETCFGWIN_NETLWF_ID, InfRmFlags);
2128
2129 return hr;
2130}
2131
2132VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinNetLwfUninstall(IN INetCfg *pNc)
2133{
2134 return vboxNetCfgWinNetLwfUninstall(pNc, 0);
2135}
2136
2137static void VBoxNetCfgWinFilterLimitWorkaround(void)
2138{
2139 /*
2140 * Need to check if the system has a limit of installed filter drivers. If it
2141 * has, bump the limit to 14, which the maximum value supported by Windows 7.
2142 * Note that we only touch the limit if it is set to the default value (8).
2143 * See @bugref{7899}.
2144 */
2145 HKEY hNetKey;
2146 DWORD dwMaxNumFilters = 0;
2147 DWORD cbMaxNumFilters = sizeof(dwMaxNumFilters);
2148 LONG hr = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
2149 _T("SYSTEM\\CurrentControlSet\\Control\\Network"),
2150 0, KEY_QUERY_VALUE | KEY_SET_VALUE, &hNetKey);
2151 if (SUCCEEDED(hr))
2152 {
2153 hr = RegQueryValueEx(hNetKey, _T("MaxNumFilters"), NULL, NULL,
2154 (LPBYTE)&dwMaxNumFilters, &cbMaxNumFilters);
2155 if (SUCCEEDED(hr) && cbMaxNumFilters == sizeof(dwMaxNumFilters) && dwMaxNumFilters == 8)
2156 {
2157 dwMaxNumFilters = 14;
2158 hr = RegSetValueEx(hNetKey, _T("MaxNumFilters"), 0, REG_DWORD,
2159 (LPBYTE)&dwMaxNumFilters, sizeof(dwMaxNumFilters));
2160 if (SUCCEEDED(hr))
2161 NonStandardLog("Adjusted the installed filter limit to 14...\n");
2162 else
2163 NonStandardLog("Failed to set MaxNumFilters, error code 0x%x\n", hr);
2164 }
2165 RegCloseKey(hNetKey);
2166 }
2167 else
2168 {
2169 NonStandardLog("Failed to open network key, error code 0x%x\n", hr);
2170 }
2171
2172}
2173
2174VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinNetLwfInstall(IN INetCfg *pNc,
2175 IN LPCWSTR const pInfFullPath)
2176{
2177 HRESULT hr = vboxNetCfgWinNetLwfUninstall(pNc, SUOI_FORCEDELETE);
2178 if (SUCCEEDED(hr))
2179 {
2180 VBoxNetCfgWinFilterLimitWorkaround();
2181 NonStandardLog("NetLwf will be installed ...\n");
2182 hr = vboxNetCfgWinInstallInfAndComponent(pNc, VBOXNETCFGWIN_NETLWF_ID,
2183 &GUID_DEVCLASS_NETSERVICE,
2184 &pInfFullPath,
2185 1,
2186 NULL);
2187 }
2188 return hr;
2189}
2190
2191#define VBOX_CONNECTION_NAME L"VirtualBox Host-Only Network"
2192VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinGenHostonlyConnectionName(PCWSTR DevName, WCHAR *pBuf, PULONG pcbBuf)
2193{
2194 const WCHAR * pSuffix = wcsrchr( DevName, L'#' );
2195 ULONG cbSize = sizeof(VBOX_CONNECTION_NAME);
2196 ULONG cbSufSize = 0;
2197
2198 if (pSuffix)
2199 {
2200 cbSize += (ULONG)wcslen(pSuffix) * 2;
2201 cbSize += 2; /* for space */
2202 }
2203
2204 if (*pcbBuf < cbSize)
2205 {
2206 *pcbBuf = cbSize;
2207 return E_FAIL;
2208 }
2209
2210 wcscpy(pBuf, VBOX_CONNECTION_NAME);
2211 if (pSuffix)
2212 {
2213 wcscat(pBuf, L" ");
2214 wcscat(pBuf, pSuffix);
2215 }
2216
2217 return S_OK;
2218}
2219
2220static BOOL vboxNetCfgWinAdjustHostOnlyNetworkInterfacePriority(IN INetCfg *pNc, IN INetCfgComponent *pNcc, PVOID pContext)
2221{
2222 INetCfgComponentBindings *pNetCfgBindings;
2223 GUID *pGuid = (GUID*)pContext;
2224
2225 /* Get component's binding. */
2226 HRESULT hr = pNcc->QueryInterface(IID_INetCfgComponentBindings, (PVOID*)&pNetCfgBindings);
2227 if (SUCCEEDED(hr))
2228 {
2229 /* Get binding path enumerator reference. */
2230 IEnumNetCfgBindingPath *pEnumNetCfgBindPath;
2231 hr = pNetCfgBindings->EnumBindingPaths(EBP_BELOW, &pEnumNetCfgBindPath);
2232 if (SUCCEEDED(hr))
2233 {
2234 bool bFoundIface = false;
2235 hr = pEnumNetCfgBindPath->Reset();
2236 do
2237 {
2238 INetCfgBindingPath *pNetCfgBindPath;
2239 hr = pEnumNetCfgBindPath->Next(1, &pNetCfgBindPath, NULL);
2240 if (hr == S_OK)
2241 {
2242 IEnumNetCfgBindingInterface *pEnumNetCfgBindIface;
2243 hr = pNetCfgBindPath->EnumBindingInterfaces(&pEnumNetCfgBindIface);
2244 if (hr == S_OK)
2245 {
2246 pEnumNetCfgBindIface->Reset();
2247 do
2248 {
2249 INetCfgBindingInterface *pNetCfgBindIfce;
2250 hr = pEnumNetCfgBindIface->Next(1, &pNetCfgBindIfce, NULL);
2251 if (hr == S_OK)
2252 {
2253 INetCfgComponent *pNetCfgCompo;
2254 hr = pNetCfgBindIfce->GetLowerComponent(&pNetCfgCompo);
2255 if (hr == S_OK)
2256 {
2257 ULONG uComponentStatus;
2258 hr = pNetCfgCompo->GetDeviceStatus(&uComponentStatus);
2259 if (hr == S_OK)
2260 {
2261 GUID guid;
2262 hr = pNetCfgCompo->GetInstanceGuid(&guid);
2263 if ( hr == S_OK
2264 && guid == *pGuid)
2265 {
2266 hr = pNetCfgBindings->MoveAfter(pNetCfgBindPath, NULL);
2267 if (FAILED(hr))
2268 NonStandardLogFlow(("Unable to move interface, hr (0x%x)\n", hr));
2269 bFoundIface = true;
2270 }
2271 }
2272 pNetCfgCompo->Release();
2273 }
2274 else
2275 NonStandardLogFlow(("GetLowerComponent failed, hr (0x%x)\n", hr));
2276 pNetCfgBindIfce->Release();
2277 }
2278 else
2279 {
2280 if (hr == S_FALSE) /* No more binding interfaces? */
2281 hr = S_OK;
2282 else
2283 NonStandardLogFlow(("Next binding interface failed, hr (0x%x)\n", hr));
2284 break;
2285 }
2286 } while (!bFoundIface);
2287 pEnumNetCfgBindIface->Release();
2288 }
2289 else
2290 NonStandardLogFlow(("EnumBindingInterfaces failed, hr (0x%x)\n", hr));
2291 pNetCfgBindPath->Release();
2292 }
2293 else
2294 {
2295 if (hr == S_FALSE) /* No more binding paths? */
2296 hr = S_OK;
2297 else
2298 NonStandardLogFlow(("Next bind path failed, hr (0x%x)\n", hr));
2299 break;
2300 }
2301 } while (!bFoundIface);
2302 pEnumNetCfgBindPath->Release();
2303 }
2304 else
2305 NonStandardLogFlow(("EnumBindingPaths failed, hr (0x%x)\n", hr));
2306 pNetCfgBindings->Release();
2307 }
2308 else
2309 NonStandardLogFlow(("QueryInterface for IID_INetCfgComponentBindings failed, hr (0x%x)\n", hr));
2310 return TRUE;
2311}
2312
2313static UINT WINAPI vboxNetCfgWinPspFileCallback(
2314 PVOID Context,
2315 UINT Notification,
2316 UINT_PTR Param1,
2317 UINT_PTR Param2
2318 )
2319{
2320 switch (Notification)
2321 {
2322 case SPFILENOTIFY_TARGETNEWER:
2323 case SPFILENOTIFY_TARGETEXISTS:
2324 return TRUE;
2325 }
2326 return SetupDefaultQueueCallback(Context, Notification, Param1, Param2);
2327}
2328
2329/* The original source of the VBoxNetAdp adapter creation/destruction code has the following copyright */
2330/*
2331 Copyright 2004 by the Massachusetts Institute of Technology
2332
2333 All rights reserved.
2334
2335 Permission to use, copy, modify, and distribute this software and its
2336 documentation for any purpose and without fee is hereby granted,
2337 provided that the above copyright notice appear in all copies and that
2338 both that copyright notice and this permission notice appear in
2339 supporting documentation, and that the name of the Massachusetts
2340 Institute of Technology (M.I.T.) not be used in advertising or publicity
2341 pertaining to distribution of the software without specific, written
2342 prior permission.
2343
2344 M.I.T. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
2345 ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
2346 M.I.T. BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
2347 ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
2348 WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
2349 ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
2350 SOFTWARE.
2351*/
2352
2353
2354/**
2355 * Use the IShellFolder API to rename the connection.
2356 */
2357static HRESULT rename_shellfolder (PCWSTR wGuid, PCWSTR wNewName)
2358{
2359 /* This is the GUID for the network connections folder. It is constant.
2360 * {7007ACC7-3202-11D1-AAD2-00805FC1270E} */
2361 const GUID CLSID_NetworkConnections = {
2362 0x7007ACC7, 0x3202, 0x11D1, {
2363 0xAA, 0xD2, 0x00, 0x80, 0x5F, 0xC1, 0x27, 0x0E
2364 }
2365 };
2366
2367 LPITEMIDLIST pidl = NULL;
2368 IShellFolder *pShellFolder = NULL;
2369 HRESULT hr;
2370
2371 /* Build the display name in the form "::{GUID}". */
2372 if (wcslen(wGuid) >= MAX_PATH)
2373 return E_INVALIDARG;
2374 WCHAR szAdapterGuid[MAX_PATH + 2] = {0};
2375 swprintf(szAdapterGuid, L"::%ls", wGuid);
2376
2377 /* Create an instance of the network connections folder. */
2378 hr = CoCreateInstance(CLSID_NetworkConnections, NULL,
2379 CLSCTX_INPROC_SERVER, IID_IShellFolder,
2380 reinterpret_cast<LPVOID *>(&pShellFolder));
2381 /* Parse the display name. */
2382 if (SUCCEEDED (hr))
2383 {
2384 hr = pShellFolder->ParseDisplayName (NULL, NULL, szAdapterGuid, NULL,
2385 &pidl, NULL);
2386 }
2387 if (SUCCEEDED (hr))
2388 {
2389 hr = pShellFolder->SetNameOf (NULL, pidl, wNewName, SHGDN_NORMAL,
2390 &pidl);
2391 }
2392
2393 CoTaskMemFree (pidl);
2394
2395 if (pShellFolder)
2396 pShellFolder->Release();
2397
2398 return hr;
2399}
2400
2401/**
2402 * Loads a system DLL.
2403 *
2404 * @returns Module handle or NULL
2405 * @param pszName The DLL name.
2406 */
2407static HMODULE loadSystemDll(const char *pszName)
2408{
2409 char szPath[MAX_PATH];
2410 UINT cchPath = GetSystemDirectoryA(szPath, sizeof(szPath));
2411 size_t cbName = strlen(pszName) + 1;
2412 if (cchPath + 1 + cbName > sizeof(szPath))
2413 return NULL;
2414 szPath[cchPath] = '\\';
2415 memcpy(&szPath[cchPath + 1], pszName, cbName);
2416 return LoadLibraryA(szPath);
2417}
2418
2419VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinRenameConnection (LPWSTR pGuid, PCWSTR NewName)
2420{
2421 typedef HRESULT (WINAPI *lpHrRenameConnection) (const GUID *, PCWSTR);
2422 lpHrRenameConnection RenameConnectionFunc = NULL;
2423 HRESULT status;
2424
2425 /* First try the IShellFolder interface, which was unimplemented
2426 * for the network connections folder before XP. */
2427 status = rename_shellfolder (pGuid, NewName);
2428 if (status == E_NOTIMPL)
2429 {
2430/** @todo that code doesn't seem to work! */
2431 /* The IShellFolder interface is not implemented on this platform.
2432 * Try the (undocumented) HrRenameConnection API in the netshell
2433 * library. */
2434 CLSID clsid;
2435 HINSTANCE hNetShell;
2436 status = CLSIDFromString ((LPOLESTR) pGuid, &clsid);
2437 if (FAILED(status))
2438 return E_FAIL;
2439 hNetShell = loadSystemDll("netshell.dll");
2440 if (hNetShell == NULL)
2441 return E_FAIL;
2442 RenameConnectionFunc =
2443 (lpHrRenameConnection) GetProcAddress (hNetShell,
2444 "HrRenameConnection");
2445 if (RenameConnectionFunc == NULL)
2446 {
2447 FreeLibrary (hNetShell);
2448 return E_FAIL;
2449 }
2450 status = RenameConnectionFunc (&clsid, NewName);
2451 FreeLibrary (hNetShell);
2452 }
2453 if (FAILED (status))
2454 return status;
2455
2456 return S_OK;
2457}
2458
2459#define DRIVERHWID _T("sun_VBoxNetAdp")
2460
2461#define SetErrBreak(strAndArgs) \
2462 if (1) { \
2463 hrc = E_FAIL; \
2464 NonStandardLog strAndArgs; \
2465 break; \
2466 } else do {} while (0)
2467
2468VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinRemoveHostOnlyNetworkInterface(IN const GUID *pGUID, OUT BSTR *pErrMsg)
2469{
2470 HRESULT hrc = S_OK;
2471
2472 do
2473 {
2474 TCHAR lszPnPInstanceId [512] = {0};
2475
2476 /* We have to find the device instance ID through a registry search */
2477
2478 HKEY hkeyNetwork = 0;
2479 HKEY hkeyConnection = 0;
2480
2481 do
2482 {
2483 WCHAR strRegLocation [256];
2484 WCHAR wszGuid[50];
2485
2486 int length = StringFromGUID2(*pGUID, wszGuid, RT_ELEMENTS(wszGuid));
2487 if (!length)
2488 SetErrBreak(("Failed to create a Guid string"));
2489
2490 swprintf (strRegLocation,
2491 L"SYSTEM\\CurrentControlSet\\Control\\Network\\"
2492 L"{4D36E972-E325-11CE-BFC1-08002BE10318}\\%s",
2493 wszGuid);
2494
2495 LONG status;
2496 status = RegOpenKeyExW (HKEY_LOCAL_MACHINE, strRegLocation, 0,
2497 KEY_READ, &hkeyNetwork);
2498 if ((status != ERROR_SUCCESS) || !hkeyNetwork)
2499 SetErrBreak (("Host interface network is not found in registry (%S) [1]",
2500 strRegLocation));
2501
2502 status = RegOpenKeyExW (hkeyNetwork, L"Connection", 0,
2503 KEY_READ, &hkeyConnection);
2504 if ((status != ERROR_SUCCESS) || !hkeyConnection)
2505 SetErrBreak (("Host interface network is not found in registry (%S) [2]",
2506 strRegLocation));
2507
2508 DWORD len = sizeof (lszPnPInstanceId);
2509 DWORD dwKeyType;
2510 status = RegQueryValueExW (hkeyConnection, L"PnPInstanceID", NULL,
2511 &dwKeyType, (LPBYTE) lszPnPInstanceId, &len);
2512 if ((status != ERROR_SUCCESS) || (dwKeyType != REG_SZ))
2513 SetErrBreak (("Host interface network is not found in registry (%S) [3]",
2514 strRegLocation));
2515 }
2516 while (0);
2517
2518 if (hkeyConnection)
2519 RegCloseKey (hkeyConnection);
2520 if (hkeyNetwork)
2521 RegCloseKey (hkeyNetwork);
2522
2523 if (FAILED (hrc))
2524 break;
2525
2526 /*
2527 * Now we are going to enumerate all network devices and
2528 * wait until we encounter the right device instance ID
2529 */
2530
2531 HDEVINFO hDeviceInfo = INVALID_HANDLE_VALUE;
2532
2533 do
2534 {
2535 BOOL ok;
2536 DWORD ret = 0;
2537 GUID netGuid;
2538 SP_DEVINFO_DATA DeviceInfoData;
2539 DWORD index = 0;
2540 BOOL found = FALSE;
2541 DWORD size = 0;
2542
2543 /* initialize the structure size */
2544 DeviceInfoData.cbSize = sizeof (SP_DEVINFO_DATA);
2545
2546 /* copy the net class GUID */
2547 memcpy (&netGuid, &GUID_DEVCLASS_NET, sizeof (GUID_DEVCLASS_NET));
2548
2549 /* return a device info set contains all installed devices of the Net class */
2550 hDeviceInfo = SetupDiGetClassDevs (&netGuid, NULL, NULL, DIGCF_PRESENT);
2551
2552 if (hDeviceInfo == INVALID_HANDLE_VALUE)
2553 SetErrBreak (("SetupDiGetClassDevs failed (0x%08X)", GetLastError()));
2554
2555 /* enumerate the driver info list */
2556 while (TRUE)
2557 {
2558 TCHAR *deviceHwid;
2559
2560 ok = SetupDiEnumDeviceInfo (hDeviceInfo, index, &DeviceInfoData);
2561
2562 if (!ok)
2563 {
2564 if (GetLastError() == ERROR_NO_MORE_ITEMS)
2565 break;
2566 else
2567 {
2568 index++;
2569 continue;
2570 }
2571 }
2572
2573 /* try to get the hardware ID registry property */
2574 ok = SetupDiGetDeviceRegistryProperty (hDeviceInfo,
2575 &DeviceInfoData,
2576 SPDRP_HARDWAREID,
2577 NULL,
2578 NULL,
2579 0,
2580 &size);
2581 if (!ok)
2582 {
2583 if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
2584 {
2585 index++;
2586 continue;
2587 }
2588
2589 deviceHwid = (TCHAR *) malloc (size);
2590 ok = SetupDiGetDeviceRegistryProperty (hDeviceInfo,
2591 &DeviceInfoData,
2592 SPDRP_HARDWAREID,
2593 NULL,
2594 (PBYTE)deviceHwid,
2595 size,
2596 NULL);
2597 if (!ok)
2598 {
2599 free (deviceHwid);
2600 deviceHwid = NULL;
2601 index++;
2602 continue;
2603 }
2604 }
2605 else
2606 {
2607 /* something is wrong. This shouldn't have worked with a NULL buffer */
2608 index++;
2609 continue;
2610 }
2611
2612 for (TCHAR *t = deviceHwid;
2613 t && *t && t < &deviceHwid[size / sizeof(TCHAR)];
2614 t += _tcslen (t) + 1)
2615 {
2616 if (!_tcsicmp (DRIVERHWID, t))
2617 {
2618 /* get the device instance ID */
2619 TCHAR devId[MAX_DEVICE_ID_LEN];
2620 if (CM_Get_Device_ID(DeviceInfoData.DevInst,
2621 devId, MAX_DEVICE_ID_LEN, 0) == CR_SUCCESS)
2622 {
2623 /* compare to what we determined before */
2624 if (wcscmp(devId, lszPnPInstanceId) == 0)
2625 {
2626 found = TRUE;
2627 break;
2628 }
2629 }
2630 }
2631 }
2632
2633 if (deviceHwid)
2634 {
2635 free (deviceHwid);
2636 deviceHwid = NULL;
2637 }
2638
2639 if (found)
2640 break;
2641
2642 index++;
2643 }
2644
2645 if (found == FALSE)
2646 SetErrBreak (("Host Interface Network driver not found (0x%08X)",
2647 GetLastError()));
2648
2649 ok = SetupDiSetSelectedDevice (hDeviceInfo, &DeviceInfoData);
2650 if (!ok)
2651 SetErrBreak (("SetupDiSetSelectedDevice failed (0x%08X)",
2652 GetLastError()));
2653
2654 ok = SetupDiCallClassInstaller (DIF_REMOVE, hDeviceInfo, &DeviceInfoData);
2655 if (!ok)
2656 SetErrBreak (("SetupDiCallClassInstaller (DIF_REMOVE) failed (0x%08X)",
2657 GetLastError()));
2658 }
2659 while (0);
2660
2661 /* clean up the device info set */
2662 if (hDeviceInfo != INVALID_HANDLE_VALUE)
2663 SetupDiDestroyDeviceInfoList (hDeviceInfo);
2664
2665 if (FAILED (hrc))
2666 break;
2667 }
2668 while (0);
2669
2670 return hrc;
2671}
2672
2673VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinUpdateHostOnlyNetworkInterface(LPCWSTR pcsxwInf, BOOL *pbRebootRequired, LPCWSTR pcsxwId)
2674{
2675 return VBoxDrvCfgDrvUpdate(pcsxwId, pcsxwInf, pbRebootRequired);
2676}
2677
2678VBOXNETCFGWIN_DECL(HRESULT) VBoxNetCfgWinCreateHostOnlyNetworkInterface(IN LPCWSTR pInfPath, IN bool bIsInfPathFile,
2679 OUT GUID *pGuid, OUT BSTR *lppszName, OUT BSTR *pErrMsg)
2680{
2681 HRESULT hrc = S_OK;
2682
2683 HDEVINFO hDeviceInfo = INVALID_HANDLE_VALUE;
2684 SP_DEVINFO_DATA DeviceInfoData;
2685 PVOID pQueueCallbackContext = NULL;
2686 DWORD ret = 0;
2687 BOOL found = FALSE;
2688 BOOL registered = FALSE;
2689 BOOL destroyList = FALSE;
2690 WCHAR pWCfgGuidString [50];
2691 WCHAR DevName[256];
2692
2693 do
2694 {
2695 GUID netGuid;
2696 SP_DRVINFO_DATA DriverInfoData;
2697 SP_DEVINSTALL_PARAMS DeviceInstallParams;
2698 TCHAR className [MAX_PATH];
2699 DWORD index = 0;
2700 PSP_DRVINFO_DETAIL_DATA pDriverInfoDetail;
2701 /* for our purposes, 2k buffer is more
2702 * than enough to obtain the hardware ID
2703 * of the VBoxNetAdp driver. */
2704 DWORD detailBuf [2048];
2705
2706 HKEY hkey = NULL;
2707 DWORD cbSize;
2708 DWORD dwValueType;
2709
2710 /* initialize the structure size */
2711 DeviceInfoData.cbSize = sizeof (SP_DEVINFO_DATA);
2712 DriverInfoData.cbSize = sizeof (SP_DRVINFO_DATA);
2713
2714 /* copy the net class GUID */
2715 memcpy(&netGuid, &GUID_DEVCLASS_NET, sizeof(GUID_DEVCLASS_NET));
2716
2717 /* create an empty device info set associated with the net class GUID */
2718 hDeviceInfo = SetupDiCreateDeviceInfoList(&netGuid, NULL);
2719 if (hDeviceInfo == INVALID_HANDLE_VALUE)
2720 SetErrBreak (("SetupDiCreateDeviceInfoList failed (0x%08X)",
2721 GetLastError()));
2722
2723 /* get the class name from GUID */
2724 BOOL fResult = SetupDiClassNameFromGuid (&netGuid, className, MAX_PATH, NULL);
2725 if (!fResult)
2726 SetErrBreak (("SetupDiClassNameFromGuid failed (0x%08X)",
2727 GetLastError()));
2728
2729 /* create a device info element and add the new device instance
2730 * key to registry */
2731 fResult = SetupDiCreateDeviceInfo (hDeviceInfo, className, &netGuid, NULL, NULL,
2732 DICD_GENERATE_ID, &DeviceInfoData);
2733 if (!fResult)
2734 SetErrBreak (("SetupDiCreateDeviceInfo failed (0x%08X)",
2735 GetLastError()));
2736
2737 /* select the newly created device info to be the currently
2738 selected member */
2739 fResult = SetupDiSetSelectedDevice (hDeviceInfo, &DeviceInfoData);
2740 if (!fResult)
2741 SetErrBreak (("SetupDiSetSelectedDevice failed (0x%08X)",
2742 GetLastError()));
2743
2744 if (pInfPath)
2745 {
2746 /* get the device install parameters and disable filecopy */
2747 DeviceInstallParams.cbSize = sizeof(SP_DEVINSTALL_PARAMS);
2748 fResult = SetupDiGetDeviceInstallParams (hDeviceInfo, &DeviceInfoData,
2749 &DeviceInstallParams);
2750 if (fResult)
2751 {
2752 memset(DeviceInstallParams.DriverPath, 0, sizeof(DeviceInstallParams.DriverPath));
2753 size_t pathLenght = wcslen(pInfPath) + 1/* null terminator */;
2754 if (pathLenght < sizeof(DeviceInstallParams.DriverPath)/sizeof(DeviceInstallParams.DriverPath[0]))
2755 {
2756 memcpy(DeviceInstallParams.DriverPath, pInfPath, pathLenght*sizeof(DeviceInstallParams.DriverPath[0]));
2757
2758 if (bIsInfPathFile)
2759 {
2760 DeviceInstallParams.Flags |= DI_ENUMSINGLEINF;
2761 }
2762
2763 fResult = SetupDiSetDeviceInstallParams(hDeviceInfo, &DeviceInfoData,
2764 &DeviceInstallParams);
2765 if (!fResult)
2766 {
2767 DWORD winEr = GetLastError();
2768 NonStandardLogFlow(("SetupDiSetDeviceInstallParams failed, winEr (%d)\n", winEr));
2769 break;
2770 }
2771 }
2772 else
2773 {
2774 NonStandardLogFlow(("SetupDiSetDeviceInstallParams faileed: INF path is too long\n"));
2775 break;
2776 }
2777 }
2778 else
2779 {
2780 DWORD winEr = GetLastError();
2781 NonStandardLogFlow(("SetupDiGetDeviceInstallParams failed, winEr (%d)\n", winEr));
2782 }
2783 }
2784
2785 /* build a list of class drivers */
2786 fResult = SetupDiBuildDriverInfoList (hDeviceInfo, &DeviceInfoData,
2787 SPDIT_CLASSDRIVER);
2788 if (!fResult)
2789 SetErrBreak (("SetupDiBuildDriverInfoList failed (0x%08X)",
2790 GetLastError()));
2791
2792 destroyList = TRUE;
2793
2794 /* enumerate the driver info list */
2795 while (TRUE)
2796 {
2797 BOOL ret;
2798
2799 ret = SetupDiEnumDriverInfo (hDeviceInfo, &DeviceInfoData,
2800 SPDIT_CLASSDRIVER, index, &DriverInfoData);
2801
2802 /* if the function failed and GetLastError() returned
2803 * ERROR_NO_MORE_ITEMS, then we have reached the end of the
2804 * list. Otherwise there was something wrong with this
2805 * particular driver. */
2806 if (!ret)
2807 {
2808 if (GetLastError() == ERROR_NO_MORE_ITEMS)
2809 break;
2810 else
2811 {
2812 index++;
2813 continue;
2814 }
2815 }
2816
2817 pDriverInfoDetail = (PSP_DRVINFO_DETAIL_DATA) detailBuf;
2818 pDriverInfoDetail->cbSize = sizeof(SP_DRVINFO_DETAIL_DATA);
2819
2820 /* if we successfully find the hardware ID and it turns out to
2821 * be the one for the loopback driver, then we are done. */
2822 if (SetupDiGetDriverInfoDetail (hDeviceInfo,
2823 &DeviceInfoData,
2824 &DriverInfoData,
2825 pDriverInfoDetail,
2826 sizeof (detailBuf),
2827 NULL))
2828 {
2829 TCHAR * t;
2830
2831 /* pDriverInfoDetail->HardwareID is a MULTISZ string. Go through the
2832 * whole list and see if there is a match somewhere. */
2833 t = pDriverInfoDetail->HardwareID;
2834 while (t && *t && t < (TCHAR *) &detailBuf [RT_ELEMENTS(detailBuf)])
2835 {
2836 if (!_tcsicmp(t, DRIVERHWID))
2837 break;
2838
2839 t += _tcslen(t) + 1;
2840 }
2841
2842 if (t && *t && t < (TCHAR *) &detailBuf [RT_ELEMENTS(detailBuf)])
2843 {
2844 found = TRUE;
2845 break;
2846 }
2847 }
2848
2849 index ++;
2850 }
2851
2852 if (!found)
2853 SetErrBreak(("Could not find Host Interface Networking driver! Please reinstall"));
2854
2855 /* set the loopback driver to be the currently selected */
2856 fResult = SetupDiSetSelectedDriver (hDeviceInfo, &DeviceInfoData,
2857 &DriverInfoData);
2858 if (!fResult)
2859 SetErrBreak(("SetupDiSetSelectedDriver failed (0x%08X)",
2860 GetLastError()));
2861
2862 /* register the phantom device to prepare for install */
2863 fResult = SetupDiCallClassInstaller (DIF_REGISTERDEVICE, hDeviceInfo,
2864 &DeviceInfoData);
2865 if (!fResult)
2866 {
2867 DWORD err = GetLastError();
2868 SetErrBreak (("SetupDiCallClassInstaller failed (0x%08X)",
2869 err));
2870 }
2871
2872 /* registered, but remove if errors occur in the following code */
2873 registered = TRUE;
2874
2875 /* ask the installer if we can install the device */
2876 fResult = SetupDiCallClassInstaller (DIF_ALLOW_INSTALL, hDeviceInfo,
2877 &DeviceInfoData);
2878 if (!fResult)
2879 {
2880 if (GetLastError() != ERROR_DI_DO_DEFAULT)
2881 SetErrBreak (("SetupDiCallClassInstaller (DIF_ALLOW_INSTALL) failed (0x%08X)",
2882 GetLastError()));
2883 /* that's fine */
2884 }
2885
2886 /* get the device install parameters and disable filecopy */
2887 DeviceInstallParams.cbSize = sizeof(SP_DEVINSTALL_PARAMS);
2888 fResult = SetupDiGetDeviceInstallParams (hDeviceInfo, &DeviceInfoData,
2889 &DeviceInstallParams);
2890 if (fResult)
2891 {
2892 pQueueCallbackContext = SetupInitDefaultQueueCallback(NULL);
2893 if (pQueueCallbackContext)
2894 {
2895 DeviceInstallParams.InstallMsgHandlerContext = pQueueCallbackContext;
2896 DeviceInstallParams.InstallMsgHandler = (PSP_FILE_CALLBACK)vboxNetCfgWinPspFileCallback;
2897 fResult = SetupDiSetDeviceInstallParams (hDeviceInfo, &DeviceInfoData,
2898 &DeviceInstallParams);
2899 if (!fResult)
2900 {
2901 DWORD winEr = GetLastError();
2902 NonStandardLogFlow(("SetupDiSetDeviceInstallParams failed, winEr (%d)\n", winEr));
2903 }
2904 Assert(fResult);
2905 }
2906 else
2907 {
2908 DWORD winEr = GetLastError();
2909 NonStandardLogFlow(("SetupInitDefaultQueueCallback failed, winEr (%d)\n", winEr));
2910 }
2911 }
2912 else
2913 {
2914 DWORD winEr = GetLastError();
2915 NonStandardLogFlow(("SetupDiGetDeviceInstallParams failed, winEr (%d)\n", winEr));
2916 }
2917
2918 /* install the files first */
2919 fResult = SetupDiCallClassInstaller (DIF_INSTALLDEVICEFILES, hDeviceInfo,
2920 &DeviceInfoData);
2921 if (!fResult)
2922 SetErrBreak (("SetupDiCallClassInstaller (DIF_INSTALLDEVICEFILES) failed (0x%08X)",
2923 GetLastError()));
2924 /* get the device install parameters and disable filecopy */
2925 DeviceInstallParams.cbSize = sizeof(SP_DEVINSTALL_PARAMS);
2926 fResult = SetupDiGetDeviceInstallParams (hDeviceInfo, &DeviceInfoData,
2927 &DeviceInstallParams);
2928 if (fResult)
2929 {
2930 DeviceInstallParams.Flags |= DI_NOFILECOPY;
2931 fResult = SetupDiSetDeviceInstallParams(hDeviceInfo, &DeviceInfoData,
2932 &DeviceInstallParams);
2933 if (!fResult)
2934 SetErrBreak (("SetupDiSetDeviceInstallParams failed (0x%08X)",
2935 GetLastError()));
2936 }
2937
2938 /*
2939 * Register any device-specific co-installers for this device,
2940 */
2941 fResult = SetupDiCallClassInstaller(DIF_REGISTER_COINSTALLERS,
2942 hDeviceInfo,
2943 &DeviceInfoData);
2944 if (!fResult)
2945 SetErrBreak (("SetupDiCallClassInstaller (DIF_REGISTER_COINSTALLERS) failed (0x%08X)",
2946 GetLastError()));
2947
2948 /*
2949 * install any installer-specified interfaces.
2950 * and then do the real install
2951 */
2952 fResult = SetupDiCallClassInstaller(DIF_INSTALLINTERFACES,
2953 hDeviceInfo,
2954 &DeviceInfoData);
2955 if (!fResult)
2956 SetErrBreak (("SetupDiCallClassInstaller (DIF_INSTALLINTERFACES) failed (0x%08X)",
2957 GetLastError()));
2958
2959 fResult = SetupDiCallClassInstaller(DIF_INSTALLDEVICE,
2960 hDeviceInfo,
2961 &DeviceInfoData);
2962 if (!fResult)
2963 SetErrBreak (("SetupDiCallClassInstaller (DIF_INSTALLDEVICE) failed (0x%08X)",
2964 GetLastError()));
2965
2966 /* Figure out NetCfgInstanceId */
2967 hkey = SetupDiOpenDevRegKey(hDeviceInfo,
2968 &DeviceInfoData,
2969 DICS_FLAG_GLOBAL,
2970 0,
2971 DIREG_DRV,
2972 KEY_READ);
2973 if (hkey == INVALID_HANDLE_VALUE)
2974 SetErrBreak(("SetupDiOpenDevRegKey failed (0x%08X)", GetLastError()));
2975
2976 if (!SetupDiGetDeviceRegistryPropertyW(hDeviceInfo, &DeviceInfoData,
2977 SPDRP_FRIENDLYNAME , /* IN DWORD Property,*/
2978 NULL, /*OUT PDWORD PropertyRegDataType, OPTIONAL*/
2979 (PBYTE)DevName, /*OUT PBYTE PropertyBuffer,*/
2980 sizeof(DevName), /* IN DWORD PropertyBufferSize,*/
2981 NULL /*OUT PDWORD RequiredSize OPTIONAL*/))
2982 {
2983 int err = GetLastError();
2984 if (err != ERROR_INVALID_DATA)
2985 {
2986 SetErrBreak (("SetupDiGetDeviceRegistryProperty failed (0x%08X)",
2987 err));
2988 }
2989
2990 if (!SetupDiGetDeviceRegistryPropertyW(hDeviceInfo, &DeviceInfoData,
2991 SPDRP_DEVICEDESC, /* IN DWORD Property,*/
2992 NULL, /*OUT PDWORD PropertyRegDataType, OPTIONAL*/
2993 (PBYTE)DevName, /*OUT PBYTE PropertyBuffer,*/
2994 sizeof(DevName), /* IN DWORD PropertyBufferSize,*/
2995 NULL /*OUT PDWORD RequiredSize OPTIONAL*/
2996 ))
2997 {
2998 err = GetLastError();
2999 SetErrBreak (("SetupDiGetDeviceRegistryProperty failed (0x%08X)",
3000 err));
3001 }
3002 }
3003
3004 /* Query the instance ID; on Windows 10, the registry key may take a short
3005 * while to appear. Microsoft recommends waiting for up to 5 seconds.
3006 */
3007 for (int retries = 0; retries < 5 * 5; ++retries)
3008 {
3009 cbSize = sizeof(pWCfgGuidString);
3010 ret = RegQueryValueExW (hkey, L"NetCfgInstanceId", NULL,
3011 &dwValueType, (LPBYTE) pWCfgGuidString, &cbSize);
3012 /* As long as the return code is FILE_NOT_FOUND, sleep and retry. */
3013 if (ret == ERROR_FILE_NOT_FOUND)
3014 Sleep(200); /* 1/5 of a second. */
3015 else
3016 break;
3017 }
3018 RegCloseKey (hkey);
3019
3020 if (ret != ERROR_SUCCESS)
3021 SetErrBreak(("Querying NetCfgInstanceId failed (0x%08X)", GetLastError()));
3022 }
3023 while (0);
3024
3025 /*
3026 * cleanup
3027 */
3028 if (pQueueCallbackContext)
3029 SetupTermDefaultQueueCallback(pQueueCallbackContext);
3030
3031 if (hDeviceInfo != INVALID_HANDLE_VALUE)
3032 {
3033 /* an error has occurred, but the device is registered, we must remove it */
3034 if (ret != 0 && registered)
3035 SetupDiCallClassInstaller(DIF_REMOVE, hDeviceInfo, &DeviceInfoData);
3036
3037 found = SetupDiDeleteDeviceInfo(hDeviceInfo, &DeviceInfoData);
3038
3039 /* destroy the driver info list */
3040 if (destroyList)
3041 SetupDiDestroyDriverInfoList(hDeviceInfo, &DeviceInfoData,
3042 SPDIT_CLASSDRIVER);
3043 /* clean up the device info set */
3044 SetupDiDestroyDeviceInfoList (hDeviceInfo);
3045 }
3046
3047 /* return the network connection GUID on success */
3048 if (SUCCEEDED(hrc))
3049 {
3050 WCHAR ConnectionName[128];
3051 ULONG cbName = sizeof(ConnectionName);
3052
3053 HRESULT hr = VBoxNetCfgWinGenHostonlyConnectionName(DevName, ConnectionName, &cbName);
3054 if (SUCCEEDED(hr))
3055 hr = VBoxNetCfgWinRenameConnection(pWCfgGuidString, ConnectionName);
3056
3057 if (lppszName)
3058 {
3059 *lppszName = SysAllocString((const OLECHAR *) DevName);
3060 if (!*lppszName)
3061 {
3062 NonStandardLogFlow(("SysAllocString failed\n"));
3063 hrc = HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY);
3064 }
3065 }
3066
3067 if (pGuid)
3068 {
3069 hrc = CLSIDFromString(pWCfgGuidString, (LPCLSID)pGuid);
3070 if (FAILED(hrc))
3071 NonStandardLogFlow(("CLSIDFromString failed, hrc (0x%x)\n", hrc));
3072 }
3073
3074 INetCfg *pNetCfg = NULL;
3075 LPWSTR lpszApp = NULL;
3076 hr = VBoxNetCfgWinQueryINetCfg(&pNetCfg, TRUE, L"VirtualBox Host-Only Creation",
3077 30 * 1000, /* on Vista we often get 6to4svc.dll holding the lock, wait for 30 sec. */
3078 /* TODO: special handling for 6to4svc.dll ???, i.e. several retrieves */
3079 &lpszApp);
3080 if (hr == S_OK)
3081 {
3082 hr = vboxNetCfgWinEnumNetCfgComponents(pNetCfg,
3083 &GUID_DEVCLASS_NETSERVICE,
3084 vboxNetCfgWinAdjustHostOnlyNetworkInterfacePriority,
3085 pGuid);
3086 if (SUCCEEDED(hr))
3087 {
3088 hr = vboxNetCfgWinEnumNetCfgComponents(pNetCfg,
3089 &GUID_DEVCLASS_NETTRANS,
3090 vboxNetCfgWinAdjustHostOnlyNetworkInterfacePriority,
3091 pGuid);
3092 if (SUCCEEDED(hr))
3093 hr = vboxNetCfgWinEnumNetCfgComponents(pNetCfg,
3094 &GUID_DEVCLASS_NETCLIENT,
3095 vboxNetCfgWinAdjustHostOnlyNetworkInterfacePriority,
3096 pGuid);
3097 }
3098
3099 if (SUCCEEDED(hr))
3100 {
3101 hr = pNetCfg->Apply();
3102 }
3103 else
3104 NonStandardLogFlow(("Enumeration failed, hr 0x%x\n", hr));
3105 VBoxNetCfgWinReleaseINetCfg(pNetCfg, TRUE);
3106 }
3107 else if (hr == NETCFG_E_NO_WRITE_LOCK && lpszApp)
3108 {
3109 NonStandardLogFlow(("Application %ws is holding the lock, failed\n", lpszApp));
3110 CoTaskMemFree(lpszApp);
3111 }
3112 else
3113 NonStandardLogFlow(("VBoxNetCfgWinQueryINetCfg failed, hr 0x%x\n", hr));
3114 }
3115 return hrc;
3116}
3117
3118#undef SetErrBreak
3119
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