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.

822 lines
19 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. arcdisk.c
  5. Abstract:
  6. Provides the routines for collecting the disk information for all the ARC
  7. disks visible in the ARC environment.
  8. Author:
  9. John Vert (jvert) 3-Nov-1993
  10. Revision History:
  11. Vijay Jayaseelan (vijayj) 2-April-2000
  12. - Added EFI partition table support
  13. --*/
  14. #include "bootlib.h"
  15. #ifdef i386
  16. #include "bldrx86.h"
  17. #endif
  18. #if defined(_IA64_)
  19. #include "bldria64.h"
  20. #endif
  21. #ifdef EFI_PARTITION_SUPPORT
  22. //
  23. // EFI partition entries
  24. //
  25. UNALIGNED EFI_PARTITION_ENTRY EfiPartitionBuffer[128] = {0};
  26. #endif
  27. BOOLEAN
  28. BlpEnumerateDisks(
  29. IN PCONFIGURATION_COMPONENT_DATA ConfigData
  30. );
  31. #if defined(_X86_) && !defined(ARCI386)
  32. static
  33. VOID
  34. BlpEnumerateXInt13(
  35. VOID
  36. )
  37. /*++
  38. Routine Description:
  39. This routine will go through all the enumerated disks and record
  40. their ability to support xInt13.
  41. Arguments:
  42. None.
  43. Return Value:
  44. None.
  45. --*/
  46. {
  47. UCHAR Partition[100];
  48. ULONG DiskId;
  49. ARC_STATUS Status;
  50. PARC_DISK_SIGNATURE DiskSignature;
  51. PARC_DISK_INFORMATION DiskInfo;
  52. PLIST_ENTRY Entry;
  53. DiskInfo = BlLoaderBlock->ArcDiskInformation;
  54. Entry = DiskInfo->DiskSignatures.Flink;
  55. while (Entry != &DiskInfo->DiskSignatures) {
  56. DiskSignature = CONTAINING_RECORD(Entry,ARC_DISK_SIGNATURE,ListEntry);
  57. //
  58. // Open partition0 on the disk and get it's device ID.
  59. //
  60. strcpy(Partition, DiskSignature->ArcName);
  61. strcat(Partition, "partition(0)");
  62. Status = ArcOpen(Partition, ArcOpenReadOnly, &DiskId);
  63. if( Status == ESUCCESS ) {
  64. //
  65. // Now we've got the DiskId. Fortunately, someone
  66. // has been keeping track of all the DiskIds on the
  67. // machine and whether or not they've got xint13 support.
  68. // All we need to do now is go lookup our diskid in
  69. // that database and get the xint13 BOOLEAN.
  70. //
  71. DiskSignature->xInt13 = BlFileTable[DiskId].u.DriveContext.xInt13;
  72. //
  73. // We don't need you anymore.
  74. //
  75. ArcClose(DiskId);
  76. } else {
  77. DiskSignature->xInt13 = FALSE;
  78. }
  79. Entry = Entry->Flink;
  80. }
  81. }
  82. #endif // for defined(_X86_) && !defined(ARCI386)
  83. ARC_STATUS
  84. BlGetArcDiskInformation(
  85. BOOLEAN XInt13Support
  86. )
  87. /*++
  88. Routine Description:
  89. Enumerates the ARC disks present in the system and collects the identifying disk
  90. information from each one.
  91. Arguments:
  92. XInt13Support : Indicates whether to find XInt13 support or not
  93. Return Value:
  94. None.
  95. --*/
  96. {
  97. PARC_DISK_INFORMATION DiskInfo;
  98. DiskInfo = BlAllocateHeap(sizeof(ARC_DISK_INFORMATION));
  99. if (DiskInfo==NULL) {
  100. return(ENOMEM);
  101. }
  102. InitializeListHead(&DiskInfo->DiskSignatures);
  103. BlLoaderBlock->ArcDiskInformation = DiskInfo;
  104. BlSearchConfigTree(BlLoaderBlock->ConfigurationRoot,
  105. PeripheralClass,
  106. DiskPeripheral,
  107. (ULONG)-1,
  108. BlpEnumerateDisks);
  109. #if defined(_X86_) && !defined(ARCI386)
  110. //
  111. // Enumerate XInt13 support on X86 only if asked for
  112. //
  113. if (XInt13Support) {
  114. BlpEnumerateXInt13();
  115. }
  116. #endif
  117. return(ESUCCESS);
  118. }
  119. BOOLEAN
  120. BlpEnumerateDisks(
  121. IN PCONFIGURATION_COMPONENT_DATA ConfigData
  122. )
  123. /*++
  124. Routine Description:
  125. Callback routine for enumerating the disks in the ARC firmware tree. It
  126. reads all the necessary information from the disk to uniquely identify
  127. it.
  128. Arguments:
  129. ConfigData - Supplies a pointer to the disk's ARC component data.
  130. Return Value:
  131. TRUE - continue searching
  132. FALSE - stop searching tree.
  133. --*/
  134. {
  135. CHAR DiskName[100];
  136. BlGetPathnameFromComponent(ConfigData, DiskName);
  137. return(BlReadSignature(DiskName,FALSE));
  138. }
  139. BOOLEAN
  140. BlReadSignature(
  141. IN PCHAR DiskName,
  142. IN BOOLEAN IsCdRom
  143. )
  144. /*++
  145. Routine Description:
  146. Given an ARC disk name, reads the MBR and adds its signature to the list of
  147. disks.
  148. Arguments:
  149. Diskname - Supplies the name of the disk.
  150. IsCdRom - Indicates whether the disk is a CD-ROM.
  151. Return Value:
  152. TRUE - Success
  153. FALSE - Failure
  154. --*/
  155. {
  156. PARC_DISK_SIGNATURE Signature;
  157. BOOLEAN Status = FALSE;
  158. Signature = BlAllocateHeap(sizeof(ARC_DISK_SIGNATURE));
  159. if (Signature==NULL) {
  160. return(FALSE);
  161. }
  162. Signature->ArcName = BlAllocateHeap(strlen(DiskName)+2);
  163. if (Signature->ArcName==NULL) {
  164. return(FALSE);
  165. }
  166. #if defined(i386)
  167. Status = BlFindDiskSignature(DiskName, Signature);
  168. #endif
  169. if(!Status) {
  170. Status = BlGetDiskSignature(DiskName, IsCdRom, Signature);
  171. }
  172. if (Status) {
  173. InsertHeadList(&BlLoaderBlock->ArcDiskInformation->DiskSignatures,
  174. &Signature->ListEntry);
  175. }
  176. return(TRUE);
  177. }
  178. BOOLEAN
  179. ArcDiskGPTDiskReadCallback(
  180. ULONGLONG StartingLBA,
  181. ULONG BytesToRead,
  182. PVOID pContext,
  183. UNALIGNED PVOID OutputBuffer
  184. )
  185. /*++
  186. Routine Description:
  187. This routine is a callback for reading data for a routine that
  188. validates the GPT partition table.
  189. NOTE: This routine changes the seek position on disk, and you must seek
  190. back to your original seek position if you plan on reading from the
  191. disk after making this call.
  192. Arguments:
  193. StartingLBA - starting logical block address to read from.
  194. BytesToRead - Indicates how many bytes are to be read.
  195. pContext - context pointer for hte function (in this case, a pointer to the disk id.)
  196. OutputBuffer - a buffer that receives the data. It's assumed that it is at least
  197. BytesToRead big enough.
  198. Return Value:
  199. TRUE - success, data has been read
  200. FALSE - failed, data has not been read.
  201. --*/
  202. {
  203. ARC_STATUS Status;
  204. LARGE_INTEGER SeekPosition;
  205. PUSHORT DataPointer;
  206. ULONG DiskId;
  207. ULONG ReadCount = 0;
  208. DiskId = *((PULONG)pContext);
  209. //
  210. // read the second LBA on the disk
  211. //
  212. SeekPosition.QuadPart = StartingLBA * SECTOR_SIZE;
  213. Status = ArcSeek(DiskId,
  214. &SeekPosition,
  215. SeekAbsolute );
  216. if (Status != ESUCCESS) {
  217. return FALSE;
  218. }
  219. DataPointer = OutputBuffer;
  220. Status = ArcRead(
  221. DiskId,
  222. DataPointer,
  223. BytesToRead,
  224. &ReadCount);
  225. if ((Status == ESUCCESS) && (ReadCount == BytesToRead)) {
  226. return(TRUE);
  227. }
  228. return(FALSE);
  229. }
  230. BOOLEAN
  231. BlGetDiskSignature(
  232. IN PCHAR DiskName,
  233. IN BOOLEAN IsCdRom,
  234. PARC_DISK_SIGNATURE Signature
  235. )
  236. /*++
  237. Routine Description:
  238. This routine gets the NTFT disk signature for a specified partition or
  239. path.
  240. Arguments:
  241. DiskName - Supplies the arcname of the partition or drive.
  242. IsCdRom - Indicates whether the disk is a CD-ROM.
  243. Signature - Returns a full ARC_DISK_SIGNATURE.
  244. Return Value:
  245. TRUE - success, Signature will be filled in.
  246. FALSE - failed, Signature will not be filled in.
  247. --*/
  248. {
  249. UCHAR SectorBuffer[2048+256] = {0};
  250. UCHAR Partition[100];
  251. ULONG DiskId;
  252. ULONG Status;
  253. LARGE_INTEGER SeekValue;
  254. PUCHAR Sector;
  255. ULONG i;
  256. ULONG Sum;
  257. ULONG Count;
  258. ULONG SectorSize;
  259. EFI_PARTITION_TABLE *EfiHdr;
  260. if (IsCdRom) {
  261. SectorSize = 2048;
  262. } else {
  263. SectorSize = 512;
  264. }
  265. #if defined(_i386_)
  266. //
  267. // NTDETECT creates an "eisa(0)..." arcname for detected
  268. // BIOS disks on an EISA machine. Change this to "multi(0)..."
  269. // in order to be consistent with the rest of the system
  270. // (particularly the arcname in boot.ini)
  271. //
  272. if (_strnicmp(DiskName,"eisa",4)==0) {
  273. strcpy(Signature->ArcName,"multi");
  274. strcpy(Partition,"multi");
  275. strcat(Signature->ArcName,DiskName+4);
  276. strcat(Partition,DiskName+4);
  277. } else {
  278. strcpy(Signature->ArcName, DiskName);
  279. strcpy(Partition, DiskName);
  280. }
  281. #else
  282. strcpy(Signature->ArcName, DiskName);
  283. strcpy(Partition, DiskName);
  284. #endif
  285. strcat(Partition, "partition(0)");
  286. Status = ArcOpen(Partition, ArcOpenReadOnly, &DiskId);
  287. if (Status != ESUCCESS) {
  288. return(FALSE);
  289. }
  290. //
  291. // Read in the first sector
  292. //
  293. Sector = ALIGN_BUFFER(SectorBuffer);
  294. if (IsCdRom) {
  295. //
  296. // For a CD-ROM, the interesting data starts at 0x8000.
  297. //
  298. SeekValue.QuadPart = 0x8000;
  299. } else {
  300. SeekValue.QuadPart = 0;
  301. }
  302. Status = ArcSeek(DiskId, &SeekValue, SeekAbsolute);
  303. if (Status == ESUCCESS) {
  304. Status = ArcRead(DiskId,
  305. Sector,
  306. SectorSize,
  307. &Count);
  308. }
  309. if (Status != ESUCCESS) {
  310. ArcClose(DiskId);
  311. return(FALSE);
  312. }
  313. //
  314. // Check to see whether this disk has a valid partition table signature or not.
  315. //
  316. if (((PUSHORT)Sector)[BOOT_SIGNATURE_OFFSET] != BOOT_RECORD_SIGNATURE) {
  317. Signature->ValidPartitionTable = FALSE;
  318. } else {
  319. Signature->ValidPartitionTable = TRUE;
  320. }
  321. Signature->Signature = ((PULONG)Sector)[PARTITION_TABLE_OFFSET/2-1];
  322. //
  323. // compute the checksum
  324. //
  325. Sum = 0;
  326. for (i=0; i<(SectorSize/4); i++) {
  327. Sum += ((PULONG)Sector)[i];
  328. }
  329. Signature->CheckSum = ~Sum + 1;
  330. //
  331. // Check for GPT disk.
  332. //
  333. Signature->IsGpt = FALSE;
  334. if (!IsCdRom) {
  335. SeekValue.QuadPart = 1 * SectorSize;
  336. Status = ArcSeek(DiskId, &SeekValue, SeekAbsolute);
  337. if (Status == ESUCCESS) {
  338. Status = ArcRead(DiskId,
  339. Sector,
  340. SectorSize,
  341. &Count);
  342. if (Status == ESUCCESS) {
  343. ULONG tmpDiskId = DiskId;
  344. //
  345. // verify EFI partition table
  346. //
  347. EfiHdr = (EFI_PARTITION_TABLE *)Sector;
  348. if (BlIsValidGUIDPartitionTable(
  349. EfiHdr,
  350. 1,
  351. &tmpDiskId,
  352. ArcDiskGPTDiskReadCallback)) {
  353. Signature->IsGpt = TRUE;
  354. memcpy(
  355. Signature->GptSignature,
  356. EfiHdr->DiskGuid,
  357. sizeof(EfiHdr->DiskGuid) );
  358. }
  359. }
  360. }
  361. }
  362. ArcClose(DiskId);
  363. return(TRUE);
  364. }
  365. #ifdef EFI_PARTITION_SUPPORT
  366. /*
  367. void DbgOut(PWSTR Str);
  368. //#define DBG_PRINT(x) DbgOut(x);
  369. ULONG BlGetKey();
  370. #if defined(_IA64_)
  371. #define STR_PREFIX L
  372. #define DBG_PRINT(x) DbgOut(x)
  373. #else
  374. #define STR_PREFIX
  375. #define DBG_PRINT(x) \
  376. {\
  377. BlPrint(x); \
  378. while (!BlGetKey()); \
  379. }
  380. #endif // _IA64_
  381. */
  382. #define DBG_PRINT(x)
  383. #define STR_PREFIX
  384. UNALIGNED EFI_PARTITION_ENTRY *
  385. BlLocateGPTPartition(
  386. IN UCHAR PartitionNumber,
  387. IN UCHAR MaxPartitions,
  388. IN PUCHAR ValidPartCount
  389. )
  390. {
  391. UNALIGNED EFI_PARTITION_ENTRY *PartEntry = NULL;
  392. UCHAR NullGuid[16] = {0};
  393. UCHAR PartIdx = 0;
  394. UCHAR PartCount = 0;
  395. #if 0
  396. BlPrint("BlLocateGPTPartition(%d,%d,%d)\r\n",
  397. PartitionNumber,
  398. MaxPartitions,
  399. ValidPartCount ? *ValidPartCount : 0);
  400. while (!BlGetKey());
  401. #endif
  402. if (ARGUMENT_PRESENT(ValidPartCount)) {
  403. PartCount = *ValidPartCount;
  404. }
  405. PartitionNumber++; // convert to one based index
  406. //
  407. // Locate the requested valid partition
  408. //
  409. while ((PartIdx < MaxPartitions) && (PartCount < PartitionNumber)) {
  410. DBG_PRINT(STR_PREFIX"Verifying GPT Partition Entry\r\n");
  411. PartEntry = (UNALIGNED EFI_PARTITION_ENTRY *)(EfiPartitionBuffer + PartIdx);
  412. if ((memcmp(PartEntry->Type, NullGuid, 16)) &&
  413. (memcmp(PartEntry->Id, NullGuid, 16)) &&
  414. (PartEntry->StartingLBA != 0) && (PartEntry->EndingLBA != 0)) {
  415. DBG_PRINT(STR_PREFIX"Found Valid GPT Partition Entry\r\n");
  416. PartCount++;
  417. if (ARGUMENT_PRESENT(ValidPartCount)) {
  418. (*ValidPartCount)++;
  419. }
  420. //
  421. // Get hold of the partition entry
  422. //
  423. if (PartCount == PartitionNumber) {
  424. break;
  425. } else {
  426. PartEntry = NULL;
  427. }
  428. } else {
  429. PartEntry = NULL;
  430. }
  431. PartIdx++;
  432. }
  433. return PartEntry;
  434. }
  435. BOOLEAN
  436. BlDiskOpenGPTDiskReadCallback(
  437. ULONGLONG StartingLBA,
  438. ULONG BytesToRead,
  439. PVOID pContext,
  440. UNALIGNED PVOID OutputBuffer
  441. )
  442. /*++
  443. Routine Description:
  444. This routine is a callback for reading data for a routine that
  445. validates the GPT partition table.
  446. Arguments:
  447. StartingLBA - starting logical block address to read from.
  448. BytesToRead - Indicates how many bytes are to be read.
  449. pContext - context pointer for hte function (in this case, a pointer to the disk id.)
  450. OutputBuffer - a buffer that receives the data. It's assumed that it is at least
  451. BytesToRead big enough.
  452. Return Value:
  453. TRUE - success, data has been read
  454. FALSE - failed, data has not been read.
  455. --*/
  456. {
  457. ARC_STATUS Status;
  458. LARGE_INTEGER SeekPosition;
  459. PUSHORT DataPointer;
  460. ULONG DiskId;
  461. ULONG ReadCount = 0;
  462. DiskId = *((PULONG)pContext);
  463. //
  464. // read the second LBA on the disk
  465. //
  466. SeekPosition.QuadPart = StartingLBA * SECTOR_SIZE;
  467. Status = (BlFileTable[DiskId].DeviceEntryTable->Seek)(DiskId,
  468. &SeekPosition,
  469. SeekAbsolute );
  470. if (Status != ESUCCESS) {
  471. return FALSE;
  472. }
  473. DataPointer = OutputBuffer;
  474. Status = (BlFileTable[DiskId].DeviceEntryTable->Read)(DiskId,
  475. DataPointer,
  476. BytesToRead,
  477. &ReadCount);
  478. if ((Status == ESUCCESS) && (ReadCount == BytesToRead)) {
  479. return(TRUE);
  480. }
  481. return(FALSE);
  482. }
  483. ARC_STATUS
  484. BlOpenGPTDiskPartition(
  485. IN ULONG FileId,
  486. IN ULONG DiskId,
  487. IN UCHAR PartitionNumber
  488. )
  489. {
  490. ARC_STATUS Status;
  491. LARGE_INTEGER SeekPosition;
  492. UCHAR DataBuffer[SECTOR_SIZE * 2] = {0};
  493. ULONG ReadCount = 0;
  494. UCHAR NullGuid[16] = {0};
  495. UCHAR PartIdx;
  496. UCHAR PartCount;
  497. UNALIGNED EFI_PARTITION_TABLE *EfiHdr;
  498. UNALIGNED EFI_PARTITION_ENTRY *PartEntry;
  499. ULONG tmpDiskId = DiskId;
  500. if (PartitionNumber >= 128)
  501. return EINVAL;
  502. DBG_PRINT(STR_PREFIX"Seeking GPT PT\r\n");
  503. //
  504. // read the second LBA on the disk
  505. //
  506. SeekPosition.QuadPart = 1 * SECTOR_SIZE;
  507. Status = (BlFileTable[DiskId].DeviceEntryTable->Seek)(DiskId,
  508. &SeekPosition,
  509. SeekAbsolute );
  510. if (Status != ESUCCESS)
  511. return Status;
  512. DBG_PRINT(STR_PREFIX"Reading GPT PT\r\n");
  513. Status = (BlFileTable[DiskId].DeviceEntryTable->Read)(DiskId,
  514. DataBuffer,
  515. SECTOR_SIZE,
  516. &ReadCount);
  517. if (Status != ESUCCESS)
  518. return Status;
  519. if (ReadCount != SECTOR_SIZE) {
  520. Status = EIO;
  521. return Status;
  522. }
  523. EfiHdr = (UNALIGNED EFI_PARTITION_TABLE *)DataBuffer;
  524. DBG_PRINT(STR_PREFIX"Verifying GPT PT\r\n");
  525. //
  526. // verify EFI partition table
  527. //
  528. if (!BlIsValidGUIDPartitionTable(
  529. EfiHdr,
  530. 1,
  531. &tmpDiskId,
  532. BlDiskOpenGPTDiskReadCallback)) {
  533. Status = EBADF;
  534. return Status;
  535. }
  536. //
  537. // Locate and read the partition entry
  538. // which is requested
  539. //
  540. SeekPosition.QuadPart = EfiHdr->PartitionEntryLBA * SECTOR_SIZE;
  541. DBG_PRINT(STR_PREFIX"Seeking GPT Partition Entries\r\n");
  542. Status = (BlFileTable[DiskId].DeviceEntryTable->Seek)(DiskId,
  543. &SeekPosition,
  544. SeekAbsolute );
  545. if (Status != ESUCCESS)
  546. return Status;
  547. RtlZeroMemory(EfiPartitionBuffer, sizeof(EfiPartitionBuffer));
  548. DBG_PRINT(STR_PREFIX"Reading GPT Partition Entries\r\n");
  549. Status = (BlFileTable[DiskId].DeviceEntryTable->Read)(DiskId,
  550. EfiPartitionBuffer,
  551. sizeof(EfiPartitionBuffer),
  552. &ReadCount);
  553. if (Status != ESUCCESS)
  554. return Status;
  555. if (ReadCount != sizeof(EfiPartitionBuffer)) {
  556. Status = EIO;
  557. return Status;
  558. }
  559. DBG_PRINT(STR_PREFIX"Locating the correct GPT partition entry\r\n");
  560. PartEntry = (UNALIGNED EFI_PARTITION_ENTRY *)BlLocateGPTPartition(PartitionNumber, 128, NULL);
  561. if (PartEntry) {
  562. DBG_PRINT(STR_PREFIX"Verifying GPT Partition Entry\r\n");
  563. if ((memcmp(PartEntry->Type, NullGuid, 16)) &&
  564. (memcmp(PartEntry->Id, NullGuid, 16)) &&
  565. (PartEntry->StartingLBA != 0) && (PartEntry->EndingLBA != 0)) {
  566. PPARTITION_CONTEXT PartContext = &(BlFileTable[FileId].u.PartitionContext);
  567. ULONG SectorCount = (ULONG)(PartEntry->EndingLBA - PartEntry->StartingLBA);
  568. DBG_PRINT(STR_PREFIX"Initializing GPT Partition Entry Context\r\n");
  569. //
  570. // Fill the partition context structure
  571. //
  572. PartContext->PartitionLength.QuadPart = SectorCount * SECTOR_SIZE;
  573. PartContext->StartingSector = (ULONG)(PartEntry->StartingLBA);
  574. PartContext->EndingSector = (ULONG)(PartEntry->EndingLBA);
  575. PartContext->DiskId = (UCHAR)DiskId;
  576. BlFileTable[FileId].Position.QuadPart = 0;
  577. #if 0
  578. BlPrint("GPT Partition opened:L:%ld,%ld:%ld,SS:%ld,ES:%ld\n",
  579. PartitionNumber,
  580. (ULONG)PartContext->PartitionLength.QuadPart,
  581. (ULONG)PartContext->StartingSector,
  582. (ULONG)PartContext->EndingSector);
  583. while (!GET_KEY());
  584. #endif
  585. Status = ESUCCESS;
  586. } else {
  587. Status = EBADF;
  588. }
  589. } else {
  590. Status = EBADF;
  591. }
  592. DBG_PRINT(STR_PREFIX"Returning from BlOpenGPTDiskPartition(...)\r\n");
  593. return Status;
  594. }
  595. #endif // for EFI_PARTITION_SUPPORT