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.

3107 lines
95 KiB

  1. #include "precomp.h"
  2. #pragma hdrstop
  3. //
  4. // Names of wizard page types, do not change order unless
  5. // the WizardPagesType enum is changed.
  6. //
  7. LPCTSTR WizardPagesTypeNames[WizPagesTypeMax] = { TEXT("Welcome"), TEXT("Mode"),
  8. TEXT("Early") , TEXT("Prenet"),
  9. TEXT("Postnet"), TEXT("Late"),
  10. TEXT("Final")
  11. };
  12. //
  13. // Name of sections and keys in infs.
  14. //
  15. LPCTSTR szComponents = TEXT("Components");
  16. LPCTSTR szOptionalComponents = TEXT("Optional Components");
  17. LPCTSTR szExtraSetupFiles = TEXT("ExtraSetupFiles");
  18. LPCTSTR szNeeds = TEXT("Needs");
  19. LPCTSTR szExclude = TEXT("Exclude");
  20. LPCTSTR szParent = TEXT("Parent");
  21. LPCTSTR szIconIndex = TEXT("IconIndex");
  22. LPCTSTR szModes = TEXT("Modes");
  23. LPCTSTR szTip = TEXT("Tip");
  24. LPCTSTR szOptionDesc = TEXT("OptionDesc");
  25. LPCTSTR szInstalledFlag = TEXT("InstalledFlag");
  26. LPCTSTR szHide = TEXT("HIDE");
  27. LPCTSTR szOSSetupOnly = TEXT("OSSetupOnly");
  28. LPCTSTR szStandaloneOnly = TEXT("StandaloneOnly");
  29. LPCTSTR szPageTitle = TEXT("PageTitles");
  30. LPCTSTR szSetupTitle = TEXT("SetupPage");
  31. LPCTSTR szGlobal = TEXT("Global");
  32. LPCTSTR szWindowTitle = TEXT("WindowTitle");
  33. LPCTSTR szWindowTitleAlone = TEXT("WindowTitle.StandAlone");
  34. LPCTSTR szSizeApproximation = TEXT("SizeApproximation");
  35. LPCTSTR szWindowTitleInternal = TEXT("*");
  36. //
  37. // Key in registry where private component data is kept.
  38. // We form a unique name within this key for the OC Manager
  39. // instantiation.
  40. //
  41. LPCTSTR szOcManagerRoot = TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Setup\\OC Manager");
  42. LPCTSTR szPrivateDataRoot = TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Setup\\OC Manager\\TemporaryData");
  43. LPCTSTR szMasterInfs = TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Setup\\OC Manager\\MasterInfs");
  44. LPCTSTR szSubcompList = TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Setup\\OC Manager\\Subcomponents");
  45. LPCTSTR szOcManagerErrors = TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Setup\\OC Manager\\Errors");
  46. //
  47. // Other string constants.
  48. //
  49. LPCTSTR szSetupDir = TEXT("Setup");
  50. //
  51. // locale information
  52. //
  53. LOCALE locale;
  54. // Structure used for string table callback when
  55. // building a subcomponent list.
  56. //
  57. typedef struct _BUILDSUBCOMPLIST_PARAMS {
  58. POC_MANAGER OcManager;
  59. UINT Pass;
  60. } BUILDSUBCOMPLIST_PARAMS, *PBUILDSUBCOMPLIST_PARAMS;
  61. //
  62. // oc manager pointer for debugging/logging
  63. //
  64. POC_MANAGER gLastOcManager = NULL;
  65. UINT
  66. pOcQueryOrSetNewInf(
  67. IN PCTSTR MasterOcInfName,
  68. OUT PTSTR SuiteName,
  69. IN DWORD Operation
  70. );
  71. VOID
  72. pOcDestroyPerOcData(
  73. IN POC_MANAGER OcManager
  74. );
  75. BOOL
  76. pOcDestroyPerOcDataStringCB(
  77. IN PVOID StringTable,
  78. IN LONG StringId,
  79. IN PCTSTR String,
  80. IN POPTIONAL_COMPONENT Oc,
  81. IN UINT OcSize,
  82. IN LPARAM Unused
  83. );
  84. VOID
  85. pOcClearAllErrorStates(
  86. IN POC_MANAGER OcManager
  87. );
  88. BOOL
  89. FileExists(
  90. IN PCTSTR FileName,
  91. OUT PWIN32_FIND_DATA FindData OPTIONAL
  92. )
  93. /*++
  94. Routine Description:
  95. Determine if a file exists and is accessible.
  96. Errormode is set (and then restored) so the user will not see
  97. any pop-ups.
  98. Arguments:
  99. FileName - supplies full path of file to check for existance.
  100. FindData - if specified, receives find data for the file.
  101. Return Value:
  102. TRUE if the file exists and is accessible.
  103. FALSE if not. GetLastError() returns extended error info.
  104. --*/
  105. {
  106. WIN32_FIND_DATA findData;
  107. HANDLE FindHandle;
  108. UINT OldMode;
  109. DWORD Error;
  110. OldMode = SetErrorMode(SEM_FAILCRITICALERRORS);
  111. FindHandle = FindFirstFile(FileName,&findData);
  112. if(FindHandle == INVALID_HANDLE_VALUE) {
  113. Error = GetLastError();
  114. } else {
  115. FindClose(FindHandle);
  116. if(FindData) {
  117. *FindData = findData;
  118. }
  119. Error = NO_ERROR;
  120. }
  121. SetErrorMode(OldMode);
  122. SetLastError(Error);
  123. return (Error == NO_ERROR);
  124. }
  125. VOID
  126. pOcFormSuitePath(
  127. IN LPCTSTR SuiteName,
  128. IN LPCTSTR FileName, OPTIONAL
  129. OUT LPTSTR FullPath
  130. )
  131. /*++
  132. Routine Description:
  133. Forms the name of the directory in the OS tree where per-suite
  134. infs and installation dlls are kept (system32\setup).
  135. Optionally also appends the name of a file to the path.
  136. Arguments:
  137. SuiteName - shortname for the suite.
  138. FileName - optionally specifies the name of a file in the per-suite
  139. directory.
  140. FullPath - receives the full path of the per-suite directory (or the
  141. file within the directory). This buffer should be MAX_PATH TCHAR
  142. elements.
  143. Return Value:
  144. None.
  145. --*/
  146. {
  147. UNREFERENCED_PARAMETER(SuiteName);
  148. GetSystemDirectory(FullPath,MAX_PATH);
  149. pSetupConcatenatePaths(FullPath,szSetupDir,MAX_PATH,NULL);
  150. //
  151. // We put all such files in a single flat directory.
  152. // This makes life easier for components that want to share
  153. // installation pieces, such as infs, dlls, etc.
  154. //
  155. // There are potential name conflict issues but we blow them off.
  156. //
  157. //pSetupConcatenatePaths(FullPath,SuiteName,MAX_PATH,NULL);
  158. if(FileName) {
  159. pSetupConcatenatePaths(FullPath,FileName,MAX_PATH,NULL);
  160. }
  161. }
  162. BOOL
  163. pOcBuildSubcomponentListStringCB(
  164. IN PVOID StringTable,
  165. IN LONG StringId,
  166. IN PCTSTR String,
  167. IN POC_INF OcInf,
  168. IN UINT OcInfSize,
  169. IN PBUILDSUBCOMPLIST_PARAMS Params
  170. )
  171. /*++
  172. Routine Description:
  173. String table callback, worker routine for pOcBuildSubcomponentLists.
  174. This routine examines the loaded per-component infs and builds the
  175. subcomponent hierarchies that are described therein via the Parent=
  176. lines in the various per-component sections.
  177. Arguments:
  178. Standard string table callback args.
  179. Return Value:
  180. Boolean indicating outcome. If FALSE, an error will have been logged.
  181. FALSE also stops the string table enumeration and causes pSetupStringTableEnum()
  182. to return FALSE.
  183. --*/
  184. {
  185. OPTIONAL_COMPONENT OptionalComponent;
  186. OPTIONAL_COMPONENT AuxOc;
  187. INFCONTEXT LineContext;
  188. INFCONTEXT SublineContext;
  189. LPCTSTR SubcompName;
  190. LPCTSTR ModuleFlags;
  191. LPCTSTR p;
  192. LONG l;
  193. LONG CurrentStringId;
  194. UINT u,n;
  195. INT IconIndex;
  196. POC_MANAGER OcManager = Params->OcManager;
  197. if(SetupFindFirstLine(OcInf->Handle,szOptionalComponents,NULL,&LineContext)) {
  198. do {
  199. if((SubcompName = pSetupGetField(&LineContext,1)) && *SubcompName) {
  200. l = pSetupStringTableLookUpStringEx(
  201. Params->OcManager->ComponentStringTable,
  202. (PTSTR)SubcompName,
  203. STRTAB_CASE_INSENSITIVE,
  204. &OptionalComponent,
  205. sizeof(OPTIONAL_COMPONENT)
  206. );
  207. if(Params->Pass == 0) {
  208. //
  209. // First pass. Add subcomponents listed in [Optional Components]
  210. // to the string table. Each one has an associated OPTIONAL_COMPONENT
  211. // structure. Top-level components already exist in the table,
  212. // so we are careful here about how we overwrite existing entries.
  213. //
  214. if(l == -1) {
  215. ZeroMemory(&OptionalComponent,sizeof(OPTIONAL_COMPONENT));
  216. OptionalComponent.ParentStringId = -1;
  217. OptionalComponent.FirstChildStringId = -1;
  218. OptionalComponent.NextSiblingStringId = -1;
  219. OptionalComponent.InfStringId = StringId;
  220. OptionalComponent.Exists = FALSE;
  221. }
  222. if (OptionalComponent.Exists) {
  223. _LogError(OcManager,
  224. OcErrLevError,
  225. MSG_OC_DUPLICATE_COMPONENT,
  226. SubcompName);
  227. continue;
  228. }
  229. OptionalComponent.Exists = TRUE;
  230. // Get the second Field of the Optional components line
  231. // Determine if this component is hidden or not
  232. ModuleFlags = pSetupGetField(&LineContext,2);
  233. if (ModuleFlags) {
  234. if (OcManager->SetupData.OperationFlags & SETUPOP_STANDALONE)
  235. p = szOSSetupOnly;
  236. else
  237. p = szStandaloneOnly;
  238. if (!_tcsicmp(ModuleFlags, szHide) || !_tcsicmp(ModuleFlags, p))
  239. OptionalComponent.InternalFlags |= OCFLAG_HIDE;
  240. }
  241. //
  242. // Fetch the description, tip, and iconindex.
  243. //
  244. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szOptionDesc,&SublineContext)
  245. && (p = pSetupGetField(&SublineContext,1))) {
  246. lstrcpyn(OptionalComponent.Description,p,MAXOCDESC);
  247. } else {
  248. OptionalComponent.Description[0] = 0;
  249. }
  250. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szTip,&SublineContext)
  251. && (p = pSetupGetField(&SublineContext,1))) {
  252. lstrcpyn(OptionalComponent.Tip,p,MAXOCTIP);
  253. } else {
  254. OptionalComponent.Tip[0] = 0;
  255. }
  256. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szIconIndex,&SublineContext)
  257. && (p = pSetupGetField(&SublineContext,1))) {
  258. LPCTSTR p2,p3;
  259. //
  260. // If we have fields 2 and 3 then assume we've got a dll
  261. // and resource name. Otherwise it's an index or *.
  262. //
  263. if((p2 = pSetupGetField(&SublineContext,2))
  264. && (p3 = pSetupGetField(&SublineContext,3))) {
  265. lstrcpyn(
  266. OptionalComponent.IconDll,
  267. p2,
  268. sizeof(OptionalComponent.IconDll)/sizeof(TCHAR)
  269. );
  270. lstrcpyn(
  271. OptionalComponent.IconResource,
  272. p3,
  273. sizeof(OptionalComponent.IconResource)/sizeof(TCHAR)
  274. );
  275. IconIndex = -2;
  276. } else {
  277. //
  278. // If the icon index is * then stick -1 in there
  279. // as a special marker value for later.
  280. // Otherwise we call SetupGetIntField because it will
  281. // validate the field for us.
  282. //
  283. if((p[0] == TEXT('*')) && (p[1] == 0)) {
  284. IconIndex = -1;
  285. } else {
  286. if(!SetupGetIntField(&SublineContext,1,&IconIndex)) {
  287. IconIndex = DEFAULT_ICON_INDEX;
  288. } else {
  289. if (IconIndex < 0 || IconIndex > 66)
  290. IconIndex = DEFAULT_ICON_INDEX;
  291. }
  292. }
  293. }
  294. } else {
  295. //
  296. // No icon index.
  297. //
  298. IconIndex = DEFAULT_ICON_INDEX;
  299. }
  300. OptionalComponent.IconIndex = IconIndex;
  301. //
  302. // if the InstalledFlag is specified, check it
  303. // and set the original selection state accordingly
  304. //
  305. OptionalComponent.InstalledState = INSTSTATE_UNKNOWN;
  306. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szInstalledFlag,&SublineContext)
  307. && (p = pSetupGetField(&SublineContext,1))) {
  308. TCHAR regkey[MAXOCIFLAG];
  309. lstrcpyn(regkey,p,MAXOCIFLAG);
  310. if (p = pSetupGetField(&SublineContext,2)) {
  311. TCHAR regval[MAXOCIFLAG];
  312. TCHAR buf[MAXOCIFLAG];
  313. HKEY hkey;
  314. DWORD size;
  315. DWORD type;
  316. DWORD rc;
  317. lstrcpyn(regval,p,MAXOCIFLAG);
  318. if (RegOpenKey(HKEY_LOCAL_MACHINE, regkey, &hkey) == ERROR_SUCCESS) {
  319. size = sizeof(buf);
  320. rc = RegQueryValueEx(hkey,
  321. regval,
  322. NULL,
  323. &type,
  324. (LPBYTE)buf,
  325. &size);
  326. RegCloseKey(hkey);
  327. if (rc == ERROR_SUCCESS) {
  328. OptionalComponent.InstalledState = INSTSTATE_YES;
  329. } else {
  330. OptionalComponent.InstalledState = INSTSTATE_NO;
  331. }
  332. }
  333. }
  334. }
  335. //
  336. // Fetch the list of modes in which the subcomponent should be
  337. // on by default. For future expandability, we'll accept any
  338. // mode values up to 31, which is the number of bits we can fit
  339. // in out UINT bitfield/
  340. //
  341. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szModes,&SublineContext)) {
  342. n = SetupGetFieldCount(&SublineContext);
  343. for(u=0; u<n; u++) {
  344. if(SetupGetIntField(&SublineContext,u+1,&IconIndex)
  345. && ((DWORD)IconIndex < 32)) {
  346. OptionalComponent.ModeBits |= (1 << IconIndex);
  347. }
  348. }
  349. }
  350. //
  351. // As an optimization, fetch the size approximation, if they
  352. // supplied one.If they didn't supply this then we have to
  353. // query them for disk space
  354. //
  355. //
  356. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szSizeApproximation,&SublineContext)
  357. && (p = pSetupGetField(&SublineContext,1))) {
  358. //
  359. // we have the text version of something that needs to be converted into
  360. // a LONGLONG...
  361. //
  362. pConvertStringToLongLong(p,&OptionalComponent.SizeApproximation);
  363. OptionalComponent.InternalFlags |= OCFLAG_APPROXSPACE;
  364. }
  365. // Find the The "TopLevelParent" for this Node
  366. // Search the list if TopLevelComponent looking for
  367. // the "INF string id" that matches the Inf String ID
  368. // of this component.
  369. for(u=0; u<OcManager->TopLevelOcCount; u++) {
  370. pSetupStringTableGetExtraData(
  371. OcManager->ComponentStringTable,
  372. OcManager->TopLevelOcStringIds[u],
  373. &AuxOc,
  374. sizeof(OPTIONAL_COMPONENT)
  375. );
  376. if(AuxOc.InfStringId == StringId) {
  377. // Found it and save to the current component
  378. OptionalComponent.TopLevelStringId = OcManager->TopLevelOcStringIds[u];
  379. u=(UINT)-1;
  380. break;
  381. }
  382. }
  383. // Check Found the Right String ID.
  384. if(u != (UINT)-1) {
  385. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  386. return(FALSE);
  387. }
  388. } else {
  389. // Pass Two - Discover Needs and Parentage
  390. // Two passs First to collect all the names second to
  391. // create the needs and parent links
  392. CurrentStringId = l;
  393. //
  394. // Deal with the needs.
  395. //
  396. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szNeeds,&SublineContext)) {
  397. n = 0;
  398. u = 0;
  399. while(p = pSetupGetField(&SublineContext,n+1)) {
  400. //
  401. // Ignore unless the subcomponent is in the string table.
  402. //
  403. l = pSetupStringTableLookUpStringEx(
  404. Params->OcManager->ComponentStringTable,
  405. (PTSTR)p,
  406. STRTAB_CASE_INSENSITIVE,
  407. &AuxOc,
  408. sizeof(OPTIONAL_COMPONENT)
  409. );
  410. if(l != -1) {
  411. //
  412. // Grow the needs array and put this item in it.
  413. //
  414. if(OptionalComponent.NeedsStringIds) {
  415. p = pSetupRealloc(
  416. OptionalComponent.NeedsStringIds,
  417. (OptionalComponent.NeedsCount+1) * sizeof(LONG)
  418. );
  419. } else {
  420. OptionalComponent.NeedsCount = 0;
  421. p = pSetupMalloc(sizeof(LONG));
  422. }
  423. if(p) {
  424. OptionalComponent.NeedsStringIds = (PVOID)p;
  425. OptionalComponent.NeedsStringIds[OptionalComponent.NeedsCount++] = l;
  426. } else {
  427. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  428. return(FALSE);
  429. }
  430. //
  431. // Insert this component in the needed component's neededby array.
  432. //
  433. if(AuxOc.NeededByStringIds) {
  434. p = pSetupRealloc(
  435. AuxOc.NeededByStringIds,
  436. (AuxOc.NeededByCount+1) * sizeof(LONG)
  437. );
  438. } else {
  439. AuxOc.NeededByCount = 0;
  440. p = pSetupMalloc(sizeof(LONG));
  441. }
  442. if(p) {
  443. AuxOc.NeededByStringIds = (PVOID)p;
  444. AuxOc.NeededByStringIds[AuxOc.NeededByCount++] = CurrentStringId;
  445. } else {
  446. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  447. return(FALSE);
  448. }
  449. pSetupStringTableSetExtraData(
  450. Params->OcManager->ComponentStringTable,
  451. l,
  452. &AuxOc,
  453. sizeof(OPTIONAL_COMPONENT)
  454. );
  455. }
  456. n++;
  457. }
  458. }
  459. //
  460. // Deal with the excludes.
  461. //
  462. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szExclude,&SublineContext)) {
  463. n = 0;
  464. u = 0;
  465. while(p = pSetupGetField(&SublineContext,n+1)) {
  466. //
  467. // Ignore unless the subcomponent is in the string table.
  468. //
  469. l = pSetupStringTableLookUpStringEx(
  470. Params->OcManager->ComponentStringTable,
  471. (PTSTR)p,
  472. STRTAB_CASE_INSENSITIVE,
  473. &AuxOc,
  474. sizeof(OPTIONAL_COMPONENT)
  475. );
  476. if(l != -1) {
  477. //
  478. // Grow the exclude array and put this item in it.
  479. //
  480. if(OptionalComponent.ExcludeStringIds) {
  481. p = pSetupRealloc(
  482. OptionalComponent.ExcludeStringIds,
  483. (OptionalComponent.ExcludeCount+1) * sizeof(LONG)
  484. );
  485. } else {
  486. OptionalComponent.ExcludeCount = 0;
  487. p = pSetupMalloc(sizeof(LONG));
  488. }
  489. if(p) {
  490. OptionalComponent.ExcludeStringIds = (PVOID)p;
  491. OptionalComponent.ExcludeStringIds[OptionalComponent.ExcludeCount++] = l;
  492. } else {
  493. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  494. return(FALSE);
  495. }
  496. //
  497. // Insert this component in the excluded component's excludedby array.
  498. //
  499. if(AuxOc.ExcludedByStringIds) {
  500. p = pSetupRealloc(
  501. AuxOc.ExcludedByStringIds,
  502. (AuxOc.ExcludedByCount+1) * sizeof(LONG)
  503. );
  504. } else {
  505. AuxOc.ExcludedByCount = 0;
  506. p = pSetupMalloc(sizeof(LONG));
  507. }
  508. if(p) {
  509. AuxOc.ExcludedByStringIds = (PVOID)p;
  510. AuxOc.ExcludedByStringIds[AuxOc.ExcludedByCount++] = CurrentStringId;
  511. } else {
  512. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  513. return(FALSE);
  514. }
  515. pSetupStringTableSetExtraData(
  516. Params->OcManager->ComponentStringTable,
  517. l,
  518. &AuxOc,
  519. sizeof(OPTIONAL_COMPONENT)
  520. );
  521. }
  522. n++;
  523. }
  524. }
  525. //
  526. // Figure out parentage. Ignore specified parent unless it exists
  527. // in the string table. We also note in the parent that it has children.
  528. //
  529. if(SetupFindFirstLine(OcInf->Handle,SubcompName,szParent,&SublineContext)
  530. && (p = (PVOID)pSetupGetField(&SublineContext,1))) {
  531. l = pSetupStringTableLookUpStringEx(
  532. Params->OcManager->ComponentStringTable,
  533. (PTSTR)p,
  534. STRTAB_CASE_INSENSITIVE,
  535. &AuxOc,
  536. sizeof(OPTIONAL_COMPONENT)
  537. );
  538. if(l != -1) {
  539. //
  540. // l is the string id of the parent, and AuxOc is filled with
  541. // the parent's optional component data.
  542. //
  543. OptionalComponent.ParentStringId = l;
  544. if(AuxOc.FirstChildStringId == -1) {
  545. //
  546. // This parent has no children yet.
  547. // Set the current component as its (first) child.
  548. // Note that in this case the current component does not yet
  549. // have any siblings.
  550. //
  551. AuxOc.FirstChildStringId = CurrentStringId;
  552. AuxOc.ChildrenCount = 1;
  553. pSetupStringTableSetExtraData(
  554. Params->OcManager->ComponentStringTable,
  555. l,
  556. &AuxOc,
  557. sizeof(OPTIONAL_COMPONENT)
  558. );
  559. } else {
  560. //
  561. // The parent already has children.
  562. // Increment the parent's count of children, then
  563. // walk the siblings list and add the new component to the end.
  564. //
  565. AuxOc.ChildrenCount++;
  566. pSetupStringTableSetExtraData(
  567. Params->OcManager->ComponentStringTable,
  568. l,
  569. &AuxOc,
  570. sizeof(OPTIONAL_COMPONENT)
  571. );
  572. l = AuxOc.FirstChildStringId;
  573. pSetupStringTableGetExtraData(
  574. Params->OcManager->ComponentStringTable,
  575. AuxOc.FirstChildStringId,
  576. &AuxOc,
  577. sizeof(OPTIONAL_COMPONENT)
  578. );
  579. while(AuxOc.NextSiblingStringId != -1) {
  580. l = AuxOc.NextSiblingStringId;
  581. pSetupStringTableGetExtraData(
  582. Params->OcManager->ComponentStringTable,
  583. l,
  584. &AuxOc,
  585. sizeof(OPTIONAL_COMPONENT)
  586. );
  587. }
  588. AuxOc.NextSiblingStringId = CurrentStringId;
  589. pSetupStringTableSetExtraData(
  590. Params->OcManager->ComponentStringTable,
  591. l,
  592. &AuxOc,
  593. sizeof(OPTIONAL_COMPONENT)
  594. );
  595. }
  596. }
  597. } else { // a node with out a parent a new Top Level node
  598. // Finally Add this String ID to the Component Strings list
  599. // UINT TopLevelParentOcCount;
  600. // PLONG TopLevelParentOcStringIds;
  601. if(OcManager->TopLevelParentOcStringIds != NULL) {
  602. p = pSetupRealloc(
  603. OcManager->TopLevelParentOcStringIds,
  604. (OcManager->TopLevelParentOcCount+1)
  605. * sizeof(OcManager->TopLevelParentOcStringIds)
  606. );
  607. } else {
  608. OcManager->TopLevelParentOcCount = 0;
  609. p = pSetupMalloc(sizeof(LONG));
  610. }
  611. if(p) {
  612. OcManager->TopLevelParentOcStringIds = (PVOID)p;
  613. OcManager->TopLevelParentOcStringIds[OcManager->TopLevelParentOcCount++] = CurrentStringId;
  614. } else {
  615. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  616. return(FALSE);
  617. }
  618. }
  619. }
  620. //
  621. // Now add the subcomponent to the string table.
  622. // We overwrite the extra data, which is not harmful since
  623. // we specifically fetched it earlier.
  624. //
  625. l = pSetupStringTableAddStringEx(
  626. Params->OcManager->ComponentStringTable,
  627. (PTSTR)SubcompName,
  628. STRTAB_NEW_EXTRADATA | STRTAB_CASE_INSENSITIVE,
  629. &OptionalComponent,
  630. sizeof(OPTIONAL_COMPONENT)
  631. );
  632. if(l == -1) {
  633. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  634. return(FALSE);
  635. }
  636. }
  637. } while(SetupFindNextLine(&LineContext,&LineContext));
  638. }
  639. return(TRUE);
  640. }
  641. BOOL
  642. pOcBuildSubcomponentLists(
  643. IN OUT POC_MANAGER OcManager,
  644. IN PVOID Log
  645. )
  646. /*++
  647. Routine Description:
  648. This routine examines the loaded per-component infs and builds the
  649. subcomponent hierarchies that are described therein via the Parent=
  650. lines in the various per-component sections.
  651. Arguments:
  652. OcManager - supplies a pointer to the context data structure
  653. for the OC Manager.
  654. Log - supplies a handle to use to log errors.
  655. Return Value:
  656. Boolean indicating outcome. If FALSE, an error will have been logged.
  657. --*/
  658. {
  659. OC_INF OcInf;
  660. BOOL b;
  661. BUILDSUBCOMPLIST_PARAMS s;
  662. s.OcManager = OcManager;
  663. //
  664. // We make 2 passes. The first adds all the subcomponent names to
  665. // the string table. The second computes parentage. If we don't do
  666. // it this way then we might have ordering problems.
  667. //
  668. s.Pass = 0;
  669. b = pSetupStringTableEnum(
  670. OcManager->InfListStringTable,
  671. &OcInf,
  672. sizeof(OC_INF),
  673. (PSTRTAB_ENUM_ROUTINE)pOcBuildSubcomponentListStringCB,
  674. (LPARAM)&s
  675. );
  676. if(b) {
  677. s.Pass = 1;
  678. b = pSetupStringTableEnum(
  679. OcManager->InfListStringTable,
  680. &OcInf,
  681. sizeof(OC_INF),
  682. (PSTRTAB_ENUM_ROUTINE)pOcBuildSubcomponentListStringCB,
  683. (LPARAM)&s
  684. );
  685. }
  686. return(b);
  687. }
  688. BOOL
  689. pOcInitPaths(
  690. IN OUT POC_MANAGER OcManager,
  691. IN PCTSTR MasterInfName
  692. )
  693. {
  694. TCHAR path[MAX_PATH];
  695. TCHAR *p;
  696. //
  697. // 1. look for master INF in specified directory.
  698. // 2. look in %systemroot%\system32\Setup directory.
  699. // 3. look in %systemroot%\inf directory.
  700. //
  701. if (!FileExists(MasterInfName, NULL)) {
  702. pOcFormSuitePath(NULL, NULL, path);
  703. p = _tcsrchr(MasterInfName, TEXT('\\'));
  704. if (!p)
  705. p = (TCHAR *)MasterInfName;
  706. pSetupConcatenatePaths(path, p, MAX_PATH, NULL);
  707. if (!FileExists(path, NULL)) {
  708. #ifdef UNICODE
  709. HMODULE hMod;
  710. FARPROC pGetSystemWindowsDirectory;
  711. hMod = LoadLibrary(L"kernel32.dll");
  712. if (hMod) {
  713. pGetSystemWindowsDirectory = GetProcAddress(hMod,"GetSystemWindowsDirectoryW");
  714. if (!pGetSystemWindowsDirectory) {
  715. pGetSystemWindowsDirectory = GetProcAddress(hMod,"GetWindowsDirectoryW");
  716. }
  717. if (pGetSystemWindowsDirectory) {
  718. pGetSystemWindowsDirectory( path, MAX_PATH );
  719. } else {
  720. GetWindowsDirectory(path,MAX_PATH);
  721. }
  722. FreeLibrary(hMod);
  723. }
  724. #else
  725. GetWindowsDirectory(path,MAX_PATH);
  726. #endif
  727. pSetupConcatenatePaths(path,TEXT("INF"),MAX_PATH,NULL);
  728. pSetupConcatenatePaths(path,p,MAX_PATH,NULL);
  729. if (!FileExists(path, NULL))
  730. return FALSE;
  731. }
  732. } else {
  733. _tcscpy(path, MasterInfName);
  734. }
  735. _tcscpy(OcManager->MasterOcInfPath, path);
  736. return TRUE;
  737. }
  738. BOOL
  739. pOcInstallSetupComponents(
  740. IN POPTIONAL_COMPONENT Oc,
  741. IN OUT POC_MANAGER OcManager,
  742. IN PCTSTR Component,
  743. IN PCTSTR DllName, OPTIONAL
  744. IN PCTSTR InfName, OPTIONAL
  745. IN HWND OwnerWindow,
  746. IN PVOID Log
  747. )
  748. /*++
  749. Routine Description:
  750. This routine makes sure that all files required for installation of
  751. a component listed in a master oc inf are properly installed in
  752. a well-known location.
  753. If the master OC inf is in the system inf directory, then we assume
  754. that all files are already in their proper locations and we do nothing.
  755. Otherwise we copy all installation files into system32\setup.
  756. Files copied include the per-component inf (if any), the installation dll,
  757. and all files listed on the ExtraSetupFiles= line in the [<component>]
  758. section in the master OC inf.
  759. Do not call this routine if the registry setting indicates that the
  760. master OC inf has been processed before.
  761. Arguments:
  762. OcManager - supplies a pointer to the context data structure
  763. for the OC Manager.
  764. MasterOcInfName - supplies the full Win32 path of the master OC inf file.
  765. Component - supplies the shortname of the component we care about.
  766. DllName - supplies the name of the component's installation dll, if any.
  767. InfName - supplies the name of the component's per-component inf,
  768. if any.
  769. OwnerWindow - supplies the handle of the window to own any UI which may be
  770. popped up by this routine.
  771. Log - supplies a handle to use to log errors.
  772. Return Value:
  773. Boolean indicating outcome. If FALSE, an error will have been logged.
  774. --*/
  775. {
  776. TCHAR Path[MAX_PATH];
  777. TCHAR TargetPath[MAX_PATH];
  778. TCHAR InfPath[MAX_PATH];
  779. PTCHAR p;
  780. UINT u;
  781. HSPFILEQ FileQueue;
  782. PVOID QueueContext;
  783. INFCONTEXT InfLine;
  784. BOOL b;
  785. TCHAR FileName[MAX_PATH];
  786. DWORD n;
  787. b = FALSE;
  788. //
  789. // All of the installation files are expected to be sourced
  790. // in the same directory as the master oc inf itself.
  791. //
  792. // We'll stick all the installation files for the component
  793. // in %windir%\system32\setup so we know where to get at them later.
  794. //
  795. // If the master inf is in the inf directory, then we instead
  796. // assume that the the component is tightly integrated into
  797. // the system and that the installation files are already in
  798. // the system32 directory.
  799. //
  800. if (!GetWindowsDirectory(Path,MAX_PATH) ||
  801. !pSetupConcatenatePaths(Path,TEXT("INF"),MAX_PATH,NULL)) {
  802. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  803. goto c0;
  804. }
  805. u = lstrlen(Path);
  806. pOcFormSuitePath(OcManager->SuiteName,NULL,TargetPath);
  807. lstrcpy(InfPath, OcManager->MasterOcInfPath);
  808. if (p = _tcsrchr(InfPath, TEXT('\\')))
  809. *p = 0;
  810. if (_tcsicmp(InfPath, Path) && _tcsicmp(InfPath, TargetPath)) {
  811. //
  812. // Inf is not in inf directory, so need to copy files.
  813. //
  814. FileQueue = SetupOpenFileQueue();
  815. if(FileQueue == INVALID_HANDLE_VALUE) {
  816. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  817. goto c0;
  818. }
  819. //
  820. // We will use the silent feature; no progress gauge but
  821. // we want errors to be displayed. Pass INVALID_HANDLE_VALUE
  822. // to get this behavior.
  823. //
  824. QueueContext = SetupInitDefaultQueueCallbackEx(OwnerWindow,INVALID_HANDLE_VALUE,0,0,0);
  825. if(!QueueContext) {
  826. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  827. goto c1;
  828. }
  829. //
  830. // Form source and target paths
  831. //
  832. lstrcpy(Path,OcManager->MasterOcInfPath);
  833. if(p = _tcsrchr(Path,TEXT('\\'))) {
  834. *p = 0;
  835. }
  836. //
  837. // Queue dll, and, if specified, inf
  838. //
  839. if (DllName && *DllName) {
  840. if ( (OcManager->SetupMode & SETUPMODE_PRIVATE_MASK) == SETUPMODE_REMOVEALL ) {
  841. b = SetupQueueDelete(
  842. FileQueue, // handle to the file queue
  843. TargetPath, // path to the file to delete
  844. DllName // optional, additional path info
  845. );
  846. } else {
  847. BOOL bCopyFile = TRUE;
  848. // check if the file is present, it may not have to be for
  849. // defered installs, where the suite will provide the Exe usally on demand
  850. // via Web download
  851. if (Oc && Oc->InterfaceFunctionName[0] == 0 ) {
  852. // No functin name means external setup
  853. lstrcpy(FileName,Path);
  854. pSetupConcatenatePaths(FileName, Oc->InstallationDllName, MAX_PATH, NULL);
  855. bCopyFile = (GetFileAttributes(FileName) == -1) ? FALSE: TRUE;
  856. b=TRUE;
  857. // bCopyFile=TRUE if we found the file
  858. }
  859. if( bCopyFile ) {
  860. b = SetupQueueCopy(
  861. FileQueue,
  862. Path,
  863. NULL,
  864. DllName,
  865. NULL,
  866. NULL,
  867. TargetPath,
  868. NULL,
  869. SP_COPY_SOURCEPATH_ABSOLUTE
  870. );
  871. }
  872. }
  873. if(!b) {
  874. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  875. goto c2;
  876. }
  877. }
  878. if(InfName && *InfName) {
  879. if ( (OcManager->SetupMode & SETUPMODE_PRIVATE_MASK) == SETUPMODE_REMOVEALL ) {
  880. b = SetupQueueDelete(
  881. FileQueue, // handle to the file queue
  882. TargetPath, // path to the file to delete
  883. InfName // optional, additional path info
  884. );
  885. } else {
  886. b = SetupQueueCopy(
  887. FileQueue,
  888. Path,
  889. NULL,
  890. InfName,
  891. NULL,
  892. NULL,
  893. TargetPath,
  894. NULL,
  895. SP_COPY_SOURCEPATH_ABSOLUTE
  896. );
  897. }
  898. if(!b) {
  899. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  900. goto c2;
  901. }
  902. }
  903. //
  904. // Queue each extra installation file.
  905. //
  906. if(SetupFindFirstLine(OcManager->MasterOcInf,Component,szExtraSetupFiles,&InfLine)) {
  907. n = 1;
  908. while(SetupGetStringField(&InfLine,n++,FileName,MAX_PATH,NULL)) {
  909. if ( (OcManager->SetupMode & SETUPMODE_PRIVATE_MASK) == SETUPMODE_REMOVEALL ) {
  910. b = SetupQueueDelete(
  911. FileQueue, // handle to the file queue
  912. TargetPath, // path to the file to delete
  913. FileName // optional, additional path info
  914. );
  915. } else {
  916. b = SetupQueueCopy(
  917. FileQueue,
  918. Path,
  919. NULL,
  920. FileName,
  921. NULL,
  922. NULL,
  923. TargetPath,
  924. NULL,
  925. SP_COPY_SOURCEPATH_ABSOLUTE
  926. );
  927. }
  928. if(!b) {
  929. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  930. goto c2;
  931. }
  932. }
  933. }
  934. //
  935. // Commit the queue.
  936. //
  937. b = SetupCommitFileQueue(OwnerWindow,FileQueue,SetupDefaultQueueCallback,QueueContext);
  938. if(!b) {
  939. _LogError(OcManager,OcErrLevError,MSG_OC_CANT_COPY_SETUP_FILES,GetLastError());
  940. goto c2;
  941. }
  942. //
  943. // Make a note that this OC inf is now "known."
  944. //
  945. u = pOcQueryOrSetNewInf(OcManager->MasterOcInfPath,OcManager->SuiteName,infSet);
  946. if(u != NO_ERROR) {
  947. _LogError(OcManager,OcErrLevWarning,MSG_OC_CANT_WRITE_REGISTRY,u);
  948. }
  949. c2:
  950. SetupTermDefaultQueueCallback(QueueContext);
  951. c1:
  952. SetupCloseFileQueue(FileQueue);
  953. c0:
  954. ;
  955. } else {
  956. b = TRUE;
  957. }
  958. return(b);
  959. }
  960. BOOL
  961. pOcLoadMasterOcInf(
  962. IN OUT POC_MANAGER OcManager,
  963. IN DWORD Flags,
  964. IN PVOID Log
  965. )
  966. /*++
  967. Routine Description:
  968. This routine loads the master OC inf and builds up the list of
  969. top-level optional components and some of the associated data
  970. inclucing the name of the installation inf, dll and the entry point.
  971. The per-components infs and dlls are not actually loaded
  972. by this routine.
  973. The wizard page ordering stuff is also initialized in the
  974. OC Manager data structure.
  975. Arguments:
  976. OcManager - supplies a pointer to the context data structure
  977. for the OC Manager.
  978. Flags - if OCINIT_FORCENEWINF, then behave as if the master OC inf is new.
  979. if OCINIT_KILLSUBCOMPS, then delete all applicable subcomponent
  980. entries from the registry before processing.
  981. Log - supplies a handle to use to log errors.
  982. Return Value:
  983. Boolean indicating outcome. If FALSE, an error will have been logged.
  984. --*/
  985. {
  986. BOOL b;
  987. INFCONTEXT InfContext;
  988. PCTSTR ComponentName;
  989. PCTSTR DllName;
  990. PCTSTR ModuleFlags;
  991. DWORD OtherFlags;
  992. PCTSTR EntryName;
  993. PCTSTR InfName;
  994. LPCTSTR chkflag;
  995. OPTIONAL_COMPONENT Oc;
  996. LONG Id;
  997. PVOID p;
  998. UINT i,j;
  999. UINT ActualCount;
  1000. WizardPagesType ReplacePages[4] = {WizPagesWelcome,WizPagesMode,WizPagesFinal,-1},
  1001. AddPages[5] = {WizPagesEarly,WizPagesPrenet,WizPagesPostnet,WizPagesLate,-1};
  1002. WizardPagesType *PageList;
  1003. PCTSTR SectionName;
  1004. BOOL NewInf;
  1005. TCHAR ComponentsSection[100];
  1006. TCHAR setupdir[MAX_PATH];
  1007. // First check and see if the setup cache directory exists.
  1008. // If not, then we should create it on this run.
  1009. pOcFormSuitePath(NULL, NULL, setupdir);
  1010. sapiAssert(*setupdir);
  1011. if (!FileExists(setupdir, NULL))
  1012. Flags |= OCINIT_FORCENEWINF;
  1013. //
  1014. // Always run pOcQueryOrSetNewInf in case it has side effects.
  1015. //
  1016. if (Flags & OCINIT_KILLSUBCOMPS)
  1017. pOcQueryOrSetNewInf(OcManager->MasterOcInfPath, OcManager->SuiteName, infReset);
  1018. NewInf = !pOcQueryOrSetNewInf(OcManager->MasterOcInfPath,OcManager->SuiteName,infQuery);
  1019. if(Flags & OCINIT_FORCENEWINF) {
  1020. NewInf = TRUE;
  1021. OcManager->InternalFlags |= OCMFLAG_NEWINF;
  1022. }
  1023. if (Flags & OCINIT_KILLSUBCOMPS)
  1024. OcManager->InternalFlags |= OCMFLAG_KILLSUBCOMPS;
  1025. if (Flags & OCINIT_RUNQUIET)
  1026. OcManager->InternalFlags |= OCMFLAG_RUNQUIET;
  1027. if (Flags & OCINIT_LANGUAGEAWARE)
  1028. OcManager->InternalFlags |= OCMFLAG_LANGUAGEAWARE;
  1029. OcManager->MasterOcInf = SetupOpenInfFile(OcManager->MasterOcInfPath,NULL,INF_STYLE_WIN4,&i);
  1030. if(OcManager->MasterOcInf == INVALID_HANDLE_VALUE) {
  1031. _LogError(OcManager,OcErrLevFatal,MSG_OC_CANT_OPEN_INF,OcManager->MasterOcInfPath,GetLastError(),i);
  1032. b = FALSE;
  1033. goto c0;
  1034. }
  1035. //
  1036. // Get the number of lines in the [Components] section and allocate
  1037. // arrays in the OC Manager context structure accordingly. This may
  1038. // overallocate the arrays (in case of duplicates, invalid lines, etc)
  1039. // but we won't worry about that here.
  1040. //
  1041. lstrcpy(ComponentsSection,szComponents);
  1042. OcManager->TopLevelOcCount = (UINT)(-1);
  1043. #if defined(_AMD64_)
  1044. lstrcat(ComponentsSection,TEXT(".amd64"));
  1045. #elif defined(_X86_)
  1046. lstrcat(ComponentsSection,TEXT(".w95"));
  1047. OcManager->TopLevelOcCount = SetupGetLineCount(OcManager->MasterOcInf,ComponentsSection);
  1048. if(OcManager->TopLevelOcCount == (UINT)(-1)) {
  1049. lstrcpy(ComponentsSection,szComponents);
  1050. lstrcat(ComponentsSection,TEXT(".x86"));
  1051. }
  1052. #elif defined(_IA64_)
  1053. lstrcat(ComponentsSection,TEXT(".ia64"));
  1054. #else
  1055. #error Unknown platform!
  1056. #endif
  1057. if(OcManager->TopLevelOcCount == (UINT)(-1)) {
  1058. OcManager->TopLevelOcCount = SetupGetLineCount(OcManager->MasterOcInf,ComponentsSection);
  1059. }
  1060. if(OcManager->TopLevelOcCount == (UINT)(-1)) {
  1061. lstrcpy(ComponentsSection,szComponents);
  1062. OcManager->TopLevelOcCount = SetupGetLineCount(OcManager->MasterOcInf,ComponentsSection);
  1063. }
  1064. if(OcManager->TopLevelOcCount == (UINT)(-1)) {
  1065. _LogError(OcManager,OcErrLevFatal,MSG_OC_INF_INVALID_NO_SECTION,OcManager->MasterOcInfPath,szComponents);
  1066. b = FALSE;
  1067. goto c1;
  1068. }
  1069. if (OcManager->TopLevelOcCount < 1) {
  1070. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  1071. goto c1;
  1072. }
  1073. if(p = pSetupRealloc(OcManager->TopLevelOcStringIds,OcManager->TopLevelOcCount*sizeof(LONG))) {
  1074. OcManager->TopLevelOcStringIds = p;
  1075. for(i=0; i<WizPagesTypeMax; i++) {
  1076. if(p = pSetupRealloc(OcManager->WizardPagesOrder[i],OcManager->TopLevelOcCount*sizeof(LONG))) {
  1077. OcManager->WizardPagesOrder[i] = p;
  1078. } else {
  1079. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  1080. b = FALSE;
  1081. goto c1;
  1082. }
  1083. }
  1084. } else {
  1085. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  1086. goto c1;
  1087. }
  1088. // get global info -
  1089. if ((OcManager->SetupData.OperationFlags & SETUPOP_STANDALONE) &&
  1090. SetupFindFirstLine(
  1091. OcManager->MasterOcInf,
  1092. szGlobal,
  1093. szWindowTitleAlone,
  1094. &InfContext)) {
  1095. // the main window title
  1096. SetupGetStringField(
  1097. &InfContext,
  1098. 1, // index of the field to get
  1099. OcManager->WindowTitle, // optional, receives the field
  1100. sizeof(OcManager->WindowTitle), // size of the provided buffer
  1101. NULL);
  1102. if( !lstrcmpi( OcManager->WindowTitle, szWindowTitleInternal)) {
  1103. //This will happen when we load sysoc.inf For MUI.
  1104. LoadString(MyModuleHandle,IDS_OCM_WINDOWTITLE,OcManager->WindowTitle,sizeof(OcManager->WindowTitle)/sizeof(TCHAR));
  1105. }
  1106. } else if(SetupFindFirstLine(
  1107. OcManager->MasterOcInf,
  1108. szGlobal,
  1109. szWindowTitle,
  1110. &InfContext)) {
  1111. // the main window title
  1112. SetupGetStringField(
  1113. &InfContext,
  1114. 1, // index of the field to get
  1115. OcManager->WindowTitle, // optional, receives the field
  1116. sizeof(OcManager->WindowTitle), // size of the provided buffer
  1117. NULL);
  1118. } else {
  1119. *OcManager->WindowTitle = 0;
  1120. }
  1121. //
  1122. // Go through the [Components] section. Each line in there is a top-level
  1123. // component spec, giving dll name, entry point name, and optionally
  1124. // the name of the per-component inf.
  1125. //
  1126. if(!SetupFindFirstLine(OcManager->MasterOcInf,ComponentsSection,NULL,&InfContext)) {
  1127. _LogError(OcManager,OcErrLevFatal,MSG_OC_INF_INVALID_NO_SECTION,OcManager->MasterOcInfPath,ComponentsSection);
  1128. b = FALSE;
  1129. goto c1;
  1130. }
  1131. ActualCount = 0;
  1132. do {
  1133. //
  1134. // Get pointers to each field in each line. Ignore invalid lines.
  1135. //
  1136. if(ComponentName = pSetupGetField(&InfContext,0)) {
  1137. DllName = pSetupGetField(&InfContext,1);
  1138. if(DllName && !*DllName) {
  1139. DllName = NULL;
  1140. }
  1141. EntryName = pSetupGetField(&InfContext,2);
  1142. if(EntryName && !*EntryName) {
  1143. EntryName = NULL;
  1144. }
  1145. //
  1146. // An empty string for the inf name is the same as
  1147. // not specifying the inf at all.
  1148. //
  1149. if((InfName = pSetupGetField(&InfContext,3)) && *InfName) {
  1150. Id = pSetupStringTableAddString(
  1151. OcManager->InfListStringTable,
  1152. (PTSTR)InfName,
  1153. STRTAB_CASE_INSENSITIVE
  1154. );
  1155. if(Id == -1) {
  1156. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  1157. goto c1;
  1158. }
  1159. } else {
  1160. Id = -1;
  1161. }
  1162. // Get the Flags Field
  1163. ModuleFlags = pSetupGetField(&InfContext,4);
  1164. if(ModuleFlags && !*ModuleFlags) {
  1165. ModuleFlags = NULL;
  1166. }
  1167. OtherFlags = 0;
  1168. SetupGetIntField(&InfContext,5,&OtherFlags);
  1169. ZeroMemory(&Oc,sizeof(OPTIONAL_COMPONENT));
  1170. //
  1171. // These guys are top-level. Also remember the string id
  1172. // of the inf name in the inf string table.
  1173. //
  1174. Oc.FirstChildStringId = -1;
  1175. Oc.NextSiblingStringId = -1;
  1176. Oc.ParentStringId = -1;
  1177. Oc.InfStringId = Id;
  1178. // Show flags allows up to have a component that is hidden
  1179. // Only on one flags now so keep processing simple
  1180. Oc.Exists = FALSE;
  1181. Oc.InternalFlags |= OCFLAG_TOPLEVELITEM;
  1182. if (ModuleFlags) {
  1183. if (OcManager->SetupData.OperationFlags & SETUPOP_STANDALONE)
  1184. chkflag = szOSSetupOnly;
  1185. else
  1186. chkflag = szStandaloneOnly;
  1187. if (!_tcsicmp(ModuleFlags, szHide) || !_tcsicmp(ModuleFlags, chkflag))
  1188. Oc.InternalFlags |= OCFLAG_HIDE;
  1189. }
  1190. if (OtherFlags & OCFLAG_NOWIZPAGES) {
  1191. Oc.InternalFlags |= OCFLAG_NOWIZARDPAGES;
  1192. }
  1193. if (OtherFlags & OCFLAG_NOQUERYSKIP) {
  1194. Oc.InternalFlags |= OCFLAG_NOQUERYSKIPPAGES;
  1195. }
  1196. if (OtherFlags & OCFLAG_NOEXTRAFLAGS) {
  1197. Oc.InternalFlags |= OCFLAG_NOEXTRAROUTINES;
  1198. }
  1199. if(DllName) {
  1200. lstrcpyn(Oc.InstallationDllName,DllName,MAX_PATH);
  1201. } else {
  1202. Oc.InstallationDllName[0] = 0;
  1203. }
  1204. //
  1205. // Interface Function Name is always ANSI -- there's no
  1206. // Unicode version of GetProcAddress.
  1207. //
  1208. if(EntryName) {
  1209. #ifdef UNICODE
  1210. WideCharToMultiByte(CP_ACP,0,EntryName,-1,Oc.InterfaceFunctionName,MAX_PATH,NULL,NULL);
  1211. #else
  1212. lstrcpyn(Oc.InterfaceFunctionName,EntryName,MAX_PATH);
  1213. #endif
  1214. } else {
  1215. Oc.InterfaceFunctionName[0] = 0;
  1216. }
  1217. Id = pSetupStringTableAddStringEx(
  1218. OcManager->ComponentStringTable,
  1219. (PTSTR)ComponentName,
  1220. STRTAB_CASE_INSENSITIVE | STRTAB_NEW_EXTRADATA,
  1221. &Oc,
  1222. sizeof(OPTIONAL_COMPONENT)
  1223. );
  1224. if(Id == -1) {
  1225. //
  1226. // OOM adding string
  1227. //
  1228. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  1229. goto c1;
  1230. }
  1231. OcManager->TopLevelOcStringIds[ActualCount++] = Id;
  1232. }
  1233. } while(SetupFindNextLine(&InfContext,&InfContext));
  1234. //
  1235. // Shrink down the various arrays.
  1236. //
  1237. OcManager->TopLevelOcStringIds = pSetupRealloc(OcManager->TopLevelOcStringIds,ActualCount*sizeof(LONG));
  1238. for(i=0; i<WizPagesTypeMax; i++) {
  1239. OcManager->WizardPagesOrder[i] = pSetupRealloc(OcManager->WizardPagesOrder[i],ActualCount*sizeof(LONG));
  1240. }
  1241. OcManager->TopLevelOcCount = ActualCount;
  1242. //
  1243. // Now for each wizard page type figure out the ordering.
  1244. //
  1245. for(i=0; i<2; i++) {
  1246. SectionName = i ? TEXT("PageAdd") : TEXT("PageReplace");
  1247. for(PageList = i ? AddPages : ReplacePages; *PageList != -1; PageList++) {
  1248. b = SetupFindFirstLine(
  1249. OcManager->MasterOcInf,
  1250. SectionName,
  1251. WizardPagesTypeNames[*PageList],
  1252. &InfContext
  1253. );
  1254. //
  1255. // Check for the "default" string, which is the same as if the line
  1256. // had not been specified at all.
  1257. //
  1258. if(b
  1259. && (ComponentName = pSetupGetField(&InfContext,1))
  1260. && !lstrcmpi(ComponentName,TEXT("Default"))) {
  1261. b = FALSE;
  1262. }
  1263. if(b) {
  1264. //
  1265. // Make sure the array is padded with -1's,
  1266. //
  1267. FillMemory(
  1268. OcManager->WizardPagesOrder[*PageList],
  1269. OcManager->TopLevelOcCount * sizeof(LONG),
  1270. (BYTE)(-1)
  1271. );
  1272. //
  1273. // Now process each element on the line, but don't allow
  1274. // overflowing the array.
  1275. //
  1276. j = 1;
  1277. ActualCount = 0;
  1278. while((ActualCount < OcManager->TopLevelOcCount)
  1279. && (ComponentName = pSetupGetField(&InfContext,j)) && *ComponentName) {
  1280. Id = pSetupStringTableLookUpString(
  1281. OcManager->ComponentStringTable,
  1282. (PTSTR)ComponentName,
  1283. STRTAB_CASE_INSENSITIVE
  1284. );
  1285. if(Id == -1) {
  1286. //
  1287. // Invalid component. Log error and keep going.
  1288. //
  1289. _LogError(OcManager,
  1290. OcErrLevWarning,
  1291. MSG_OC_INVALID_COMP_IN_SECT,
  1292. OcManager->MasterOcInfPath,
  1293. SectionName,
  1294. ComponentName
  1295. );
  1296. } else {
  1297. //
  1298. // Remember the string id for this component.
  1299. //
  1300. OcManager->WizardPagesOrder[*PageList][ActualCount++] = Id;
  1301. }
  1302. j++;
  1303. }
  1304. } else {
  1305. //
  1306. // Default ordering, which is the order in which the components
  1307. // were listed in the [Components] section
  1308. //
  1309. CopyMemory(
  1310. OcManager->WizardPagesOrder[*PageList],
  1311. OcManager->TopLevelOcStringIds,
  1312. OcManager->TopLevelOcCount * sizeof(LONG)
  1313. );
  1314. }
  1315. }
  1316. }
  1317. // get the caption for various pages
  1318. if(SetupFindFirstLine(OcManager->MasterOcInf,szPageTitle,szSetupTitle,&InfContext)) {
  1319. // Found it
  1320. SetupGetStringField(
  1321. &InfContext,
  1322. 1, // index of the field to get
  1323. OcManager->SetupPageTitle, // optional, receives the field
  1324. sizeof(OcManager->SetupPageTitle), // size of the provided buffer
  1325. NULL);
  1326. }
  1327. return(TRUE);
  1328. c1:
  1329. sapiAssert(OcManager->MasterOcInf != INVALID_HANDLE_VALUE);
  1330. SetupCloseInfFile(OcManager->MasterOcInf);
  1331. c0:
  1332. return(b);
  1333. }
  1334. BOOL
  1335. pOcSetOcManagerDirIds(
  1336. IN HINF InfHandle,
  1337. IN LPCTSTR MasterOcInfName,
  1338. IN LPCTSTR ComponentName
  1339. )
  1340. /*++
  1341. Routine Description:
  1342. This routine sets up the pre-defined OC Manager directory ids for
  1343. per-component infs.
  1344. DIRID_OCM_MASTERINF
  1345. DIRID_OCM_MASTERINF_PLAT
  1346. DIRID_OCM_MASTERINF_COMP
  1347. DIRID_OCM_MASTERINF_COMP_PLAT
  1348. Arguments:
  1349. InfHandle - supplies handle to open inf file
  1350. MasterOcInfName - win32 path to master oc inf
  1351. ComponentName - simple shortname for the component
  1352. Return Value:
  1353. Boolean value indicating outcome. If FALSE, caller can assume OOM.
  1354. --*/
  1355. {
  1356. TCHAR Path[MAX_PATH];
  1357. TCHAR *p;
  1358. #if defined(_AMD64_)
  1359. LPCTSTR Platform = TEXT("AMD64");
  1360. #elif defined(_X86_)
  1361. LPCTSTR Platform = (IsNEC_98) ? TEXT("NEC98") : TEXT("I386");
  1362. #elif defined(_IA64_)
  1363. LPCTSTR Platform = TEXT("IA64");
  1364. #else
  1365. #error "No Target Architecture"
  1366. #endif
  1367. lstrcpy(Path,MasterOcInfName);
  1368. if(p = _tcsrchr(Path,TEXT('\\'))) {
  1369. *p = 0;
  1370. } else {
  1371. //
  1372. // Something is very broken
  1373. //
  1374. return(FALSE);
  1375. }
  1376. if(!SetupSetDirectoryId(InfHandle,DIRID_OCM_MASTERINF,Path)) {
  1377. return(FALSE);
  1378. }
  1379. if(!SetupSetDirectoryIdEx(InfHandle,DIRID_OCM_PLATFORM,Platform,SETDIRID_NOT_FULL_PATH,0,0)) {
  1380. return(FALSE);
  1381. }
  1382. if(!SetupSetDirectoryIdEx(
  1383. InfHandle,DIRID_OCM_PLATFORM_ALTERNATE,
  1384. #ifdef _X86_
  1385. TEXT("X86"),
  1386. #else
  1387. Platform,
  1388. #endif
  1389. SETDIRID_NOT_FULL_PATH,0,0)) {
  1390. return(FALSE);
  1391. }
  1392. if(!SetupSetDirectoryIdEx(InfHandle,DIRID_OCM_COMPONENT,ComponentName,SETDIRID_NOT_FULL_PATH,0,0)) {
  1393. return(FALSE);
  1394. }
  1395. return(TRUE);
  1396. }
  1397. BOOL
  1398. pOcLoadInfsAndDlls(
  1399. IN OUT POC_MANAGER OcManager,
  1400. OUT PBOOL Canceled,
  1401. IN PVOID Log
  1402. )
  1403. /*++
  1404. Routine Description:
  1405. Loads per-component INFs and installation DLLs.
  1406. This includes invoking the preinitialization and initialization
  1407. entry points in the DLLs.
  1408. Arguments:
  1409. OcManager - supplies OC manager context structure
  1410. Cancelled - receives a flag that is valid when this routine fails,
  1411. indicating whether the failure was caused by a component
  1412. canceling.
  1413. Log - supplies a handle to use to log errors.
  1414. Return Value:
  1415. Boolean value indicating outcome
  1416. --*/
  1417. {
  1418. UINT i;
  1419. OPTIONAL_COMPONENT Oc;
  1420. OC_INF OcInf;
  1421. UINT ErrorLine;
  1422. UINT Flags;
  1423. UINT ErrorCode;
  1424. PCTSTR InfName;
  1425. PCTSTR ComponentName;
  1426. TCHAR Library[MAX_PATH];
  1427. BOOL b;
  1428. INFCONTEXT Context;
  1429. *Canceled = FALSE;
  1430. //
  1431. // The top-level component strctures have everything we need.
  1432. // Spin through them.
  1433. //
  1434. for(i=0; i<OcManager->TopLevelOcCount; i++) {
  1435. //
  1436. // Get the OC data from the string table.
  1437. //
  1438. pSetupStringTableGetExtraData(
  1439. OcManager->ComponentStringTable,
  1440. OcManager->TopLevelOcStringIds[i],
  1441. &Oc,
  1442. sizeof(OPTIONAL_COMPONENT)
  1443. );
  1444. if(Oc.InfStringId == -1) {
  1445. //
  1446. // The master OC inf specified that this component has no per-component inf.
  1447. //
  1448. OcInf.Handle = INVALID_HANDLE_VALUE;
  1449. } else {
  1450. //
  1451. // Load the per-component INF if it has not already been loaded.
  1452. //
  1453. pSetupStringTableGetExtraData(OcManager->InfListStringTable,Oc.InfStringId,&OcInf,sizeof(OC_INF));
  1454. if(!OcInf.Handle) {
  1455. InfName = pSetupStringTableStringFromId(OcManager->InfListStringTable,Oc.InfStringId);
  1456. if (!InfName) {
  1457. _LogError(OcManager,OcErrLevError,MSG_OC_OOM);
  1458. return(FALSE);
  1459. }
  1460. if(OcManager->InternalFlags & OCMFLAG_NEWINF) {
  1461. // First try the directory the master inf is in
  1462. lstrcpy(Library, OcManager->SourceDir);
  1463. pSetupConcatenatePaths(Library, InfName, MAX_PATH, NULL);
  1464. } else {
  1465. pOcFormSuitePath(OcManager->SuiteName,InfName,Library);
  1466. }
  1467. if(FileExists(Library, NULL)) {
  1468. OcInf.Handle = SetupOpenInfFile(Library,NULL,INF_STYLE_WIN4,&ErrorLine);
  1469. } else {
  1470. //
  1471. // Use standard inf search rules if the inf can't be found
  1472. // in the special suite directory.
  1473. //
  1474. OcInf.Handle = SetupOpenInfFile(InfName,NULL,INF_STYLE_WIN4,&ErrorLine);
  1475. }
  1476. if(OcInf.Handle == INVALID_HANDLE_VALUE) {
  1477. //
  1478. // Log error.
  1479. //
  1480. _LogError(OcManager,OcErrLevError,MSG_OC_CANT_OPEN_INF,InfName,GetLastError(),ErrorLine);
  1481. return(FALSE);
  1482. } else {
  1483. // open the layout file
  1484. SetupOpenAppendInfFile(NULL, OcInf.Handle, NULL);
  1485. //
  1486. // Remember inf handle and set OC Manager DIRIDs.
  1487. //
  1488. pSetupStringTableSetExtraData(
  1489. OcManager->InfListStringTable,
  1490. Oc.InfStringId,
  1491. &OcInf,
  1492. sizeof(OC_INF)
  1493. );
  1494. b = pOcSetOcManagerDirIds(
  1495. OcInf.Handle,
  1496. OcManager->MasterOcInfPath,
  1497. pSetupStringTableStringFromId(OcManager->ComponentStringTable,OcManager->TopLevelOcStringIds[i])
  1498. );
  1499. if(!b) {
  1500. _LogError(OcManager,OcErrLevError,MSG_OC_OOM);
  1501. return(FALSE);
  1502. }
  1503. OcManager->SubComponentsPresent = TRUE;
  1504. }
  1505. }
  1506. }
  1507. //
  1508. // Load the DLL and get the entry point address.
  1509. // We make no attempt to track duplicates like we do for infs --
  1510. // the underlying OS does that for us.
  1511. //
  1512. // The dll could be either in the special suite directory for this
  1513. // master oc inf, or in a standard place. LoadLibraryEx does
  1514. // exactly what we want.
  1515. //
  1516. if(Oc.InstallationDllName[0] && Oc.InterfaceFunctionName[0]) {
  1517. if (OcManager->InternalFlags & OCMFLAG_NEWINF) {
  1518. // First try the directory the master inf is in
  1519. lstrcpy(Library, OcManager->SourceDir);
  1520. pSetupConcatenatePaths(Library, Oc.InstallationDllName, MAX_PATH, NULL);
  1521. Oc.InstallationDll = LoadLibraryEx(Library,NULL,LOAD_WITH_ALTERED_SEARCH_PATH);
  1522. }
  1523. // Try the Setup Directory
  1524. if (! Oc.InstallationDll ) {
  1525. pOcFormSuitePath(OcManager->SuiteName,Oc.InstallationDllName,Library);
  1526. Oc.InstallationDll = LoadLibraryEx(Library,NULL,LOAD_WITH_ALTERED_SEARCH_PATH);
  1527. }
  1528. // lastly try anywhere in the path
  1529. if ( ! Oc.InstallationDll ) {
  1530. Oc.InstallationDll = LoadLibraryEx(Oc.InstallationDllName,NULL,LOAD_WITH_ALTERED_SEARCH_PATH);
  1531. }
  1532. //
  1533. // Failure is taken care of below....
  1534. //
  1535. if (Oc.InstallationDll) {
  1536. Oc.InstallationRoutine = (POCSETUPPROC)GetProcAddress(
  1537. Oc.InstallationDll,
  1538. Oc.InterfaceFunctionName);
  1539. } else {
  1540. Oc.InstallationRoutine = NULL;
  1541. }
  1542. } else {
  1543. Oc.InstallationDll = MyModuleHandle;
  1544. Oc.InstallationRoutine = StandAloneSetupAppInterfaceRoutine;
  1545. }
  1546. if(Oc.InstallationDll && Oc.InstallationRoutine) {
  1547. //
  1548. // Success. Call the init-related entry points. Note that we can't call
  1549. // any entry point except the preinit one until after we've stored
  1550. // the ansi/unicode flag into the OPTIONAL_COMPONENT structure.
  1551. //
  1552. // Also note that before we do this we have to store the Oc structure,
  1553. // otherwise the interface routine is NULL and we fault.
  1554. //
  1555. pSetupStringTableSetExtraData(
  1556. OcManager->ComponentStringTable,
  1557. OcManager->TopLevelOcStringIds[i],
  1558. &Oc,
  1559. sizeof(OPTIONAL_COMPONENT)
  1560. );
  1561. Oc.Flags = OcInterfacePreinitialize(OcManager,OcManager->TopLevelOcStringIds[i]);
  1562. if(!Oc.Flags) {
  1563. //
  1564. // If this fails, then assume the DLL is written for a different
  1565. // platform or version, for example a Unicode/ANSI problem.
  1566. //
  1567. _LogError(OcManager,
  1568. OcErrLevError,
  1569. MSG_OC_DLL_PREINIT_FAILED,
  1570. pSetupStringTableStringFromId(
  1571. OcManager->ComponentStringTable,
  1572. OcManager->TopLevelOcStringIds[i]
  1573. ),
  1574. Oc.InstallationDllName
  1575. );
  1576. return(FALSE);
  1577. }
  1578. } else {
  1579. //
  1580. // Failure, log error.
  1581. //
  1582. _LogError(OcManager,
  1583. OcErrLevError,
  1584. MSG_OC_DLL_LOAD_FAIL,
  1585. Oc.InstallationDllName,
  1586. Oc.InterfaceFunctionName,
  1587. GetLastError()
  1588. );
  1589. return(FALSE);
  1590. }
  1591. //
  1592. // Set the OC data back into the string table.
  1593. // After this we can call other interface entry points since
  1594. // the ansi/unicode flag will now be stored in the OPTIONAL_COMPONENT
  1595. // structure for the component.
  1596. //
  1597. pSetupStringTableSetExtraData(
  1598. OcManager->ComponentStringTable,
  1599. OcManager->TopLevelOcStringIds[i],
  1600. &Oc,
  1601. sizeof(OPTIONAL_COMPONENT)
  1602. );
  1603. ErrorCode = OcInterfaceInitComponent(
  1604. OcManager,
  1605. OcManager->TopLevelOcStringIds[i]
  1606. );
  1607. if(ErrorCode == NO_ERROR) {
  1608. // Send down extra helper routines.
  1609. if ((Oc.InternalFlags & OCFLAG_NOEXTRAROUTINES)==0) {
  1610. ErrorCode = OcInterfaceExtraRoutines(
  1611. OcManager,
  1612. OcManager->TopLevelOcStringIds[i]
  1613. );
  1614. }
  1615. }
  1616. if(ErrorCode == NO_ERROR) {
  1617. if (OcManager->InternalFlags & OCMFLAG_LANGUAGEAWARE) {
  1618. //
  1619. // Send down a set-language request.
  1620. // Ignore the result.
  1621. //
  1622. OcInterfaceSetLanguage(
  1623. OcManager,
  1624. OcManager->TopLevelOcStringIds[i],
  1625. LANGIDFROMLCID(GetThreadLocale())
  1626. );
  1627. }
  1628. } else {
  1629. if(ErrorCode == ERROR_CANCELLED) {
  1630. // cancel will stop oc manager only if
  1631. // we aren't running in gui-mode setup
  1632. *Canceled = TRUE;
  1633. if (OcManager->SetupData.OperationFlags & SETUPOP_STANDALONE)
  1634. return FALSE;
  1635. } else {
  1636. _LogError(OcManager,
  1637. OcErrLevError,
  1638. MSG_OC_DLL_INIT_FAILED,
  1639. pSetupStringTableStringFromId(
  1640. OcManager->ComponentStringTable,
  1641. OcManager->TopLevelOcStringIds[i]
  1642. )
  1643. );
  1644. }
  1645. pOcRemoveComponent(OcManager, OcManager->TopLevelOcStringIds[i], pidLoadComponent);
  1646. }
  1647. }
  1648. //
  1649. // Now go gather additional information about subcomponents
  1650. // (descriptions, parentage, needs=, etc).
  1651. //
  1652. return(pOcBuildSubcomponentLists(OcManager,Log));
  1653. }
  1654. BOOL
  1655. pOcUnloadInfsAndDlls(
  1656. IN OUT POC_MANAGER OcManager,
  1657. IN PVOID Log
  1658. )
  1659. /*++
  1660. Routine Description:
  1661. Unloads per-component INFs and installation DLLs that were
  1662. previously loaded by pOcLoadInfsAndDlls().
  1663. This routine does NOT call the interface entry points to
  1664. uninitialize the installation DLLs.
  1665. Arguments:
  1666. OcManager - supplies OC manager context structure
  1667. Log - supplies a handle to use to log errors.
  1668. Return Value:
  1669. Boolean value indicating outcome.
  1670. If FALSE, an error will have been logged to indicate what failed.
  1671. --*/
  1672. {
  1673. OPTIONAL_COMPONENT Oc;
  1674. OC_INF OcInf;
  1675. UINT i;
  1676. //
  1677. // Unload Dlls.
  1678. //
  1679. for(i=0; i<OcManager->TopLevelOcCount; i++) {
  1680. pSetupStringTableGetExtraData(
  1681. OcManager->ComponentStringTable,
  1682. OcManager->TopLevelOcStringIds[i],
  1683. &Oc,
  1684. sizeof(OPTIONAL_COMPONENT)
  1685. );
  1686. if(Oc.InstallationDll && (Oc.InstallationDll != MyModuleHandle)) {
  1687. FreeLibrary(Oc.InstallationDll);
  1688. Oc.InstallationDll = NULL;
  1689. }
  1690. Oc.InstallationRoutine = NULL;
  1691. Oc.InstallationDllName[0] = 0;
  1692. Oc.InterfaceFunctionName[0]= 0;
  1693. if(Oc.InfStringId != -1) {
  1694. pSetupStringTableGetExtraData(
  1695. OcManager->InfListStringTable,
  1696. Oc.InfStringId,
  1697. &OcInf,
  1698. sizeof(OC_INF)
  1699. );
  1700. if(OcInf.Handle && (OcInf.Handle != INVALID_HANDLE_VALUE)) {
  1701. SetupCloseInfFile(OcInf.Handle);
  1702. OcInf.Handle = INVALID_HANDLE_VALUE;
  1703. pSetupStringTableSetExtraData(
  1704. OcManager->InfListStringTable,
  1705. Oc.InfStringId,
  1706. &OcInf,
  1707. sizeof(OC_INF)
  1708. );
  1709. }
  1710. Oc.InfStringId = -1;
  1711. }
  1712. pSetupStringTableSetExtraData(
  1713. OcManager->ComponentStringTable,
  1714. OcManager->TopLevelOcStringIds[i],
  1715. &Oc,
  1716. sizeof(OPTIONAL_COMPONENT)
  1717. );
  1718. }
  1719. return(TRUE);
  1720. }
  1721. BOOL
  1722. pOcManagerInitPrivateDataStore(
  1723. IN OUT POC_MANAGER OcManager,
  1724. IN PVOID Log
  1725. )
  1726. /*++
  1727. Routine Description:
  1728. Initializes the private data store for components.
  1729. The registry is used as the backing store for private data.
  1730. We make use of volatile keys to help ensure that this data is temporary
  1731. in nature.
  1732. Arguments:
  1733. OcManager - supplies OC Manager context stucture.
  1734. Log - supplies a handle to use to log errors.
  1735. Return Value:
  1736. Boolean value indicating outcome. If FALSE an error will have
  1737. been logged.
  1738. --*/
  1739. {
  1740. LONG l;
  1741. DWORD Disposition;
  1742. //
  1743. // Start out by forming a unique name for this instantiation
  1744. // of the OC Manager.
  1745. //
  1746. wsprintf(OcManager->PrivateDataSubkey,TEXT("%x:%x"),GetCurrentProcessId(),OcManager);
  1747. //
  1748. // Open/create the private data root. Save the handle.
  1749. //
  1750. l = RegCreateKeyEx(
  1751. HKEY_LOCAL_MACHINE,
  1752. szPrivateDataRoot,
  1753. 0,
  1754. NULL,
  1755. REG_OPTION_VOLATILE,
  1756. KEY_CREATE_SUB_KEY | KEY_ENUMERATE_SUB_KEYS | KEY_QUERY_VALUE | KEY_SET_VALUE,
  1757. NULL,
  1758. &OcManager->hKeyPrivateDataRoot,
  1759. &Disposition
  1760. );
  1761. if(l != NO_ERROR) {
  1762. OcManager->hKeyPrivateDataRoot = NULL;
  1763. OcManager->hKeyPrivateData = NULL;
  1764. _LogError(OcManager,OcErrLevWarning,MSG_OC_CREATE_KEY_FAILED,szPrivateDataRoot,l);
  1765. return(FALSE);
  1766. }
  1767. //
  1768. // Get rid of the private data tree if it already exists for some reason.
  1769. //
  1770. if(Disposition == REG_OPENED_EXISTING_KEY) {
  1771. pSetupRegistryDelnode(OcManager->hKeyPrivateDataRoot,OcManager->PrivateDataSubkey);
  1772. }
  1773. //
  1774. // Create the private data tree. Save the handle.
  1775. //
  1776. l = RegCreateKeyEx(
  1777. OcManager->hKeyPrivateDataRoot,
  1778. OcManager->PrivateDataSubkey,
  1779. 0,
  1780. NULL,
  1781. REG_OPTION_VOLATILE,
  1782. KEY_CREATE_SUB_KEY | KEY_QUERY_VALUE | KEY_SET_VALUE,
  1783. NULL,
  1784. &OcManager->hKeyPrivateData,
  1785. &Disposition
  1786. );
  1787. if(l != NO_ERROR) {
  1788. RegCloseKey(OcManager->hKeyPrivateDataRoot);
  1789. OcManager->hKeyPrivateDataRoot = NULL;
  1790. OcManager->hKeyPrivateData = NULL;
  1791. _LogError(OcManager,OcErrLevWarning,MSG_OC_CREATE_KEY_FAILED,OcManager->PrivateDataSubkey,l);
  1792. return(FALSE);
  1793. }
  1794. return(TRUE);
  1795. }
  1796. VOID
  1797. pOcManagerTearDownPrivateDataStore(
  1798. IN OUT POC_MANAGER OcManager
  1799. )
  1800. {
  1801. RegCloseKey(OcManager->hKeyPrivateData);
  1802. pSetupRegistryDelnode(OcManager->hKeyPrivateDataRoot,OcManager->PrivateDataSubkey);
  1803. RegCloseKey(OcManager->hKeyPrivateDataRoot);
  1804. OcManager->hKeyPrivateDataRoot = NULL;
  1805. OcManager->hKeyPrivateData = NULL;
  1806. }
  1807. UINT
  1808. pOcQueryOrSetNewInf(
  1809. IN PCTSTR MasterOcInfName,
  1810. OUT PTSTR SuiteName,
  1811. IN DWORD operation
  1812. )
  1813. /*++
  1814. Routine Description:
  1815. Determine whether a master OC inf has been encountered before,
  1816. or remember that a master inf has been encountered.
  1817. This information is stored in the registry.
  1818. Arguments:
  1819. MasterOcInfName - supplies the full Win32 path of the master OC inf.
  1820. SuiteName - receives the filename part of the .inf, without any
  1821. extension. This is suitable for use as a tag representing the
  1822. suite that the master INF is for. This buffer should be
  1823. MAX_PATH TCHAR elements.
  1824. QueryOnly - if TRUE, then the routine is to query whether the
  1825. master inf has been previously processed. If FALSE, then
  1826. the routine is to remember that the inf has been processed.
  1827. Return Value:
  1828. If QueryOnly is TRUE:
  1829. TRUE if the INF has been encountered before, FALSE if not.
  1830. If QueryOnly is FALSE:
  1831. Win32 error code indicating outcome.
  1832. --*/
  1833. {
  1834. PTCHAR p;
  1835. HKEY hKey;
  1836. LONG l;
  1837. DWORD Type;
  1838. DWORD Size;
  1839. DWORD Data;
  1840. //
  1841. // Form the suite name. The MasterOcInfName is expected to be
  1842. // a full path so this is pretty easy. We'll try to be at least
  1843. // a little more robust.
  1844. //
  1845. if(p = _tcsrchr(MasterOcInfName,TEXT('\\'))) {
  1846. p++;
  1847. } else {
  1848. p = (PTCHAR)MasterOcInfName;
  1849. }
  1850. lstrcpyn(SuiteName,p,MAX_PATH);
  1851. if(p = _tcsrchr(SuiteName,TEXT('.'))) {
  1852. *p = 0;
  1853. }
  1854. //
  1855. // Look in the registry to see if there is a value entry
  1856. // with the suite name in there.
  1857. //
  1858. l = RegCreateKeyEx(
  1859. HKEY_LOCAL_MACHINE,
  1860. szMasterInfs,
  1861. 0,
  1862. NULL,
  1863. REG_OPTION_NON_VOLATILE,
  1864. (operation == infQuery) ? KEY_QUERY_VALUE : KEY_SET_VALUE,
  1865. NULL,
  1866. &hKey,
  1867. &Type
  1868. );
  1869. if (l != NO_ERROR)
  1870. return (operation == infQuery) ? 0 : l;
  1871. // do the job
  1872. switch (operation) {
  1873. case infQuery:
  1874. Size = sizeof(DWORD);
  1875. l = RegQueryValueEx(hKey,SuiteName,NULL,&Type,(LPBYTE)&Data,&Size);
  1876. if((l == NO_ERROR) && (Type == REG_DWORD) && Data)
  1877. l = TRUE;
  1878. else
  1879. l = FALSE;
  1880. break;
  1881. case infSet:
  1882. Data = 1;
  1883. l = RegSetValueEx(hKey,SuiteName,0,REG_DWORD,(LPBYTE)&Data,sizeof(DWORD));
  1884. break;
  1885. case infReset:
  1886. l = RegDeleteValue(hKey,SuiteName);
  1887. break;
  1888. }
  1889. RegCloseKey(hKey);
  1890. return l;
  1891. }
  1892. PVOID
  1893. OcInitialize(
  1894. IN POCM_CLIENT_CALLBACKS Callbacks,
  1895. IN LPCTSTR MasterOcInfName,
  1896. IN UINT Flags,
  1897. OUT PBOOL ShowError,
  1898. IN PVOID Log
  1899. )
  1900. /*++
  1901. Routine Description:
  1902. Initializes the OC Manager. The master OC INF is loaded and
  1903. processed, which includes INFs, loading setup interface DLLs and
  1904. querying interface entry points. A set of in-memory structures
  1905. is built up.
  1906. If the OC INF hasn't been processed before then we copy the
  1907. installation files for the component(s) into %windir%\system32\setup.
  1908. Files are expected to be in the same directory as the OC inf.
  1909. Arguments:
  1910. Callbacks - supplies a set of routines used by the OC Manager
  1911. to perform various functions.
  1912. MasterOcInfName - supplies the full Win32 path of the master OC inf.
  1913. Flags - supplies various flags that control operation.
  1914. ShowError - receives a flag that is valid if this routine fails,
  1915. advising the caller whether he should show an error message.
  1916. Log - supplies a handle to use to log errors.
  1917. Return Value:
  1918. Opaque pointer to internal context structure or NULL if failure.
  1919. If NULL, an error will have been logged to indicate what failed.
  1920. --*/
  1921. {
  1922. POC_MANAGER OcManager;
  1923. UINT i;
  1924. HKEY hKey;
  1925. DWORD DontCare;
  1926. LONG l;
  1927. BOOL Canceled;
  1928. BOOL rc;
  1929. TCHAR *p;
  1930. *ShowError = TRUE;
  1931. // init the wizard handle
  1932. WizardDialogHandle = NULL;
  1933. //
  1934. // Allocate a new OC MANAGER structure
  1935. //
  1936. OcManager = pSetupMalloc(sizeof(OC_MANAGER));
  1937. if(!OcManager) {
  1938. //
  1939. // Make the callback work
  1940. //
  1941. OC_MANAGER ocm;
  1942. ocm.Callbacks = *Callbacks;
  1943. _LogError(&ocm,OcErrLevFatal,MSG_OC_OOM);
  1944. goto c0;
  1945. }
  1946. ZeroMemory(OcManager,sizeof(OC_MANAGER));
  1947. OcManager->Callbacks = *Callbacks;
  1948. OcManager->CurrentComponentStringId = -1;
  1949. OcManager->SetupMode = SETUPMODE_CUSTOM;
  1950. OcManager->UnattendedInf = INVALID_HANDLE_VALUE;
  1951. gLastOcManager = (POC_MANAGER) OcManager;
  1952. if (!pOcInitPaths(OcManager, MasterOcInfName)) {
  1953. _LogError(OcManager,OcErrLevFatal,MSG_OC_MASTER_INF_LOAD_FAILED);
  1954. goto c1;
  1955. }
  1956. sapiAssert(*OcManager->MasterOcInfPath);
  1957. #ifdef UNICODE
  1958. OcFillInSetupDataW(&(OcManager->SetupData));
  1959. #else
  1960. OcFillInSetupDataA(&(OcManager->SetupData));
  1961. #endif
  1962. // Say the user canceled until we successfully complete an install
  1963. // This will prevent Inf Files from being copied to system32\setup
  1964. OcManager->InternalFlags |= OCMFLAG_USERCANCELED;
  1965. // Make sure the OC Manager key exists in the registry as a non-volatile key.
  1966. // Other parts of the code deal in volatile keys so some care is needed to
  1967. // avoid getting a volatile key under which we want to create non-volatile entries.
  1968. //
  1969. l = RegCreateKeyEx(
  1970. HKEY_LOCAL_MACHINE,
  1971. szOcManagerRoot,
  1972. 0,
  1973. NULL,
  1974. REG_OPTION_NON_VOLATILE,
  1975. KEY_CREATE_SUB_KEY | KEY_QUERY_VALUE | KEY_SET_VALUE,
  1976. NULL,
  1977. &hKey,
  1978. &DontCare
  1979. );
  1980. if(l == NO_ERROR) {
  1981. RegCloseKey(hKey);
  1982. } else {
  1983. _LogError(OcManager,OcErrLevWarning,MSG_OC_CREATE_KEY_FAILED,szOcManagerRoot,l);
  1984. }
  1985. // get the locale info
  1986. locale.lcid = GetSystemDefaultLCID();
  1987. GetLocaleInfo(locale.lcid,
  1988. LOCALE_SDECIMAL,
  1989. locale.DecimalSeparator,
  1990. sizeof(locale.DecimalSeparator)/sizeof(TCHAR));
  1991. //
  1992. // Initialize string tables.
  1993. //
  1994. OcManager->InfListStringTable = pSetupStringTableInitializeEx(sizeof(OC_INF),0);
  1995. if(!OcManager->InfListStringTable) {
  1996. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  1997. goto c1;
  1998. }
  1999. OcManager->ComponentStringTable = pSetupStringTableInitializeEx(sizeof(OPTIONAL_COMPONENT),0);
  2000. if(!OcManager->ComponentStringTable) {
  2001. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  2002. goto c2;
  2003. }
  2004. //
  2005. // Initialize various arrays. We alloc 0-length blocks here to allow
  2006. // realloc later without special casing.
  2007. //
  2008. OcManager->TopLevelOcStringIds = pSetupMalloc(0);
  2009. if(!OcManager->TopLevelOcStringIds) {
  2010. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  2011. goto c3;
  2012. }
  2013. for(i=0; i<WizPagesTypeMax; i++) {
  2014. OcManager->WizardPagesOrder[i] = pSetupMalloc(0);
  2015. if(!OcManager->WizardPagesOrder[i]) {
  2016. _LogError(OcManager,OcErrLevFatal,MSG_OC_OOM);
  2017. goto c4;
  2018. }
  2019. }
  2020. //
  2021. // Initialize the private data store.
  2022. //
  2023. if(!pOcManagerInitPrivateDataStore(OcManager,Log)) {
  2024. _LogError(OcManager,OcErrLevFatal,MSG_OC_PRIVATEDATASTORE_INIT_FAILED);
  2025. goto c4;
  2026. }
  2027. //
  2028. // Load the master OC inf.
  2029. //
  2030. if(!pOcLoadMasterOcInf(OcManager, Flags, Log)) {
  2031. _LogError(OcManager,OcErrLevFatal,MSG_OC_MASTER_INF_LOAD_FAILED);
  2032. goto c5;
  2033. }
  2034. // set the source path
  2035. lstrcpy(OcManager->SourceDir, OcManager->MasterOcInfPath);
  2036. if (p = _tcsrchr(OcManager->SourceDir, TEXT('\\')))
  2037. *p = 0;
  2038. else
  2039. GetCurrentDirectory(MAX_PATH, OcManager->SourceDir);
  2040. //
  2041. // Load the unattend file
  2042. //
  2043. if(OcManager->SetupData.UnattendFile[0]) {
  2044. OcManager->UnattendedInf = SetupOpenInfFile(
  2045. OcManager->SetupData.UnattendFile,
  2046. NULL,
  2047. INF_STYLE_WIN4,
  2048. NULL
  2049. );
  2050. if (OcManager->UnattendedInf == INVALID_HANDLE_VALUE && GetLastError() == ERROR_WRONG_INF_STYLE) {
  2051. OcManager->UnattendedInf = SetupOpenInfFile(
  2052. OcManager->SetupData.UnattendFile,
  2053. NULL,
  2054. INF_STYLE_OLDNT,
  2055. NULL
  2056. );
  2057. }
  2058. if(OcManager->UnattendedInf == INVALID_HANDLE_VALUE) {
  2059. _LogError(OcManager,OcErrLevFatal,MSG_OC_CANT_OPEN_INF,OcManager->SetupData.UnattendFile,GetLastError());
  2060. goto c6;
  2061. }
  2062. }
  2063. //
  2064. // Load component infs and DLLs.
  2065. //
  2066. rc = pOcLoadInfsAndDlls(OcManager, &Canceled, Log);
  2067. //
  2068. // Error already logged if necessary
  2069. //
  2070. if (Canceled)
  2071. *ShowError = FALSE;
  2072. if (!rc)
  2073. goto c6;
  2074. pOcClearAllErrorStates(OcManager);
  2075. if(!pOcFetchInstallStates(OcManager)) {
  2076. //
  2077. // Error already logged.
  2078. //
  2079. goto c6;
  2080. }
  2081. //
  2082. // Ask the components to give up a rough estimate
  2083. // of their sizes so we can say something meaningful in the list boxes.
  2084. //
  2085. pOcGetApproximateDiskSpace(OcManager);
  2086. return(OcManager);
  2087. c6:
  2088. sapiAssert(OcManager->MasterOcInf != INVALID_HANDLE_VALUE);
  2089. SetupCloseInfFile(OcManager->MasterOcInf);
  2090. if (OcManager->UnattendedInf != INVALID_HANDLE_VALUE)
  2091. SetupCloseInfFile(OcManager->UnattendedInf);
  2092. c5:
  2093. //
  2094. // Tear down private data store.
  2095. //
  2096. pOcManagerTearDownPrivateDataStore(OcManager);
  2097. c4:
  2098. if(OcManager->TopLevelOcStringIds) {
  2099. pSetupFree(OcManager->TopLevelOcStringIds);
  2100. for(i=0; OcManager->WizardPagesOrder[i] && (i<WizPagesTypeMax); i++) {
  2101. pSetupFree(OcManager->WizardPagesOrder[i]);
  2102. }
  2103. }
  2104. // free up the list of aborted components
  2105. if (OcManager->AbortedComponentIds) {
  2106. pSetupFree(OcManager->AbortedComponentIds);
  2107. }
  2108. c3:
  2109. pOcDestroyPerOcData(OcManager);
  2110. pSetupStringTableDestroy(OcManager->ComponentStringTable);
  2111. c2:
  2112. pSetupStringTableDestroy(OcManager->InfListStringTable);
  2113. c1:
  2114. pSetupFree(OcManager);
  2115. c0:
  2116. return(NULL);
  2117. }
  2118. VOID
  2119. OcTerminate(
  2120. IN OUT PVOID *OcManagerContext
  2121. )
  2122. /*++
  2123. Routine Description:
  2124. This routine shuts down the OC Manager, including calling the subcomponents
  2125. to clean themselves up, release resources, etc.
  2126. Arguments:
  2127. OcManagerContext - in input, supplies a pointer to a context value returned
  2128. by OcInitialize. On output, receives NULL.
  2129. Return Value:
  2130. None.
  2131. --*/
  2132. {
  2133. POC_MANAGER OcManager;
  2134. UINT u;
  2135. BOOL b;
  2136. OPTIONAL_COMPONENT Oc;
  2137. OC_INF OcInf;
  2138. LPCTSTR ComponentName;
  2139. LPCTSTR InfName;
  2140. sapiAssert(OcManagerContext && *OcManagerContext);
  2141. OcManager = *OcManagerContext;
  2142. *OcManagerContext = NULL;
  2143. //
  2144. // Run down the top-level OCs, calling the dlls to indicate that we're done.
  2145. // We don't tear down any infrastructure until after we've told all the dlls
  2146. // that we're done.
  2147. //
  2148. for(u=0; u<OcManager->TopLevelOcCount; u++) {
  2149. OcInterfaceCleanup(OcManager,OcManager->TopLevelOcStringIds[u]);
  2150. }
  2151. // if the user did not cancel setup
  2152. //
  2153. if (!(OcManager->InternalFlags & OCMFLAG_USERCANCELED)){
  2154. //
  2155. // Remember persistent install state.
  2156. //
  2157. pOcRememberInstallStates(OcManager);
  2158. // copy the master INF file over
  2159. if ( OcManager->InternalFlags & OCMFLAG_NEWINF ) {
  2160. b = pOcInstallSetupComponents(
  2161. NULL,
  2162. OcManager,
  2163. OcManager->SuiteName,
  2164. NULL,
  2165. _tcsrchr(OcManager->MasterOcInfPath,TEXT('\\')), // InfName,
  2166. NULL,
  2167. NULL
  2168. );
  2169. }
  2170. }
  2171. //
  2172. // Run down the top-level OCs and free DLLs and per-component infs.
  2173. //
  2174. for(u=0; u<OcManager->TopLevelOcCount; u++) {
  2175. ComponentName = pSetupStringTableStringFromId( // ComponentName
  2176. OcManager->ComponentStringTable,
  2177. OcManager->TopLevelOcStringIds[u]);
  2178. pSetupStringTableGetExtraData(
  2179. OcManager->ComponentStringTable,
  2180. OcManager->TopLevelOcStringIds[u],
  2181. &Oc,
  2182. sizeof(OPTIONAL_COMPONENT)
  2183. );
  2184. if(Oc.InstallationDll && (Oc.InstallationDll != MyModuleHandle)) {
  2185. //FreeLibrary(Oc.InstallationDll);
  2186. }
  2187. if(Oc.InfStringId != -1) {
  2188. pSetupStringTableGetExtraData(
  2189. OcManager->InfListStringTable,
  2190. Oc.InfStringId,
  2191. &OcInf,
  2192. sizeof(OC_INF)
  2193. );
  2194. if(OcInf.Handle && (OcInf.Handle != INVALID_HANDLE_VALUE)) {
  2195. SetupCloseInfFile(OcInf.Handle);
  2196. //
  2197. // Mark the handle as closed, is to components are shareing
  2198. // the same INF file, should only close the file once.
  2199. //
  2200. OcInf.Handle = INVALID_HANDLE_VALUE;
  2201. pSetupStringTableSetExtraData(
  2202. OcManager->InfListStringTable,
  2203. Oc.InfStringId,
  2204. &OcInf,
  2205. sizeof(OC_INF)
  2206. );
  2207. }
  2208. }
  2209. // This is a new install and we did not cancel out of setup
  2210. // copy the setup components to the setup dir otherwise a canceled
  2211. // setup will leave an upgrade dead in the water.
  2212. if ( ( (OcManager->InternalFlags & OCMFLAG_NEWINF )
  2213. || ( (OcManager->SetupMode & SETUPMODE_PRIVATE_MASK) == SETUPMODE_REMOVEALL ) )
  2214. && (! (OcManager->InternalFlags & OCMFLAG_USERCANCELED))){
  2215. if (Oc.InfStringId == -1) {
  2216. InfName = NULL;
  2217. } else {
  2218. InfName = pSetupStringTableStringFromId(OcManager->InfListStringTable,
  2219. Oc.InfStringId);
  2220. }
  2221. b = pOcInstallSetupComponents(
  2222. &Oc,
  2223. OcManager,
  2224. ComponentName,
  2225. Oc.InstallationDllName,
  2226. InfName,
  2227. NULL,
  2228. NULL
  2229. );
  2230. if(!b) {
  2231. _LogError(OcManager,OcErrLevFatal,MSG_OC_CANT_INSTALL_SETUP,ComponentName);
  2232. }
  2233. }
  2234. }
  2235. //
  2236. // Free string tables.
  2237. //
  2238. pOcDestroyPerOcData(OcManager);
  2239. pSetupStringTableDestroy(OcManager->ComponentStringTable);
  2240. pSetupStringTableDestroy(OcManager->InfListStringTable);
  2241. //
  2242. // Free the wizard page ordering arrays.
  2243. //
  2244. for(u=0; u<WizPagesTypeMax; u++) {
  2245. if(OcManager->WizardPagesOrder[u]) {
  2246. pSetupFree(OcManager->WizardPagesOrder[u]);
  2247. }
  2248. }
  2249. //
  2250. // Tear down the private data store.
  2251. //
  2252. pOcManagerTearDownPrivateDataStore(OcManager);
  2253. //
  2254. // Free the master oc inf and finally the oc manager context structure itself.
  2255. //
  2256. sapiAssert(OcManager->MasterOcInf != INVALID_HANDLE_VALUE);
  2257. SetupCloseInfFile(OcManager->MasterOcInf);
  2258. if (OcManager->UnattendedInf && OcManager->UnattendedInf != INVALID_HANDLE_VALUE)
  2259. SetupCloseInfFile(OcManager->UnattendedInf);
  2260. // If this is a remove all then Delete the master inf file too!
  2261. if ( !(OcManager->InternalFlags & OCMFLAG_USERCANCELED)
  2262. && ( (OcManager->SetupMode & SETUPMODE_PRIVATE_MASK)
  2263. == SETUPMODE_REMOVEALL )) {
  2264. TCHAR InFSuitePath[MAX_PATH];
  2265. // Se do it this way, so if you run remove all from the
  2266. // orginal source location we don't blow away that copy of the
  2267. // suite inf file.
  2268. pOcFormSuitePath(NULL,OcManager->MasterOcInfPath,InFSuitePath);
  2269. DeleteFile(InFSuitePath);
  2270. }
  2271. // free up the list of aborted components
  2272. if (OcManager->AbortedComponentIds) {
  2273. pSetupFree(OcManager->AbortedComponentIds);
  2274. }
  2275. if (OcManager->TopLevelOcStringIds) {
  2276. pSetupFree(OcManager->TopLevelOcStringIds);
  2277. }
  2278. if (OcManager->TopLevelParentOcStringIds) {
  2279. pSetupFree(OcManager->TopLevelParentOcStringIds);
  2280. }
  2281. //
  2282. // free the oc manager context structure itself.
  2283. //
  2284. pSetupFree(OcManager);
  2285. gLastOcManager = NULL;
  2286. }
  2287. BOOL
  2288. OcSubComponentsPresent(
  2289. IN PVOID OcManagerContext
  2290. )
  2291. /*++
  2292. Routine Description:
  2293. This routine tells the caller if there are any
  2294. subcomponents available on the details page.
  2295. Arguments:
  2296. OcManager - supplies pointer to OC Manager context structure.
  2297. Return Value:
  2298. TRUE = yes, there are subcomponents
  2299. FALSE = no way
  2300. --*/
  2301. {
  2302. POC_MANAGER OcManager = (POC_MANAGER)OcManagerContext;
  2303. if (!OcManagerContext) {
  2304. return FALSE;
  2305. }
  2306. return OcManager->SubComponentsPresent;
  2307. }
  2308. VOID
  2309. pOcDestroyPerOcData(
  2310. IN POC_MANAGER OcManager
  2311. )
  2312. /*++
  2313. Routine Description:
  2314. This routine frees all data allocated as part of the per-subcomponent
  2315. data structures. The component list string table is enumerated;
  2316. the OPTIONAL_COMPONENT structures have several pointers for arrays
  2317. that must be freed.
  2318. Arguments:
  2319. OcManager - supplies pointer to OC Manager context structure.
  2320. Return Value:
  2321. None.
  2322. --*/
  2323. {
  2324. OPTIONAL_COMPONENT OptionalComponent;
  2325. pSetupStringTableEnum(
  2326. OcManager->ComponentStringTable,
  2327. &OptionalComponent,
  2328. sizeof(OPTIONAL_COMPONENT),
  2329. pOcDestroyPerOcDataStringCB,
  2330. 0
  2331. );
  2332. }
  2333. BOOL
  2334. pOcDestroyPerOcDataStringCB(
  2335. IN PVOID StringTable,
  2336. IN LONG StringId,
  2337. IN PCTSTR String,
  2338. IN POPTIONAL_COMPONENT Oc,
  2339. IN UINT OcSize,
  2340. IN LPARAM Unused
  2341. )
  2342. /*++
  2343. Routine Description:
  2344. String table callback routine that is the worker routine for
  2345. pOcDestroyPerOcData.
  2346. Arguments:
  2347. Standard string table callback arguments.
  2348. Return Value:
  2349. Always returns TRUE to continue enumeration.
  2350. --*/
  2351. {
  2352. UNREFERENCED_PARAMETER(StringTable);
  2353. UNREFERENCED_PARAMETER(StringId);
  2354. UNREFERENCED_PARAMETER(String);
  2355. UNREFERENCED_PARAMETER(OcSize);
  2356. UNREFERENCED_PARAMETER(Unused);
  2357. if(Oc->NeedsStringIds) {
  2358. pSetupFree(Oc->NeedsStringIds);
  2359. }
  2360. if(Oc->NeededByStringIds) {
  2361. pSetupFree(Oc->NeededByStringIds);
  2362. }
  2363. if (Oc->ExcludeStringIds){
  2364. pSetupFree(Oc->ExcludeStringIds);
  2365. }
  2366. if(Oc->ExcludedByStringIds){
  2367. pSetupFree(Oc->ExcludedByStringIds);
  2368. }
  2369. if (Oc->HelperContext) {
  2370. pSetupFree(Oc->HelperContext);
  2371. }
  2372. return(TRUE);
  2373. }
  2374. BOOL
  2375. pOcClearAllErrorStatesCB(
  2376. IN PVOID StringTable,
  2377. IN LONG StringId,
  2378. IN PCTSTR String,
  2379. IN POPTIONAL_COMPONENT Oc,
  2380. IN UINT OcSize,
  2381. IN LPARAM OcManager
  2382. )
  2383. {
  2384. UNREFERENCED_PARAMETER(StringTable);
  2385. UNREFERENCED_PARAMETER(StringId);
  2386. UNREFERENCED_PARAMETER(String);
  2387. UNREFERENCED_PARAMETER(OcSize);
  2388. OcHelperClearExternalError ((POC_MANAGER)OcManager, StringId ,0);
  2389. return(TRUE);
  2390. }
  2391. VOID
  2392. pOcClearAllErrorStates(
  2393. IN POC_MANAGER OcManager
  2394. )
  2395. /*++
  2396. Routine Description:
  2397. This routine clears out the past registry entries for error reports
  2398. for all components
  2399. Arguments:
  2400. OcManagerContext - in input, supplies a pointer to a context value returned
  2401. by OcInitialize. On output, receives NULL.
  2402. Return Value:
  2403. None.
  2404. --*/
  2405. {
  2406. OPTIONAL_COMPONENT OptionalComponent;
  2407. pSetupStringTableEnum(
  2408. OcManager->ComponentStringTable,
  2409. &OptionalComponent,
  2410. sizeof(OPTIONAL_COMPONENT),
  2411. pOcClearAllErrorStatesCB,
  2412. (LPARAM)OcManager
  2413. );
  2414. }
  2415. BOOL
  2416. pOcRemoveComponent(
  2417. IN POC_MANAGER OcManager,
  2418. IN LONG ComponentId,
  2419. IN DWORD PhaseId
  2420. )
  2421. /*++
  2422. Routine Description:
  2423. This routine adds a specified component to the list of aborted components,
  2424. preventing it's entry function from being called any more.
  2425. Arguments:
  2426. OcManager - in input, supplies a pointer to a context value returned
  2427. by OcInitialize. On output, receives NULL.
  2428. ComponentId - in input, this string names the faulty component to be removed
  2429. Return Value:
  2430. None.
  2431. --*/
  2432. {
  2433. PVOID p;
  2434. OPTIONAL_COMPONENT Oc;
  2435. // test for valid component to remove
  2436. if (ComponentId <= 0)
  2437. return FALSE;
  2438. if (pOcComponentWasRemoved(OcManager, ComponentId))
  2439. return FALSE;
  2440. // add component to list of aborted components
  2441. if (!OcManager->AbortedCount) {
  2442. OcManager->AbortedComponentIds = pSetupMalloc(sizeof(UINT));
  2443. if (!OcManager->AbortedComponentIds)
  2444. return FALSE;
  2445. }
  2446. OcManager->AbortedCount++;
  2447. p = pSetupRealloc(OcManager->AbortedComponentIds, sizeof(UINT) * OcManager->AbortedCount);
  2448. if (!p) {
  2449. OcManager->AbortedCount--;
  2450. return FALSE;
  2451. }
  2452. OcManager->AbortedComponentIds = (UINT *)p;
  2453. OcManager->AbortedComponentIds[OcManager->AbortedCount - 1] = ComponentId;
  2454. // stop display of component in the listbox, if it isn't too late
  2455. pSetupStringTableGetExtraData(
  2456. OcManager->ComponentStringTable,
  2457. ComponentId,
  2458. &Oc,
  2459. sizeof(OPTIONAL_COMPONENT)
  2460. );
  2461. Oc.InternalFlags |= OCFLAG_HIDE;
  2462. pSetupStringTableSetExtraData(
  2463. OcManager->ComponentStringTable,
  2464. ComponentId,
  2465. &Oc,
  2466. sizeof(OPTIONAL_COMPONENT)
  2467. );
  2468. _LogError(OcManager,
  2469. OcErrLevInfo | OcErrBatch,
  2470. MSG_OC_REMOVE_COMPONENT,
  2471. pSetupStringTableStringFromId(OcManager->ComponentStringTable,ComponentId),
  2472. PhaseId
  2473. );
  2474. return TRUE;
  2475. }
  2476. BOOL
  2477. pOcComponentWasRemoved(
  2478. IN POC_MANAGER OcManager,
  2479. IN LONG ComponentId
  2480. )
  2481. /*++
  2482. Routine Description:
  2483. This routine indicates if a components has been aborted.
  2484. Arguments:
  2485. OcManager - in input, supplies a pointer to a context value returned
  2486. by OcInitialize. On output, receives NULL.
  2487. ComponentId - in input, this string names the component to check for
  2488. Return Value:
  2489. BOOL - true if it was aborted - else false
  2490. --*/
  2491. {
  2492. UINT i;
  2493. for (i = 0; i < OcManager->AbortedCount; i++) {
  2494. if (OcManager->AbortedComponentIds[i] == ComponentId) {
  2495. return TRUE;
  2496. }
  2497. }
  2498. return FALSE;
  2499. }