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.

977 lines
23 KiB

  1. /*++
  2. Copyright (c) 2000 Microsoft Corporation
  3. Module Name:
  4. components.c
  5. Abstract:
  6. Implements a set of APIs for the purposes of allowing the application layer to select
  7. module functionality.
  8. Author:
  9. Jim Schmidt (jimschm) 07-Aug-2000
  10. Revision History:
  11. <alias> <date> <comments>
  12. --*/
  13. //
  14. // Includes
  15. //
  16. #include "pch.h"
  17. #include "ism.h"
  18. #include "ismp.h"
  19. #define DBG_COMP "Comp"
  20. //
  21. // Strings
  22. //
  23. #define S_COMPONENT_ROOT TEXT("Components")
  24. #define S_USER_SUPPLIED TEXT("User")
  25. #define S_MODULE_SUPPLIED TEXT("Module")
  26. //
  27. // Constants
  28. //
  29. #define MAX_COMPONENT_SPEC 127
  30. #define MAX_COMPONENT_SPEC_PLUS_NUL (MAX_COMPONENT_SPEC+1)
  31. #define MAX_CONTROLLED_NODE_SIZE (MAX_COMPONENT_SPEC_PLUS_NUL + \
  32. ARRAYSIZE(S_COMPONENT_ROOT) + \
  33. ARRAYSIZE(S_MODULE_SUPPLIED) + 16)
  34. #define MEMDB_FLAG_PREFERRED 1
  35. #define MEMDB_FLAG_SELECTED 1
  36. //
  37. // Macros
  38. //
  39. // none
  40. //
  41. // Types
  42. //
  43. typedef enum {
  44. CES_DONE = 0,
  45. CES_FIRST_COMPONENT,
  46. CES_NEXT_COMPONENT,
  47. CES_FIRST_ALIAS,
  48. CES_NEXT_ALIAS,
  49. CES_CHECK_ALIAS_FLAGS
  50. } COMPONENTENUMSTATE;
  51. typedef struct {
  52. MEMDB_ENUM ComponentEnumStruct;
  53. MEMDB_ENUM AliasEnumStruct;
  54. BOOL EnumAliases;
  55. UINT GroupIdFilter;
  56. BOOL EnumEnabled;
  57. BOOL EnumDisabled;
  58. BOOL EnumPreferredOnly;
  59. BOOL EnumNonPreferredOnly;
  60. COMPONENTENUMSTATE State;
  61. } COMPONENTENUM_HANDLE, *PCOMPONENTENUM_HANDLE;
  62. //
  63. // Globals
  64. //
  65. // none
  66. //
  67. // Macro expansion list
  68. //
  69. // none
  70. //
  71. // Private function prototypes
  72. //
  73. // none
  74. //
  75. // Macro expansion definition
  76. //
  77. // none
  78. //
  79. // Code
  80. //
  81. BOOL
  82. pCheckCompChar (
  83. IN CHARTYPE Char,
  84. IN BOOL CheckDecoration
  85. )
  86. {
  87. //
  88. // Process decoration chars
  89. //
  90. if (Char == TEXT('$') || Char == TEXT('@') || Char == TEXT('~') || Char == TEXT('#')) {
  91. return CheckDecoration;
  92. }
  93. if (CheckDecoration) {
  94. return FALSE;
  95. }
  96. //
  97. // Block illegal chars
  98. //
  99. if (Char == TEXT('\"') || Char == TEXT('*') || Char == TEXT('?') || Char== TEXT('\\') ||
  100. Char == TEXT('%') || Char == TEXT(';')
  101. ) {
  102. return FALSE;
  103. }
  104. //
  105. // Make sure char is printable
  106. //
  107. if (Char < 33 || Char > 126) {
  108. return FALSE;
  109. }
  110. return TRUE;
  111. }
  112. BOOL
  113. pCheckComponentName (
  114. IN PCTSTR ComponentString
  115. )
  116. {
  117. BOOL result = FALSE;
  118. PCTSTR end;
  119. PCTSTR begin;
  120. //
  121. // Check for a non-empty spec
  122. //
  123. if (ComponentString && ComponentString[0]) {
  124. //
  125. // Allow for decoration
  126. //
  127. end = ComponentString;
  128. while (pCheckCompChar ((CHARTYPE) _tcsnextc (end), TRUE)) {
  129. end = _tcsinc (end);
  130. }
  131. //
  132. // Now enforce the name character set: non-decorated characters and no
  133. // more than MAX_COMPONENT_SPEC characters. Allow spaces in the middle.
  134. //
  135. begin = end;
  136. while (*end) {
  137. if (!pCheckCompChar ((CHARTYPE) _tcsnextc (end), FALSE)) {
  138. if (_tcsnextc (end) == TEXT(' ')) {
  139. if (!end[1] || end == begin) {
  140. break;
  141. }
  142. } else {
  143. break;
  144. }
  145. }
  146. end = _tcsinc (end);
  147. }
  148. if (!(*end) && *begin) {
  149. if (end - ComponentString <= MAX_COMPONENT_SPEC) {
  150. result = TRUE;
  151. }
  152. }
  153. }
  154. if (!result) {
  155. SetLastError (ERROR_INVALID_PARAMETER);
  156. DEBUGMSG ((DBG_ERROR, "%s is not a valid component name", ComponentString));
  157. }
  158. return result;
  159. }
  160. BOOL
  161. pFindComponent (
  162. IN PCTSTR LocalizedAlias,
  163. IN UINT ComponentGroupId,
  164. OUT PCTSTR *ComponentKey, OPTIONAL
  165. OUT PCTSTR *AliasKey OPTIONAL
  166. )
  167. {
  168. MEMDB_ENUM component;
  169. PCTSTR memdbNode = NULL;
  170. PCTSTR encodedUserAlias;
  171. PCTSTR encodedModuleAlias;
  172. TCHAR number[32];
  173. BOOL result = FALSE;
  174. //
  175. // Find the component based on the localized alias
  176. //
  177. wsprintf (number, TEXT("\\%s\\%04u"), S_USER_SUPPLIED, ComponentGroupId);
  178. encodedUserAlias = JoinPaths (number, LocalizedAlias);
  179. wsprintf (number, TEXT("\\%s\\%04u"), S_MODULE_SUPPLIED, ComponentGroupId);
  180. encodedModuleAlias = JoinPaths (number, LocalizedAlias);
  181. if (MemDbEnumFirst (
  182. &component,
  183. S_COMPONENT_ROOT TEXT("\\*"),
  184. ENUMFLAG_ALL,
  185. 1,
  186. 1
  187. )) {
  188. do {
  189. memdbNode = JoinText (component.FullKeyName, encodedModuleAlias);
  190. if (MemDbTestKey (memdbNode)) {
  191. break;
  192. }
  193. FreeText (memdbNode);
  194. memdbNode = NULL;
  195. memdbNode = JoinText (component.FullKeyName, encodedUserAlias);
  196. if (MemDbTestKey (memdbNode)) {
  197. break;
  198. }
  199. FreeText (memdbNode);
  200. memdbNode = NULL;
  201. } while (MemDbEnumNext (&component));
  202. }
  203. if (memdbNode) {
  204. if (ComponentKey) {
  205. *ComponentKey = DuplicateText (component.FullKeyName);
  206. }
  207. if (AliasKey) {
  208. *AliasKey = memdbNode;
  209. memdbNode = NULL;
  210. }
  211. MemDbAbortEnum (&component);
  212. result = TRUE;
  213. }
  214. FreeText (memdbNode);
  215. INVALID_POINTER (memdbNode);
  216. FreePathString (encodedUserAlias);
  217. INVALID_POINTER (encodedUserAlias);
  218. FreePathString (encodedModuleAlias);
  219. INVALID_POINTER (encodedModuleAlias);
  220. return result;
  221. }
  222. BOOL
  223. WINAPI
  224. IsmSelectPreferredAlias (
  225. IN PCTSTR ComponentString,
  226. IN PCTSTR LocalizedAlias, OPTIONAL
  227. IN UINT ComponentGroupId OPTIONAL
  228. )
  229. /*++
  230. Routine Description:
  231. IsmSelectPreferredAlias marks a specific alias as the "preferred" one, so
  232. that the UI knows what to display. If LocalizedAlias is not specified, none
  233. of the aliases are preferred.
  234. A component can have only one preferred localized alias. If another alias is
  235. selected as preferred, it will be deselected automatically.
  236. Arguments:
  237. ComponentString - Specifies the non-displayed component identifier
  238. LocalizedAlias - Specifies the displayable string to mark as "preferred,"
  239. or NULL to remove the preferred flag from the component.
  240. ComponentGroupId - Specifies the group ID for LocalizedAlias. Required if
  241. LocalizedAlias is not NULL.
  242. Return Value:
  243. TRUE if selection (or deselection) succeeded, FALSE if LocalizedAlias does
  244. not exist.
  245. --*/
  246. {
  247. MEMDB_ENUM e;
  248. TCHAR number[32];
  249. PCTSTR memdbNode = NULL;
  250. PCTSTR baseOfPattern;
  251. PCTSTR enumPattern;
  252. PCTSTR groupedAlias;
  253. BOOL result;
  254. if (!ComponentString || (LocalizedAlias && !ComponentGroupId)) {
  255. MYASSERT (FALSE);
  256. SetLastError (ERROR_INVALID_PARAMETER);
  257. return FALSE;
  258. }
  259. result = (LocalizedAlias == NULL);
  260. //
  261. // Build enumeration string Components\<component>\*
  262. //
  263. baseOfPattern = JoinPaths (S_COMPONENT_ROOT, ComponentString);
  264. enumPattern = JoinPaths (baseOfPattern, TEXT("*"));
  265. FreePathString (baseOfPattern);
  266. INVALID_POINTER (baseOfPattern);
  267. if (LocalizedAlias) {
  268. wsprintf (number, TEXT("%04u"), ComponentGroupId);
  269. groupedAlias = JoinPaths (number, LocalizedAlias);
  270. } else {
  271. groupedAlias = NULL;
  272. }
  273. if (MemDbEnumFirst (
  274. &e,
  275. enumPattern,
  276. ENUMFLAG_NORMAL,
  277. 3,
  278. ENUMLEVEL_ALLLEVELS
  279. )) {
  280. do {
  281. if (groupedAlias && StringIMatch (e.KeyName, groupedAlias)) {
  282. MemDbSetFlags (e.FullKeyName, MEMDB_FLAG_PREFERRED, MEMDB_FLAG_PREFERRED);
  283. result = TRUE;
  284. } else {
  285. MemDbSetFlags (e.FullKeyName, 0, MEMDB_FLAG_PREFERRED);
  286. }
  287. } while (MemDbEnumNext (&e));
  288. }
  289. FreePathString (enumPattern);
  290. INVALID_POINTER (enumPattern);
  291. FreePathString (groupedAlias);
  292. INVALID_POINTER (groupedAlias);
  293. return result;
  294. }
  295. BOOL
  296. WINAPI
  297. IsmAddComponentAlias (
  298. IN PCTSTR ComponentString, OPTIONAL
  299. IN UINT MasterGroup,
  300. IN PCTSTR LocalizedAlias,
  301. IN UINT ComponentGroupId,
  302. IN BOOL UserSupplied
  303. )
  304. /*++
  305. Routine Description:
  306. IsmAddComponentAlias associates a display string (LocalizedAlias) with a
  307. logical component tag (ComponentString).
  308. Arguments:
  309. ComponentString - Specifies the identifier of the component. This
  310. identifier is not used for display purposes.
  311. MasterGroup - Specifies a MASTERGROUP_xxx constant, which organizes
  312. the components into major groups such as system settings
  313. and app settings (to simplify selection).
  314. LocalizedAliais - The displayable text. It is a localized component name,
  315. a path, a file, etc.
  316. ComponentGroupId - An arbitrary numeric ID defined outside of the ISM. This
  317. ID is used to implement requirements specific to the app
  318. layer. It allows for arbitrary idenfication and grouping.
  319. UserSupplied - Specifies TRUE if the end-user supplied this info, FALSE
  320. if it is built into the migration package.
  321. Return Value:
  322. A flag indicating success or failure.
  323. --*/
  324. {
  325. PCTSTR memdbNode;
  326. TCHAR workNode[MAX_CONTROLLED_NODE_SIZE];
  327. static UINT sequencer = 0;
  328. TCHAR madeUpComponent[MAX_COMPONENT_SPEC_PLUS_NUL];
  329. BOOL b;
  330. BOOL newComponent = FALSE;
  331. //
  332. // Components are kept in memdb in the form of
  333. //
  334. // Component\<Module|User>\<GroupId>\<LocalizedAlias> = <preferred flag>
  335. //
  336. // <GroupId> is stored as a 4 digit number (such as 0001)
  337. //
  338. // Component\<Module|User> = <enable/disable>,<master group>
  339. //
  340. //
  341. // Validate arguments
  342. //
  343. if (ComponentGroupId > 9999) {
  344. SetLastError (ERROR_INVALID_PARAMETER);
  345. return FALSE;
  346. }
  347. if (!ComponentString || !ComponentString[0]) {
  348. wsprintf (madeUpComponent, TEXT("Component %u"), ++sequencer);
  349. ComponentString = madeUpComponent;
  350. } else if (!pCheckComponentName (ComponentString)) {
  351. return FALSE;
  352. }
  353. if (!MasterGroup || MasterGroup >= MASTERGROUP_ALL) {
  354. SetLastError (ERROR_INVALID_PARAMETER);
  355. DEBUGMSG ((DBG_ERROR, "MasterGroup is invalid"));
  356. return FALSE;
  357. }
  358. //
  359. // See if the component already exists
  360. //
  361. wsprintf (workNode, TEXT("%s\\%s"), S_COMPONENT_ROOT, ComponentString);
  362. if (pFindComponent (LocalizedAlias, ComponentGroupId, &memdbNode, NULL)) {
  363. if (StringIMatch (workNode, memdbNode)) {
  364. DEBUGMSG ((DBG_VERBOSE, "Alias %s already exists; not adding it again", LocalizedAlias));
  365. } else {
  366. DEBUGMSG ((
  367. DBG_WARNING,
  368. "Alias %s is in use by component %s; not adding it again",
  369. LocalizedAlias,
  370. _tcschr (memdbNode, TEXT('\\')) + 1
  371. ));
  372. }
  373. FreeText (memdbNode);
  374. return FALSE;
  375. }
  376. //
  377. // Create the component if it doesn't exist, and then add the alias
  378. //
  379. if (!MemDbTestKey (workNode)) {
  380. if (!MemDbSetValueAndFlags (workNode, MasterGroup, MEMDB_FLAG_SELECTED, MEMDB_FLAG_SELECTED)) {
  381. EngineError ();
  382. return FALSE;
  383. }
  384. newComponent = TRUE;
  385. }
  386. wsprintf (
  387. workNode,
  388. TEXT("%s\\%s\\%s\\%04u"),
  389. S_COMPONENT_ROOT,
  390. ComponentString,
  391. UserSupplied ? S_USER_SUPPLIED : S_MODULE_SUPPLIED,
  392. ComponentGroupId
  393. );
  394. memdbNode = JoinPaths (workNode, LocalizedAlias);
  395. if (newComponent) {
  396. b = MemDbSetFlags (memdbNode, MEMDB_FLAG_PREFERRED, MEMDB_FLAG_PREFERRED);
  397. } else {
  398. b = MemDbSetKey (memdbNode);
  399. }
  400. FreePathString (memdbNode);
  401. if (!b) {
  402. EngineError ();
  403. return FALSE;
  404. }
  405. return TRUE;
  406. }
  407. BOOL
  408. WINAPI
  409. IsmSelectComponent (
  410. IN PCTSTR ComponentOrAlias,
  411. IN UINT ComponentGroupId, OPTIONAL
  412. IN BOOL Enable
  413. )
  414. {
  415. PCTSTR memdbNode = NULL;
  416. UINT flags;
  417. BOOL b;
  418. if (ComponentGroupId > 9999) {
  419. SetLastError (ERROR_INVALID_PARAMETER);
  420. DEBUGMSG ((DBG_ERROR, "Invalid component group"));
  421. return FALSE;
  422. }
  423. if (!ComponentOrAlias || !ComponentOrAlias[0]) {
  424. SetLastError (ERROR_INVALID_PARAMETER);
  425. DEBUGMSG ((DBG_ERROR, "Invalid localized alias"));
  426. return FALSE;
  427. }
  428. if (!ComponentGroupId) {
  429. if (!pCheckComponentName (ComponentOrAlias)) {
  430. return FALSE;
  431. }
  432. memdbNode = JoinText (S_COMPONENT_ROOT TEXT("\\"), ComponentOrAlias);
  433. if (!MemDbTestKey (memdbNode)) {
  434. FreeText (memdbNode);
  435. return FALSE;
  436. }
  437. } else {
  438. if (!pFindComponent (ComponentOrAlias, ComponentGroupId, &memdbNode, NULL)) {
  439. SetLastError (ERROR_NO_SUCH_ALIAS);
  440. return FALSE;
  441. }
  442. }
  443. flags = Enable ? MEMDB_FLAG_SELECTED : 0;
  444. b = MemDbSetFlags (memdbNode, flags, MEMDB_FLAG_SELECTED);
  445. FreeText (memdbNode);
  446. if (!b) {
  447. EngineError ();
  448. return FALSE;
  449. }
  450. return TRUE;
  451. }
  452. BOOL
  453. pEnumWorker (
  454. IN OUT PMIG_COMPONENT_ENUM EnumPtr,
  455. IN OUT PCOMPONENTENUM_HANDLE Handle
  456. )
  457. {
  458. BOOL result = FALSE;
  459. PCTSTR pattern;
  460. PCTSTR p;
  461. BOOL enabled;
  462. while (Handle->State != CES_DONE) {
  463. switch (Handle->State) {
  464. case CES_FIRST_COMPONENT:
  465. if (!MemDbEnumFirst (
  466. &Handle->ComponentEnumStruct,
  467. S_COMPONENT_ROOT TEXT("\\*"),
  468. ENUMFLAG_ALL,
  469. 1,
  470. 1
  471. )) {
  472. Handle->State = CES_DONE;
  473. } else {
  474. if (Handle->EnumAliases) {
  475. Handle->State = CES_FIRST_ALIAS;
  476. } else {
  477. Handle->State = CES_NEXT_COMPONENT;
  478. result = TRUE;
  479. }
  480. }
  481. break;
  482. case CES_FIRST_ALIAS:
  483. enabled = ((Handle->ComponentEnumStruct.Flags & MEMDB_FLAG_SELECTED) != 0);
  484. if ((!Handle->EnumEnabled && enabled) ||
  485. (!Handle->EnumDisabled && !enabled)
  486. ) {
  487. Handle->State = CES_NEXT_COMPONENT;
  488. continue;
  489. }
  490. EnumPtr->Instance = 0;
  491. pattern = JoinPaths (Handle->ComponentEnumStruct.FullKeyName, TEXT("*"));
  492. if (!MemDbEnumFirst (
  493. &Handle->AliasEnumStruct,
  494. pattern,
  495. ENUMFLAG_NORMAL,
  496. 4,
  497. ENUMLEVEL_ALLLEVELS
  498. )) {
  499. Handle->State = CES_NEXT_COMPONENT;
  500. } else {
  501. Handle->State = CES_CHECK_ALIAS_FLAGS;
  502. }
  503. FreePathString (pattern);
  504. break;
  505. case CES_NEXT_ALIAS:
  506. if (EnumPtr->SkipToNextComponent) {
  507. MemDbAbortEnum (&Handle->AliasEnumStruct);
  508. Handle->State = CES_NEXT_COMPONENT;
  509. EnumPtr->SkipToNextComponent = FALSE;
  510. break;
  511. }
  512. if (!MemDbEnumNext (&Handle->AliasEnumStruct)) {
  513. Handle->State = CES_NEXT_COMPONENT;
  514. } else {
  515. Handle->State = CES_CHECK_ALIAS_FLAGS;
  516. }
  517. break;
  518. case CES_CHECK_ALIAS_FLAGS:
  519. EnumPtr->Preferred = ((Handle->AliasEnumStruct.Flags & MEMDB_FLAG_PREFERRED) != 0);
  520. Handle->State = CES_NEXT_ALIAS;
  521. if (Handle->EnumPreferredOnly) {
  522. result = EnumPtr->Preferred;
  523. } else if (Handle->EnumNonPreferredOnly) {
  524. result = !EnumPtr->Preferred;
  525. } else {
  526. result = TRUE;
  527. }
  528. break;
  529. case CES_NEXT_COMPONENT:
  530. if (!MemDbEnumNext (&Handle->ComponentEnumStruct)) {
  531. Handle->State = CES_DONE;
  532. } else {
  533. if (Handle->EnumAliases) {
  534. Handle->State = CES_FIRST_ALIAS;
  535. } else {
  536. Handle->State = CES_NEXT_COMPONENT;
  537. result = TRUE;
  538. }
  539. }
  540. break;
  541. default:
  542. Handle->State = CES_DONE;
  543. break;
  544. }
  545. if (result) {
  546. //
  547. // Fill in all of the caller enum struct fields
  548. //
  549. EnumPtr->SkipToNextComponent = FALSE;
  550. EnumPtr->ComponentString = Handle->ComponentEnumStruct.KeyName;
  551. EnumPtr->Enabled = ((Handle->ComponentEnumStruct.Flags & MEMDB_FLAG_SELECTED) != 0);
  552. EnumPtr->MasterGroup = Handle->ComponentEnumStruct.Value;
  553. if (Handle->EnumAliases) {
  554. p = _tcschr (Handle->AliasEnumStruct.FullKeyName, TEXT('\\'));
  555. MYASSERT (p);
  556. if (p) {
  557. p = _tcschr (p + 1, TEXT('\\'));
  558. MYASSERT (p);
  559. }
  560. if (p) {
  561. p++;
  562. if (_totlower (p[0]) == TEXT('u')) {
  563. EnumPtr->UserSupplied = TRUE;
  564. } else {
  565. EnumPtr->UserSupplied = FALSE;
  566. }
  567. p = _tcschr (p, TEXT('\\'));
  568. MYASSERT (p);
  569. }
  570. if (p) {
  571. p++;
  572. EnumPtr->GroupId = _tcstoul (p, (PTSTR *) (&p), 10);
  573. MYASSERT (p && p[0] == TEXT('\\'));
  574. }
  575. if (p) {
  576. EnumPtr->LocalizedAlias = p + 1;
  577. }
  578. //
  579. // If group ID filter was specified, loop until a match is found
  580. //
  581. if (Handle->GroupIdFilter && Handle->GroupIdFilter != EnumPtr->GroupId) {
  582. result = FALSE;
  583. continue;
  584. }
  585. } else {
  586. EnumPtr->Preferred = FALSE;
  587. EnumPtr->UserSupplied = FALSE;
  588. EnumPtr->GroupId = 0;
  589. EnumPtr->LocalizedAlias = NULL;
  590. }
  591. EnumPtr->Instance++;
  592. break;
  593. }
  594. }
  595. if (!result) {
  596. IsmAbortComponentEnum (EnumPtr);
  597. }
  598. return result;
  599. }
  600. BOOL
  601. WINAPI
  602. IsmEnumFirstComponent (
  603. OUT PMIG_COMPONENT_ENUM EnumPtr,
  604. IN DWORD Flags,
  605. IN UINT GroupIdFilter OPTIONAL
  606. )
  607. {
  608. PCOMPONENTENUM_HANDLE handle;
  609. ZeroMemory (EnumPtr, sizeof (MIG_COMPONENT_ENUM));
  610. if (Flags & (COMPONENTENUM_PREFERRED_ONLY|COMPONENTENUM_NON_PREFERRED_ONLY)) {
  611. if (!(Flags & COMPONENTENUM_ALIASES)) {
  612. SetLastError (ERROR_INVALID_PARAMETER);
  613. return FALSE;
  614. }
  615. if ((Flags & (COMPONENTENUM_PREFERRED_ONLY|COMPONENTENUM_NON_PREFERRED_ONLY)) ==
  616. (COMPONENTENUM_PREFERRED_ONLY|COMPONENTENUM_NON_PREFERRED_ONLY)
  617. ) {
  618. SetLastError (ERROR_INVALID_PARAMETER);
  619. return FALSE;
  620. }
  621. }
  622. EnumPtr->Handle = MemAllocUninit (sizeof (COMPONENTENUM_HANDLE));
  623. handle = (PCOMPONENTENUM_HANDLE) EnumPtr->Handle;
  624. handle->EnumAliases = Flags & COMPONENTENUM_ALIASES ? TRUE : FALSE;
  625. handle->EnumEnabled = Flags & COMPONENTENUM_ENABLED ? TRUE : FALSE;
  626. handle->EnumDisabled = Flags & COMPONENTENUM_DISABLED ? TRUE : FALSE;
  627. handle->GroupIdFilter = GroupIdFilter;
  628. handle->EnumPreferredOnly = Flags & COMPONENTENUM_PREFERRED_ONLY;
  629. handle->EnumNonPreferredOnly = Flags & COMPONENTENUM_NON_PREFERRED_ONLY;
  630. handle->State = CES_FIRST_COMPONENT;
  631. return pEnumWorker (EnumPtr, handle);
  632. }
  633. BOOL
  634. WINAPI
  635. IsmEnumNextComponent (
  636. IN OUT PMIG_COMPONENT_ENUM EnumPtr
  637. )
  638. {
  639. PCOMPONENTENUM_HANDLE handle;
  640. handle = (PCOMPONENTENUM_HANDLE) EnumPtr->Handle;
  641. return pEnumWorker (EnumPtr, handle);
  642. }
  643. VOID
  644. WINAPI
  645. IsmAbortComponentEnum (
  646. IN PMIG_COMPONENT_ENUM EnumPtr ZEROED
  647. )
  648. {
  649. PCOMPONENTENUM_HANDLE handle;
  650. handle = (PCOMPONENTENUM_HANDLE) EnumPtr->Handle;
  651. if (handle) {
  652. if (handle->State == CES_NEXT_COMPONENT) {
  653. MemDbAbortEnum (&handle->ComponentEnumStruct);
  654. }
  655. if (handle->EnumAliases) {
  656. if (handle->State == CES_NEXT_ALIAS) {
  657. MemDbAbortEnum (&handle->AliasEnumStruct);
  658. }
  659. }
  660. FreeAlloc (handle);
  661. }
  662. ZeroMemory (EnumPtr, sizeof (MIG_COMPONENT_ENUM));
  663. }
  664. VOID
  665. WINAPI
  666. IsmRemoveAllUserSuppliedComponents (
  667. VOID
  668. )
  669. {
  670. MEMDB_ENUM e;
  671. MULTISZ_ENUM listEnum;
  672. GROWBUFFER list = INIT_GROWBUFFER;
  673. //
  674. // Collect all the components that have user-supplied aliases. Then after
  675. // enum completes, delete them. We don't delete during the enum because it
  676. // is never a good idea to delete the item just enumerated, and then try
  677. // to continue enumerating.
  678. //
  679. if (MemDbEnumFirst (
  680. &e,
  681. S_COMPONENT_ROOT TEXT("\\*\\") S_USER_SUPPLIED TEXT("\\*"),
  682. ENUMFLAG_ALL,
  683. 2,
  684. 2
  685. )) {
  686. do {
  687. GbMultiSzAppend (&list, e.FullKeyName);
  688. } while (MemDbEnumNext (&e));
  689. }
  690. if (EnumFirstMultiSz (&listEnum, (PCTSTR) list.Buf)) {
  691. do {
  692. MemDbDeleteTree (listEnum.CurrentString);
  693. } while (EnumNextMultiSz (&listEnum));
  694. }
  695. GbFree (&list);
  696. }
  697. BOOL
  698. WINAPI
  699. IsmSelectMasterGroup (
  700. IN UINT MasterGroup,
  701. IN BOOL Enable
  702. )
  703. {
  704. MEMDB_ENUM e;
  705. UINT flags;
  706. if (MasterGroup > MASTERGROUP_ALL) {
  707. SetLastError (ERROR_INVALID_PARAMETER);
  708. DEBUGMSG ((DBG_ERROR, "Can't select invalid MasterGroup"));
  709. return FALSE;
  710. }
  711. //
  712. // Enumerate all components and mark them enabled or disabled
  713. // depending on the master group
  714. //
  715. if (MemDbEnumFirst (
  716. &e,
  717. S_COMPONENT_ROOT TEXT("\\*"),
  718. ENUMFLAG_NORMAL,
  719. 1,
  720. 1
  721. )) {
  722. do {
  723. if (MasterGroup == MASTERGROUP_ALL ||
  724. MasterGroup == e.Value
  725. ) {
  726. flags = Enable ? MEMDB_FLAG_SELECTED : 0;
  727. } else {
  728. flags = 0;
  729. }
  730. if (!MemDbSetFlags (e.FullKeyName, flags, MEMDB_FLAG_SELECTED)) {
  731. EngineError ();
  732. MemDbAbortEnum (&e);
  733. return FALSE;
  734. }
  735. } while (MemDbEnumNext (&e));
  736. }
  737. return TRUE;
  738. }
  739. BOOL
  740. WINAPI
  741. IsmIsComponentSelected (
  742. IN PCTSTR ComponentOrAlias,
  743. IN UINT ComponentGroupId OPTIONAL
  744. )
  745. {
  746. UINT flags = 0;
  747. TCHAR memdbNode[MAX_CONTROLLED_NODE_SIZE];
  748. PCTSTR componentNode;
  749. if (!ComponentGroupId) {
  750. if (!pCheckComponentName (ComponentOrAlias)) {
  751. return FALSE;
  752. }
  753. wsprintf (memdbNode, TEXT("%s\\%s"), S_COMPONENT_ROOT, ComponentOrAlias);
  754. MemDbGetFlags (memdbNode, &flags);
  755. } else {
  756. if (pFindComponent (ComponentOrAlias, ComponentGroupId, &componentNode, NULL)) {
  757. MemDbGetFlags (componentNode, &flags);
  758. FreeText (componentNode);
  759. }
  760. }
  761. return (flags & MEMDB_FLAG_SELECTED) != 0;
  762. }