Leaked source code of windows server 2003
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.

2788 lines
89 KiB

  1. /*++
  2. Copyright (c) 1998 Microsoft Corporation
  3. Module Name:
  4. hwprofil.c
  5. Abstract:
  6. This module contains support for changing the Hardware profile
  7. based on the current docking state, either at boot time or by
  8. ACPI dock.
  9. Author:
  10. Kenneth D. Ray (kenray) Jan 1998
  11. Revision History:
  12. --*/
  13. #include "cmp.h"
  14. NTSTATUS
  15. CmDeleteKeyRecursive(
  16. HANDLE hKeyRoot,
  17. PWSTR Key,
  18. PVOID TemporaryBuffer,
  19. ULONG LengthTemporaryBuffer,
  20. BOOLEAN ThisKeyToo
  21. );
  22. NTSTATUS
  23. CmpGetAcpiProfileInformation (
  24. IN HANDLE IDConfigDB,
  25. OUT PCM_HARDWARE_PROFILE_LIST * ProfileList,
  26. OUT PCM_HARDWARE_PROFILE_ACPI_ALIAS_LIST * AliasList,
  27. IN PWCHAR NameBuffer,
  28. IN PUCHAR ValueBuffer,
  29. IN ULONG Len
  30. );
  31. NTSTATUS
  32. CmpFilterAcpiDockingState (
  33. IN PPROFILE_ACPI_DOCKING_STATE NewDockState,
  34. IN ULONG CurrentDockingState,
  35. IN PWCHAR CurrentAcpiSN,
  36. IN ULONG CurrentProfileNumber,
  37. IN OUT PCM_HARDWARE_PROFILE_LIST ProfileList,
  38. IN OUT PCM_HARDWARE_PROFILE_ACPI_ALIAS_LIST AliasList
  39. );
  40. NTSTATUS
  41. CmpMoveBiosAliasTable (
  42. IN HANDLE IDConfigDB,
  43. IN HANDLE CurrentInfo,
  44. IN ULONG CurrentProfileNumber,
  45. IN ULONG NewProfileNumber,
  46. IN PWCHAR nameBuffer,
  47. IN PCHAR valueBuffer,
  48. IN ULONG bufferLen
  49. );
  50. #pragma alloc_text(PAGE,CmDeleteKeyRecursive)
  51. #pragma alloc_text(PAGE,CmpCloneHwProfile)
  52. #pragma alloc_text(PAGE,CmSetAcpiHwProfile)
  53. #pragma alloc_text(PAGE,CmpFilterAcpiDockingState)
  54. #pragma alloc_text(PAGE,CmpGetAcpiProfileInformation)
  55. #pragma alloc_text(PAGE,CmpAddAcpiAliasEntry)
  56. #pragma alloc_text(PAGE,CmpMoveBiosAliasTable)
  57. #pragma alloc_text(PAGE,CmpCreateHwProfileFriendlyName)
  58. extern UNICODE_STRING CmSymbolicLinkValueName;
  59. NTSTATUS
  60. CmpGetAcpiProfileInformation (
  61. IN HANDLE IDConfigDB,
  62. OUT PCM_HARDWARE_PROFILE_LIST * ProfileList,
  63. OUT PCM_HARDWARE_PROFILE_ACPI_ALIAS_LIST * AliasList,
  64. IN PWCHAR nameBuffer,
  65. IN PUCHAR valueBuffer,
  66. IN ULONG bufferLen
  67. )
  68. /*++
  69. Routine Description:
  70. Obtain the alias and hardware profile information from the registry.
  71. --*/
  72. {
  73. NTSTATUS status = STATUS_SUCCESS;
  74. HANDLE acpiAlias = NULL;
  75. HANDLE profiles = NULL;
  76. HANDLE entry = NULL;
  77. ULONG len = 0;
  78. ULONG i, j;
  79. OBJECT_ATTRIBUTES attributes;
  80. UNICODE_STRING name;
  81. KEY_FULL_INFORMATION keyInfo;
  82. PKEY_VALUE_FULL_INFORMATION value;
  83. PKEY_BASIC_INFORMATION basicInfo;
  84. PAGED_CODE ();
  85. UNREFERENCED_PARAMETER (nameBuffer);
  86. *ProfileList = NULL;
  87. *AliasList = NULL;
  88. value = (PKEY_VALUE_FULL_INFORMATION) valueBuffer;
  89. basicInfo = (PKEY_BASIC_INFORMATION) valueBuffer;
  90. //
  91. // Open a handle to the Profile information
  92. //
  93. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_HARDWARE_PROFILES);
  94. InitializeObjectAttributes (&attributes,
  95. &name,
  96. OBJ_CASE_INSENSITIVE,
  97. IDConfigDB,
  98. NULL);
  99. status = ZwOpenKey (&profiles,
  100. KEY_READ,
  101. &attributes);
  102. if (!NT_SUCCESS (status)) {
  103. profiles = NULL;
  104. goto Clean;
  105. }
  106. //
  107. // Find the number of profile Sub Keys
  108. //
  109. status = ZwQueryKey (profiles,
  110. KeyFullInformation,
  111. &keyInfo,
  112. sizeof (keyInfo),
  113. &len);
  114. if (!NT_SUCCESS (status)) {
  115. goto Clean;
  116. }
  117. ASSERT (0 < keyInfo.SubKeys);
  118. len = sizeof (CM_HARDWARE_PROFILE_LIST)
  119. + (sizeof (CM_HARDWARE_PROFILE) * (keyInfo.SubKeys - 1));
  120. * ProfileList = ExAllocatePool (PagedPool, len);
  121. if (NULL == *ProfileList) {
  122. status = STATUS_INSUFFICIENT_RESOURCES;
  123. goto Clean;
  124. }
  125. RtlZeroMemory (*ProfileList, len);
  126. (*ProfileList)->MaxProfileCount = keyInfo.SubKeys;
  127. (*ProfileList)->CurrentProfileCount = 0;
  128. //
  129. // Iterrate the profiles
  130. //
  131. for (i = 0; i < keyInfo.SubKeys; i++) {
  132. CM_HARDWARE_PROFILE TempProfile;
  133. //
  134. // Get the first key in the list.
  135. //
  136. status = ZwEnumerateKey (profiles,
  137. i,
  138. KeyBasicInformation,
  139. basicInfo,
  140. bufferLen - sizeof (UNICODE_NULL), // term 0
  141. &len);
  142. if (!NT_SUCCESS (status)) {
  143. //
  144. // This should never happen.
  145. //
  146. break;
  147. }
  148. basicInfo->Name [basicInfo->NameLength/sizeof(WCHAR)] = 0;
  149. name.Length = (USHORT) basicInfo->NameLength;
  150. name.MaximumLength = (USHORT) basicInfo->NameLength + sizeof (UNICODE_NULL);
  151. name.Buffer = basicInfo->Name;
  152. InitializeObjectAttributes (&attributes,
  153. &name,
  154. OBJ_CASE_INSENSITIVE,
  155. profiles,
  156. NULL);
  157. status = ZwOpenKey (&entry,
  158. KEY_READ,
  159. &attributes);
  160. if (!NT_SUCCESS (status)) {
  161. break;
  162. }
  163. //
  164. // Fill in the temporary profile structure with this
  165. // profile's data.
  166. //
  167. RtlUnicodeStringToInteger(&name, 0, &TempProfile.Id);
  168. //
  169. // Find the pref order of this entry.
  170. //
  171. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PREFERENCE_ORDER);
  172. status = NtQueryValueKey(entry,
  173. &name,
  174. KeyValueFullInformation,
  175. valueBuffer,
  176. bufferLen,
  177. &len);
  178. if ((!NT_SUCCESS (status)) || (value->Type != REG_DWORD)) {
  179. TempProfile.PreferenceOrder = (ULONG)-1;
  180. } else {
  181. TempProfile.PreferenceOrder
  182. = * (PULONG) ((PUCHAR) value + value->DataOffset);
  183. }
  184. //
  185. // Extract the friendly name
  186. //
  187. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_FRIENDLY_NAME);
  188. status = NtQueryValueKey(entry,
  189. &name,
  190. KeyValueFullInformation,
  191. valueBuffer,
  192. bufferLen,
  193. &len);
  194. if (!NT_SUCCESS (status) || (value->Type != REG_SZ)) {
  195. WCHAR tmpname[] = L"-------"; // as taken from cmboot.c
  196. ULONG length;
  197. PVOID buffer;
  198. length = sizeof (tmpname);
  199. buffer = ExAllocatePool (PagedPool, length);
  200. TempProfile.NameLength = length;
  201. TempProfile.FriendlyName = buffer;
  202. if (NULL == buffer) {
  203. status = STATUS_INSUFFICIENT_RESOURCES;
  204. ZwClose (entry);
  205. goto Clean;
  206. }
  207. RtlCopyMemory (buffer, tmpname, value->DataLength);
  208. } else {
  209. PVOID buffer;
  210. buffer = ExAllocatePool (PagedPool, value->DataLength);
  211. TempProfile.NameLength = value->DataLength;
  212. TempProfile.FriendlyName = buffer;
  213. if (NULL == buffer) {
  214. status = STATUS_INSUFFICIENT_RESOURCES;
  215. ZwClose (entry);
  216. goto Clean;
  217. }
  218. RtlCopyMemory (buffer,
  219. (PUCHAR) value + value->DataOffset,
  220. value->DataLength);
  221. }
  222. TempProfile.Flags = 0;
  223. //
  224. // Is this aliasable?
  225. //
  226. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ALIASABLE);
  227. status = NtQueryValueKey(entry,
  228. &name,
  229. KeyValueFullInformation,
  230. valueBuffer,
  231. bufferLen,
  232. &len);
  233. if (NT_SUCCESS (status) && (value->Type == REG_DWORD)) {
  234. if (* (PULONG) ((PUCHAR) value + value->DataOffset)) {
  235. TempProfile.Flags |= CM_HP_FLAGS_ALIASABLE;
  236. }
  237. } else {
  238. TempProfile.Flags |= CM_HP_FLAGS_ALIASABLE;
  239. }
  240. //
  241. // Is this pristine?
  242. //
  243. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PRISTINE);
  244. status = NtQueryValueKey(entry,
  245. &name,
  246. KeyValueFullInformation,
  247. valueBuffer,
  248. bufferLen,
  249. &len);
  250. if (NT_SUCCESS (status) && (value->Type == REG_DWORD)) {
  251. if (* (PULONG) ((PUCHAR) value + value->DataOffset)) {
  252. TempProfile.Flags = CM_HP_FLAGS_PRISTINE;
  253. // No other flags set;
  254. }
  255. }
  256. //
  257. // If we see a profile with the ID of zero (AKA an illegal)
  258. // ID for a hardware profile to possess, then we know that this
  259. // must be a pristine profile.
  260. //
  261. if (0 == TempProfile.Id) {
  262. TempProfile.Flags = CM_HP_FLAGS_PRISTINE;
  263. // NO other flags set.
  264. TempProfile.PreferenceOrder = (ULONG)-1; // move to the end of the list.
  265. }
  266. //
  267. // Insert this new profile into the appropriate spot in the
  268. // profile array. Entries are sorted by preference order.
  269. //
  270. for (j=0; j < (*ProfileList)->CurrentProfileCount; j++) {
  271. if ((*ProfileList)->Profile[j].PreferenceOrder >=
  272. TempProfile.PreferenceOrder) {
  273. //
  274. // Insert at position j.
  275. //
  276. RtlMoveMemory(&(*ProfileList)->Profile[j+1],
  277. &(*ProfileList)->Profile[j],
  278. sizeof(CM_HARDWARE_PROFILE) *
  279. ((*ProfileList)->MaxProfileCount-j-1));
  280. break;
  281. }
  282. }
  283. (*ProfileList)->Profile[j] = TempProfile;
  284. ++(*ProfileList)->CurrentProfileCount;
  285. ZwClose (entry);
  286. }
  287. //
  288. // Open a handle to the ACPI Alias information
  289. //
  290. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ACPI_ALIAS);
  291. InitializeObjectAttributes (&attributes,
  292. &name,
  293. OBJ_CASE_INSENSITIVE,
  294. IDConfigDB,
  295. NULL);
  296. status = ZwOpenKey (&acpiAlias,
  297. KEY_READ,
  298. &attributes);
  299. if (!NT_SUCCESS (status)) {
  300. //
  301. // So we don't have an alias table. This is ok.
  302. //
  303. status = STATUS_SUCCESS;
  304. acpiAlias = NULL;
  305. goto Clean;
  306. }
  307. //
  308. // Find the number of Acpi Alias Sub Keys
  309. //
  310. status = ZwQueryKey (acpiAlias,
  311. KeyFullInformation,
  312. &keyInfo,
  313. sizeof (keyInfo),
  314. &len);
  315. if (!NT_SUCCESS (status)) {
  316. goto Clean;
  317. }
  318. ASSERT (0 < keyInfo.SubKeys);
  319. * AliasList = ExAllocatePool (
  320. PagedPool,
  321. sizeof (CM_HARDWARE_PROFILE_LIST) +
  322. (sizeof (CM_HARDWARE_PROFILE) * (keyInfo.SubKeys - 1)));
  323. if (NULL == *AliasList) {
  324. status = STATUS_INSUFFICIENT_RESOURCES;
  325. goto Clean;
  326. }
  327. (*AliasList)->MaxAliasCount =
  328. (*AliasList)->CurrentAliasCount = keyInfo.SubKeys;
  329. //
  330. // Iterrate the alias entries
  331. //
  332. for (i = 0; i < keyInfo.SubKeys; i++) {
  333. //
  334. // Get the first key in the list.
  335. //
  336. status = ZwEnumerateKey (acpiAlias,
  337. i,
  338. KeyBasicInformation,
  339. basicInfo,
  340. bufferLen - sizeof (UNICODE_NULL), // term 0
  341. &len);
  342. if (!NT_SUCCESS (status)) {
  343. //
  344. // This should never happen.
  345. //
  346. break;
  347. }
  348. basicInfo->Name [basicInfo->NameLength/sizeof(WCHAR)] = 0;
  349. name.Length = (USHORT) basicInfo->NameLength;
  350. name.MaximumLength = (USHORT) basicInfo->NameLength + sizeof (UNICODE_NULL);
  351. name.Buffer = basicInfo->Name;
  352. InitializeObjectAttributes (&attributes,
  353. &name,
  354. OBJ_CASE_INSENSITIVE,
  355. acpiAlias,
  356. NULL);
  357. status = ZwOpenKey (&entry,
  358. KEY_READ,
  359. &attributes);
  360. if (!NT_SUCCESS (status)) {
  361. break;
  362. }
  363. //
  364. // Extract The Profile number to which this alias refers.
  365. //
  366. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PROFILE_NUMBER);
  367. status = NtQueryValueKey(entry,
  368. &name,
  369. KeyValueFullInformation,
  370. valueBuffer,
  371. bufferLen,
  372. &len);
  373. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  374. status = STATUS_REGISTRY_CORRUPT;
  375. ZwClose (entry);
  376. goto Clean;
  377. }
  378. (*AliasList)->Alias[i].ProfileNumber =
  379. * (PULONG) ((PUCHAR) value + value->DataOffset);
  380. //
  381. // Extract The Docking State.
  382. //
  383. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKING_STATE);
  384. status = NtQueryValueKey(entry,
  385. &name,
  386. KeyValueFullInformation,
  387. valueBuffer,
  388. bufferLen,
  389. &len);
  390. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  391. status = STATUS_REGISTRY_CORRUPT;
  392. ZwClose (entry);
  393. goto Clean;
  394. }
  395. (*AliasList)->Alias[i].DockState =
  396. * (PULONG) ((PUCHAR) value + value->DataOffset);
  397. //
  398. // Find the SerialNumber
  399. //
  400. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ACPI_SERIAL_NUMBER);
  401. status = NtQueryValueKey(entry,
  402. &name,
  403. KeyValueFullInformation,
  404. valueBuffer,
  405. bufferLen,
  406. &len);
  407. if (!NT_SUCCESS (status) || (value->Type != REG_BINARY)) {
  408. status = STATUS_REGISTRY_CORRUPT;
  409. ZwClose (entry);
  410. goto Clean;
  411. }
  412. (*AliasList)->Alias[i].SerialLength = value->DataLength;
  413. (*AliasList)->Alias[i].SerialNumber =
  414. (value->DataLength) ?
  415. ExAllocatePool (PagedPool, value->DataLength) :
  416. 0;
  417. if (value->DataLength && (NULL == (*AliasList)->Alias[i].SerialNumber)) {
  418. status = STATUS_INSUFFICIENT_RESOURCES;
  419. ZwClose (entry);
  420. goto Clean;
  421. }
  422. if (value->DataLength) {
  423. RtlCopyMemory ((*AliasList)->Alias[i].SerialNumber,
  424. (PUCHAR) value + value->DataOffset,
  425. value->DataLength);
  426. }
  427. ZwClose (entry);
  428. }
  429. Clean:
  430. if (NULL != acpiAlias) {
  431. NtClose (acpiAlias);
  432. }
  433. if (NULL != profiles) {
  434. NtClose (profiles);
  435. }
  436. if (!NT_SUCCESS (status)) {
  437. if (NULL != *ProfileList) {
  438. for (i = 0; i < (*ProfileList)->CurrentProfileCount; i++) {
  439. if ((*ProfileList)->Profile[i].FriendlyName) {
  440. ExFreePool ((*ProfileList)->Profile[i].FriendlyName);
  441. }
  442. }
  443. ExFreePool (*ProfileList);
  444. *ProfileList = 0;
  445. }
  446. if (NULL != *AliasList) {
  447. for (i = 0; i < (*AliasList)->CurrentAliasCount; i++) {
  448. if ((*AliasList)->Alias[i].SerialNumber) {
  449. ExFreePool ((*AliasList)->Alias[i].SerialNumber);
  450. }
  451. }
  452. ExFreePool (*AliasList);
  453. *AliasList = 0;
  454. }
  455. }
  456. return status;
  457. }
  458. NTSTATUS
  459. CmpAddAcpiAliasEntry (
  460. IN HANDLE IDConfigDB,
  461. IN PPROFILE_ACPI_DOCKING_STATE NewDockState,
  462. IN ULONG ProfileNumber,
  463. IN PWCHAR nameBuffer,
  464. IN PVOID valueBuffer,
  465. IN ULONG valueBufferLength,
  466. IN BOOLEAN PreventDuplication
  467. )
  468. /*++
  469. Routine Description:
  470. Set the Acpi Alais entry.
  471. Routine Description:
  472. Create an alias entry in the IDConfigDB database for the given
  473. hardware profile.
  474. Create the "AcpiAlias" key if it does not exist.
  475. Parameters:
  476. IDConfigDB - Pointer to "..\CurrentControlSet\Control\IDConfigDB"
  477. NewDockState - The new docking state for which this alias points.
  478. ProfileNumber -The profile number to which this alias points.
  479. nameBuffer - a temp scratch space for writing things.
  480. (assumed to be at least 128 WCHARS)
  481. --*/
  482. {
  483. OBJECT_ATTRIBUTES attributes;
  484. NTSTATUS status = STATUS_SUCCESS;
  485. UNICODE_STRING name;
  486. HANDLE aliasKey = NULL;
  487. HANDLE aliasEntry = NULL;
  488. ULONG value;
  489. ULONG disposition;
  490. ULONG aliasNumber = 0;
  491. ULONG len;
  492. PKEY_VALUE_FULL_INFORMATION keyInfo;
  493. PAGED_CODE ();
  494. keyInfo = (PKEY_VALUE_FULL_INFORMATION) valueBuffer;
  495. //
  496. // Find the Alias Key or Create it if it does not already exist.
  497. //
  498. RtlInitUnicodeString (&name,CM_HARDWARE_PROFILE_STR_ACPI_ALIAS);
  499. InitializeObjectAttributes (&attributes,
  500. &name,
  501. OBJ_CASE_INSENSITIVE,
  502. IDConfigDB,
  503. NULL);
  504. status = NtOpenKey (&aliasKey,
  505. KEY_READ | KEY_WRITE,
  506. &attributes);
  507. if (STATUS_OBJECT_NAME_NOT_FOUND == status) {
  508. status = NtCreateKey (&aliasKey,
  509. KEY_READ | KEY_WRITE,
  510. &attributes,
  511. 0, // no title
  512. NULL, // no class
  513. 0, // no options
  514. &disposition);
  515. }
  516. if (!NT_SUCCESS (status)) {
  517. aliasKey = NULL;
  518. goto Exit;
  519. }
  520. //
  521. // Create an entry key
  522. //
  523. while (aliasNumber < 200) {
  524. aliasNumber++;
  525. swprintf (nameBuffer, L"%04d", aliasNumber);
  526. RtlInitUnicodeString (&name, nameBuffer);
  527. InitializeObjectAttributes(&attributes,
  528. &name,
  529. OBJ_CASE_INSENSITIVE,
  530. aliasKey,
  531. NULL);
  532. status = NtOpenKey (&aliasEntry,
  533. KEY_READ | KEY_WRITE,
  534. &attributes);
  535. if (NT_SUCCESS (status)) {
  536. if (PreventDuplication) {
  537. //
  538. // If we have a matching DockingState, SerialNumber, and
  539. // Profile Number, then we should not make this alias
  540. //
  541. //
  542. // Extract The DockingState to which this alias refers.
  543. //
  544. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKING_STATE);
  545. status = NtQueryValueKey(aliasEntry,
  546. &name,
  547. KeyValueFullInformation,
  548. valueBuffer,
  549. valueBufferLength,
  550. &len);
  551. if (!NT_SUCCESS (status) || (keyInfo->Type != REG_DWORD)) {
  552. status = STATUS_REGISTRY_CORRUPT;
  553. goto Exit;
  554. }
  555. if (NewDockState->DockingState !=
  556. * (PULONG) ((PUCHAR) keyInfo + keyInfo->DataOffset)) {
  557. //
  558. // Not a dupe
  559. //
  560. NtClose (aliasEntry);
  561. aliasEntry = NULL;
  562. continue;
  563. }
  564. //
  565. // Extract the SerialNumber
  566. //
  567. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ACPI_SERIAL_NUMBER);
  568. status = NtQueryValueKey(aliasEntry,
  569. &name,
  570. KeyValueFullInformation,
  571. valueBuffer,
  572. valueBufferLength,
  573. &len);
  574. if (!NT_SUCCESS (status) || (keyInfo->Type != REG_BINARY)) {
  575. status = STATUS_REGISTRY_CORRUPT;
  576. goto Exit;
  577. }
  578. if (NewDockState->SerialLength != keyInfo->DataLength) {
  579. //
  580. // Not a dupe
  581. //
  582. NtClose (aliasEntry);
  583. aliasEntry = NULL;
  584. continue;
  585. }
  586. if (!RtlEqualMemory (NewDockState->SerialNumber,
  587. ((PUCHAR) keyInfo + keyInfo->DataOffset),
  588. NewDockState->SerialLength)) {
  589. //
  590. // Not a dupe
  591. //
  592. NtClose (aliasEntry);
  593. aliasEntry = NULL;
  594. continue;
  595. }
  596. status = STATUS_SUCCESS;
  597. goto Exit;
  598. }
  599. } else if (STATUS_OBJECT_NAME_NOT_FOUND == status) {
  600. status = STATUS_SUCCESS;
  601. break;
  602. } else {
  603. break;
  604. }
  605. }
  606. if (!NT_SUCCESS (status)) {
  607. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: cmpCreateAcpiAliasEntry error finding new set %08lx\n",status));
  608. aliasEntry = 0;
  609. goto Exit;
  610. }
  611. status = NtCreateKey (&aliasEntry,
  612. KEY_READ | KEY_WRITE,
  613. &attributes,
  614. 0,
  615. NULL,
  616. 0,
  617. &disposition);
  618. if (!NT_SUCCESS (status)) {
  619. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: cmpCreateAcpiAliasEntry error creating new set %08lx\n",status));
  620. aliasEntry = 0;
  621. goto Exit;
  622. }
  623. //
  624. // Write the Docking State;
  625. //
  626. value = NewDockState->DockingState;
  627. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKING_STATE);
  628. status = NtSetValueKey (aliasEntry,
  629. &name,
  630. 0,
  631. REG_DWORD,
  632. &value,
  633. sizeof (value));
  634. //
  635. // Write the Serial Number
  636. //
  637. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ACPI_SERIAL_NUMBER);
  638. status = NtSetValueKey (aliasEntry,
  639. &name,
  640. 0,
  641. REG_BINARY,
  642. NewDockState->SerialNumber,
  643. NewDockState->SerialLength);
  644. //
  645. // Write the Profile Number
  646. //
  647. value = ProfileNumber;
  648. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PROFILE_NUMBER);
  649. status = NtSetValueKey (aliasEntry,
  650. &name,
  651. 0,
  652. REG_DWORD,
  653. &value,
  654. sizeof (value));
  655. Exit:
  656. if (aliasKey) {
  657. NtClose (aliasKey);
  658. }
  659. if (aliasEntry) {
  660. NtClose (aliasEntry);
  661. }
  662. return status;
  663. }
  664. NTSTATUS
  665. CmSetAcpiHwProfile (
  666. IN PPROFILE_ACPI_DOCKING_STATE NewDockState,
  667. IN PCM_ACPI_SELECTION_ROUTINE Select,
  668. IN PVOID Context,
  669. OUT PHANDLE NewProfile,
  670. OUT PBOOLEAN ProfileChanged
  671. )
  672. /*++
  673. Routine Description:
  674. The ACPI docking state of the machine has changed.
  675. Based on the new change calculate the new HW Profile(s) consitent with the
  676. new ACPI docking state.
  677. Pass the list of known profiles to the callers selection routine.
  678. Set the new current profile.
  679. Patch up any ACPI alias entries if a new profile for this ACPI state has
  680. been used.
  681. Arguments:
  682. NewDockStateArray - The list of possible Docking States that we might enter.
  683. Select - Call back to select which profile to enter, given the list of
  684. possible profiles.
  685. --*/
  686. {
  687. NTSTATUS status = STATUS_SUCCESS;
  688. HANDLE IDConfigDB = NULL;
  689. HANDLE HardwareProfile = NULL;
  690. HANDLE currentInfo = NULL;
  691. HANDLE currentSymLink = NULL;
  692. HANDLE parent = NULL;
  693. WCHAR nameBuffer[128];
  694. UNICODE_STRING name;
  695. CHAR valueBuffer[256];
  696. ULONG len;
  697. ULONG i;
  698. ULONG selectedElement;
  699. ULONG profileNum;
  700. ULONG currentDockingState;
  701. ULONG currentProfileNumber;
  702. ULONG disposition;
  703. ULONG flags;
  704. PWCHAR currentAcpiSN = NULL;
  705. PCM_HARDWARE_PROFILE_ACPI_ALIAS_LIST AliasList = NULL;
  706. PCM_HARDWARE_PROFILE_LIST ProfileList = NULL;
  707. PKEY_VALUE_FULL_INFORMATION value;
  708. OBJECT_ATTRIBUTES attributes;
  709. PAGED_CODE ();
  710. *ProfileChanged = FALSE;
  711. value = (PKEY_VALUE_FULL_INFORMATION) valueBuffer;
  712. //
  713. // Open The Hardware Profile Database
  714. //
  715. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DATABASE);
  716. InitializeObjectAttributes (&attributes,
  717. &name,
  718. OBJ_CASE_INSENSITIVE,
  719. NULL,
  720. NULL);
  721. status = ZwOpenKey (&IDConfigDB,
  722. KEY_READ,
  723. &attributes);
  724. if (!NT_SUCCESS (status)) {
  725. IDConfigDB = NULL;
  726. goto Clean;
  727. }
  728. //
  729. // Obtain the total list of profiles
  730. //
  731. status = CmpGetAcpiProfileInformation (IDConfigDB,
  732. &ProfileList,
  733. &AliasList,
  734. nameBuffer,
  735. (PUCHAR) valueBuffer,
  736. sizeof (valueBuffer));
  737. if (!NT_SUCCESS (status)) {
  738. goto Clean;
  739. }
  740. //
  741. // Determine the current Dock information.
  742. //
  743. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CCS_CURRENT);
  744. InitializeObjectAttributes (&attributes,
  745. &name,
  746. OBJ_CASE_INSENSITIVE,
  747. NULL,
  748. NULL);
  749. status = ZwOpenKey (&HardwareProfile,
  750. KEY_READ,
  751. &attributes);
  752. if (!NT_SUCCESS (status)) {
  753. HardwareProfile = NULL;
  754. goto Clean;
  755. }
  756. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CURRENT_DOCK_INFO);
  757. InitializeObjectAttributes (&attributes,
  758. &name,
  759. OBJ_CASE_INSENSITIVE,
  760. IDConfigDB,
  761. NULL);
  762. status = ZwOpenKey (&currentInfo,
  763. KEY_READ,
  764. &attributes);
  765. if (!NT_SUCCESS (status)) {
  766. currentInfo = NULL;
  767. goto Clean;
  768. }
  769. //
  770. // The current Docking State
  771. //
  772. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKING_STATE);
  773. status = NtQueryValueKey (currentInfo,
  774. &name,
  775. KeyValueFullInformation,
  776. valueBuffer,
  777. sizeof (valueBuffer),
  778. &len);
  779. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  780. status = STATUS_REGISTRY_CORRUPT;
  781. goto Clean;
  782. }
  783. currentDockingState = * (PULONG) ((PUCHAR) value + value->DataOffset);
  784. //
  785. // The current ACPI Serial Number
  786. //
  787. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ACPI_SERIAL_NUMBER);
  788. status = NtQueryValueKey(currentInfo,
  789. &name,
  790. KeyValueFullInformation,
  791. valueBuffer,
  792. sizeof (valueBuffer),
  793. &len);
  794. if (NT_SUCCESS (status) && (value->Type == REG_BINARY)) {
  795. currentAcpiSN = ExAllocatePool (PagedPool, value->DataLength);
  796. if (NULL == currentAcpiSN) {
  797. status = STATUS_INSUFFICIENT_RESOURCES;
  798. goto Clean;
  799. }
  800. RtlCopyMemory (currentAcpiSN,
  801. (PUCHAR) value + value->DataOffset,
  802. value->DataLength);
  803. } else {
  804. currentAcpiSN = 0;
  805. }
  806. //
  807. // The current Profile Number
  808. //
  809. RtlInitUnicodeString(&name, L"CurrentConfig");
  810. status = NtQueryValueKey(IDConfigDB,
  811. &name,
  812. KeyValueFullInformation,
  813. valueBuffer,
  814. sizeof (valueBuffer),
  815. &len);
  816. if (!NT_SUCCESS(status) || (value->Type != REG_DWORD)) {
  817. status = STATUS_REGISTRY_CORRUPT;
  818. goto Clean;
  819. }
  820. currentProfileNumber = *(PULONG)((PUCHAR)value + value->DataOffset);
  821. //
  822. // Filter the current list of hardware profiles based on the current
  823. // docking state, the new acpi state, and the acpi alias tables
  824. //
  825. status = CmpFilterAcpiDockingState (NewDockState,
  826. currentDockingState,
  827. currentAcpiSN,
  828. currentProfileNumber,
  829. ProfileList,
  830. AliasList);
  831. if (!NT_SUCCESS (status)) {
  832. goto Clean;
  833. }
  834. //
  835. // Allow the caller a chance to select from the filtered list.
  836. //
  837. status = Select (ProfileList, &selectedElement, Context);
  838. //
  839. // If the user selected -1 then he is not interested in selecting any of
  840. // the profiles.
  841. //
  842. if (-1 == selectedElement) {
  843. ASSERT (STATUS_MORE_PROCESSING_REQUIRED == status);
  844. goto Clean;
  845. }
  846. if (!NT_SUCCESS (status)) {
  847. goto Clean;
  848. }
  849. //
  850. // Fine! We have finally made the new selection.
  851. // Set it.
  852. //
  853. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CCS_HWPROFILE);
  854. InitializeObjectAttributes (&attributes,
  855. &name,
  856. OBJ_CASE_INSENSITIVE,
  857. NULL,
  858. NULL);
  859. status = ZwOpenKey (&parent, KEY_READ, &attributes);
  860. if (!NT_SUCCESS (status)) {
  861. parent = NULL;
  862. goto Clean;
  863. }
  864. //
  865. // How did we get here?
  866. //
  867. flags = ProfileList->Profile[selectedElement].Flags;
  868. profileNum = ProfileList->Profile[selectedElement].Id;
  869. //
  870. // Check for duplicate
  871. //
  872. if (flags & CM_HP_FLAGS_DUPLICATE) {
  873. //
  874. // If there is a duplicate then we need to adjust the pnp
  875. // bios alias table.
  876. //
  877. // This happens if we booted PnP bios detected docked, and then
  878. // we received a set state for ACPI as docked, then we have the
  879. // potential for duplicates. See Comment in CmpFilterAcpiDockingState
  880. // for details.
  881. //
  882. // We need to find any pnp bios alias entries that match the current
  883. // state and point them to the duplicate entry.
  884. //
  885. ASSERT (flags & CM_HP_FLAGS_TRUE_MATCH);
  886. ASSERT (!(flags & CM_HP_FLAGS_PRISTINE));
  887. status = CmpMoveBiosAliasTable (IDConfigDB,
  888. currentInfo,
  889. currentProfileNumber,
  890. profileNum,
  891. nameBuffer,
  892. valueBuffer,
  893. sizeof (valueBuffer));
  894. if (!NT_SUCCESS (status)) {
  895. goto Clean;
  896. }
  897. }
  898. if ((flags & CM_HP_FLAGS_PRISTINE) || (profileNum != currentProfileNumber)){
  899. //
  900. // The profile Number Changed or will change.
  901. //
  902. *ProfileChanged = TRUE;
  903. ASSERT (currentInfo);
  904. ZwClose (currentInfo);
  905. currentInfo = NULL;
  906. if (flags & CM_HP_FLAGS_PRISTINE) {
  907. //
  908. // If the selected profile is pristine then we need to clone.
  909. //
  910. ASSERT (!(flags & CM_HP_FLAGS_TRUE_MATCH));
  911. status = CmpCloneHwProfile (IDConfigDB,
  912. parent,
  913. HardwareProfile,
  914. profileNum,
  915. NewDockState->DockingState,
  916. &HardwareProfile,
  917. &profileNum);
  918. if (!NT_SUCCESS (status)) {
  919. HardwareProfile = 0;
  920. goto Clean;
  921. }
  922. } else {
  923. ASSERT (HardwareProfile);
  924. ZwClose (HardwareProfile);
  925. //
  926. // Open the new profile
  927. //
  928. swprintf (nameBuffer, L"%04d\0", profileNum);
  929. RtlInitUnicodeString (&name, nameBuffer);
  930. InitializeObjectAttributes (&attributes,
  931. &name,
  932. OBJ_CASE_INSENSITIVE,
  933. parent,
  934. NULL);
  935. status = ZwOpenKey (&HardwareProfile, KEY_READ, &attributes);
  936. if (!NT_SUCCESS (status)) {
  937. HardwareProfile = NULL;
  938. goto Clean;
  939. }
  940. }
  941. ASSERT (currentProfileNumber != profileNum);
  942. //
  943. // Open the current info for the profile.
  944. //
  945. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CURRENT_DOCK_INFO);
  946. InitializeObjectAttributes (&attributes,
  947. &name,
  948. OBJ_CASE_INSENSITIVE,
  949. IDConfigDB,
  950. NULL);
  951. status = NtCreateKey (&currentInfo,
  952. KEY_READ | KEY_WRITE,
  953. &attributes,
  954. 0,
  955. NULL,
  956. REG_OPTION_VOLATILE,
  957. &disposition);
  958. if (!NT_SUCCESS (status)) {
  959. currentInfo = NULL;
  960. goto Clean;
  961. }
  962. //
  963. // Set CurrentConfig in the Database
  964. //
  965. RtlInitUnicodeString(&name, L"CurrentConfig");
  966. status = NtSetValueKey(IDConfigDB,
  967. &name,
  968. 0,
  969. REG_DWORD,
  970. &profileNum,
  971. sizeof (profileNum));
  972. if (!NT_SUCCESS(status)) {
  973. status = STATUS_REGISTRY_CORRUPT;
  974. goto Clean;
  975. }
  976. }
  977. //
  978. // Write the new Docking State to the current Info key
  979. //
  980. i = NewDockState->DockingState;
  981. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKING_STATE);
  982. status = ZwSetValueKey (currentInfo,
  983. &name,
  984. 0,
  985. REG_DWORD,
  986. &i,
  987. sizeof (ULONG));
  988. //
  989. // Write the new ACPI information to the current Info key
  990. //
  991. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ACPI_SERIAL_NUMBER);
  992. status = ZwSetValueKey (currentInfo,
  993. &name,
  994. 0,
  995. REG_BINARY,
  996. NewDockState->SerialNumber,
  997. NewDockState->SerialLength);
  998. if (!(flags & CM_HP_FLAGS_TRUE_MATCH)) {
  999. //
  1000. // Add the alias entry for this profile.
  1001. //
  1002. status = CmpAddAcpiAliasEntry (IDConfigDB,
  1003. NewDockState,
  1004. profileNum,
  1005. nameBuffer,
  1006. valueBuffer,
  1007. sizeof (valueBuffer),
  1008. FALSE); // Don't Prevent Duplication
  1009. }
  1010. if (profileNum != currentProfileNumber) {
  1011. //
  1012. // Move the symbolic link.
  1013. //
  1014. RtlInitUnicodeString(&name, CM_HARDWARE_PROFILE_STR_CCS_CURRENT);
  1015. InitializeObjectAttributes(&attributes,
  1016. &name,
  1017. OBJ_CASE_INSENSITIVE | OBJ_OPENLINK,
  1018. NULL,
  1019. NULL);
  1020. status = NtCreateKey(&currentSymLink,
  1021. KEY_CREATE_LINK,
  1022. &attributes,
  1023. 0,
  1024. NULL,
  1025. REG_OPTION_OPEN_LINK,
  1026. &disposition);
  1027. ASSERT (STATUS_SUCCESS == status);
  1028. ASSERT (REG_OPENED_EXISTING_KEY == disposition);
  1029. swprintf (nameBuffer,
  1030. L"\\Registry\\Machine\\System\\CurrentControlSet\\Hardware Profiles\\%04d",
  1031. profileNum);
  1032. RtlInitUnicodeString (&name, nameBuffer);
  1033. status = NtSetValueKey (currentSymLink,
  1034. &CmSymbolicLinkValueName,
  1035. 0,
  1036. REG_LINK,
  1037. name.Buffer,
  1038. name.Length);
  1039. ASSERT (STATUS_SUCCESS == status);
  1040. }
  1041. Clean:
  1042. if (NT_SUCCESS (status)) {
  1043. // NB more process required is not a success code.
  1044. *NewProfile = HardwareProfile;
  1045. } else if (NULL != HardwareProfile) {
  1046. ZwClose (HardwareProfile);
  1047. }
  1048. if (NULL != IDConfigDB) {
  1049. ZwClose (IDConfigDB);
  1050. }
  1051. if (NULL != currentInfo) {
  1052. ZwClose (currentInfo);
  1053. }
  1054. if (NULL != parent) {
  1055. ZwClose (parent);
  1056. }
  1057. if (NULL != currentAcpiSN) {
  1058. ExFreePool (currentAcpiSN);
  1059. }
  1060. if (NULL != ProfileList) {
  1061. for (i = 0; i < ProfileList->CurrentProfileCount; i++) {
  1062. if (ProfileList->Profile[i].FriendlyName) {
  1063. ExFreePool (ProfileList->Profile[i].FriendlyName);
  1064. }
  1065. }
  1066. ExFreePool (ProfileList);
  1067. }
  1068. if (NULL != AliasList) {
  1069. for (i = 0; i < AliasList->CurrentAliasCount; i++) {
  1070. if (AliasList->Alias[i].SerialNumber) {
  1071. ExFreePool (AliasList->Alias[i].SerialNumber);
  1072. }
  1073. }
  1074. ExFreePool (AliasList);
  1075. }
  1076. return status;
  1077. }
  1078. NTSTATUS
  1079. CmpFilterAcpiDockingState (
  1080. IN PPROFILE_ACPI_DOCKING_STATE NewDockingState,
  1081. IN ULONG CurrentDockState,
  1082. IN PWCHAR CurrentAcpiSN,
  1083. IN ULONG CurrentProfileNumber,
  1084. IN OUT PCM_HARDWARE_PROFILE_LIST ProfileList,
  1085. IN OUT PCM_HARDWARE_PROFILE_ACPI_ALIAS_LIST AliasList
  1086. )
  1087. /*++
  1088. Routine Description:
  1089. Given the new state of things and the current state of things,
  1090. prune the given list of profiles.
  1091. --*/
  1092. {
  1093. NTSTATUS status = STATUS_SUCCESS;
  1094. ULONG i = 0;
  1095. ULONG j;
  1096. ULONG len;
  1097. ULONG mask = HW_PROFILE_DOCKSTATE_UNDOCKED | HW_PROFILE_DOCKSTATE_DOCKED;
  1098. ULONG flags;
  1099. PCM_HARDWARE_PROFILE_ACPI_ALIAS alias;
  1100. BOOLEAN trueMatch = FALSE;
  1101. BOOLEAN dupDetect = FALSE;
  1102. BOOLEAN currentListed = FALSE;
  1103. BOOLEAN keepCurrent = FALSE;
  1104. PAGED_CODE ();
  1105. //
  1106. // Check for duplicate:
  1107. //
  1108. // If the user boots undocked, and then hot docks. We will generate
  1109. // a profile alias for the pnp reported undocked state [A], and one for the
  1110. // ACPI reported docked state [B}. If the use subsequently reboots docked,
  1111. // then we will create a third pnp reported docked state [C] profile alias.
  1112. // {C] is really a duplicate of [B}, but we wont know this until such time
  1113. // as the ACPI state for {B] is reported.
  1114. //
  1115. // The same can happen for undocked scenerios.
  1116. //
  1117. // Detection: If the Current Dock State is the same as
  1118. // NewDockingState.DockingState then there is a potential for a duplicate.
  1119. // In order to also have a duplicate we must have an acpi already pointing
  1120. // to a profile different than the current one.
  1121. // This must also be the first ACPI change since we booted, therefore
  1122. // CurrentAcpiSn Should be Zero.
  1123. // In other words there must be at least one true match and none of the
  1124. // true matches can point to the current profile.
  1125. //
  1126. if (AliasList) {
  1127. while (i < AliasList->CurrentAliasCount) {
  1128. alias = &AliasList->Alias[i];
  1129. if (((alias->DockState & mask) != 0) &&
  1130. ((alias->DockState & mask) !=
  1131. (NewDockingState->DockingState & mask))) {
  1132. //
  1133. // This alias claims to be docked or undocked, but does not
  1134. // match the current state. Therefore skip it.
  1135. //
  1136. ;
  1137. } else if (alias->SerialLength != NewDockingState->SerialLength) {
  1138. //
  1139. // This alias has an incompatible serial number
  1140. //
  1141. ;
  1142. } else if (alias->SerialLength ==
  1143. RtlCompareMemory (NewDockingState->SerialNumber,
  1144. alias->SerialNumber,
  1145. alias->SerialLength)) {
  1146. //
  1147. // NB RtlCompareMemory can work with zero length memory
  1148. // addresses. This is a requirement here.
  1149. //
  1150. //
  1151. // This alias matches so mark the profile.
  1152. //
  1153. for (j = 0; j < ProfileList->CurrentProfileCount; j++) {
  1154. if (ProfileList->Profile[j].Id == alias->ProfileNumber) {
  1155. //
  1156. // Alias entries should never point to a pristine profile
  1157. //
  1158. ASSERT (!(ProfileList->Profile[j].Flags &
  1159. CM_HP_FLAGS_PRISTINE));
  1160. ProfileList->Profile[j].Flags |= CM_HP_FLAGS_TRUE_MATCH;
  1161. trueMatch = TRUE;
  1162. }
  1163. if ((CurrentDockState == NewDockingState->DockingState) &&
  1164. (NULL == CurrentAcpiSN)) {
  1165. //
  1166. // The dock state did not change during this acpi
  1167. // event; therefore, we might just have a duplicate
  1168. // on our hands.
  1169. //
  1170. dupDetect = TRUE;
  1171. }
  1172. if (alias->ProfileNumber == CurrentProfileNumber) {
  1173. //
  1174. // There exists an entry in the acpi alias table that
  1175. // if chosen would result in no change of Hardware
  1176. // Profile. Therefore, we should chose this one, and
  1177. // ignore the duplicate.
  1178. //
  1179. currentListed = TRUE;
  1180. }
  1181. }
  1182. }
  1183. i++;
  1184. }
  1185. }
  1186. if ((!dupDetect) &&
  1187. (NULL == CurrentAcpiSN) &&
  1188. (!trueMatch) &&
  1189. (CurrentDockState == NewDockingState->DockingState)) {
  1190. //
  1191. // (1) The docking state did not change,
  1192. // (2) the current profile has not yet, on this boot, been marked with
  1193. // an ACPI serial number.
  1194. // (3) There was no Alias match.
  1195. //
  1196. // Therefore we should keep the current profile regardless of it being
  1197. // aliasable.
  1198. //
  1199. keepCurrent = TRUE;
  1200. trueMatch = TRUE; // prevent pristine from being listed.
  1201. }
  1202. i = 0;
  1203. while (i < ProfileList->CurrentProfileCount) {
  1204. flags = ProfileList->Profile[i].Flags;
  1205. if (dupDetect) {
  1206. if (flags & CM_HP_FLAGS_TRUE_MATCH) {
  1207. if (currentListed) {
  1208. if (ProfileList->Profile[i].Id == CurrentProfileNumber) {
  1209. //
  1210. // Let this one live. This results in no change of
  1211. // profile number.
  1212. //
  1213. i++;
  1214. continue;
  1215. }
  1216. //
  1217. // Bounce any true matches that do not result in no change
  1218. // of profile.
  1219. //
  1220. ;
  1221. } else {
  1222. //
  1223. // We did not find the current one listed so we definately
  1224. // have a duplicate.
  1225. //
  1226. // Mark it as such. and list it live.
  1227. //
  1228. ProfileList->Profile[i].Flags |= CM_HP_FLAGS_DUPLICATE;
  1229. i++;
  1230. continue;
  1231. }
  1232. }
  1233. //
  1234. // Bounce all non True matches in a duplicate detected situation.
  1235. //
  1236. ;
  1237. } else if ((flags & CM_HP_FLAGS_PRISTINE) && !trueMatch) {
  1238. //
  1239. // Leave this one in the list
  1240. //
  1241. i++;
  1242. continue;
  1243. } else if (flags & CM_HP_FLAGS_ALIASABLE) {
  1244. //
  1245. // Leave this one in the list
  1246. //
  1247. ASSERT (! (flags & CM_HP_FLAGS_PRISTINE));
  1248. i++;
  1249. continue;
  1250. } else if (flags & CM_HP_FLAGS_TRUE_MATCH) {
  1251. //
  1252. // Leave this one in the list
  1253. //
  1254. i++;
  1255. continue;
  1256. } else if (keepCurrent &&
  1257. (ProfileList->Profile[i].Id == CurrentProfileNumber)) {
  1258. //
  1259. // Leave this one in the list
  1260. //
  1261. i++;
  1262. continue;
  1263. }
  1264. //
  1265. // discard this profile by (1) shifting remaining profiles in
  1266. // array to fill in the space of this discarded profile
  1267. // and (2) decrementing profile count
  1268. //
  1269. len = ProfileList->CurrentProfileCount - i - 1;
  1270. if (0 < len) {
  1271. RtlMoveMemory(&ProfileList->Profile[i],
  1272. &ProfileList->Profile[i+1],
  1273. sizeof(CM_HARDWARE_PROFILE) * len);
  1274. }
  1275. --ProfileList->CurrentProfileCount;
  1276. }
  1277. return status;
  1278. }
  1279. NTSTATUS
  1280. CmpMoveBiosAliasTable (
  1281. IN HANDLE IDConfigDB,
  1282. IN HANDLE CurrentInfo,
  1283. IN ULONG CurrentProfileNumber,
  1284. IN ULONG NewProfileNumber,
  1285. IN PWCHAR nameBuffer,
  1286. IN PCHAR valueBuffer,
  1287. IN ULONG bufferLen
  1288. )
  1289. /*++
  1290. Routine Description:
  1291. Search the Alias table for bios entries which match the current
  1292. docking state, and point from current profile number to new profile number.
  1293. Assumption: If the profile is cloned (therefore created by
  1294. CmpCloneHwProfile, and we have just moved the bios table to point
  1295. away from this entry, then we *should* be able to safely delete
  1296. the old hardware profile key.
  1297. (in both IDConfigDB\HardwareProfiles and CCS\HardwareProfiles
  1298. --*/
  1299. {
  1300. NTSTATUS status = STATUS_SUCCESS;
  1301. HANDLE alias = NULL;
  1302. HANDLE entry = NULL;
  1303. HANDLE hwprofile = NULL;
  1304. UNICODE_STRING name;
  1305. ULONG currentDockId;
  1306. ULONG currentSerialNumber;
  1307. ULONG len;
  1308. ULONG i;
  1309. OBJECT_ATTRIBUTES attributes;
  1310. KEY_FULL_INFORMATION keyInfo;
  1311. PKEY_BASIC_INFORMATION basicInfo;
  1312. PKEY_VALUE_FULL_INFORMATION value;
  1313. PAGED_CODE ();
  1314. value = (PKEY_VALUE_FULL_INFORMATION) valueBuffer;
  1315. basicInfo = (PKEY_BASIC_INFORMATION) valueBuffer;
  1316. //
  1317. // Extract the current Serial Number and DockID
  1318. //
  1319. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_SERIAL_NUMBER);
  1320. status = NtQueryValueKey(CurrentInfo,
  1321. &name,
  1322. KeyValueFullInformation,
  1323. valueBuffer,
  1324. bufferLen,
  1325. &len);
  1326. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  1327. status = STATUS_REGISTRY_CORRUPT;
  1328. goto Clean;
  1329. }
  1330. currentSerialNumber = * (PULONG) ((PUCHAR) value + value->DataOffset);
  1331. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKID);
  1332. status = NtQueryValueKey(CurrentInfo,
  1333. &name,
  1334. KeyValueFullInformation,
  1335. valueBuffer,
  1336. bufferLen,
  1337. &len);
  1338. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  1339. status = STATUS_REGISTRY_CORRUPT;
  1340. goto Clean;
  1341. }
  1342. currentDockId = * (PULONG) ((PUCHAR) value + value->DataOffset);
  1343. //
  1344. // Open a handle to the Alias information
  1345. //
  1346. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ALIAS);
  1347. InitializeObjectAttributes (&attributes,
  1348. &name,
  1349. OBJ_CASE_INSENSITIVE,
  1350. IDConfigDB,
  1351. NULL);
  1352. status = ZwOpenKey (&alias,
  1353. KEY_READ,
  1354. &attributes);
  1355. if (!NT_SUCCESS (status)) {
  1356. //
  1357. // So we don't have an alias table. This is ok, albeit a bit strange
  1358. //
  1359. status = STATUS_SUCCESS;
  1360. alias = NULL;
  1361. goto Clean;
  1362. }
  1363. status = ZwQueryKey (alias,
  1364. KeyFullInformation,
  1365. &keyInfo,
  1366. sizeof (keyInfo),
  1367. &len);
  1368. if (!NT_SUCCESS (status)) {
  1369. goto Clean;
  1370. }
  1371. ASSERT (0 < keyInfo.SubKeys);
  1372. //
  1373. // Iterrate the alias entries
  1374. //
  1375. for (i = 0; i < keyInfo.SubKeys; i++) {
  1376. //
  1377. // Get the first key in the list.
  1378. //
  1379. status = ZwEnumerateKey (alias,
  1380. i,
  1381. KeyBasicInformation,
  1382. basicInfo,
  1383. bufferLen - sizeof (UNICODE_NULL), // term 0
  1384. &len);
  1385. if (!NT_SUCCESS (status)) {
  1386. //
  1387. // This should never happen.
  1388. //
  1389. break;
  1390. }
  1391. basicInfo->Name [basicInfo->NameLength/sizeof(WCHAR)] = 0;
  1392. name.Length = (USHORT) basicInfo->NameLength;
  1393. name.MaximumLength = (USHORT) basicInfo->NameLength + sizeof (UNICODE_NULL);
  1394. name.Buffer = basicInfo->Name;
  1395. InitializeObjectAttributes (&attributes,
  1396. &name,
  1397. OBJ_CASE_INSENSITIVE,
  1398. alias,
  1399. NULL);
  1400. status = ZwOpenKey (&entry,
  1401. KEY_READ | KEY_WRITE,
  1402. &attributes);
  1403. if (!NT_SUCCESS (status)) {
  1404. break;
  1405. }
  1406. //
  1407. // Extract The Profile number to which this alias refers.
  1408. //
  1409. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PROFILE_NUMBER);
  1410. status = NtQueryValueKey(entry,
  1411. &name,
  1412. KeyValueFullInformation,
  1413. valueBuffer,
  1414. bufferLen,
  1415. &len);
  1416. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  1417. status = STATUS_REGISTRY_CORRUPT;
  1418. goto Clean;
  1419. }
  1420. if (CurrentProfileNumber != *(PULONG)((PUCHAR)value + value->DataOffset)) {
  1421. //
  1422. // Not a match
  1423. //
  1424. ZwClose (entry);
  1425. entry = NULL;
  1426. continue;
  1427. }
  1428. //
  1429. // Compare the Dock ID
  1430. //
  1431. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_DOCKID);
  1432. status = NtQueryValueKey(entry,
  1433. &name,
  1434. KeyValueFullInformation,
  1435. valueBuffer,
  1436. bufferLen,
  1437. &len);
  1438. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  1439. status = STATUS_REGISTRY_CORRUPT;
  1440. goto Clean;
  1441. }
  1442. if (currentDockId != * (PULONG) ((PUCHAR) value + value->DataOffset)) {
  1443. //
  1444. // Not a match
  1445. //
  1446. ZwClose (entry);
  1447. entry = NULL;
  1448. continue;
  1449. }
  1450. //
  1451. // Compare the SerialNumber
  1452. //
  1453. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_SERIAL_NUMBER);
  1454. status = NtQueryValueKey(entry,
  1455. &name,
  1456. KeyValueFullInformation,
  1457. valueBuffer,
  1458. bufferLen,
  1459. &len);
  1460. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  1461. status = STATUS_REGISTRY_CORRUPT;
  1462. goto Clean;
  1463. }
  1464. if (currentSerialNumber != *(PULONG)((PUCHAR)value + value->DataOffset)) {
  1465. //
  1466. // Not a match
  1467. //
  1468. ZwClose (entry);
  1469. entry = NULL;
  1470. continue;
  1471. }
  1472. //
  1473. // This must be a match.
  1474. // move the profile number
  1475. //
  1476. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PROFILE_NUMBER);
  1477. status = NtSetValueKey (entry,
  1478. &name,
  1479. 0,
  1480. REG_DWORD,
  1481. &NewProfileNumber,
  1482. sizeof (NewProfileNumber));
  1483. ASSERT (STATUS_SUCCESS == status);
  1484. ZwClose (entry);
  1485. entry = NULL;
  1486. //
  1487. // We most likely have left a dangling profile here.
  1488. // Try to attempt to clean it up.
  1489. //
  1490. // If this profile is cloned then we created it and can therefore
  1491. // get rid of it.
  1492. //
  1493. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_HARDWARE_PROFILES);
  1494. InitializeObjectAttributes (&attributes,
  1495. &name,
  1496. OBJ_CASE_INSENSITIVE,
  1497. IDConfigDB,
  1498. NULL);
  1499. status = ZwOpenKey (&hwprofile, KEY_READ | KEY_WRITE, &attributes);
  1500. if (!NT_SUCCESS (status)) {
  1501. hwprofile = NULL;
  1502. status = STATUS_REGISTRY_CORRUPT;
  1503. goto Clean;
  1504. }
  1505. swprintf (nameBuffer, L"%04d\0", CurrentProfileNumber);
  1506. RtlInitUnicodeString (&name, nameBuffer);
  1507. InitializeObjectAttributes (&attributes,
  1508. &name,
  1509. OBJ_CASE_INSENSITIVE,
  1510. hwprofile,
  1511. NULL);
  1512. status = ZwOpenKey (&entry, KEY_ALL_ACCESS, &attributes);
  1513. if (!NT_SUCCESS (status)) {
  1514. entry = NULL;
  1515. status = STATUS_REGISTRY_CORRUPT;
  1516. goto Clean;
  1517. }
  1518. //
  1519. // Test for the Cloned Bit.
  1520. //
  1521. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CLONED);
  1522. status = NtQueryValueKey(entry,
  1523. &name,
  1524. KeyValueFullInformation,
  1525. valueBuffer,
  1526. bufferLen,
  1527. &len);
  1528. if (!NT_SUCCESS (status) || (value->Type != REG_DWORD)) {
  1529. status = STATUS_REGISTRY_CORRUPT;
  1530. goto Clean;
  1531. }
  1532. if (*(PULONG)((PUCHAR)value + value->DataOffset)) {
  1533. //
  1534. // We cloned this one.
  1535. //
  1536. status = ZwDeleteKey (entry);
  1537. ASSERT (NT_SUCCESS (status));
  1538. ZwClose (entry);
  1539. ZwClose (hwprofile);
  1540. entry = hwprofile = NULL;
  1541. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CCS_HWPROFILE);
  1542. InitializeObjectAttributes (&attributes,
  1543. &name,
  1544. OBJ_CASE_INSENSITIVE,
  1545. NULL,
  1546. NULL);
  1547. status = ZwOpenKey (&hwprofile, KEY_READ | KEY_WRITE, &attributes);
  1548. if (!NT_SUCCESS (status)) {
  1549. hwprofile = NULL;
  1550. status = STATUS_REGISTRY_CORRUPT;
  1551. goto Clean;
  1552. }
  1553. swprintf (nameBuffer, L"%04d\0", CurrentProfileNumber);
  1554. status = CmDeleteKeyRecursive (hwprofile,
  1555. nameBuffer,
  1556. valueBuffer,
  1557. bufferLen,
  1558. TRUE);
  1559. ASSERT (NT_SUCCESS (status));
  1560. ZwClose (hwprofile);
  1561. hwprofile = NULL;
  1562. } else {
  1563. //
  1564. // We didn't clone this one.
  1565. // don't do anything else.
  1566. //
  1567. ZwClose (entry);
  1568. ZwClose (hwprofile);
  1569. entry = hwprofile = NULL;
  1570. }
  1571. CM_HARDWARE_PROFILE_STR_CCS_HWPROFILE;
  1572. }
  1573. Clean:
  1574. if (alias) {
  1575. ZwClose (alias);
  1576. }
  1577. if (entry) {
  1578. ZwClose (entry);
  1579. }
  1580. if (hwprofile) {
  1581. ZwClose (hwprofile);
  1582. }
  1583. return status;
  1584. }
  1585. NTSTATUS
  1586. CmpCloneHwProfile (
  1587. IN HANDLE IDConfigDB,
  1588. IN HANDLE Parent,
  1589. IN HANDLE OldProfile,
  1590. IN ULONG OldProfileNumber,
  1591. IN USHORT DockingState,
  1592. OUT PHANDLE NewProfile,
  1593. OUT PULONG NewProfileNumber
  1594. )
  1595. /*++
  1596. Routine Description
  1597. The given hardware profile key needs cloning.
  1598. Clone the key and then return the new profile.
  1599. Return:
  1600. STATUS_SUCCESS - if the profile has been cloned, in which case the new
  1601. profile key has been opened for read / write privs. The old profile
  1602. will be closed.
  1603. <unsuccessful> - for a given error. NewProfile is invalid and the Old
  1604. Profile has also been closed.
  1605. (Copied lovingly from CmpCloneControlSet)
  1606. --*/
  1607. {
  1608. NTSTATUS status = STATUS_SUCCESS;
  1609. UNICODE_STRING newProfileName;
  1610. UNICODE_STRING name;
  1611. UNICODE_STRING friendlyName;
  1612. UNICODE_STRING guidStr;
  1613. PCM_KEY_BODY oldProfileKey;
  1614. PCM_KEY_BODY newProfileKey;
  1615. OBJECT_ATTRIBUTES attributes;
  1616. PSECURITY_DESCRIPTOR security;
  1617. ULONG securityLength;
  1618. WCHAR nameBuffer [64];
  1619. HANDLE IDConfigDBEntry = NULL;
  1620. ULONG disposition;
  1621. ULONG value;
  1622. UUID uuid;
  1623. PKEY_BASIC_INFORMATION keyBasicInfo;
  1624. PKEY_FULL_INFORMATION keyFullInfo;
  1625. PKEY_VALUE_FULL_INFORMATION keyValueInfo;
  1626. ULONG length, profileSubKeys, i;
  1627. UCHAR valueBuffer[256];
  1628. HANDLE hardwareProfiles=NULL;
  1629. HANDLE profileEntry=NULL;
  1630. PAGED_CODE ();
  1631. keyFullInfo = (PKEY_FULL_INFORMATION) valueBuffer;
  1632. keyBasicInfo = (PKEY_BASIC_INFORMATION) valueBuffer;
  1633. keyValueInfo = (PKEY_VALUE_FULL_INFORMATION) valueBuffer;
  1634. *NewProfile = 0;
  1635. *NewProfileNumber = OldProfileNumber;
  1636. //
  1637. // Find the new profile number.
  1638. //
  1639. while (*NewProfileNumber < 200) {
  1640. (*NewProfileNumber)++;
  1641. swprintf (nameBuffer, L"%04d", *NewProfileNumber);
  1642. RtlInitUnicodeString (&newProfileName, nameBuffer);
  1643. InitializeObjectAttributes(&attributes,
  1644. &newProfileName,
  1645. OBJ_CASE_INSENSITIVE,
  1646. Parent,
  1647. NULL);
  1648. status = NtOpenKey (NewProfile,
  1649. KEY_READ | KEY_WRITE,
  1650. &attributes);
  1651. if (NT_SUCCESS (status)) {
  1652. NtClose (*NewProfile);
  1653. } else if (STATUS_OBJECT_NAME_NOT_FOUND == status) {
  1654. status = STATUS_SUCCESS;
  1655. break;
  1656. } else {
  1657. break;
  1658. }
  1659. }
  1660. if (!NT_SUCCESS (status)) {
  1661. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile error finding new profile key %08lx\n", status));
  1662. goto Exit;
  1663. }
  1664. //
  1665. // Get the security descriptor from the old key to create the new clone one.
  1666. //
  1667. status = NtQuerySecurityObject (OldProfile,
  1668. DACL_SECURITY_INFORMATION,
  1669. NULL,
  1670. 0,
  1671. &securityLength);
  1672. if (STATUS_BUFFER_TOO_SMALL == status) {
  1673. security = ExAllocatePool (PagedPool, securityLength);
  1674. if (security != NULL) {
  1675. status = NtQuerySecurityObject(OldProfile,
  1676. DACL_SECURITY_INFORMATION,
  1677. security,
  1678. securityLength,
  1679. &securityLength);
  1680. if (!NT_SUCCESS (status)) {
  1681. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile"
  1682. " - NtQuerySecurityObject failed %08lx\n", status));
  1683. ExFreePool(security);
  1684. security=NULL;
  1685. }
  1686. }
  1687. } else {
  1688. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile"
  1689. " - NtQuerySecurityObject returned %08lx\n", status));
  1690. security=NULL;
  1691. }
  1692. //
  1693. // Create the new key
  1694. //
  1695. InitializeObjectAttributes (&attributes,
  1696. &newProfileName,
  1697. OBJ_CASE_INSENSITIVE,
  1698. Parent,
  1699. security);
  1700. status = NtCreateKey (NewProfile,
  1701. KEY_READ | KEY_WRITE,
  1702. &attributes,
  1703. 0,
  1704. NULL,
  1705. 0,
  1706. &disposition);
  1707. if (NULL != security) {
  1708. ExFreePool (security);
  1709. }
  1710. if (!NT_SUCCESS (status)) {
  1711. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile couldn't create Clone %08lx\n",status));
  1712. goto Exit;
  1713. }
  1714. //
  1715. // Check to make sure the key was created. If it already exists,
  1716. // something is wrong.
  1717. //
  1718. if (disposition != REG_CREATED_NEW_KEY) {
  1719. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile: Clone tree already exists!\n"));
  1720. //
  1721. // WARNNOTE:
  1722. // If somebody somehow managed to create a key in our way,
  1723. // they'll thwart duplication of the prestine. Tough luck.
  1724. // Claim it worked and go on.
  1725. //
  1726. status = STATUS_SUCCESS;
  1727. goto Exit;
  1728. }
  1729. //
  1730. // Create the IDConfigDB Entry
  1731. //
  1732. swprintf (nameBuffer, L"Hardware Profiles\\%04d", *NewProfileNumber);
  1733. RtlInitUnicodeString (&name, nameBuffer);
  1734. InitializeObjectAttributes (&attributes,
  1735. &name,
  1736. OBJ_CASE_INSENSITIVE,
  1737. IDConfigDB,
  1738. NULL);
  1739. status = NtCreateKey (&IDConfigDBEntry,
  1740. KEY_READ | KEY_WRITE,
  1741. &attributes,
  1742. 0,
  1743. NULL,
  1744. 0,
  1745. &disposition);
  1746. if (!NT_SUCCESS (status)) {
  1747. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile couldn't create Clone %08lx\n",status));
  1748. IDConfigDBEntry = NULL;
  1749. goto Exit;
  1750. }
  1751. //
  1752. // Determine the next PreferenceOrder for the new profile. (The
  1753. // PrefenceOrder for the new profile will be incrementally next from the
  1754. // greatest PreferenceOrder value of all the current profiles; assumes
  1755. // current set of PreferenceOrder values is incremental)
  1756. //
  1757. //
  1758. // Open the Hardware Profiles key
  1759. //
  1760. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_HARDWARE_PROFILES);
  1761. InitializeObjectAttributes (&attributes,
  1762. &name,
  1763. OBJ_CASE_INSENSITIVE,
  1764. IDConfigDB,
  1765. NULL);
  1766. status = ZwOpenKey (&hardwareProfiles,
  1767. KEY_READ,
  1768. &attributes);
  1769. if (!NT_SUCCESS (status)) {
  1770. hardwareProfiles = NULL;
  1771. goto Exit;
  1772. }
  1773. //
  1774. // Find the number of profile Sub Keys
  1775. //
  1776. status = ZwQueryKey (hardwareProfiles,
  1777. KeyFullInformation,
  1778. valueBuffer,
  1779. sizeof (valueBuffer),
  1780. &length);
  1781. if (!NT_SUCCESS (status)) {
  1782. goto Exit;
  1783. }
  1784. //
  1785. // At very least, the Pristine and the new profile key we just created,
  1786. // should be there.
  1787. //
  1788. profileSubKeys = keyFullInfo->SubKeys;
  1789. ASSERT (1 < profileSubKeys);
  1790. //
  1791. // Initialize the highest PreferenceOrder value found to -1.
  1792. //
  1793. value = (ULONG)-1;
  1794. //
  1795. // Iterrate the profiles
  1796. //
  1797. for (i = 0; i < profileSubKeys; i++) {
  1798. //
  1799. // Enumerate all profile subkeys, noting their PreferenceOrder values.
  1800. //
  1801. status = ZwEnumerateKey (hardwareProfiles,
  1802. i,
  1803. KeyBasicInformation,
  1804. valueBuffer,
  1805. sizeof(valueBuffer) - sizeof (UNICODE_NULL), //term 0
  1806. &length);
  1807. if(!NT_SUCCESS(status)) {
  1808. break;
  1809. }
  1810. //
  1811. // Zero-terminate the subkey name just in case.
  1812. //
  1813. keyBasicInfo->Name[keyBasicInfo->NameLength/sizeof(WCHAR)] = 0;
  1814. //
  1815. // If this is the Pristine, or the NewProfile key, ignore it.
  1816. //
  1817. if ((!_wtoi(keyBasicInfo->Name)) ||
  1818. ((ULONG)(_wtoi(keyBasicInfo->Name)) == *NewProfileNumber)) {
  1819. continue;
  1820. }
  1821. //
  1822. // Open this profile key
  1823. //
  1824. name.Length = (USHORT) keyBasicInfo->NameLength;
  1825. name.MaximumLength = (USHORT) keyBasicInfo->NameLength + sizeof (UNICODE_NULL);
  1826. name.Buffer = keyBasicInfo->Name;
  1827. InitializeObjectAttributes (&attributes,
  1828. &name,
  1829. OBJ_CASE_INSENSITIVE,
  1830. hardwareProfiles,
  1831. NULL);
  1832. status = ZwOpenKey (&profileEntry,
  1833. KEY_READ,
  1834. &attributes);
  1835. if (!NT_SUCCESS (status)) {
  1836. profileEntry = NULL;
  1837. continue;
  1838. }
  1839. //
  1840. // Extract The PreferenceOrder value for this Profile.
  1841. //
  1842. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PREFERENCE_ORDER);
  1843. status = NtQueryValueKey(profileEntry,
  1844. &name,
  1845. KeyValueFullInformation,
  1846. valueBuffer,
  1847. sizeof(valueBuffer),
  1848. &length);
  1849. if (!NT_SUCCESS (status) || (keyValueInfo->Type != REG_DWORD)) {
  1850. //
  1851. // No PreferenceOrder; continue on as best we can
  1852. //
  1853. ZwClose(profileEntry);
  1854. profileEntry=NULL;
  1855. continue;
  1856. }
  1857. //
  1858. // If this is a the highest PreferenceOrder so far, reassign value to
  1859. // this PreferenceOrder, OR assign it this valid PreferenceOrder if
  1860. // value is still unassigned.
  1861. //
  1862. if (((*(PULONG) ((PUCHAR)keyValueInfo + keyValueInfo->DataOffset)) > value) ||
  1863. (value == -1)) {
  1864. value = (* (PULONG) ((PUCHAR)keyValueInfo + keyValueInfo->DataOffset));
  1865. }
  1866. ZwClose(profileEntry);
  1867. profileEntry=NULL;
  1868. }
  1869. //
  1870. // Increment value one above the greatest PreferenceOrder found.
  1871. // (If no other profiles were found, (value+=1) == 0, the most preferred
  1872. // profile)
  1873. //
  1874. value += 1;
  1875. //
  1876. // Give the new profile a preference order.
  1877. //
  1878. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_PREFERENCE_ORDER);
  1879. status = NtSetValueKey (IDConfigDBEntry,
  1880. &name,
  1881. 0,
  1882. REG_DWORD,
  1883. &value,
  1884. sizeof (value));
  1885. //
  1886. // Give the new profile a friendly name, based on the DockingState
  1887. //
  1888. status = CmpCreateHwProfileFriendlyName(IDConfigDB,
  1889. DockingState,
  1890. *NewProfileNumber,
  1891. &friendlyName);
  1892. if (NT_SUCCESS(status)) {
  1893. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_FRIENDLY_NAME);
  1894. status = NtSetValueKey (IDConfigDBEntry,
  1895. &name,
  1896. 0,
  1897. REG_SZ,
  1898. friendlyName.Buffer,
  1899. friendlyName.Length + sizeof(UNICODE_NULL));
  1900. RtlFreeUnicodeString(&friendlyName);
  1901. }
  1902. //
  1903. // Set the aliasable flag on the new "cloned profile" to be false
  1904. //
  1905. value = FALSE;
  1906. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_ALIASABLE);
  1907. status = NtSetValueKey (IDConfigDBEntry,
  1908. &name,
  1909. 0,
  1910. REG_DWORD,
  1911. &value,
  1912. sizeof (value));
  1913. //
  1914. // Set the cloned profile on the new "cloned profile" to be true;
  1915. //
  1916. value = TRUE;
  1917. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_CLONED);
  1918. status = NtSetValueKey (IDConfigDBEntry,
  1919. &name,
  1920. 0,
  1921. REG_DWORD,
  1922. &value,
  1923. sizeof (value));
  1924. //
  1925. // Set the HwProfileGuid for the brand new profile
  1926. //
  1927. status = ExUuidCreate (&uuid);
  1928. if (NT_SUCCESS (status)) {
  1929. status = RtlStringFromGUID (&uuid, &guidStr);
  1930. if (NT_SUCCESS (status)) {
  1931. RtlInitUnicodeString (&name, CM_HARDWARE_PROFILE_STR_HW_PROFILE_GUID);
  1932. status = NtSetValueKey (IDConfigDBEntry,
  1933. &name,
  1934. 0,
  1935. REG_SZ,
  1936. guidStr.Buffer,
  1937. guidStr.MaximumLength);
  1938. RtlFreeUnicodeString(&guidStr);
  1939. } else {
  1940. //
  1941. // What's a fella to do?
  1942. // let's just go on.
  1943. //
  1944. status = STATUS_SUCCESS;
  1945. }
  1946. } else {
  1947. //
  1948. // let's just go on.
  1949. //
  1950. status = STATUS_SUCCESS;
  1951. }
  1952. //
  1953. // Clone the key
  1954. //
  1955. // (Copied lovingly from CmpCloneControlSet)
  1956. //
  1957. //
  1958. status = ObReferenceObjectByHandle (OldProfile,
  1959. KEY_READ,
  1960. CmpKeyObjectType,
  1961. KernelMode,
  1962. (PVOID *)(&oldProfileKey),
  1963. NULL);
  1964. if (!NT_SUCCESS(status)) {
  1965. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHWProfile: couldn't reference CurrentHandle %08lx\n",
  1966. status));
  1967. goto Exit;
  1968. }
  1969. status = ObReferenceObjectByHandle (*NewProfile,
  1970. KEY_WRITE,
  1971. CmpKeyObjectType,
  1972. KernelMode,
  1973. (PVOID *)(&newProfileKey),
  1974. NULL);
  1975. if (!NT_SUCCESS(status)) {
  1976. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHWProfile: couldn't reference CurrentHandle %08lx\n",
  1977. status));
  1978. goto Exit;
  1979. }
  1980. CmpLockRegistryExclusive();
  1981. //
  1982. // Note: This copy tree command does not copy the values in the
  1983. // root keys. We are relying on this, since the values stored there
  1984. // are things like "pristine" which we do not wish to have moved to the
  1985. // new tree.
  1986. //
  1987. if (CmpCopyTree(oldProfileKey->KeyControlBlock->KeyHive,
  1988. oldProfileKey->KeyControlBlock->KeyCell,
  1989. newProfileKey->KeyControlBlock->KeyHive,
  1990. newProfileKey->KeyControlBlock->KeyCell)) {
  1991. //
  1992. // Set the max subkey name property for the new target key.
  1993. //
  1994. PCM_KEY_NODE SourceNode;
  1995. PCM_KEY_NODE DestNode;
  1996. SourceNode = (PCM_KEY_NODE)HvGetCell(oldProfileKey->KeyControlBlock->KeyHive,oldProfileKey->KeyControlBlock->KeyCell);
  1997. if( SourceNode != NULL ) {
  1998. DestNode = (PCM_KEY_NODE)HvGetCell(newProfileKey->KeyControlBlock->KeyHive,newProfileKey->KeyControlBlock->KeyCell);
  1999. if( DestNode != NULL ) {
  2000. //
  2001. // CmpCopyTree doesn't do this.
  2002. //
  2003. ASSERT_CELL_DIRTY(newProfileKey->KeyControlBlock->KeyHive,newProfileKey->KeyControlBlock->KeyCell);
  2004. DestNode->MaxNameLen = SourceNode->MaxNameLen;
  2005. DestNode->MaxClassLen = SourceNode->MaxClassLen;
  2006. HvReleaseCell(newProfileKey->KeyControlBlock->KeyHive,newProfileKey->KeyControlBlock->KeyCell);
  2007. CmpRebuildKcbCache(newProfileKey->KeyControlBlock);
  2008. status = STATUS_SUCCESS;
  2009. } else {
  2010. status = STATUS_INSUFFICIENT_RESOURCES;
  2011. }
  2012. HvReleaseCell(oldProfileKey->KeyControlBlock->KeyHive,oldProfileKey->KeyControlBlock->KeyCell);
  2013. } else {
  2014. status = STATUS_INSUFFICIENT_RESOURCES;
  2015. }
  2016. } else {
  2017. CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BUGCHECK,"CM: CmpCloneHwProfile: tree copy failed.\n"));
  2018. status = STATUS_REGISTRY_CORRUPT;
  2019. }
  2020. CmpUnlockRegistry();
  2021. Exit:
  2022. NtClose (OldProfile);
  2023. if (IDConfigDBEntry) {
  2024. NtClose (IDConfigDBEntry);
  2025. }
  2026. if (hardwareProfiles) {
  2027. NtClose (hardwareProfiles);
  2028. }
  2029. if (!NT_SUCCESS (status)) {
  2030. if (*NewProfile) {
  2031. NtClose (*NewProfile);
  2032. }
  2033. }
  2034. return status;
  2035. }
  2036. NTSTATUS
  2037. CmDeleteKeyRecursive(
  2038. HANDLE hKeyRoot,
  2039. PWSTR Key,
  2040. PVOID TemporaryBuffer,
  2041. ULONG LengthTemporaryBuffer,
  2042. BOOLEAN ThisKeyToo
  2043. )
  2044. /*++
  2045. Routine Description:
  2046. Routine to recursively delete all subkeys under the given
  2047. key, including the key given.
  2048. Arguments:
  2049. hKeyRoot: Handle to root relative to which the key to be deleted is
  2050. specified.
  2051. Key: Root relative path of the key which is to be recursively deleted.
  2052. ThisKeyToo: Whether after deletion of all subkeys, this key itself is to
  2053. be deleted.
  2054. Return Value:
  2055. Status is returned.
  2056. --*/
  2057. {
  2058. ULONG ResultLength;
  2059. PKEY_BASIC_INFORMATION KeyInfo;
  2060. NTSTATUS Status;
  2061. UNICODE_STRING UnicodeString;
  2062. OBJECT_ATTRIBUTES Obja;
  2063. PWSTR SubkeyName;
  2064. HANDLE hKey;
  2065. //
  2066. // Initialize
  2067. //
  2068. KeyInfo = (PKEY_BASIC_INFORMATION)TemporaryBuffer;
  2069. //
  2070. // Open the key
  2071. //
  2072. RtlInitUnicodeString (&UnicodeString,Key);
  2073. InitializeObjectAttributes(&Obja,
  2074. &UnicodeString,
  2075. OBJ_CASE_INSENSITIVE,
  2076. hKeyRoot,
  2077. NULL);
  2078. Status = ZwOpenKey(&hKey,KEY_ALL_ACCESS,&Obja);
  2079. if( !NT_SUCCESS(Status) ) {
  2080. return(Status);
  2081. }
  2082. //
  2083. // Enumerate all subkeys of the current key. if any exist they should
  2084. // be deleted first. since deleting the subkey affects the subkey
  2085. // index, we always enumerate on subkeyindex 0
  2086. //
  2087. while(1) {
  2088. Status = ZwEnumerateKey(
  2089. hKey,
  2090. 0,
  2091. KeyBasicInformation,
  2092. TemporaryBuffer,
  2093. LengthTemporaryBuffer,
  2094. &ResultLength
  2095. );
  2096. if(!NT_SUCCESS(Status)) {
  2097. break;
  2098. }
  2099. //
  2100. // Zero-terminate the subkey name just in case.
  2101. //
  2102. KeyInfo->Name[KeyInfo->NameLength/sizeof(WCHAR)] = 0;
  2103. //
  2104. // Make a duplicate of the subkey name because the name is
  2105. // in TemporaryBuffer, which might get clobbered by recursive
  2106. // calls to this routine.
  2107. //
  2108. SubkeyName = ExAllocatePool (PagedPool,
  2109. ((wcslen (KeyInfo->Name) + 1) *
  2110. sizeof (WCHAR)));
  2111. if (!SubkeyName) {
  2112. Status = STATUS_INSUFFICIENT_RESOURCES;
  2113. break;
  2114. }
  2115. wcscpy(SubkeyName, KeyInfo->Name);
  2116. Status = CmDeleteKeyRecursive( hKey,
  2117. SubkeyName,
  2118. TemporaryBuffer,
  2119. LengthTemporaryBuffer,
  2120. TRUE);
  2121. ExFreePool(SubkeyName);
  2122. if(!NT_SUCCESS(Status)) {
  2123. break;
  2124. }
  2125. }
  2126. //
  2127. // Check the status, if the status is anything other than
  2128. // STATUS_NO_MORE_ENTRIES we failed in deleting some subkey,
  2129. // so we cannot delete this key too
  2130. //
  2131. if( Status == STATUS_NO_MORE_ENTRIES) {
  2132. Status = STATUS_SUCCESS;
  2133. }
  2134. if (!NT_SUCCESS (Status)) {
  2135. ZwClose(hKey);
  2136. return (Status);
  2137. }
  2138. //
  2139. // else delete the current key if asked to do so
  2140. //
  2141. if( ThisKeyToo ) {
  2142. Status = ZwDeleteKey (hKey);
  2143. }
  2144. ZwClose(hKey);
  2145. return(Status);
  2146. }
  2147. NTSTATUS
  2148. CmpCreateHwProfileFriendlyName (
  2149. IN HANDLE IDConfigDB,
  2150. IN ULONG DockingState,
  2151. IN ULONG NewProfileNumber,
  2152. OUT PUNICODE_STRING FriendlyName
  2153. )
  2154. /*++
  2155. Routine Description:
  2156. Create a new FriendlyName for a new Hardware Profile, given the DockState.
  2157. If a new profile name based on the DockState cannot be created, an attempt
  2158. is made to create a default FriendlyName based on NewProfileNumber. If
  2159. successful, a unicode string with the new profile friendlyName is created.
  2160. It is the responsibility of the caller to free this using
  2161. RtlFreeUnicodeString. If unsuccesful, no string is returned.
  2162. Arguments:
  2163. IDConfigDB: Handle to the IDConfigDB registry key.
  2164. DockingState: The Docking State of the profile for which the new
  2165. FriendlyName is being created. This should be one of:
  2166. HW_PROFILE_DOCKSTATE_DOCKED,
  2167. HW_PROFILE_DOCKSTATE_UNDOCKED, or
  2168. HW_PROFILE_DOCKSTATE_UNKNOWN
  2169. NewProfileNumber: The number of the new profile being created. If unable to
  2170. create a DockState specific FriendlyName, this value will
  2171. be used to create a (not-so) FriendlyName.
  2172. FriendlyName: Supplies a unicode string to receive the FriendlyName for this
  2173. new profile. The caller is expected to free this with
  2174. RtlFreeUnicodeString.
  2175. Return:
  2176. NTSTATUS code. Currently returns STATUS_SUCCESS, or STATUS_UNSUCCESSFUL.
  2177. Notes:
  2178. The new FriendlyName is generated from the DockState and appropriate
  2179. counter, and may not necessarily be unique among the existing Hardware
  2180. Profiles.
  2181. The naming scheme used here (including the localized strings in the kernel
  2182. message table) should be kept in sync with that provided to the user through
  2183. the Hardware Profile control panel applet.
  2184. --*/
  2185. {
  2186. NTSTATUS status = STATUS_SUCCESS;
  2187. ANSI_STRING ansiString;
  2188. UNICODE_STRING unicodeString;
  2189. UNICODE_STRING labelName, keyName;
  2190. PMESSAGE_RESOURCE_ENTRY messageEntry;
  2191. PKLDR_DATA_TABLE_ENTRY dataTableEntry;
  2192. ULONG messageId;
  2193. UCHAR valueBuffer[256];
  2194. WCHAR friendlyNameBuffer[MAX_FRIENDLY_NAME_LENGTH/sizeof(WCHAR)];
  2195. PKEY_VALUE_FULL_INFORMATION keyValueInfo;
  2196. ULONG length, index;
  2197. HANDLE hardwareProfiles=NULL;
  2198. OBJECT_ATTRIBUTES attributes;
  2199. PAGED_CODE ();
  2200. //
  2201. // Make sure we were given a place to put the FriendlyName
  2202. //
  2203. if (!FriendlyName) {
  2204. return STATUS_INVALID_PARAMETER;
  2205. }
  2206. //
  2207. // If we don't have a handle to IDConfigDB, try to assign a default
  2208. // FriendlyName on the way out.
  2209. //
  2210. if (!IDConfigDB) {
  2211. status = STATUS_INVALID_PARAMETER;
  2212. goto Exit;
  2213. }
  2214. //
  2215. // Determine the appropriate message to use, based on the DockState.
  2216. //
  2217. if ((DockingState & HW_PROFILE_DOCKSTATE_UNKNOWN) == HW_PROFILE_DOCKSTATE_UNKNOWN){
  2218. messageId = HARDWARE_PROFILE_UNKNOWN_STRING;
  2219. RtlInitUnicodeString(&labelName, CM_HARDWARE_PROFILE_STR_UNKNOWN);
  2220. } else if (DockingState & HW_PROFILE_DOCKSTATE_DOCKED) {
  2221. messageId = HARDWARE_PROFILE_DOCKED_STRING;
  2222. RtlInitUnicodeString(&labelName, CM_HARDWARE_PROFILE_STR_DOCKED);
  2223. } else if (DockingState & HW_PROFILE_DOCKSTATE_UNDOCKED) {
  2224. messageId = HARDWARE_PROFILE_UNDOCKED_STRING;
  2225. RtlInitUnicodeString(&labelName, CM_HARDWARE_PROFILE_STR_UNDOCKED);
  2226. } else {
  2227. messageId = HARDWARE_PROFILE_UNKNOWN_STRING;
  2228. RtlInitUnicodeString(&labelName, CM_HARDWARE_PROFILE_STR_UNKNOWN);
  2229. }
  2230. //
  2231. // Find the message entry in the kernel's own message table. KeLoaderBlock
  2232. // is available when we're creating hardware profiles during system
  2233. // initialization only; for profiles created thereafter, use the the first
  2234. // entry of the PsLoadedModuleList to get the image base of the kernel.
  2235. //
  2236. if (KeLoaderBlock) {
  2237. dataTableEntry = CONTAINING_RECORD(KeLoaderBlock->LoadOrderListHead.Flink,
  2238. KLDR_DATA_TABLE_ENTRY,
  2239. InLoadOrderLinks);
  2240. } else if (PsLoadedModuleList.Flink) {
  2241. dataTableEntry = CONTAINING_RECORD(PsLoadedModuleList.Flink,
  2242. KLDR_DATA_TABLE_ENTRY,
  2243. InLoadOrderLinks);
  2244. } else {
  2245. status = STATUS_UNSUCCESSFUL;
  2246. goto Exit;
  2247. }
  2248. status = RtlFindMessage(dataTableEntry->DllBase,
  2249. (ULONG_PTR)11, // RT_MESSAGETABLE
  2250. MAKELANGID(LANG_NEUTRAL,SUBLANG_SYS_DEFAULT), // System default language
  2251. messageId,
  2252. &messageEntry);
  2253. if (!NT_SUCCESS(status)) {
  2254. goto Exit;
  2255. }
  2256. if(!(messageEntry->Flags & MESSAGE_RESOURCE_UNICODE)) {
  2257. //
  2258. // If the message is not unicode, convert to unicode.
  2259. // Let the conversion routine allocate the buffer.
  2260. //
  2261. RtlInitAnsiString(&ansiString,(PCSZ)messageEntry->Text);
  2262. status = RtlAnsiStringToUnicodeString(&unicodeString,&ansiString,TRUE);
  2263. } else {
  2264. //
  2265. // Message is already unicode. Make a copy.
  2266. //
  2267. status = RtlCreateUnicodeString(&unicodeString,(PWSTR)messageEntry->Text);
  2268. }
  2269. if(!NT_SUCCESS(status)) {
  2270. goto Exit;
  2271. }
  2272. //
  2273. // Strip the trailing CRLF.
  2274. //
  2275. if (unicodeString.Length > 2 * sizeof(WCHAR)) {
  2276. unicodeString.Length -= 2 * sizeof(WCHAR);
  2277. unicodeString.Buffer[unicodeString.Length / sizeof(WCHAR)] = UNICODE_NULL;
  2278. }
  2279. //
  2280. // Check that the size of the label, with any numeric tag that may
  2281. // potentially be added (up to 4 digits, preceded by a space) is not too
  2282. // big.
  2283. //
  2284. if ((unicodeString.Length + 5*sizeof(WCHAR) + sizeof(UNICODE_NULL)) >
  2285. MAX_FRIENDLY_NAME_LENGTH) {
  2286. status = STATUS_UNSUCCESSFUL;
  2287. goto Clean;
  2288. }
  2289. //
  2290. // Open the Hardware Profiles key.
  2291. //
  2292. RtlInitUnicodeString(&keyName, CM_HARDWARE_PROFILE_STR_HARDWARE_PROFILES);
  2293. InitializeObjectAttributes(&attributes,
  2294. &keyName,
  2295. OBJ_CASE_INSENSITIVE,
  2296. IDConfigDB,
  2297. NULL);
  2298. status = ZwOpenKey(&hardwareProfiles,
  2299. KEY_READ,
  2300. &attributes);
  2301. if (!NT_SUCCESS(status)) {
  2302. hardwareProfiles = NULL;
  2303. goto Clean;
  2304. }
  2305. //
  2306. // Retrieve the counter of FriendlyNames we have previously assigned, based
  2307. // on this DockState.
  2308. //
  2309. keyValueInfo = (PKEY_VALUE_FULL_INFORMATION) valueBuffer;
  2310. status = ZwQueryValueKey(hardwareProfiles,
  2311. &labelName,
  2312. KeyValueFullInformation,
  2313. valueBuffer,
  2314. sizeof(valueBuffer),
  2315. &length);
  2316. if (NT_SUCCESS(status) && (keyValueInfo->Type == REG_DWORD)) {
  2317. //
  2318. // Increment the counter.
  2319. //
  2320. index = (* (PULONG) ((PUCHAR)keyValueInfo + keyValueInfo->DataOffset));
  2321. index++;
  2322. } else {
  2323. //
  2324. // Missing or invalid counter value; start the counter at "1".
  2325. //
  2326. index = 1;
  2327. }
  2328. //
  2329. // Update the counter in the registry.
  2330. //
  2331. status = ZwSetValueKey(hardwareProfiles,
  2332. &labelName,
  2333. 0,
  2334. REG_DWORD,
  2335. &index,
  2336. sizeof(index));
  2337. if (!NT_SUCCESS(status)) {
  2338. goto Clean;
  2339. }
  2340. //
  2341. // Copy the FriendlyName, adding the index if necessary.
  2342. //
  2343. if ((messageId == HARDWARE_PROFILE_UNKNOWN_STRING) || (index > 1)) {
  2344. swprintf(friendlyNameBuffer, L"%s %u",
  2345. unicodeString.Buffer, index);
  2346. } else {
  2347. wcscpy(friendlyNameBuffer, unicodeString.Buffer);
  2348. }
  2349. Clean:
  2350. RtlFreeUnicodeString(&unicodeString);
  2351. if (hardwareProfiles!=NULL) {
  2352. ZwClose(hardwareProfiles);
  2353. }
  2354. Exit:
  2355. if (!NT_SUCCESS(status)) {
  2356. //
  2357. // If we failed to assign a counter-based FriendlyName for whatever
  2358. // reason, give the new profile a new (not so) friendly name as a last
  2359. // resort.
  2360. //
  2361. swprintf (friendlyNameBuffer, L"%04d", NewProfileNumber);
  2362. status = STATUS_SUCCESS;
  2363. }
  2364. //
  2365. // Create the unicode string to return to the caller.
  2366. //
  2367. if (!RtlCreateUnicodeString(FriendlyName, (PWSTR)friendlyNameBuffer)) {
  2368. status = STATUS_UNSUCCESSFUL;
  2369. }
  2370. return status;
  2371. } // CmpCreateHwProfileFriendlyName