/*++ Copyright (C) Microsoft Corporation, 1991 - 1999 Module Name: geometry.c Abstract: SCSI disk class driver - this module contains all the code for generating disk geometries. Environment: kernel mode only Notes: Revision History: --*/ #include "disk.h" #include "ntddstor.h" #ifdef DEBUG_USE_WPP #include "geometry.tmh" #endif #if defined(_X86_) || defined(_AMD64_) DISK_GEOMETRY_SOURCE DiskUpdateGeometry( IN PFUNCTIONAL_DEVICE_EXTENSION DeviceExtension ); NTSTATUS DiskUpdateRemovableGeometry ( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension ); VOID DiskScanBusDetectInfo( IN PDRIVER_OBJECT DriverObject, IN HANDLE BusKey ); NTSTATUS DiskSaveBusDetectInfo( IN PDRIVER_OBJECT DriverObject, IN HANDLE TargetKey, IN ULONG DiskNumber ); NTSTATUS DiskSaveGeometryDetectInfo( IN PDRIVER_OBJECT DriverObject, IN HANDLE HardwareKey ); NTSTATUS DiskGetPortGeometry( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension, OUT PDISK_GEOMETRY Geometry ); typedef struct _DISK_DETECT_INFO { BOOLEAN Initialized; ULONG Style; ULONG Signature; ULONG MbrCheckSum; PDEVICE_OBJECT Device; CM_INT13_DRIVE_PARAMETER DriveParameters; } DISK_DETECT_INFO, *PDISK_DETECT_INFO; // // Information about the disk geometries collected and saved into the registry // by NTDETECT.COM or the system firmware. // PDISK_DETECT_INFO DetectInfoList = NULL; ULONG DetectInfoCount = 0; LONG DetectInfoUsedCount = 0; #define GET_STARTING_SECTOR(p) ( \ (ULONG) (p->StartingSectorLsb0) + \ (ULONG) (p->StartingSectorLsb1 << 8 ) + \ (ULONG) (p->StartingSectorMsb0 << 16) + \ (ULONG) (p->StartingSectorMsb1 << 24) ) #define GET_ENDING_S_OF_CHS(p) ( \ (UCHAR) (p->EndingCylinderLsb & 0x3F) ) // // Definitions from hal.h // // // Boot record disk partition table entry structure format // typedef struct _PARTITION_DESCRIPTOR { UCHAR ActiveFlag; UCHAR StartingTrack; UCHAR StartingCylinderLsb; UCHAR StartingCylinderMsb; UCHAR PartitionType; UCHAR EndingTrack; UCHAR EndingCylinderLsb; UCHAR EndingCylinderMsb; UCHAR StartingSectorLsb0; UCHAR StartingSectorLsb1; UCHAR StartingSectorMsb0; UCHAR StartingSectorMsb1; UCHAR PartitionLengthLsb0; UCHAR PartitionLengthLsb1; UCHAR PartitionLengthMsb0; UCHAR PartitionLengthMsb1; } PARTITION_DESCRIPTOR, *PPARTITION_DESCRIPTOR; // // Number of partition table entries // #define NUM_PARTITION_TABLE_ENTRIES 4 // // Partition table record and boot signature offsets in 16-bit words // #define PARTITION_TABLE_OFFSET ( 0x1be / 2) #define BOOT_SIGNATURE_OFFSET ((0x200 / 2) - 1) // // Boot record signature value // #define BOOT_RECORD_SIGNATURE (0xaa55) #ifdef ALLOC_PRAGMA #pragma alloc_text(INIT, DiskSaveDetectInfo) #pragma alloc_text(INIT, DiskScanBusDetectInfo) #pragma alloc_text(INIT, DiskSaveBusDetectInfo) #pragma alloc_text(INIT, DiskSaveGeometryDetectInfo) #pragma alloc_text(PAGE, DiskUpdateGeometry) #pragma alloc_text(PAGE, DiskUpdateRemovableGeometry) #pragma alloc_text(PAGE, DiskGetPortGeometry) #pragma alloc_text(PAGE, DiskIsNT4Geometry) #pragma alloc_text(PAGE, DiskGetDetectInfo) #pragma alloc_text(PAGE, DiskReadSignature) #endif NTSTATUS DiskSaveDetectInfo( PDRIVER_OBJECT DriverObject ) /*++ Routine Description: This routine saves away the firmware information about the disks which has been saved in the registry. It generates a list (DetectInfoList) which contains the disk geometries, signatures & checksums of all drives which were examined by NtDetect. This list is later used to assign geometries to disks as they are initialized. Arguments: DriverObject - the driver being initialized. This is used to get to the hardware database. Return Value: status. --*/ { OBJECT_ATTRIBUTES objectAttributes = {0}; HANDLE hardwareKey; UNICODE_STRING unicodeString; HANDLE busKey; NTSTATUS status; PAGED_CODE(); InitializeObjectAttributes( &objectAttributes, DriverObject->HardwareDatabase, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, NULL, NULL); // // Create the hardware base key. // status = ZwOpenKey(&hardwareKey, KEY_READ, &objectAttributes); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveDetectInfo: Cannot open hardware data. " "Name: %wZ\n", DriverObject->HardwareDatabase)); return status; } status = DiskSaveGeometryDetectInfo(DriverObject, hardwareKey); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveDetectInfo: Can't query configuration data " "(%#08lx)\n", status)); ZwClose(hardwareKey); return status; } // // Open EISA bus key. // RtlInitUnicodeString(&unicodeString, L"EisaAdapter"); InitializeObjectAttributes(&objectAttributes, &unicodeString, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, hardwareKey, NULL); status = ZwOpenKey(&busKey, KEY_READ, &objectAttributes); if(NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskSaveDetectInfo: Opened EisaAdapter key\n")); DiskScanBusDetectInfo(DriverObject, busKey); ZwClose(busKey); } // // Open MultiFunction bus key. // RtlInitUnicodeString(&unicodeString, L"MultifunctionAdapter"); InitializeObjectAttributes(&objectAttributes, &unicodeString, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, hardwareKey, NULL); status = ZwOpenKey(&busKey, KEY_READ, &objectAttributes); if(NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskSaveDetectInfo: Opened MultifunctionAdapter key\n")); DiskScanBusDetectInfo(DriverObject, busKey); ZwClose(busKey); } ZwClose(hardwareKey); return STATUS_SUCCESS; } VOID DiskCleanupDetectInfo( IN PDRIVER_OBJECT DriverObject ) /*++ Routine Description: This routine will cleanup the data structure built by DiskSaveDetectInfo. Arguments: DriverObject - a pointer to the kernel object for this driver. Return Value: none --*/ { UNREFERENCED_PARAMETER(DriverObject); FREE_POOL(DetectInfoList); return; } NTSTATUS DiskSaveGeometryDetectInfo( IN PDRIVER_OBJECT DriverObject, IN HANDLE HardwareKey ) { UNICODE_STRING unicodeString; PKEY_VALUE_FULL_INFORMATION keyData; ULONG length; PCM_FULL_RESOURCE_DESCRIPTOR fullDescriptor; PCM_PARTIAL_RESOURCE_DESCRIPTOR partialDescriptor; PCM_INT13_DRIVE_PARAMETER driveParameters; ULONG numberOfDrives; ULONG i; NTSTATUS status; PAGED_CODE(); UNREFERENCED_PARAMETER(DriverObject); // // Get disk BIOS geometry information. // RtlInitUnicodeString(&unicodeString, L"Configuration Data"); keyData = ExAllocatePoolWithTag(PagedPool, VALUE_BUFFER_SIZE, DISK_TAG_UPDATE_GEOM); if(keyData == NULL) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveGeometryDetectInfo: Can't allocate config " "data buffer\n")); return STATUS_INSUFFICIENT_RESOURCES; } status = ZwQueryValueKey(HardwareKey, &unicodeString, KeyValueFullInformation, keyData, VALUE_BUFFER_SIZE, &length); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveGeometryDetectInfo: Can't query configuration " "data (%#08lx)\n", status)); FREE_POOL(keyData); return status; } // // Extract the resource list out of the key data. // fullDescriptor = (PCM_FULL_RESOURCE_DESCRIPTOR) (((PUCHAR) keyData) + keyData->DataOffset); partialDescriptor = fullDescriptor->PartialResourceList.PartialDescriptors; length = partialDescriptor->u.DeviceSpecificData.DataSize; if((keyData->DataLength < sizeof(CM_FULL_RESOURCE_DESCRIPTOR)) || (fullDescriptor->PartialResourceList.Count == 0) || (partialDescriptor->Type != CmResourceTypeDeviceSpecific) || (length < sizeof(ULONG))) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveGeometryDetectInfo: BIOS header data too small " "or invalid\n")); FREE_POOL(keyData); return STATUS_INVALID_PARAMETER; } // // Point to the BIOS data. THe BIOS data is located after the first // partial Resource list which should be device specific data. // { PUCHAR buffer = (PUCHAR) keyData; buffer += keyData->DataOffset; buffer += sizeof(CM_FULL_RESOURCE_DESCRIPTOR); driveParameters = (PCM_INT13_DRIVE_PARAMETER) buffer; } numberOfDrives = length / sizeof(CM_INT13_DRIVE_PARAMETER); // // Allocate our detect info list now that we know how many entries there // are going to be. No other routine allocates detect info and this is // done out of DriverEntry so we don't need to synchronize it's creation. // length = sizeof(DISK_DETECT_INFO) * numberOfDrives; DetectInfoList = ExAllocatePoolWithTag(PagedPool, length, DISK_TAG_UPDATE_GEOM); if(DetectInfoList == NULL) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveGeometryDetectInfo: Couldn't allocate %x bytes " "for DetectInfoList\n", length)); FREE_POOL(keyData); return STATUS_INSUFFICIENT_RESOURCES; } DetectInfoCount = numberOfDrives; RtlZeroMemory(DetectInfoList, length); // // Copy the information out of the key data and into the list we've // allocated. // for(i = 0; i < numberOfDrives; i++) { #ifdef _MSC_VER #pragma warning(suppress: 6386) // PREFast bug means it doesn't correctly remember the size of DetectInfoList #endif DetectInfoList[i].DriveParameters = driveParameters[i]; } FREE_POOL(keyData); return STATUS_SUCCESS; } VOID DiskScanBusDetectInfo( IN PDRIVER_OBJECT DriverObject, IN HANDLE BusKey ) /*++ Routine Description: The routine queries the registry to determine which disks are visible to the BIOS. If a disk is visable to the BIOS then the geometry information is updated with the disk's signature and MBR checksum. Arguments: DriverObject - the object for this driver. BusKey - handle to the bus key to be enumerated. Return Value: status --*/ { ULONG busNumber; NTSTATUS status; for(busNumber = 0; ; busNumber++) { WCHAR buffer[32] = { 0 }; UNICODE_STRING unicodeString; OBJECT_ATTRIBUTES objectAttributes = {0}; HANDLE spareKey; HANDLE adapterKey; ULONG adapterNumber; TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Scanning bus %d\n", busNumber)); // // Open controller name key. // status = RtlStringCchPrintfW(buffer, sizeof(buffer) / sizeof(buffer[0]) - 1, L"%d", busNumber); if (!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Format symbolic link failed with error: 0x%X\n", status)); break; } RtlInitUnicodeString(&unicodeString, buffer); InitializeObjectAttributes(&objectAttributes, &unicodeString, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, BusKey, NULL); status = ZwOpenKey(&spareKey, KEY_READ, &objectAttributes); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Error %#08lx opening bus " "key %#x\n", status, busNumber)); break; } // // Open up a controller ordinal key. // RtlInitUnicodeString(&unicodeString, L"DiskController"); InitializeObjectAttributes(&objectAttributes, &unicodeString, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, spareKey, NULL); status = ZwOpenKey(&adapterKey, KEY_READ, &objectAttributes); ZwClose(spareKey); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Error %#08lx opening " "DiskController key\n", status)); continue; } for(adapterNumber = 0; ; adapterNumber++) { HANDLE diskKey; ULONG diskNumber; // // Open disk key. // TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Scanning disk key " "%d\\DiskController\\%d\\DiskPeripheral\n", busNumber, adapterNumber)); status = RtlStringCchPrintfW(buffer, sizeof(buffer) / sizeof(buffer[0]) - 1, L"%d\\DiskPeripheral", adapterNumber); if (!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Format symbolic link failed with error: 0x%X\n", status)); break; } RtlInitUnicodeString(&unicodeString, buffer); InitializeObjectAttributes(&objectAttributes, &unicodeString, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, adapterKey, NULL); status = ZwOpenKey(&diskKey, KEY_READ, &objectAttributes); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Error %#08lx opening " "disk key\n", status)); break; } for(diskNumber = 0; ; diskNumber++) { HANDLE targetKey; TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Scanning target key " "%d\\DiskController\\%d\\DiskPeripheral\\%d\n", busNumber, adapterNumber, diskNumber)); status = RtlStringCchPrintfW(buffer, sizeof(buffer) / sizeof(buffer[0]) - 1, L"%d", diskNumber); if (!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Format symbolic link failed with error: 0x%X\n", status)); break; } RtlInitUnicodeString(&unicodeString, buffer); InitializeObjectAttributes(&objectAttributes, &unicodeString, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, diskKey, NULL); status = ZwOpenKey(&targetKey, KEY_READ, &objectAttributes); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskScanBusDetectInfo: Error %#08lx " "opening target key\n", status)); break; } DiskSaveBusDetectInfo(DriverObject, targetKey, diskNumber); ZwClose(targetKey); } ZwClose(diskKey); } ZwClose(adapterKey); } return; } NTSTATUS DiskSaveBusDetectInfo( IN PDRIVER_OBJECT DriverObject, IN HANDLE TargetKey, IN ULONG DiskNumber ) /*++ Routine Description: This routine will transfer the firmware/ntdetect reported information in the specified target key into the appropriate entry in the DetectInfoList. Arguments: DriverObject - the object for this driver. TargetKey - the key for the disk being saved. DiskNumber - the ordinal of the entry in the DiskPeripheral tree for this entry Return Value: status --*/ { PDISK_DETECT_INFO diskInfo; UNICODE_STRING unicodeString; PKEY_VALUE_FULL_INFORMATION keyData; ULONG length; NTSTATUS status; PAGED_CODE(); UNREFERENCED_PARAMETER(DriverObject); if (DiskNumber >= DetectInfoCount) { return STATUS_UNSUCCESSFUL; } diskInfo = &(DetectInfoList[DiskNumber]); if(diskInfo->Initialized) { NT_ASSERT(FALSE); TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveBusDetectInfo: disk entry %#x already has a " "signature of %#08lx and mbr checksum of %#08lx\n", DiskNumber, diskInfo->Signature, diskInfo->MbrCheckSum)); return STATUS_UNSUCCESSFUL; } RtlInitUnicodeString(&unicodeString, L"Identifier"); keyData = ExAllocatePoolWithTag(PagedPool, VALUE_BUFFER_SIZE, DISK_TAG_UPDATE_GEOM); if(keyData == NULL) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveBusDetectInfo: Couldn't allocate space for " "registry data\n")); return STATUS_INSUFFICIENT_RESOURCES; } // // Get disk peripheral identifier. // status = ZwQueryValueKey(TargetKey, &unicodeString, KeyValueFullInformation, keyData, VALUE_BUFFER_SIZE, &length); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveBusDetectInfo: Error %#08lx getting " "Identifier\n", status)); FREE_POOL(keyData); return status; } else if (keyData->DataLength < 9*sizeof(WCHAR)) { // // the data is too short to use (we subtract 9 chars in normal path) // TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveBusDetectInfo: Saved data was invalid, " "not enough data in registry!\n")); FREE_POOL(keyData); return STATUS_UNSUCCESSFUL; } else { UNICODE_STRING identifier; ULONG value; // // Complete unicode string. // identifier.Buffer = (PWSTR) ((PUCHAR)keyData + keyData->DataOffset); identifier.Length = (USHORT) keyData->DataLength; identifier.MaximumLength = (USHORT) keyData->DataLength; // // Get the first value out of the identifier - this will be the MBR // checksum. // status = RtlUnicodeStringToInteger(&identifier, 16, &value); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveBusDetectInfo: Error %#08lx converting " "identifier %wZ into MBR xsum\n", status, &identifier)); FREE_POOL(keyData); return status; } diskInfo->MbrCheckSum = value; // // Shift the string over to get the disk signature // identifier.Buffer += 9; identifier.Length -= 9 * sizeof(WCHAR); identifier.MaximumLength -= 9 * sizeof(WCHAR); status = RtlUnicodeStringToInteger(&identifier, 16, &value); if(!NT_SUCCESS(status)) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskSaveBusDetectInfo: Error %#08lx converting " "identifier %wZ into disk signature\n", status, &identifier)); value = 0; } diskInfo->Signature = value; } // // Here is where we would save away the extended int13 data. // // // Mark this entry as initialized so we can make sure not to do it again. // diskInfo->Initialized = TRUE; FREE_POOL(keyData); return STATUS_SUCCESS; } DISK_GEOMETRY_SOURCE DiskUpdateGeometry( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension ) /*++ Routine Description: This routine checks the DetectInfoList saved away during disk driver init to see if any geometry information was reported for this drive. If the geometry data exists (determined by matching non-zero signatures or non-zero MBR checksums) then it will be saved in the RealGeometry member of the disk data block. ClassReadDriveCapacity MUST be called after calling this routine to update the cylinder count based on the size of the disk and the presence of any disk management software. Arguments: DeviceExtension - Supplies a pointer to the device information for disk. Return Value: Inidicates whether the "RealGeometry" in the data block is now valid. --*/ { PDISK_DATA diskData = FdoExtension->CommonExtension.DriverData; ULONG i; PDISK_DETECT_INFO diskInfo = NULL; BOOLEAN found = FALSE; NTSTATUS status; PAGED_CODE(); NT_ASSERT((FdoExtension->DeviceObject->Characteristics & FILE_REMOVABLE_MEDIA) == 0); // // If we've already set a non-default geometry for this drive then there's // no need to try and update again. // if(diskData->GeometrySource != DiskGeometryUnknown) { return diskData->GeometrySource; } // // Scan through the saved detect info to see if we can find a match // for this device. // for(i = 0; i < DetectInfoCount; i++) { NT_ASSERT(DetectInfoList != NULL); diskInfo = &(DetectInfoList[i]); if((diskData->Mbr.Signature != 0) && (diskData->Mbr.Signature == diskInfo->Signature)) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskUpdateGeometry: found match for signature " "%#08lx\n", diskData->Mbr.Signature)); found = TRUE; break; } else if((diskData->Mbr.Signature == 0) && (diskData->Mbr.MbrCheckSum != 0) && (diskData->Mbr.MbrCheckSum == diskInfo->MbrCheckSum)) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskUpdateGeometry: found match for xsum %#08lx\n", diskData->Mbr.MbrCheckSum)); found = TRUE; break; } } if(found) { ULONG cylinders; ULONG sectorsPerTrack; ULONG tracksPerCylinder; ULONG length; // // Point to the array of drive parameters. // cylinders = diskInfo->DriveParameters.MaxCylinders + 1; sectorsPerTrack = diskInfo->DriveParameters.SectorsPerTrack; tracksPerCylinder = diskInfo->DriveParameters.MaxHeads + 1; // // Since the BIOS may not report the full drive, recalculate the drive // size based on the volume size and the BIOS values for tracks per // cylinder and sectors per track.. // length = tracksPerCylinder * sectorsPerTrack; if (length == 0) { // // The BIOS information is bogus. // TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskUpdateGeometry: H (%d) or S(%d) is zero\n", tracksPerCylinder, sectorsPerTrack)); return DiskGeometryUnknown; } // // since we are copying the structure RealGeometry here, we should // really initialize all the fields, especially since a zero'd // BytesPerSector field would cause a trap in xHalReadPartitionTable() // diskData->RealGeometry = FdoExtension->DiskGeometry; // // Save the geometry information away in the disk data block and // set the bit indicating that we found a valid one. // diskData->RealGeometry.SectorsPerTrack = sectorsPerTrack; diskData->RealGeometry.TracksPerCylinder = tracksPerCylinder; diskData->RealGeometry.Cylinders.QuadPart = (LONGLONG)cylinders; TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskUpdateGeometry: BIOS spt %#x, #heads %#x, " "#cylinders %#x\n", sectorsPerTrack, tracksPerCylinder, cylinders)); diskData->GeometrySource = DiskGeometryFromBios; diskInfo->Device = FdoExtension->DeviceObject; // // Increment the count of used geometry entries. // InterlockedIncrement(&DetectInfoUsedCount); } else { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskUpdateGeometry: no match found for signature %#08lx\n", diskData->Mbr.Signature)); } if(diskData->GeometrySource == DiskGeometryUnknown) { // // We couldn't find a geometry from the BIOS. Check with the port // driver and see if it can provide one. // status = DiskGetPortGeometry(FdoExtension, &(diskData->RealGeometry)); if(NT_SUCCESS(status)) { // // Check the geometry to make sure it's valid. // if((diskData->RealGeometry.TracksPerCylinder * diskData->RealGeometry.SectorsPerTrack) != 0) { diskData->GeometrySource = DiskGeometryFromPort; TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskUpdateGeometry: using Port geometry for disk %#p\n", FdoExtension)); if (diskData->RealGeometry.BytesPerSector == 0) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskDriverReinit: Port driver failed to " "set BytesPerSector in the RealGeometry\n")); diskData->RealGeometry.BytesPerSector = FdoExtension->DiskGeometry.BytesPerSector; if (diskData->RealGeometry.BytesPerSector == 0) { NT_ASSERT(!"BytesPerSector is still zero!"); } } } } } // // If we came up with a "real" geometry for this drive then set it in the // device extension. // if (diskData->GeometrySource != DiskGeometryUnknown) { FdoExtension->DiskGeometry = diskData->RealGeometry; } return diskData->GeometrySource; } NTSTATUS DiskUpdateRemovableGeometry ( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension ) /*++ Routine Description: This routine updates the geometry of the disk. It will query the port driver to see if it can provide any geometry info. If not it will use the current head & sector count. Based on these values & the capacity of the drive as reported by ClassReadDriveCapacity it will determine a new cylinder count for the device. Arguments: Fdo - Supplies the functional device object whos size needs to be updated. Return Value: Returns the status of the opertion. --*/ { PCOMMON_DEVICE_EXTENSION commonExtension = &(FdoExtension->CommonExtension); PDISK_DATA diskData = commonExtension->DriverData; PDISK_GEOMETRY geometry = &(diskData->RealGeometry); NTSTATUS status; PAGED_CODE(); if (FdoExtension->DeviceDescriptor) { NT_ASSERT(FdoExtension->DeviceDescriptor->RemovableMedia); } NT_ASSERT(TEST_FLAG(FdoExtension->DeviceObject->Characteristics, FILE_REMOVABLE_MEDIA)); // // Attempt to determine the disk geometry. First we'll check with the // port driver to see what it suggests for a value. // status = DiskGetPortGeometry(FdoExtension, geometry); if(NT_SUCCESS(status) && ((geometry->TracksPerCylinder * geometry->SectorsPerTrack) != 0)) { FdoExtension->DiskGeometry = (*geometry); } return status; } NTSTATUS DiskGetPortGeometry( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension, OUT PDISK_GEOMETRY Geometry ) /*++ Routine Description: This routine will query the port driver for disk geometry. Some port drivers (in particular IDEPORT) may be able to provide geometry for the device. Arguments: FdoExtension - the device object for the disk. Geometry - a structure to save the geometry information into (if any is available) Return Value: STATUS_SUCCESS if geometry information can be provided or error status indicating why it can't. --*/ { PCOMMON_DEVICE_EXTENSION commonExtension = &(FdoExtension->CommonExtension); PIRP irp; PIO_STACK_LOCATION irpStack; KEVENT event; NTSTATUS status; PAGED_CODE(); // // Build an irp to send IOCTL_DISK_GET_DRIVE_GEOMETRY to the lower driver. // irp = IoAllocateIrp(commonExtension->LowerDeviceObject->StackSize, FALSE); if(irp == NULL) { return STATUS_INSUFFICIENT_RESOURCES; } irpStack = IoGetNextIrpStackLocation(irp); irpStack->MajorFunction = IRP_MJ_DEVICE_CONTROL; irpStack->Parameters.DeviceIoControl.IoControlCode = IOCTL_DISK_GET_DRIVE_GEOMETRY; irpStack->Parameters.DeviceIoControl.OutputBufferLength = sizeof(DISK_GEOMETRY); irp->AssociatedIrp.SystemBuffer = Geometry; KeInitializeEvent(&event, SynchronizationEvent, FALSE); IoSetCompletionRoutine(irp, ClassSignalCompletion, &event, TRUE, TRUE, TRUE); status = IoCallDriver(commonExtension->LowerDeviceObject, irp); if (status == STATUS_PENDING) { KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL); status = irp->IoStatus.Status; } IoFreeIrp(irp); return status; } BOOLEAN DiskIsNT4Geometry( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension ) /*++ Routine Description: The default geometry that was used in partitioning disks under Windows NT 4.0 was Sectors per Track = 0x20 = 32 Tracks per Cylinder = 0x40 = 64 This was changed in Windows 2000 to Sectors per Track = 0x3F = 63 Tracks per Cylinder = 0xFF = 255 If neither the BIOS nor the port driver can report the correct geometry, we will default to the new numbers on such disks. Now LVM uses the geometry when creating logical volumes and dynamic disks. So reporting an incorrect geometry will cause the entire extended partition / dynamic disk to be destroyed In this routine, we will look at the Master Boot Record. In 90% of the cases, the first entry corresponds to a partition that starts on the first track. If this is so, we shall retrieve the logical block address associated with it and calculate the correct geometry. Now, all partitions start on a cylinder boundary. So, for the remaining 10% we will look at the ending CHS number to determine the geometry --*/ { PUSHORT readBuffer = NULL; BOOLEAN bFoundNT4 = FALSE; PAGED_CODE(); readBuffer = ExAllocatePoolWithTag(NonPagedPoolNx, FdoExtension->DiskGeometry.BytesPerSector, DISK_TAG_UPDATE_GEOM); if (readBuffer) { KEVENT event; LARGE_INTEGER diskOffset; IO_STATUS_BLOCK ioStatus = { 0 }; PIRP irp; KeInitializeEvent(&event, NotificationEvent, FALSE); // // Read the Master Boot Record at disk offset 0 // diskOffset.QuadPart = 0; irp = IoBuildSynchronousFsdRequest(IRP_MJ_READ, FdoExtension->DeviceObject, readBuffer, FdoExtension->DiskGeometry.BytesPerSector, &diskOffset, &event, &ioStatus); if (irp) { PIO_STACK_LOCATION irpSp = IoGetNextIrpStackLocation(irp); NTSTATUS status; irpSp->Flags |= SL_OVERRIDE_VERIFY_VOLUME; status = IoCallDriver(FdoExtension->DeviceObject, irp); if (status == STATUS_PENDING) { KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL); status = ioStatus.Status; } if (NT_SUCCESS(status)) { // // Match the boot record signature // if (readBuffer[BOOT_SIGNATURE_OFFSET] == BOOT_RECORD_SIGNATURE) { PPARTITION_DESCRIPTOR partitionTableEntry = (PPARTITION_DESCRIPTOR)&readBuffer[PARTITION_TABLE_OFFSET]; ULONG uCount = 0; // // Walk the entries looking for a clue as to what the geometry might be // for (uCount = 0; uCount < NUM_PARTITION_TABLE_ENTRIES; uCount++) { // // We are only concerned if there might be a logical volume or if this disk is part of a dynamic set // if (IsContainerPartition(partitionTableEntry->PartitionType) || partitionTableEntry->PartitionType == PARTITION_LDM) { // // In 90% of the cases, the first entry corresponds to a partition that starts on the first track // if (partitionTableEntry->StartingTrack == 1 && GET_STARTING_SECTOR(partitionTableEntry) == 0x20) { bFoundNT4 = TRUE; break; } // // In almost every case, the ending CHS number is on a cylinder boundary // if (partitionTableEntry->EndingTrack == 0x3F && GET_ENDING_S_OF_CHS(partitionTableEntry) == 0x20) { bFoundNT4 = TRUE; break; } } partitionTableEntry++; } } else { // // The Master Boot Record is invalid // } } } FREE_POOL(readBuffer); } return bFoundNT4; } NTSTATUS DiskReadDriveCapacity( IN PDEVICE_OBJECT Fdo ) /*++ Routine Description: This routine is used by disk.sys as a wrapper for the classpnp API ClassReadDriveCapacity. It will perform some additional operations to attempt to determine drive geometry before it calls the classpnp version of the routine. For fixed disks this involves calling DiskUpdateGeometry which will check various sources (the BIOS, the port driver) for geometry information. Arguments: Fdo - a pointer to the device object to be checked. Return Value: status of ClassReadDriveCapacity. --*/ { PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension; NTSTATUS status; if (TEST_FLAG(Fdo->Characteristics, FILE_REMOVABLE_MEDIA)) { DiskUpdateRemovableGeometry(fdoExtension); } else { DiskUpdateGeometry(fdoExtension); } status = ClassReadDriveCapacity(Fdo); return status; } VOID DiskDriverReinitialization( IN PDRIVER_OBJECT DriverObject, IN PVOID Nothing, IN ULONG Count ) /*++ Routine Description: This routine will scan through the current list of disks and attempt to match them to any remaining geometry information. This will only be done on the first call to the routine. Note: This routine assumes that the system will not be adding or removing devices during this phase of the init process. This is very likely a bad assumption but it greatly simplifies the code. Arguments: DriverObject - a pointer to the object for the disk driver. Nothing - unused Count - an indication of how many times this routine has been called. Return Value: none --*/ { PDEVICE_OBJECT deviceObject; PFUNCTIONAL_DEVICE_EXTENSION fdoExtension; PDISK_DATA diskData; ULONG unmatchedDiskCount; PDEVICE_OBJECT unmatchedDisk = NULL; ULONG i; PDISK_DETECT_INFO diskInfo = NULL; UNREFERENCED_PARAMETER(Nothing); if(Count != 1) { TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_GENERAL, "DiskDriverReinitialization: ignoring call %d\n", Count)); return; } // // Check to see how many entries in the detect info list have been matched. // If there's only one remaining we'll see if we can find a disk to go with // it. // if(DetectInfoCount == 0) { TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_GENERAL, "DiskDriverReinitialization: no detect info saved\n")); return; } if((DetectInfoCount - DetectInfoUsedCount) != 1) { TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_GENERAL, "DiskDriverReinitialization: %d of %d geometry entries " "used - will not attempt match\n", DetectInfoUsedCount, DetectInfoCount)); return; } // // Scan through the list of disks and see if any of them are missing // geometry information. If there is only one such disk we'll try to // match it to the unmatched geometry. // // // ISSUE-2000/5/24-henrygab - figure out if there's a way to keep // removals from happening while doing this. // unmatchedDiskCount = 0; for(deviceObject = DriverObject->DeviceObject; deviceObject != NULL; #ifdef _MSC_VER #pragma prefast(suppress:28175, "Need to access the opaque field to scan through the list of disks") #endif deviceObject = deviceObject->NextDevice) { fdoExtension = deviceObject->DeviceExtension; if (!fdoExtension->CommonExtension.IsFdo) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskDriverReinit: %#p is not an FDO\n", deviceObject)); continue; } // // If the geometry for this one is already known then skip it. // diskData = fdoExtension->CommonExtension.DriverData; if(diskData->GeometrySource != DiskGeometryUnknown) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskDriverReinit: FDO %#p has a geometry\n", deviceObject)); continue; } TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_GENERAL, "DiskDriverReinit: FDO %#p has no geometry\n", deviceObject)); // // Mark this one as using the default. It's past the time when disk // might blunder across the geometry info. If we set the geometry // from the bios we'll reset this field down below. // diskData->GeometrySource = DiskGeometryFromDefault; // // As long as we've only got one unmatched disk we're fine. // unmatchedDiskCount++; if(unmatchedDiskCount > 1) { NT_ASSERT(unmatchedDisk != NULL); TracePrint((TRACE_LEVEL_WARNING, TRACE_FLAG_GENERAL, "DiskDriverReinit: FDO %#p also has no geometry\n", unmatchedDisk)); unmatchedDisk = NULL; break; } unmatchedDisk = deviceObject; } // // If there's more or less than one ungeometried disk then we can't do // anything about the geometry. // if(unmatchedDiskCount != 1) { TracePrint((TRACE_LEVEL_ERROR, TRACE_FLAG_GENERAL, "DiskDriverReinit: Unable to match geometry\n")); return; } fdoExtension = unmatchedDisk->DeviceExtension; diskData = fdoExtension->CommonExtension.DriverData; TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskDriverReinit: Found possible match\n")); // // Find the geometry which wasn't assigned. // for(i = 0; i < DetectInfoCount; i++) { if(DetectInfoList[i].Device == NULL) { diskInfo = &(DetectInfoList[i]); break; } } if (diskInfo != NULL) { // // Save the geometry information away in the disk data block and // set the bit indicating that we found a valid one. // ULONG cylinders; ULONG sectorsPerTrack; ULONG tracksPerCylinder; ULONG length; // // Point to the array of drive parameters. // cylinders = diskInfo->DriveParameters.MaxCylinders + 1; sectorsPerTrack = diskInfo->DriveParameters.SectorsPerTrack; tracksPerCylinder = diskInfo->DriveParameters.MaxHeads + 1; // // Since the BIOS may not report the full drive, recalculate the drive // size based on the volume size and the BIOS values for tracks per // cylinder and sectors per track.. // length = tracksPerCylinder * sectorsPerTrack; if (length == 0) { // // The BIOS information is bogus. // TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskDriverReinit: H (%d) or S(%d) is zero\n", tracksPerCylinder, sectorsPerTrack)); return; } // // since we are copying the structure RealGeometry here, we should // really initialize all the fields, especially since a zero'd // BytesPerSector field would cause a trap in xHalReadPartitionTable() // diskData->RealGeometry = fdoExtension->DiskGeometry; // // Save the geometry information away in the disk data block and // set the bit indicating that we found a valid one. // diskData->RealGeometry.SectorsPerTrack = sectorsPerTrack; diskData->RealGeometry.TracksPerCylinder = tracksPerCylinder; diskData->RealGeometry.Cylinders.QuadPart = (LONGLONG)cylinders; TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskDriverReinit: BIOS spt %#x, #heads %#x, " "#cylinders %#x\n", sectorsPerTrack, tracksPerCylinder, cylinders)); diskData->GeometrySource = DiskGeometryGuessedFromBios; diskInfo->Device = unmatchedDisk; // // Now copy the geometry over to the fdo extension and call // classpnp to redetermine the disk size and cylinder count. // fdoExtension->DiskGeometry = diskData->RealGeometry; (VOID)ClassReadDriveCapacity(unmatchedDisk); if (diskData->RealGeometry.BytesPerSector == 0) { // // if the BytesPerSector field is set to zero for a disk // listed in the bios, then the system will bugcheck in // xHalReadPartitionTable(). assert here since it is // easier to determine what is happening this way. // NT_ASSERT(!"RealGeometry not set to non-zero bps\n"); } } return; } NTSTATUS DiskGetDetectInfo( IN PFUNCTIONAL_DEVICE_EXTENSION FdoExtension, OUT PDISK_DETECTION_INFO DetectInfo ) /*++ Routine Description: Get the Int13 information from the BIOS DetectInfoList. Arguments: FdoExtension - Supplies a pointer to the FDO extension that we want to obtain the detect information for. DetectInfo - A buffer where the detect information will be copied to. Return Value: NTSTATUS code. --*/ { ULONG i; BOOLEAN found = FALSE; PDISK_DETECT_INFO diskInfo = NULL; PDISK_DATA diskData = FdoExtension->CommonExtension.DriverData; PAGED_CODE (); // // Fail for non-fixed drives. // if (TEST_FLAG (FdoExtension->DeviceObject->Characteristics, FILE_REMOVABLE_MEDIA)) { return STATUS_NOT_SUPPORTED; } // // There is no GPT detection info, so fail this. // if (diskData->PartitionStyle == PARTITION_STYLE_GPT) { return STATUS_NOT_SUPPORTED; } for(i = 0; i < DetectInfoCount; i++) { NT_ASSERT(DetectInfoList != NULL); diskInfo = &(DetectInfoList[i]); if((diskData->Mbr.Signature != 0) && (diskData->Mbr.Signature == diskInfo->Signature)) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskGetDetectInfo: found match for signature " "%#08lx\n", diskData->Mbr.Signature)); found = TRUE; break; } else if((diskData->Mbr.Signature == 0) && (diskData->Mbr.MbrCheckSum != 0) && (diskData->Mbr.MbrCheckSum == diskInfo->MbrCheckSum)) { TracePrint((TRACE_LEVEL_INFORMATION, TRACE_FLAG_GENERAL, "DiskGetDetectInfo: found match for xsum %#08lx\n", diskData->Mbr.MbrCheckSum)); found = TRUE; break; } } if ( found ) { DetectInfo->DetectionType = DetectInt13; DetectInfo->Int13.DriveSelect = diskInfo->DriveParameters.DriveSelect; DetectInfo->Int13.MaxCylinders = diskInfo->DriveParameters.MaxCylinders; DetectInfo->Int13.SectorsPerTrack = diskInfo->DriveParameters.SectorsPerTrack; DetectInfo->Int13.MaxHeads = diskInfo->DriveParameters.MaxHeads; DetectInfo->Int13.NumberDrives = diskInfo->DriveParameters.NumberDrives; RtlZeroMemory (&DetectInfo->ExInt13, sizeof (DetectInfo->ExInt13)); } return (found ? STATUS_SUCCESS : STATUS_UNSUCCESSFUL); } NTSTATUS DiskReadSignature( IN PDEVICE_OBJECT Fdo ) /*++ Routine Description: Read the disks signature from the drive. The signature can be either a MBR signature or a GPT/EFI signature. The low-level signature reading is done by IoReadDiskSignature(). Arguments: Fdo - Pointer to the FDO of a disk to read the signature for. Return Value: NTSTATUS code. --*/ { NTSTATUS Status; PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension; PDISK_DATA diskData = fdoExtension->CommonExtension.DriverData; DISK_SIGNATURE Signature = { 0 }; PAGED_CODE(); Status = IoReadDiskSignature (Fdo, fdoExtension->DiskGeometry.BytesPerSector, &Signature); if (!NT_SUCCESS (Status)) { return Status; } if (Signature.PartitionStyle == PARTITION_STYLE_GPT) { diskData->PartitionStyle = PARTITION_STYLE_GPT; diskData->Efi.DiskId = Signature.Gpt.DiskId; } else if (Signature.PartitionStyle == PARTITION_STYLE_MBR) { diskData->PartitionStyle = PARTITION_STYLE_MBR; diskData->Mbr.Signature = Signature.Mbr.Signature; diskData->Mbr.MbrCheckSum = Signature.Mbr.CheckSum; } else { NT_ASSERT (FALSE); Status = STATUS_UNSUCCESSFUL; } return Status; } #endif // defined(_X86_) || defined(_AMD64_)