Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

4053 lines
117 KiB

  1. //+---------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation, 2000.
  5. //
  6. // File: I C S U P G R D . CPP
  7. //
  8. // Contents: Functions that is related to
  9. // o upgrade of ICS from Win98 SE, WinMe and Win2K to Whistler
  10. // o unattended clean install of Homenet on Whistler or later
  11. //
  12. // Date: 20-Sept-2000
  13. //
  14. //----------------------------------------------------------------------------
  15. #include "pch.h"
  16. #pragma hdrstop
  17. #include <winsock2.h>
  18. #include <netcon.h>
  19. #include <netconp.h>
  20. #include <shfolder.h>
  21. #include <hnetcfg.h>
  22. #include <atlbase.h>
  23. extern CComModule _Module; // required by atlcom.h
  24. #include <atlcom.h>
  25. #include "ncatl.h"
  26. #include "ncui.h"
  27. #include "ncstring.h"
  28. #include "ncnetcfg.h"
  29. #include "kkutils.h"
  30. #include "kkcwinf.h"
  31. #include "nslog.h"
  32. #include "afilexp.h"
  33. #include "ncras.h"
  34. #include "ncreg.h"
  35. extern "C"
  36. {
  37. // Fix IA64 conflict with Shell Macro
  38. #undef IS_ALIGNED
  39. #include <ipnat.h>
  40. }
  41. #include "IcsUpgrd.h"
  42. #include "resource.h" // need to use string resource ids
  43. /*++
  44. Routine Description:
  45. This function drives the FIcsUpgrade call by
  46. passing a init'ed and opened CWInfFile object.
  47. Arguments:
  48. None
  49. Returns: TRUE if succeeded
  50. Notes:
  51. --*/
  52. BOOL FDoIcsUpgradeIfNecessary()
  53. {
  54. BOOL fRet = FALSE;
  55. HRESULT hr;
  56. tstring strAnswerFileName;
  57. hr = HrGetAnswerFileName(&strAnswerFileName);
  58. if (S_OK == hr)
  59. {
  60. CWInfFile wifIcsAnswerFile;
  61. // initialize answer file
  62. if (wifIcsAnswerFile.Init())
  63. {
  64. if (wifIcsAnswerFile.Open(strAnswerFileName.c_str()))
  65. {
  66. TraceTag(ttidNetSetup, "calling FIcsUpgrade now...");
  67. fRet = FIcsUpgrade(&wifIcsAnswerFile);
  68. wifIcsAnswerFile.Close();
  69. }
  70. else
  71. {
  72. TraceTag(ttidNetSetup, "wifIcsAnswerFile.Open failed");
  73. }
  74. }
  75. else
  76. {
  77. TraceTag(ttidNetSetup, "wifIcsAnswerFile.Init failed");
  78. }
  79. }
  80. else
  81. {
  82. TraceTag(ttidNetSetup, "HrGetAnswerFileName failed");
  83. }
  84. return fRet;
  85. }
  86. /*++
  87. Routine Description:
  88. Perform the upgrade of
  89. o ICS from Win98 SE, WinMe and Win2K
  90. o Unattended Homenet clean install
  91. Arguments:
  92. [in] pwifAnswerFile The answer-file contains upgrade info from Win9x or
  93. Windows XP unattended clean install.
  94. Returns: TRUE if succeeded
  95. Notes:
  96. --*/
  97. BOOL FIcsUpgrade(CWInfFile* pwifAnswerFile)
  98. {
  99. HRESULT hr = S_OK;
  100. ICS_UPGRADE_SETTINGS iusSettings;
  101. BOOL fUpgradeIcsToRrasNat = FALSE;
  102. BOOL fCoUninitialize = FALSE;
  103. DefineFunctionName("FIcsUpgrade");
  104. TraceFunctionEntry(ttidNetSetup);
  105. hr = CoInitialize(NULL);
  106. if (SUCCEEDED(hr))
  107. {
  108. fCoUninitialize = TRUE;
  109. }
  110. else
  111. {
  112. if (RPC_E_CHANGED_MODE == hr)
  113. {
  114. hr = S_OK;
  115. }
  116. else
  117. {
  118. TraceTag(ttidError,
  119. "%s: CoInitialize failed: 0x%lx",
  120. __FUNCNAME__, hr);
  121. NetSetupLogStatusV(
  122. LogSevError,
  123. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  124. return FALSE;
  125. }
  126. }
  127. do
  128. {
  129. SetIcsDefaultSettings(&iusSettings);
  130. // check if we are upgrading from Windows 2000
  131. if (FNeedIcsUpgradeFromWin2K())
  132. {
  133. hr = BuildIcsUpgradeSettingsFromWin2K(&iusSettings);
  134. if (FAILED(hr))
  135. {
  136. TraceTag(ttidError,
  137. "%s: BuildIcsUpgradeSettingsFromWin2K failed: 0x%lx",
  138. __FUNCNAME__, hr);
  139. NetSetupLogStatusV(
  140. LogSevInformation,
  141. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  142. break;
  143. }
  144. // ICS installed on Win2K if we are here
  145. if (FOsIsAdvServerOrHigher())
  146. {
  147. TraceTag(ttidNetSetup,
  148. "%s: We're running on ADS/DTC SKUs, won't upgrade ICS from Win2K.",
  149. __FUNCNAME__);
  150. fUpgradeIcsToRrasNat = TRUE;
  151. break;
  152. }
  153. // OS version is less than Advanced Server if we are here
  154. }
  155. else
  156. {
  157. // we need to check if we are doing ICS upgrade from Win9x or
  158. // doing unattended Homenet install on Windows XP or later
  159. if (NULL == pwifAnswerFile)
  160. {
  161. TraceTag(ttidNetSetup,
  162. "%s: Not an ICS Upgrade from Win2K and no answer-file.",
  163. __FUNCNAME__);
  164. break;
  165. }
  166. // Try to load "Homenet" section data from Answer-File
  167. hr = LoadIcsSettingsFromAnswerFile(pwifAnswerFile, &iusSettings);
  168. if (S_FALSE == hr)
  169. {
  170. // no Homenet section
  171. hr = S_OK;
  172. break;
  173. }
  174. if (FAILED(hr))
  175. {
  176. TraceTag(ttidNetSetup,
  177. "%s: LoadIcsSettingsFromAnswerFile failed: 0x%lx",
  178. __FUNCNAME__, hr);
  179. // this may not be an error because of bug# 253074 in Win9x
  180. // OEM ICS upgrade
  181. // I'm not going to log this to setup log
  182. if (iusSettings.fEnableICS && iusSettings.fWin9xUpgrade)
  183. {
  184. hr = S_OK; // ICS enabled but no internal/external adapters.
  185. }
  186. break;
  187. }
  188. // Assert: Answer file has valid [Homenet] section.
  189. // make sure OS SKU is less than Advanced Server
  190. if (FOsIsAdvServerOrHigher())
  191. {
  192. TraceTag(ttidNetSetup,
  193. "%s: We're running on ADS/DTC SKUs, won't do any homenet unattended setup.",
  194. __FUNCNAME__);
  195. NetSetupLogStatusV(
  196. LogSevInformation,
  197. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS_ADS_DTC));
  198. break;
  199. }
  200. }
  201. // upgrade ics settings to XP or do unattended Homenet if we
  202. // can reach here
  203. hr = UpgradeIcsSettings(&iusSettings);
  204. if (S_OK != hr)
  205. {
  206. TraceTag(ttidError,
  207. "%s: UpgradeIcsSettings failed : 0x%lx",
  208. __FUNCNAME__, hr);
  209. NetSetupLogStatusV(
  210. LogSevInformation,
  211. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  212. }
  213. } while (FALSE);
  214. if (fUpgradeIcsToRrasNat)
  215. {
  216. // If we are here, ICS installed on Win2K and this is an upgrade of OS
  217. // to ADS/DTC SKUs. We won't upgrade ICS on ADS/DTC SKUs
  218. TraceTag(ttidNetSetup,
  219. "%s: We are not upgrading ICS on ADS/DTC SKUs",
  220. __FUNCNAME__);
  221. // During the "Report System Compatiblilty" stage (winnt32.exe) of this
  222. // OS upgrade process, user has already been notified that we're not
  223. // upgrading ICS on ADS/DTC SKUs.
  224. // we will still log a message to setupact.log file.
  225. NetSetupLogStatusV(
  226. LogSevInformation,
  227. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS_ADS_DTC));
  228. // we need to delete and backup the old ICS registry settings
  229. hr = BackupAndDelIcsRegistryValuesOnWin2k();
  230. if (FAILED(hr))
  231. {
  232. TraceTag(ttidNetSetup,
  233. "%s: BackupAndDelIcsRegistryValuesOnWin2k failed: 0x%lx",
  234. __FUNCNAME__, hr);
  235. }
  236. }
  237. // Free iusSettings contents if needed
  238. FreeIcsUpgradeSettings(&iusSettings);
  239. if (fCoUninitialize)
  240. {
  241. CoUninitialize();
  242. }
  243. return (S_OK == hr? TRUE : FALSE);
  244. }
  245. //--------- ICS Upgrade helpers begin----------------------------
  246. /*++
  247. Routine Description:
  248. Gets the current shared ICS connection On Win2K
  249. Arguments:
  250. [out] pConnection The shared connection information
  251. Returns: S_OK if succeeded
  252. Notes: Returns the interface connected to the internet
  253. --*/
  254. HRESULT GetSharedConnectionOnWin2k(LPRASSHARECONN pConnection)
  255. {
  256. HKEY hKey = NULL;
  257. HRESULT hr = S_OK;
  258. DefineFunctionName("GetSharedConnectionOnWin2k");
  259. TraceFunctionEntry(ttidNetSetup);
  260. Assert(pConnection);
  261. hr = HrRegOpenKeyEx(HKEY_LOCAL_MACHINE, c_wszRegKeySharedAccessParams, KEY_ALL_ACCESS, &hKey);
  262. if (FAILED(hr))
  263. {
  264. TraceTag(ttidError,
  265. "%s: HrRegOpenKeyEx failed: 0x%lx",
  266. __FUNCNAME__, hr);
  267. return hr;
  268. }
  269. Assert(hKey);
  270. DWORD dwType = REG_BINARY;
  271. PBYTE pByte = NULL;
  272. DWORD dwValueSize = 0;
  273. hr = HrRegQueryValueWithAlloc(hKey, c_wszRegValSharedConnection, &dwType, &pByte, &dwValueSize);
  274. if (FAILED(hr))
  275. {
  276. TraceTag(ttidError,
  277. "%s: HrRegQueryValueWithAlloc %S failed: 0x%lx",
  278. __FUNCNAME__, c_wszRegValSharedConnection, hr);
  279. goto Exit;
  280. }
  281. if (dwValueSize > sizeof(RASSHARECONN))
  282. {
  283. TraceTag(ttidError,
  284. "%s: RASSHARECONN size too big: 0x%lx bytes",
  285. __FUNCNAME__, dwValueSize);
  286. hr = E_FAIL;
  287. goto Exit;
  288. }
  289. // transfer value
  290. memcpy(pConnection, pByte, dwValueSize);
  291. Exit:
  292. // Close the key, if opened
  293. RegSafeCloseKey(hKey);
  294. // Clean up the registry buffer
  295. if (pByte)
  296. {
  297. MemFree(pByte);
  298. }
  299. return hr;
  300. }
  301. /*++
  302. Routine Description:
  303. Gets the current private LAN connection on Win2K
  304. Arguments:
  305. [out] pGuid The private LAN information
  306. Returns: S_OK if succeeded
  307. Notes: Returns the interface connected to the private LAN
  308. On Win2K, there is only 1 private LAN connection
  309. --*/
  310. HRESULT GetSharedPrivateLanOnWin2K(GUID *pGuid)
  311. {
  312. HKEY hKey = NULL;
  313. HRESULT hr = S_OK;
  314. DefineFunctionName("GetSharedPrivateLanOnWin2K");
  315. TraceFunctionEntry(ttidNetSetup);
  316. Assert(pGuid);
  317. hr = HrRegOpenKeyEx(HKEY_LOCAL_MACHINE, c_wszRegKeySharedAccessParams, KEY_ALL_ACCESS, &hKey);
  318. if (FAILED(hr))
  319. {
  320. TraceTag(ttidError,
  321. "%s: HrRegOpenKeyEx failed: 0x%lx",
  322. __FUNCNAME__, hr);
  323. return hr;
  324. }
  325. Assert(hKey);
  326. DWORD dwType = REG_SZ;
  327. PBYTE pByte = NULL;
  328. DWORD dwValueSize = 0;
  329. hr = HrRegQueryValueWithAlloc(hKey, c_wszRegValSharedPrivateLan, &dwType, &pByte, &dwValueSize);
  330. if (FAILED(hr))
  331. {
  332. TraceTag(ttidError,
  333. "%s: HrRegQueryValueWithAlloc %S failed: 0x%lx",
  334. __FUNCNAME__, c_wszRegValSharedPrivateLan, hr);
  335. goto Exit;
  336. }
  337. // SharedPrivateLan is of type REG_SZ, pByte should have included
  338. // the terminating null character.
  339. hr = CLSIDFromString(reinterpret_cast<PWCHAR>(pByte), pGuid);
  340. Exit:
  341. // Close the key, if opened
  342. RegSafeCloseKey(hKey);
  343. // Clean up the registry buffer
  344. if (pByte)
  345. {
  346. MemFree(pByte);
  347. }
  348. return hr;
  349. }
  350. /*++
  351. Routine Description:
  352. Delete and backup SharedConnection and SharedPrivateLan registry values
  353. Arguments: none
  354. Returns: S_OK if succeeded
  355. Notes:
  356. --*/
  357. HRESULT BackupAndDelIcsRegistryValuesOnWin2k()
  358. {
  359. HKEY hKey = NULL;
  360. HRESULT hr = S_OK;
  361. DefineFunctionName("BackupAndDelIcsRegistryValuesOnWin2k");
  362. TraceFunctionEntry(ttidNetSetup);
  363. hr = HrRegOpenKeyEx(HKEY_LOCAL_MACHINE, c_wszRegKeySharedAccessParams, KEY_ALL_ACCESS, &hKey);
  364. if (FAILED(hr))
  365. {
  366. TraceTag(ttidError,
  367. "%s: HrRegOpenKeyEx failed: 0x%lx",
  368. __FUNCNAME__, hr);
  369. return hr;
  370. }
  371. Assert(hKey);
  372. DWORD dwType = REG_BINARY;
  373. PBYTE pByte = NULL;
  374. DWORD dwValueSize = 0;
  375. // backup SharedConnection
  376. hr = HrRegQueryValueWithAlloc(hKey, c_wszRegValSharedConnection, &dwType, &pByte, &dwValueSize);
  377. if (FAILED(hr))
  378. {
  379. TraceTag(ttidError,
  380. "%s: HrRegQueryValueWithAlloc %S failed: 0x%lx",
  381. __FUNCNAME__, c_wszRegValSharedConnection, hr);
  382. goto Exit;
  383. }
  384. hr = HrRegSetValueEx(hKey, c_wszRegValBackupSharedConnection, dwType, pByte, dwValueSize);
  385. if (FAILED(hr))
  386. {
  387. TraceTag(ttidError,
  388. "%s: HrRegSetValueEx %S failed: 0x%lx",
  389. __FUNCNAME__, c_wszRegValBackupSharedConnection, hr);
  390. goto Exit;
  391. }
  392. if (pByte)
  393. {
  394. MemFree(pByte);
  395. }
  396. // Reset values
  397. pByte = NULL;
  398. dwValueSize = 0;
  399. dwType = REG_SZ;
  400. // backup SharedPrivateLan
  401. hr = HrRegQueryValueWithAlloc(hKey, c_wszRegValSharedPrivateLan, &dwType, &pByte, &dwValueSize);
  402. if (FAILED(hr))
  403. {
  404. TraceTag(ttidError,
  405. "%s: HrRegQueryValueWithAlloc %S failed: 0x%lx",
  406. __FUNCNAME__, c_wszRegValSharedPrivateLan, hr);
  407. goto Exit;
  408. }
  409. hr = HrRegSetValueEx(hKey, c_wszRegValBackupSharedPrivateLan, dwType, pByte, dwValueSize);
  410. if (FAILED(hr))
  411. {
  412. TraceTag(ttidError,
  413. "%s: HrRegSetValueEx %S failed: 0x%lx",
  414. __FUNCNAME__, c_wszRegValBackupSharedPrivateLan, hr);
  415. goto Exit;
  416. }
  417. Exit:
  418. // Delete SharedConnection and ShardPrivateLan named values, ignore any errors
  419. HrRegDeleteValue(hKey, c_wszRegValSharedConnection);
  420. HrRegDeleteValue(hKey, c_wszRegValSharedPrivateLan);
  421. RegSafeCloseKey(hKey);
  422. if (pByte)
  423. {
  424. MemFree(pByte);
  425. }
  426. return hr;
  427. }
  428. /*++
  429. Routine Description:
  430. Gets the RAS phonebook directory
  431. Arguments:
  432. [out] pszPathBuf The phone book path
  433. Returns: S_OK if succeeded
  434. Notes:
  435. --*/
  436. HRESULT GetPhonebookDirectory(WCHAR* pwszPathBuf)
  437. // Loads caller's 'pszPathBuf' (should have length MAX_PATH + 1) with the
  438. // path to the directory containing phonebook files for the given mode,
  439. // e.g. c:\nt\system32\ras\" for mode PBM_Router. Note the
  440. // trailing backslash.
  441. //
  442. // Returns true if successful, false otherwise. Caller is guaranteed that
  443. // an 8.3 filename will fit on the end of the directory without exceeding
  444. // MAX_PATH.
  445. //
  446. {
  447. DefineFunctionName("GetPhonebookDirectory");
  448. TraceFunctionEntry(ttidNetSetup);
  449. HANDLE hToken = NULL;
  450. PFNSHGETFOLDERPATHW pfnSHGetFolderPathW;
  451. HINSTANCE Hinstance;
  452. //
  453. // Load ShFolder.dll, which contains the 'SHGetFolderPath' entrypoint
  454. // that we will use to get the Application data for all users path
  455. if (!(Hinstance = LoadLibraryW(c_wszShFolder)) ||
  456. !(pfnSHGetFolderPathW = (PFNSHGETFOLDERPATHW)
  457. GetProcAddress(Hinstance, c_szSHGetFolderPathW)))
  458. {
  459. if (Hinstance)
  460. {
  461. FreeLibrary(Hinstance);
  462. }
  463. return E_FAIL;
  464. }
  465. if ((OpenThreadToken(
  466. GetCurrentThread(),
  467. TOKEN_QUERY | TOKEN_IMPERSONATE,
  468. TRUE,
  469. &hToken)
  470. || OpenProcessToken(
  471. GetCurrentProcess(),
  472. TOKEN_QUERY | TOKEN_IMPERSONATE,
  473. &hToken)))
  474. {
  475. HRESULT hr;
  476. INT csidl = CSIDL_COMMON_APPDATA;
  477. hr = pfnSHGetFolderPathW(NULL, csidl, hToken, 0, pwszPathBuf);
  478. if (SUCCEEDED(hr))
  479. {
  480. if(lstrlen(pwszPathBuf) <=
  481. (MAX_PATH -
  482. (lstrlen(c_wszPhoneBookPath))))
  483. {
  484. lstrcat(pwszPathBuf, c_wszPhoneBookPath);
  485. CloseHandle(hToken);
  486. FreeLibrary(Hinstance);
  487. return S_OK;
  488. }
  489. }
  490. else
  491. {
  492. TraceTag(ttidError, "%s: SHGetFolderPath failed: 0x%lx",
  493. __FUNCNAME__, hr);
  494. }
  495. CloseHandle(hToken);
  496. }
  497. FreeLibrary(Hinstance);
  498. return E_FAIL;
  499. }
  500. /*++
  501. Routine Description:
  502. CSharedAccessServer::CSharedAccessServer() Constructor
  503. Arguments: none
  504. Return Value: none
  505. Notes:
  506. --*/
  507. CSharedAccessServer::CSharedAccessServer()
  508. {
  509. m_wPort = 0;
  510. m_wInternalPort = 0;
  511. m_bBuiltIn = FALSE;
  512. m_bSelected = FALSE;
  513. m_dwSectionNum = 0;
  514. };
  515. /*++
  516. Routine Description:
  517. Returns the current list of Shared Access Servers
  518. Arguments:
  519. [in,out] lstSharedAccessServers List of SharedAccessServers
  520. Returns: S_OK if succeeded
  521. Notes: Reads the sharedaccess.ini phonebook file for server mappings.
  522. --*/
  523. HRESULT GetServerMappings(list<CSharedAccessServer> &lstSharedAccessServers)
  524. {
  525. WCHAR wszPathBuf[MAX_PATH + 1];
  526. HRESULT hr = GetPhonebookDirectory(wszPathBuf);
  527. if FAILED(hr)
  528. return hr;
  529. tstring strIniFile; // full pathname to sharedaccess.ini
  530. strIniFile = wszPathBuf;
  531. strIniFile += c_wszFileSharedAccess;
  532. const DWORD dwBufSize = 32768; // limit the section buffer size to the same value as Win9x max. buffer size
  533. PWCHAR wszLargeReturnedString = new WCHAR[dwBufSize];
  534. if (NULL == wszLargeReturnedString)
  535. {
  536. return E_OUTOFMEMORY;
  537. }
  538. DWORD dwResult;
  539. dwResult = GetPrivateProfileSection(c_wszContentsServer, wszLargeReturnedString, dwBufSize-1, strIniFile.c_str());
  540. if (dwResult)
  541. {
  542. PWCHAR wszEnd = wszLargeReturnedString + dwResult;
  543. for (PWCHAR wszServer = wszLargeReturnedString; (*wszServer) && (wszServer < wszEnd); wszServer += wcslen(wszServer) + 1)
  544. {
  545. WCHAR wszReturnedString[MAX_PATH];
  546. PWCHAR wszSectionNum = NULL;
  547. PWCHAR wszEnabled;
  548. // prefix bug# 295834
  549. wszSectionNum = new WCHAR[wcslen(wszServer)+1];
  550. if (NULL == wszSectionNum)
  551. {
  552. delete [] wszLargeReturnedString;
  553. return E_OUTOFMEMORY;
  554. }
  555. wcscpy(wszSectionNum, wszServer);
  556. PWCHAR wszAssign = wcschr(wszSectionNum, L'=');
  557. if (wszAssign)
  558. {
  559. *wszAssign = NULL;
  560. wszEnabled = wszAssign + 1;
  561. }
  562. else
  563. {
  564. wszEnabled = FALSE;
  565. }
  566. tstring strSectionName;
  567. strSectionName = c_wszServerPrefix; // prefix bug# 295835
  568. strSectionName += wszSectionNum; // prefix bug# 295836
  569. CSharedAccessServer SharedAccessServer;
  570. SharedAccessServer.m_dwSectionNum = _wtoi(wszSectionNum);
  571. SharedAccessServer.m_bSelected = _wtoi(wszEnabled);
  572. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszInternalName, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  573. if (dwResult)
  574. {
  575. SharedAccessServer.m_szInternalName = wszReturnedString;
  576. }
  577. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszTitle, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  578. if (dwResult)
  579. {
  580. SharedAccessServer.m_szTitle = wszReturnedString;
  581. }
  582. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszInternalPort, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  583. if (dwResult)
  584. {
  585. SharedAccessServer.m_wInternalPort = static_cast<USHORT>(_wtoi(wszReturnedString));
  586. }
  587. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszPort, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  588. if (dwResult)
  589. {
  590. SharedAccessServer.m_wPort = static_cast<USHORT>(_wtoi(wszReturnedString));
  591. }
  592. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszReservedAddress, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  593. if (dwResult)
  594. {
  595. SharedAccessServer.m_szReservedAddress = wszReturnedString;
  596. }
  597. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszProtocol, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  598. if (dwResult)
  599. {
  600. SharedAccessServer.m_szProtocol = wszReturnedString;
  601. }
  602. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszBuiltIn, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  603. if (dwResult)
  604. {
  605. SharedAccessServer.m_bBuiltIn = static_cast<USHORT>(_wtoi(wszReturnedString));
  606. }
  607. lstSharedAccessServers.insert(lstSharedAccessServers.end(), SharedAccessServer);
  608. delete [] wszSectionNum;
  609. wszSectionNum = NULL;
  610. }
  611. }
  612. delete [] wszLargeReturnedString;
  613. return S_OK;
  614. }
  615. /*++
  616. Routine Description:
  617. Returns the current list of Shared Access Applications
  618. Arguments:
  619. [in,out] lstSharedAccessApplications List of SharedAccessApplications
  620. Returns: S_OK if succeeded
  621. Notes: Reads the sharedaccess.ini phonebook file for application mappings.
  622. --*/
  623. HRESULT GetApplicationMappings(list<CSharedAccessApplication> &lstSharedAccessApplications)
  624. {
  625. WCHAR wszPathBuf[MAX_PATH + 1];
  626. HRESULT hr = GetPhonebookDirectory(wszPathBuf);
  627. if FAILED(hr)
  628. return hr;
  629. tstring strIniFile; // full pathname to sharedaccess.ini
  630. strIniFile = wszPathBuf;
  631. strIniFile += c_wszFileSharedAccess;
  632. const DWORD dwBufSize = 32768; // limit the section buffer size to the same value as Win9x max. buffer size
  633. PWCHAR wszLargeReturnedString = new WCHAR[dwBufSize];
  634. if (NULL == wszLargeReturnedString)
  635. {
  636. return E_OUTOFMEMORY;
  637. }
  638. DWORD dwResult;
  639. dwResult = GetPrivateProfileSection(c_wszContentsApplication, wszLargeReturnedString, dwBufSize-1, strIniFile.c_str());
  640. if (dwResult)
  641. {
  642. PWCHAR wszEnd = wszLargeReturnedString + dwResult;
  643. for (PWCHAR wszApplication = wszLargeReturnedString; (*wszApplication) && (wszApplication < wszEnd); wszApplication += wcslen(wszApplication) + 1)
  644. {
  645. WCHAR wszReturnedString[MAX_PATH];
  646. PWCHAR wszSectionNum = NULL;
  647. PWCHAR wszEnabled;
  648. // prefix bug# 295838
  649. wszSectionNum = new WCHAR[wcslen(wszApplication)+1];
  650. if (NULL == wszSectionNum)
  651. {
  652. delete [] wszLargeReturnedString;
  653. return E_OUTOFMEMORY;
  654. }
  655. wcscpy(wszSectionNum, wszApplication);
  656. PWCHAR wszAssign = wcschr(wszSectionNum, L'=');
  657. if (wszAssign)
  658. {
  659. *wszAssign = NULL;
  660. wszEnabled = wszAssign + 1;
  661. }
  662. else
  663. {
  664. wszEnabled = FALSE;
  665. }
  666. tstring strSectionName;
  667. strSectionName = c_wszApplicationPrefix; // prefix bug# 295839
  668. strSectionName += wszSectionNum; // prefix bug# 295840
  669. CSharedAccessApplication SharedAccessApplication;
  670. SharedAccessApplication.m_dwSectionNum = _wtoi(wszSectionNum);
  671. SharedAccessApplication.m_bSelected = _wtoi(wszEnabled);
  672. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszTitle, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  673. if (dwResult)
  674. {
  675. SharedAccessApplication.m_szTitle = wszReturnedString;
  676. }
  677. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszProtocol, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  678. if (dwResult)
  679. {
  680. SharedAccessApplication.m_szProtocol = wszReturnedString;
  681. }
  682. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszPort, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  683. if (dwResult)
  684. {
  685. SharedAccessApplication.m_wPort = static_cast<USHORT>(_wtoi(wszReturnedString));
  686. }
  687. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszTcpResponseList, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  688. if (dwResult)
  689. {
  690. SharedAccessApplication.m_szTcpResponseList = wszReturnedString;
  691. }
  692. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszUdpResponseList, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  693. if (dwResult)
  694. {
  695. SharedAccessApplication.m_szUdpResponseList = wszReturnedString;
  696. }
  697. dwResult = GetPrivateProfileString(strSectionName.c_str(), c_wszBuiltIn, L"", wszReturnedString, MAX_PATH, strIniFile.c_str());
  698. if (dwResult)
  699. {
  700. SharedAccessApplication.m_bBuiltIn = static_cast<USHORT>(_wtoi(wszReturnedString));
  701. }
  702. lstSharedAccessApplications.insert(lstSharedAccessApplications.end(), SharedAccessApplication);
  703. delete [] wszSectionNum;
  704. wszSectionNum = NULL;
  705. }
  706. }
  707. delete [] wszLargeReturnedString;
  708. return S_OK;
  709. }
  710. /*++
  711. Routine Description:
  712. Add ResponseList string into vector of HNET_RESPONSE_RANGE
  713. Arguments:
  714. [in] rssaAppProt - ref. to the CSharedAccessApplication obj.
  715. [in] ucProtocol - NAT_PROTOCOL_TCP or NAT_PROTOCOL_UDP
  716. [out] rvecResponseRange - add new HNET_RESPONSE_RANGE(s) to this vector
  717. Returns: S_OK if succeeded
  718. Notes: In case of failure, contents of rvecResponseRange will be erased.
  719. Design by contract: ucProtocol could only be NAT_PROTOCOL_TCP or
  720. NAT_PROTOCOL_UDP
  721. --*/
  722. HRESULT AddResponseStringToVector(
  723. CSharedAccessApplication& rsaaAppProt,
  724. UCHAR ucProtocol,
  725. vector<HNET_RESPONSE_RANGE>& rvecResponseRange // vector of response range
  726. )
  727. {
  728. WCHAR* Endp;
  729. USHORT EndPort; // the End port # in the range of response port #
  730. USHORT StartPort; // the starting port # in the range of response port #
  731. const WCHAR* pwszValue = NULL; // the tcp or udp ResponseString to be converted.
  732. // (e.g "1300-1310,1450" or "1245")
  733. HNET_RESPONSE_RANGE hnrrResponseRange; // the response range
  734. HRESULT hr = S_OK;
  735. // select the ResponseList to be added
  736. if (NAT_PROTOCOL_TCP == ucProtocol)
  737. {
  738. pwszValue = rsaaAppProt.m_szTcpResponseList.c_str();
  739. }
  740. else if (NAT_PROTOCOL_UDP == ucProtocol)
  741. {
  742. pwszValue = rsaaAppProt.m_szUdpResponseList.c_str();
  743. }
  744. if (NULL == pwszValue)
  745. {
  746. // no operation
  747. goto Exit;
  748. }
  749. while (*pwszValue)
  750. {
  751. // Read either a single port or a range of ports.
  752. if (!(StartPort = (USHORT)wcstoul(pwszValue, &Endp, 10)))
  753. {
  754. hr = E_FAIL;
  755. goto Exit;
  756. }
  757. else if (!*Endp || *Endp == L',')
  758. {
  759. EndPort = StartPort;
  760. pwszValue = (!*Endp ? Endp : Endp + 1);
  761. }
  762. else if (*Endp != L'-')
  763. {
  764. hr = E_FAIL;
  765. goto Exit;
  766. }
  767. else if (!(EndPort = (USHORT)wcstoul(++Endp, (WCHAR**)&pwszValue, 10)))
  768. {
  769. hr = E_FAIL;
  770. goto Exit;
  771. }
  772. else if (EndPort < StartPort)
  773. {
  774. hr = E_FAIL;
  775. goto Exit;
  776. }
  777. else if (*pwszValue && *pwszValue++ != L',')
  778. {
  779. hr = E_FAIL;
  780. goto Exit;
  781. }
  782. // transfer values
  783. hnrrResponseRange.ucIPProtocol = ucProtocol;
  784. hnrrResponseRange.usStartPort = HTONS(StartPort);
  785. hnrrResponseRange.usEndPort = HTONS(EndPort);
  786. rvecResponseRange.push_back(hnrrResponseRange);
  787. }
  788. Exit:
  789. if (FAILED(hr))
  790. {
  791. rvecResponseRange.erase(rvecResponseRange.begin(), rvecResponseRange.end());
  792. }
  793. return hr;
  794. }
  795. /*++
  796. Routine Description:
  797. Converts ResponseList string into array of HNET_RESPONSE_RANGE
  798. Arguments:
  799. [in] rssaAppProt - ref. to the CSharedAccessApplication obj.
  800. [out] puscResponse - number of HNET_RESPONSE_RANGE converted
  801. [out] pphnrrResponseRange - array of HNET_RESPONSE_RANGE converted
  802. Returns: S_OK if succeeded
  803. Notes: User is responsible to free pphnrrResponseRange by
  804. "delete [] (BYTE*)(*pphnrrResponseRange)" if *puscResponse > 0
  805. --*/
  806. HRESULT PutResponseStringIntoArray(
  807. CSharedAccessApplication& rsaaAppProt, // Application Protocol
  808. USHORT* puscResponse,
  809. HNET_RESPONSE_RANGE** pphnrrResponseRange
  810. )
  811. {
  812. HRESULT hr;
  813. DWORD dwIdx = 0;
  814. Assert(pphnrrResponseRange != NULL);
  815. Assert(puscResponse != NULL);
  816. *pphnrrResponseRange = NULL;
  817. *puscResponse = 0;
  818. vector<HNET_RESPONSE_RANGE> vecResponseRange; // vector of response range
  819. if (! rsaaAppProt.m_szTcpResponseList.empty())
  820. {
  821. //TcpResponseList is not empty
  822. hr = AddResponseStringToVector(rsaaAppProt, NAT_PROTOCOL_TCP, vecResponseRange);
  823. if (FAILED(hr))
  824. {
  825. return hr;
  826. }
  827. }
  828. if (! rsaaAppProt.m_szUdpResponseList.empty())
  829. {
  830. // UdpResponseList is not empty
  831. hr = AddResponseStringToVector(rsaaAppProt, NAT_PROTOCOL_UDP, vecResponseRange);
  832. if (FAILED(hr))
  833. {
  834. return hr;
  835. }
  836. }
  837. HNET_RESPONSE_RANGE* phnrrResponseRange = NULL;
  838. USHORT uscResponseRange = (USHORT) vecResponseRange.size();
  839. if (1 > uscResponseRange)
  840. {
  841. // we should have at least 1 ReponseRange
  842. hr = E_FAIL;
  843. goto Exit;
  844. }
  845. phnrrResponseRange = (HNET_RESPONSE_RANGE*)
  846. new BYTE[sizeof(HNET_RESPONSE_RANGE) * uscResponseRange];
  847. if (phnrrResponseRange == NULL)
  848. {
  849. hr = E_OUTOFMEMORY;
  850. goto Exit;
  851. }
  852. for (vector<HNET_RESPONSE_RANGE>::iterator iter = vecResponseRange.begin();
  853. iter != vecResponseRange.end();
  854. ++iter, ++dwIdx)
  855. {
  856. phnrrResponseRange[dwIdx] = *iter;
  857. }
  858. // transfer values
  859. *pphnrrResponseRange = phnrrResponseRange;
  860. *puscResponse = uscResponseRange;
  861. Exit:
  862. vecResponseRange.erase(vecResponseRange.begin(), vecResponseRange.end());
  863. return hr;
  864. }
  865. /*++
  866. Routine Description:
  867. Check if we need to upgrade ICS from Win2K
  868. Returns: TRUE or FALSE
  869. Notes:
  870. 1. Check if registry value names (SharedConnection and
  871. SharedPrivateLan) exist.
  872. 2. Check if sharedaccess.ini file exists
  873. After upgrade, the registry names should be deleted because
  874. ICS will use settings from WMI repository.
  875. We don't Check if the IPNATHLP service exists, because it is
  876. removed in ADS and DTC SKUs
  877. --*/
  878. BOOL FNeedIcsUpgradeFromWin2K()
  879. {
  880. // 1. Check if registry key values exist
  881. HKEY hKey = NULL;
  882. HRESULT hr = S_OK;
  883. hr = HrRegOpenKeyEx(HKEY_LOCAL_MACHINE, c_wszRegKeySharedAccessParams, KEY_ALL_ACCESS, &hKey);
  884. if (FAILED(hr))
  885. {
  886. return FALSE;
  887. }
  888. Assert(hKey);
  889. DWORD dwType = REG_BINARY;
  890. PBYTE pByte = NULL;
  891. DWORD dwValueSize = 0;
  892. hr = HrRegQueryValueWithAlloc(hKey, c_wszRegValSharedConnection, &dwType, &pByte, &dwValueSize);
  893. if (FAILED(hr))
  894. {
  895. RegSafeCloseKey(hKey);
  896. return FALSE;
  897. }
  898. if (pByte)
  899. {
  900. MemFree(pByte);
  901. }
  902. // Reset values
  903. pByte = NULL;
  904. dwValueSize = 0;
  905. dwType = REG_SZ;
  906. hr = HrRegQueryValueWithAlloc(hKey, c_wszRegValSharedPrivateLan, &dwType, &pByte, &dwValueSize);
  907. if (FAILED(hr))
  908. {
  909. RegSafeCloseKey(hKey);
  910. return FALSE;
  911. }
  912. if (pByte)
  913. {
  914. MemFree(pByte);
  915. }
  916. // 2. Check if sharedaccess.ini file exists
  917. WCHAR wszPathBuf[MAX_PATH + 1];
  918. hr = GetPhonebookDirectory(wszPathBuf);
  919. if FAILED(hr)
  920. return FALSE;
  921. tstring strShareAccessFile;
  922. strShareAccessFile = wszPathBuf;
  923. strShareAccessFile += c_wszFileSharedAccess;
  924. HANDLE hOpenFile =
  925. CreateFileW(strShareAccessFile.c_str(), // open file
  926. GENERIC_READ, // open for reading
  927. FILE_SHARE_READ, // share for reading
  928. NULL, // no security
  929. OPEN_EXISTING, // existing file only
  930. FILE_ATTRIBUTE_NORMAL, // normal file
  931. NULL); // no attr. template
  932. if (INVALID_HANDLE_VALUE == hOpenFile)
  933. {
  934. return FALSE;
  935. }
  936. CloseHandle(hOpenFile);
  937. return TRUE;
  938. }
  939. /*++
  940. Routine Description:
  941. Upgrade ICS Settings
  942. Arguments:
  943. [in] pIcsUpgradeSettings - the ICS UPGRADE SETTINGS
  944. Returns: standard HRESULT
  945. Notes:
  946. --*/
  947. HRESULT UpgradeIcsSettings(ICS_UPGRADE_SETTINGS * pIcsUpgrdSettings)
  948. {
  949. HRESULT hr;
  950. DefineFunctionName("UpgradeIcsSettings");
  951. TraceFunctionEntry(ttidNetSetup);
  952. CIcsUpgrade IcsUpgradeObj;
  953. hr = IcsUpgradeObj.Init(pIcsUpgrdSettings);
  954. if (S_OK != hr)
  955. {
  956. TraceTag(ttidError,
  957. "%s: IcsUpgradeObj.Init failed: 0x%lx",
  958. __FUNCNAME__, hr);
  959. return hr;
  960. }
  961. return IcsUpgradeObj.StartUpgrade();
  962. }
  963. /*++
  964. Routine Description:
  965. Build the ICS_UPGRADE_SETTINGS struct from Win2K ICS settings
  966. Arguments:
  967. [in] pIcsUpgrdSettings - pointer to the ICS_UPGRADE_SETTINGS
  968. Return Value: standard HRESULT
  969. Notes:
  970. --*/
  971. HRESULT BuildIcsUpgradeSettingsFromWin2K(ICS_UPGRADE_SETTINGS* pIcsUpgrdSettings)
  972. {
  973. HRESULT hr;
  974. DefineFunctionName("BuildIcsUpgradeSettingsFromWin2K");
  975. TraceFunctionEntry(ttidNetSetup);
  976. Assert(pIcsUpgrdSettings != NULL);
  977. GUID guid;
  978. hr = GetSharedPrivateLanOnWin2K(&guid);
  979. if (FAILED(hr))
  980. {
  981. TraceTag(ttidError, "%s: GetSharedPrivateLanOnWin2K failed: 0x%lx",
  982. __FUNCNAME__, hr);
  983. return hr;
  984. }
  985. hr = GetSharedConnectionOnWin2k(&(pIcsUpgrdSettings->rscExternal));
  986. if (FAILED(hr))
  987. {
  988. TraceTag(ttidError, "%s: GetSharedConnectionOnWin2k failed: 0x%lx",
  989. __FUNCNAME__, hr);
  990. return hr;
  991. }
  992. hr = GetServerMappings(pIcsUpgrdSettings->listSvrPortmappings);
  993. if (FAILED(hr))
  994. {
  995. TraceTag(ttidError, "%s: Ignore error from GetServerMappings : 0x%lx",
  996. __FUNCNAME__, hr);
  997. }
  998. hr = GetApplicationMappings(pIcsUpgrdSettings->listAppPortmappings);
  999. if (FAILED(hr))
  1000. {
  1001. TraceTag(ttidError, "%s: Ignore error from GetApplicationMappings : 0x%lx",
  1002. __FUNCNAME__, hr);
  1003. }
  1004. // transfer values
  1005. pIcsUpgrdSettings->guidInternal = guid;
  1006. pIcsUpgrdSettings->fInternalAdapterFound = TRUE;
  1007. pIcsUpgrdSettings->fDialOnDemand = TRUE;
  1008. pIcsUpgrdSettings->fEnableICS = TRUE;
  1009. pIcsUpgrdSettings->fShowTrayIcon = TRUE;
  1010. pIcsUpgrdSettings->fWin2KUpgrade = TRUE;
  1011. return S_OK;
  1012. }
  1013. /*++
  1014. Routine Description:
  1015. Free resources inside the ICS_UPGRADE_SETTINGS
  1016. Arguments:
  1017. [in] pIcsUpgrdSettings - pointer to the ICS_UPGRADE_SETTINGS
  1018. Return Value: none
  1019. Note.
  1020. --*/
  1021. void FreeIcsUpgradeSettings(ICS_UPGRADE_SETTINGS* pIcsUpgrdSettings)
  1022. {
  1023. Assert(pIcsUpgrdSettings != NULL);
  1024. list<GUID>& rlistPersonalFirewall = pIcsUpgrdSettings->listPersonalFirewall;
  1025. list<GUID>& rlistBridge = pIcsUpgrdSettings->listBridge;
  1026. list<CSharedAccessServer>& rlistSvrPortmappings = pIcsUpgrdSettings->listSvrPortmappings;
  1027. list<CSharedAccessApplication>& rlistAppPortmappings = pIcsUpgrdSettings->listAppPortmappings;
  1028. rlistPersonalFirewall.erase(rlistPersonalFirewall.begin(), rlistPersonalFirewall.end());
  1029. rlistBridge.erase(rlistBridge.begin(), rlistBridge.end());
  1030. rlistSvrPortmappings.erase(rlistSvrPortmappings.begin(), rlistSvrPortmappings.end());
  1031. rlistAppPortmappings.erase(rlistAppPortmappings.begin(), rlistAppPortmappings.end());
  1032. }
  1033. /*++
  1034. Routine Description:
  1035. convert a list of adapter strings into a list of its corresponding GUID
  1036. Arguments:
  1037. [in] rslAdapters - reference to the list of adapter strings
  1038. [in out] rlistGuid - a list of the converted GUIDs
  1039. Return Value:
  1040. S_OK - if either all of the adapter strings converted to its corresponding
  1041. GUIDs or the input is an empty list of adapter strings.
  1042. S_FALSE - if it can convert at least one of the adaper strings.
  1043. E_FAIL - if none of the adapter strings could be converted to its
  1044. corresponding GUIDs
  1045. --*/
  1046. HRESULT ConvertAdapterStringListToGuidList(IN TStringList& rslAdapters,
  1047. IN OUT list<GUID>& rlistGuid)
  1048. {
  1049. HRESULT hr = S_OK;
  1050. GUID guidTemp = {0};
  1051. tstring* pstrTemp = NULL;
  1052. TStringList::iterator iter;
  1053. if (rslAdapters.empty())
  1054. return hr;
  1055. // init the [in out] parameter first
  1056. rlistGuid.erase(rlistGuid.begin(), rlistGuid.end());
  1057. for (iter = rslAdapters.begin(); iter != rslAdapters.end(); ++iter)
  1058. {
  1059. pstrTemp = *iter;
  1060. Assert(pstrTemp);
  1061. if (!FGetInstanceGuidOfComponentFromAnswerFileMap(pstrTemp->c_str(), &guidTemp))
  1062. {
  1063. TraceTag(ttidError,
  1064. "FGetInstanceGuidOfComponentFromAnswerFileMap failed to match GUID for adapter %S",
  1065. pstrTemp->c_str());
  1066. NetSetupLogStatusV(
  1067. LogSevInformation,
  1068. SzLoadIds (IDS_TXT_E_ADAPTER_TO_GUID),
  1069. pstrTemp->c_str());
  1070. }
  1071. else
  1072. {
  1073. rlistGuid.push_back(guidTemp);
  1074. }
  1075. }
  1076. if (rlistGuid.empty())
  1077. {
  1078. hr = E_FAIL;
  1079. }
  1080. if (rlistGuid.size() < rslAdapters.size())
  1081. {
  1082. hr = S_FALSE;
  1083. }
  1084. return hr;
  1085. }
  1086. /*++
  1087. Routine Description:
  1088. remove duplicates in a TStringList
  1089. Arguments:
  1090. [in, out] sl - the string list contains duplicates to be removed
  1091. Return Value: none
  1092. --*/
  1093. void RemoveDupsInTStringList(IN OUT TStringList& sl)
  1094. {
  1095. for (list<tstring*>::iterator i = sl.begin(); i != sl.end(); ++i)
  1096. {
  1097. list<tstring*>::iterator tmpI = i;
  1098. for (list<tstring*>::iterator j = ++tmpI; j != sl.end(); ++j)
  1099. {
  1100. list<tstring*>::iterator tmpJ = j;
  1101. list<tstring*>::iterator k = --tmpJ;
  1102. if (!_wcsicmp((*i)->c_str(), (*j)->c_str()))
  1103. {
  1104. TraceTag(ttidNetSetup,
  1105. "RemoveDupsInTStringList: found duplicated string %S",
  1106. (*i)->c_str());
  1107. delete *j; // free the resource
  1108. sl.erase(j); // erase the list item
  1109. j = k; // adjust the iterator to the previous one
  1110. }
  1111. }
  1112. }
  1113. }
  1114. /*++
  1115. Routine Description:
  1116. remove items in slDest if the item is also in slSrc
  1117. Arguments:
  1118. [in] slSrc - the source string list contains items to be matched
  1119. [in, out] slDest - the string list where equal items are removed
  1120. Return Value: none
  1121. Note : Both slSrc and slDest MUST not contain duplicated items.
  1122. --*/
  1123. void RemoveEqualItems(IN const TStringList& slSrc, IN OUT TStringList& slDest)
  1124. {
  1125. list<tstring*>::iterator _F = slSrc.begin(), _L = slSrc.end(), pos;
  1126. if (slDest.empty())
  1127. return;
  1128. while (_F != _L)
  1129. {
  1130. if (FIsInStringList(slDest, (*_F)->c_str(), &pos))
  1131. {
  1132. TraceTag(ttidNetSetup,
  1133. "RemoveEqualItems: '%S' appeared in both slSrc and slDest lists, remove it from slDest.",
  1134. (*_F)->c_str());
  1135. // found a matched item.
  1136. delete *pos;
  1137. slDest.erase(pos);
  1138. if (slDest.empty())
  1139. return;
  1140. }
  1141. ++_F;
  1142. }
  1143. }
  1144. /*++
  1145. Routine Description:
  1146. remove item in slDest if the item is the same as pstrSrc
  1147. Arguments:
  1148. [in] strSrc - the source string to be matched
  1149. [in, out] slDest - the string list where equal item is removed
  1150. Return Value: none
  1151. Note : Both slDest MUST not contain duplicated items.
  1152. --*/
  1153. void RemoveEqualItems(IN const tstring& strSrc, IN OUT TStringList& slDest)
  1154. {
  1155. list<tstring*>::iterator pos;
  1156. if (FIsInStringList(slDest, strSrc.c_str(), &pos))
  1157. {
  1158. TraceTag(ttidNetSetup,
  1159. "RemoveEqualItems: remove '%S' from slDest.",
  1160. strSrc.c_str());
  1161. // found a matched item.
  1162. delete *pos;
  1163. slDest.erase(pos);
  1164. if (slDest.empty())
  1165. return;
  1166. }
  1167. }
  1168. /*++
  1169. Routine Description:
  1170. Load Homenet settings from the answer file and put the
  1171. settings into the ICS_UPGRADE_SETTINGS structure
  1172. Arguments:
  1173. [in] pwifAnswerFile the access point of the answer file
  1174. [in/out] pSettings it contains the settings when the
  1175. routine returns S_OK
  1176. Returns: S_OK if there are Homenet settings in the answer file
  1177. S_FALSE: no Homenet settings present in the answer file
  1178. otherwise the failure code
  1179. Notes:
  1180. Bug# 253047, the Answer-File for a Win9x ICS upgrade will
  1181. look like the following (if Clean install WinMe, selecting custom
  1182. installation, and select ICS as an optional component.):
  1183. [Homenet]
  1184. IsW9xUpgrade='1'
  1185. EnableICS="1"
  1186. ShowTrayIcon="1"
  1187. Bug# 304474, in this case, Answer-File for a Win9x ICS upgrade will look
  1188. like (There is no value for ExternalAdapter or
  1189. ExternalConnectionName key.):
  1190. [Homenet]
  1191. IsW9xUpgrade='1'
  1192. InternalAdapter="Adapter2"
  1193. DialOnDemand="0"
  1194. EnableICS="1"
  1195. ShowTrayIcon="1"
  1196. More notes regarding validation of Answer File:
  1197. 1. if adapter found in both the PersonalFirewall list and Bridge list,
  1198. remove that adaper from the PersonalFirewall list.
  1199. 2. if ICS public LAN found in Bridge list, it will be removed from the
  1200. Bridge list.
  1201. 3. if ICS private found in Firewall list, remove that adapter from the
  1202. PersonalFirewall list.
  1203. --*/
  1204. HRESULT LoadIcsSettingsFromAnswerFile(
  1205. CWInfFile* pwifAnswerFile,
  1206. PICS_UPGRADE_SETTINGS pSettings
  1207. )
  1208. {
  1209. HRESULT hr = S_OK;
  1210. BOOL fRet;
  1211. PCWInfSection pwifSection = NULL;
  1212. //BOOL fInternalAdapterFound = FALSE;
  1213. DefineFunctionName("LoadIcsSettingsFromAnswerFile");
  1214. TraceFunctionEntry(ttidNetSetup);
  1215. Assert(pwifAnswerFile);
  1216. Assert(pSettings);
  1217. pwifSection = pwifAnswerFile->FindSection(c_wszHomenetSection);
  1218. if (NULL == pwifSection)
  1219. return S_FALSE;
  1220. tstring strTemp;
  1221. TStringList slICF, slBridge; // hold comma-delimited string into a list
  1222. GUID guidTemp = {0};
  1223. BOOL fTemp = FALSE;
  1224. do
  1225. {
  1226. // ICS Upgrade from Win9x or Unattended Homenet clean install
  1227. // determine if this answer file is for Win9x upgrade or
  1228. // Homenet unattended clean install
  1229. pSettings->fWin9xUpgrade = pwifSection->GetBoolValue(c_wszIsW9xUpgrade, FALSE);
  1230. pSettings->fXpUnattended = !pSettings->fWin9xUpgrade;
  1231. // default value for EnableICS is FALSE
  1232. pSettings->fEnableICS = pwifSection->GetBoolValue(c_wszICSEnabled, FALSE);
  1233. // default value for InternalIsBridge is FALSE
  1234. pSettings->fInternalIsBridge = pwifSection->GetBoolValue(c_wszInternalIsBridge, FALSE);
  1235. TraceTag(ttidNetSetup,
  1236. "IsW9xUpgrade: %S, EnableICS: %S, InternalIsBridge: %S",
  1237. pSettings->fWin9xUpgrade ? L"Yes" : L"No",
  1238. pSettings->fEnableICS ? L"Yes" : L"No",
  1239. pSettings->fInternalIsBridge ? L"Yes" : L"No");
  1240. //
  1241. // get a list of interface GUID for the Internet Connection Firewall
  1242. // (ICF) and a list of interface GUID that forms the bridge. We'll also
  1243. // do validation before converting adapter names to interface GUID
  1244. //
  1245. fRet = pwifSection->GetStringListValue(c_wszPersonalFirewall, slICF);
  1246. if (fRet)
  1247. {
  1248. // remove duplicates within slICF
  1249. RemoveDupsInTStringList(slICF);
  1250. }
  1251. fRet = pwifSection->GetStringListValue(c_wszBridge, slBridge);
  1252. if (fRet)
  1253. {
  1254. // remove duplicates within slBridge
  1255. RemoveDupsInTStringList(slBridge);
  1256. }
  1257. if (!slICF.empty() && !slBridge.empty())
  1258. {
  1259. // remove items in slICF if the item is also in slBridge
  1260. RemoveEqualItems(slBridge, slICF);
  1261. }
  1262. // if ICS is enabled, get the share connection.
  1263. // If the share connection is a LAN connection, make sure it is
  1264. // not a member of the bridge.
  1265. if (pSettings->fEnableICS)
  1266. {
  1267. fRet = pwifSection->GetStringValue(c_wszExternalAdapter, strTemp);
  1268. if (fRet)
  1269. {
  1270. if (!FGetInstanceGuidOfComponentFromAnswerFileMap(strTemp.c_str(), &guidTemp))
  1271. {
  1272. TraceTag(ttidError,
  1273. "FGetInstanceGuidOfComponentFromAnswerFileMap failed to match GUID for adapter %S",
  1274. strTemp.c_str());
  1275. NetSetupLogStatusV(
  1276. LogSevInformation,
  1277. SzLoadIds (IDS_TXT_E_ADAPTER_TO_GUID),
  1278. strTemp.c_str());
  1279. hr = E_FAIL;
  1280. break;
  1281. }
  1282. pSettings->rscExternal.fIsLanConnection = TRUE;
  1283. pSettings->rscExternal.guid = guidTemp;
  1284. // Another validation:
  1285. // ICS public LAN adapter couldn't be a member of a bridge
  1286. if (! slBridge.empty())
  1287. {
  1288. RemoveEqualItems(strTemp, slBridge);
  1289. }
  1290. }
  1291. else
  1292. {
  1293. // Can't find "ExternalAdapter" key in answer-file.
  1294. // We'll try the "ExternalConnectionName" key.
  1295. TraceTag(ttidNetSetup,
  1296. "pwifSection->GetStringValue failed to get %S, we will try to get %S",
  1297. c_wszExternalAdapter, c_wszExternalConnectionName);
  1298. fRet = pwifSection->GetStringValue(c_wszExternalConnectionName, strTemp);
  1299. if (! fRet)
  1300. {
  1301. if (pSettings->fWin9xUpgrade)
  1302. {
  1303. TraceTag(ttidNetSetup,
  1304. "pwifSection->GetStringValue failed to get %S too.",
  1305. c_wszExternalConnectionName);
  1306. TraceTag(ttidNetSetup,
  1307. "We will try to look for a WAN connectoid as the ExternalConnectionName.");
  1308. // Bug# 304474
  1309. // Cook up a fake WAN connectoid name, the code in
  1310. // GetINetConnectionByName will later try to resolve it
  1311. // to a WAN connectoid if there is *only 1* WAN
  1312. // connectoid left on the system. WAN connectoid has
  1313. // media types of either NCM_PHONE, NCM_ISDN or
  1314. // NCM_TUNNEL.
  1315. lstrcpynW(pSettings->rscExternal.name.szEntryName,
  1316. L"X1-2Bogus-3WAN-4Conn-5Name",
  1317. celems(pSettings->rscExternal.name.szEntryName));
  1318. }
  1319. else
  1320. {
  1321. TraceTag(ttidError,
  1322. "pwifSection->GetStringValue failed to get %S too.",
  1323. c_wszExternalConnectionName);
  1324. hr = E_FAIL;
  1325. break;
  1326. }
  1327. }
  1328. else
  1329. {
  1330. lstrcpynW(pSettings->rscExternal.name.szEntryName,
  1331. strTemp.c_str(),
  1332. celems(pSettings->rscExternal.name.szEntryName));
  1333. }
  1334. pSettings->rscExternal.fIsLanConnection = FALSE;
  1335. }
  1336. }
  1337. // convert bridge adapter names to interface GUID
  1338. if (!slBridge.empty())
  1339. {
  1340. hr = ConvertAdapterStringListToGuidList(slBridge, pSettings->listBridge);
  1341. if (FAILED(hr))
  1342. {
  1343. TraceTag(
  1344. ttidError,
  1345. "Error in converting Personal Firewall string list to GUID list");
  1346. break;
  1347. }
  1348. if (S_FALSE == hr)
  1349. {
  1350. hr = S_OK; // convert any S_FALSE to S_OK
  1351. }
  1352. }
  1353. // we will convert the slICF list later because we need to check if
  1354. // ICS private adapter is also in slICF, if yes, we need to remove it
  1355. // from the slICF list.
  1356. //
  1357. // get the internal adapter GUID
  1358. //
  1359. fRet = pwifSection->GetStringValue(c_wszInternalAdapter, strTemp);
  1360. if (fRet && !pSettings->fInternalIsBridge && !pSettings->fEnableICS)
  1361. {
  1362. // invalid parameters from answer file (AF)
  1363. TraceTag(ttidError,
  1364. "Invalid AF settings: InternalAdapter=%S, InternalIsBridge='0', EnableICS='0'.",
  1365. strTemp.c_str());
  1366. NetSetupLogStatusV(
  1367. LogSevInformation,
  1368. SzLoadIds (IDS_TXT_HOMENET_INVALID_AF_SETTINGS));
  1369. hr = E_FAIL;
  1370. break;
  1371. }
  1372. if (fRet && pSettings->fInternalIsBridge)
  1373. {
  1374. TraceTag(ttidError,
  1375. "Invalid AF settings: InternalAdapter=%S, InternalIsBridge='1'.",
  1376. strTemp.c_str());
  1377. NetSetupLogStatusV(
  1378. LogSevInformation,
  1379. SzLoadIds (IDS_TXT_HOMENET_INVALID_AF_SETTINGS));
  1380. hr = E_FAIL;
  1381. break;
  1382. }
  1383. if (fRet)
  1384. {
  1385. // get the internal adapter GUID if InternalIsBridge is '0'
  1386. Assert(!pSettings->fInternalIsBridge);
  1387. if (!FGetInstanceGuidOfComponentFromAnswerFileMap(strTemp.c_str(), &guidTemp))
  1388. {
  1389. TraceTag(ttidError,
  1390. "FGetInstanceGuidOfComponentFromAnswerFileMap failed to match GUID for adapter %S",
  1391. strTemp.c_str());
  1392. NetSetupLogStatusV(
  1393. LogSevInformation,
  1394. SzLoadIds (IDS_TXT_E_ADAPTER_TO_GUID),
  1395. strTemp.c_str());
  1396. hr = E_FAIL;
  1397. break;
  1398. }
  1399. else
  1400. {
  1401. pSettings->fInternalAdapterFound = TRUE;
  1402. pSettings->guidInternal = guidTemp;
  1403. // another validation:
  1404. // remove ICS private from the firewall list (slICF)
  1405. // if necessary.
  1406. if (!slICF.empty())
  1407. {
  1408. RemoveEqualItems(strTemp, slICF);
  1409. }
  1410. }
  1411. }
  1412. // check for other answer file errors
  1413. if (pSettings->fWin9xUpgrade && !pSettings->fEnableICS)
  1414. {
  1415. TraceTag(
  1416. ttidError,
  1417. "Invalid AF settings: IsW9xUpgrade='1', EnableICS='0'");
  1418. hr = E_FAIL;
  1419. break;
  1420. }
  1421. // special case for Win9x ICS upgrade.
  1422. if ( pSettings->fWin9xUpgrade)
  1423. {
  1424. // Bug# 315265
  1425. if ( pSettings->fInternalIsBridge &&
  1426. 1 == (pSettings->listBridge).size() )
  1427. {
  1428. // for Win9x ICS upgrade, one of the 2 internal adapters is
  1429. // broken, we will continue to do the upgrade without creating
  1430. // the internal bridge.
  1431. // adjust answer file parameters
  1432. pSettings->fWin9xUpgradeAtLeastOneInternalAdapterBroken = TRUE;
  1433. pSettings->fInternalAdapterFound = TRUE;
  1434. pSettings->guidInternal = *((pSettings->listBridge).begin());
  1435. // note: On Win9x ICS upgrade, there will be no firewall list,
  1436. // so, we don't try to remove this ICS private from the
  1437. // firewall list (slICF).
  1438. pSettings->fInternalIsBridge = FALSE;
  1439. pSettings->listBridge.erase(pSettings->listBridge.begin(),
  1440. pSettings->listBridge.end());
  1441. TraceTag(
  1442. ttidNetSetup,
  1443. "On Win9x ICS upgrade, one internal adapter couldn't be upgraded.");
  1444. }
  1445. }
  1446. if ( (pSettings->fInternalIsBridge && (pSettings->listBridge).size() < 2) ||
  1447. (1 == (pSettings->listBridge).size()) )
  1448. {
  1449. TraceTag(
  1450. ttidError,
  1451. "Invalid setting: fInternalIsBridge: %S, size of listBridge: %d",
  1452. pSettings->fInternalIsBridge ? L"TRUE" : L"FALSE",
  1453. (pSettings->listBridge).size());
  1454. NetSetupLogStatusV(
  1455. LogSevInformation,
  1456. SzLoadIds (IDS_TXT_HOMENET_INVALID_AF_SETTINGS));
  1457. hr = E_FAIL;
  1458. break;
  1459. }
  1460. if ( pSettings->fEnableICS &&
  1461. !pSettings->fInternalIsBridge &&
  1462. !pSettings->fInternalAdapterFound)
  1463. {
  1464. // Bug# 304474 falls under this case
  1465. TraceTag(
  1466. ttidError,
  1467. "Invalid AF settings: no InternalAdapter, InternalIsBridge='0', EnableICS='1'.");
  1468. NetSetupLogStatusV(
  1469. LogSevInformation,
  1470. SzLoadIds (IDS_TXT_HOMENET_INVALID_AF_SETTINGS));
  1471. hr = E_FAIL;
  1472. break;
  1473. }
  1474. // convert firewall adapter names to interface GUID
  1475. if (!slICF.empty())
  1476. {
  1477. hr = ConvertAdapterStringListToGuidList(slICF, pSettings->listPersonalFirewall);
  1478. if (FAILED(hr))
  1479. {
  1480. TraceTag(
  1481. ttidError,
  1482. "Error in converting Personal Firewall string list to GUID list");
  1483. break;
  1484. }
  1485. if (S_FALSE == hr)
  1486. {
  1487. hr = S_OK; // convert any S_FALSE to S_OK
  1488. }
  1489. }
  1490. // default value for DialOnDemand is FALSE
  1491. pSettings->fDialOnDemand = pwifSection->GetBoolValue(c_wszDialOnDemand, FALSE);
  1492. // default value for ShowTrayIcon is TRUE
  1493. pSettings->fShowTrayIcon = pwifSection->GetBoolValue(c_wszShowTrayIcon, TRUE);
  1494. } while(FALSE);
  1495. // free resource if necessary
  1496. EraseAndDeleteAll(&slICF);
  1497. EraseAndDeleteAll(&slBridge);
  1498. return hr;
  1499. }
  1500. /*++
  1501. Routine Description:
  1502. Set some default values into ICS_UPGRADE_SETTINGS
  1503. Arguments:
  1504. [in] pIcsUpgrdSettings - pointer to the ICS_UPGRADE_SETTINGS
  1505. Return Value: none
  1506. Note.
  1507. --*/
  1508. void SetIcsDefaultSettings(ICS_UPGRADE_SETTINGS * pSettings)
  1509. {
  1510. //initialize the settings here
  1511. pSettings->fDialOnDemand = FALSE;
  1512. pSettings->fEnableICS = FALSE;
  1513. pSettings->fShowTrayIcon = TRUE;
  1514. pSettings->fWin9xUpgrade = FALSE;
  1515. pSettings->fWin9xUpgradeAtLeastOneInternalAdapterBroken = FALSE;
  1516. pSettings->fWin2KUpgrade = FALSE;
  1517. pSettings->fXpUnattended = FALSE;
  1518. pSettings->fInternalIsBridge = FALSE;
  1519. pSettings->fInternalAdapterFound = FALSE;
  1520. }
  1521. /*++
  1522. Routine Description:
  1523. Check current OS version
  1524. Arguments:
  1525. None
  1526. Return Value: returns TRUE if current OS version is Adv. Server or higher
  1527. Note.
  1528. --*/
  1529. BOOL FOsIsAdvServerOrHigher()
  1530. {
  1531. OSVERSIONINFOEXW verInfo = {0};
  1532. ULONGLONG ConditionMask = 0;
  1533. verInfo.dwOSVersionInfoSize = sizeof(verInfo);
  1534. verInfo.wSuiteMask = VER_SUITE_ENTERPRISE;
  1535. verInfo.wProductType = VER_NT_SERVER;
  1536. VER_SET_CONDITION(ConditionMask, VER_PRODUCT_TYPE, VER_GREATER_EQUAL);
  1537. VER_SET_CONDITION(ConditionMask, VER_SUITENAME, VER_AND);
  1538. return VerifyVersionInfo(&verInfo,
  1539. VER_PRODUCT_TYPE | VER_SUITENAME,
  1540. ConditionMask);
  1541. }
  1542. //--------- ICS Upgrade helpers end -----------------------------
  1543. //--------- HNet helpers begin ----------------------------------
  1544. // extract from %sdxroot%\net\rras\ras\ui\common\nouiutil\noui.c
  1545. DWORD
  1546. IpPszToHostAddr(
  1547. IN LPCTSTR cp )
  1548. // Converts an IP address represented as a string to
  1549. // host byte order.
  1550. //
  1551. {
  1552. DWORD val, base, n;
  1553. TCHAR c;
  1554. DWORD parts[4], *pp = parts;
  1555. again:
  1556. // Collect number up to ``.''.
  1557. // Values are specified as for C:
  1558. // 0x=hex, 0=octal, other=decimal.
  1559. //
  1560. val = 0; base = 10;
  1561. if (*cp == TEXT('0'))
  1562. base = 8, cp++;
  1563. if (*cp == TEXT('x') || *cp == TEXT('X'))
  1564. base = 16, cp++;
  1565. while (c = *cp)
  1566. {
  1567. if ((c >= TEXT('0')) && (c <= TEXT('9')))
  1568. {
  1569. val = (val * base) + (c - TEXT('0'));
  1570. cp++;
  1571. continue;
  1572. }
  1573. if ((base == 16) &&
  1574. ( ((c >= TEXT('0')) && (c <= TEXT('9'))) ||
  1575. ((c >= TEXT('A')) && (c <= TEXT('F'))) ||
  1576. ((c >= TEXT('a')) && (c <= TEXT('f'))) ))
  1577. {
  1578. val = (val << 4) + (c + 10 - (
  1579. ((c >= TEXT('a')) && (c <= TEXT('f')))
  1580. ? TEXT('a')
  1581. : TEXT('A') ) );
  1582. cp++;
  1583. continue;
  1584. }
  1585. break;
  1586. }
  1587. if (*cp == TEXT('.'))
  1588. {
  1589. // Internet format:
  1590. // a.b.c.d
  1591. // a.b.c (with c treated as 16-bits)
  1592. // a.b (with b treated as 24 bits)
  1593. //
  1594. if (pp >= parts + 3)
  1595. return (DWORD) -1;
  1596. *pp++ = val, cp++;
  1597. goto again;
  1598. }
  1599. // Check for trailing characters.
  1600. //
  1601. if (*cp && (*cp != TEXT(' ')))
  1602. return 0xffffffff;
  1603. *pp++ = val;
  1604. // Concoct the address according to
  1605. // the number of parts specified.
  1606. //
  1607. n = (DWORD) (pp - parts);
  1608. switch (n)
  1609. {
  1610. case 1: // a -- 32 bits
  1611. val = parts[0];
  1612. break;
  1613. case 2: // a.b -- 8.24 bits
  1614. val = (parts[0] << 24) | (parts[1] & 0xffffff);
  1615. break;
  1616. case 3: // a.b.c -- 8.8.16 bits
  1617. val = (parts[0] << 24) | ((parts[1] & 0xff) << 16) |
  1618. (parts[2] & 0xffff);
  1619. break;
  1620. case 4: // a.b.c.d -- 8.8.8.8 bits
  1621. val = (parts[0] << 24) | ((parts[1] & 0xff) << 16) |
  1622. ((parts[2] & 0xff) << 8) | (parts[3] & 0xff);
  1623. break;
  1624. default:
  1625. return 0xffffffff;
  1626. }
  1627. return val;
  1628. }
  1629. /*++
  1630. Routine Description:
  1631. Sets the standard COM security settings on the proxy for an
  1632. object.
  1633. Arguments:
  1634. pUnk - the object to set the proxy blanket on
  1635. Return Value:
  1636. Note. Even if the CoSetProxyBlanket calls fail, pUnk remains
  1637. in a usable state. Failure is expected in certain contexts, such
  1638. as when, for example, we're being called w/in the netman process --
  1639. in this case, we have direct pointers to the netman objects, instead
  1640. of going through a proxy.
  1641. --*/
  1642. VOID SetProxyBlanket(IUnknown *pUnk)
  1643. {
  1644. HRESULT hr;
  1645. Assert(pUnk);
  1646. hr = CoSetProxyBlanket(
  1647. pUnk,
  1648. RPC_C_AUTHN_WINNT, // use NT default security
  1649. RPC_C_AUTHZ_NONE, // use NT default authentication
  1650. NULL, // must be null if default
  1651. RPC_C_AUTHN_LEVEL_CALL, // call
  1652. RPC_C_IMP_LEVEL_IMPERSONATE,
  1653. NULL, // use process token
  1654. EOAC_NONE
  1655. );
  1656. if (SUCCEEDED(hr))
  1657. {
  1658. IUnknown * pUnkSet = NULL;
  1659. hr = pUnk->QueryInterface(&pUnkSet);
  1660. if (SUCCEEDED(hr))
  1661. {
  1662. hr = CoSetProxyBlanket(
  1663. pUnkSet,
  1664. RPC_C_AUTHN_WINNT, // use NT default security
  1665. RPC_C_AUTHZ_NONE, // use NT default authentication
  1666. NULL, // must be null if default
  1667. RPC_C_AUTHN_LEVEL_CALL, // call
  1668. RPC_C_IMP_LEVEL_IMPERSONATE,
  1669. NULL, // use process token
  1670. EOAC_NONE
  1671. );
  1672. pUnkSet->Release();
  1673. }
  1674. }
  1675. }
  1676. /////////////////////////////////////////////////////////////////////////////
  1677. // CIcsUpgrade public methods
  1678. /////////////////////////////////////////////////////////////////////////////
  1679. /*++
  1680. Routine Description:
  1681. Init ourself with the pIcsUpgradeSettings, we also cache the
  1682. IEnumNetConnection object in our m_spEnum.
  1683. Arguments:
  1684. [in] pIcsUpgradeSettings - the ICS UPGRADE SETTINGS
  1685. Return Value:
  1686. standard HRESULT
  1687. --*/
  1688. HRESULT CIcsUpgrade::Init(ICS_UPGRADE_SETTINGS * pIcsUpgradeSettings)
  1689. {
  1690. HRESULT hr;
  1691. if (!pIcsUpgradeSettings)
  1692. {
  1693. return E_INVALIDARG;
  1694. }
  1695. if (pIcsUpgradeSettings->fWin9xUpgrade)
  1696. {
  1697. // validate parameters for Win9x ICS upgrade
  1698. if (!pIcsUpgradeSettings->fEnableICS)
  1699. {
  1700. return E_INVALIDARG;
  1701. }
  1702. if (pIcsUpgradeSettings->fInternalIsBridge)
  1703. {
  1704. // internal is a bridge
  1705. if ( (pIcsUpgradeSettings->listBridge).size() < 2 )
  1706. {
  1707. return E_INVALIDARG;
  1708. }
  1709. }
  1710. else
  1711. {
  1712. if (!pIcsUpgradeSettings->fInternalAdapterFound)
  1713. {
  1714. return E_INVALIDARG;
  1715. }
  1716. }
  1717. }
  1718. else if (pIcsUpgradeSettings->fWin2KUpgrade)
  1719. {
  1720. // validate parameters for Win2K ICS upgrade
  1721. if (pIcsUpgradeSettings->fInternalIsBridge ||
  1722. !pIcsUpgradeSettings->fEnableICS ||
  1723. !pIcsUpgradeSettings->fInternalAdapterFound)
  1724. {
  1725. return E_INVALIDARG;
  1726. }
  1727. }
  1728. if (!m_fInited)
  1729. {
  1730. // Get the net connection manager
  1731. INetConnectionManager *pConnMan;
  1732. hr = CoCreateInstance(CLSID_ConnectionManager, NULL,
  1733. CLSCTX_ALL,
  1734. IID_INetConnectionManager,
  1735. (LPVOID *)&pConnMan);
  1736. if (S_OK == hr)
  1737. {
  1738. // Get the enumeration of connections
  1739. SetProxyBlanket(pConnMan);
  1740. hr = pConnMan->EnumConnections(NCME_DEFAULT, &m_spEnum);
  1741. pConnMan->Release();
  1742. }
  1743. else
  1744. {
  1745. return E_FAIL;
  1746. }
  1747. if (S_OK == hr)
  1748. {
  1749. SetProxyBlanket(m_spEnum);
  1750. m_fInited = TRUE;
  1751. }
  1752. else
  1753. {
  1754. return E_FAIL;
  1755. }
  1756. }
  1757. m_pIcsUpgradeSettings = pIcsUpgradeSettings;
  1758. return S_OK;
  1759. }
  1760. /*++
  1761. Routine Description:
  1762. Enable ICS on external adapter and internal adapters.
  1763. Enable personal firewall on the external adapter.
  1764. Migrating the server and application port mappings.
  1765. Arguments:
  1766. None
  1767. Return Value:
  1768. standard HRESULT
  1769. --*/
  1770. HRESULT CIcsUpgrade::StartUpgrade()
  1771. {
  1772. HRESULT hr;
  1773. DefineFunctionName("CIcsUpgrade::StartUpGrade");
  1774. TraceFunctionEntry(ttidNetSetup);
  1775. if (m_fInited)
  1776. {
  1777. if (m_pIcsUpgradeSettings->fWin2KUpgrade)
  1778. {
  1779. // we need to delete Win2K registry named value
  1780. hr = BackupAndDelIcsRegistryValuesOnWin2k();
  1781. if (FAILED(hr))
  1782. {
  1783. TraceTag(ttidError,
  1784. "%s: Ignore BackupAndDelIcsRegistryValuesOnWin2k failed: 0x%lx",
  1785. __FUNCNAME__, hr);
  1786. }
  1787. hr = SetupApplicationProtocol();
  1788. if (FAILED(hr))
  1789. {
  1790. TraceTag(ttidError, "%s: SetupApplicationProtocol failed: 0x%lx",
  1791. __FUNCNAME__, hr);
  1792. return hr;
  1793. }
  1794. hr = SetupServerPortMapping();
  1795. if (FAILED(hr))
  1796. {
  1797. TraceTag(ttidError, "%s: SetupServerPortMapping failed: 0x%lx",
  1798. __FUNCNAME__, hr);
  1799. return hr;
  1800. }
  1801. }
  1802. hr = SetupHomenetConnections();
  1803. if (FAILED(hr))
  1804. {
  1805. TraceTag(ttidError, "%s: SetupHomenetConnections failed: 0x%lx",
  1806. __FUNCNAME__, hr);
  1807. return hr;
  1808. }
  1809. hr = SetupIcsMiscItems();
  1810. if (FAILED(hr))
  1811. {
  1812. TraceTag(ttidError,
  1813. "%s: Ignore SetupIcsMiscItems failed: 0x%lx",
  1814. __FUNCNAME__, hr);
  1815. }
  1816. }
  1817. return S_OK;
  1818. }
  1819. /////////////////////////////////////////////////////////////////////////////
  1820. // CIcsUpgrade private methods
  1821. /////////////////////////////////////////////////////////////////////////////
  1822. /*++
  1823. Routine Description:
  1824. Free all cached resources,
  1825. CComPtr smart pointer will be released when this object is out of scope or
  1826. being deleted.
  1827. Arguments:
  1828. None
  1829. Return Value:
  1830. None
  1831. --*/
  1832. void CIcsUpgrade::FinalRelease()
  1833. {
  1834. if (m_pExternalNetConn)
  1835. {
  1836. m_pExternalNetConn->Release();
  1837. m_pExternalNetConn = NULL;
  1838. }
  1839. if (m_hIcsUpgradeEvent)
  1840. {
  1841. CloseHandle(m_hIcsUpgradeEvent);
  1842. m_hIcsUpgradeEvent = NULL;
  1843. }
  1844. }
  1845. /*++
  1846. Routine Description:
  1847. Setup Homenet connections based on parameters from m_pIcsUpgradeSettings.
  1848. Arguments:
  1849. None
  1850. Return Value:
  1851. standard HRESULT
  1852. --*/
  1853. HRESULT CIcsUpgrade::SetupHomenetConnections()
  1854. {
  1855. DefineFunctionName("CIcsUpgrade::SetupHomenetConnections");
  1856. TraceFunctionEntry(ttidNetSetup);
  1857. HRESULT hr = S_OK;
  1858. INetConnection** prgINetConn = NULL; // array of INetConnection*
  1859. DWORD cINetConn = 0; // number of INetConnection* in the array
  1860. INetConnection* pExternalNetConn = NULL;
  1861. INetConnection* pInternalNetConn = NULL;
  1862. // Create a named event to notify other components that we're
  1863. // in GUI Mode ICS Upgrade. Reason: sharedaccess service
  1864. // fails to start during the GUI Mode Setup.
  1865. hr = CreateIcsUpgradeNamedEvent();
  1866. if (FAILED(hr))
  1867. {
  1868. TraceTag(ttidError,
  1869. "%s: CreateIcsUpgradeNamedEvent failed: 0x%lx",
  1870. __FUNCNAME__, hr);
  1871. return hr;
  1872. }
  1873. // create a bridge if we have more than 1 adapter guid
  1874. if ( (m_pIcsUpgradeSettings->listBridge).size() > 1 )
  1875. {
  1876. do
  1877. {
  1878. hr = GetINetConnectionArray(m_pIcsUpgradeSettings->listBridge,
  1879. &prgINetConn, &cINetConn);
  1880. TraceTag(ttidNetSetup,
  1881. "after GetINetConnectionArray for bridge.");
  1882. if (FAILED(hr))
  1883. {
  1884. TraceTag(ttidError, "%s: GetINetConnectionArray failed: 0x%lx line: %d",
  1885. __FUNCNAME__, hr, __LINE__);
  1886. NetSetupLogStatusV(
  1887. LogSevInformation,
  1888. SzLoadIds (IDS_TXT_CANT_CREATE_BRIDGE));
  1889. break;
  1890. }
  1891. TraceTag(ttidNetSetup, "calling HNetCreateBridge.");
  1892. hr = HNetCreateBridge(prgINetConn, NULL);
  1893. if (FAILED(hr))
  1894. {
  1895. TraceTag(ttidError, "%s: HNetCreateBridge failed: 0x%lx",
  1896. __FUNCNAME__, hr);
  1897. NetSetupLogStatusV(
  1898. LogSevInformation,
  1899. SzLoadIds (IDS_TXT_CANT_CREATE_BRIDGE));
  1900. break;
  1901. }
  1902. } while (FALSE);
  1903. // free resource if necessary
  1904. for (DWORD i = 0; i < cINetConn ; ++i)
  1905. {
  1906. if (prgINetConn[i])
  1907. prgINetConn[i]->Release();
  1908. }
  1909. if (prgINetConn)
  1910. {
  1911. delete [] (BYTE*)prgINetConn;
  1912. }
  1913. prgINetConn = NULL;
  1914. cINetConn = 0;
  1915. }
  1916. // create ICS
  1917. if (m_pIcsUpgradeSettings->fEnableICS)
  1918. {
  1919. do
  1920. {
  1921. if (m_pExternalNetConn == NULL)
  1922. { // no cached copy
  1923. hr = GetExternalINetConnection(&pExternalNetConn);
  1924. if (FAILED(hr))
  1925. {
  1926. TraceTag(ttidError, "%s: GetExternalINetConnection failed: 0x%lx",
  1927. __FUNCNAME__, hr);
  1928. NetSetupLogStatusV(
  1929. LogSevInformation,
  1930. SzLoadIds (IDS_TXT_CANT_CREATE_ICS));
  1931. break;
  1932. }
  1933. }
  1934. else
  1935. {
  1936. pExternalNetConn = m_pExternalNetConn; // use the cached copy
  1937. }
  1938. if (m_pIcsUpgradeSettings->fInternalIsBridge)
  1939. {
  1940. hr = GetBridgeINetConn(&pInternalNetConn);
  1941. if (FAILED(hr))
  1942. {
  1943. TraceTag(ttidError, "%s: GetBridgeINetConn failed: 0x%lx",
  1944. __FUNCNAME__, hr);
  1945. NetSetupLogStatusV(
  1946. LogSevInformation,
  1947. SzLoadIds (IDS_TXT_CANT_CREATE_ICS));
  1948. break;
  1949. }
  1950. }
  1951. else
  1952. {
  1953. hr = GetINetConnectionByGuid(
  1954. &m_pIcsUpgradeSettings->guidInternal,
  1955. &pInternalNetConn);
  1956. if (FAILED(hr))
  1957. {
  1958. TraceTag(ttidError, "%s: GetINetConnectionByGuid failed: 0x%lx line: %d",
  1959. __FUNCNAME__, hr, __LINE__);
  1960. NetSetupLogStatusV(
  1961. LogSevInformation,
  1962. SzLoadIds (IDS_TXT_CANT_CREATE_ICS));
  1963. break;
  1964. }
  1965. }
  1966. TraceTag(ttidNetSetup, "calling HrCreateICS.");
  1967. hr = HrCreateICS(pExternalNetConn, pInternalNetConn);
  1968. if (FAILED(hr))
  1969. {
  1970. TraceTag(ttidError, "%s: HNetCreateBridge failed: 0x%lx",
  1971. __FUNCNAME__, hr);
  1972. NetSetupLogStatusV(
  1973. LogSevInformation,
  1974. SzLoadIds (IDS_TXT_CANT_CREATE_ICS));
  1975. break;
  1976. }
  1977. m_fICSCreated = TRUE;
  1978. } while (FALSE);
  1979. // cleanup if necessary
  1980. if (pInternalNetConn)
  1981. {
  1982. pInternalNetConn->Release();
  1983. }
  1984. if (FAILED(hr))
  1985. {
  1986. if (pExternalNetConn && !m_pExternalNetConn)
  1987. {
  1988. pExternalNetConn->Release();
  1989. }
  1990. }
  1991. else
  1992. {
  1993. // cache External INetConnections if necessary
  1994. if (!m_pExternalNetConn)
  1995. {
  1996. m_pExternalNetConn = pExternalNetConn;
  1997. }
  1998. }
  1999. }
  2000. // enable firewall on each connection specified from answer file
  2001. // note: ICS private couldn't be firewalled, bridge couldn't be firewalled.
  2002. if ( (m_pIcsUpgradeSettings->listPersonalFirewall).size() > 0 )
  2003. {
  2004. do
  2005. {
  2006. hr = GetINetConnectionArray(
  2007. m_pIcsUpgradeSettings->listPersonalFirewall,
  2008. &prgINetConn, &cINetConn);
  2009. if (FAILED(hr))
  2010. {
  2011. TraceTag(ttidError, "%s: GetINetConnectionArray failed: 0x%lx line: %d",
  2012. __FUNCNAME__, hr, __LINE__);
  2013. NetSetupLogStatusV(
  2014. LogSevInformation,
  2015. SzLoadIds (IDS_TXT_CANT_FIREWALL));
  2016. break;
  2017. }
  2018. TraceTag(ttidNetSetup, "calling HrEnablePersonalFirewall.");
  2019. hr = HrEnablePersonalFirewall(prgINetConn);
  2020. if (FAILED(hr))
  2021. {
  2022. TraceTag(ttidError, "%s: HrEnablePersonalFirewall failed: 0x%lx",
  2023. __FUNCNAME__, hr);
  2024. NetSetupLogStatusV(
  2025. LogSevInformation,
  2026. SzLoadIds (IDS_TXT_CANT_FIREWALL));
  2027. break;
  2028. }
  2029. } while(FALSE);
  2030. // free resource if necessary
  2031. for (DWORD i = 0; i < cINetConn ; ++i)
  2032. {
  2033. if (prgINetConn[i])
  2034. prgINetConn[i]->Release();
  2035. }
  2036. if (prgINetConn)
  2037. {
  2038. delete [] (BYTE*)prgINetConn;
  2039. }
  2040. prgINetConn = NULL;
  2041. cINetConn = 0;
  2042. }
  2043. return hr;
  2044. }
  2045. /*++
  2046. Routine Description:
  2047. Get an array of INetConnection interface pointers
  2048. corresponds to a list of GUID in rlistGuid.
  2049. Arguments:
  2050. [in] rlistGuid - a list of interface Guid.
  2051. [in, out] pprgInternalNetConn - an array of INetConnection*
  2052. [in, out] pcInternalNetConn - number of INetConnection* in the array
  2053. Return Value:
  2054. standard HRESULT
  2055. Note :
  2056. -User needs to release the returned INetConnection interface pointers in
  2057. the array.
  2058. -User needs to free the memory allocated for the array by using
  2059. "delete [] (BYTE*)prgInternalNetConn".
  2060. -The array is NULL terminated (similar to "char** argv" argument in
  2061. main())
  2062. --*/
  2063. HRESULT CIcsUpgrade::GetINetConnectionArray(
  2064. IN list<GUID>& rlistGuid,
  2065. IN OUT INetConnection*** pprgINetConn,
  2066. IN OUT DWORD* pcINetConn)
  2067. {
  2068. DefineFunctionName("CIcsUpgrade::GetINetConnectionArray");
  2069. TraceFunctionEntry(ttidNetSetup);
  2070. HRESULT hr;
  2071. Assert(pprgINetConn);
  2072. Assert(pcINetConn);
  2073. *pprgINetConn = NULL;
  2074. *pcINetConn = 0;
  2075. DWORD cConnections = rlistGuid.size();
  2076. if (cConnections < 1)
  2077. {
  2078. return E_FAIL;
  2079. }
  2080. // Note that we allocated an extra entry since this is a null-terminated
  2081. // array
  2082. DWORD dwSize = (cConnections + 1) * sizeof(INetConnection*);
  2083. INetConnection** prgINetConn = (INetConnection**) new BYTE[dwSize];
  2084. if (prgINetConn)
  2085. {
  2086. ZeroMemory((PVOID) prgINetConn, dwSize);
  2087. DWORD i = 0;
  2088. for (list<GUID>::iterator iter = rlistGuid.begin();
  2089. iter != rlistGuid.end(); ++iter, ++i)
  2090. {
  2091. hr = GetINetConnectionByGuid( &(*iter), &prgINetConn[i]);
  2092. if (FAILED(hr))
  2093. {
  2094. for (DWORD j = 0; j < i; ++j)
  2095. {
  2096. if (prgINetConn[j])
  2097. {
  2098. prgINetConn[j]->Release();
  2099. }
  2100. }
  2101. delete [] (BYTE*)prgINetConn;
  2102. return hr;
  2103. }
  2104. }
  2105. }
  2106. else
  2107. {
  2108. hr = E_OUTOFMEMORY;
  2109. }
  2110. if (SUCCEEDED(hr))
  2111. {
  2112. *pcINetConn = cConnections;
  2113. *pprgINetConn = prgINetConn;
  2114. }
  2115. return hr;
  2116. }
  2117. /*++
  2118. Routine Description:
  2119. Retrieves the INetConnection that corresponds to the LPRASSHARECONN
  2120. in m_pIcsUpgradeSettings->pExternal.
  2121. Arguments:
  2122. [out] ppNetConn - receives the interface
  2123. Return Value:
  2124. standard HRESULT
  2125. --*/
  2126. HRESULT CIcsUpgrade::GetExternalINetConnection(INetConnection** ppNetConn)
  2127. {
  2128. LPRASSHARECONN pRasShareConn = &(m_pIcsUpgradeSettings->rscExternal);
  2129. if (pRasShareConn->fIsLanConnection)
  2130. {
  2131. return GetINetConnectionByGuid(&(pRasShareConn->guid), ppNetConn);
  2132. }
  2133. else
  2134. {
  2135. return GetINetConnectionByName((pRasShareConn->name).szEntryName,
  2136. ppNetConn);
  2137. }
  2138. }
  2139. /*++
  2140. Routine Description:
  2141. Retrieves the INetConnection that corresponds to the given GUID.
  2142. We will enumerate the INetConnection using our cached m_spEnum.
  2143. Arguments:
  2144. [in] pGuid - the guid of the connection
  2145. [out] ppNetCon - receives the interface
  2146. Return Value:
  2147. standard HRESULT
  2148. --*/
  2149. HRESULT
  2150. CIcsUpgrade::GetINetConnectionByGuid(
  2151. GUID* pGuid,
  2152. INetConnection** ppNetCon)
  2153. {
  2154. HRESULT hr;
  2155. INetConnection* pConn;
  2156. Assert(NULL != pGuid);
  2157. Assert(NULL != ppNetCon);
  2158. *ppNetCon = NULL;
  2159. // Search for the connection with the correct guid
  2160. ULONG ulCount;
  2161. BOOLEAN fFound = FALSE;
  2162. #if DBG
  2163. WCHAR szGuid[c_cchGuidWithTerm];
  2164. if (SUCCEEDED(StringFromGUID2(*pGuid, szGuid, c_cchGuidWithTerm)))
  2165. {
  2166. TraceTag(ttidNetSetup, "GetINetConnectionByGuid: pGuid is: %S", szGuid);
  2167. }
  2168. #endif
  2169. // reset our cache m_spEnum
  2170. m_spEnum->Reset();
  2171. do
  2172. {
  2173. NETCON_PROPERTIES* pProps;
  2174. hr = m_spEnum->Next(1, &pConn, &ulCount);
  2175. if (SUCCEEDED(hr) && 1 == ulCount)
  2176. {
  2177. SetProxyBlanket(pConn);
  2178. hr = pConn->GetProperties(&pProps);
  2179. if (S_OK == hr)
  2180. {
  2181. if (IsEqualGUID(pProps->guidId, *pGuid))
  2182. {
  2183. fFound = TRUE;
  2184. *ppNetCon = pConn;
  2185. (*ppNetCon)->AddRef();
  2186. }
  2187. NcFreeNetconProperties(pProps);
  2188. }
  2189. pConn->Release();
  2190. }
  2191. } while (FALSE == fFound && SUCCEEDED(hr) && 1 == ulCount);
  2192. // Normalize hr
  2193. hr = (fFound ? S_OK : E_FAIL);
  2194. return hr;
  2195. }
  2196. /*++
  2197. Routine Description:
  2198. Retrieves the INetConnection that corresponds to the given Name
  2199. of the Connection Icon in Network Connection Folder.
  2200. We will enumerate the INetConnection using our cached m_spEnum.
  2201. Arguments:
  2202. [in] pwszConnName - the connection name to search
  2203. [out] ppNetCon - receives the interface
  2204. Return Value:
  2205. standard HRESULT
  2206. Note: Bug# 304474: If we couldn't find the INetConnection corresponding
  2207. to the given name from the pwszConnName parameter and there is
  2208. exactly one WAN MediaType connection left, we'll return the
  2209. INetconnection for that WAN connection. WAN media types are either
  2210. NCM_PHONE, NCM_ISDN or NCM_TUNNEL.
  2211. --*/
  2212. HRESULT CIcsUpgrade::GetINetConnectionByName(
  2213. WCHAR* pwszConnName,
  2214. INetConnection** ppNetCon)
  2215. {
  2216. HRESULT hr;
  2217. INetConnection* pConn;
  2218. Assert(NULL != pwszConnName);
  2219. Assert(NULL != ppNetCon);
  2220. *ppNetCon = NULL;
  2221. TraceTag(ttidNetSetup, "GetINetConnectionByName: pwszConnName: %S", pwszConnName);
  2222. // Search for the connection with the correct connection name
  2223. ULONG ulCount;
  2224. BOOLEAN fFound = FALSE;
  2225. INetConnection* pWANConn = NULL;
  2226. ULONG ulcWANConn = 0;
  2227. // reset our cache m_spEnum
  2228. m_spEnum->Reset();
  2229. do
  2230. {
  2231. NETCON_PROPERTIES* pProps;
  2232. hr = m_spEnum->Next(1, &pConn, &ulCount);
  2233. if (SUCCEEDED(hr) && 1 == ulCount)
  2234. {
  2235. SetProxyBlanket(pConn);
  2236. hr = pConn->GetProperties(&pProps);
  2237. if (S_OK == hr)
  2238. {
  2239. TraceTag(ttidNetSetup, "GetINetConnectionByName: Connection Name: %S", pProps->pszwName);
  2240. if ( (NCM_PHONE == pProps->MediaType) ||
  2241. (NCM_ISDN == pProps->MediaType) ||
  2242. (NCM_TUNNEL == pProps->MediaType)
  2243. )
  2244. {
  2245. TraceTag(
  2246. ttidNetSetup,
  2247. "GetINetConnectionByName: Connection Name: %S is WAN, MediaType is %d",
  2248. pProps->pszwName, pProps->MediaType);
  2249. ulcWANConn++;
  2250. if (1 == ulcWANConn)
  2251. {
  2252. // save the 1st WAN connection found
  2253. TraceTag(ttidNetSetup, "GetINetConnectionByName: Connection Name: %S is 1st WAN found", pProps->pszwName);
  2254. pWANConn = pConn;
  2255. pWANConn->AddRef();
  2256. }
  2257. }
  2258. if (wcscmp(pProps->pszwName, pwszConnName) == 0)
  2259. {
  2260. fFound = TRUE;
  2261. *ppNetCon = pConn;
  2262. (*ppNetCon)->AddRef();
  2263. }
  2264. NcFreeNetconProperties(pProps);
  2265. }
  2266. pConn->Release();
  2267. }
  2268. }
  2269. while (FALSE == fFound && SUCCEEDED(hr) && 1 == ulCount);
  2270. if (fFound)
  2271. {
  2272. if (pWANConn)
  2273. {
  2274. // No matter how many WAN connections we have,
  2275. // release the saved WAN connection.
  2276. pWANConn->Release();
  2277. }
  2278. }
  2279. else
  2280. {
  2281. TraceTag(ttidNetSetup, "GetINetConnectionByName: Can't find pwszConnName: %S", pwszConnName);
  2282. // no connection has the name specified by the pwszConnName [in] parameter
  2283. if (1 == ulcWANConn && pWANConn)
  2284. {
  2285. // There is exactly ONE WAN connection left on the sysetm.
  2286. // Transfer this as the INetConnection output.
  2287. TraceTag(ttidNetSetup, "GetINetConnectionByName: will use the only one WAN connectoid left");
  2288. *ppNetCon = pWANConn;
  2289. fFound = TRUE;
  2290. }
  2291. else if (ulcWANConn > 1 && pWANConn)
  2292. {
  2293. // not found and there are more than 1 WAN connections
  2294. TraceTag(ttidNetSetup, "GetINetConnectionByName: more than one WAN connectoid found");
  2295. pWANConn->Release();
  2296. }
  2297. }
  2298. // Normalize hr
  2299. hr = (fFound ? S_OK : E_FAIL);
  2300. return hr;
  2301. }
  2302. /*++
  2303. Routine Description:
  2304. Setup the global Application Protocol
  2305. Arguments:
  2306. None
  2307. Return Value:
  2308. standard HRESULT
  2309. --*/
  2310. HRESULT CIcsUpgrade::SetupApplicationProtocol()
  2311. {
  2312. DefineFunctionName("CIcsUpgrade::SetupApplicationProtocol");
  2313. TraceFunctionEntry(ttidNetSetup);
  2314. HRESULT hr;
  2315. list<CSharedAccessApplication>& rListAppProt =
  2316. m_pIcsUpgradeSettings->listAppPortmappings;
  2317. if ( rListAppProt.size() == 0)
  2318. {
  2319. // no Application Protocol to set
  2320. return S_OK;
  2321. }
  2322. // 1. Get the INetProtocolSettings interface to control system-wide
  2323. // ICS and firewall settings (i.e port mappings and applications).
  2324. // Create Homenet Configuration Manager COM Instance
  2325. CComPtr<IHNetCfgMgr> spIHNetCfgMgr;
  2326. hr = CoCreateInstance(CLSID_HNetCfgMgr, NULL,
  2327. CLSCTX_ALL,
  2328. IID_IHNetCfgMgr,
  2329. (LPVOID *)&spIHNetCfgMgr);
  2330. if (FAILED(hr))
  2331. {
  2332. TraceTag(ttidError, "%s: CoCreateInstance CLSID_HNetCfgMgr failed: 0x%lx",
  2333. __FUNCNAME__, hr);
  2334. return hr;
  2335. }
  2336. // Get the IHNetProtocolSettings
  2337. CComPtr<IHNetProtocolSettings> spHNetProtocolSettings;
  2338. hr = spIHNetCfgMgr->QueryInterface(IID_IHNetProtocolSettings,
  2339. (LPVOID *)&spHNetProtocolSettings);
  2340. if (FAILED(hr))
  2341. {
  2342. TraceTag(ttidError, "%s: QueryInterface IID_IHNetProtocolSettings failed: 0x%lx",
  2343. __FUNCNAME__, hr);
  2344. return hr;
  2345. }
  2346. // 2. For each enabled CSharedAccessApplication
  2347. // if (there is no existing matching ApplicationProtocol
  2348. // invoke INetProtocolSettings::CreateApplicationProtocol
  2349. // else
  2350. // update the existing ApplicaitonProtocol
  2351. for (list<CSharedAccessApplication>::iterator iter = rListAppProt.begin();
  2352. iter != rListAppProt.end();
  2353. ++iter)
  2354. {
  2355. HNET_RESPONSE_RANGE* pHNetResponseArray = NULL;
  2356. USHORT uscResponseRange = 0; // number of HNET_RESPONSE_RANGE returned
  2357. if ((*iter).m_bSelected)
  2358. {
  2359. // migrate the enabled one only
  2360. UCHAR ucProtocol;
  2361. if (lstrcmpiW( ((*iter).m_szProtocol).c_str(), c_wszTCP) == 0)
  2362. {
  2363. ucProtocol = NAT_PROTOCOL_TCP;
  2364. }
  2365. else if (lstrcmpiW( ((*iter).m_szProtocol).c_str(), c_wszUDP) == 0)
  2366. {
  2367. ucProtocol = NAT_PROTOCOL_UDP;
  2368. }
  2369. else
  2370. {
  2371. // ignore others
  2372. continue;
  2373. }
  2374. USHORT usPort = HTONS((*iter).m_wPort);
  2375. WCHAR* pwszTitle = (WCHAR*)((*iter).m_szTitle).c_str();
  2376. hr = PutResponseStringIntoArray(*iter, &uscResponseRange,
  2377. &pHNetResponseArray);
  2378. if (FAILED(hr))
  2379. {
  2380. TraceTag(ttidError,
  2381. "%s: Ignore error in PutResponseStringIntoArray : 0x%lx",
  2382. __FUNCNAME__, hr);
  2383. }
  2384. if (S_OK == hr)
  2385. {
  2386. Assert(pHNetResponseArray != NULL);
  2387. CComPtr<IHNetApplicationProtocol> spHNetAppProt;
  2388. hr = FindMatchingApplicationProtocol(
  2389. spHNetProtocolSettings, ucProtocol, usPort,
  2390. &spHNetAppProt);
  2391. if (S_OK == hr)
  2392. {
  2393. TraceTag(ttidNetSetup,
  2394. "%s: Update existing ApplicationProtocol for %S",
  2395. __FUNCNAME__, pwszTitle);
  2396. // there is existing ApplicatonProtocol
  2397. hr = spHNetAppProt->SetName(pwszTitle);
  2398. if (S_OK != hr)
  2399. {
  2400. TraceTag(ttidError, "%s: spHNetAppProt->SetName failed: 0x%lx",
  2401. __FUNCNAME__, hr);
  2402. }
  2403. hr = spHNetAppProt->SetResponseRanges(uscResponseRange,
  2404. pHNetResponseArray);
  2405. if (S_OK != hr)
  2406. {
  2407. TraceTag(ttidError, "%s: spHNetAppProt->SetResponseRanges failed: 0x%lx",
  2408. __FUNCNAME__, hr);
  2409. }
  2410. hr = spHNetAppProt->SetEnabled(TRUE);
  2411. if (S_OK != hr)
  2412. {
  2413. TraceTag(ttidError, "%s: spHNetAppProt->SetEnabled failed: 0x%lx",
  2414. __FUNCNAME__, hr);
  2415. }
  2416. }
  2417. else
  2418. {
  2419. // no existing ApplicationProtocol, create a new one
  2420. hr = spHNetProtocolSettings->CreateApplicationProtocol(
  2421. pwszTitle,
  2422. ucProtocol,
  2423. usPort,
  2424. uscResponseRange,
  2425. pHNetResponseArray,
  2426. &spHNetAppProt);
  2427. if (S_OK != hr)
  2428. {
  2429. TraceTag(ttidError,
  2430. "%s: CreateApplicationProtocol failed: 0x%lx",
  2431. __FUNCNAME__, hr);
  2432. }
  2433. else
  2434. {
  2435. hr = spHNetAppProt->SetEnabled(TRUE);
  2436. if (S_OK != hr)
  2437. {
  2438. TraceTag(ttidError,
  2439. "%s: spHNetAppProt->SetEnabled failed: 0x%lx",
  2440. __FUNCNAME__, hr);
  2441. }
  2442. }
  2443. }
  2444. delete [] (BYTE*) pHNetResponseArray;
  2445. pHNetResponseArray = NULL;
  2446. uscResponseRange = 0;
  2447. }
  2448. }
  2449. } //end for
  2450. return S_OK;
  2451. }
  2452. /*++
  2453. Routine Description:
  2454. Setup the Server Port Mapping Protocol on the external connection.
  2455. Arguments:
  2456. None
  2457. Return Value:
  2458. standard HRESULT
  2459. --*/
  2460. HRESULT CIcsUpgrade::SetupServerPortMapping()
  2461. {
  2462. HRESULT hr;
  2463. list<CSharedAccessServer>& rListSvrPortMappings =
  2464. m_pIcsUpgradeSettings->listSvrPortmappings;
  2465. DefineFunctionName("CIcsUpgrade::SetupServerPortMapping");
  2466. TraceFunctionEntry(ttidNetSetup);
  2467. if ( rListSvrPortMappings.size() == 0 )
  2468. {
  2469. // no Server Port Mappings to set
  2470. return S_OK;
  2471. }
  2472. // 1. Get the INetProtocolSettings interface to control system-wide
  2473. // ICS and firewall settings (i.e port mappings and applications).
  2474. // Create Homenet Configuration Manager COM Instance
  2475. CComPtr<IHNetCfgMgr> spIHNetCfgMgr;
  2476. hr = CoCreateInstance(CLSID_HNetCfgMgr, NULL,
  2477. CLSCTX_ALL,
  2478. IID_IHNetCfgMgr,
  2479. (LPVOID *)&spIHNetCfgMgr);
  2480. if (FAILED(hr))
  2481. {
  2482. TraceTag(ttidError, "%s: CoCreateInstance CLSID_HNetCfgMgr failed: 0x%lx",
  2483. __FUNCNAME__, hr);
  2484. return hr;
  2485. }
  2486. // Get the IHNetProtocolSettings
  2487. CComPtr<IHNetProtocolSettings> spHNetProtocolSettings;
  2488. hr = spIHNetCfgMgr->QueryInterface(IID_IHNetProtocolSettings,
  2489. (LPVOID *)&spHNetProtocolSettings);
  2490. if (FAILED(hr))
  2491. {
  2492. TraceTag(ttidError, "%s: QueryInterface IID_IHNetProtocolSettings failed: 0x%lx",
  2493. __FUNCNAME__, hr);
  2494. return hr;
  2495. }
  2496. // 2. For each enabled CSharedAccessServer x
  2497. // if (there is no PortMapping Protocol corresponding to the
  2498. // x.m_szProtocol and x.m_wInternalPort)
  2499. // call INetProtocolSettings::CreatePortMappingProtocol
  2500. // to get an IHNetPortMappingProtocol interface ptr;
  2501. // else
  2502. // get an existing IHNetPortMappingProtocol interface ptr
  2503. //
  2504. // Get IHNetPortMappingBinding interface ptr by calling
  2505. // IHNetConnection::GetBindingForPortMappingProtocol by passing the
  2506. // IHNetPortMappingProtocol iface ptr on the public IHNetConnection;
  2507. // call IHNetPortMappingBinding::SetTargetComputerName;
  2508. for (list<CSharedAccessServer>::iterator iter =
  2509. rListSvrPortMappings.begin();
  2510. iter != rListSvrPortMappings.end();
  2511. ++iter)
  2512. {
  2513. // migrate the enabled one only
  2514. if ((*iter).m_bSelected)
  2515. {
  2516. UCHAR ucProtocol;
  2517. if (lstrcmpiW( ((*iter).m_szProtocol).c_str(), c_wszTCP) == 0)
  2518. {
  2519. ucProtocol = NAT_PROTOCOL_TCP;
  2520. }
  2521. else if (lstrcmpiW( ((*iter).m_szProtocol).c_str(), c_wszUDP) == 0)
  2522. {
  2523. ucProtocol = NAT_PROTOCOL_UDP;
  2524. }
  2525. else
  2526. {
  2527. // ignore others
  2528. continue;
  2529. }
  2530. USHORT usPort = HTONS((*iter).m_wInternalPort);
  2531. WCHAR* pwszTitle = (WCHAR*) ((*iter).m_szTitle).c_str();
  2532. // get the IHNetPortMappingProtocol iface ptr
  2533. CComPtr<IHNetPortMappingProtocol> spHNetPortMappingProt;
  2534. hr = FindMatchingPortMappingProtocol(
  2535. spHNetProtocolSettings, ucProtocol, usPort,
  2536. &spHNetPortMappingProt);
  2537. if (S_OK == hr)
  2538. {
  2539. TraceTag(ttidNetSetup,
  2540. "%s: Update existing PortMappingProtocol for %S",
  2541. __FUNCNAME__, pwszTitle);
  2542. // change the name of the title
  2543. hr= spHNetPortMappingProt->SetName(pwszTitle);
  2544. if (FAILED(hr))
  2545. {
  2546. TraceTag(ttidError,
  2547. "%s: spHNetPortMappingProt->SetName failed: 0x%lx",
  2548. __FUNCNAME__, hr);
  2549. hr = S_OK; // we still want to invoke
  2550. // IHNetPortMappingBinding::SetEnabled() below.
  2551. }
  2552. }
  2553. else
  2554. {
  2555. TraceTag(ttidNetSetup,
  2556. "%s: CreatePortMappingProtocol for %S",
  2557. __FUNCNAME__, pwszTitle);
  2558. // no existing PortMappingProtocol, create a new one
  2559. hr = spHNetProtocolSettings->CreatePortMappingProtocol(
  2560. pwszTitle, // Title
  2561. ucProtocol, // Protocol
  2562. usPort, // InternalPort
  2563. &spHNetPortMappingProt); // returned iface ptr
  2564. if (FAILED(hr))
  2565. {
  2566. TraceTag(ttidError, "%s: CreatePortMappingProtocol failed: 0x%lx",
  2567. __FUNCNAME__, hr);
  2568. }
  2569. }
  2570. if (S_OK == hr)
  2571. {
  2572. CComPtr<IHNetConnection> spExternalHNetConn;
  2573. if (m_pExternalNetConn == NULL)
  2574. {
  2575. hr = GetExternalINetConnection(&m_pExternalNetConn);
  2576. if (FAILED(hr))
  2577. {
  2578. TraceTag(ttidError, "%s: GetExternalINetConnection failed: 0x%lx",
  2579. __FUNCNAME__, hr);
  2580. return hr; // fatal return
  2581. }
  2582. }
  2583. // get the external/public IHNetConnection iface ptr
  2584. hr = spIHNetCfgMgr->GetIHNetConnectionForINetConnection(
  2585. m_pExternalNetConn,
  2586. &spExternalHNetConn);
  2587. if (FAILED(hr))
  2588. {
  2589. TraceTag(ttidError, "%s: GetIHNetConnectionForINetConnection failed: 0x%lx",
  2590. __FUNCNAME__, hr);
  2591. }
  2592. if (S_OK == hr)
  2593. {
  2594. // Get IHNetPortMappingBinding interface ptr
  2595. CComPtr<IHNetPortMappingBinding> spHNetPortMappingBinding;
  2596. hr = spExternalHNetConn->GetBindingForPortMappingProtocol(
  2597. spHNetPortMappingProt,
  2598. &spHNetPortMappingBinding);
  2599. if (FAILED(hr))
  2600. {
  2601. TraceTag(ttidError,
  2602. "%s: GetBindingForPortMappingProtocol failed: 0x%lx",
  2603. __FUNCNAME__, hr);
  2604. return hr;
  2605. }
  2606. if (S_OK == hr)
  2607. {
  2608. ULONG ulAddress = INADDR_NONE;
  2609. WCHAR* pwszInternalName = (OLECHAR *)((*iter).m_szInternalName).c_str();
  2610. if ( ((*iter).m_szInternalName).length() > 7)
  2611. {
  2612. // 1.2.3.4 -- minimum of 7 characters
  2613. ulAddress = IpPszToHostAddr(pwszInternalName);
  2614. }
  2615. if (INADDR_NONE == ulAddress)
  2616. {
  2617. hr = spHNetPortMappingBinding->SetTargetComputerName(
  2618. pwszInternalName);
  2619. if (FAILED(hr))
  2620. {
  2621. TraceTag(ttidError,
  2622. "%s: SetTargetComputerName failed: 0x%lx",
  2623. __FUNCNAME__, hr);
  2624. }
  2625. }
  2626. else
  2627. {
  2628. hr = spHNetPortMappingBinding->SetTargetComputerAddress(
  2629. HTONL(ulAddress));
  2630. if (FAILED(hr))
  2631. {
  2632. TraceTag(ttidError,
  2633. "%s: SetTargetComputerAddress failed: 0x%lx",
  2634. __FUNCNAME__, hr);
  2635. }
  2636. }
  2637. if (S_OK == hr)
  2638. {
  2639. hr = spHNetPortMappingBinding->SetEnabled(TRUE);
  2640. if (FAILED(hr))
  2641. {
  2642. TraceTag(ttidError,
  2643. "%s: SetEnabled failed: 0x%lx",
  2644. __FUNCNAME__, hr);
  2645. }
  2646. }
  2647. }
  2648. }
  2649. }
  2650. } // end if ((*iter).m_bSelected)
  2651. } //end for
  2652. return S_OK;
  2653. }
  2654. /*++
  2655. Routine Description:
  2656. retrive a matching IHNetPortMappingProtocol object which matches the
  2657. given protocol (ucProtocol) and port number (usPort)
  2658. Arguments:
  2659. [in] pHNetProtocolSettings - IHNetProtocolSettings iface ptr
  2660. [in] ucProtocol - NAT_PROTOCOL_TCP or NAT_PROTOCOL_UDP
  2661. [in] usPort - port number used by this server PortMapping Protocol
  2662. [out] ppHNetPortMappingProtocol - return the matching
  2663. IHNetPortMappingProtocol
  2664. Return Value:
  2665. standard HRESULT
  2666. --*/
  2667. HRESULT CIcsUpgrade::FindMatchingPortMappingProtocol(
  2668. IHNetProtocolSettings* pHNetProtocolSettings,
  2669. UCHAR ucProtocol,
  2670. USHORT usPort,
  2671. IHNetPortMappingProtocol** ppHNetPortMappingProtocol)
  2672. {
  2673. HRESULT hr;
  2674. Assert(pHNetProtocolSettings != NULL);
  2675. Assert(ppHNetPortMappingProtocol != NULL);
  2676. DefineFunctionName("CIcsUpgrade::FindMatchingPortMappingProtocol");
  2677. TraceFunctionEntry(ttidNetSetup);
  2678. *ppHNetPortMappingProtocol = NULL;
  2679. CComPtr<IEnumHNetPortMappingProtocols> spServerEnum;
  2680. hr = pHNetProtocolSettings->EnumPortMappingProtocols(
  2681. &spServerEnum);
  2682. if (FAILED(hr))
  2683. {
  2684. return hr;
  2685. }
  2686. IHNetPortMappingProtocol* pServer;
  2687. ULONG ulCount;
  2688. do
  2689. {
  2690. UCHAR ucFoundProtocol;
  2691. USHORT usFoundPort;
  2692. hr = spServerEnum->Next(
  2693. 1,
  2694. &pServer,
  2695. &ulCount
  2696. );
  2697. if (SUCCEEDED(hr) && 1 == ulCount)
  2698. {
  2699. hr = pServer->GetIPProtocol(&ucFoundProtocol);
  2700. if (FAILED(hr))
  2701. {
  2702. pServer->Release();
  2703. return hr;
  2704. }
  2705. hr = pServer->GetPort(&usFoundPort);
  2706. if (FAILED(hr))
  2707. {
  2708. pServer->Release();
  2709. return hr;
  2710. }
  2711. if (ucFoundProtocol == ucProtocol &&
  2712. usFoundPort == usPort)
  2713. {
  2714. // found mathcing one, transfer value
  2715. *ppHNetPortMappingProtocol = pServer;
  2716. return S_OK;
  2717. }
  2718. pServer->Release();
  2719. }
  2720. } while (SUCCEEDED(hr) && 1 == ulCount);
  2721. return E_FAIL; // not found
  2722. }
  2723. /*++
  2724. Routine Description:
  2725. retrive a matching IHNetApplicationProtocol object which matches the
  2726. given protocol (ucProtocol) and port number (usPort)
  2727. Arguments:
  2728. [in] pHNetProtocolSettings - IHNetProtocolSettings iface ptr
  2729. [in] ucProtocol - NAT_PROTOCOL_TCP or NAT_PROTOCOL_UDP
  2730. [in] usPort - port number used by this server PortMapping Protocol
  2731. [out] ppHNetApplicationProtocol - return the matching
  2732. IHNetApplicationProtocol
  2733. Return Value:
  2734. standard HRESULT
  2735. --*/
  2736. HRESULT CIcsUpgrade::FindMatchingApplicationProtocol(
  2737. IHNetProtocolSettings* pHNetProtocolSettings,
  2738. UCHAR ucProtocol,
  2739. USHORT usPort,
  2740. IHNetApplicationProtocol** ppHNetApplicationProtocol)
  2741. {
  2742. HRESULT hr;
  2743. Assert(pHNetProtocolSettings != NULL);
  2744. Assert(ppHNetApplicationProtocol != NULL);
  2745. DefineFunctionName("CIcsUpgrade::FindMatchingApplicationProtocol");
  2746. TraceFunctionEntry(ttidNetSetup);
  2747. *ppHNetApplicationProtocol = NULL;
  2748. CComPtr<IEnumHNetApplicationProtocols> spAppEnum;
  2749. hr = pHNetProtocolSettings->EnumApplicationProtocols(
  2750. FALSE,
  2751. &spAppEnum);
  2752. if (FAILED(hr))
  2753. {
  2754. return hr;
  2755. }
  2756. IHNetApplicationProtocol* pApp;
  2757. ULONG ulCount;
  2758. do
  2759. {
  2760. UCHAR ucFoundProtocol;
  2761. USHORT usFoundPort;
  2762. hr = spAppEnum->Next(
  2763. 1,
  2764. &pApp,
  2765. &ulCount
  2766. );
  2767. if (SUCCEEDED(hr) && 1 == ulCount)
  2768. {
  2769. hr = pApp->GetOutgoingIPProtocol(&ucFoundProtocol);
  2770. if (FAILED(hr))
  2771. {
  2772. pApp->Release();
  2773. return hr;
  2774. }
  2775. hr = pApp->GetOutgoingPort(&usFoundPort);
  2776. if (FAILED(hr))
  2777. {
  2778. pApp->Release();
  2779. return hr;
  2780. }
  2781. if (ucFoundProtocol == ucProtocol &&
  2782. usFoundPort == usPort)
  2783. {
  2784. // found mathcing one, transfer value
  2785. *ppHNetApplicationProtocol = pApp;
  2786. return S_OK;
  2787. }
  2788. pApp->Release();
  2789. }
  2790. } while (SUCCEEDED(hr) && 1 == ulCount);
  2791. return E_FAIL; // not found
  2792. }
  2793. /*++
  2794. Routine Description:
  2795. Setup ICS misc. settings from m_pIcsUpgradeSettings
  2796. Arguments:
  2797. Return Value:
  2798. standard HRESULT
  2799. --*/
  2800. HRESULT CIcsUpgrade::SetupIcsMiscItems()
  2801. {
  2802. HRESULT hr = S_OK;
  2803. if (NULL == m_pIcsUpgradeSettings)
  2804. {
  2805. return E_UNEXPECTED;
  2806. }
  2807. DefineFunctionName("CIcsUpgrade::SetupIcsMiscItems");
  2808. TraceFunctionEntry(ttidNetSetup);
  2809. if (!m_fICSCreated)
  2810. {
  2811. // no need to continue if ICS has not been created
  2812. return hr;
  2813. }
  2814. // If Win9x/Win2K ICS upgrade, firewall the ICS public connection
  2815. if (m_pIcsUpgradeSettings->fWin9xUpgrade ||
  2816. m_pIcsUpgradeSettings->fWin2KUpgrade)
  2817. {
  2818. INetConnection* rgINetConn[2] = {0, 0};
  2819. if (m_pExternalNetConn)
  2820. {
  2821. rgINetConn[0] = m_pExternalNetConn;
  2822. hr = HrEnablePersonalFirewall(rgINetConn);
  2823. if (FAILED(hr))
  2824. {
  2825. TraceTag(ttidError, "%s: HrEnablePersonalFirewall failed: 0x%lx line: %d",
  2826. __FUNCNAME__, hr, __LINE__);
  2827. NetSetupLogStatusV(
  2828. LogSevInformation,
  2829. SzLoadIds (IDS_TXT_CANT_FIREWALL));
  2830. hr = S_OK; // continue with the rest.
  2831. }
  2832. }
  2833. }
  2834. // Bug# 315265, 315242
  2835. // post-processing to fix up IP Configuration on private connection
  2836. // (1) if there is at least one internal adapter couldn't be upgraded from
  2837. // Win9x ICS upgrade, we have to make sure the survival internal adapter
  2838. // has static IP address 192.168.0.1, subnet mask 255.255.255.0
  2839. // (2) if we create a bridge for Win9x ICS upgrade, we need to set static
  2840. // IP address 192.168.0.1, subnet mask 255.255.255.0 for TcpIp bound to
  2841. // the bridge.
  2842. // Note: (1) and (2) are exclusive.
  2843. //
  2844. // For WinXP Unattended clean install,
  2845. // (3) if an internal ICS adapter is found, set static IP address
  2846. // 192.168.0.1, subnet mask 255.255.255.0 for the TCPIP bound to the
  2847. // internal adapter.
  2848. // (4) if ICS is enabled and the bridge is the ICS private connection.
  2849. // Set static IP address 192.168.0.1, subnet mask 255.255.255.0 for
  2850. // TcpIp bound to the bridge.
  2851. // Note: (3) and (4) are exclusive.
  2852. if ( m_pIcsUpgradeSettings->fInternalAdapterFound &&
  2853. (m_pIcsUpgradeSettings->fWin9xUpgradeAtLeastOneInternalAdapterBroken ||
  2854. m_pIcsUpgradeSettings->fXpUnattended) )
  2855. {
  2856. hr = SetPrivateIpConfiguration(m_pIcsUpgradeSettings->guidInternal);
  2857. if (FAILED(hr))
  2858. {
  2859. // logging, but we'll continue to do the rest of the ICS upgrade though.
  2860. TraceTag(ttidError,
  2861. "%s: SetPrivateIpConfiguration on private adapter failed: 0x%lx",
  2862. __FUNCNAME__, hr);
  2863. NetSetupLogStatusV(
  2864. LogSevInformation,
  2865. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  2866. }
  2867. }
  2868. else if ( ((m_pIcsUpgradeSettings->listBridge).size() >= 2) &&
  2869. !m_pIcsUpgradeSettings->fInternalAdapterFound)
  2870. {
  2871. // A bridge was created as ICS private.
  2872. // We need to setup its static IP Configuration since
  2873. // these settings are not configured when it was created a while ago.
  2874. GUID guidBridge;
  2875. hr = GetBridgeGuid(guidBridge);
  2876. if (SUCCEEDED(hr))
  2877. {
  2878. hr = SetPrivateIpConfiguration(guidBridge);
  2879. if (FAILED(hr))
  2880. {
  2881. // logging, but we'll continue to do the rest of the ICS upgrade though.
  2882. TraceTag(ttidError,
  2883. "%s: SetPrivateIpConfiguration for bridge failed: 0x%lx",
  2884. __FUNCNAME__, hr);
  2885. NetSetupLogStatusV(
  2886. LogSevInformation,
  2887. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  2888. }
  2889. }
  2890. else
  2891. {
  2892. // logging, but we'll continue to do the rest of the ICS upgrade though.
  2893. TraceTag(ttidError,
  2894. "%s: GetBridgeGuid failed: 0x%lx",
  2895. __FUNCNAME__, hr);
  2896. NetSetupLogStatusV(
  2897. LogSevInformation,
  2898. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  2899. }
  2900. }
  2901. LPRASSHARECONN pRasShareConn = &(m_pIcsUpgradeSettings->rscExternal);
  2902. // set the fShowTrayIcon to the external INetConnection
  2903. // note: this property has already migrated for Win2K upgrade
  2904. if ( (m_pExternalNetConn) &&
  2905. (m_pIcsUpgradeSettings->fWin9xUpgrade ||
  2906. m_pIcsUpgradeSettings->fXpUnattended) )
  2907. {
  2908. if (pRasShareConn->fIsLanConnection)
  2909. {
  2910. INetLanConnection* pLanConn = NULL;
  2911. hr = HrQIAndSetProxyBlanket(m_pExternalNetConn, &pLanConn);
  2912. if (SUCCEEDED(hr) && pLanConn)
  2913. {
  2914. LANCON_INFO linfo = {0};
  2915. linfo.fShowIcon = m_pIcsUpgradeSettings->fShowTrayIcon;
  2916. // Set new value of show icon property
  2917. hr = pLanConn->SetInfo(LCIF_ICON, &linfo);
  2918. if (FAILED(hr))
  2919. {
  2920. TraceTag(ttidError, "%s: pLanConn->SetInfo failed: 0x%lx",
  2921. __FUNCNAME__, hr);
  2922. }
  2923. pLanConn->Release();
  2924. }
  2925. else
  2926. {
  2927. TraceTag(
  2928. ttidError,
  2929. "%s: HrQIAndSetProxyBlanket for INetLanConnection failed: 0x%lx",
  2930. __FUNCNAME__, hr);
  2931. }
  2932. }
  2933. else
  2934. {
  2935. // this is a WAN connection
  2936. RASCON_INFO rci;
  2937. LPRASENTRY pRasEntry = NULL;
  2938. DWORD dwRasEntrySize;
  2939. hr = HrRciGetRasConnectionInfo(m_pExternalNetConn, &rci);
  2940. if (SUCCEEDED(hr))
  2941. {
  2942. hr = HrRasGetEntryProperties(rci.pszwPbkFile, rci.pszwEntryName,
  2943. &pRasEntry, &dwRasEntrySize);
  2944. if (SUCCEEDED(hr) && pRasEntry)
  2945. {
  2946. DWORD dwRet;
  2947. if (m_pIcsUpgradeSettings->fShowTrayIcon)
  2948. pRasEntry->dwfOptions |= RASEO_ModemLights;
  2949. else
  2950. pRasEntry->dwfOptions &= (~RASEO_ModemLights);
  2951. dwRet = RasSetEntryProperties(rci.pszwPbkFile, rci.pszwEntryName,
  2952. pRasEntry, dwRasEntrySize, 0, 0);
  2953. if (0 != dwRet)
  2954. {
  2955. TraceTag(
  2956. ttidError,
  2957. "%s: RasSetEntryProperties failed: 0x%lx",
  2958. __FUNCNAME__, GetLastError());
  2959. hr = HRESULT_FROM_WIN32(GetLastError());
  2960. }
  2961. MemFree(pRasEntry);
  2962. }
  2963. else
  2964. {
  2965. TraceTag(
  2966. ttidError,
  2967. "%s: HrRasGetEntryProperties failed: 0x%lx",
  2968. __FUNCNAME__, hr);
  2969. }
  2970. RciFree (&rci);
  2971. }
  2972. else
  2973. {
  2974. TraceTag(
  2975. ttidError,
  2976. "%s: HrRciGetRasConnectionInfo failed: 0x%lx",
  2977. __FUNCNAME__, hr);
  2978. }
  2979. }
  2980. }
  2981. // Set the dial on demand setting.
  2982. // We don't need to do it for 2 cases:
  2983. // 1. Win2K ICS upgrade because the registry settings is already migrated.
  2984. // 2. the ICS public is a LAN connection.
  2985. if (pRasShareConn->fIsLanConnection || m_pIcsUpgradeSettings->fWin2KUpgrade)
  2986. {
  2987. // no need to set dial on-demand for external lan connection
  2988. return hr;
  2989. }
  2990. // Get the IHNetIcsSettings interface to control system-wide
  2991. // ICS settings
  2992. // Create Homenet Configuration Manager COM Instance
  2993. CComPtr<IHNetCfgMgr> spIHNetCfgMgr;
  2994. hr = CoCreateInstance(CLSID_HNetCfgMgr, NULL,
  2995. CLSCTX_ALL,
  2996. IID_IHNetCfgMgr,
  2997. (LPVOID *)&spIHNetCfgMgr);
  2998. if (FAILED(hr))
  2999. {
  3000. TraceTag(ttidError, "%s: CoCreateInstance CLSID_HNetCfgMgr failed: 0x%lx",
  3001. __FUNCNAME__, hr);
  3002. return hr;
  3003. }
  3004. // Get the IHNetIcsSettings
  3005. CComPtr<IHNetIcsSettings> spHNetIcsSettings;
  3006. hr = spIHNetCfgMgr->QueryInterface(IID_IHNetIcsSettings,
  3007. (LPVOID *)&spHNetIcsSettings);
  3008. if (FAILED(hr))
  3009. {
  3010. TraceTag(ttidError,
  3011. "%s: QueryInterface IID_IHNetIcsSettings failed: 0x%lx",
  3012. __FUNCNAME__, hr);
  3013. return hr;
  3014. }
  3015. hr = spHNetIcsSettings->SetAutodialSettings(!!(m_pIcsUpgradeSettings->fDialOnDemand));
  3016. if (FAILED(hr))
  3017. {
  3018. TraceTag(ttidError,
  3019. "%s: Ignore SetAutodialSettings failed: 0x%lx",
  3020. __FUNCNAME__, hr);
  3021. hr = S_OK;
  3022. }
  3023. return hr;
  3024. }
  3025. /*++
  3026. Routine Description:
  3027. The interface given by rAdapterGuid will be configured to
  3028. use ICS static IP 192.168.0.1, subnet mask 255.255.255.0
  3029. Arguments:
  3030. [in] rInterfaceGuid -- the Guid of the TcpIp interface
  3031. Return Value:
  3032. standard HRESULT
  3033. --*/
  3034. HRESULT CIcsUpgrade::SetPrivateIpConfiguration(IN GUID& rInterfaceGuid)
  3035. {
  3036. HRESULT hr = S_OK;
  3037. HKEY hkeyTcpipInterface = NULL;
  3038. DefineFunctionName("CIcsUpgrade::SetPrivateIpConfiguration");
  3039. TraceFunctionEntry(ttidNetSetup);
  3040. hr = OpenTcpipInterfaceKey(rInterfaceGuid, &hkeyTcpipInterface);
  3041. if (FAILED(hr))
  3042. {
  3043. TraceTag(ttidError,
  3044. "%s: OpenTcpipInterfaceKey failed: 0x%lx",
  3045. __FUNCNAME__, hr);
  3046. return hr;
  3047. }
  3048. Assert(hkeyTcpipInterface);
  3049. // IPAddress
  3050. hr = HrRegSetMultiSz(hkeyTcpipInterface,
  3051. c_wszIPAddress,
  3052. c_mszScopeAddress);
  3053. if(FAILED(hr))
  3054. {
  3055. TraceTag(ttidError,
  3056. "%s: failed to set %S to the registry. hr = 0x%lx.",
  3057. __FUNCNAME__, c_wszIPAddress,
  3058. hr);
  3059. goto Error;
  3060. }
  3061. // SubnetMask
  3062. hr = HrRegSetMultiSz(hkeyTcpipInterface,
  3063. c_wszSubnetMask,
  3064. c_mszScopeMask);
  3065. if(FAILED(hr))
  3066. {
  3067. TraceTag(ttidError,
  3068. "%s: failed to set %S to the registry. hr = 0x%lx.",
  3069. __FUNCNAME__, c_wszIPAddress,
  3070. hr);
  3071. goto Error;
  3072. }
  3073. // EnableDHCP
  3074. hr = HrRegSetDword(hkeyTcpipInterface,
  3075. c_wszEnableDHCP,
  3076. 0);
  3077. if(FAILED(hr))
  3078. {
  3079. TraceTag(ttidError,
  3080. "%s: failed to set %S to the registry. hr = 0x%lx.",
  3081. __FUNCNAME__, c_wszEnableDHCP,
  3082. hr);
  3083. goto Error;
  3084. }
  3085. Error:
  3086. RegSafeCloseKey(hkeyTcpipInterface);
  3087. return hr;
  3088. }
  3089. /*++
  3090. Routine Description:
  3091. Get the interface guid of a bridge.
  3092. Arguments:
  3093. [out] rInterfaceGuid -- receive the Guid from the bridge interface
  3094. Return Value:
  3095. standard HRESULT
  3096. --*/
  3097. HRESULT CIcsUpgrade::GetBridgeGuid(OUT GUID& rInterfaceGuid)
  3098. {
  3099. HRESULT hr = S_OK;
  3100. BOOLEAN fFound = FALSE;
  3101. INetConnection* pConn = NULL;
  3102. ULONG ulCount = 0;
  3103. DefineFunctionName("CIcsUpgrade::GetBridgeGuid");
  3104. TraceFunctionEntry(ttidNetSetup);
  3105. // we don't use the cached m_spEnum because it may be in a stale
  3106. // state (a bridge has just been created).
  3107. CComPtr<IEnumNetConnection> spEnum;
  3108. // Get the net connection manager
  3109. INetConnectionManager* pConnMan = NULL;
  3110. hr = CoCreateInstance(CLSID_ConnectionManager, NULL,
  3111. CLSCTX_ALL,
  3112. IID_INetConnectionManager,
  3113. (LPVOID *)&pConnMan);
  3114. if (SUCCEEDED(hr))
  3115. {
  3116. // Get the enumeration of connections
  3117. SetProxyBlanket(pConnMan);
  3118. hr = pConnMan->EnumConnections(NCME_DEFAULT, &spEnum);
  3119. pConnMan->Release(); // don't need this anymore
  3120. if (SUCCEEDED(hr))
  3121. {
  3122. SetProxyBlanket(spEnum);
  3123. }
  3124. else
  3125. {
  3126. TraceTag(ttidError,
  3127. "%s: EnumConnections failed: 0x%lx.",
  3128. __FUNCNAME__, hr);
  3129. return hr;
  3130. }
  3131. }
  3132. else
  3133. {
  3134. TraceTag(ttidError,
  3135. "%s: CoCreateInstance failed: 0x%lx.",
  3136. __FUNCNAME__, hr);
  3137. return hr;
  3138. }
  3139. hr = spEnum->Reset();
  3140. if (FAILED(hr))
  3141. {
  3142. TraceTag(ttidError,
  3143. "%s: Reset failed: 0x%lx.",
  3144. __FUNCNAME__, hr);
  3145. return hr;
  3146. }
  3147. do
  3148. {
  3149. NETCON_PROPERTIES* pProps;
  3150. hr = spEnum->Next(1, &pConn, &ulCount);
  3151. if (SUCCEEDED(hr) && 1 == ulCount)
  3152. {
  3153. SetProxyBlanket(pConn);
  3154. hr = pConn->GetProperties(&pProps);
  3155. if (SUCCEEDED(hr))
  3156. {
  3157. if (NCM_BRIDGE == pProps->MediaType)
  3158. {
  3159. // transfer value
  3160. rInterfaceGuid = pProps->guidId;
  3161. fFound = TRUE;
  3162. }
  3163. NcFreeNetconProperties(pProps);
  3164. }
  3165. pConn->Release();
  3166. }
  3167. } while (FALSE == fFound && SUCCEEDED(hr) && 1 == ulCount);
  3168. // Normalize hr
  3169. hr = (fFound ? S_OK : E_FAIL);
  3170. return hr;
  3171. }
  3172. /*++
  3173. Routine Description:
  3174. Get the INetConnection of a bridge.
  3175. Arguments:
  3176. [out] ppINetConn -- receive the INetConnection from the bridge interface
  3177. Return Value:
  3178. standard HRESULT
  3179. --*/
  3180. HRESULT CIcsUpgrade::GetBridgeINetConn(OUT INetConnection** ppINetConn)
  3181. {
  3182. HRESULT hr = S_OK;
  3183. BOOLEAN fFound = FALSE;
  3184. INetConnection* pConn = NULL;
  3185. ULONG ulCount = 0;
  3186. DefineFunctionName("CIcsUpgrade::GetBridgeINetConn");
  3187. TraceFunctionEntry(ttidNetSetup);
  3188. Assert(ppINetConn);
  3189. *ppINetConn = NULL;
  3190. // we don't use the cached m_spEnum because it may be in a stale
  3191. // state (a bridge has just been created).
  3192. CComPtr<IEnumNetConnection> spEnum;
  3193. // Get the net connection manager
  3194. INetConnectionManager* pConnMan = NULL;
  3195. hr = CoCreateInstance(CLSID_ConnectionManager, NULL,
  3196. CLSCTX_ALL,
  3197. IID_INetConnectionManager,
  3198. (LPVOID *)&pConnMan);
  3199. if (SUCCEEDED(hr))
  3200. {
  3201. // Get the enumeration of connections
  3202. SetProxyBlanket(pConnMan);
  3203. hr = pConnMan->EnumConnections(NCME_DEFAULT, &spEnum);
  3204. pConnMan->Release(); // don't need this anymore
  3205. if (SUCCEEDED(hr))
  3206. {
  3207. SetProxyBlanket(spEnum);
  3208. }
  3209. else
  3210. {
  3211. TraceTag(ttidError,
  3212. "%s: EnumConnections failed: 0x%lx.",
  3213. __FUNCNAME__, hr);
  3214. return hr;
  3215. }
  3216. }
  3217. else
  3218. {
  3219. TraceTag(ttidError,
  3220. "%s: CoCreateInstance failed: 0x%lx.",
  3221. __FUNCNAME__, hr);
  3222. return hr;
  3223. }
  3224. hr = spEnum->Reset();
  3225. if (FAILED(hr))
  3226. {
  3227. TraceTag(ttidError,
  3228. "%s: Reset failed: 0x%lx.",
  3229. __FUNCNAME__, hr);
  3230. return hr;
  3231. }
  3232. do
  3233. {
  3234. NETCON_PROPERTIES* pProps;
  3235. hr = spEnum->Next(1, &pConn, &ulCount);
  3236. if (SUCCEEDED(hr) && 1 == ulCount)
  3237. {
  3238. SetProxyBlanket(pConn);
  3239. hr = pConn->GetProperties(&pProps);
  3240. if (SUCCEEDED(hr))
  3241. {
  3242. if (NCM_BRIDGE == pProps->MediaType)
  3243. {
  3244. // transfer value
  3245. *ppINetConn = pConn;
  3246. fFound = TRUE;
  3247. }
  3248. NcFreeNetconProperties(pProps);
  3249. }
  3250. if (!fFound)
  3251. {
  3252. pConn->Release();
  3253. }
  3254. }
  3255. } while (FALSE == fFound && SUCCEEDED(hr) && 1 == ulCount);
  3256. // Normalize hr
  3257. hr = (fFound ? S_OK : E_FAIL);
  3258. return hr;
  3259. }
  3260. /*++
  3261. Routine Description:
  3262. Open the Ip Configuration registry Key of an interface
  3263. Arguments:
  3264. [in] rGuid -- the Guid of the TcpIp interface
  3265. [out] phKey -- receives the opened key
  3266. Return Value:
  3267. standard HRESULT
  3268. --*/
  3269. HRESULT CIcsUpgrade::OpenTcpipInterfaceKey(
  3270. IN GUID& rGuid,
  3271. OUT PHKEY phKey)
  3272. {
  3273. HRESULT hr;
  3274. LPWSTR wszSubKeyName;
  3275. ULONG ulSubKeyNameLength;
  3276. LPWSTR wszGuid;
  3277. Assert(phKey);
  3278. hr = StringFromCLSID(rGuid, &wszGuid);
  3279. if (SUCCEEDED(hr))
  3280. {
  3281. ulSubKeyNameLength =
  3282. wcslen(c_wszTcpipParametersKey) + 1 +
  3283. wcslen(c_wszInterfaces) + 1 +
  3284. wcslen(wszGuid) + 2;
  3285. wszSubKeyName = new WCHAR[ulSubKeyNameLength];
  3286. if (NULL != wszSubKeyName)
  3287. {
  3288. swprintf(
  3289. wszSubKeyName,
  3290. L"%ls\\%ls\\%ls",
  3291. c_wszTcpipParametersKey,
  3292. c_wszInterfaces,
  3293. wszGuid
  3294. );
  3295. }
  3296. else
  3297. {
  3298. hr = E_OUTOFMEMORY;
  3299. }
  3300. CoTaskMemFree(wszGuid);
  3301. }
  3302. if (SUCCEEDED(hr))
  3303. {
  3304. hr = HrRegOpenKeyEx(HKEY_LOCAL_MACHINE, wszSubKeyName, KEY_ALL_ACCESS, phKey);
  3305. delete [] wszSubKeyName;
  3306. }
  3307. return hr;
  3308. }
  3309. /*++
  3310. Routine Description:
  3311. Create a named event to notify other components that we're
  3312. in GUI Mode Setup of Win2K ICS Upgrade
  3313. Arguments:
  3314. Return Value:
  3315. standard HRESULT
  3316. --*/
  3317. HRESULT CIcsUpgrade::CreateIcsUpgradeNamedEvent()
  3318. {
  3319. DefineFunctionName("CIcsUpgrade::CreateIcsUpgradeNamedEvent");
  3320. TraceFunctionEntry(ttidNetSetup);
  3321. // create an auto-reset, nonsingaled, named event
  3322. m_hIcsUpgradeEvent = CreateEvent(NULL, FALSE, FALSE, c_wszIcsUpgradeEventName);
  3323. if (NULL == m_hIcsUpgradeEvent)
  3324. {
  3325. TraceTag(ttidError,
  3326. "%s: CreateEvent failed: 0x%lx",
  3327. __FUNCNAME__, GetLastError());
  3328. NetSetupLogStatusV(
  3329. LogSevError,
  3330. SzLoadIds (IDS_TXT_CANT_UPGRADE_ICS));
  3331. return HRESULT_FROM_WIN32(GetLastError());
  3332. }
  3333. return S_OK;
  3334. }
  3335. //--------- HNet helpers end ------------------------------------