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.

1470 lines
42 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. spdskreg.c
  5. Abstract:
  6. Code for building and manipulating the disk registry. Used in the Win9x Upgrade
  7. case.
  8. Author:
  9. Marc R. Whitten (marcw) 11-Mar-1997
  10. Revision History:
  11. --*/
  12. #include "spprecmp.h"
  13. #pragma hdrstop
  14. PUCHAR DiskRegistryKey = DISK_REGISTRY_KEY;
  15. PUCHAR DiskRegistryClass = "Disk and fault tolerance information.";
  16. PUCHAR DiskRegistryValue = DISK_REGISTRY_VALUE;
  17. #define WORK_BUFFER_SIZE 4096
  18. //
  19. // In spw9xupg.c - Should be moved to a header file.
  20. //
  21. PDISK_REGION
  22. SpFirstPartitionedRegion (
  23. IN PDISK_REGION Region,
  24. IN BOOLEAN Primary
  25. );
  26. PDISK_REGION
  27. SpNextPartitionedRegion (
  28. IN PDISK_REGION Region,
  29. IN BOOLEAN Primary
  30. );
  31. //
  32. // In spupgcfg.c
  33. //
  34. NTSTATUS
  35. SppCopyKeyRecursive(
  36. HANDLE hKeyRootSrc,
  37. HANDLE hKeyRootDst,
  38. PWSTR SrcKeyPath, OPTIONAL
  39. PWSTR DstKeyPath, OPTIONAL
  40. BOOLEAN CopyAlways,
  41. BOOLEAN ApplyACLsAlways
  42. );
  43. //
  44. // wrapper functions to allow linking with diskreg.lib.
  45. //
  46. //
  47. // Have to turn off this warning temporarily.
  48. //
  49. #define TESTANDFREE(Memory) {if (Memory) {SpMemFree(Memory);}}
  50. NTSTATUS
  51. FtCreateKey(
  52. PHANDLE HandlePtr,
  53. PUCHAR KeyName,
  54. PUCHAR KeyClass
  55. )
  56. {
  57. NTSTATUS status;
  58. STRING keyString;
  59. UNICODE_STRING unicodeKeyName;
  60. STRING classString;
  61. UNICODE_STRING unicodeClassName;
  62. OBJECT_ATTRIBUTES objectAttributes;
  63. ULONG disposition;
  64. HANDLE tempHandle;
  65. //
  66. // Initialize the object for the key.
  67. //
  68. RtlInitString(&keyString,
  69. KeyName);
  70. (VOID)RtlAnsiStringToUnicodeString(&unicodeKeyName,
  71. &keyString,
  72. TRUE);
  73. memset(&objectAttributes, 0, sizeof(OBJECT_ATTRIBUTES));
  74. InitializeObjectAttributes(&objectAttributes,
  75. &unicodeKeyName,
  76. OBJ_CASE_INSENSITIVE,
  77. NULL,
  78. NULL);
  79. //
  80. // Setup the unicode class value.
  81. //
  82. RtlInitString(&classString,
  83. KeyClass);
  84. (VOID)RtlAnsiStringToUnicodeString(&unicodeClassName,
  85. &classString,
  86. TRUE);
  87. //
  88. // Create the key.
  89. //
  90. status = ZwCreateKey(&tempHandle,
  91. KEY_READ | KEY_WRITE,
  92. &objectAttributes,
  93. 0,
  94. &unicodeClassName,
  95. REG_OPTION_NON_VOLATILE,
  96. &disposition);
  97. if (NT_SUCCESS(status)) {
  98. switch (disposition)
  99. {
  100. case REG_CREATED_NEW_KEY:
  101. break;
  102. case REG_OPENED_EXISTING_KEY:
  103. break;
  104. default:
  105. break;
  106. }
  107. }
  108. //
  109. // Free all allocated space.
  110. //
  111. RtlFreeUnicodeString(&unicodeKeyName);
  112. RtlFreeUnicodeString(&unicodeClassName);
  113. if (HandlePtr != NULL) {
  114. *HandlePtr = tempHandle;
  115. } else {
  116. NtClose(tempHandle);
  117. }
  118. return status;
  119. }
  120. NTSTATUS
  121. FtOpenKey(
  122. PHANDLE HandlePtr,
  123. PUCHAR KeyName,
  124. PUCHAR CreateKeyClass
  125. )
  126. {
  127. NTSTATUS status;
  128. STRING keyString;
  129. OBJECT_ATTRIBUTES objectAttributes;
  130. UNICODE_STRING unicodeKeyName;
  131. RtlInitString(&keyString,
  132. KeyName);
  133. (VOID)RtlAnsiStringToUnicodeString(&unicodeKeyName,
  134. &keyString,
  135. TRUE);
  136. memset(&objectAttributes, 0, sizeof(OBJECT_ATTRIBUTES));
  137. InitializeObjectAttributes(&objectAttributes,
  138. &unicodeKeyName,
  139. OBJ_CASE_INSENSITIVE,
  140. NULL,
  141. NULL);
  142. status = ZwOpenKey(HandlePtr,
  143. MAXIMUM_ALLOWED,
  144. &objectAttributes);
  145. RtlFreeUnicodeString(&unicodeKeyName);
  146. if ((!NT_SUCCESS(status)) && (CreateKeyClass)) {
  147. status = FtCreateKey(HandlePtr,
  148. KeyName,
  149. CreateKeyClass);
  150. }
  151. return status;
  152. }
  153. NTSTATUS
  154. FtRegistryQuery(
  155. IN PUCHAR ValueName,
  156. OUT PVOID *FreeToken,
  157. OUT PVOID *Buffer,
  158. OUT ULONG *LengthReturned,
  159. OUT PHANDLE HandlePtr
  160. )
  161. {
  162. NTSTATUS status;
  163. HANDLE handle;
  164. ULONG resultLength;
  165. STRING valueString;
  166. UNICODE_STRING unicodeValueName;
  167. PDISK_CONFIG_HEADER regHeader;
  168. PKEY_VALUE_FULL_INFORMATION keyValueInformation= NULL;
  169. *LengthReturned = 0;
  170. status = FtOpenKey(&handle,
  171. DiskRegistryKey,
  172. NULL);
  173. if (NT_SUCCESS(status)) {
  174. RtlInitString(&valueString,
  175. ValueName);
  176. RtlAnsiStringToUnicodeString(&unicodeValueName,
  177. &valueString,
  178. TRUE);
  179. resultLength = WORK_BUFFER_SIZE;
  180. while (1) {
  181. keyValueInformation = (PKEY_VALUE_FULL_INFORMATION)
  182. SpMemAlloc(resultLength);
  183. status = ZwQueryValueKey(handle,
  184. &unicodeValueName,
  185. KeyValueFullInformation,
  186. keyValueInformation,
  187. resultLength,
  188. &resultLength);
  189. if (status == STATUS_BUFFER_OVERFLOW) {
  190. TESTANDFREE(keyValueInformation);
  191. //
  192. // Loop again and get a larger buffer.
  193. //
  194. } else {
  195. //
  196. // Either a real error or the information fit.
  197. //
  198. break;
  199. }
  200. }
  201. RtlFreeUnicodeString(&unicodeValueName);
  202. if (HandlePtr != NULL) {
  203. *HandlePtr = handle;
  204. } else {
  205. NtClose(handle);
  206. }
  207. if (NT_SUCCESS(status)) {
  208. if (keyValueInformation->DataLength == 0) {
  209. //
  210. // Treat this as if there was not disk information.
  211. //
  212. TESTANDFREE(keyValueInformation);
  213. *FreeToken = (PVOID) NULL;
  214. return STATUS_OBJECT_NAME_NOT_FOUND;
  215. } else {
  216. //
  217. // Set up the pointers for the caller.
  218. //
  219. regHeader = (PDISK_CONFIG_HEADER)
  220. ((PUCHAR) keyValueInformation + keyValueInformation->DataOffset);
  221. *LengthReturned = regHeader->FtInformationOffset +
  222. regHeader->FtInformationSize;
  223. *Buffer = (PVOID) regHeader;
  224. }
  225. }
  226. *FreeToken = (PVOID) keyValueInformation;
  227. } else {
  228. *FreeToken = (PVOID) NULL;
  229. }
  230. return status;
  231. }
  232. NTSTATUS
  233. FtSetValue(
  234. HANDLE KeyHandle,
  235. PUCHAR ValueName,
  236. PVOID DataBuffer,
  237. ULONG DataLength,
  238. ULONG Type
  239. )
  240. {
  241. NTSTATUS status;
  242. STRING valueString;
  243. UNICODE_STRING unicodeValueName;
  244. RtlInitString(&valueString,
  245. ValueName);
  246. RtlAnsiStringToUnicodeString(&unicodeValueName,
  247. &valueString,
  248. TRUE);
  249. status = ZwSetValueKey(KeyHandle,
  250. &unicodeValueName,
  251. 0,
  252. Type,
  253. DataBuffer,
  254. DataLength);
  255. RtlFreeUnicodeString(&unicodeValueName);
  256. return status;
  257. }
  258. NTSTATUS
  259. FtDeleteValue(
  260. HANDLE KeyHandle,
  261. PUCHAR ValueName
  262. )
  263. {
  264. NTSTATUS status;
  265. STRING valueString;
  266. UNICODE_STRING unicodeValueName;
  267. RtlInitString(&valueString,
  268. ValueName);
  269. status = RtlAnsiStringToUnicodeString(&unicodeValueName,
  270. &valueString,
  271. TRUE);
  272. if (!NT_SUCCESS(status)) {
  273. return status;
  274. }
  275. status = ZwDeleteValueKey(KeyHandle,
  276. &unicodeValueName);
  277. RtlFreeUnicodeString(&unicodeValueName);
  278. return status;
  279. }
  280. VOID
  281. FtBackup(
  282. IN HANDLE KeyHandle
  283. )
  284. {
  285. //
  286. // For the time being (i.e. rename doesn't work), just attempt
  287. // to delete the value.
  288. //
  289. (VOID) FtDeleteValue(KeyHandle,
  290. DiskRegistryKey);
  291. }
  292. BOOLEAN
  293. SpDiskRegistryAssignDriveLetter(
  294. ULONG Signature,
  295. LARGE_INTEGER StartingOffset,
  296. LARGE_INTEGER Length,
  297. UCHAR DriveLetter
  298. )
  299. /*++
  300. Routine Description:
  301. This routine will get the information from the disk registry
  302. and update the drive letter assigned for the partition in
  303. the registry information. This includes any cleanup for FT
  304. sets when they change drive letter.
  305. Arguments:
  306. Signature - disk signature for disk containing partition for letter.
  307. StartingOffset - Starting offset of partition for the letter.
  308. Length - lenght of affected partition.
  309. DriveLetter - New drive letter for affected partition.
  310. Return Value:
  311. TRUE if all works.
  312. --*/
  313. {
  314. BOOLEAN writeRegistry= FALSE;
  315. PVOID freeToken = NULL;
  316. ULONG lengthReturned,
  317. i,
  318. j,
  319. k,
  320. l;
  321. NTSTATUS status;
  322. USHORT type,
  323. group;
  324. PDISK_CONFIG_HEADER regHeader;
  325. PDISK_REGISTRY diskRegistry;
  326. PDISK_DESCRIPTION diskDescription;
  327. PDISK_PARTITION diskPartition;
  328. PUCHAR endOfDiskInfo;
  329. HANDLE handle;
  330. PFT_REGISTRY ftRegistry;
  331. PFT_DESCRIPTION ftDescription;
  332. PFT_MEMBER_DESCRIPTION ftMember;
  333. //
  334. // Get the registry information.
  335. //
  336. status = FtRegistryQuery(DiskRegistryValue,
  337. &freeToken,
  338. (PVOID *) &regHeader,
  339. &lengthReturned,
  340. &handle);
  341. if (!NT_SUCCESS(status)) {
  342. //
  343. // Could be permission problem, or there is no registry information.
  344. //
  345. lengthReturned = 0;
  346. //
  347. // Try to open the key for later use when setting the new value.
  348. //
  349. status = FtOpenKey(&handle,
  350. DiskRegistryKey,
  351. NULL);
  352. }
  353. if (!NT_SUCCESS(status)) {
  354. //
  355. // There is no registry key for the disk information.
  356. // Return FALSE and force caller to create registry information.
  357. //
  358. return FALSE;
  359. }
  360. if (lengthReturned == 0) {
  361. //
  362. // There is currently no registry information.
  363. //
  364. NtClose(handle);
  365. TESTANDFREE(freeToken);
  366. return FALSE;
  367. }
  368. //
  369. // Search for the disk signature.
  370. //
  371. diskRegistry = (PDISK_REGISTRY)
  372. ((PUCHAR)regHeader + regHeader->DiskInformationOffset);
  373. diskDescription = &diskRegistry->Disks[0];
  374. for (i = 0; i < diskRegistry->NumberOfDisks; i++) {
  375. if (diskDescription->Signature == Signature) {
  376. //
  377. // Now locate the partition.
  378. //
  379. for (j = 0; j < diskDescription->NumberOfPartitions; j++) {
  380. diskPartition = &diskDescription->Partitions[j];
  381. if ((StartingOffset.QuadPart == diskPartition->StartingOffset.QuadPart) &&
  382. (Length.QuadPart == diskPartition->Length.QuadPart)) {
  383. if (diskPartition->FtType == NotAnFtMember) {
  384. //
  385. // Found the affected partition simple partition
  386. // i.e. not a part of an FT set.
  387. //
  388. writeRegistry= TRUE;
  389. if (DriveLetter == ' ') {
  390. diskPartition->AssignDriveLetter = FALSE;
  391. } else {
  392. diskPartition->AssignDriveLetter = TRUE;
  393. }
  394. diskPartition->DriveLetter = DriveLetter;
  395. } else {
  396. //
  397. // For FT sets work from the FT information area,
  398. // not from this partition location.
  399. //
  400. type = diskPartition->FtType;
  401. group = diskPartition->FtGroup;
  402. if (!regHeader->FtInformationOffset) {
  403. //
  404. // This is really a corrupt hive! The partition
  405. // affected is part of an FT set, but there is no
  406. // FT information.
  407. //
  408. NtClose(handle);
  409. TESTANDFREE(freeToken);
  410. return FALSE;
  411. }
  412. //
  413. // This is an FT set member, must correct the
  414. // drive letter for all FT set members in the
  415. // registry.
  416. //
  417. ftRegistry = (PFT_REGISTRY)
  418. ((PUCHAR)regHeader + regHeader->FtInformationOffset);
  419. ftDescription = &ftRegistry->FtDescription[0];
  420. for (k = 0; k < ftRegistry->NumberOfComponents; k++) {
  421. if (ftDescription->Type == type) {
  422. //
  423. // For each member, chase back to the diskPartition
  424. // information and if this is the correct FtGroup
  425. // update the drive letter.
  426. //
  427. for (l = 0; l < ftDescription->NumberOfMembers; l++) {
  428. ftMember = &ftDescription->FtMemberDescription[l];
  429. diskPartition = (PDISK_PARTITION)
  430. ((PUCHAR)regHeader + ftMember->OffsetToPartitionInfo);
  431. //
  432. // This could be a different FtGroup for the
  433. // same FT type. Check the group before
  434. // changing.
  435. //
  436. if (diskPartition->FtGroup == group) {
  437. writeRegistry= TRUE;
  438. diskPartition->DriveLetter = DriveLetter;
  439. //
  440. // Maintain the AssignDriveLetter flag on
  441. // the zero member of the set only.
  442. //
  443. if (diskPartition->FtMember == 0) {
  444. if (DriveLetter == ' ') {
  445. diskPartition->AssignDriveLetter = FALSE;
  446. } else {
  447. diskPartition->AssignDriveLetter = TRUE;
  448. }
  449. }
  450. } else {
  451. //
  452. // Not the same group, go to the next
  453. // FT set description.
  454. //
  455. break;
  456. }
  457. }
  458. //
  459. // break out to write the registry information
  460. // once the correct set has been found.
  461. //
  462. if (writeRegistry) {
  463. break;
  464. }
  465. }
  466. ftDescription = (PFT_DESCRIPTION)
  467. &ftDescription->FtMemberDescription[ftDescription->NumberOfMembers];
  468. }
  469. //
  470. // If this actually falls through as opposed to the
  471. // break statement in the for loop above, it indicates a
  472. // bad disk information structure.
  473. //
  474. }
  475. //
  476. // Only write this back out if it is believed that things
  477. // worked correctly.
  478. //
  479. if (writeRegistry) {
  480. //
  481. // All done with setting new drive letter in registry.
  482. // Backup the previous value.
  483. //
  484. FtBackup(handle);
  485. //
  486. // Set the new value.
  487. //
  488. status = FtSetValue(handle,
  489. DiskRegistryValue,
  490. regHeader,
  491. sizeof(DISK_CONFIG_HEADER) +
  492. regHeader->DiskInformationSize +
  493. regHeader->FtInformationSize,
  494. REG_BINARY);
  495. NtClose(handle);
  496. TESTANDFREE(freeToken);
  497. return TRUE;
  498. }
  499. }
  500. }
  501. }
  502. //
  503. // Look at the next disk
  504. //
  505. diskDescription = (PDISK_DESCRIPTION)
  506. &diskDescription->Partitions[diskDescription->NumberOfPartitions];
  507. }
  508. return TRUE;
  509. }
  510. NTSTATUS
  511. SpDiskRegistryAssignCdRomLetter(
  512. IN PWSTR CdromName,
  513. IN WCHAR DriveLetter
  514. )
  515. {
  516. NTSTATUS status;
  517. HANDLE handle;
  518. WCHAR newValue[4];
  519. UNICODE_STRING unicodeValueName;
  520. //
  521. // Try to open the key for later use when setting the new value.
  522. //
  523. status = FtOpenKey(&handle,
  524. DiskRegistryKey,
  525. DiskRegistryClass);
  526. if (NT_SUCCESS(status)) {
  527. unicodeValueName.MaximumLength =
  528. unicodeValueName.Length = (wcslen(CdromName) * sizeof(WCHAR)) + sizeof(WCHAR);
  529. unicodeValueName.Buffer = CdromName;
  530. unicodeValueName.Length -= sizeof(WCHAR); // don't count the eos
  531. newValue[0] = DriveLetter;
  532. newValue[1] = (WCHAR) ':';
  533. newValue[2] = 0;
  534. status = ZwSetValueKey(handle,
  535. &unicodeValueName,
  536. 0,
  537. REG_SZ,
  538. &newValue,
  539. 3 * sizeof(WCHAR));
  540. NtClose(handle);
  541. }
  542. return status;
  543. }
  544. //
  545. // This is a modified SppMigrateFtKeys.
  546. //
  547. NTSTATUS
  548. SpMigrateDiskRegistry(
  549. IN HANDLE hDestSystemHive
  550. )
  551. /*++
  552. Routine Description:
  553. Arguments:
  554. hDestSystemHive - Handle to the root of the system hive on the system
  555. being upgraded.
  556. Return Value:
  557. Status value indicating outcome of operation.
  558. --*/
  559. {
  560. NTSTATUS Status;
  561. NTSTATUS SavedStatus;
  562. OBJECT_ATTRIBUTES Obja;
  563. UNICODE_STRING UnicodeString;
  564. PWSTR FtDiskKeys[] = {
  565. L"Disk"
  566. };
  567. WCHAR KeyPath[MAX_PATH];
  568. HANDLE SrcKey;
  569. ULONG i;
  570. SavedStatus = STATUS_SUCCESS;
  571. for( i = 0; i < sizeof(FtDiskKeys)/sizeof(PWSTR); i++ ) {
  572. //
  573. // Open the source key
  574. //
  575. swprintf( KeyPath, L"\\registry\\machine\\system\\%ls", FtDiskKeys[i] );
  576. INIT_OBJA(&Obja,&UnicodeString,KeyPath);
  577. Obja.RootDirectory = NULL;
  578. Status = ZwOpenKey(&SrcKey,KEY_ALL_ACCESS,&Obja);
  579. if( !NT_SUCCESS( Status ) ) {
  580. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: Unable to open %ls on the setup hive. Status = %lx \n", KeyPath, Status));
  581. if( SavedStatus == STATUS_SUCCESS ) {
  582. SavedStatus = Status;
  583. }
  584. continue;
  585. }
  586. Status = SppCopyKeyRecursive( SrcKey,
  587. hDestSystemHive,
  588. NULL,
  589. FtDiskKeys[i],
  590. TRUE,
  591. FALSE
  592. );
  593. if( !NT_SUCCESS( Status ) ) {
  594. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: Unable to migrate %ls to SYSTEM\\%ls. Status = %lx\n", KeyPath, FtDiskKeys[i], Status));
  595. if( SavedStatus == STATUS_SUCCESS ) {
  596. SavedStatus = Status;
  597. }
  598. }
  599. ZwClose( SrcKey );
  600. }
  601. return( SavedStatus );
  602. }
  603. VOID
  604. SpGetPartitionStartingOffsetAndLength(
  605. IN DWORD DiskIndex,
  606. IN PDISK_REGION Region,
  607. IN BOOL ExtendedPartition,
  608. OUT PLARGE_INTEGER Offset,
  609. OUT PLARGE_INTEGER Length
  610. )
  611. {
  612. ULONGLONG bytesPerSector;
  613. bytesPerSector = (ULONGLONG)PartitionedDisks[DiskIndex].HardDisk->Geometry.BytesPerSector;
  614. //
  615. // Calculate Offset and Legnth.
  616. //
  617. Offset -> QuadPart = Region->StartSector * bytesPerSector;
  618. Length -> QuadPart = Region->SectorCount * bytesPerSector;
  619. }
  620. BOOL
  621. SpFillInDiskPartitionStructure (
  622. IN DWORD DiskIndex,
  623. IN PDISK_REGION Region,
  624. IN USHORT LogicalNumber,
  625. IN BOOL ExtendedPartition,
  626. OUT PDISK_PARTITION Partition
  627. )
  628. {
  629. LARGE_INTEGER ftLength;
  630. ftLength.QuadPart = 0;
  631. RtlZeroMemory(Partition, sizeof(DISK_PARTITION));
  632. Partition -> FtType = NotAnFtMember;
  633. //
  634. // Set the offset and length.
  635. //
  636. SpGetPartitionStartingOffsetAndLength(
  637. DiskIndex,
  638. Region,
  639. ExtendedPartition,
  640. &(Partition -> StartingOffset),
  641. &(Partition -> Length)
  642. );
  643. //
  644. // set the Drive Letter to an uninitialized drive letter (for now)
  645. // Note that this is *NOT* Unicode.
  646. //
  647. Partition -> DriveLetter = ' ';
  648. Partition -> AssignDriveLetter = TRUE;
  649. Partition -> Modified = TRUE;
  650. Partition -> ReservedChars[0] = 0;
  651. Partition -> ReservedChars[1] = 0;
  652. Partition -> ReservedChars[2] = 0;
  653. Partition -> ReservedTwoLongs[0] = 0;
  654. Partition -> ReservedTwoLongs[1] = 0;
  655. Partition -> LogicalNumber = LogicalNumber;
  656. return TRUE;
  657. }
  658. //
  659. // cut/copied and modified from SpMigrateFtKeys in spupgcfg.c
  660. //
  661. NTSTATUS
  662. SpCopySetupDiskRegistryToTargetDiskRegistry(
  663. IN HANDLE hDestSystemHive
  664. )
  665. {
  666. NTSTATUS Status;
  667. NTSTATUS SavedStatus;
  668. OBJECT_ATTRIBUTES Obja;
  669. UNICODE_STRING UnicodeString;
  670. PWSTR FtDiskKeys[] = {L"Disk"};
  671. WCHAR KeyPath[MAX_PATH];
  672. HANDLE SrcKey;
  673. ULONG i;
  674. SavedStatus = STATUS_SUCCESS;
  675. for( i = 0; i < sizeof(FtDiskKeys)/sizeof(PWSTR); i++ ) {
  676. //
  677. // Open the source key
  678. //
  679. swprintf( KeyPath, L"\\registry\\machine\\system\\%ls", FtDiskKeys[i] );
  680. INIT_OBJA(&Obja,&UnicodeString,KeyPath);
  681. Obja.RootDirectory = NULL;
  682. Status = ZwOpenKey(&SrcKey,KEY_ALL_ACCESS,&Obja);
  683. if( !NT_SUCCESS( Status ) ) {
  684. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: Unable to open %ls on the setup hive. Status = %lx \n", KeyPath, Status));
  685. if( SavedStatus == STATUS_SUCCESS ) {
  686. SavedStatus = Status;
  687. }
  688. continue;
  689. }
  690. Status = SppCopyKeyRecursive( SrcKey,
  691. hDestSystemHive,
  692. NULL,
  693. FtDiskKeys[i],
  694. TRUE,
  695. FALSE
  696. );
  697. if( !NT_SUCCESS( Status ) ) {
  698. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: Unable to migrate %ls to SYSTEM\\%ls. Status = %lx\n", KeyPath, FtDiskKeys[i], Status));
  699. if( SavedStatus == STATUS_SUCCESS ) {
  700. SavedStatus = Status;
  701. }
  702. }
  703. ZwClose( SrcKey );
  704. }
  705. return( SavedStatus );
  706. }
  707. DWORD
  708. SpDetermineNecessarySizeForDiskRegistry(
  709. VOID
  710. )
  711. {
  712. DWORD rSize;
  713. PDISK_REGION region;
  714. DWORD index;
  715. DWORD partitionCount;
  716. //
  717. // Need one overall DISK_REGISTRY.
  718. //
  719. rSize = sizeof(DISK_REGISTRY);
  720. //
  721. // Need HardDiskCount DISK_DESCRIPTIONS.
  722. //
  723. rSize += sizeof(DISK_DESCRIPTION) * HardDiskCount;
  724. //
  725. // Need One DISK_PARTITION per partition for every disk.
  726. //
  727. for (index = 0, partitionCount = 0;index < HardDiskCount; index++) {
  728. region = SpFirstPartitionedRegion(PartitionedDisks[index].PrimaryDiskRegions, TRUE);
  729. while(region) {
  730. partitionCount++;
  731. region = SpNextPartitionedRegion(region, TRUE);
  732. }
  733. region = SpFirstPartitionedRegion(PartitionedDisks[index].PrimaryDiskRegions, FALSE);
  734. while(region) {
  735. partitionCount++;
  736. region = SpNextPartitionedRegion(region, FALSE);
  737. }
  738. }
  739. rSize += sizeof(DISK_PARTITION) * partitionCount;
  740. return rSize;
  741. }
  742. NTSTATUS
  743. SpDiskRegistrySet(
  744. IN PDISK_REGISTRY Registry
  745. )
  746. {
  747. typedef struct _MEMCHAIN {
  748. PDISK_DESCRIPTION Disk;
  749. PDISK_PARTITION Partition;
  750. ULONG MemberNumber;
  751. PVOID NextMember;
  752. } MEMCHAIN, *PMEMCHAIN;
  753. typedef struct _COMPONENT {
  754. PVOID NextComponent;
  755. PMEMCHAIN MemberChain;
  756. FT_TYPE Type;
  757. ULONG Group;
  758. } COMPONENT, *PCOMPONENT;
  759. NTSTATUS status;
  760. HANDLE handle;
  761. DISK_CONFIG_HEADER regHeader;
  762. PDISK_DESCRIPTION disk;
  763. PDISK_PARTITION partition;
  764. ULONG outer; // outer loop index
  765. ULONG i; // inner loop index
  766. PCOMPONENT ftBase = NULL;
  767. PCOMPONENT ftComponent = NULL;
  768. PCOMPONENT ftLastComponent = NULL;
  769. PMEMCHAIN ftMemChain;
  770. PVOID outBuffer = NULL;
  771. ULONG countFtComponents = 0;
  772. ULONG ftMemberCount = 0;
  773. ULONG ftComponentCount = 0;
  774. PFT_REGISTRY ftRegistry = NULL;
  775. PFT_DESCRIPTION ftComponentDescription = NULL;
  776. PFT_MEMBER_DESCRIPTION ftMember = NULL;
  777. status = FtOpenKey(&handle,
  778. DiskRegistryKey,
  779. DiskRegistryClass);
  780. if (NT_SUCCESS(status)) {
  781. //
  782. // Initialize the registry header.
  783. //
  784. regHeader.Version = DISK_INFORMATION_VERSION;
  785. regHeader.CheckSum = 0;
  786. regHeader.Reserved[0] = 0;
  787. regHeader.Reserved[1] = 0;
  788. regHeader.Reserved[2] = 0;
  789. regHeader.NameOffset = 0;
  790. regHeader.NameSize = 0;
  791. regHeader.FtInformationOffset = 0;
  792. regHeader.FtInformationSize = 0;
  793. regHeader.DiskInformationOffset = sizeof(DISK_CONFIG_HEADER);
  794. //
  795. // Walk through the disk information provided and count FT items.
  796. //
  797. disk = &Registry->Disks[0];
  798. for (outer = 0; outer < Registry->NumberOfDisks; outer++) {
  799. //
  800. // Walk through the partition information.
  801. //
  802. for (i = 0; i < disk->NumberOfPartitions; i++) {
  803. partition = &disk->Partitions[i];
  804. if (partition->FtType != NotAnFtMember) {
  805. //
  806. // Have a member of an FT item.
  807. //
  808. if (ftBase == NULL) {
  809. ftBase = (PCOMPONENT) SpMemAlloc(sizeof(COMPONENT));
  810. if (ftBase == NULL) {
  811. return STATUS_NO_MEMORY;
  812. }
  813. ftBase->Type = partition->FtType;
  814. ftBase->Group = partition->FtGroup;
  815. ftBase->NextComponent = NULL;
  816. ftMemChain = (PMEMCHAIN) SpMemAlloc(sizeof(MEMCHAIN));
  817. if (ftMemChain == NULL) {
  818. return STATUS_NO_MEMORY;
  819. }
  820. ftBase->MemberChain = ftMemChain;
  821. ftMemChain->Disk = disk;
  822. ftMemChain->Partition = partition;
  823. ftMemChain->MemberNumber = partition->FtMember;
  824. ftMemChain->NextMember = NULL;
  825. ftComponentCount++;
  826. ftMemberCount++;
  827. } else {
  828. //
  829. // Search the existing chain to see if this is
  830. // a member of a previously encountered FT component.
  831. //
  832. ftComponent = ftBase;
  833. while (ftComponent) {
  834. if ((ftComponent->Type == partition->FtType) &&
  835. (ftComponent->Group == partition->FtGroup)){
  836. //
  837. // Member of same group.
  838. //
  839. ftMemChain = ftComponent->MemberChain;
  840. //
  841. // Go to end of chain.
  842. //
  843. while (ftMemChain->NextMember != NULL) {
  844. ftMemChain = ftMemChain->NextMember;
  845. }
  846. //
  847. // Add new member at end.
  848. //
  849. ftMemChain->NextMember = (PMEMCHAIN) SpMemAlloc(sizeof(MEMCHAIN));
  850. if (ftMemChain->NextMember == NULL) {
  851. return STATUS_NO_MEMORY;
  852. }
  853. ftMemChain = ftMemChain->NextMember;
  854. ftMemChain->NextMember = NULL;
  855. ftMemChain->Disk = disk;
  856. ftMemChain->Partition = partition;
  857. ftMemChain->MemberNumber = partition->FtMember;
  858. ftMemberCount++;
  859. break;
  860. }
  861. ftLastComponent = ftComponent;
  862. ftComponent = ftComponent->NextComponent;
  863. }
  864. if (ftComponent == NULL) {
  865. //
  866. // New FT component volume.
  867. //
  868. ftComponent = (PCOMPONENT)SpMemAlloc(sizeof(COMPONENT));
  869. if (ftComponent == NULL) {
  870. return STATUS_NO_MEMORY;
  871. }
  872. if (ftLastComponent != NULL) {
  873. ftLastComponent->NextComponent = ftComponent;
  874. }
  875. ftComponent->Type = partition->FtType;
  876. ftComponent->Group = partition->FtGroup;
  877. ftComponent->NextComponent = NULL;
  878. ftMemChain = (PMEMCHAIN) SpMemAlloc(sizeof(MEMCHAIN));
  879. if (ftMemChain == NULL) {
  880. return STATUS_NO_MEMORY;
  881. }
  882. ftComponent->MemberChain = ftMemChain;
  883. ftMemChain->Disk = disk;
  884. ftMemChain->Partition = partition;
  885. ftMemChain->MemberNumber = partition->FtMember;
  886. ftMemChain->NextMember = NULL;
  887. ftComponentCount++;
  888. ftMemberCount++;
  889. }
  890. }
  891. }
  892. }
  893. //
  894. // The next disk description occurs immediately after the
  895. // last partition infomation.
  896. //
  897. disk =(PDISK_DESCRIPTION)&disk->Partitions[i];
  898. }
  899. //
  900. // Update the registry header with the length of the disk information.
  901. //
  902. regHeader.DiskInformationSize = ((PUCHAR)disk - (PUCHAR)Registry);
  903. regHeader.FtInformationOffset = sizeof(DISK_CONFIG_HEADER) +
  904. regHeader.DiskInformationSize;
  905. //
  906. // Now walk the ftBase chain constructed above and build
  907. // the FT component of the registry.
  908. //
  909. if (ftBase != NULL) {
  910. //
  911. // Calculate size needed for the FT portion of the
  912. // registry information.
  913. //
  914. i = (ftMemberCount * sizeof(FT_MEMBER_DESCRIPTION)) +
  915. (ftComponentCount * sizeof(FT_DESCRIPTION)) +
  916. sizeof(FT_REGISTRY);
  917. ftRegistry = (PFT_REGISTRY) SpMemAlloc(i);
  918. if (ftRegistry == NULL) {
  919. return STATUS_NO_MEMORY;
  920. }
  921. ftRegistry->NumberOfComponents = 0;
  922. regHeader.FtInformationSize = i;
  923. //
  924. // Construct FT entries.
  925. //
  926. ftComponentDescription = &ftRegistry->FtDescription[0];
  927. ftComponent = ftBase;
  928. while (ftComponent != NULL) {
  929. ftRegistry->NumberOfComponents++;
  930. ftComponentDescription->FtVolumeState = FtStateOk;
  931. ftComponentDescription->Type = ftComponent->Type;
  932. ftComponentDescription->Reserved = 0;
  933. //
  934. // Sort the member list into the ft registry section.
  935. //
  936. i = 0;
  937. while (1) {
  938. ftMemChain = ftComponent->MemberChain;
  939. while (ftMemChain->MemberNumber != i) {
  940. ftMemChain = ftMemChain->NextMember;
  941. if (ftMemChain == NULL) {
  942. break;
  943. }
  944. }
  945. if (ftMemChain == NULL) {
  946. break;
  947. }
  948. ftMember = &ftComponentDescription->FtMemberDescription[i];
  949. ftMember->State = 0;
  950. ftMember->ReservedShort = 0;
  951. ftMember->Signature = ftMemChain->Disk->Signature;
  952. ftMember->OffsetToPartitionInfo = (ULONG)
  953. ((PUCHAR) ftMemChain->Partition -
  954. (PUCHAR) Registry) +
  955. sizeof(DISK_CONFIG_HEADER);
  956. ftMember->LogicalNumber =
  957. ftMemChain->Partition->LogicalNumber;
  958. i++;
  959. }
  960. ftComponentDescription->NumberOfMembers = (USHORT)i;
  961. //
  962. // Set up base for next registry component.
  963. //
  964. ftComponentDescription = (PFT_DESCRIPTION)
  965. &ftComponentDescription->FtMemberDescription[i];
  966. //
  967. // Move forward on the chain.
  968. //
  969. ftLastComponent = ftComponent;
  970. ftComponent = ftComponent->NextComponent;
  971. //
  972. // Free the member chain and component.
  973. //
  974. ftMemChain = ftLastComponent->MemberChain;
  975. while (ftMemChain != NULL) {
  976. PMEMCHAIN nextChain;
  977. nextChain = ftMemChain->NextMember;
  978. TESTANDFREE(ftMemChain);
  979. ftMemChain = nextChain;
  980. }
  981. TESTANDFREE(ftLastComponent);
  982. }
  983. }
  984. i = regHeader.FtInformationSize +
  985. regHeader.DiskInformationSize +
  986. sizeof(DISK_CONFIG_HEADER);
  987. outBuffer = SpMemAlloc(i);
  988. if (outBuffer == NULL) {
  989. TESTANDFREE(ftRegistry);
  990. return STATUS_NO_MEMORY;
  991. }
  992. //
  993. // Move all of the pieces together.
  994. //
  995. RtlMoveMemory(outBuffer,
  996. &regHeader,
  997. sizeof(DISK_CONFIG_HEADER));
  998. RtlMoveMemory((PUCHAR)outBuffer + sizeof(DISK_CONFIG_HEADER),
  999. Registry,
  1000. regHeader.DiskInformationSize);
  1001. RtlMoveMemory((PUCHAR)outBuffer + regHeader.FtInformationOffset,
  1002. ftRegistry,
  1003. regHeader.FtInformationSize);
  1004. TESTANDFREE(ftRegistry);
  1005. //
  1006. // Backup the previous value.
  1007. //
  1008. FtBackup(handle);
  1009. //
  1010. // Set the new value.
  1011. //
  1012. status = FtSetValue(handle,
  1013. DiskRegistryValue,
  1014. outBuffer,
  1015. sizeof(DISK_CONFIG_HEADER) +
  1016. regHeader.DiskInformationSize +
  1017. regHeader.FtInformationSize,
  1018. REG_BINARY);
  1019. TESTANDFREE(outBuffer);
  1020. ZwFlushKey(handle);
  1021. ZwClose(handle);
  1022. }
  1023. return status;
  1024. }
  1025. BOOL
  1026. SpBuildDiskRegistry(
  1027. VOID
  1028. )
  1029. {
  1030. PDISK_DESCRIPTION curDisk;
  1031. DWORD diskRegistrySize;
  1032. PBYTE curOffset;
  1033. PDISK_REGISTRY diskRegistry;
  1034. PDISK_REGION region;
  1035. DWORD diskIndex;
  1036. USHORT logicalNumber;
  1037. NTSTATUS ntStatus;
  1038. //
  1039. // First, allocate enough space for the DiskRegistry structure.
  1040. //
  1041. diskRegistrySize = SpDetermineNecessarySizeForDiskRegistry();
  1042. diskRegistry = SpMemAlloc(diskRegistrySize);
  1043. if (!diskRegistry) {
  1044. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: Could not allocate enough space to create a disk registry.\n"));
  1045. return FALSE;
  1046. }
  1047. //
  1048. // Set the number of disks in the system into the header.
  1049. //
  1050. diskRegistry -> NumberOfDisks = (USHORT) HardDiskCount;
  1051. diskRegistry -> ReservedShort = 0;
  1052. //
  1053. // Initialize curOffset to the Disks element of diskRegistry.
  1054. //
  1055. curOffset = (PBYTE) diskRegistry -> Disks;
  1056. //
  1057. // Now, enumerate all of these hard disks filling in the information for each of them.
  1058. //
  1059. for (diskIndex = 0;diskIndex < diskRegistry -> NumberOfDisks; diskIndex++) {
  1060. //
  1061. // Claim PDISK_DESCRIPTION worth of data.
  1062. //
  1063. curDisk = (PDISK_DESCRIPTION) curOffset;
  1064. //
  1065. // Set the disk signature and clear the reserved data.
  1066. //
  1067. curDisk -> Signature = PartitionedDisks[diskIndex].HardDisk -> Signature;
  1068. curDisk -> ReservedShort = 0;
  1069. //
  1070. // Initialize the NumberOfPartitions member to 0. This will be bumped for
  1071. // each partition found.
  1072. //
  1073. curDisk -> NumberOfPartitions = 0;
  1074. //
  1075. // Initialize curOffset to the Partitions element of the current disk description.
  1076. //
  1077. curOffset = (PBYTE) curDisk -> Partitions;
  1078. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "Creating Disk Description structure in registry.\n"));
  1079. //
  1080. // Initialize the logical number var for this disk.
  1081. //
  1082. logicalNumber = 1;
  1083. //
  1084. // Enumerate the Primary regions, creating a DISK_PARTITION structure for
  1085. // each partition.
  1086. //
  1087. region = SpFirstPartitionedRegion(PartitionedDisks[diskIndex].PrimaryDiskRegions, TRUE);
  1088. while(region) {
  1089. SpFillInDiskPartitionStructure(
  1090. diskIndex,
  1091. region,
  1092. logicalNumber,
  1093. FALSE,
  1094. (PDISK_PARTITION) curOffset
  1095. );
  1096. //
  1097. // Increment the partition count and set the curOffset to the next
  1098. // free spot.
  1099. //
  1100. curDisk -> NumberOfPartitions++;
  1101. curOffset += sizeof(DISK_PARTITION);
  1102. region = SpNextPartitionedRegion(region, TRUE);
  1103. logicalNumber++;
  1104. }
  1105. //
  1106. // Enumerate the Extended regions, creating a DISK_PARTITION structure for
  1107. // each partition.
  1108. //
  1109. region = SpFirstPartitionedRegion(PartitionedDisks[diskIndex].PrimaryDiskRegions, FALSE);
  1110. while(region) {
  1111. SpFillInDiskPartitionStructure(
  1112. diskIndex,
  1113. region,
  1114. logicalNumber,
  1115. TRUE,
  1116. (PDISK_PARTITION) curOffset
  1117. );
  1118. //
  1119. // Increment the partition count and set the curOffset to the next
  1120. // free spot.
  1121. //
  1122. curDisk -> NumberOfPartitions++;
  1123. curOffset += sizeof(DISK_PARTITION);
  1124. region = SpNextPartitionedRegion(region, FALSE);
  1125. logicalNumber++;
  1126. }
  1127. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "Disk contained %u partitions.\n",curDisk -> NumberOfPartitions));
  1128. }
  1129. //
  1130. // Now that the structure has been built, create its registry key and
  1131. // save it. Then free the associated memory.
  1132. //
  1133. ntStatus = SpDiskRegistrySet(diskRegistry);
  1134. if (!NT_SUCCESS(ntStatus)) {
  1135. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "Could not set the Disk Registry information. %u (%x)\n",ntStatus,ntStatus));
  1136. }
  1137. return NT_SUCCESS(ntStatus);
  1138. }