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.

6589 lines
208 KiB

  1. /*++
  2. Copyright (C) Microsoft Corporation, 1991 - 1999
  3. Module Name:
  4. floppy.c
  5. Abstract:
  6. This is the NEC PD756 (aka AT, aka ISA, aka ix86) and Intel 82077
  7. (aka MIPS) floppy diskette driver for NT.
  8. Environment:
  9. Kernel mode only.
  10. --*/
  11. //
  12. // Include files.
  13. //
  14. #include "stdio.h"
  15. #include "ntddk.h" // various NT definitions
  16. #include "ntdddisk.h" // disk device driver I/O control codes
  17. #include "ntddfdc.h" // fdc I/O control codes and parameters
  18. #include "initguid.h"
  19. #include "mountdev.h"
  20. #include "acpiioct.h"
  21. #include <flo_data.h> // this driver's data declarations
  22. //
  23. // This is the actual definition of FloppyDebugLevel.
  24. // Note that it is only defined if this is a "debug"
  25. // build.
  26. //
  27. #if DBG
  28. extern ULONG FloppyDebugLevel = 0;
  29. #endif
  30. #ifdef ALLOC_PRAGMA
  31. #pragma alloc_text(INIT,DriverEntry)
  32. #pragma alloc_text(PAGE,FloppyAddDevice)
  33. #pragma alloc_text(PAGE,FloppyPnp)
  34. #pragma alloc_text(PAGE,FloppyPower)
  35. #pragma alloc_text(PAGE,FlConfigCallBack)
  36. #pragma alloc_text(PAGE,FlInitializeControllerHardware)
  37. #pragma alloc_text(PAGE,FlInterpretError)
  38. #pragma alloc_text(PAGE,FlDatarateSpecifyConfigure)
  39. #pragma alloc_text(PAGE,FlRecalibrateDrive)
  40. #pragma alloc_text(PAGE,FlDetermineMediaType)
  41. #pragma alloc_text(PAGE,FlCheckBootSector)
  42. #pragma alloc_text(PAGE,FlConsolidateMediaTypeWithBootSector)
  43. #pragma alloc_text(PAGE,FlIssueCommand)
  44. #pragma alloc_text(PAGE,FlReadWriteTrack)
  45. #pragma alloc_text(PAGE,FlReadWrite)
  46. #pragma alloc_text(PAGE,FlFormat)
  47. #pragma alloc_text(PAGE,FlFinishOperation)
  48. #pragma alloc_text(PAGE,FlStartDrive)
  49. #pragma alloc_text(PAGE,FloppyThread)
  50. #pragma alloc_text(PAGE,FlAllocateIoBuffer)
  51. #pragma alloc_text(PAGE,FlFreeIoBuffer)
  52. #pragma alloc_text(PAGE,FloppyCreateClose)
  53. #pragma alloc_text(PAGE,FloppyDeviceControl)
  54. #pragma alloc_text(PAGE,FloppyReadWrite)
  55. #pragma alloc_text(PAGE,FlCheckFormatParameters)
  56. #pragma alloc_text(PAGE,FlFdcDeviceIo)
  57. #pragma alloc_text(PAGE,FlHdbit)
  58. #pragma alloc_text(PAGE,FloppySystemControl)
  59. #endif
  60. #ifdef POOL_TAGGING
  61. #ifdef ExAllocatePool
  62. #undef ExAllocatePool
  63. #endif
  64. #define ExAllocatePool(a,b) ExAllocatePoolWithTag(a,b,'polF')
  65. #endif
  66. // #define KEEP_COUNTERS 1
  67. #ifdef KEEP_COUNTERS
  68. ULONG FloppyUsedSeek = 0;
  69. ULONG FloppyNoSeek = 0;
  70. #endif
  71. //
  72. // Used for paging the driver.
  73. //
  74. ULONG PagingReferenceCount = 0;
  75. PFAST_MUTEX PagingMutex = NULL;
  76. NTSTATUS
  77. DriverEntry(
  78. IN PDRIVER_OBJECT DriverObject,
  79. IN PUNICODE_STRING RegistryPath
  80. )
  81. /*++
  82. Routine Description:
  83. This routine is the driver's entry point, called by the I/O system
  84. to load the driver. The driver's entry points are initialized and
  85. a mutex to control paging is initialized.
  86. In DBG mode, this routine also examines the registry for special
  87. debug parameters.
  88. Arguments:
  89. DriverObject - a pointer to the object that represents this device
  90. driver.
  91. RegistryPath - a pointer to this driver's key in the Services tree.
  92. Return Value:
  93. STATUS_SUCCESS unless we can't allocate a mutex.
  94. --*/
  95. {
  96. NTSTATUS ntStatus = STATUS_SUCCESS;
  97. #if DBG
  98. //
  99. // We use this to query into the registry as to whether we
  100. // should break at driver entry.
  101. //
  102. RTL_QUERY_REGISTRY_TABLE paramTable[3];
  103. ULONG zero = 0;
  104. ULONG one = 1;
  105. ULONG debugLevel = 0;
  106. ULONG shouldBreak = 0;
  107. ULONG notConfigurable = 0;
  108. PWCHAR path;
  109. ULONG pathLength;
  110. //
  111. // Since the registry path parameter is a "counted" UNICODE string, it
  112. // might not be zero terminated. For a very short time allocate memory
  113. // to hold the registry path zero terminated so that we can use it to
  114. // delve into the registry.
  115. //
  116. // NOTE NOTE!!!! This is not an architected way of breaking into
  117. // a driver. It happens to work for this driver because the author
  118. // likes to do things this way.
  119. //
  120. pathLength = RegistryPath->Length + sizeof(WCHAR);
  121. if ( path = ExAllocatePool(PagedPool, pathLength) ) {
  122. RtlZeroMemory( &paramTable[0], sizeof(paramTable) );
  123. RtlZeroMemory( path, pathLength);
  124. RtlMoveMemory( path, RegistryPath->Buffer, RegistryPath->Length );
  125. paramTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
  126. paramTable[0].Name = L"BreakOnEntry";
  127. paramTable[0].EntryContext = &shouldBreak;
  128. paramTable[0].DefaultType = REG_DWORD;
  129. paramTable[0].DefaultData = &zero;
  130. paramTable[0].DefaultLength = sizeof(ULONG);
  131. paramTable[1].Flags = RTL_QUERY_REGISTRY_DIRECT;
  132. paramTable[1].Name = L"DebugLevel";
  133. paramTable[1].EntryContext = &debugLevel;
  134. paramTable[1].DefaultType = REG_DWORD;
  135. paramTable[1].DefaultData = &zero;
  136. paramTable[1].DefaultLength = sizeof(ULONG);
  137. if (!NT_SUCCESS(RtlQueryRegistryValues(
  138. RTL_REGISTRY_ABSOLUTE | RTL_REGISTRY_OPTIONAL,
  139. path,
  140. &paramTable[0],
  141. NULL,
  142. NULL))) {
  143. shouldBreak = 0;
  144. debugLevel = 0;
  145. }
  146. ExFreePool( path );
  147. }
  148. FloppyDebugLevel = debugLevel;
  149. if ( shouldBreak ) {
  150. DbgBreakPoint();
  151. }
  152. #endif
  153. FloppyDump(FLOPSHOW, ("Floppy: DriverEntry\n") );
  154. //
  155. // Initialize the driver object with this driver's entry points.
  156. //
  157. DriverObject->MajorFunction[IRP_MJ_CREATE] = FloppyCreateClose;
  158. DriverObject->MajorFunction[IRP_MJ_CLOSE] = FloppyCreateClose;
  159. DriverObject->MajorFunction[IRP_MJ_READ] = FloppyReadWrite;
  160. DriverObject->MajorFunction[IRP_MJ_WRITE] = FloppyReadWrite;
  161. DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = FloppyDeviceControl;
  162. DriverObject->MajorFunction[IRP_MJ_PNP] = FloppyPnp;
  163. DriverObject->MajorFunction[IRP_MJ_POWER] = FloppyPower;
  164. DriverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = FloppySystemControl;
  165. DriverObject->DriverUnload = FloppyUnload;
  166. DriverObject->DriverExtension->AddDevice = FloppyAddDevice;
  167. //
  168. // Allocate and initialize a mutex for paging the driver.
  169. //
  170. PagingMutex = ExAllocatePool( NonPagedPool, sizeof(FAST_MUTEX) );
  171. if ( PagingMutex == NULL ) {
  172. return STATUS_INSUFFICIENT_RESOURCES;
  173. }
  174. ExInitializeFastMutex(PagingMutex);
  175. //
  176. // Now page out the driver and wait for a call to FloppyAddDevice.
  177. //
  178. MmPageEntireDriver(DriverEntry);
  179. DriveMediaLimits =
  180. (IsNEC_98 ? (PDRIVE_MEDIA_LIMITS)&_DriveMediaLimits_NEC98[0] : &_DriveMediaLimits[0]);
  181. DriveMediaConstants =
  182. (IsNEC_98 ? (PDRIVE_MEDIA_CONSTANTS)&_DriveMediaConstants_NEC98[0] : &_DriveMediaConstants[0]);
  183. return ntStatus;
  184. }
  185. VOID
  186. FloppyUnload(
  187. IN PDRIVER_OBJECT DriverObject
  188. )
  189. /*++
  190. Routine Description:
  191. Unload the driver from the system. The paging mutex is freed before
  192. final unload.
  193. Arguments:
  194. DriverObject - a pointer to the object that represents this device
  195. driver.
  196. Return Value:
  197. none
  198. --*/
  199. {
  200. FloppyDump( FLOPSHOW, ("FloppyUnload:\n"));
  201. //
  202. // The device object(s) should all be gone by now.
  203. //
  204. ASSERT( DriverObject->DeviceObject == NULL );
  205. //
  206. // Free the paging mutex that was allocated in DriverEntry.
  207. //
  208. ExFreePool( PagingMutex );
  209. return;
  210. }
  211. NTSTATUS
  212. FloppyAddDevice(
  213. IN PDRIVER_OBJECT DriverObject,
  214. IN OUT PDEVICE_OBJECT PhysicalDeviceObject
  215. )
  216. /*++
  217. Routine Description:
  218. This routine is the driver's pnp add device entry point. It is
  219. called by the pnp manager to initialize the driver.
  220. Add device creates and initializes a device object for this FDO and
  221. attaches to the underlying PDO.
  222. Arguments:
  223. DriverObject - a pointer to the object that represents this device
  224. driver.
  225. PhysicalDeviceObject - a pointer to the underlying PDO to which this
  226. new device will attach.
  227. Return Value:
  228. If we successfully create a device object, STATUS_SUCCESS is
  229. returned. Otherwise, return the appropriate error code.
  230. --*/
  231. {
  232. NTSTATUS ntStatus;
  233. PDEVICE_OBJECT deviceObject;
  234. PDISKETTE_EXTENSION disketteExtension;
  235. FDC_INFO fdcInfo;
  236. UCHAR arcNameBuffer[256];
  237. STRING arcNameString;
  238. WCHAR deviceNameBuffer[20];
  239. UNICODE_STRING deviceName;
  240. ntStatus = STATUS_SUCCESS;
  241. FloppyDump( FLOPSHOW, ("FloppyAddDevice: CreateDeviceObject\n"));
  242. //
  243. // Get some device information from the underlying PDO.
  244. //
  245. fdcInfo.BufferCount = 0;
  246. fdcInfo.BufferSize = 0;
  247. ntStatus = FlFdcDeviceIo( PhysicalDeviceObject,
  248. IOCTL_DISK_INTERNAL_GET_FDC_INFO,
  249. &fdcInfo );
  250. if ( NT_SUCCESS(ntStatus) ) {
  251. USHORT i = 0;
  252. //
  253. // Create a device. We will use the first available device name for
  254. // this device.
  255. //
  256. do {
  257. swprintf( deviceNameBuffer, L"\\Device\\Floppy%d", i++ );
  258. RtlInitUnicodeString( &deviceName, deviceNameBuffer );
  259. ntStatus = IoCreateDevice( DriverObject,
  260. sizeof( DISKETTE_EXTENSION ),
  261. &deviceName,
  262. FILE_DEVICE_DISK,
  263. (FILE_REMOVABLE_MEDIA |
  264. FILE_FLOPPY_DISKETTE |
  265. FILE_DEVICE_SECURE_OPEN),
  266. FALSE,
  267. &deviceObject );
  268. } while ( ntStatus == STATUS_OBJECT_NAME_COLLISION );
  269. if ( NT_SUCCESS(ntStatus) ) {
  270. disketteExtension = (PDISKETTE_EXTENSION)deviceObject->DeviceExtension;
  271. //
  272. // Save the device name.
  273. //
  274. FloppyDump( FLOPSHOW | FLOPPNP,
  275. ("FloppyAddDevice - Device Object Name - %S\n", deviceNameBuffer) );
  276. disketteExtension->DeviceName.Buffer = ExAllocatePool( PagedPool | POOL_COLD_ALLOCATION, deviceName.Length );
  277. if ( disketteExtension->DeviceName.Buffer == NULL ) {
  278. IoDeleteDevice( deviceObject );
  279. return STATUS_INSUFFICIENT_RESOURCES;
  280. }
  281. disketteExtension->DeviceName.Length = 0;
  282. disketteExtension->DeviceName.MaximumLength = deviceName.Length;
  283. RtlCopyUnicodeString( &disketteExtension->DeviceName, &deviceName );
  284. IoGetConfigurationInformation()->FloppyCount++;
  285. //
  286. // Create a symbolic link from the disk name to the corresponding
  287. // ARC name, to be used if we're booting off the disk. This will
  288. // if it's not system initialization time; that's fine. The ARC
  289. // name looks something like \ArcName\multi(0)disk(0)rdisk(0).
  290. //
  291. sprintf( arcNameBuffer,
  292. "%s(%d)disk(%d)fdisk(%d)",
  293. "\\ArcName\\multi",
  294. fdcInfo.BusNumber,
  295. fdcInfo.ControllerNumber,
  296. fdcInfo.PeripheralNumber );
  297. RtlInitString( &arcNameString, arcNameBuffer );
  298. ntStatus = RtlAnsiStringToUnicodeString( &disketteExtension->ArcName,
  299. &arcNameString,
  300. TRUE );
  301. if ( NT_SUCCESS( ntStatus ) ) {
  302. IoAssignArcName( &disketteExtension->ArcName, &deviceName );
  303. }
  304. deviceObject->Flags |= DO_DIRECT_IO | DO_POWER_PAGABLE;
  305. if ( deviceObject->AlignmentRequirement < FILE_WORD_ALIGNMENT ) {
  306. deviceObject->AlignmentRequirement = FILE_WORD_ALIGNMENT;
  307. }
  308. deviceObject->Flags &= ~DO_DEVICE_INITIALIZING;
  309. disketteExtension->DriverObject = DriverObject;
  310. // Set the PDO for use with PlugPlay functions
  311. disketteExtension->UnderlyingPDO = PhysicalDeviceObject;
  312. FloppyDump( FLOPSHOW,
  313. ("FloppyAddDevice: Attaching %p to %p\n",
  314. deviceObject,
  315. PhysicalDeviceObject));
  316. disketteExtension->TargetObject =
  317. IoAttachDeviceToDeviceStack( deviceObject,
  318. PhysicalDeviceObject );
  319. FloppyDump( FLOPSHOW,
  320. ("FloppyAddDevice: TargetObject = %p\n",
  321. disketteExtension->TargetObject) );
  322. KeInitializeSemaphore( &disketteExtension->RequestSemaphore,
  323. 0L,
  324. MAXLONG );
  325. ExInitializeFastMutex( &disketteExtension->PowerDownMutex );
  326. KeInitializeSpinLock( &disketteExtension->ListSpinLock );
  327. ExInitializeFastMutex( &disketteExtension->ThreadReferenceMutex );
  328. ExInitializeFastMutex( &disketteExtension->HoldNewReqMutex );
  329. InitializeListHead( &disketteExtension->ListEntry );
  330. disketteExtension->ThreadReferenceCount = -1;
  331. disketteExtension->IsStarted = FALSE;
  332. disketteExtension->IsRemoved = FALSE;
  333. disketteExtension->HoldNewRequests = FALSE;
  334. InitializeListHead( &disketteExtension->NewRequestQueue );
  335. KeInitializeSpinLock( &disketteExtension->NewRequestQueueSpinLock );
  336. KeInitializeSpinLock( &disketteExtension->FlCancelSpinLock );
  337. KeInitializeEvent(&disketteExtension->QueryPowerEvent,
  338. SynchronizationEvent,
  339. FALSE);
  340. disketteExtension->FloppyControllerAllocated = FALSE;
  341. disketteExtension->ReleaseFdcWithMotorRunning = FALSE;
  342. disketteExtension->DeviceObject = deviceObject;
  343. disketteExtension->IsReadOnly = FALSE;
  344. disketteExtension->MediaType = Undetermined;
  345. disketteExtension->ControllerConfigurable = (IsNEC_98) ? FALSE : TRUE;
  346. }
  347. }
  348. return ntStatus;
  349. }
  350. NTSTATUS
  351. FloppySystemControl(
  352. IN PDEVICE_OBJECT DeviceObject,
  353. IN PIRP Irp
  354. )
  355. /*+++
  356. Routine Description ;
  357. This is the dispatch routine for IRP_MJ_SYSTEM_CONTROL IRPs.
  358. Currently we don't handle it. Just pass it down to the lower
  359. device.
  360. Arguments:
  361. DeviceObject - a pointer to the object that represents the device
  362. Irp - a pointer to the I/O Request Packet for this request.
  363. Return Value :
  364. Status returned by lower device.
  365. --*/
  366. {
  367. PDISKETTE_EXTENSION disketteExtension = DeviceObject->DeviceExtension;
  368. IoSkipCurrentIrpStackLocation(Irp);
  369. return IoCallDriver(disketteExtension->TargetObject, Irp);
  370. }
  371. NTSTATUS
  372. FlConfigCallBack(
  373. IN PVOID Context,
  374. IN PUNICODE_STRING PathName,
  375. IN INTERFACE_TYPE BusType,
  376. IN ULONG BusNumber,
  377. IN PKEY_VALUE_FULL_INFORMATION *BusInformation,
  378. IN CONFIGURATION_TYPE ControllerType,
  379. IN ULONG ControllerNumber,
  380. IN PKEY_VALUE_FULL_INFORMATION *ControllerInformation,
  381. IN CONFIGURATION_TYPE PeripheralType,
  382. IN ULONG PeripheralNumber,
  383. IN PKEY_VALUE_FULL_INFORMATION *PeripheralInformation
  384. )
  385. /*++
  386. Routine Description:
  387. This routine is used to acquire all of the configuration
  388. information for each floppy disk controller and the
  389. peripheral driver attached to that controller.
  390. Arguments:
  391. Context - Pointer to the confuration information we are building
  392. up.
  393. PathName - unicode registry path. Not Used.
  394. BusType - Internal, Isa, ...
  395. BusNumber - Which bus if we are on a multibus system.
  396. BusInformation - Configuration information about the bus. Not Used.
  397. ControllerType - Should always be DiskController.
  398. ControllerNumber - Which controller if there is more than one
  399. controller in the system.
  400. ControllerInformation - Array of pointers to the three pieces of
  401. registry information.
  402. PeripheralType - Should always be FloppyDiskPeripheral.
  403. PeripheralNumber - Which floppy if this controller is maintaining
  404. more than one.
  405. PeripheralInformation - Arrya of pointers to the three pieces of
  406. registry information.
  407. Return Value:
  408. STATUS_SUCCESS if everything went ok, or STATUS_INSUFFICIENT_RESOURCES
  409. if it couldn't map the base csr or acquire the adapter object, or
  410. all of the resource information couldn't be acquired.
  411. --*/
  412. {
  413. //
  414. // So we don't have to typecast the context.
  415. //
  416. PDISKETTE_EXTENSION disketteExtension = Context;
  417. //
  418. // Simple iteration variable.
  419. //
  420. ULONG i;
  421. PCM_FULL_RESOURCE_DESCRIPTOR peripheralData;
  422. NTSTATUS ntStatus;
  423. ASSERT(ControllerType == DiskController);
  424. ASSERT(PeripheralType == FloppyDiskPeripheral);
  425. //
  426. // Check if the infprmation from the registry for this device
  427. // is valid.
  428. //
  429. if (!(((PUCHAR)PeripheralInformation[IoQueryDeviceConfigurationData]) +
  430. PeripheralInformation[IoQueryDeviceConfigurationData]->DataLength)) {
  431. ASSERT(FALSE);
  432. return STATUS_INVALID_PARAMETER;
  433. }
  434. peripheralData = (PCM_FULL_RESOURCE_DESCRIPTOR)
  435. (((PUCHAR)PeripheralInformation[IoQueryDeviceConfigurationData]) +
  436. PeripheralInformation[IoQueryDeviceConfigurationData]->DataOffset);
  437. //
  438. // With Version 2.0 or greater for this resource list, we will get
  439. // the full int13 information for the drive. So get that if available.
  440. //
  441. // Otherwise, the only thing that we want out of the peripheral information
  442. // is the maximum drive capacity.
  443. //
  444. // Drop any information on the floor other than the
  445. // device specfic floppy information.
  446. //
  447. for ( i = 0; i < peripheralData->PartialResourceList.Count; i++ ) {
  448. PCM_PARTIAL_RESOURCE_DESCRIPTOR partial =
  449. &peripheralData->PartialResourceList.PartialDescriptors[i];
  450. if ( partial->Type == CmResourceTypeDeviceSpecific ) {
  451. //
  452. // Point to right after this partial. This will take
  453. // us to the beginning of the "real" device specific.
  454. //
  455. PCM_FLOPPY_DEVICE_DATA fDeviceData;
  456. UCHAR driveType;
  457. PDRIVE_MEDIA_CONSTANTS biosDriveMediaConstants =
  458. &(disketteExtension->BiosDriveMediaConstants);
  459. fDeviceData = (PCM_FLOPPY_DEVICE_DATA)(partial + 1);
  460. //
  461. // Get the driver density
  462. //
  463. switch ( fDeviceData->MaxDensity ) {
  464. case 360: driveType = DRIVE_TYPE_0360; break;
  465. case 1200: driveType = DRIVE_TYPE_1200; break;
  466. case 1185: driveType = DRIVE_TYPE_1200; break;
  467. case 1423: driveType = DRIVE_TYPE_1440; break;
  468. case 1440: driveType = DRIVE_TYPE_1440; break;
  469. case 2880: driveType = DRIVE_TYPE_2880; break;
  470. case 1201: if (IsNEC_98) {
  471. driveType = DRIVE_TYPE_1200_E; break;
  472. } // (IsNEC_98)
  473. default:
  474. FloppyDump(
  475. FLOPDBGP,
  476. ("Floppy: Bad DriveCapacity!\n"
  477. "------ density is %d\n",
  478. fDeviceData->MaxDensity)
  479. );
  480. driveType = DRIVE_TYPE_1200;
  481. FloppyDump(
  482. FLOPDBGP,
  483. ("Floppy: run a setup program to set the floppy\n"
  484. "------ drive type; assuming 1.2mb\n"
  485. "------ (type is %x)\n",fDeviceData->MaxDensity)
  486. );
  487. break;
  488. }
  489. disketteExtension->DriveType = driveType;
  490. //
  491. // Pick up all the default from our own table and override
  492. // with the BIOS information
  493. //
  494. *biosDriveMediaConstants = DriveMediaConstants[
  495. DriveMediaLimits[driveType].HighestDriveMediaType];
  496. //
  497. // If the version is high enough, get the rest of the
  498. // information. DeviceSpecific information with a version >= 2
  499. // should have this information
  500. //
  501. if ( fDeviceData->Version >= 2 ) {
  502. // biosDriveMediaConstants->MediaType =
  503. biosDriveMediaConstants->StepRateHeadUnloadTime =
  504. fDeviceData->StepRateHeadUnloadTime;
  505. biosDriveMediaConstants->HeadLoadTime =
  506. fDeviceData->HeadLoadTime;
  507. biosDriveMediaConstants->MotorOffTime =
  508. fDeviceData->MotorOffTime;
  509. biosDriveMediaConstants->SectorLengthCode =
  510. fDeviceData->SectorLengthCode;
  511. // biosDriveMediaConstants->BytesPerSector =
  512. if (fDeviceData->SectorPerTrack == 0) {
  513. // This is not a valid sector per track value.
  514. // We don't recognize this drive. This bogus
  515. // value is often returned by SCSI floppies.
  516. return STATUS_SUCCESS;
  517. }
  518. if (fDeviceData->MaxDensity == 0 ) {
  519. //
  520. // This values are returned by the LS-120 atapi drive.
  521. // BIOS function 8, in int 13 is returned in bl, which
  522. // is mapped to this field. The LS-120 returns 0x10
  523. // which is mapped to 0. Thats why we wont pick it up
  524. // as a normal floppy.
  525. //
  526. return STATUS_SUCCESS;
  527. }
  528. biosDriveMediaConstants->SectorsPerTrack =
  529. fDeviceData->SectorPerTrack;
  530. biosDriveMediaConstants->ReadWriteGapLength =
  531. fDeviceData->ReadWriteGapLength;
  532. biosDriveMediaConstants->FormatGapLength =
  533. fDeviceData->FormatGapLength;
  534. biosDriveMediaConstants->FormatFillCharacter =
  535. fDeviceData->FormatFillCharacter;
  536. biosDriveMediaConstants->HeadSettleTime =
  537. fDeviceData->HeadSettleTime;
  538. biosDriveMediaConstants->MotorSettleTimeRead =
  539. fDeviceData->MotorSettleTime * 1000 / 8;
  540. biosDriveMediaConstants->MotorSettleTimeWrite =
  541. fDeviceData->MotorSettleTime * 1000 / 8;
  542. if (fDeviceData->MaximumTrackValue == 0) {
  543. // This is not a valid maximum track value.
  544. // We don't recognize this drive. This bogus
  545. // value is often returned by SCSI floppies.
  546. return STATUS_SUCCESS;
  547. }
  548. biosDriveMediaConstants->MaximumTrack =
  549. fDeviceData->MaximumTrackValue;
  550. biosDriveMediaConstants->DataLength =
  551. fDeviceData->DataTransferLength;
  552. }
  553. }
  554. }
  555. return STATUS_SUCCESS;
  556. }
  557. NTSTATUS
  558. FlAcpiConfigureFloppy(
  559. PDISKETTE_EXTENSION DisketteExtension,
  560. PFDC_INFO FdcInfo
  561. )
  562. /*++
  563. Routine Description:
  564. Arguments:
  565. Return Value:
  566. --*/
  567. {
  568. UCHAR driveType;
  569. PDRIVE_MEDIA_CONSTANTS biosDriveMediaConstants =
  570. &(DisketteExtension->BiosDriveMediaConstants);
  571. if ( !FdcInfo->AcpiFdiSupported ) {
  572. return STATUS_UNSUCCESSFUL;
  573. }
  574. //
  575. // Get the driver density
  576. //
  577. // JB:TBD - review this drive type list.
  578. //
  579. switch ( (ACPI_FDI_DEVICE_TYPE)FdcInfo->AcpiFdiData.DeviceType ) {
  580. case Form525Capacity360: driveType = DRIVE_TYPE_0360; break;
  581. case Form525Capacity1200: driveType = DRIVE_TYPE_1200; break;
  582. case Form35Capacity720: driveType = DRIVE_TYPE_0720; break;
  583. case Form35Capacity1440: driveType = DRIVE_TYPE_1440; break;
  584. case Form35Capacity2880: driveType = DRIVE_TYPE_2880; break;
  585. default: driveType = DRIVE_TYPE_1200; break;
  586. }
  587. DisketteExtension->DriveType = driveType;
  588. //
  589. // Pick up all the default from our own table and override
  590. // with the BIOS information
  591. //
  592. *biosDriveMediaConstants = DriveMediaConstants[
  593. DriveMediaLimits[driveType].HighestDriveMediaType];
  594. biosDriveMediaConstants->StepRateHeadUnloadTime = (UCHAR) FdcInfo->AcpiFdiData.StepRateHeadUnloadTime;
  595. biosDriveMediaConstants->HeadLoadTime = (UCHAR) FdcInfo->AcpiFdiData.HeadLoadTime;
  596. biosDriveMediaConstants->MotorOffTime = (UCHAR) FdcInfo->AcpiFdiData.MotorOffTime;
  597. biosDriveMediaConstants->SectorLengthCode = (UCHAR) FdcInfo->AcpiFdiData.SectorLengthCode;
  598. biosDriveMediaConstants->SectorsPerTrack = (UCHAR) FdcInfo->AcpiFdiData.SectorPerTrack;
  599. biosDriveMediaConstants->ReadWriteGapLength = (UCHAR) FdcInfo->AcpiFdiData.ReadWriteGapLength;
  600. biosDriveMediaConstants->FormatGapLength = (UCHAR) FdcInfo->AcpiFdiData.FormatGapLength;
  601. biosDriveMediaConstants->FormatFillCharacter = (UCHAR) FdcInfo->AcpiFdiData.FormatFillCharacter;
  602. biosDriveMediaConstants->HeadSettleTime = (UCHAR) FdcInfo->AcpiFdiData.HeadSettleTime;
  603. biosDriveMediaConstants->MotorSettleTimeRead = (UCHAR) FdcInfo->AcpiFdiData.MotorSettleTime * 1000 / 8;
  604. biosDriveMediaConstants->MotorSettleTimeWrite = (USHORT) FdcInfo->AcpiFdiData.MotorSettleTime * 1000 / 8;
  605. biosDriveMediaConstants->MaximumTrack = (UCHAR) FdcInfo->AcpiFdiData.MaxCylinderNumber;
  606. biosDriveMediaConstants->DataLength = (UCHAR) FdcInfo->AcpiFdiData.DataTransferLength;
  607. return STATUS_SUCCESS;
  608. }
  609. NTSTATUS
  610. FlQueueIrpToThread(
  611. IN OUT PIRP Irp,
  612. IN OUT PDISKETTE_EXTENSION DisketteExtension
  613. )
  614. /*++
  615. Routine Description:
  616. This routine queues the given irp to be serviced by the controller's
  617. thread. If the thread is down then this routine creates the thread.
  618. Arguments:
  619. Irp - Supplies the IRP to queue to the controller's thread.
  620. ControllerData - Supplies the controller data.
  621. Return Value:
  622. May return an error if PsCreateSystemThread fails.
  623. Otherwise returns STATUS_PENDING and marks the IRP pending.
  624. --*/
  625. {
  626. KIRQL oldIrql;
  627. NTSTATUS status;
  628. HANDLE threadHandle;
  629. PIO_STACK_LOCATION irpSp = IoGetCurrentIrpStackLocation( Irp );
  630. //
  631. // Verify if the system is powering down. If so we fail
  632. // the irps.
  633. //
  634. ExAcquireFastMutex(&DisketteExtension->PowerDownMutex);
  635. if (DisketteExtension->PoweringDown == TRUE) {
  636. ExReleaseFastMutex(&DisketteExtension->PowerDownMutex);
  637. FloppyDump( FLOPDBGP,
  638. ("Queue IRP: Bailing out since power irp is waiting.\n"));
  639. Irp->IoStatus.Status = STATUS_POWER_STATE_INVALID;
  640. Irp->IoStatus.Information = 0;
  641. return STATUS_POWER_STATE_INVALID;
  642. }
  643. ExReleaseFastMutex(&DisketteExtension->PowerDownMutex);
  644. FloppyDump( FLOPSHOW, ("Queue IRP: No power irp waiting.\n"));
  645. ExAcquireFastMutex(&DisketteExtension->ThreadReferenceMutex);
  646. if (++(DisketteExtension->ThreadReferenceCount) == 0) {
  647. OBJECT_ATTRIBUTES ObjAttributes;
  648. DisketteExtension->ThreadReferenceCount++;
  649. FloppyResetDriverPaging();
  650. //
  651. // Create the thread.
  652. //
  653. ASSERT(DisketteExtension->FloppyThread == NULL);
  654. InitializeObjectAttributes(&ObjAttributes, NULL,
  655. OBJ_KERNEL_HANDLE,
  656. NULL,
  657. NULL);
  658. status = PsCreateSystemThread(&threadHandle,
  659. (ACCESS_MASK) 0L,
  660. &ObjAttributes,
  661. (HANDLE) 0L,
  662. NULL,
  663. FloppyThread,
  664. DisketteExtension);
  665. if (!NT_SUCCESS(status)) {
  666. DisketteExtension->ThreadReferenceCount = -1;
  667. FloppyPageEntireDriver();
  668. ExReleaseFastMutex(&DisketteExtension->ThreadReferenceMutex);
  669. return status;
  670. }
  671. status = ObReferenceObjectByHandle( threadHandle,
  672. SYNCHRONIZE,
  673. NULL,
  674. KernelMode,
  675. &DisketteExtension->FloppyThread,
  676. NULL );
  677. ASSERT(NT_SUCCESS(status));
  678. ZwClose(threadHandle);
  679. if (!NT_SUCCESS(status)) {
  680. DisketteExtension->ThreadReferenceCount = -1;
  681. DisketteExtension->FloppyThread = NULL;
  682. FloppyPageEntireDriver();
  683. ExReleaseFastMutex(&DisketteExtension->ThreadReferenceMutex);
  684. return status;
  685. }
  686. ExReleaseFastMutex(&DisketteExtension->ThreadReferenceMutex);
  687. } else {
  688. ExReleaseFastMutex(&DisketteExtension->ThreadReferenceMutex);
  689. }
  690. IoMarkIrpPending(Irp);
  691. ExInterlockedInsertTailList(
  692. &DisketteExtension->ListEntry,
  693. &Irp->Tail.Overlay.ListEntry,
  694. &DisketteExtension->ListSpinLock );
  695. KeReleaseSemaphore(
  696. &DisketteExtension->RequestSemaphore,
  697. (KPRIORITY) 0,
  698. 1,
  699. FALSE );
  700. return STATUS_PENDING;
  701. }
  702. NTSTATUS
  703. FloppyCreateClose(
  704. IN PDEVICE_OBJECT DeviceObject,
  705. IN PIRP Irp
  706. )
  707. /*++
  708. Routine Description:
  709. This routine is called only rarely by the I/O system; it's mainly
  710. for layered drivers to call. All it does is complete the IRP
  711. successfully.
  712. Arguments:
  713. DeviceObject - a pointer to the object that represents the device
  714. that I/O is to be done on.
  715. Irp - a pointer to the I/O Request Packet for this request.
  716. Return Value:
  717. Always returns STATUS_SUCCESS, since this is a null operation.
  718. --*/
  719. {
  720. UNREFERENCED_PARAMETER( DeviceObject );
  721. FloppyDump(
  722. FLOPSHOW,
  723. ("FloppyCreateClose...\n")
  724. );
  725. //
  726. // Null operation. Do not give an I/O boost since
  727. // no I/O was actually done. IoStatus.Information should be
  728. // FILE_OPENED for an open; it's undefined for a close.
  729. //
  730. Irp->IoStatus.Status = STATUS_SUCCESS;
  731. Irp->IoStatus.Information = FILE_OPENED;
  732. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  733. return STATUS_SUCCESS;
  734. }
  735. NTSTATUS
  736. FloppyDeviceControl(
  737. IN PDEVICE_OBJECT DeviceObject,
  738. IN PIRP Irp
  739. )
  740. /*++
  741. Routine Description:
  742. This routine is called by the I/O system to perform a device I/O
  743. control function.
  744. Arguments:
  745. DeviceObject - a pointer to the object that represents the device
  746. that I/O is to be done on.
  747. Irp - a pointer to the I/O Request Packet for this request.
  748. Return Value:
  749. STATUS_SUCCESS or STATUS_PENDING if recognized I/O control code,
  750. STATUS_INVALID_DEVICE_REQUEST otherwise.
  751. --*/
  752. {
  753. PIO_STACK_LOCATION irpSp;
  754. PDISKETTE_EXTENSION disketteExtension;
  755. PDISK_GEOMETRY outputBuffer;
  756. NTSTATUS ntStatus;
  757. ULONG outputBufferLength;
  758. UCHAR i;
  759. DRIVE_MEDIA_TYPE lowestDriveMediaType;
  760. DRIVE_MEDIA_TYPE highestDriveMediaType;
  761. ULONG formatExParametersSize;
  762. PFORMAT_EX_PARAMETERS formatExParameters;
  763. FloppyDump( FLOPSHOW, ("FloppyDeviceControl...\n") );
  764. disketteExtension = DeviceObject->DeviceExtension;
  765. irpSp = IoGetCurrentIrpStackLocation( Irp );
  766. //
  767. // We need to check if we are currently holding requests.
  768. //
  769. ExAcquireFastMutex(&(disketteExtension->HoldNewReqMutex));
  770. if ( disketteExtension->HoldNewRequests ) {
  771. //
  772. // Queue request only if this is not an ACPI exec method. There is
  773. // a nasty recursion with ACPI and fdc/flpy that requires that these
  774. // requests get through in order to avoid a deadlock.
  775. //
  776. if ( irpSp->Parameters.DeviceIoControl.IoControlCode != IOCTL_ACPI_ASYNC_EVAL_METHOD ) {
  777. ntStatus = FloppyQueueRequest( disketteExtension, Irp );
  778. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  779. return ntStatus;
  780. }
  781. }
  782. //
  783. // If the device has been removed we will just fail this request outright.
  784. //
  785. if ( disketteExtension->IsRemoved ) {
  786. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  787. Irp->IoStatus.Information = 0;
  788. Irp->IoStatus.Status = STATUS_DELETE_PENDING;
  789. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  790. return STATUS_DELETE_PENDING;
  791. }
  792. //
  793. // If the device hasn't been started we will let the IOCTL through. This
  794. // is another hack for ACPI.
  795. //
  796. if (!disketteExtension->IsStarted) {
  797. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  798. IoSkipCurrentIrpStackLocation( Irp );
  799. return IoCallDriver( disketteExtension->TargetObject, Irp );
  800. }
  801. switch( irpSp->Parameters.DeviceIoControl.IoControlCode ) {
  802. case IOCTL_MOUNTDEV_QUERY_DEVICE_NAME: {
  803. PMOUNTDEV_NAME mountName;
  804. FloppyDump( FLOPSHOW, ("FloppyDeviceControl: IOCTL_MOUNTDEV_QUERY_DEVICE_NAME\n") );
  805. ASSERT(disketteExtension->DeviceName.Buffer);
  806. if ( irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  807. sizeof(MOUNTDEV_NAME) ) {
  808. ntStatus = STATUS_INVALID_PARAMETER;
  809. break;
  810. }
  811. mountName = Irp->AssociatedIrp.SystemBuffer;
  812. mountName->NameLength = disketteExtension->DeviceName.Length;
  813. if ( irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  814. sizeof(USHORT) + mountName->NameLength) {
  815. ntStatus = STATUS_BUFFER_OVERFLOW;
  816. Irp->IoStatus.Information = sizeof(MOUNTDEV_NAME);
  817. break;
  818. }
  819. RtlCopyMemory( mountName->Name, disketteExtension->DeviceName.Buffer,
  820. mountName->NameLength);
  821. ntStatus = STATUS_SUCCESS;
  822. Irp->IoStatus.Information = sizeof(USHORT) + mountName->NameLength;
  823. break;
  824. }
  825. case IOCTL_MOUNTDEV_QUERY_UNIQUE_ID: {
  826. PMOUNTDEV_UNIQUE_ID uniqueId;
  827. FloppyDump( FLOPSHOW, ("FloppyDeviceControl: IOCTL_MOUNTDEV_QUERY_UNIQUE_ID\n") );
  828. if ( !disketteExtension->InterfaceString.Buffer ||
  829. irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  830. sizeof(MOUNTDEV_UNIQUE_ID)) {
  831. ntStatus = STATUS_INVALID_PARAMETER;
  832. break;
  833. }
  834. uniqueId = Irp->AssociatedIrp.SystemBuffer;
  835. uniqueId->UniqueIdLength =
  836. disketteExtension->InterfaceString.Length;
  837. if (irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  838. sizeof(USHORT) + uniqueId->UniqueIdLength) {
  839. ntStatus = STATUS_BUFFER_OVERFLOW;
  840. Irp->IoStatus.Information = sizeof(MOUNTDEV_UNIQUE_ID);
  841. break;
  842. }
  843. RtlCopyMemory( uniqueId->UniqueId,
  844. disketteExtension->InterfaceString.Buffer,
  845. uniqueId->UniqueIdLength );
  846. ntStatus = STATUS_SUCCESS;
  847. Irp->IoStatus.Information = sizeof(USHORT) +
  848. uniqueId->UniqueIdLength;
  849. break;
  850. }
  851. case IOCTL_DISK_FORMAT_TRACKS:
  852. case IOCTL_DISK_FORMAT_TRACKS_EX:
  853. //
  854. // Make sure that we got all the necessary format parameters.
  855. //
  856. if ( irpSp->Parameters.DeviceIoControl.InputBufferLength <
  857. sizeof( FORMAT_PARAMETERS ) ) {
  858. FloppyDump(
  859. FLOPDBGP,
  860. ("Floppy: invalid FORMAT buffer length\n")
  861. );
  862. ntStatus = STATUS_INVALID_PARAMETER;
  863. break;
  864. }
  865. //
  866. // Make sure the parameters we got are reasonable.
  867. //
  868. if ( !FlCheckFormatParameters(
  869. disketteExtension,
  870. (PFORMAT_PARAMETERS) Irp->AssociatedIrp.SystemBuffer ) ) {
  871. FloppyDump(
  872. FLOPDBGP,
  873. ("Floppy: invalid FORMAT parameters\n")
  874. );
  875. ntStatus = STATUS_INVALID_PARAMETER;
  876. break;
  877. }
  878. //
  879. // If this is an EX request then make a couple of extra checks
  880. //
  881. if (irpSp->Parameters.DeviceIoControl.IoControlCode ==
  882. IOCTL_DISK_FORMAT_TRACKS_EX) {
  883. if (irpSp->Parameters.DeviceIoControl.InputBufferLength <
  884. sizeof(FORMAT_EX_PARAMETERS)) {
  885. ntStatus = STATUS_INVALID_PARAMETER;
  886. break;
  887. }
  888. formatExParameters = (PFORMAT_EX_PARAMETERS)
  889. Irp->AssociatedIrp.SystemBuffer;
  890. formatExParametersSize =
  891. FIELD_OFFSET(FORMAT_EX_PARAMETERS, SectorNumber) +
  892. formatExParameters->SectorsPerTrack*sizeof(USHORT);
  893. if (irpSp->Parameters.DeviceIoControl.InputBufferLength <
  894. formatExParametersSize ||
  895. formatExParameters->FormatGapLength >= 0x100 ||
  896. formatExParameters->SectorsPerTrack >= 0x100) {
  897. ntStatus = STATUS_INVALID_PARAMETER;
  898. break;
  899. }
  900. }
  901. //
  902. // Fall through to queue the request.
  903. //
  904. case IOCTL_DISK_CHECK_VERIFY:
  905. case IOCTL_STORAGE_CHECK_VERIFY:
  906. case IOCTL_DISK_GET_DRIVE_GEOMETRY:
  907. case IOCTL_DISK_IS_WRITABLE:
  908. //
  909. // The thread must know which diskette to operate on, but the
  910. // request list only passes the IRP. So we'll stick a pointer
  911. // to the diskette extension in Type3InputBuffer, which is
  912. // a field that isn't used for floppy ioctls.
  913. //
  914. //
  915. // Add the request to the queue, and wake up the thread to
  916. // process it.
  917. //
  918. // irpSp->Parameters.DeviceIoControl.Type3InputBuffer = (PVOID)
  919. // disketteExtension;
  920. FloppyDump(
  921. FLOPIRPPATH,
  922. ("Floppy: Enqueing up IRP: %p\n",Irp)
  923. );
  924. ntStatus = FlQueueIrpToThread(Irp, disketteExtension);
  925. break;
  926. case IOCTL_DISK_GET_MEDIA_TYPES:
  927. case IOCTL_STORAGE_GET_MEDIA_TYPES: {
  928. FloppyDump(
  929. FLOPSHOW,
  930. ("Floppy: IOCTL_DISK_GET_MEDIA_TYPES called\n")
  931. );
  932. lowestDriveMediaType = DriveMediaLimits[
  933. disketteExtension->DriveType].LowestDriveMediaType;
  934. highestDriveMediaType = DriveMediaLimits[
  935. disketteExtension->DriveType].HighestDriveMediaType;
  936. outputBufferLength =
  937. irpSp->Parameters.DeviceIoControl.OutputBufferLength;
  938. //
  939. // Make sure that the input buffer has enough room to return
  940. // at least one descriptions of a supported media type.
  941. //
  942. if ( outputBufferLength < ( sizeof( DISK_GEOMETRY ) ) ) {
  943. FloppyDump(
  944. FLOPDBGP,
  945. ("Floppy: invalid GET_MEDIA_TYPES buffer size\n")
  946. );
  947. ntStatus = STATUS_BUFFER_TOO_SMALL;
  948. break;
  949. }
  950. //
  951. // Assume success, although we might modify it to a buffer
  952. // overflow warning below (if the buffer isn't big enough
  953. // to hold ALL of the media descriptions).
  954. //
  955. ntStatus = STATUS_SUCCESS;
  956. if ( outputBufferLength < ( sizeof( DISK_GEOMETRY ) *
  957. ( highestDriveMediaType - lowestDriveMediaType + 1 ) ) ) {
  958. //
  959. // The buffer is too small for all of the descriptions;
  960. // calculate what CAN fit in the buffer.
  961. //
  962. FloppyDump(
  963. FLOPDBGP,
  964. ("Floppy: GET_MEDIA_TYPES buffer size too small\n")
  965. );
  966. ntStatus = STATUS_BUFFER_OVERFLOW;
  967. highestDriveMediaType =
  968. (DRIVE_MEDIA_TYPE)( ( lowestDriveMediaType - 1 ) +
  969. ( outputBufferLength /
  970. sizeof( DISK_GEOMETRY ) ) );
  971. }
  972. outputBuffer = (PDISK_GEOMETRY) Irp->AssociatedIrp.SystemBuffer;
  973. for (
  974. i = (UCHAR)lowestDriveMediaType;
  975. i <= (UCHAR)highestDriveMediaType;
  976. i++ ) {
  977. outputBuffer->MediaType = DriveMediaConstants[i].MediaType;
  978. outputBuffer->Cylinders.LowPart =
  979. DriveMediaConstants[i].MaximumTrack + 1;
  980. outputBuffer->Cylinders.HighPart = 0;
  981. outputBuffer->TracksPerCylinder =
  982. DriveMediaConstants[i].NumberOfHeads;
  983. outputBuffer->SectorsPerTrack =
  984. DriveMediaConstants[i].SectorsPerTrack;
  985. outputBuffer->BytesPerSector =
  986. DriveMediaConstants[i].BytesPerSector;
  987. FloppyDump(
  988. FLOPSHOW,
  989. ("Floppy: media types supported [%d]\n"
  990. "------- Cylinders low: 0x%x\n"
  991. "------- Cylinders high: 0x%x\n"
  992. "------- Track/Cyl: 0x%x\n"
  993. "------- Sectors/Track: 0x%x\n"
  994. "------- Bytes/Sector: 0x%x\n"
  995. "------- Media Type: %d\n",
  996. i,
  997. outputBuffer->Cylinders.LowPart,
  998. outputBuffer->Cylinders.HighPart,
  999. outputBuffer->TracksPerCylinder,
  1000. outputBuffer->SectorsPerTrack,
  1001. outputBuffer->BytesPerSector,
  1002. outputBuffer->MediaType)
  1003. );
  1004. outputBuffer++;
  1005. Irp->IoStatus.Information += sizeof( DISK_GEOMETRY );
  1006. }
  1007. break;
  1008. }
  1009. case IOCTL_MOUNTDEV_QUERY_SUGGESTED_LINK_NAME: {
  1010. if (IsNEC_98) {
  1011. PMOUNTDEV_SUGGESTED_LINK_NAME suggestedName;
  1012. WCHAR driveLetterNameBuffer[10];
  1013. RTL_QUERY_REGISTRY_TABLE queryTable[2];
  1014. PWSTR valueName;
  1015. UNICODE_STRING driveLetterName;
  1016. FloppyDump(FLOPDBGP,("FLOPPY: IOCTL_MOUNTDEV_QUERY_SUGGESTED_LINK_NAME to device %#08lx"
  1017. " through irp %#08lx\n",
  1018. DeviceObject, Irp));
  1019. if (!(DeviceObject->Characteristics & FILE_REMOVABLE_MEDIA)) {
  1020. ntStatus = STATUS_NOT_FOUND;
  1021. break;
  1022. }
  1023. if (irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  1024. sizeof(MOUNTDEV_SUGGESTED_LINK_NAME)) {
  1025. ntStatus = STATUS_INVALID_PARAMETER;
  1026. break;
  1027. }
  1028. valueName = ExAllocatePool(PagedPool, sizeof(WCHAR) * 64);
  1029. if (!valueName) {
  1030. ntStatus = STATUS_INSUFFICIENT_RESOURCES;
  1031. break;
  1032. }
  1033. RtlZeroMemory( valueName, sizeof(WCHAR) * 64 );
  1034. RtlCopyMemory( valueName,
  1035. disketteExtension->DeviceName.Buffer,
  1036. disketteExtension->DeviceName.Length );
  1037. driveLetterName.Buffer = driveLetterNameBuffer;
  1038. driveLetterName.MaximumLength = 20;
  1039. driveLetterName.Length = 0;
  1040. RtlZeroMemory(queryTable, 2*sizeof(RTL_QUERY_REGISTRY_TABLE));
  1041. queryTable[0].Flags = RTL_QUERY_REGISTRY_REQUIRED |
  1042. RTL_QUERY_REGISTRY_DIRECT;
  1043. queryTable[0].Name = valueName;
  1044. queryTable[0].EntryContext = &driveLetterName;
  1045. ntStatus = RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
  1046. L"\\Registry\\Machine\\System\\DISK",
  1047. queryTable, NULL, NULL);
  1048. if (!NT_SUCCESS(ntStatus)) {
  1049. ExFreePool(valueName);
  1050. break;
  1051. }
  1052. if (driveLetterName.Length != 4 ||
  1053. driveLetterName.Buffer[0] < 'A' ||
  1054. driveLetterName.Buffer[0] > 'Z' ||
  1055. driveLetterName.Buffer[1] != ':') {
  1056. ntStatus = STATUS_NOT_FOUND;
  1057. ExFreePool(valueName);
  1058. break;
  1059. }
  1060. suggestedName = Irp->AssociatedIrp.SystemBuffer;
  1061. suggestedName->UseOnlyIfThereAreNoOtherLinks = TRUE;
  1062. suggestedName->NameLength = 28;
  1063. Irp->IoStatus.Information =
  1064. FIELD_OFFSET(MOUNTDEV_SUGGESTED_LINK_NAME, Name) + 28;
  1065. if (irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  1066. Irp->IoStatus.Information) {
  1067. Irp->IoStatus.Information =
  1068. sizeof(MOUNTDEV_SUGGESTED_LINK_NAME);
  1069. ntStatus = STATUS_BUFFER_OVERFLOW;
  1070. ExFreePool(valueName);
  1071. break;
  1072. }
  1073. RtlDeleteRegistryValue(RTL_REGISTRY_ABSOLUTE,
  1074. L"\\Registry\\Machine\\System\\DISK",
  1075. valueName);
  1076. ExFreePool(valueName);
  1077. RtlCopyMemory(suggestedName->Name, L"\\DosDevices\\", 24);
  1078. suggestedName->Name[12] = driveLetterName.Buffer[0];
  1079. suggestedName->Name[13] = ':';
  1080. break;
  1081. } // (IsNEC_98)
  1082. // Pass this to the default.
  1083. }
  1084. case IOCTL_DISK_SENSE_DEVICE: {
  1085. if (IsNEC_98) {
  1086. FloppyDump(
  1087. FLOPDBGP,
  1088. ("Floppy: SENSE_DEVISE_STATUS \n")
  1089. );
  1090. //
  1091. // Make sure that we got all the necessary IOCTL read write parameters.
  1092. //
  1093. if ( irpSp->Parameters.DeviceIoControl.OutputBufferLength <
  1094. sizeof( SENSE_DEVISE_STATUS_PTOS ) ) {
  1095. FloppyDump(
  1096. FLOPDBGP,
  1097. ("Floppy: invalid SENSE_DEVISE_STATUS buffer length\n")
  1098. );
  1099. ntStatus = STATUS_INVALID_PARAMETER;
  1100. break;
  1101. }
  1102. // irpSp->Parameters.DeviceIoControl.Type3InputBuffer = (PVOID)
  1103. // disketteExtension;
  1104. FloppyDump(
  1105. FLOPIRPPATH,
  1106. ("Floppy: Enqueing up IRP: %p\n",Irp)
  1107. );
  1108. ntStatus = FlQueueIrpToThread(Irp, disketteExtension);
  1109. break;
  1110. } // (IsNEC_98)
  1111. }
  1112. default: {
  1113. //
  1114. // We pass down IOCTL's because ACPI uses this as a communications
  1115. // method. ACPI *should* have used a PNP Interface mechanism, but
  1116. // it's too late now.
  1117. //
  1118. #if 0
  1119. FloppyDump(
  1120. FLOPDBGP,
  1121. ("Floppy: invalid device request %x\n",
  1122. irpSp->Parameters.DeviceIoControl.IoControlCode)
  1123. );
  1124. ntStatus = STATUS_INVALID_DEVICE_REQUEST;
  1125. break;
  1126. #endif
  1127. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1128. IoSkipCurrentIrpStackLocation( Irp );
  1129. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1130. return ntStatus;
  1131. }
  1132. }
  1133. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1134. if ( ntStatus != STATUS_PENDING ) {
  1135. Irp->IoStatus.Status = ntStatus;
  1136. if (!NT_SUCCESS( ntStatus ) &&
  1137. IoIsErrorUserInduced( ntStatus )) {
  1138. IoSetHardErrorOrVerifyDevice( Irp, DeviceObject );
  1139. }
  1140. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  1141. }
  1142. return ntStatus;
  1143. }
  1144. NTSTATUS
  1145. FloppyPnp(
  1146. IN PDEVICE_OBJECT DeviceObject,
  1147. IN PIRP Irp
  1148. )
  1149. /*++
  1150. Routine Description:
  1151. Arguments:
  1152. DeviceObject - a pointer to the object that represents the device
  1153. that I/O is to be done on.
  1154. Irp - a pointer to the I/O Request Packet for this request.
  1155. Return Value:
  1156. --*/
  1157. {
  1158. PIO_STACK_LOCATION irpSp;
  1159. PDISKETTE_EXTENSION disketteExtension;
  1160. NTSTATUS ntStatus = STATUS_SUCCESS;
  1161. ULONG i;
  1162. FloppyDump( FLOPSHOW, ("FloppyPnp:\n") );
  1163. //
  1164. // Lock down the driver if it is not already locked.
  1165. //
  1166. FloppyResetDriverPaging();
  1167. disketteExtension = DeviceObject->DeviceExtension;
  1168. irpSp = IoGetCurrentIrpStackLocation( Irp );
  1169. if ( disketteExtension->IsRemoved ) {
  1170. //
  1171. // Since the device is stopped, but we don't hold IRPs,
  1172. // this is a surprise removal. Just fail it.
  1173. //
  1174. Irp->IoStatus.Information = 0;
  1175. Irp->IoStatus.Status = STATUS_DELETE_PENDING;
  1176. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  1177. return STATUS_DELETE_PENDING;
  1178. }
  1179. switch ( irpSp->MinorFunction ) {
  1180. case IRP_MN_START_DEVICE:
  1181. ntStatus = FloppyStartDevice( DeviceObject, Irp );
  1182. break;
  1183. case IRP_MN_QUERY_STOP_DEVICE:
  1184. case IRP_MN_QUERY_REMOVE_DEVICE:
  1185. if ( irpSp->MinorFunction == IRP_MN_QUERY_STOP_DEVICE ) {
  1186. FloppyDump( FLOPPNP,("FloppyPnp: IRP_MN_QUERY_STOP_DEVICE - Irp: %p\n", Irp) );
  1187. } else {
  1188. FloppyDump( FLOPPNP,("FloppyPnp: IRP_MN_QUERY_REMOVE_DEVICE - Irp: %p\n", Irp) );
  1189. }
  1190. if ( !disketteExtension->IsStarted ) {
  1191. //
  1192. // If we aren't started, we'll just pass the irp down.
  1193. //
  1194. IoSkipCurrentIrpStackLocation (Irp);
  1195. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1196. return ntStatus;
  1197. }
  1198. //
  1199. // Hold all new requests.
  1200. //
  1201. ExAcquireFastMutex(&(disketteExtension->HoldNewReqMutex));
  1202. disketteExtension->HoldNewRequests = TRUE;
  1203. //
  1204. // Queue this irp to the floppy thread, this will shutdown the
  1205. // floppy thread without waiting for the typical 3 second motor
  1206. // timeout.
  1207. //
  1208. ntStatus = FlQueueIrpToThread( Irp, disketteExtension );
  1209. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1210. //
  1211. // Wait for the floppy thread to finish. This could take a few hundred
  1212. // milliseconds if the motor needs to be shut down.
  1213. //
  1214. if ( ntStatus == STATUS_PENDING ) {
  1215. ASSERT(disketteExtension->FloppyThread != NULL);
  1216. FlTerminateFloppyThread(disketteExtension);
  1217. Irp->IoStatus.Status = STATUS_SUCCESS;
  1218. IoSkipCurrentIrpStackLocation( Irp );
  1219. IoCallDriver( disketteExtension->TargetObject, Irp );
  1220. ntStatus = STATUS_PENDING;
  1221. } else {
  1222. //
  1223. // We failed to either start the thread or get a pointer to the
  1224. // thread object. Either way veto the Query.
  1225. //
  1226. ntStatus = STATUS_UNSUCCESSFUL;
  1227. Irp->IoStatus.Status = ntStatus;
  1228. Irp->IoStatus.Information = 0;
  1229. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  1230. }
  1231. break;
  1232. case IRP_MN_CANCEL_STOP_DEVICE:
  1233. case IRP_MN_CANCEL_REMOVE_DEVICE:
  1234. if ( irpSp->MinorFunction == IRP_MN_CANCEL_STOP_DEVICE ) {
  1235. FloppyDump( FLOPPNP,("FloppyPnp: IRP_MN_CANCEL_STOP_DEVICE - Irp: %p\n", Irp) );
  1236. } else {
  1237. FloppyDump( FLOPPNP,("FloppyPnp: IRP_MN_CANCEL_REMOVE_DEVICE - Irp: %p\n", Irp) );
  1238. }
  1239. if ( !disketteExtension->IsStarted ) {
  1240. //
  1241. // Nothing to do, just pass the irp down:
  1242. // no need to start the device
  1243. //
  1244. // Set Status to SUCCESS before passing the irp down
  1245. //
  1246. Irp->IoStatus.Status = STATUS_SUCCESS;
  1247. IoSkipCurrentIrpStackLocation (Irp);
  1248. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1249. } else {
  1250. KEVENT doneEvent;
  1251. //
  1252. // Set the status to STATUS_SUCCESS
  1253. //
  1254. Irp->IoStatus.Status = STATUS_SUCCESS;
  1255. //
  1256. // We need to wait for the lower drivers to do their job.
  1257. //
  1258. IoCopyCurrentIrpStackLocationToNext (Irp);
  1259. //
  1260. // Clear the event: it will be set in the completion
  1261. // routine.
  1262. //
  1263. KeInitializeEvent( &doneEvent,
  1264. SynchronizationEvent,
  1265. FALSE);
  1266. IoSetCompletionRoutine( Irp,
  1267. FloppyPnpComplete,
  1268. &doneEvent,
  1269. TRUE, TRUE, TRUE );
  1270. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1271. if ( ntStatus == STATUS_PENDING ) {
  1272. KeWaitForSingleObject( &doneEvent,
  1273. Executive,
  1274. KernelMode,
  1275. FALSE,
  1276. NULL );
  1277. ntStatus = Irp->IoStatus.Status;
  1278. }
  1279. ExAcquireFastMutex(&(disketteExtension->HoldNewReqMutex));
  1280. disketteExtension->HoldNewRequests = FALSE;
  1281. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1282. //
  1283. // Process the queued requests
  1284. //
  1285. FloppyProcessQueuedRequests( disketteExtension );
  1286. //
  1287. // We must now complete the IRP, since we stopped it in the
  1288. // completetion routine with MORE_PROCESSING_REQUIRED.
  1289. //
  1290. Irp->IoStatus.Status = ntStatus;
  1291. Irp->IoStatus.Information = 0;
  1292. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  1293. }
  1294. break;
  1295. case IRP_MN_STOP_DEVICE:
  1296. FloppyDump( FLOPPNP,("FloppyPnp: IRP_MN_STOP_DEVICE - Irp: %p\n", Irp) );
  1297. disketteExtension->IsStarted = FALSE;
  1298. Irp->IoStatus.Status = STATUS_SUCCESS;
  1299. IoSkipCurrentIrpStackLocation( Irp );
  1300. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1301. break;
  1302. case IRP_MN_REMOVE_DEVICE:
  1303. FloppyDump( FLOPPNP,("FloppyPnp: IRP_MN_REMOVE_DEVICE - Irp: %p\n", Irp) );
  1304. FlTerminateFloppyThread(disketteExtension);
  1305. //
  1306. // We need to mark the fact that we don't hold requests first, since
  1307. // we asserted earlier that we are holding requests only if
  1308. // we're not removed.
  1309. //
  1310. ExAcquireFastMutex(&(disketteExtension->HoldNewReqMutex));
  1311. disketteExtension->HoldNewRequests = FALSE;
  1312. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1313. disketteExtension->IsStarted = FALSE;
  1314. disketteExtension->IsRemoved = TRUE;
  1315. //
  1316. // Here we either have completed all the requests in a personal
  1317. // queue when IRP_MN_QUERY_REMOVE was received, or will have to
  1318. // fail all of them if this is a surprise removal.
  1319. // Note that fdoData->IsRemoved is TRUE, so pSD_ProcessQueuedRequests
  1320. // will simply flush the queue, completing each IRP with
  1321. // STATUS_DELETE_PENDING
  1322. //
  1323. FloppyProcessQueuedRequests( disketteExtension );
  1324. //
  1325. // Forward this Irp to the underlying PDO
  1326. //
  1327. IoSkipCurrentIrpStackLocation( Irp );
  1328. Irp->IoStatus.Status = STATUS_SUCCESS;
  1329. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1330. //
  1331. // Send notification that we are going away.
  1332. //
  1333. if ( disketteExtension->InterfaceString.Buffer != NULL ) {
  1334. IoSetDeviceInterfaceState( &disketteExtension->InterfaceString,
  1335. FALSE);
  1336. RtlFreeUnicodeString( &disketteExtension->InterfaceString );
  1337. RtlInitUnicodeString( &disketteExtension->InterfaceString, NULL );
  1338. }
  1339. RtlFreeUnicodeString( &disketteExtension->DeviceName );
  1340. RtlInitUnicodeString( &disketteExtension->DeviceName, NULL );
  1341. if ( disketteExtension->ArcName.Length != 0 ) {
  1342. IoDeassignArcName( &disketteExtension->ArcName );
  1343. RtlFreeUnicodeString( &disketteExtension->ArcName );
  1344. RtlInitUnicodeString( &disketteExtension->ArcName, NULL );
  1345. }
  1346. //
  1347. // Detatch from the undelying device.
  1348. //
  1349. IoDetachDevice( disketteExtension->TargetObject );
  1350. //
  1351. // And delete the device.
  1352. //
  1353. IoDeleteDevice( DeviceObject );
  1354. IoGetConfigurationInformation()->FloppyCount--;
  1355. break;
  1356. default:
  1357. FloppyDump( FLOPPNP, ("FloppyPnp: Unsupported PNP Request %x - Irp: %p\n",irpSp->MinorFunction, Irp) );
  1358. IoSkipCurrentIrpStackLocation( Irp );
  1359. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1360. }
  1361. //
  1362. // Page out the driver if it is not busy elsewhere.
  1363. //
  1364. FloppyPageEntireDriver();
  1365. return ntStatus;
  1366. }
  1367. NTSTATUS
  1368. FloppyStartDevice(
  1369. IN PDEVICE_OBJECT DeviceObject,
  1370. IN PIRP Irp
  1371. )
  1372. {
  1373. NTSTATUS ntStatus;
  1374. NTSTATUS pnpStatus;
  1375. KEVENT doneEvent;
  1376. FDC_INFO fdcInfo;
  1377. CONFIGURATION_TYPE Dc = DiskController;
  1378. CONFIGURATION_TYPE Fp = FloppyDiskPeripheral;
  1379. PDISKETTE_EXTENSION disketteExtension = (PDISKETTE_EXTENSION)DeviceObject->DeviceExtension;
  1380. PIO_STACK_LOCATION irpSp = IoGetCurrentIrpStackLocation( Irp );
  1381. FloppyDump( FLOPSHOW,("FloppyStartDevice: Irp: %p\n", Irp) );
  1382. FloppyDump( FLOPSHOW, (" AllocatedResources = %08x\n",irpSp->Parameters.StartDevice.AllocatedResources));
  1383. FloppyDump( FLOPSHOW, (" AllocatedResourcesTranslated = %08x\n",irpSp->Parameters.StartDevice.AllocatedResourcesTranslated));
  1384. //
  1385. // First we must pass this Irp on to the PDO.
  1386. //
  1387. KeInitializeEvent( &doneEvent, NotificationEvent, FALSE );
  1388. IoCopyCurrentIrpStackLocationToNext( Irp );
  1389. IoSetCompletionRoutine( Irp,
  1390. FloppyPnpComplete,
  1391. &doneEvent,
  1392. TRUE, TRUE, TRUE );
  1393. ntStatus = IoCallDriver( disketteExtension->TargetObject, Irp );
  1394. if ( ntStatus == STATUS_PENDING ) {
  1395. ntStatus = KeWaitForSingleObject( &doneEvent,
  1396. Executive,
  1397. KernelMode,
  1398. FALSE,
  1399. NULL );
  1400. ASSERT( ntStatus == STATUS_SUCCESS );
  1401. ntStatus = Irp->IoStatus.Status;
  1402. }
  1403. fdcInfo.BufferCount = 0;
  1404. fdcInfo.BufferSize = 0;
  1405. ntStatus = FlFdcDeviceIo( disketteExtension->TargetObject,
  1406. IOCTL_DISK_INTERNAL_GET_FDC_INFO,
  1407. &fdcInfo );
  1408. if ( NT_SUCCESS(ntStatus) ) {
  1409. disketteExtension->MaxTransferSize = fdcInfo.MaxTransferSize;
  1410. if ( (fdcInfo.AcpiBios) &&
  1411. (fdcInfo.AcpiFdiSupported) ) {
  1412. ntStatus = FlAcpiConfigureFloppy( disketteExtension, &fdcInfo );
  1413. if ( disketteExtension->DriveType == DRIVE_TYPE_2880 ) {
  1414. disketteExtension->PerpendicularMode |= 1 << fdcInfo.PeripheralNumber;
  1415. }
  1416. } else {
  1417. INTERFACE_TYPE InterfaceType;
  1418. if ( disketteExtension->DriveType == DRIVE_TYPE_2880 ) {
  1419. disketteExtension->PerpendicularMode |= 1 << fdcInfo.PeripheralNumber;
  1420. }
  1421. //
  1422. // Query the registry till we find the correct interface type,
  1423. // since we do not know what type of interface we are on.
  1424. //
  1425. for ( InterfaceType = 0;
  1426. InterfaceType < MaximumInterfaceType;
  1427. InterfaceType++ ) {
  1428. fdcInfo.BusType = InterfaceType;
  1429. ntStatus = IoQueryDeviceDescription( &fdcInfo.BusType,
  1430. &fdcInfo.BusNumber,
  1431. &Dc,
  1432. &fdcInfo.ControllerNumber,
  1433. &Fp,
  1434. &fdcInfo.PeripheralNumber,
  1435. FlConfigCallBack,
  1436. disketteExtension );
  1437. if (NT_SUCCESS(ntStatus)) {
  1438. //
  1439. // We found the interface we are on.
  1440. //
  1441. FloppyDump(FLOPSHOW,
  1442. ("Interface Type is %x\n", InterfaceType));
  1443. break;
  1444. }
  1445. }
  1446. }
  1447. if ( NT_SUCCESS(ntStatus) ) {
  1448. if (IsNEC_98) {
  1449. disketteExtension->DeviceUnit = (UCHAR)fdcInfo.UnitNumber;
  1450. disketteExtension->DriveOnValue = (UCHAR)fdcInfo.UnitNumber;
  1451. } else { // (IsNEC_98)
  1452. disketteExtension->DeviceUnit = (UCHAR)fdcInfo.PeripheralNumber;
  1453. disketteExtension->DriveOnValue =
  1454. (UCHAR)(fdcInfo.PeripheralNumber | ( DRVCTL_DRIVE_0 << fdcInfo.PeripheralNumber ));
  1455. } // (IsNEC_98)
  1456. pnpStatus = IoRegisterDeviceInterface( disketteExtension->UnderlyingPDO,
  1457. (LPGUID)&MOUNTDEV_MOUNTED_DEVICE_GUID,
  1458. NULL,
  1459. &disketteExtension->InterfaceString );
  1460. if ( NT_SUCCESS(pnpStatus) ) {
  1461. pnpStatus = IoSetDeviceInterfaceState( &disketteExtension->InterfaceString,
  1462. TRUE );
  1463. }
  1464. disketteExtension->IsStarted = TRUE;
  1465. ExAcquireFastMutex(&(disketteExtension->HoldNewReqMutex));
  1466. disketteExtension->HoldNewRequests = FALSE;
  1467. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1468. FloppyProcessQueuedRequests( disketteExtension );
  1469. }
  1470. }
  1471. Irp->IoStatus.Status = ntStatus;
  1472. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  1473. return ntStatus;
  1474. }
  1475. NTSTATUS
  1476. FloppyPnpComplete (
  1477. IN PDEVICE_OBJECT DeviceObject,
  1478. IN PIRP Irp,
  1479. IN PVOID Context
  1480. )
  1481. /*++
  1482. Routine Description:
  1483. A completion routine for use when calling the lower device objects to
  1484. which our bus (FDO) is attached.
  1485. --*/
  1486. {
  1487. KeSetEvent ((PKEVENT) Context, 1, FALSE);
  1488. // No special priority
  1489. // No Wait
  1490. return STATUS_MORE_PROCESSING_REQUIRED; // Keep this IRP
  1491. }
  1492. VOID
  1493. FlTerminateFloppyThread(
  1494. PDISKETTE_EXTENSION DisketteExtension
  1495. )
  1496. {
  1497. if (DisketteExtension->FloppyThread != NULL) {
  1498. KeWaitForSingleObject( DisketteExtension->FloppyThread,
  1499. Executive,
  1500. KernelMode,
  1501. FALSE,
  1502. NULL );
  1503. //
  1504. // Make sure again FloppyThread is not NULL.
  1505. //
  1506. if (DisketteExtension->FloppyThread != NULL) {
  1507. ObDereferenceObject(DisketteExtension->FloppyThread);
  1508. }
  1509. DisketteExtension->FloppyThread = NULL;
  1510. }
  1511. }
  1512. NTSTATUS
  1513. FloppyPower(
  1514. IN PDEVICE_OBJECT DeviceObject,
  1515. IN PIRP Irp
  1516. )
  1517. /*++
  1518. Routine Description:
  1519. Arguments:
  1520. DeviceObject - a pointer to the object that represents the device
  1521. that I/O is to be done on.
  1522. Irp - a pointer to the I/O Request Packet for this request.
  1523. Return Value:
  1524. --*/
  1525. {
  1526. PDISKETTE_EXTENSION disketteExtension;
  1527. NTSTATUS ntStatus = Irp->IoStatus.Status;
  1528. PIO_STACK_LOCATION irpSp;
  1529. POWER_STATE_TYPE type;
  1530. POWER_STATE state;
  1531. BOOLEAN WaitForCompletion = TRUE;
  1532. FloppyDump( FLOPSHOW, ("FloppyPower:\n"));
  1533. disketteExtension = DeviceObject->DeviceExtension;
  1534. irpSp = IoGetCurrentIrpStackLocation( Irp );
  1535. type = irpSp->Parameters.Power.Type;
  1536. state = irpSp->Parameters.Power.State;
  1537. switch(irpSp->MinorFunction) {
  1538. case IRP_MN_QUERY_POWER: {
  1539. FloppyDump( FLOPDBGP,
  1540. ("IRP_MN_QUERY_POWER : Type - %d, State %d\n",
  1541. type, state));
  1542. if ((type == SystemPowerState) &&
  1543. (state.SystemState > PowerSystemHibernate)) {
  1544. //
  1545. // This is a shutdown request. Pass that.
  1546. //
  1547. ntStatus = STATUS_SUCCESS;
  1548. break;
  1549. }
  1550. //
  1551. // If there are no requests being processed or queued up
  1552. // for floppy, ThreadReferenceCount will be -1. It can be 0 if
  1553. // there was only one request and that has been dequeued and is
  1554. // currently being processed.
  1555. //
  1556. ExAcquireFastMutex(&disketteExtension->ThreadReferenceMutex);
  1557. if (disketteExtension->ThreadReferenceCount > 0) {
  1558. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  1559. FloppyDump(FLOPDBGP,
  1560. ("Floppy: Requests pending. Cannot powerdown!\n"));
  1561. PoStartNextPowerIrp(Irp);
  1562. Irp->IoStatus.Information = 0;
  1563. Irp->IoStatus.Status = STATUS_DEVICE_BUSY;
  1564. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  1565. return STATUS_DEVICE_BUSY;
  1566. } else if ((disketteExtension->ThreadReferenceCount == 0) &&
  1567. ((disketteExtension->FloppyThread) != NULL)) {
  1568. FloppyDump(FLOPDBGP,
  1569. ("Ref count 0. No request pending.\n"));
  1570. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  1571. ExAcquireFastMutex(&disketteExtension->PowerDownMutex);
  1572. disketteExtension->ReceivedQueryPower = TRUE;
  1573. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  1574. KeWaitForSingleObject(&disketteExtension->QueryPowerEvent,
  1575. Executive,
  1576. KernelMode,
  1577. FALSE,
  1578. NULL);
  1579. } else {
  1580. FloppyDump(FLOPDBGP,
  1581. ("No IRPs pending. Let system hibernate"));
  1582. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  1583. }
  1584. ntStatus = STATUS_SUCCESS;
  1585. break;
  1586. }
  1587. case IRP_MN_SET_POWER: {
  1588. //
  1589. // Indicate that we are going to power down or power up
  1590. // so that FloppyThread can process queued requests
  1591. // accordingly.
  1592. //
  1593. if (type == SystemPowerState) {
  1594. ExAcquireFastMutex(&disketteExtension->PowerDownMutex);
  1595. if (state.SystemState == PowerSystemWorking) {
  1596. FloppyDump( FLOPDBGP, ("Powering Up\n"));
  1597. disketteExtension->PoweringDown = FALSE;
  1598. WaitForCompletion = FALSE;
  1599. } else {
  1600. FloppyDump( FLOPDBGP, ("Powering down\n"));
  1601. WaitForCompletion = TRUE;
  1602. disketteExtension->PoweringDown = TRUE;
  1603. }
  1604. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  1605. //
  1606. // Wait till FloppyThread signals that it is done with
  1607. // the queued requests.
  1608. //
  1609. if ((disketteExtension->FloppyThread != NULL) &&
  1610. (WaitForCompletion == TRUE)) {
  1611. KeWaitForSingleObject( disketteExtension->FloppyThread,
  1612. Executive,
  1613. KernelMode,
  1614. FALSE,
  1615. NULL );
  1616. }
  1617. }
  1618. FloppyDump( FLOPSHOW, ("Processing power irp : %p\n", Irp));
  1619. ntStatus = STATUS_SUCCESS;
  1620. break;
  1621. }
  1622. default: {
  1623. break;
  1624. }
  1625. }
  1626. PoStartNextPowerIrp( Irp );
  1627. IoSkipCurrentIrpStackLocation( Irp );
  1628. ntStatus = PoCallDriver( disketteExtension->TargetObject, Irp );
  1629. return ntStatus;
  1630. }
  1631. NTSTATUS
  1632. FloppyReadWrite(
  1633. IN PDEVICE_OBJECT DeviceObject,
  1634. IN PIRP Irp
  1635. )
  1636. /*++
  1637. Routine Description:
  1638. This routine is called by the I/O system to read or write to a
  1639. device that we control.
  1640. Arguments:
  1641. DeviceObject - a pointer to the object that represents the device
  1642. that I/O is to be done on.
  1643. Irp - a pointer to the I/O Request Packet for this request.
  1644. Return Value:
  1645. STATUS_INVALID_PARAMETER if parameters are invalid,
  1646. STATUS_PENDING otherwise.
  1647. --*/
  1648. {
  1649. PIO_STACK_LOCATION irpSp;
  1650. NTSTATUS ntStatus;
  1651. PDISKETTE_EXTENSION disketteExtension;
  1652. FloppyDump( FLOPSHOW, ("FloppyReadWrite...\n") );
  1653. disketteExtension = DeviceObject->DeviceExtension;
  1654. irpSp = IoGetCurrentIrpStackLocation( Irp );
  1655. //
  1656. // This IRP was sent to the function driver.
  1657. // We need to check if we are currently holding requests.
  1658. //
  1659. ExAcquireFastMutex(&(disketteExtension->HoldNewReqMutex));
  1660. if ( disketteExtension->HoldNewRequests ) {
  1661. ntStatus = FloppyQueueRequest( disketteExtension, Irp );
  1662. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1663. return ntStatus;
  1664. }
  1665. //
  1666. // If the device is not active (not started yet or removed) we will
  1667. // just fail this request outright.
  1668. //
  1669. if ( disketteExtension->IsRemoved || !disketteExtension->IsStarted) {
  1670. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1671. if ( disketteExtension->IsRemoved) {
  1672. ntStatus = STATUS_DELETE_PENDING;
  1673. } else {
  1674. ntStatus = STATUS_UNSUCCESSFUL;
  1675. }
  1676. Irp->IoStatus.Information = 0;
  1677. Irp->IoStatus.Status = ntStatus;
  1678. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  1679. return ntStatus;
  1680. }
  1681. if ( (disketteExtension->MediaType > Unknown) &&
  1682. ((irpSp->Parameters.Read.ByteOffset.LowPart +
  1683. irpSp->Parameters.Read.Length > disketteExtension->ByteCapacity) ||
  1684. ((irpSp->Parameters.Read.Length &
  1685. (disketteExtension->BytesPerSector - 1)) != 0 ))) {
  1686. FloppyDump( FLOPDBGP,
  1687. ("Floppy: Invalid Parameter, rejecting request\n") );
  1688. FloppyDump( FLOPWARN,
  1689. ("Floppy: Starting offset = %lx\n"
  1690. "------ I/O Length = %lx\n"
  1691. "------ ByteCapacity = %lx\n"
  1692. "------ BytesPerSector = %lx\n",
  1693. irpSp->Parameters.Read.ByteOffset.LowPart,
  1694. irpSp->Parameters.Read.Length,
  1695. disketteExtension->ByteCapacity,
  1696. disketteExtension->BytesPerSector) );
  1697. ntStatus = STATUS_INVALID_PARAMETER;
  1698. } else {
  1699. //
  1700. // verify that user is really expecting some I/O operation to
  1701. // occur.
  1702. //
  1703. if (irpSp->Parameters.Read.Length) {
  1704. //
  1705. // Queue request to thread.
  1706. //
  1707. FloppyDump( FLOPIRPPATH,
  1708. ("Floppy: Enqueing up IRP: %p\n",Irp) );
  1709. ntStatus = FlQueueIrpToThread(Irp, disketteExtension);
  1710. } else {
  1711. //
  1712. // Complete this zero length request with no boost.
  1713. //
  1714. Irp->IoStatus.Information = 0;
  1715. Irp->IoStatus.Status = STATUS_SUCCESS;
  1716. FloppyDump(FLOPDBGP,
  1717. ("Zero length r/w request. Completing IRP.\n"));
  1718. ntStatus = STATUS_SUCCESS;
  1719. }
  1720. }
  1721. ExReleaseFastMutex(&(disketteExtension->HoldNewReqMutex));
  1722. if ( ntStatus != STATUS_PENDING ) {
  1723. Irp->IoStatus.Status = ntStatus;
  1724. FloppyDump(FLOPDBGP,
  1725. ("Completing request. NTStatus %x\n",
  1726. ntStatus));
  1727. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  1728. }
  1729. return ntStatus;
  1730. }
  1731. NTSTATUS
  1732. FlInterpretError(
  1733. IN UCHAR StatusRegister1,
  1734. IN UCHAR StatusRegister2
  1735. )
  1736. /*++
  1737. Routine Description:
  1738. This routine is called when the floppy controller returns an error.
  1739. Status registers 1 and 2 are passed in, and this returns an appropriate
  1740. error status.
  1741. Arguments:
  1742. StatusRegister1 - the controller's status register #1.
  1743. StatusRegister2 - the controller's status register #2.
  1744. Return Value:
  1745. An NTSTATUS error determined from the status registers.
  1746. --*/
  1747. {
  1748. if ( ( StatusRegister1 & STREG1_CRC_ERROR ) ||
  1749. ( StatusRegister2 & STREG2_CRC_ERROR ) ) {
  1750. FloppyDump(
  1751. FLOPSHOW,
  1752. ("FlInterpretError: STATUS_CRC_ERROR\n")
  1753. );
  1754. return STATUS_CRC_ERROR;
  1755. }
  1756. if ( StatusRegister1 & STREG1_DATA_OVERRUN ) {
  1757. FloppyDump(
  1758. FLOPSHOW,
  1759. ("FlInterpretError: STATUS_DATA_OVERRUN\n")
  1760. );
  1761. return STATUS_DATA_OVERRUN;
  1762. }
  1763. if ( ( StatusRegister1 & STREG1_SECTOR_NOT_FOUND ) ||
  1764. ( StatusRegister1 & STREG1_END_OF_DISKETTE ) ) {
  1765. FloppyDump(
  1766. FLOPSHOW,
  1767. ("FlInterpretError: STATUS_NONEXISTENT_SECTOR\n")
  1768. );
  1769. return STATUS_NONEXISTENT_SECTOR;
  1770. }
  1771. if ( ( StatusRegister2 & STREG2_DATA_NOT_FOUND ) ||
  1772. ( StatusRegister2 & STREG2_BAD_CYLINDER ) ||
  1773. ( StatusRegister2 & STREG2_DELETED_DATA ) ) {
  1774. FloppyDump(
  1775. FLOPSHOW,
  1776. ("FlInterpretError: STATUS_DEVICE_DATA_ERROR\n")
  1777. );
  1778. return STATUS_DEVICE_DATA_ERROR;
  1779. }
  1780. if ( StatusRegister1 & STREG1_WRITE_PROTECTED ) {
  1781. FloppyDump(
  1782. FLOPSHOW,
  1783. ("FlInterpretError: STATUS_MEDIA_WRITE_PROTECTED\n")
  1784. );
  1785. return STATUS_MEDIA_WRITE_PROTECTED;
  1786. }
  1787. if ( StatusRegister1 & STREG1_ID_NOT_FOUND ) {
  1788. FloppyDump(
  1789. FLOPSHOW,
  1790. ("FlInterpretError: STATUS_FLOPPY_ID_MARK_NOT_FOUND\n")
  1791. );
  1792. return STATUS_FLOPPY_ID_MARK_NOT_FOUND;
  1793. }
  1794. if ( StatusRegister2 & STREG2_WRONG_CYLINDER ) {
  1795. FloppyDump(
  1796. FLOPSHOW,
  1797. ("FlInterpretError: STATUS_FLOPPY_WRONG_CYLINDER\n")
  1798. );
  1799. return STATUS_FLOPPY_WRONG_CYLINDER;
  1800. }
  1801. //
  1802. // There's other error bits, but no good status values to map them
  1803. // to. Just return a generic one.
  1804. //
  1805. FloppyDump(
  1806. FLOPSHOW,
  1807. ("FlInterpretError: STATUS_FLOPPY_UNKNOWN_ERROR\n")
  1808. );
  1809. return STATUS_FLOPPY_UNKNOWN_ERROR;
  1810. }
  1811. VOID
  1812. FlFinishOperation(
  1813. IN OUT PIRP Irp,
  1814. IN PDISKETTE_EXTENSION DisketteExtension
  1815. )
  1816. /*++
  1817. Routine Description:
  1818. This routine is called by FloppyThread at the end of any operation
  1819. whether it succeeded or not.
  1820. If the packet is failing due to a hardware error, this routine will
  1821. reinitialize the hardware and retry once.
  1822. When the packet is done, this routine will start the timer to turn
  1823. off the motor, and complete the IRP.
  1824. Arguments:
  1825. Irp - a pointer to the IO Request Packet being processed.
  1826. DisketteExtension - a pointer to the diskette extension for the
  1827. diskette on which the operation occurred.
  1828. Return Value:
  1829. None.
  1830. --*/
  1831. {
  1832. NTSTATUS ntStatus;
  1833. FloppyDump(
  1834. FLOPSHOW,
  1835. ("Floppy: FloppyFinishOperation...\n")
  1836. );
  1837. //
  1838. // See if this packet is being failed due to a hardware error.
  1839. //
  1840. if ( ( Irp->IoStatus.Status != STATUS_SUCCESS ) &&
  1841. ( DisketteExtension->HardwareFailed ) ) {
  1842. DisketteExtension->HardwareFailCount++;
  1843. if ( DisketteExtension->HardwareFailCount <
  1844. HARDWARE_RESET_RETRY_COUNT ) {
  1845. //
  1846. // This is our first time through (that is, we're not retrying
  1847. // the packet after a hardware failure). If it failed this first
  1848. // time because of a hardware problem, set the HardwareFailed flag
  1849. // and put the IRP at the beginning of the request queue.
  1850. //
  1851. ntStatus = FlInitializeControllerHardware( DisketteExtension );
  1852. if ( NT_SUCCESS( ntStatus ) ) {
  1853. FloppyDump(
  1854. FLOPINFO,
  1855. ("Floppy: packet failed; hardware reset. Retry.\n")
  1856. );
  1857. //
  1858. // Force media to be redetermined, in case we messed up
  1859. // and to make sure FlDatarateSpecifyConfigure() gets
  1860. // called.
  1861. //
  1862. DisketteExtension->MediaType = Undetermined;
  1863. FloppyDump(
  1864. FLOPIRPPATH,
  1865. ("Floppy: irp %x failed - back on the queue with it\n",
  1866. Irp)
  1867. );
  1868. ExAcquireFastMutex(&DisketteExtension->ThreadReferenceMutex);
  1869. ASSERT(DisketteExtension->ThreadReferenceCount >= 0);
  1870. (DisketteExtension->ThreadReferenceCount)++;
  1871. ExReleaseFastMutex(&DisketteExtension->ThreadReferenceMutex);
  1872. ExInterlockedInsertHeadList(
  1873. &DisketteExtension->ListEntry,
  1874. &Irp->Tail.Overlay.ListEntry,
  1875. &DisketteExtension->ListSpinLock );
  1876. return;
  1877. }
  1878. FloppyDump(
  1879. FLOPDBGP,
  1880. ("Floppy: packet AND hardware reset failed.\n")
  1881. );
  1882. }
  1883. }
  1884. //
  1885. // If we didn't already RETURN, we're done with this packet so
  1886. // reset the HardwareFailCount for the next packet.
  1887. //
  1888. DisketteExtension->HardwareFailCount = 0;
  1889. //
  1890. // If this request was unsuccessful and the error is one that can be
  1891. // remedied by the user, save the Device Object so that the file system,
  1892. // after reaching its original entry point, can know the real device.
  1893. //
  1894. if ( !NT_SUCCESS( Irp->IoStatus.Status ) &&
  1895. IoIsErrorUserInduced( Irp->IoStatus.Status ) ) {
  1896. IoSetHardErrorOrVerifyDevice( Irp, DisketteExtension->DeviceObject );
  1897. }
  1898. //
  1899. // Even if the operation failed, it probably had to wait for the drive
  1900. // to spin up or somesuch so we'll always complete the request with the
  1901. // standard priority boost.
  1902. //
  1903. if ( ( Irp->IoStatus.Status != STATUS_SUCCESS ) &&
  1904. ( Irp->IoStatus.Status != STATUS_VERIFY_REQUIRED ) &&
  1905. ( Irp->IoStatus.Status != STATUS_NO_MEDIA_IN_DEVICE ) ) {
  1906. FloppyDump(
  1907. FLOPDBGP,
  1908. ("Floppy: IRP failed with error %lx\n", Irp->IoStatus.Status)
  1909. );
  1910. } else {
  1911. FloppyDump(
  1912. FLOPINFO,
  1913. ("Floppy: IoStatus.Status = %x\n", Irp->IoStatus.Status)
  1914. );
  1915. }
  1916. FloppyDump(
  1917. FLOPINFO,
  1918. ("Floppy: IoStatus.Information = %x\n", Irp->IoStatus.Information)
  1919. );
  1920. FloppyDump(
  1921. FLOPIRPPATH,
  1922. ("Floppy: Finishing up IRP: %p\n",Irp)
  1923. );
  1924. //
  1925. // In order to get explorer to request a format of unformatted media
  1926. // the STATUS_UNRECOGNIZED_MEDIA error must be translated to a generic
  1927. // STATUS_UNSUCCESSFUL error.
  1928. //
  1929. // if ( Irp->IoStatus.Status == STATUS_UNRECOGNIZED_MEDIA ) {
  1930. // Irp->IoStatus.Status = STATUS_UNSUCCESSFUL;
  1931. // }
  1932. IoCompleteRequest( Irp, IO_DISK_INCREMENT );
  1933. }
  1934. NTSTATUS
  1935. FlStartDrive(
  1936. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  1937. IN PIRP Irp,
  1938. IN BOOLEAN WriteOperation,
  1939. IN BOOLEAN SetUpMedia,
  1940. IN BOOLEAN IgnoreChange
  1941. )
  1942. /*++
  1943. Routine Description:
  1944. This routine is called at the beginning of every operation. It cancels
  1945. the motor timer if it's on, turns the motor on and waits for it to
  1946. spin up if it was off, resets the disk change line and returns
  1947. VERIFY_REQUIRED if the disk has been changed, determines the diskette
  1948. media type if it's not known and SetUpMedia=TRUE, and makes sure that
  1949. the disk isn't write protected if WriteOperation = TRUE.
  1950. Arguments:
  1951. DisketteExtension - a pointer to our data area for the drive being
  1952. started.
  1953. Irp - Supplies the I/O request packet.
  1954. WriteOperation - TRUE if the diskette will be written to, FALSE
  1955. otherwise.
  1956. SetUpMedia - TRUE if the media type of the diskette in the drive
  1957. should be determined.
  1958. IgnoreChange - Do not return VERIFY_REQUIRED eventhough we are mounting
  1959. for the first time.
  1960. Return Value:
  1961. STATUS_SUCCESS if the drive is started properly; appropriate error
  1962. propogated otherwise.
  1963. --*/
  1964. {
  1965. LARGE_INTEGER delay;
  1966. BOOLEAN motorStarted;
  1967. BOOLEAN diskChanged;
  1968. UCHAR driveStatus;
  1969. NTSTATUS ntStatus = STATUS_SUCCESS;
  1970. FDC_ENABLE_PARMS fdcEnableParms;
  1971. FDC_DISK_CHANGE_PARMS fdcDiskChangeParms;
  1972. FloppyDump(
  1973. FLOPSHOW,
  1974. ("Floppy: FloppyStartDrive...\n")
  1975. );
  1976. //
  1977. // IMPORTANT
  1978. // NOTE
  1979. // COMMENT
  1980. //
  1981. // Here we will copy the BIOS floppy configuration on top of the
  1982. // highest media value in our global array so that any type of processing
  1983. // that will recalibrate the drive can have it done here.
  1984. // An optimization would be to only do it when we will try to recalibrate
  1985. // the driver or media in it.
  1986. // At this point, we ensure that on any processing of a command we
  1987. // are going to have the real values inthe first entry of the array for
  1988. // driver constants.
  1989. //
  1990. DriveMediaConstants[DriveMediaLimits[DisketteExtension->DriveType].
  1991. HighestDriveMediaType] = DisketteExtension->BiosDriveMediaConstants;
  1992. if ((DisketteExtension->MediaType == Undetermined) ||
  1993. (DisketteExtension->MediaType == Unknown)) {
  1994. DisketteExtension->DriveMediaConstants = DriveMediaConstants[0];
  1995. }
  1996. //
  1997. // Grab the timer spin lock and cancel the timer, since we want the
  1998. // motor to run for the whole operation. If the proper drive is
  1999. // already running, great; if not, start the motor and wait for it
  2000. // to spin up.
  2001. //
  2002. fdcEnableParms.DriveOnValue = DisketteExtension->DriveOnValue;
  2003. if ( WriteOperation ) {
  2004. fdcEnableParms.TimeToWait =
  2005. DisketteExtension->DriveMediaConstants.MotorSettleTimeWrite;
  2006. } else {
  2007. fdcEnableParms.TimeToWait =
  2008. DisketteExtension->DriveMediaConstants.MotorSettleTimeRead;
  2009. }
  2010. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  2011. IOCTL_DISK_INTERNAL_ENABLE_FDC_DEVICE,
  2012. &fdcEnableParms );
  2013. motorStarted = fdcEnableParms.MotorStarted;
  2014. if (NT_SUCCESS(ntStatus)) {
  2015. fdcDiskChangeParms.DriveOnValue = DisketteExtension->DriveOnValue;
  2016. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  2017. IOCTL_DISK_INTERNAL_GET_FDC_DISK_CHANGE,
  2018. &fdcDiskChangeParms );
  2019. driveStatus = fdcDiskChangeParms.DriveStatus;
  2020. }
  2021. if (!NT_SUCCESS(ntStatus)) {
  2022. return ntStatus;
  2023. }
  2024. //
  2025. // Support for 360K drives:
  2026. // They have no change line, so we will assume a power up of the motor
  2027. // to be equivalent to a change of floppy (we assume noone will
  2028. // change the floppy while it is turning.
  2029. // So force a VERIFY here (unless the file system explicitly turned
  2030. // it off).
  2031. //
  2032. if ( ((DisketteExtension->DriveType == DRIVE_TYPE_0360) &&
  2033. motorStarted) ||
  2034. ((DisketteExtension->DriveType != DRIVE_TYPE_0360) &&
  2035. driveStatus & DSKCHG_DISKETTE_REMOVED) ) {
  2036. FloppyDump(
  2037. FLOPSHOW,
  2038. ("Floppy: disk changed...\n")
  2039. );
  2040. DisketteExtension->MediaType = Undetermined;
  2041. //
  2042. // If the volume is mounted, we must tell the filesystem to
  2043. // verify that the media in the drive is the same volume.
  2044. //
  2045. if ( DisketteExtension->DeviceObject->Vpb->Flags & VPB_MOUNTED ) {
  2046. if (Irp) {
  2047. IoSetHardErrorOrVerifyDevice( Irp,
  2048. DisketteExtension->DeviceObject );
  2049. }
  2050. DisketteExtension->DeviceObject->Flags |= DO_VERIFY_VOLUME;
  2051. }
  2052. //
  2053. // Only go through the device reset if we did get the flag set
  2054. // We really only want to go throught here if the diskette changed,
  2055. // but on 360 it will always say the diskette has changed.
  2056. // So based on our previous test, only proceed if it is NOT
  2057. // a 360K driver
  2058. if (DisketteExtension->DriveType != DRIVE_TYPE_0360) {
  2059. if (IsNEC_98) {
  2060. //
  2061. // Before seek, make sure that disk has been removed.
  2062. //
  2063. DisketteExtension->FifoBuffer[0] = COMMND_SENSE_DRIVE_STATUS;
  2064. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2065. ntStatus = FlIssueCommand( DisketteExtension,
  2066. DisketteExtension->FifoBuffer,
  2067. DisketteExtension->FifoBuffer,
  2068. NULL,
  2069. 0,
  2070. 0 );
  2071. if ( !NT_SUCCESS( ntStatus ) ) {
  2072. FloppyDump(
  2073. FLOPWARN,
  2074. ("Floppy: SENSE_DRIVE (1) returned%x\n", ntStatus)
  2075. );
  2076. return ntStatus;
  2077. }
  2078. if ( DisketteExtension->FifoBuffer[0] & STREG3_DRIVE_READY ) {
  2079. driveStatus = DSKCHG_RESERVED;
  2080. } else {
  2081. driveStatus = DSKCHG_DISKETTE_REMOVED;
  2082. }
  2083. if ( driveStatus & DSKCHG_DISKETTE_REMOVED ) {
  2084. //
  2085. // If "disk changed" is still set after the double seek, the
  2086. // drive door must be opened.
  2087. //
  2088. FloppyDump(
  2089. FLOPINFO,
  2090. ("Floppy: close the door! (1)\n")
  2091. );
  2092. //
  2093. // Turn off the flag for now so that we will not get so many
  2094. // gratuitous verifys. It will be set again the next time.
  2095. //
  2096. if(DisketteExtension->DeviceObject->Vpb->Flags & VPB_MOUNTED) {
  2097. DisketteExtension->DeviceObject->Flags &= ~DO_VERIFY_VOLUME;
  2098. }
  2099. return STATUS_NO_MEDIA_IN_DEVICE;
  2100. }
  2101. } // (IsNEC_98)
  2102. //
  2103. // Now seek twice to reset the "disk changed" line. First
  2104. // seek to 1.
  2105. //
  2106. // Normally we'd do a READ ID after a seek. However, we don't
  2107. // even know if this disk is formatted. We're not really
  2108. // trying to get anywhere; we're just doing this to reset the
  2109. // "disk changed" line so we'll skip the READ ID.
  2110. //
  2111. DisketteExtension->FifoBuffer[0] = COMMND_SEEK;
  2112. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2113. DisketteExtension->FifoBuffer[2] = 1;
  2114. ntStatus = FlIssueCommand( DisketteExtension,
  2115. DisketteExtension->FifoBuffer,
  2116. DisketteExtension->FifoBuffer,
  2117. NULL,
  2118. 0,
  2119. 0 );
  2120. if ( !NT_SUCCESS( ntStatus ) ) {
  2121. FloppyDump( FLOPWARN,
  2122. ("Floppy: seek to 1 returned %x\n", ntStatus) );
  2123. return ntStatus;
  2124. } else {
  2125. if (!( DisketteExtension->FifoBuffer[0] & STREG0_SEEK_COMPLETE)
  2126. || ( DisketteExtension->FifoBuffer[1] != 1 ) ) {
  2127. FloppyDump(
  2128. FLOPWARN,
  2129. ("Floppy: Seek to 1 had bad return registers\n")
  2130. );
  2131. DisketteExtension->HardwareFailed = TRUE;
  2132. return STATUS_FLOPPY_BAD_REGISTERS;
  2133. }
  2134. }
  2135. //
  2136. // Seek back to 0. We can once again skip the READ ID.
  2137. //
  2138. DisketteExtension->FifoBuffer[0] = COMMND_SEEK;
  2139. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2140. DisketteExtension->FifoBuffer[2] = 0;
  2141. //
  2142. // Floppy drives use by Toshiba systems require a delay
  2143. // when this operation is performed.
  2144. //
  2145. delay.LowPart = (ULONG) -900;
  2146. delay.HighPart = -1;
  2147. KeDelayExecutionThread( KernelMode, FALSE, &delay );
  2148. ntStatus = FlIssueCommand( DisketteExtension,
  2149. DisketteExtension->FifoBuffer,
  2150. DisketteExtension->FifoBuffer,
  2151. NULL,
  2152. 0,
  2153. 0 );
  2154. //
  2155. // Again, for Toshiba floppy drives, a delay is required.
  2156. //
  2157. delay.LowPart = (ULONG) -5;
  2158. delay.HighPart = -1;
  2159. KeDelayExecutionThread( KernelMode, FALSE, &delay );
  2160. if ( !NT_SUCCESS( ntStatus ) ) {
  2161. FloppyDump( FLOPWARN,
  2162. ("Floppy: seek to 0 returned %x\n", ntStatus) );
  2163. return ntStatus;
  2164. } else {
  2165. if (!(DisketteExtension->FifoBuffer[0] & STREG0_SEEK_COMPLETE)
  2166. || ( DisketteExtension->FifoBuffer[1] != 0 ) ) {
  2167. FloppyDump(
  2168. FLOPWARN,
  2169. ("Floppy: Seek to 0 had bad return registers\n")
  2170. );
  2171. DisketteExtension->HardwareFailed = TRUE;
  2172. return STATUS_FLOPPY_BAD_REGISTERS;
  2173. }
  2174. }
  2175. if (IsNEC_98) {
  2176. //
  2177. // Before seek, make sure that disk has been removed.
  2178. //
  2179. DisketteExtension->FifoBuffer[0] = COMMND_SENSE_DRIVE_STATUS;
  2180. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2181. ntStatus = FlIssueCommand( DisketteExtension,
  2182. DisketteExtension->FifoBuffer,
  2183. DisketteExtension->FifoBuffer,
  2184. NULL,
  2185. 0,
  2186. 0 );
  2187. if ( !NT_SUCCESS( ntStatus ) ) {
  2188. FloppyDump(
  2189. FLOPWARN,
  2190. ("Floppy: SENSE_DRIVE (1) returned%x\n", ntStatus)
  2191. );
  2192. return ntStatus;
  2193. }
  2194. if ( DisketteExtension->FifoBuffer[0] & STREG3_DRIVE_READY ) {
  2195. driveStatus = DSKCHG_RESERVED;
  2196. } else {
  2197. driveStatus = DSKCHG_DISKETTE_REMOVED;
  2198. }
  2199. } else { // (IsNEC_98)
  2200. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  2201. IOCTL_DISK_INTERNAL_GET_FDC_DISK_CHANGE,
  2202. &fdcDiskChangeParms );
  2203. driveStatus = fdcDiskChangeParms.DriveStatus;
  2204. if (!NT_SUCCESS(ntStatus)) {
  2205. return ntStatus;
  2206. }
  2207. } // (IsNEC_98)
  2208. if ( driveStatus & DSKCHG_DISKETTE_REMOVED ) {
  2209. //
  2210. // If "disk changed" is still set after the double seek, the
  2211. // drive door must be opened.
  2212. //
  2213. FloppyDump(
  2214. FLOPINFO,
  2215. ("Floppy: close the door!\n")
  2216. );
  2217. //
  2218. // Turn off the flag for now so that we will not get so many
  2219. // gratuitous verifys. It will be set again the next time.
  2220. //
  2221. if(DisketteExtension->DeviceObject->Vpb->Flags & VPB_MOUNTED) {
  2222. DisketteExtension->DeviceObject->Flags &= ~DO_VERIFY_VOLUME;
  2223. }
  2224. return STATUS_NO_MEDIA_IN_DEVICE;
  2225. }
  2226. }
  2227. //
  2228. // IgnoreChange indicates the file system is in the process
  2229. // of performing a verify so do not return verify required.
  2230. //
  2231. if ( IgnoreChange == FALSE ) {
  2232. if ( DisketteExtension->DeviceObject->Vpb->Flags & VPB_MOUNTED ) {
  2233. //
  2234. // Drive WAS mounted, but door was opened since the last time
  2235. // we checked so tell the file system to verify the diskette.
  2236. //
  2237. FloppyDump(
  2238. FLOPSHOW,
  2239. ("Floppy: start drive - verify required because door opened\n")
  2240. );
  2241. return STATUS_VERIFY_REQUIRED;
  2242. } else {
  2243. return STATUS_IO_DEVICE_ERROR;
  2244. }
  2245. }
  2246. }else{
  2247. if (IsNEC_98) {
  2248. FlHdbit(DisketteExtension);
  2249. } // (IsNEC_98)
  2250. }
  2251. if ( SetUpMedia ) {
  2252. if ( DisketteExtension->MediaType == Undetermined ) {
  2253. ntStatus = FlDetermineMediaType( DisketteExtension );
  2254. } else {
  2255. if ( DisketteExtension->MediaType == Unknown ) {
  2256. //
  2257. // We've already tried to determine the media type and
  2258. // failed. It's probably not formatted.
  2259. //
  2260. FloppyDump(
  2261. FLOPSHOW,
  2262. ("Floppy - start drive - media type was unknown\n")
  2263. );
  2264. return STATUS_UNRECOGNIZED_MEDIA;
  2265. } else {
  2266. if ( DisketteExtension->DriveMediaType !=
  2267. DisketteExtension->LastDriveMediaType ) {
  2268. //
  2269. // Last drive/media combination accessed by the
  2270. // controller was different, so set up the controller.
  2271. //
  2272. ntStatus = FlDatarateSpecifyConfigure( DisketteExtension );
  2273. if (!NT_SUCCESS(ntStatus)) {
  2274. FloppyDump(
  2275. FLOPWARN,
  2276. ("Floppy: start drive - bad status from datarate"
  2277. "------ specify %x\n",
  2278. ntStatus)
  2279. );
  2280. }
  2281. }
  2282. }
  2283. }
  2284. }
  2285. //
  2286. // If this is a WRITE, check the drive to make sure it's not write
  2287. // protected. If so, return an error.
  2288. //
  2289. if ( ( WriteOperation ) && ( NT_SUCCESS( ntStatus ) ) ) {
  2290. DisketteExtension->FifoBuffer[0] = COMMND_SENSE_DRIVE_STATUS;
  2291. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2292. ntStatus = FlIssueCommand( DisketteExtension,
  2293. DisketteExtension->FifoBuffer,
  2294. DisketteExtension->FifoBuffer,
  2295. NULL,
  2296. 0,
  2297. 0 );
  2298. if ( !NT_SUCCESS( ntStatus ) ) {
  2299. FloppyDump(
  2300. FLOPWARN,
  2301. ("Floppy: SENSE_DRIVE returned %x\n", ntStatus)
  2302. );
  2303. return ntStatus;
  2304. }
  2305. if (IsNEC_98) {
  2306. //
  2307. // Check if media has be ejected.
  2308. //
  2309. if (!(DisketteExtension->FifoBuffer[0] & STREG3_DRIVE_READY)) {
  2310. FloppyDump(
  2311. FLOPDBGP,
  2312. ("Floppy: start drive - media is not ready\n")
  2313. );
  2314. return STATUS_NO_MEDIA_IN_DEVICE;
  2315. }
  2316. } // (IsNEC_98)
  2317. if ( DisketteExtension->FifoBuffer[0] & STREG3_WRITE_PROTECTED ) {
  2318. FloppyDump(
  2319. FLOPSHOW,
  2320. ("Floppy: start drive - media is write protected\n")
  2321. );
  2322. return STATUS_MEDIA_WRITE_PROTECTED;
  2323. }
  2324. }
  2325. return ntStatus;
  2326. }
  2327. NTSTATUS
  2328. FlDatarateSpecifyConfigure(
  2329. IN PDISKETTE_EXTENSION DisketteExtension
  2330. )
  2331. /*++
  2332. Routine Description:
  2333. This routine is called to set up the controller every time a new type
  2334. of diskette is to be accessed. It issues the CONFIGURE command if
  2335. it's available, does a SPECIFY, sets the data rate, and RECALIBRATEs
  2336. the drive.
  2337. The caller must set DisketteExtension->DriveMediaType before calling
  2338. this routine.
  2339. Arguments:
  2340. DisketteExtension - pointer to our data area for the drive to be
  2341. prepared.
  2342. Return Value:
  2343. STATUS_SUCCESS if the controller is properly prepared; appropriate
  2344. error propogated otherwise.
  2345. --*/
  2346. {
  2347. NTSTATUS ntStatus = STATUS_SUCCESS;
  2348. //
  2349. // If the controller has a CONFIGURE command, use it to enable implied
  2350. // seeks. If it doesn't, we'll find out here the first time through.
  2351. //
  2352. if ( DisketteExtension->ControllerConfigurable ) {
  2353. DisketteExtension->FifoBuffer[0] = COMMND_CONFIGURE;
  2354. DisketteExtension->FifoBuffer[1] = 0;
  2355. DisketteExtension->FifoBuffer[2] = COMMND_CONFIGURE_FIFO_THRESHOLD;
  2356. DisketteExtension->FifoBuffer[2] += COMMND_CONFIGURE_DISABLE_POLLING;
  2357. if (!DisketteExtension->DriveMediaConstants.CylinderShift) {
  2358. DisketteExtension->FifoBuffer[2] += COMMND_CONFIGURE_IMPLIED_SEEKS;
  2359. }
  2360. DisketteExtension->FifoBuffer[3] = 0;
  2361. ntStatus = FlIssueCommand( DisketteExtension,
  2362. DisketteExtension->FifoBuffer,
  2363. DisketteExtension->FifoBuffer,
  2364. NULL,
  2365. 0,
  2366. 0 );
  2367. if ( ntStatus == STATUS_DEVICE_NOT_READY ) {
  2368. DisketteExtension->ControllerConfigurable = FALSE;
  2369. ntStatus = STATUS_SUCCESS;
  2370. }
  2371. }
  2372. //
  2373. // Issue SPECIFY command to program the head load and unload
  2374. // rates, the drive step rate, and the DMA data transfer mode.
  2375. //
  2376. if ( NT_SUCCESS( ntStatus ) ||
  2377. ntStatus == STATUS_DEVICE_NOT_READY ) {
  2378. DisketteExtension->FifoBuffer[0] = COMMND_SPECIFY;
  2379. DisketteExtension->FifoBuffer[1] =
  2380. DisketteExtension->DriveMediaConstants.StepRateHeadUnloadTime;
  2381. DisketteExtension->FifoBuffer[2] =
  2382. DisketteExtension->DriveMediaConstants.HeadLoadTime;
  2383. ntStatus = FlIssueCommand( DisketteExtension,
  2384. DisketteExtension->FifoBuffer,
  2385. DisketteExtension->FifoBuffer,
  2386. NULL,
  2387. 0,
  2388. 0 );
  2389. if ( NT_SUCCESS( ntStatus ) ) {
  2390. //
  2391. // Program the data rate
  2392. //
  2393. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  2394. IOCTL_DISK_INTERNAL_SET_FDC_DATA_RATE,
  2395. &DisketteExtension->
  2396. DriveMediaConstants.DataTransferRate );
  2397. //
  2398. // Recalibrate the drive, now that we've changed all its
  2399. // parameters.
  2400. //
  2401. if (NT_SUCCESS(ntStatus)) {
  2402. ntStatus = FlRecalibrateDrive( DisketteExtension );
  2403. }
  2404. } else {
  2405. FloppyDump(
  2406. FLOPINFO,
  2407. ("Floppy: Failed specify %x\n", ntStatus)
  2408. );
  2409. }
  2410. } else {
  2411. FloppyDump(
  2412. FLOPINFO,
  2413. ("Floppy: Failed configuration %x\n", ntStatus)
  2414. );
  2415. }
  2416. if ( NT_SUCCESS( ntStatus ) ) {
  2417. DisketteExtension->LastDriveMediaType =
  2418. DisketteExtension->DriveMediaType;
  2419. } else {
  2420. DisketteExtension->LastDriveMediaType = Unknown;
  2421. FloppyDump(
  2422. FLOPINFO,
  2423. ("Floppy: Failed recalibrate %x\n", ntStatus)
  2424. );
  2425. }
  2426. return ntStatus;
  2427. }
  2428. NTSTATUS
  2429. FlRecalibrateDrive(
  2430. IN PDISKETTE_EXTENSION DisketteExtension
  2431. )
  2432. /*++
  2433. Routine Description:
  2434. This routine recalibrates a drive. It is called whenever we're
  2435. setting up to access a new diskette, and after certain errors. It
  2436. will actually recalibrate twice, since many controllers stop after
  2437. 77 steps and many disks have 80 tracks.
  2438. Arguments:
  2439. DisketteExtension - pointer to our data area for the drive to be
  2440. recalibrated.
  2441. Return Value:
  2442. STATUS_SUCCESS if the drive is successfully recalibrated; appropriate
  2443. error is propogated otherwise.
  2444. --*/
  2445. {
  2446. NTSTATUS ntStatus;
  2447. UCHAR recalibrateCount;
  2448. recalibrateCount = 0;
  2449. do {
  2450. //
  2451. // Issue the recalibrate command
  2452. //
  2453. DisketteExtension->FifoBuffer[0] = COMMND_RECALIBRATE;
  2454. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2455. ntStatus = FlIssueCommand( DisketteExtension,
  2456. DisketteExtension->FifoBuffer,
  2457. DisketteExtension->FifoBuffer,
  2458. NULL,
  2459. 0,
  2460. 0 );
  2461. if ( !NT_SUCCESS( ntStatus ) ) {
  2462. FloppyDump(
  2463. FLOPWARN,
  2464. ("Floppy: recalibrate returned %x\n", ntStatus)
  2465. );
  2466. }
  2467. if ( NT_SUCCESS( ntStatus ) ) {
  2468. if (IsNEC_98) {
  2469. UCHAR fifoBuffer[2];
  2470. //
  2471. // Procedure for media is not ready
  2472. //
  2473. fifoBuffer[0] = DisketteExtension->FifoBuffer[0];
  2474. fifoBuffer[1] = DisketteExtension->FifoBuffer[1];
  2475. //
  2476. // Sense target drive and get all data at transition of condistion.
  2477. //
  2478. DisketteExtension->FifoBuffer[0] = COMMND_SENSE_DRIVE_STATUS;
  2479. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  2480. ntStatus = FlIssueCommand( DisketteExtension,
  2481. DisketteExtension->FifoBuffer,
  2482. DisketteExtension->FifoBuffer,
  2483. NULL,
  2484. 0,
  2485. 0 );
  2486. if ( !NT_SUCCESS( ntStatus ) ) {
  2487. FloppyDump(
  2488. FLOPWARN,
  2489. ("Floppy: SENSE_DRIVE returned %x\n", ntStatus)
  2490. );
  2491. return ntStatus;
  2492. }
  2493. DisketteExtension->FifoBuffer[0] = fifoBuffer[0];
  2494. DisketteExtension->FifoBuffer[1] = fifoBuffer[1];
  2495. } // (IsNEC_98)
  2496. if ( !( DisketteExtension->FifoBuffer[0] & STREG0_SEEK_COMPLETE ) ||
  2497. ( DisketteExtension->FifoBuffer[1] != 0 ) ) {
  2498. FloppyDump(
  2499. FLOPWARN,
  2500. ("Floppy: recalibrate had bad registers\n")
  2501. );
  2502. DisketteExtension->HardwareFailed = TRUE;
  2503. ntStatus = STATUS_FLOPPY_BAD_REGISTERS;
  2504. }
  2505. }
  2506. recalibrateCount++;
  2507. } while ( ( !NT_SUCCESS( ntStatus ) ) && ( recalibrateCount < 2 ) );
  2508. FloppyDump( FLOPSHOW,
  2509. ("Floppy: FloppyRecalibrateDrive: status %x, count %d\n",
  2510. ntStatus, recalibrateCount)
  2511. );
  2512. return ntStatus;
  2513. }
  2514. NTSTATUS
  2515. FlDetermineMediaType(
  2516. IN OUT PDISKETTE_EXTENSION DisketteExtension
  2517. )
  2518. /*++
  2519. Routine Description:
  2520. This routine is called by FlStartDrive() when the media type is
  2521. unknown. It assumes the largest media supported by the drive is
  2522. available, and keeps trying lower values until it finds one that
  2523. works.
  2524. Arguments:
  2525. DisketteExtension - pointer to our data area for the drive whose
  2526. media is to checked.
  2527. Return Value:
  2528. STATUS_SUCCESS if the type of the media is determined; appropriate
  2529. error propogated otherwise.
  2530. --*/
  2531. {
  2532. NTSTATUS ntStatus;
  2533. PDRIVE_MEDIA_CONSTANTS driveMediaConstants;
  2534. BOOLEAN mediaTypesExhausted;
  2535. ULONG retries = 0;
  2536. USHORT sectorLengthCode;
  2537. PBOOT_SECTOR_INFO bootSector;
  2538. LARGE_INTEGER offset;
  2539. PIRP irp;
  2540. FloppyDump(
  2541. FLOPSHOW,
  2542. ("FlDetermineMediaType...\n")
  2543. );
  2544. DisketteExtension->IsReadOnly = FALSE;
  2545. //
  2546. // Try up to three times for read the media id.
  2547. //
  2548. for ( retries = 0; retries < 3; retries++ ) {
  2549. if (retries) {
  2550. //
  2551. // We're retrying the media determination because
  2552. // some controllers don't always want to work
  2553. // at setup. First we'll reset the device to give
  2554. // it a better chance of working.
  2555. //
  2556. FloppyDump(
  2557. FLOPINFO,
  2558. ("FlDetermineMediaType: Resetting controller\n")
  2559. );
  2560. FlInitializeControllerHardware( DisketteExtension );
  2561. }
  2562. //
  2563. // Assume that the largest supported media is in the drive. If that
  2564. // turns out to be untrue, we'll try successively smaller media types
  2565. // until we find what's really in there (or we run out and decide
  2566. // that the media isn't formatted).
  2567. //
  2568. DisketteExtension->DriveMediaType =
  2569. DriveMediaLimits[DisketteExtension->DriveType].HighestDriveMediaType;
  2570. DisketteExtension->DriveMediaConstants =
  2571. DriveMediaConstants[DisketteExtension->DriveMediaType];
  2572. mediaTypesExhausted = FALSE;
  2573. do {
  2574. if (IsNEC_98) {
  2575. sectorLengthCode = DriveMediaConstants[DisketteExtension->DriveMediaType].SectorLengthCode;
  2576. FlHdbit(DisketteExtension);
  2577. } // (IsNEC_98)
  2578. ntStatus = FlDatarateSpecifyConfigure( DisketteExtension );
  2579. if ( !NT_SUCCESS( ntStatus ) ) {
  2580. //
  2581. // The SPECIFY or CONFIGURE commands resulted in an error.
  2582. // Force ourselves out of this loop and return error.
  2583. //
  2584. FloppyDump(
  2585. FLOPINFO,
  2586. ("FlDetermineMediaType: DatarateSpecify failed %x\n", ntStatus)
  2587. );
  2588. mediaTypesExhausted = TRUE;
  2589. } else {
  2590. //
  2591. // Use the media constants table when trying to determine
  2592. // media type.
  2593. //
  2594. driveMediaConstants =
  2595. &DriveMediaConstants[DisketteExtension->DriveMediaType];
  2596. //
  2597. // Now try to read the ID from wherever we're at.
  2598. //
  2599. DisketteExtension->FifoBuffer[1] = (UCHAR)
  2600. ( DisketteExtension->DeviceUnit |
  2601. ( ( driveMediaConstants->NumberOfHeads - 1 ) << 2 ) );
  2602. DisketteExtension->FifoBuffer[0] =
  2603. COMMND_READ_ID + COMMND_OPTION_MFM;
  2604. ntStatus = FlIssueCommand( DisketteExtension,
  2605. DisketteExtension->FifoBuffer,
  2606. DisketteExtension->FifoBuffer,
  2607. NULL,
  2608. 0,
  2609. 0 );
  2610. if ((!NT_SUCCESS( ntStatus)) ||
  2611. ((DisketteExtension->FifoBuffer[0]&(~STREG0_SEEK_COMPLETE)) !=
  2612. (UCHAR)( ( DisketteExtension->DeviceUnit ) |
  2613. ((driveMediaConstants->NumberOfHeads - 1 ) << 2 ))) ||
  2614. ( DisketteExtension->FifoBuffer[1] != 0 ) ||
  2615. ( DisketteExtension->FifoBuffer[2] != 0 ) ||
  2616. ( IsNEC_98 && ( DisketteExtension->FifoBuffer[6] != sectorLengthCode ))
  2617. ) {
  2618. FloppyDump(
  2619. FLOPINFO,
  2620. ("Floppy: READID failed trying lower media\n"
  2621. "------ status = %x\n"
  2622. "------ SR0 = %x\n"
  2623. "------ SR1 = %x\n"
  2624. "------ SR2 = %x\n",
  2625. ntStatus,
  2626. DisketteExtension->FifoBuffer[0],
  2627. DisketteExtension->FifoBuffer[1],
  2628. DisketteExtension->FifoBuffer[2])
  2629. );
  2630. DisketteExtension->DriveMediaType--;
  2631. DisketteExtension->DriveMediaConstants =
  2632. DriveMediaConstants[DisketteExtension->DriveMediaType];
  2633. if (ntStatus != STATUS_DEVICE_NOT_READY) {
  2634. ntStatus = STATUS_UNRECOGNIZED_MEDIA;
  2635. }
  2636. //
  2637. // Next comparison must be signed, for when
  2638. // LowestDriveMediaType = 0.
  2639. //
  2640. if ( (CHAR)( DisketteExtension->DriveMediaType ) <
  2641. (CHAR)( DriveMediaLimits[DisketteExtension->DriveType].
  2642. LowestDriveMediaType ) ) {
  2643. DisketteExtension->MediaType = Unknown;
  2644. mediaTypesExhausted = TRUE;
  2645. FloppyDump(
  2646. FLOPINFO,
  2647. ("Floppy: Unrecognized media.\n")
  2648. );
  2649. }
  2650. } else {
  2651. if (IsNEC_98) {
  2652. //
  2653. // Read boot sector by current media type's parameters to determine.
  2654. //
  2655. DisketteExtension->MediaType = driveMediaConstants->MediaType;
  2656. DisketteExtension->BytesPerSector =
  2657. driveMediaConstants->BytesPerSector;
  2658. FloppyDump(
  2659. FLOPINFO,
  2660. ("Floppy: MediaType is %x ---\n", DisketteExtension->MediaType)
  2661. );
  2662. DisketteExtension->ByteCapacity =
  2663. ( driveMediaConstants->BytesPerSector ) *
  2664. driveMediaConstants->SectorsPerTrack *
  2665. ( 1 + driveMediaConstants->MaximumTrack ) *
  2666. driveMediaConstants->NumberOfHeads;
  2667. //
  2668. // Structure copy the media constants into the diskette extension.
  2669. //
  2670. DisketteExtension->DriveMediaConstants =
  2671. DriveMediaConstants[DisketteExtension->DriveMediaType];
  2672. //
  2673. // Check the boot sector for any overriding geometry information.
  2674. //
  2675. //FlCheckBootSector(DisketteExtension);
  2676. // Set up the IRP to read the boot sector.
  2677. bootSector = ExAllocatePool(NonPagedPoolCacheAligned, BOOT_SECTOR_SIZE);
  2678. if (!bootSector) {
  2679. return STATUS_INSUFFICIENT_RESOURCES;
  2680. }
  2681. offset.LowPart = offset.HighPart = 0;
  2682. irp = IoBuildAsynchronousFsdRequest(IRP_MJ_READ,
  2683. DisketteExtension->DeviceObject,
  2684. bootSector,
  2685. BOOT_SECTOR_SIZE,
  2686. &offset,
  2687. NULL);
  2688. if (!irp) {
  2689. FloppyDump(
  2690. FLOPWARN,
  2691. ( "Floppy: Returned from IoBuildAsynchronousFsdRequest with error\n" ));
  2692. ExFreePool(bootSector);
  2693. return STATUS_INSUFFICIENT_RESOURCES;
  2694. }
  2695. irp->CurrentLocation--;
  2696. irp->Tail.Overlay.CurrentStackLocation = IoGetNextIrpStackLocation(irp);
  2697. // Allocate an adapter channel, do read, free adapter channel.
  2698. ntStatus = FlReadWrite(DisketteExtension, irp, TRUE);
  2699. FloppyDump(
  2700. FLOPSHOW,
  2701. ( "Floppy: Read boot sector (called FlReadWrite) with status=%x\n" ,
  2702. ntStatus ));
  2703. MmUnlockPages(irp->MdlAddress);
  2704. IoFreeMdl(irp->MdlAddress);
  2705. IoFreeIrp(irp);
  2706. ExFreePool(bootSector);
  2707. if ( !NT_SUCCESS( ntStatus ) ) {
  2708. FloppyDump(
  2709. FLOPINFO,
  2710. ("Floppy: READID failed trying lower media\n"
  2711. "------ Status = %x\n"
  2712. "------ SecLen = %x\n"
  2713. "------ SR0 = %x\n"
  2714. "------ SR1 = %x\n"
  2715. "------ SR2 = %x\n"
  2716. "------ SR6 = %x\n",
  2717. ntStatus,
  2718. sectorLengthCode,
  2719. DisketteExtension->FifoBuffer[0],
  2720. DisketteExtension->FifoBuffer[1],
  2721. DisketteExtension->FifoBuffer[2],
  2722. DisketteExtension->FifoBuffer[6])
  2723. );
  2724. DisketteExtension->DriveMediaType--;
  2725. DisketteExtension->DriveMediaConstants =
  2726. DriveMediaConstants[DisketteExtension->DriveMediaType];
  2727. if (ntStatus != STATUS_DEVICE_NOT_READY) {
  2728. ntStatus = STATUS_UNRECOGNIZED_MEDIA;
  2729. }
  2730. //
  2731. // Next comparison must be signed, for when
  2732. // LowestDriveMediaType = 0.
  2733. //
  2734. if ( (CHAR)( DisketteExtension->DriveMediaType ) <
  2735. (CHAR)( DriveMediaLimits[DisketteExtension->DriveType].
  2736. LowestDriveMediaType ) ) {
  2737. DisketteExtension->MediaType = Unknown;
  2738. mediaTypesExhausted = TRUE;
  2739. FloppyDump(
  2740. FLOPINFO,
  2741. ("Floppy: Unrecognized media. 2\n")
  2742. );
  2743. }
  2744. }
  2745. } // (IsNEC_98)
  2746. }
  2747. }
  2748. } while ( ( !NT_SUCCESS( ntStatus ) ) && !( mediaTypesExhausted ) );
  2749. if (NT_SUCCESS(ntStatus)) {
  2750. //
  2751. // We determined the media type. Time to move on.
  2752. //
  2753. FloppyDump(
  2754. FLOPINFO,
  2755. ("Floppy: Determined media type %d\n", retries)
  2756. );
  2757. break;
  2758. }
  2759. }
  2760. if ( (!NT_SUCCESS( ntStatus )) || mediaTypesExhausted) {
  2761. FloppyDump(
  2762. FLOPINFO,
  2763. ("Floppy: failed determine types status = %x %s\n",
  2764. ntStatus,
  2765. mediaTypesExhausted ? "media types exhausted" : "")
  2766. );
  2767. return ntStatus;
  2768. }
  2769. DisketteExtension->MediaType = driveMediaConstants->MediaType;
  2770. DisketteExtension->BytesPerSector = driveMediaConstants->BytesPerSector;
  2771. DisketteExtension->ByteCapacity =
  2772. ( driveMediaConstants->BytesPerSector ) *
  2773. driveMediaConstants->SectorsPerTrack *
  2774. ( 1 + driveMediaConstants->MaximumTrack ) *
  2775. driveMediaConstants->NumberOfHeads;
  2776. FloppyDump(
  2777. FLOPINFO,
  2778. ("FlDetermineMediaType: MediaType is %x, bytes per sector %d, capacity %d\n",
  2779. DisketteExtension->MediaType,
  2780. DisketteExtension->BytesPerSector,
  2781. DisketteExtension->ByteCapacity)
  2782. );
  2783. //
  2784. // Structure copy the media constants into the diskette extension.
  2785. //
  2786. DisketteExtension->DriveMediaConstants =
  2787. DriveMediaConstants[DisketteExtension->DriveMediaType];
  2788. //
  2789. // Check the boot sector for any overriding geometry information.
  2790. //
  2791. FlCheckBootSector(DisketteExtension);
  2792. return ntStatus;
  2793. }
  2794. VOID
  2795. FlAllocateIoBuffer(
  2796. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  2797. IN ULONG BufferSize
  2798. )
  2799. /*++
  2800. Routine Description:
  2801. This routine allocates a PAGE_SIZE io buffer.
  2802. Arguments:
  2803. ControllerData - Supplies the controller data.
  2804. BufferSize - Supplies the number of bytes to allocate.
  2805. Return Value:
  2806. None.
  2807. --*/
  2808. {
  2809. BOOLEAN allocateContiguous;
  2810. LARGE_INTEGER maxDmaAddress;
  2811. if (DisketteExtension->IoBuffer) {
  2812. if (DisketteExtension->IoBufferSize >= BufferSize) {
  2813. return;
  2814. }
  2815. FlFreeIoBuffer(DisketteExtension);
  2816. }
  2817. if (BufferSize > DisketteExtension->MaxTransferSize ) {
  2818. allocateContiguous = TRUE;
  2819. } else {
  2820. allocateContiguous = FALSE;
  2821. }
  2822. if (allocateContiguous) {
  2823. maxDmaAddress.QuadPart = MAXIMUM_DMA_ADDRESS;
  2824. DisketteExtension->IoBuffer = MmAllocateContiguousMemory(BufferSize,
  2825. maxDmaAddress);
  2826. } else {
  2827. DisketteExtension->IoBuffer = ExAllocatePool(NonPagedPoolCacheAligned,
  2828. BufferSize);
  2829. }
  2830. if (!DisketteExtension->IoBuffer) {
  2831. return;
  2832. }
  2833. DisketteExtension->IoBufferMdl = IoAllocateMdl(DisketteExtension->IoBuffer,
  2834. BufferSize, FALSE, FALSE, NULL);
  2835. if (!DisketteExtension->IoBufferMdl) {
  2836. if (allocateContiguous) {
  2837. MmFreeContiguousMemory(DisketteExtension->IoBuffer);
  2838. } else {
  2839. ExFreePool(DisketteExtension->IoBuffer);
  2840. }
  2841. DisketteExtension->IoBuffer = NULL;
  2842. return;
  2843. }
  2844. try {
  2845. MmProbeAndLockPages(DisketteExtension->IoBufferMdl, KernelMode,
  2846. IoModifyAccess);
  2847. } except(EXCEPTION_EXECUTE_HANDLER) {
  2848. FloppyDump(FLOPWARN,
  2849. ("MmProbeAndLockPages failed. Status = %x\n",
  2850. GetExceptionCode())
  2851. );
  2852. if (allocateContiguous) {
  2853. MmFreeContiguousMemory(DisketteExtension->IoBuffer);
  2854. } else {
  2855. ExFreePool(DisketteExtension->IoBuffer);
  2856. }
  2857. DisketteExtension->IoBuffer = NULL;
  2858. return;
  2859. }
  2860. DisketteExtension->IoBufferSize = BufferSize;
  2861. }
  2862. VOID
  2863. FlFreeIoBuffer(
  2864. IN OUT PDISKETTE_EXTENSION DisketteExtension
  2865. )
  2866. /*++
  2867. Routine Description:
  2868. This routine free's the controller's IoBuffer.
  2869. Arguments:
  2870. DisketteExtension - Supplies the controller data.
  2871. Return Value:
  2872. None.
  2873. --*/
  2874. {
  2875. BOOLEAN contiguousBuffer;
  2876. if (!DisketteExtension->IoBuffer) {
  2877. return;
  2878. }
  2879. if (DisketteExtension->IoBufferSize >
  2880. DisketteExtension->MaxTransferSize) {
  2881. contiguousBuffer = TRUE;
  2882. } else {
  2883. contiguousBuffer = FALSE;
  2884. }
  2885. DisketteExtension->IoBufferSize = 0;
  2886. MmUnlockPages(DisketteExtension->IoBufferMdl);
  2887. IoFreeMdl(DisketteExtension->IoBufferMdl);
  2888. DisketteExtension->IoBufferMdl = NULL;
  2889. if (contiguousBuffer) {
  2890. MmFreeContiguousMemory(DisketteExtension->IoBuffer);
  2891. } else {
  2892. ExFreePool(DisketteExtension->IoBuffer);
  2893. }
  2894. DisketteExtension->IoBuffer = NULL;
  2895. }
  2896. VOID
  2897. FloppyThread(
  2898. PVOID Context
  2899. )
  2900. /*++
  2901. Routine Description:
  2902. This is the code executed by the system thread created when the
  2903. floppy driver initializes. This thread loops forever (or until a
  2904. flag is set telling the thread to kill itself) processing packets
  2905. put into the queue by the dispatch routines.
  2906. For each packet, this thread calls appropriate routines to process
  2907. the request, and then calls FlFinishOperation() to complete the
  2908. packet.
  2909. Arguments:
  2910. Context - a pointer to our data area for the controller being
  2911. supported (there is one thread per controller).
  2912. Return Value:
  2913. None.
  2914. --*/
  2915. {
  2916. PIRP irp;
  2917. PIO_STACK_LOCATION irpSp;
  2918. PLIST_ENTRY request;
  2919. PDISKETTE_EXTENSION disketteExtension = Context;
  2920. NTSTATUS ntStatus = STATUS_SUCCESS;
  2921. NTSTATUS waitStatus;
  2922. LARGE_INTEGER queueWait;
  2923. LARGE_INTEGER acquireWait;
  2924. ULONG inx;
  2925. //
  2926. // Set thread priority to lowest realtime level.
  2927. //
  2928. KeSetPriorityThread(KeGetCurrentThread(), LOW_REALTIME_PRIORITY);
  2929. queueWait.QuadPart = -(1 * 1000 * 10000);
  2930. acquireWait.QuadPart = -(15 * 1000 * 10000);
  2931. do {
  2932. //
  2933. // Wait for a request from the dispatch routines.
  2934. // KeWaitForSingleObject won't return error here - this thread
  2935. // isn't alertable and won't take APCs, and we're not passing in
  2936. // a timeout.
  2937. //
  2938. for (inx = 0; inx < 3; inx++) {
  2939. waitStatus = KeWaitForSingleObject(
  2940. (PVOID) &disketteExtension->RequestSemaphore,
  2941. Executive,
  2942. KernelMode,
  2943. FALSE,
  2944. &queueWait );
  2945. if (waitStatus == STATUS_TIMEOUT) {
  2946. ExAcquireFastMutex(&disketteExtension->PowerDownMutex);
  2947. if (disketteExtension->ReceivedQueryPower == TRUE) {
  2948. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  2949. break;
  2950. } else {
  2951. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  2952. }
  2953. } else {
  2954. break;
  2955. }
  2956. }
  2957. if (waitStatus == STATUS_TIMEOUT) {
  2958. if (!IsNEC_98 &&
  2959. disketteExtension->FloppyControllerAllocated) {
  2960. FloppyDump(FLOPSHOW,
  2961. ("Floppy: Timed Out - Turning off the motor\n")
  2962. );
  2963. FlFdcDeviceIo( disketteExtension->TargetObject,
  2964. IOCTL_DISK_INTERNAL_DISABLE_FDC_DEVICE,
  2965. NULL );
  2966. FlFdcDeviceIo( disketteExtension->TargetObject,
  2967. IOCTL_DISK_INTERNAL_RELEASE_FDC,
  2968. disketteExtension->DeviceObject );
  2969. disketteExtension->FloppyControllerAllocated = FALSE;
  2970. }
  2971. ExAcquireFastMutex(&disketteExtension->ThreadReferenceMutex);
  2972. if (disketteExtension->ThreadReferenceCount == 0) {
  2973. disketteExtension->ThreadReferenceCount = -1;
  2974. ASSERT(disketteExtension->FloppyThread != NULL);
  2975. //
  2976. // FloppyThread could be NULL in the unlikely event the
  2977. // ObReferenceObjectByHandle failed when we created the
  2978. // thread.
  2979. //
  2980. if (disketteExtension->FloppyThread != NULL) {
  2981. ObDereferenceObject(disketteExtension->FloppyThread);
  2982. disketteExtension->FloppyThread = NULL;
  2983. }
  2984. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  2985. if (IsNEC_98) {
  2986. if (disketteExtension->ReleaseFdcWithMotorRunning) {
  2987. ntStatus = FlFdcDeviceIo(
  2988. disketteExtension->TargetObject,
  2989. IOCTL_DISK_INTERNAL_ACQUIRE_FDC,
  2990. &acquireWait );
  2991. FlFdcDeviceIo( disketteExtension->TargetObject,
  2992. IOCTL_DISK_INTERNAL_DISABLE_FDC_DEVICE,
  2993. NULL );
  2994. FlFdcDeviceIo( disketteExtension->TargetObject,
  2995. IOCTL_DISK_INTERNAL_RELEASE_FDC,
  2996. disketteExtension->DeviceObject );
  2997. disketteExtension->FloppyControllerAllocated = FALSE;
  2998. disketteExtension->ReleaseFdcWithMotorRunning = FALSE;
  2999. }
  3000. }
  3001. FloppyPageEntireDriver();
  3002. FloppyDump(FLOPDBGP,
  3003. ("Floppy: Terminating thread in FloppyThread.\n")
  3004. );
  3005. ExAcquireFastMutex(&disketteExtension->PowerDownMutex);
  3006. if (disketteExtension->ReceivedQueryPower == TRUE) {
  3007. disketteExtension->ReceivedQueryPower = FALSE;
  3008. KeSetEvent(&disketteExtension->QueryPowerEvent,
  3009. IO_NO_INCREMENT,
  3010. FALSE);
  3011. }
  3012. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  3013. PsTerminateSystemThread( STATUS_SUCCESS );
  3014. }
  3015. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  3016. continue;
  3017. }
  3018. while (request = ExInterlockedRemoveHeadList(
  3019. &disketteExtension->ListEntry,
  3020. &disketteExtension->ListSpinLock)) {
  3021. ExAcquireFastMutex(&disketteExtension->ThreadReferenceMutex);
  3022. ASSERT(disketteExtension->ThreadReferenceCount > 0);
  3023. (disketteExtension->ThreadReferenceCount)--;
  3024. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  3025. disketteExtension->HardwareFailed = FALSE;
  3026. irp = CONTAINING_RECORD( request, IRP, Tail.Overlay.ListEntry );
  3027. //
  3028. // Verify if the system is powering down. If so we fail
  3029. // the irps.
  3030. //
  3031. ExAcquireFastMutex(&disketteExtension->PowerDownMutex);
  3032. if (disketteExtension->PoweringDown == TRUE) {
  3033. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  3034. FloppyDump( FLOPDBGP,
  3035. ("Bailing out since power irp is waiting.\n"));
  3036. irp = CONTAINING_RECORD( request, IRP, Tail.Overlay.ListEntry );
  3037. irp->IoStatus.Status = STATUS_POWER_STATE_INVALID;
  3038. irp->IoStatus.Information = 0;
  3039. IoCompleteRequest(irp, IO_NO_INCREMENT);
  3040. continue;
  3041. }
  3042. ExReleaseFastMutex(&disketteExtension->PowerDownMutex);
  3043. FloppyDump( FLOPSHOW, ("No power irp waiting.\n"));
  3044. irpSp = IoGetCurrentIrpStackLocation( irp );
  3045. FloppyDump(
  3046. FLOPIRPPATH,
  3047. ("Floppy: Starting up IRP: %p for extension %p\n",
  3048. irp,irpSp->Parameters.Others.Argument4)
  3049. );
  3050. switch ( irpSp->MajorFunction ) {
  3051. case IRP_MJ_PNP:
  3052. FloppyDump( FLOPSHOW, ("FloppyThread: IRP_MN_QUERY_REMOVE_DEVICE\n") );
  3053. if ( irpSp->MinorFunction == IRP_MN_QUERY_REMOVE_DEVICE ||
  3054. irpSp->MinorFunction == IRP_MN_QUERY_STOP_DEVICE ) {
  3055. if (IsNEC_98) {
  3056. if (disketteExtension->ReleaseFdcWithMotorRunning) {
  3057. FlFdcDeviceIo( disketteExtension->TargetObject,
  3058. IOCTL_DISK_INTERNAL_ACQUIRE_FDC,
  3059. &acquireWait );
  3060. disketteExtension->ReleaseFdcWithMotorRunning = FALSE;
  3061. disketteExtension->FloppyControllerAllocated = TRUE;
  3062. }
  3063. }
  3064. if ( disketteExtension->FloppyControllerAllocated ) {
  3065. FlFdcDeviceIo( disketteExtension->TargetObject,
  3066. IOCTL_DISK_INTERNAL_DISABLE_FDC_DEVICE,
  3067. NULL );
  3068. FlFdcDeviceIo( disketteExtension->TargetObject,
  3069. IOCTL_DISK_INTERNAL_RELEASE_FDC,
  3070. disketteExtension->DeviceObject );
  3071. disketteExtension->FloppyControllerAllocated = FALSE;
  3072. }
  3073. ExAcquireFastMutex( &disketteExtension->ThreadReferenceMutex );
  3074. ASSERT(disketteExtension->ThreadReferenceCount == 0);
  3075. disketteExtension->ThreadReferenceCount = -1;
  3076. ExReleaseFastMutex(&disketteExtension->ThreadReferenceMutex);
  3077. FloppyPageEntireDriver();
  3078. PsTerminateSystemThread( STATUS_SUCCESS );
  3079. } else {
  3080. ntStatus = STATUS_INVALID_DEVICE_REQUEST;
  3081. }
  3082. break;
  3083. case IRP_MJ_READ:
  3084. case IRP_MJ_WRITE: {
  3085. //
  3086. // Get the diskette extension from where it was hidden
  3087. // in the IRP.
  3088. //
  3089. // disketteExtension = (PDISKETTE_EXTENSION)
  3090. // irpSp->Parameters.Others.Argument4;
  3091. if (!disketteExtension->FloppyControllerAllocated) {
  3092. ntStatus = FlFdcDeviceIo(
  3093. disketteExtension->TargetObject,
  3094. IOCTL_DISK_INTERNAL_ACQUIRE_FDC,
  3095. &acquireWait );
  3096. if (NT_SUCCESS(ntStatus)) {
  3097. disketteExtension->FloppyControllerAllocated = TRUE;
  3098. if (IsNEC_98) disketteExtension->ReleaseFdcWithMotorRunning = FALSE;
  3099. } else {
  3100. break;
  3101. }
  3102. }
  3103. //
  3104. // Until the file system clears the DO_VERIFY_VOLUME
  3105. // flag, we should return all requests with error.
  3106. //
  3107. if (( disketteExtension->DeviceObject->Flags &
  3108. DO_VERIFY_VOLUME ) &&
  3109. !(irpSp->Flags & SL_OVERRIDE_VERIFY_VOLUME))
  3110. {
  3111. FloppyDump(
  3112. FLOPINFO,
  3113. ("Floppy: clearing queue; verify required\n")
  3114. );
  3115. //
  3116. // The disk changed, and we set this bit. Fail
  3117. // all current IRPs for this device; when all are
  3118. // returned, the file system will clear
  3119. // DO_VERIFY_VOLUME.
  3120. //
  3121. ntStatus = STATUS_VERIFY_REQUIRED;
  3122. } else {
  3123. ntStatus = FlReadWrite( disketteExtension, irp, FALSE );
  3124. }
  3125. break;
  3126. }
  3127. case IRP_MJ_DEVICE_CONTROL: {
  3128. // disketteExtension = (PDISKETTE_EXTENSION)
  3129. // irpSp->Parameters.DeviceIoControl.Type3InputBuffer;
  3130. if (!disketteExtension->FloppyControllerAllocated) {
  3131. ntStatus = FlFdcDeviceIo(
  3132. disketteExtension->TargetObject,
  3133. IOCTL_DISK_INTERNAL_ACQUIRE_FDC,
  3134. &acquireWait );
  3135. if (NT_SUCCESS(ntStatus)) {
  3136. disketteExtension->FloppyControllerAllocated = TRUE;
  3137. if (IsNEC_98) disketteExtension->ReleaseFdcWithMotorRunning = FALSE;
  3138. } else {
  3139. break;
  3140. }
  3141. }
  3142. //
  3143. // Until the file system clears the DO_VERIFY_VOLUME
  3144. // flag, we should return all requests with error.
  3145. //
  3146. if (( disketteExtension->DeviceObject->Flags &
  3147. DO_VERIFY_VOLUME ) &&
  3148. !(irpSp->Flags & SL_OVERRIDE_VERIFY_VOLUME))
  3149. {
  3150. FloppyDump(
  3151. FLOPINFO,
  3152. ("Floppy: clearing queue; verify required\n")
  3153. );
  3154. //
  3155. // The disk changed, and we set this bit. Fail
  3156. // all current IRPs; when all are returned, the
  3157. // file system will clear DO_VERIFY_VOLUME.
  3158. //
  3159. ntStatus = STATUS_VERIFY_REQUIRED;
  3160. } else {
  3161. switch (
  3162. irpSp->Parameters.DeviceIoControl.IoControlCode ) {
  3163. case IOCTL_STORAGE_CHECK_VERIFY:
  3164. case IOCTL_DISK_CHECK_VERIFY: {
  3165. //
  3166. // Just start the drive; it will
  3167. // automatically check whether or not the
  3168. // disk has been changed.
  3169. //
  3170. FloppyDump(
  3171. FLOPSHOW,
  3172. ("Floppy: IOCTL_DISK_CHECK_VERIFY called\n")
  3173. );
  3174. //
  3175. // IgnoreChange should be TRUE if
  3176. // SL_OVERRIDE_VERIFY_VOLUME flag is set
  3177. // in the irp. Else, IgnoreChange should
  3178. // be FALSE
  3179. //
  3180. ntStatus = FlStartDrive(
  3181. disketteExtension,
  3182. irp,
  3183. FALSE,
  3184. FALSE,
  3185. (BOOLEAN)!!(irpSp->Flags &
  3186. SL_OVERRIDE_VERIFY_VOLUME));
  3187. break;
  3188. }
  3189. case IOCTL_DISK_IS_WRITABLE: {
  3190. //
  3191. // Start the drive with the WriteOperation
  3192. // flag set to TRUE.
  3193. //
  3194. FloppyDump(
  3195. FLOPSHOW,
  3196. ("Floppy: IOCTL_DISK_IS_WRITABLE called\n")
  3197. );
  3198. if (disketteExtension->IsReadOnly) {
  3199. ntStatus = STATUS_INVALID_PARAMETER;
  3200. } else {
  3201. ntStatus = FlStartDrive(
  3202. disketteExtension,
  3203. irp,
  3204. TRUE,
  3205. FALSE,
  3206. TRUE);
  3207. }
  3208. break;
  3209. }
  3210. case IOCTL_DISK_GET_DRIVE_GEOMETRY: {
  3211. FloppyDump(
  3212. FLOPSHOW,
  3213. ("Floppy: IOCTL_DISK_GET_DRIVE_GEOMETRY\n")
  3214. );
  3215. //
  3216. // If there's enough room to write the
  3217. // data, start the drive to make sure we
  3218. // know what type of media is in the drive.
  3219. //
  3220. if ( irpSp->Parameters.DeviceIoControl.
  3221. OutputBufferLength <
  3222. sizeof( DISK_GEOMETRY ) ) {
  3223. ntStatus = STATUS_INVALID_PARAMETER;
  3224. } else {
  3225. ntStatus = FlStartDrive(
  3226. disketteExtension,
  3227. irp,
  3228. FALSE,
  3229. TRUE,
  3230. (BOOLEAN)!!(irpSp->Flags &
  3231. SL_OVERRIDE_VERIFY_VOLUME));
  3232. }
  3233. //
  3234. // If the media wasn't formatted, FlStartDrive
  3235. // returned STATUS_UNRECOGNIZED_MEDIA.
  3236. //
  3237. if ( NT_SUCCESS( ntStatus ) ||
  3238. ( ntStatus == STATUS_UNRECOGNIZED_MEDIA )) {
  3239. PDISK_GEOMETRY outputBuffer =
  3240. (PDISK_GEOMETRY)
  3241. irp->AssociatedIrp.SystemBuffer;
  3242. // Always return the media type, even if
  3243. // it's unknown.
  3244. //
  3245. ntStatus = STATUS_SUCCESS;
  3246. outputBuffer->MediaType =
  3247. disketteExtension->MediaType;
  3248. //
  3249. // The rest of the fields only have meaning
  3250. // if the media type is known.
  3251. //
  3252. if ( disketteExtension->MediaType ==
  3253. Unknown ) {
  3254. FloppyDump(
  3255. FLOPSHOW,
  3256. ("Floppy: geometry unknown\n")
  3257. );
  3258. //
  3259. // Just zero out everything. The
  3260. // caller shouldn't look at it.
  3261. //
  3262. outputBuffer->Cylinders.LowPart = 0;
  3263. outputBuffer->Cylinders.HighPart = 0;
  3264. outputBuffer->TracksPerCylinder = 0;
  3265. outputBuffer->SectorsPerTrack = 0;
  3266. outputBuffer->BytesPerSector = 0;
  3267. } else {
  3268. //
  3269. // Return the geometry of the current
  3270. // media.
  3271. //
  3272. FloppyDump(
  3273. FLOPSHOW,
  3274. ("Floppy: geomentry is known\n")
  3275. );
  3276. outputBuffer->Cylinders.LowPart =
  3277. disketteExtension->
  3278. DriveMediaConstants.MaximumTrack + 1;
  3279. outputBuffer->Cylinders.HighPart = 0;
  3280. outputBuffer->TracksPerCylinder =
  3281. disketteExtension->
  3282. DriveMediaConstants.NumberOfHeads;
  3283. outputBuffer->SectorsPerTrack =
  3284. disketteExtension->
  3285. DriveMediaConstants.SectorsPerTrack;
  3286. outputBuffer->BytesPerSector =
  3287. disketteExtension->
  3288. DriveMediaConstants.BytesPerSector;
  3289. }
  3290. FloppyDump(
  3291. FLOPSHOW,
  3292. ("Floppy: Geometry\n"
  3293. "------- Cylinders low: 0x%x\n"
  3294. "------- Cylinders high: 0x%x\n"
  3295. "------- Track/Cyl: 0x%x\n"
  3296. "------- Sectors/Track: 0x%x\n"
  3297. "------- Bytes/Sector: 0x%x\n"
  3298. "------- Media Type: %d\n",
  3299. outputBuffer->Cylinders.LowPart,
  3300. outputBuffer->Cylinders.HighPart,
  3301. outputBuffer->TracksPerCylinder,
  3302. outputBuffer->SectorsPerTrack,
  3303. outputBuffer->BytesPerSector,
  3304. outputBuffer->MediaType)
  3305. );
  3306. }
  3307. irp->IoStatus.Information =
  3308. sizeof( DISK_GEOMETRY );
  3309. break;
  3310. }
  3311. case IOCTL_DISK_FORMAT_TRACKS_EX:
  3312. case IOCTL_DISK_FORMAT_TRACKS: {
  3313. FloppyDump(
  3314. FLOPSHOW,
  3315. ("Floppy: IOCTL_DISK_FORMAT_TRACKS\n")
  3316. );
  3317. //
  3318. // Start the drive, and make sure it's not
  3319. // write protected.
  3320. //
  3321. ntStatus = FlStartDrive(
  3322. disketteExtension,
  3323. irp,
  3324. TRUE,
  3325. FALSE,
  3326. FALSE );
  3327. //
  3328. // Note that FlStartDrive could have returned
  3329. // STATUS_UNRECOGNIZED_MEDIA if the drive
  3330. // wasn't formatted.
  3331. //
  3332. if ( NT_SUCCESS( ntStatus ) ||
  3333. ( ntStatus == STATUS_UNRECOGNIZED_MEDIA)) {
  3334. //
  3335. // We need a single page to do FORMATs.
  3336. // If we already allocated a buffer,
  3337. // we'll use that. If not, let's
  3338. // allocate a single page. Note that
  3339. // we'd have to do this anyway if there's
  3340. // not enough map registers.
  3341. //
  3342. FlAllocateIoBuffer( disketteExtension,
  3343. PAGE_SIZE);
  3344. if (disketteExtension->IoBuffer) {
  3345. ntStatus = FlFormat(disketteExtension,
  3346. irp);
  3347. } else {
  3348. ntStatus = STATUS_INSUFFICIENT_RESOURCES;
  3349. }
  3350. }
  3351. break;
  3352. } //end of case format
  3353. case IOCTL_DISK_SENSE_DEVICE: {
  3354. if (IsNEC_98) {
  3355. disketteExtension->FifoBuffer[0] = COMMND_SENSE_DRIVE_STATUS;
  3356. disketteExtension->FifoBuffer[1] = disketteExtension->DeviceUnit;
  3357. ntStatus = FlIssueCommand( disketteExtension,
  3358. disketteExtension->FifoBuffer,
  3359. disketteExtension->FifoBuffer,
  3360. NULL,
  3361. 0,
  3362. 0 );
  3363. if ( NT_SUCCESS( ntStatus ) ) {
  3364. PSENSE_DEVISE_STATUS_PTOS outputBuffer
  3365. =(PSENSE_DEVISE_STATUS_PTOS)irp->AssociatedIrp.SystemBuffer;
  3366. ((PSENSE_DEVISE_STATUS_PTOS)outputBuffer)->ST3_PTOS
  3367. =Result_Status3_PTOS[0].ST3_PTOS;
  3368. irp->IoStatus.Information = sizeof( SENSE_DEVISE_STATUS_PTOS );
  3369. }
  3370. break;
  3371. }
  3372. } //end of case sense device
  3373. } //end of switch controlcode
  3374. }
  3375. break;
  3376. } //end of case IOCTL
  3377. default: {
  3378. FloppyDump(
  3379. FLOPDBGP,
  3380. ("Floppy: bad majorfunction %x\n",irpSp->MajorFunction)
  3381. );
  3382. ntStatus = STATUS_NOT_IMPLEMENTED;
  3383. }
  3384. } //end of switch on majorfunction
  3385. if (ntStatus == STATUS_DEVICE_BUSY) {
  3386. // If the status is DEVICE_BUSY then this indicates that the
  3387. // qic117 has control of the controller. Therefore complete
  3388. // all remaining requests with STATUS_DEVICE_BUSY.
  3389. for (;;) {
  3390. disketteExtension->HardwareFailed = FALSE;
  3391. irp->IoStatus.Status = STATUS_DEVICE_BUSY;
  3392. IoCompleteRequest(irp, IO_DISK_INCREMENT);
  3393. request = ExInterlockedRemoveHeadList(
  3394. &disketteExtension->ListEntry,
  3395. &disketteExtension->ListSpinLock );
  3396. if (!request) {
  3397. break;
  3398. }
  3399. ExAcquireFastMutex(
  3400. &disketteExtension->ThreadReferenceMutex);
  3401. ASSERT(disketteExtension->ThreadReferenceCount > 0);
  3402. (disketteExtension->ThreadReferenceCount)--;
  3403. ExReleaseFastMutex(
  3404. &disketteExtension->ThreadReferenceMutex);
  3405. irp = CONTAINING_RECORD( request,
  3406. IRP,
  3407. Tail.Overlay.ListEntry);
  3408. }
  3409. } else {
  3410. //
  3411. // All operations leave a final status in ntStatus. Copy it
  3412. // to the IRP, and then complete the operation.
  3413. //
  3414. irp->IoStatus.Status = ntStatus;
  3415. //
  3416. // If we'd allocated an I/O buffer free it now
  3417. //
  3418. if (disketteExtension->IoBuffer) {
  3419. FlFreeIoBuffer(disketteExtension);
  3420. }
  3421. FlFinishOperation( irp, disketteExtension );
  3422. }
  3423. } // while there are packets to process
  3424. if (IsNEC_98) {
  3425. if (disketteExtension->FloppyControllerAllocated) {
  3426. FlFdcDeviceIo( disketteExtension->TargetObject,
  3427. IOCTL_DISK_INTERNAL_RELEASE_FDC,
  3428. disketteExtension->DeviceObject );
  3429. disketteExtension->FloppyControllerAllocated = FALSE;
  3430. disketteExtension->ReleaseFdcWithMotorRunning = TRUE;
  3431. }
  3432. }
  3433. } while ( TRUE );
  3434. }
  3435. VOID
  3436. FlConsolidateMediaTypeWithBootSector(
  3437. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  3438. IN PBOOT_SECTOR_INFO BootSector
  3439. )
  3440. /*++
  3441. Routine Description:
  3442. This routine adjusts the DisketteExtension data according
  3443. to the BPB values if this is appropriate.
  3444. Arguments:
  3445. DisketteExtension - Supplies the diskette extension.
  3446. BootSector - Supplies the boot sector information.
  3447. Return Value:
  3448. None.
  3449. --*/
  3450. {
  3451. USHORT bpbNumberOfSectors, bpbNumberOfHeads;
  3452. USHORT bpbSectorsPerTrack, bpbBytesPerSector;
  3453. USHORT bpbMediaByte, bpbMaximumTrack;
  3454. MEDIA_TYPE bpbMediaType;
  3455. ULONG i, n;
  3456. PDRIVE_MEDIA_CONSTANTS readidDriveMediaConstants;
  3457. BOOLEAN changeToBpbMedia;
  3458. FloppyDump(
  3459. FLOPSHOW,
  3460. ("Floppy: First sector read: media descriptor is: 0x%x\n",
  3461. BootSector->MediaByte[0])
  3462. );
  3463. if (BootSector->JumpByte[0] != 0xeb &&
  3464. BootSector->JumpByte[0] != 0xe9) {
  3465. // This is not a formatted floppy so ignore the BPB.
  3466. return;
  3467. }
  3468. bpbNumberOfSectors = BootSector->NumberOfSectors[1]*0x100 +
  3469. BootSector->NumberOfSectors[0];
  3470. bpbNumberOfHeads = BootSector->NumberOfHeads[1]*0x100 +
  3471. BootSector->NumberOfHeads[0];
  3472. bpbSectorsPerTrack = BootSector->SectorsPerTrack[1]*0x100 +
  3473. BootSector->SectorsPerTrack[0];
  3474. bpbBytesPerSector = BootSector->BytesPerSector[1]*0x100 +
  3475. BootSector->BytesPerSector[0];
  3476. bpbMediaByte = BootSector->MediaByte[0];
  3477. if (!bpbNumberOfHeads || !bpbSectorsPerTrack) {
  3478. // Invalid BPB, avoid dividing by zero.
  3479. return;
  3480. }
  3481. bpbMaximumTrack =
  3482. bpbNumberOfSectors/bpbNumberOfHeads/bpbSectorsPerTrack - 1;
  3483. // First figure out if this BPB specifies a known media type
  3484. // independantly of the current drive type.
  3485. bpbMediaType = Unknown;
  3486. for (i = 0; i < NUMBER_OF_DRIVE_MEDIA_COMBINATIONS; i++) {
  3487. if (bpbBytesPerSector == DriveMediaConstants[i].BytesPerSector &&
  3488. bpbSectorsPerTrack == DriveMediaConstants[i].SectorsPerTrack &&
  3489. bpbMaximumTrack == DriveMediaConstants[i].MaximumTrack &&
  3490. bpbNumberOfHeads == DriveMediaConstants[i].NumberOfHeads &&
  3491. bpbMediaByte == DriveMediaConstants[i].MediaByte) {
  3492. bpbMediaType = DriveMediaConstants[i].MediaType;
  3493. break;
  3494. }
  3495. }
  3496. //
  3497. // If DriveType is 3.5", we change 5.25" to 3.5".
  3498. // The following media's BPB is the same between 5.25" and 3.5",
  3499. // so 5.25" media types are always found first.
  3500. //
  3501. if (DisketteExtension->DriveType == DRIVE_TYPE_1440) {
  3502. switch (bpbMediaType) {
  3503. case F5_640_512: bpbMediaType = F3_640_512; break;
  3504. case F5_720_512: bpbMediaType = F3_720_512; break;
  3505. case F5_1Pt2_512: bpbMediaType = F3_1Pt2_512; break;
  3506. case F5_1Pt23_1024: bpbMediaType = F3_1Pt23_1024; break;
  3507. default: break;
  3508. }
  3509. }
  3510. FloppyDump(
  3511. FLOPSHOW,
  3512. ("FLOPPY: After switch media type is: %x\n",bpbMediaType)
  3513. );
  3514. FloppyDump(
  3515. FLOPINFO,
  3516. ("FloppyBpb: Media type ")
  3517. );
  3518. if (bpbMediaType == DisketteExtension->MediaType) {
  3519. // No conflict between BPB and readId result.
  3520. changeToBpbMedia = FALSE;
  3521. FloppyDump(
  3522. FLOPINFO,
  3523. ("is same\n")
  3524. );
  3525. } else {
  3526. // There is a conflict between the BPB and the readId
  3527. // media type. If the new parameters are acceptable
  3528. // then go with them.
  3529. readidDriveMediaConstants = &(DisketteExtension->DriveMediaConstants);
  3530. if (bpbBytesPerSector == readidDriveMediaConstants->BytesPerSector &&
  3531. bpbSectorsPerTrack < 0x100 &&
  3532. bpbMaximumTrack == readidDriveMediaConstants->MaximumTrack &&
  3533. bpbNumberOfHeads <= readidDriveMediaConstants->NumberOfHeads) {
  3534. changeToBpbMedia = TRUE;
  3535. } else {
  3536. changeToBpbMedia = FALSE;
  3537. }
  3538. FloppyDump( FLOPINFO,
  3539. ("%s",
  3540. changeToBpbMedia ?
  3541. "will change to Bpb\n" : "will not change\n")
  3542. );
  3543. // If we didn't derive a new media type from the BPB then
  3544. // just use the one from readId. Also override any
  3545. // skew compensation since we don't really know anything
  3546. // about this new media type.
  3547. if (bpbMediaType == Unknown) {
  3548. bpbMediaType = readidDriveMediaConstants->MediaType;
  3549. DisketteExtension->DriveMediaConstants.SkewDelta = 0;
  3550. }
  3551. }
  3552. if (changeToBpbMedia) {
  3553. // Change the DriveMediaType only if this new media type
  3554. // falls in line with what is supported by the drive.
  3555. i = DriveMediaLimits[DisketteExtension->DriveType].LowestDriveMediaType;
  3556. n = DriveMediaLimits[DisketteExtension->DriveType].HighestDriveMediaType;
  3557. for (; i <= n; i++) {
  3558. if (bpbMediaType == DriveMediaConstants[i].MediaType) {
  3559. DisketteExtension->DriveMediaType = i;
  3560. break;
  3561. }
  3562. }
  3563. DisketteExtension->MediaType = bpbMediaType;
  3564. DisketteExtension->ByteCapacity = bpbNumberOfSectors*bpbBytesPerSector;
  3565. DisketteExtension->DriveMediaConstants.SectorsPerTrack =
  3566. (UCHAR) bpbSectorsPerTrack;
  3567. DisketteExtension->DriveMediaConstants.NumberOfHeads =
  3568. (UCHAR) bpbNumberOfHeads;
  3569. // If the MSDMF3. signature is there then make this floppy
  3570. // read-only.
  3571. if (RtlCompareMemory(BootSector->OemData, "MSDMF3.", 7) == 7) {
  3572. DisketteExtension->IsReadOnly = TRUE;
  3573. }
  3574. }
  3575. }
  3576. VOID
  3577. FlCheckBootSector(
  3578. IN OUT PDISKETTE_EXTENSION DisketteExtension
  3579. )
  3580. /*++
  3581. Routine Description:
  3582. This routine reads the boot sector and then figures
  3583. out whether or not the boot sector contains new geometry
  3584. information.
  3585. Arguments:
  3586. DisketteExtension - Supplies the diskette extension.
  3587. Return Value:
  3588. None.
  3589. --*/
  3590. {
  3591. PBOOT_SECTOR_INFO bootSector;
  3592. LARGE_INTEGER offset;
  3593. PIRP irp;
  3594. NTSTATUS status;
  3595. // Set up the IRP to read the boot sector.
  3596. bootSector = ExAllocatePool(NonPagedPoolCacheAligned, BOOT_SECTOR_SIZE);
  3597. if (!bootSector) {
  3598. return;
  3599. }
  3600. offset.LowPart = offset.HighPart = 0;
  3601. irp = IoBuildAsynchronousFsdRequest(IRP_MJ_READ,
  3602. DisketteExtension->DeviceObject,
  3603. bootSector,
  3604. BOOT_SECTOR_SIZE,
  3605. &offset,
  3606. NULL);
  3607. if (!irp) {
  3608. ExFreePool(bootSector);
  3609. return;
  3610. }
  3611. irp->CurrentLocation--;
  3612. irp->Tail.Overlay.CurrentStackLocation = IoGetNextIrpStackLocation(irp);
  3613. // Allocate an adapter channel, do read, free adapter channel.
  3614. status = FlReadWrite(DisketteExtension, irp, TRUE);
  3615. MmUnlockPages(irp->MdlAddress);
  3616. IoFreeMdl(irp->MdlAddress);
  3617. IoFreeIrp(irp);
  3618. ExFreePool(bootSector);
  3619. }
  3620. NTSTATUS
  3621. FlReadWriteTrack(
  3622. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  3623. IN OUT PMDL IoMdl,
  3624. IN OUT ULONG IoOffset,
  3625. IN BOOLEAN WriteOperation,
  3626. IN UCHAR Cylinder,
  3627. IN UCHAR Head,
  3628. IN UCHAR Sector,
  3629. IN UCHAR NumberOfSectors,
  3630. IN BOOLEAN NeedSeek
  3631. )
  3632. /*++
  3633. Routine Description:
  3634. This routine reads a portion of a track. It transfers the to or from the
  3635. device from or to the given IoBuffer and IoMdl.
  3636. Arguments:
  3637. DisketteExtension - Supplies the diskette extension.
  3638. IoMdl - Supplies the Mdl for transfering from/to the device.
  3639. IoBuffer - Supplies the buffer to transfer from/to the device.
  3640. WriteOperation - Supplies whether or not this is a write operation.
  3641. Cylinder - Supplies the cylinder number for this track.
  3642. Head - Supplies the head number for this track.
  3643. Sector - Supplies the starting sector of the transfer.
  3644. NumberOfSectors - Supplies the number of sectors to transfer.
  3645. NeedSeek - Supplies whether or not we need to do a seek.
  3646. Return Value:
  3647. An NTSTATUS code.
  3648. --*/
  3649. {
  3650. PDRIVE_MEDIA_CONSTANTS driveMediaConstants;
  3651. ULONG byteToSectorShift;
  3652. ULONG transferBytes;
  3653. LARGE_INTEGER headSettleTime;
  3654. NTSTATUS status;
  3655. ULONG seekRetry, ioRetry;
  3656. BOOLEAN recalibrateDrive = FALSE;
  3657. UCHAR i;
  3658. FloppyDump( FLOPSHOW,
  3659. ("\nFlReadWriteTrack:%sseek for %s at chs %d/%d/%d for %d sectors\n",
  3660. NeedSeek ? " need " : " ",
  3661. WriteOperation ? "write" : "read",
  3662. Cylinder,
  3663. Head,
  3664. Sector,
  3665. NumberOfSectors) );
  3666. driveMediaConstants = &DisketteExtension->DriveMediaConstants;
  3667. byteToSectorShift = SECTORLENGTHCODE_TO_BYTESHIFT +
  3668. driveMediaConstants->SectorLengthCode;
  3669. transferBytes = ((ULONG) NumberOfSectors)<<byteToSectorShift;
  3670. headSettleTime.LowPart = -(10*1000*driveMediaConstants->HeadSettleTime);
  3671. headSettleTime.HighPart = -1;
  3672. for (seekRetry = 0, ioRetry = 0; seekRetry < 3; seekRetry++) {
  3673. if (recalibrateDrive) {
  3674. // Something failed, so recalibrate the drive.
  3675. FloppyDump(
  3676. FLOPINFO,
  3677. ("FlReadWriteTrack: performing recalibrate\n")
  3678. );
  3679. FlRecalibrateDrive(DisketteExtension);
  3680. }
  3681. // Do a seek if we have to.
  3682. if (recalibrateDrive ||
  3683. (NeedSeek &&
  3684. (!DisketteExtension->ControllerConfigurable ||
  3685. driveMediaConstants->CylinderShift != 0))) {
  3686. DisketteExtension->FifoBuffer[0] = COMMND_SEEK;
  3687. DisketteExtension->FifoBuffer[1] = (Head<<2) |
  3688. DisketteExtension->DeviceUnit;
  3689. DisketteExtension->FifoBuffer[2] = Cylinder<<
  3690. driveMediaConstants->CylinderShift;
  3691. status = FlIssueCommand( DisketteExtension,
  3692. DisketteExtension->FifoBuffer,
  3693. DisketteExtension->FifoBuffer,
  3694. NULL,
  3695. 0,
  3696. 0 );
  3697. if (NT_SUCCESS(status)) {
  3698. // Check the completion state of the controller.
  3699. if (!(DisketteExtension->FifoBuffer[0]&STREG0_SEEK_COMPLETE) ||
  3700. DisketteExtension->FifoBuffer[1] !=
  3701. Cylinder<<driveMediaConstants->CylinderShift) {
  3702. DisketteExtension->HardwareFailed = TRUE;
  3703. status = STATUS_FLOPPY_BAD_REGISTERS;
  3704. }
  3705. if (NT_SUCCESS(status)) {
  3706. // Delay after doing seek.
  3707. KeDelayExecutionThread(KernelMode, FALSE, &headSettleTime);
  3708. // SEEKs should always be followed by a READID.
  3709. DisketteExtension->FifoBuffer[0] =
  3710. COMMND_READ_ID + COMMND_OPTION_MFM;
  3711. DisketteExtension->FifoBuffer[1] =
  3712. (Head<<2) | DisketteExtension->DeviceUnit;
  3713. status = FlIssueCommand( DisketteExtension,
  3714. DisketteExtension->FifoBuffer,
  3715. DisketteExtension->FifoBuffer,
  3716. NULL,
  3717. 0,
  3718. 0 );
  3719. if (NT_SUCCESS(status)) {
  3720. if (IsNEC_98) {
  3721. if(DisketteExtension->FifoBuffer[0] & STREG0_DRIVE_NOT_READY) {
  3722. return STATUS_DEVICE_NOT_READY;
  3723. }
  3724. } // (IsNEC_98)
  3725. if (DisketteExtension->FifoBuffer[0] !=
  3726. ((Head<<2) | DisketteExtension->DeviceUnit) ||
  3727. DisketteExtension->FifoBuffer[1] != 0 ||
  3728. DisketteExtension->FifoBuffer[2] != 0 ||
  3729. DisketteExtension->FifoBuffer[3] != Cylinder) {
  3730. DisketteExtension->HardwareFailed = TRUE;
  3731. status = FlInterpretError(
  3732. DisketteExtension->FifoBuffer[1],
  3733. DisketteExtension->FifoBuffer[2]);
  3734. }
  3735. } else {
  3736. FloppyDump(
  3737. FLOPINFO,
  3738. ("FlReadWriteTrack: Read ID failed %x\n", status)
  3739. );
  3740. }
  3741. }
  3742. } else {
  3743. FloppyDump(
  3744. FLOPINFO,
  3745. ("FlReadWriteTrack: SEEK failed %x\n", status)
  3746. );
  3747. }
  3748. } else {
  3749. status = STATUS_SUCCESS;
  3750. }
  3751. if (!NT_SUCCESS(status)) {
  3752. // The seek failed so try again.
  3753. FloppyDump(
  3754. FLOPINFO,
  3755. ("FlReadWriteTrack: setup failure %x - recalibrating\n", status)
  3756. );
  3757. recalibrateDrive = TRUE;
  3758. continue;
  3759. }
  3760. for (;; ioRetry++) {
  3761. //
  3762. // Issue the READ or WRITE command
  3763. //
  3764. DisketteExtension->FifoBuffer[1] = (Head<<2) |
  3765. DisketteExtension->DeviceUnit;
  3766. DisketteExtension->FifoBuffer[2] = Cylinder;
  3767. DisketteExtension->FifoBuffer[3] = Head;
  3768. DisketteExtension->FifoBuffer[4] = Sector + 1;
  3769. DisketteExtension->FifoBuffer[5] =
  3770. driveMediaConstants->SectorLengthCode;
  3771. DisketteExtension->FifoBuffer[6] = Sector + NumberOfSectors;
  3772. DisketteExtension->FifoBuffer[7] =
  3773. driveMediaConstants->ReadWriteGapLength;
  3774. DisketteExtension->FifoBuffer[8] = driveMediaConstants->DataLength;
  3775. if (WriteOperation) {
  3776. DisketteExtension->FifoBuffer[0] =
  3777. COMMND_WRITE_DATA + COMMND_OPTION_MFM;
  3778. } else {
  3779. DisketteExtension->FifoBuffer[0] =
  3780. COMMND_READ_DATA + COMMND_OPTION_MFM;
  3781. }
  3782. FloppyDump(FLOPINFO,
  3783. ("FlReadWriteTrack: Params - %x,%x,%x,%x,%x,%x,%x,%x\n",
  3784. DisketteExtension->FifoBuffer[1],
  3785. DisketteExtension->FifoBuffer[2],
  3786. DisketteExtension->FifoBuffer[3],
  3787. DisketteExtension->FifoBuffer[4],
  3788. DisketteExtension->FifoBuffer[5],
  3789. DisketteExtension->FifoBuffer[6],
  3790. DisketteExtension->FifoBuffer[7],
  3791. DisketteExtension->FifoBuffer[8])
  3792. );
  3793. status = FlIssueCommand( DisketteExtension,
  3794. DisketteExtension->FifoBuffer,
  3795. DisketteExtension->FifoBuffer,
  3796. IoMdl,
  3797. IoOffset,
  3798. transferBytes );
  3799. if (NT_SUCCESS(status)) {
  3800. if (IsNEC_98) {
  3801. if(DisketteExtension->FifoBuffer[0] & STREG0_DRIVE_NOT_READY) {
  3802. return STATUS_DEVICE_NOT_READY;
  3803. }
  3804. } // (IsNEC_98)
  3805. if ((DisketteExtension->FifoBuffer[0] & STREG0_END_MASK) !=
  3806. STREG0_END_NORMAL) {
  3807. DisketteExtension->HardwareFailed = TRUE;
  3808. status = FlInterpretError(DisketteExtension->FifoBuffer[1],
  3809. DisketteExtension->FifoBuffer[2]);
  3810. } else {
  3811. //
  3812. // The floppy controller may return no errors but not have
  3813. // read all of the requested data. If this is the case,
  3814. // record it as an error and retru the operation.
  3815. //
  3816. if (DisketteExtension->FifoBuffer[5] != 1) {
  3817. DisketteExtension->HardwareFailed = TRUE;
  3818. status = STATUS_FLOPPY_UNKNOWN_ERROR;
  3819. }
  3820. }
  3821. } else {
  3822. FloppyDump( FLOPINFO,
  3823. ("FlReadWriteTrack: %s command failed %x\n",
  3824. WriteOperation ? "write" : "read",
  3825. status) );
  3826. }
  3827. if (NT_SUCCESS(status)) {
  3828. break;
  3829. }
  3830. if (ioRetry >= 2) {
  3831. FloppyDump(FLOPINFO,
  3832. ("FlReadWriteTrack: too many retries - failing\n"));
  3833. break;
  3834. }
  3835. }
  3836. if (NT_SUCCESS(status)) {
  3837. break;
  3838. }
  3839. // We failed quite a bit so make seeks mandatory.
  3840. recalibrateDrive = TRUE;
  3841. }
  3842. if (!NT_SUCCESS(status) && NumberOfSectors > 1) {
  3843. // Retry one sector at a time.
  3844. FloppyDump( FLOPINFO,
  3845. ("FlReadWriteTrack: Attempting sector at a time\n") );
  3846. for (i = 0; i < NumberOfSectors; i++) {
  3847. status = FlReadWriteTrack( DisketteExtension,
  3848. IoMdl,
  3849. IoOffset+(((ULONG)i)<<byteToSectorShift),
  3850. WriteOperation,
  3851. Cylinder,
  3852. Head,
  3853. (UCHAR) (Sector + i),
  3854. 1,
  3855. FALSE );
  3856. if (!NT_SUCCESS(status)) {
  3857. FloppyDump( FLOPINFO,
  3858. ("FlReadWriteTrack: failed sector %d status %x\n",
  3859. i,
  3860. status) );
  3861. DisketteExtension->HardwareFailed = TRUE;
  3862. break;
  3863. }
  3864. }
  3865. }
  3866. return status;
  3867. }
  3868. NTSTATUS
  3869. FlReadWrite(
  3870. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  3871. IN OUT PIRP Irp,
  3872. IN BOOLEAN DriveStarted
  3873. )
  3874. /*++
  3875. Routine Description:
  3876. This routine is called by the floppy thread to read/write data
  3877. to/from the diskette. It breaks the request into pieces called
  3878. "transfers" (their size depends on the buffer size, where the end of
  3879. the track is, etc) and retries each transfer until it succeeds or
  3880. the retry count is exceeded.
  3881. Arguments:
  3882. DisketteExtension - a pointer to our data area for the drive to be
  3883. accessed.
  3884. Irp - a pointer to the IO Request Packet.
  3885. DriveStarted - indicated whether or not the drive has been started.
  3886. Return Value:
  3887. STATUS_SUCCESS if the packet was successfully read or written; the
  3888. appropriate error is propogated otherwise.
  3889. --*/
  3890. {
  3891. PIO_STACK_LOCATION irpSp;
  3892. BOOLEAN writeOperation;
  3893. NTSTATUS status;
  3894. PDRIVE_MEDIA_CONSTANTS driveMediaConstants;
  3895. ULONG byteToSectorShift;
  3896. ULONG currentSector, firstSector, lastSector;
  3897. ULONG trackSize;
  3898. UCHAR sectorsPerTrack, numberOfHeads;
  3899. UCHAR currentHead, currentCylinder, trackSector;
  3900. PCHAR userBuffer;
  3901. UCHAR skew, skewDelta;
  3902. UCHAR numTransferSectors;
  3903. PMDL mdl;
  3904. PCHAR ioBuffer;
  3905. ULONG ioOffset;
  3906. irpSp = IoGetCurrentIrpStackLocation(Irp);
  3907. FloppyDump(
  3908. FLOPSHOW,
  3909. ("FlReadWrite: for %s at offset %x size %x ",
  3910. irpSp->MajorFunction == IRP_MJ_WRITE ? "write" : "read",
  3911. irpSp->Parameters.Read.ByteOffset.LowPart,
  3912. irpSp->Parameters.Read.Length)
  3913. );
  3914. // Check for valid operation on this device.
  3915. if (irpSp->MajorFunction == IRP_MJ_WRITE) {
  3916. if (DisketteExtension->IsReadOnly) {
  3917. FloppyDump( FLOPSHOW, ("is read-only\n") );
  3918. return STATUS_INVALID_PARAMETER;
  3919. }
  3920. writeOperation = TRUE;
  3921. } else {
  3922. writeOperation = FALSE;
  3923. }
  3924. FloppyDump( FLOPSHOW, ("\n") );
  3925. // Start up the drive.
  3926. if (DriveStarted) {
  3927. status = STATUS_SUCCESS;
  3928. } else {
  3929. status = FlStartDrive( DisketteExtension,
  3930. Irp,
  3931. writeOperation,
  3932. TRUE,
  3933. (BOOLEAN)
  3934. !!(irpSp->Flags&SL_OVERRIDE_VERIFY_VOLUME));
  3935. }
  3936. if (!NT_SUCCESS(status)) {
  3937. FloppyDump(
  3938. FLOPSHOW,
  3939. ("FlReadWrite: error on start %x\n", status)
  3940. );
  3941. return status;
  3942. }
  3943. if (IsNEC_98) {
  3944. FlHdbit(DisketteExtension);
  3945. } // (IsNEC_98)
  3946. if (DisketteExtension->MediaType == Unknown) {
  3947. FloppyDump( FLOPSHOW, ("not recognized\n") );
  3948. return STATUS_UNRECOGNIZED_MEDIA;
  3949. }
  3950. // The drive has started up with a recognized media.
  3951. // Gather some relavant parameters.
  3952. driveMediaConstants = &DisketteExtension->DriveMediaConstants;
  3953. byteToSectorShift = SECTORLENGTHCODE_TO_BYTESHIFT +
  3954. driveMediaConstants->SectorLengthCode;
  3955. firstSector = irpSp->Parameters.Read.ByteOffset.LowPart>>
  3956. byteToSectorShift;
  3957. lastSector = firstSector + (irpSp->Parameters.Read.Length>>
  3958. byteToSectorShift);
  3959. sectorsPerTrack = driveMediaConstants->SectorsPerTrack;
  3960. numberOfHeads = driveMediaConstants->NumberOfHeads;
  3961. userBuffer = MmGetSystemAddressForMdlSafe(Irp->MdlAddress,
  3962. HighPagePriority);
  3963. if (userBuffer == NULL) {
  3964. FloppyDump(FLOPWARN,
  3965. ("MmGetSystemAddressForMdlSafe returned NULL in FlReadWrite\n")
  3966. );
  3967. return STATUS_INSUFFICIENT_RESOURCES;
  3968. }
  3969. trackSize = ((ULONG) sectorsPerTrack)<<byteToSectorShift;
  3970. skew = 0;
  3971. skewDelta = driveMediaConstants->SkewDelta;
  3972. for (currentSector = firstSector;
  3973. currentSector < lastSector;
  3974. currentSector += numTransferSectors) {
  3975. // Compute cylinder, head and sector from absolute sector.
  3976. currentCylinder = (UCHAR) (currentSector/sectorsPerTrack/numberOfHeads);
  3977. trackSector = (UCHAR) (currentSector%sectorsPerTrack);
  3978. currentHead = (UCHAR) (currentSector/sectorsPerTrack%numberOfHeads);
  3979. numTransferSectors = sectorsPerTrack - trackSector;
  3980. if (lastSector - currentSector < numTransferSectors) {
  3981. numTransferSectors = (UCHAR) (lastSector - currentSector);
  3982. }
  3983. //
  3984. // If we're using a temporary IO buffer because of
  3985. // insufficient registers in the DMA and we're
  3986. // doing a write then copy the write buffer to
  3987. // the contiguous buffer.
  3988. //
  3989. if (trackSize > DisketteExtension->MaxTransferSize) {
  3990. FloppyDump(FLOPSHOW,
  3991. ("FlReadWrite allocating an IoBuffer\n")
  3992. );
  3993. FlAllocateIoBuffer(DisketteExtension, trackSize);
  3994. if (!DisketteExtension->IoBuffer) {
  3995. FloppyDump(
  3996. FLOPSHOW,
  3997. ("FlReadWrite: no resources\n")
  3998. );
  3999. return STATUS_INSUFFICIENT_RESOURCES;
  4000. }
  4001. mdl = DisketteExtension->IoBufferMdl;
  4002. ioBuffer = DisketteExtension->IoBuffer;
  4003. ioOffset = 0;
  4004. if (writeOperation) {
  4005. RtlMoveMemory(ioBuffer,
  4006. userBuffer + ((currentSector - firstSector)<<
  4007. byteToSectorShift),
  4008. ((ULONG) numTransferSectors)<<byteToSectorShift);
  4009. }
  4010. } else {
  4011. mdl = Irp->MdlAddress;
  4012. ioOffset = (currentSector - firstSector) << byteToSectorShift;
  4013. }
  4014. //
  4015. // Transfer the track.
  4016. // Do what we can to avoid missing revs.
  4017. //
  4018. // Alter the skew to be in the range of what
  4019. // we're transfering.
  4020. if (skew >= numTransferSectors + trackSector) {
  4021. skew = 0;
  4022. }
  4023. if (skew < trackSector) {
  4024. skew = trackSector;
  4025. }
  4026. // Go from skew to the end of the irp.
  4027. status = FlReadWriteTrack(
  4028. DisketteExtension,
  4029. mdl,
  4030. ioOffset + (((ULONG) skew - trackSector)<<byteToSectorShift),
  4031. writeOperation,
  4032. currentCylinder,
  4033. currentHead,
  4034. skew,
  4035. (UCHAR) (numTransferSectors + trackSector - skew),
  4036. TRUE);
  4037. // Go from start of irp to skew.
  4038. if (NT_SUCCESS(status) && skew > trackSector) {
  4039. status = FlReadWriteTrack( DisketteExtension,
  4040. mdl,
  4041. ioOffset,
  4042. writeOperation,
  4043. currentCylinder,
  4044. currentHead,
  4045. trackSector,
  4046. (UCHAR) (skew - trackSector),
  4047. FALSE);
  4048. } else {
  4049. skew = (numTransferSectors + trackSector)%sectorsPerTrack;
  4050. }
  4051. if (!NT_SUCCESS(status)) {
  4052. break;
  4053. }
  4054. //
  4055. // If we used the temporary IO buffer to do the
  4056. // read then copy the contents back to the IRPs buffer.
  4057. //
  4058. if (!writeOperation &&
  4059. trackSize > DisketteExtension->MaxTransferSize) {
  4060. RtlMoveMemory( userBuffer + ((currentSector - firstSector) <<
  4061. byteToSectorShift),
  4062. ioBuffer,
  4063. ((ULONG) numTransferSectors)<<byteToSectorShift);
  4064. }
  4065. //
  4066. // Increment the skew. Do this even if just switching sides
  4067. // for National Super I/O chips.
  4068. //
  4069. skew = (skew + skewDelta)%sectorsPerTrack;
  4070. }
  4071. Irp->IoStatus.Information =
  4072. (currentSector - firstSector) << byteToSectorShift;
  4073. // If the read was successful then consolidate the
  4074. // boot sector with the determined density.
  4075. if (NT_SUCCESS(status) && firstSector == 0) {
  4076. FlConsolidateMediaTypeWithBootSector(DisketteExtension,
  4077. (PBOOT_SECTOR_INFO) userBuffer);
  4078. }
  4079. FloppyDump( FLOPSHOW,
  4080. ("FlReadWrite: completed status %x information %d\n",
  4081. status, Irp->IoStatus.Information)
  4082. );
  4083. return status;
  4084. }
  4085. NTSTATUS
  4086. FlFormat(
  4087. IN PDISKETTE_EXTENSION DisketteExtension,
  4088. IN PIRP Irp
  4089. )
  4090. /*++
  4091. Routine Description:
  4092. This routine is called by the floppy thread to format some tracks on
  4093. the diskette. This won't take TOO long because the FORMAT utility
  4094. is written to only format a few tracks at a time so that it can keep
  4095. a display of what percentage of the disk has been formatted.
  4096. Arguments:
  4097. DisketteExtension - pointer to our data area for the diskette to be
  4098. formatted.
  4099. Irp - pointer to the IO Request Packet.
  4100. Return Value:
  4101. STATUS_SUCCESS if the tracks were formatted; appropriate error
  4102. propogated otherwise.
  4103. --*/
  4104. {
  4105. LARGE_INTEGER headSettleTime;
  4106. PIO_STACK_LOCATION irpSp;
  4107. PBAD_TRACK_NUMBER badTrackBuffer;
  4108. PFORMAT_PARAMETERS formatParameters;
  4109. PFORMAT_EX_PARAMETERS formatExParameters;
  4110. PDRIVE_MEDIA_CONSTANTS driveMediaConstants;
  4111. NTSTATUS ntStatus;
  4112. ULONG badTrackBufferLength;
  4113. DRIVE_MEDIA_TYPE driveMediaType;
  4114. UCHAR driveStatus;
  4115. UCHAR numberOfBadTracks = 0;
  4116. UCHAR currentTrack;
  4117. UCHAR endTrack;
  4118. UCHAR whichSector;
  4119. UCHAR retryCount;
  4120. BOOLEAN bufferOverflow = FALSE;
  4121. FDC_DISK_CHANGE_PARMS fdcDiskChangeParms;
  4122. FloppyDump(
  4123. FLOPSHOW,
  4124. ("Floppy: FlFormat...\n")
  4125. );
  4126. irpSp = IoGetCurrentIrpStackLocation( Irp );
  4127. formatParameters = (PFORMAT_PARAMETERS) Irp->AssociatedIrp.SystemBuffer;
  4128. if (irpSp->Parameters.DeviceIoControl.IoControlCode ==
  4129. IOCTL_DISK_FORMAT_TRACKS_EX) {
  4130. formatExParameters =
  4131. (PFORMAT_EX_PARAMETERS) Irp->AssociatedIrp.SystemBuffer;
  4132. } else {
  4133. formatExParameters = NULL;
  4134. }
  4135. FloppyDump(
  4136. FLOPFORMAT,
  4137. ("Floppy: Format Params - MediaType: %d\n"
  4138. "------ Start Cyl: %x\n"
  4139. "------ End Cyl: %x\n"
  4140. "------ Start Hd: %d\n"
  4141. "------ End Hd: %d\n",
  4142. formatParameters->MediaType,
  4143. formatParameters->StartCylinderNumber,
  4144. formatParameters->EndCylinderNumber,
  4145. formatParameters->StartHeadNumber,
  4146. formatParameters->EndHeadNumber)
  4147. );
  4148. badTrackBufferLength =
  4149. irpSp->Parameters.DeviceIoControl.OutputBufferLength;
  4150. //
  4151. // Figure out which entry in the DriveMediaConstants table to use.
  4152. // We know we'll find one, or FlCheckFormatParameters() would have
  4153. // rejected the request.
  4154. //
  4155. driveMediaType =
  4156. DriveMediaLimits[DisketteExtension->DriveType].HighestDriveMediaType;
  4157. while ( ( DriveMediaConstants[driveMediaType].MediaType !=
  4158. formatParameters->MediaType ) &&
  4159. ( driveMediaType > DriveMediaLimits[DisketteExtension->DriveType].
  4160. LowestDriveMediaType ) ) {
  4161. driveMediaType--;
  4162. }
  4163. driveMediaConstants = &DriveMediaConstants[driveMediaType];
  4164. //
  4165. // Set some values in the diskette extension to indicate what we
  4166. // know about the media type.
  4167. //
  4168. DisketteExtension->MediaType = formatParameters->MediaType;
  4169. DisketteExtension->DriveMediaType = driveMediaType;
  4170. DisketteExtension->DriveMediaConstants =
  4171. DriveMediaConstants[driveMediaType];
  4172. if (formatExParameters) {
  4173. DisketteExtension->DriveMediaConstants.SectorsPerTrack =
  4174. (UCHAR) formatExParameters->SectorsPerTrack;
  4175. DisketteExtension->DriveMediaConstants.FormatGapLength =
  4176. (UCHAR) formatExParameters->FormatGapLength;
  4177. }
  4178. driveMediaConstants = &(DisketteExtension->DriveMediaConstants);
  4179. DisketteExtension->BytesPerSector = driveMediaConstants->BytesPerSector;
  4180. DisketteExtension->ByteCapacity =
  4181. ( driveMediaConstants->BytesPerSector ) *
  4182. driveMediaConstants->SectorsPerTrack *
  4183. ( 1 + driveMediaConstants->MaximumTrack ) *
  4184. driveMediaConstants->NumberOfHeads;
  4185. currentTrack = (UCHAR)( ( formatParameters->StartCylinderNumber *
  4186. driveMediaConstants->NumberOfHeads ) +
  4187. formatParameters->StartHeadNumber );
  4188. endTrack = (UCHAR)( ( formatParameters->EndCylinderNumber *
  4189. driveMediaConstants->NumberOfHeads ) +
  4190. formatParameters->EndHeadNumber );
  4191. FloppyDump(
  4192. FLOPFORMAT,
  4193. ("Floppy: Format - Starting/ending tracks: %x/%x\n",
  4194. currentTrack,
  4195. endTrack)
  4196. );
  4197. //
  4198. // Set the data rate (which depends on the drive/media
  4199. // type).
  4200. //
  4201. if (IsNEC_98) {
  4202. FlHdbit(DisketteExtension);
  4203. } // (IsNEC_98)
  4204. if ( DisketteExtension->LastDriveMediaType != driveMediaType ) {
  4205. ntStatus = FlDatarateSpecifyConfigure( DisketteExtension );
  4206. if ( !NT_SUCCESS( ntStatus ) ) {
  4207. return ntStatus;
  4208. }
  4209. }
  4210. //
  4211. // Since we're doing a format, make this drive writable.
  4212. //
  4213. DisketteExtension->IsReadOnly = FALSE;
  4214. //
  4215. // Format each track.
  4216. //
  4217. do {
  4218. //
  4219. // Seek to proper cylinder
  4220. //
  4221. DisketteExtension->FifoBuffer[0] = COMMND_SEEK;
  4222. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  4223. DisketteExtension->FifoBuffer[2] = (UCHAR)( ( currentTrack /
  4224. driveMediaConstants->NumberOfHeads ) <<
  4225. driveMediaConstants->CylinderShift );
  4226. FloppyDump(
  4227. FLOPFORMAT,
  4228. ("Floppy: Format seek to cylinder: %x\n",
  4229. DisketteExtension->FifoBuffer[1])
  4230. );
  4231. ntStatus = FlIssueCommand( DisketteExtension,
  4232. DisketteExtension->FifoBuffer,
  4233. DisketteExtension->FifoBuffer,
  4234. NULL,
  4235. 0,
  4236. 0 );
  4237. if ( NT_SUCCESS( ntStatus ) ) {
  4238. if ( ( DisketteExtension->FifoBuffer[0] & STREG0_SEEK_COMPLETE ) &&
  4239. ( DisketteExtension->FifoBuffer[1] == (UCHAR)( ( currentTrack /
  4240. driveMediaConstants->NumberOfHeads ) <<
  4241. driveMediaConstants->CylinderShift ) ) ) {
  4242. //
  4243. // Must delay HeadSettleTime milliseconds before
  4244. // doing anything after a SEEK.
  4245. //
  4246. headSettleTime.LowPart = - ( 10 * 1000 *
  4247. driveMediaConstants->HeadSettleTime );
  4248. headSettleTime.HighPart = -1;
  4249. KeDelayExecutionThread(
  4250. KernelMode,
  4251. FALSE,
  4252. &headSettleTime );
  4253. if (IsNEC_98) {
  4254. //
  4255. // We don't need READ ID at format.
  4256. //
  4257. } else { // (IsNEC_98)
  4258. //
  4259. // Read ID. Note that we don't bother checking the return
  4260. // registers, because if this media wasn't formatted we'd
  4261. // get an error.
  4262. //
  4263. DisketteExtension->FifoBuffer[0] =
  4264. COMMND_READ_ID + COMMND_OPTION_MFM;
  4265. DisketteExtension->FifoBuffer[1] =
  4266. DisketteExtension->DeviceUnit;
  4267. ntStatus = FlIssueCommand( DisketteExtension,
  4268. DisketteExtension->FifoBuffer,
  4269. DisketteExtension->FifoBuffer,
  4270. NULL,
  4271. 0,
  4272. 0 );
  4273. } // (IsNEC_98)
  4274. } else {
  4275. FloppyDump(
  4276. FLOPWARN,
  4277. ("Floppy: format's seek returned bad registers\n"
  4278. "------ Statusreg0 = %x\n"
  4279. "------ Statusreg1 = %x\n",
  4280. DisketteExtension->FifoBuffer[0],
  4281. DisketteExtension->FifoBuffer[1])
  4282. );
  4283. DisketteExtension->HardwareFailed = TRUE;
  4284. ntStatus = STATUS_FLOPPY_BAD_REGISTERS;
  4285. }
  4286. }
  4287. if ( !NT_SUCCESS( ntStatus ) ) {
  4288. FloppyDump(
  4289. FLOPWARN,
  4290. ("Floppy: format's seek/readid returned %x\n", ntStatus)
  4291. );
  4292. return ntStatus;
  4293. }
  4294. //
  4295. // Fill the buffer with the format of this track.
  4296. //
  4297. for (whichSector = 0;
  4298. whichSector < driveMediaConstants->SectorsPerTrack;
  4299. whichSector++) {
  4300. DisketteExtension->IoBuffer[whichSector*4] =
  4301. currentTrack/driveMediaConstants->NumberOfHeads;
  4302. DisketteExtension->IoBuffer[whichSector*4 + 1] =
  4303. currentTrack%driveMediaConstants->NumberOfHeads;
  4304. if (formatExParameters) {
  4305. DisketteExtension->IoBuffer[whichSector*4 + 2] =
  4306. (UCHAR) formatExParameters->SectorNumber[whichSector];
  4307. } else {
  4308. DisketteExtension->IoBuffer[whichSector*4 + 2] =
  4309. whichSector + 1;
  4310. }
  4311. DisketteExtension->IoBuffer[whichSector*4 + 3] =
  4312. driveMediaConstants->SectorLengthCode;
  4313. FloppyDump(
  4314. FLOPFORMAT,
  4315. ("Floppy - Format table entry %x - %x/%x/%x/%x\n",
  4316. whichSector,
  4317. DisketteExtension->IoBuffer[whichSector*4],
  4318. DisketteExtension->IoBuffer[whichSector*4 + 1],
  4319. DisketteExtension->IoBuffer[whichSector*4 + 2],
  4320. DisketteExtension->IoBuffer[whichSector*4 + 3])
  4321. );
  4322. }
  4323. //
  4324. // Retry until success or too many retries.
  4325. //
  4326. retryCount = 0;
  4327. do {
  4328. ULONG length;
  4329. length = driveMediaConstants->BytesPerSector;
  4330. //
  4331. // Issue command to format track
  4332. //
  4333. DisketteExtension->FifoBuffer[0] =
  4334. COMMND_FORMAT_TRACK + COMMND_OPTION_MFM;
  4335. DisketteExtension->FifoBuffer[1] = (UCHAR)
  4336. ( ( ( currentTrack % driveMediaConstants->NumberOfHeads ) << 2 )
  4337. | DisketteExtension->DeviceUnit );
  4338. DisketteExtension->FifoBuffer[2] =
  4339. driveMediaConstants->SectorLengthCode;
  4340. DisketteExtension->FifoBuffer[3] =
  4341. driveMediaConstants->SectorsPerTrack;
  4342. DisketteExtension->FifoBuffer[4] =
  4343. driveMediaConstants->FormatGapLength;
  4344. DisketteExtension->FifoBuffer[5] =
  4345. driveMediaConstants->FormatFillCharacter;
  4346. FloppyDump(
  4347. FLOPFORMAT,
  4348. ("Floppy: format command parameters\n"
  4349. "------ Head/Unit: %x\n"
  4350. "------ Bytes/Sector: %x\n"
  4351. "------ Sectors/Cylinder: %x\n"
  4352. "------ Gap 3: %x\n"
  4353. "------ Filler Byte: %x\n",
  4354. DisketteExtension->FifoBuffer[1],
  4355. DisketteExtension->FifoBuffer[2],
  4356. DisketteExtension->FifoBuffer[3],
  4357. DisketteExtension->FifoBuffer[4],
  4358. DisketteExtension->FifoBuffer[5])
  4359. );
  4360. ntStatus = FlIssueCommand( DisketteExtension,
  4361. DisketteExtension->FifoBuffer,
  4362. DisketteExtension->FifoBuffer,
  4363. DisketteExtension->IoBufferMdl,
  4364. 0,
  4365. length );
  4366. if ( !NT_SUCCESS( ntStatus ) ) {
  4367. FloppyDump(
  4368. FLOPDBGP,
  4369. ("Floppy: format returned %x\n", ntStatus)
  4370. );
  4371. }
  4372. if ( NT_SUCCESS( ntStatus ) ) {
  4373. //
  4374. // Check the return bytes from the controller.
  4375. //
  4376. if ( ( DisketteExtension->FifoBuffer[0] &
  4377. ( STREG0_DRIVE_FAULT |
  4378. STREG0_END_INVALID_COMMAND |
  4379. STREG0_END_ERROR
  4380. ) )
  4381. || ( DisketteExtension->FifoBuffer[1] &
  4382. STREG1_DATA_OVERRUN ) ||
  4383. ( DisketteExtension->FifoBuffer[2] != 0 ) ) {
  4384. FloppyDump(
  4385. FLOPWARN,
  4386. ("Floppy: format had bad registers\n"
  4387. "------ Streg0 = %x\n"
  4388. "------ Streg1 = %x\n"
  4389. "------ Streg2 = %x\n",
  4390. DisketteExtension->FifoBuffer[0],
  4391. DisketteExtension->FifoBuffer[1],
  4392. DisketteExtension->FifoBuffer[2])
  4393. );
  4394. DisketteExtension->HardwareFailed = TRUE;
  4395. ntStatus = FlInterpretError(
  4396. DisketteExtension->FifoBuffer[1],
  4397. DisketteExtension->FifoBuffer[2] );
  4398. }
  4399. }
  4400. } while ( ( !NT_SUCCESS( ntStatus ) ) &&
  4401. ( retryCount++ < RECALIBRATE_RETRY_COUNT ) );
  4402. if ( !NT_SUCCESS( ntStatus ) ) {
  4403. if (IsNEC_98) {
  4404. DisketteExtension->FifoBuffer[0] = COMMND_SENSE_DRIVE_STATUS;
  4405. DisketteExtension->FifoBuffer[1] = DisketteExtension->DeviceUnit;
  4406. ntStatus = FlIssueCommand( DisketteExtension,
  4407. DisketteExtension->FifoBuffer,
  4408. DisketteExtension->FifoBuffer,
  4409. NULL,
  4410. 0,
  4411. 0 );
  4412. if ( !NT_SUCCESS( ntStatus ) ) {
  4413. FloppyDump(
  4414. FLOPWARN,
  4415. ("Floppy: SENSE_DRIVE returned %x\n", ntStatus)
  4416. );
  4417. return ntStatus;
  4418. }
  4419. if ( DisketteExtension->FifoBuffer[0] & STREG3_DRIVE_READY ) {
  4420. driveStatus = DSKCHG_RESERVED;
  4421. } else {
  4422. driveStatus = DSKCHG_DISKETTE_REMOVED;
  4423. }
  4424. } else { // (IsNEC_98)
  4425. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  4426. IOCTL_DISK_INTERNAL_GET_FDC_DISK_CHANGE,
  4427. &fdcDiskChangeParms );
  4428. driveStatus = fdcDiskChangeParms.DriveStatus;
  4429. } // (IsNEC_98)
  4430. if ( (DisketteExtension->DriveType != DRIVE_TYPE_0360) &&
  4431. driveStatus & DSKCHG_DISKETTE_REMOVED ) {
  4432. //
  4433. // The user apparently popped the floppy. Return error
  4434. // rather than logging bad track.
  4435. //
  4436. return STATUS_NO_MEDIA_IN_DEVICE;
  4437. }
  4438. //
  4439. // Log the bad track.
  4440. //
  4441. FloppyDump(
  4442. FLOPDBGP,
  4443. ("Floppy: track %x is bad\n", currentTrack)
  4444. );
  4445. if ( badTrackBufferLength >= (ULONG)
  4446. ( ( numberOfBadTracks + 1 ) * sizeof( BAD_TRACK_NUMBER ) ) ) {
  4447. badTrackBuffer = (PBAD_TRACK_NUMBER)
  4448. Irp->AssociatedIrp.SystemBuffer;
  4449. badTrackBuffer[numberOfBadTracks] = ( BAD_TRACK_NUMBER )
  4450. currentTrack;
  4451. } else {
  4452. bufferOverflow = TRUE;
  4453. }
  4454. numberOfBadTracks++;
  4455. }
  4456. currentTrack++;
  4457. } while ( currentTrack <= endTrack );
  4458. if ( ( NT_SUCCESS( ntStatus ) ) && ( bufferOverflow ) ) {
  4459. ntStatus = STATUS_BUFFER_OVERFLOW;
  4460. }
  4461. return ntStatus;
  4462. }
  4463. BOOLEAN
  4464. FlCheckFormatParameters(
  4465. IN PDISKETTE_EXTENSION DisketteExtension,
  4466. IN PFORMAT_PARAMETERS FormatParameters
  4467. )
  4468. /*++
  4469. Routine Description:
  4470. This routine checks the supplied format parameters to make sure that
  4471. they'll work on the drive to be formatted.
  4472. Arguments:
  4473. DisketteExtension - a pointer to our data area for the diskette to
  4474. be formatted.
  4475. FormatParameters - a pointer to the caller's parameters for the FORMAT.
  4476. Return Value:
  4477. TRUE if parameters are OK.
  4478. FALSE if the parameters are bad.
  4479. --*/
  4480. {
  4481. PDRIVE_MEDIA_CONSTANTS driveMediaConstants;
  4482. DRIVE_MEDIA_TYPE driveMediaType;
  4483. //
  4484. // Figure out which entry in the DriveMediaConstants table to use.
  4485. //
  4486. driveMediaType =
  4487. DriveMediaLimits[DisketteExtension->DriveType].HighestDriveMediaType;
  4488. while ( ( DriveMediaConstants[driveMediaType].MediaType !=
  4489. FormatParameters->MediaType ) &&
  4490. ( driveMediaType > DriveMediaLimits[DisketteExtension->DriveType].
  4491. LowestDriveMediaType ) ) {
  4492. driveMediaType--;
  4493. }
  4494. if ( DriveMediaConstants[driveMediaType].MediaType !=
  4495. FormatParameters->MediaType ) {
  4496. return FALSE;
  4497. } else {
  4498. driveMediaConstants = &DriveMediaConstants[driveMediaType];
  4499. if ( ( FormatParameters->StartHeadNumber >
  4500. (ULONG)( driveMediaConstants->NumberOfHeads - 1 ) ) ||
  4501. ( FormatParameters->EndHeadNumber >
  4502. (ULONG)( driveMediaConstants->NumberOfHeads - 1 ) ) ||
  4503. ( FormatParameters->StartCylinderNumber >
  4504. driveMediaConstants->MaximumTrack ) ||
  4505. ( FormatParameters->EndCylinderNumber >
  4506. driveMediaConstants->MaximumTrack ) ||
  4507. ( FormatParameters->EndCylinderNumber <
  4508. FormatParameters->StartCylinderNumber ) ) {
  4509. return FALSE;
  4510. } else {
  4511. if (IsNEC_98) {
  4512. if((FormatParameters->MediaType == F5_360_512)||
  4513. (FormatParameters->MediaType == F5_320_512)||
  4514. (FormatParameters->MediaType == F5_320_1024)||
  4515. (FormatParameters->MediaType == F5_180_512)||
  4516. (FormatParameters->MediaType == F5_160_512)){
  4517. return FALSE;
  4518. }
  4519. } // (IsNEC_98)
  4520. return TRUE;
  4521. }
  4522. }
  4523. }
  4524. NTSTATUS
  4525. FlIssueCommand(
  4526. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  4527. IN PUCHAR FifoInBuffer,
  4528. OUT PUCHAR FifoOutBuffer,
  4529. IN PMDL IoMdl,
  4530. IN OUT ULONG IoOffset,
  4531. IN ULONG TransferBytes
  4532. )
  4533. /*++
  4534. Routine Description:
  4535. This routine sends the command and all parameters to the controller,
  4536. waits for the command to interrupt if necessary, and reads the result
  4537. bytes from the controller, if any.
  4538. Before calling this routine, the caller should put the parameters for
  4539. the command in ControllerData->FifoBuffer[]. The result bytes will
  4540. be returned in the same place.
  4541. This routine runs off the CommandTable. For each command, this says
  4542. how many parameters there are, whether or not there is an interrupt
  4543. to wait for, and how many result bytes there are. Note that commands
  4544. without result bytes actually have two, since the ISR will issue a
  4545. SENSE INTERRUPT STATUS command on their behalf.
  4546. Arguments:
  4547. Command - a byte specifying the command to be sent to the controller.
  4548. FloppyExtension - a pointer to our data area for the drive being
  4549. accessed (any drive if a controller command is being given).
  4550. Return Value:
  4551. STATUS_SUCCESS if the command was sent and bytes received properly;
  4552. appropriate error propogated otherwise.
  4553. --*/
  4554. {
  4555. NTSTATUS ntStatus;
  4556. UCHAR i;
  4557. PIRP irp;
  4558. KEVENT DoneEvent;
  4559. IO_STATUS_BLOCK IoStatus;
  4560. PIO_STACK_LOCATION irpSp;
  4561. ISSUE_FDC_COMMAND_PARMS issueCommandParms;
  4562. //
  4563. // Set the command parameters
  4564. //
  4565. issueCommandParms.FifoInBuffer = FifoInBuffer;
  4566. issueCommandParms.FifoOutBuffer = FifoOutBuffer;
  4567. issueCommandParms.IoHandle = (PVOID)IoMdl;
  4568. issueCommandParms.IoOffset = IoOffset;
  4569. issueCommandParms.TransferBytes = TransferBytes;
  4570. issueCommandParms.TimeOut = FDC_TIMEOUT;
  4571. FloppyDump( FLOPSHOW,
  4572. ("Floppy: FloppyIssueCommand %2x...\n",
  4573. DisketteExtension->FifoBuffer[0])
  4574. );
  4575. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  4576. IOCTL_DISK_INTERNAL_ISSUE_FDC_COMMAND,
  4577. &issueCommandParms );
  4578. //
  4579. // If it appears like the floppy controller is not responding
  4580. // set the HardwareFailed flag which will force a reset.
  4581. //
  4582. if ( ntStatus == STATUS_DEVICE_NOT_READY ||
  4583. ntStatus == STATUS_FLOPPY_BAD_REGISTERS ) {
  4584. DisketteExtension->HardwareFailed = TRUE;
  4585. }
  4586. return ntStatus;
  4587. }
  4588. NTSTATUS
  4589. FlInitializeControllerHardware(
  4590. IN OUT PDISKETTE_EXTENSION DisketteExtension
  4591. )
  4592. /*++
  4593. Routine Description:
  4594. This routine is called to reset and initialize the floppy controller device.
  4595. Arguments:
  4596. disketteExtension - Supplies the diskette extension.
  4597. Return Value:
  4598. --*/
  4599. {
  4600. NTSTATUS ntStatus;
  4601. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  4602. IOCTL_DISK_INTERNAL_RESET_FDC,
  4603. NULL );
  4604. if (NT_SUCCESS(ntStatus)) {
  4605. if ( DisketteExtension->PerpendicularMode != 0 ) {
  4606. DisketteExtension->FifoBuffer[0] = COMMND_PERPENDICULAR_MODE;
  4607. DisketteExtension->FifoBuffer[1] =
  4608. (UCHAR) (COMMND_PERPENDICULAR_MODE_OW |
  4609. (DisketteExtension->PerpendicularMode << 2));
  4610. ntStatus = FlIssueCommand( DisketteExtension,
  4611. DisketteExtension->FifoBuffer,
  4612. DisketteExtension->FifoBuffer,
  4613. NULL,
  4614. 0,
  4615. 0 );
  4616. }
  4617. }
  4618. return ntStatus;
  4619. }
  4620. NTSTATUS
  4621. FlFdcDeviceIo(
  4622. IN PDEVICE_OBJECT DeviceObject,
  4623. IN ULONG Ioctl,
  4624. IN OUT PVOID Data
  4625. )
  4626. {
  4627. NTSTATUS ntStatus;
  4628. PIRP irp;
  4629. PIO_STACK_LOCATION irpStack;
  4630. KEVENT doneEvent;
  4631. IO_STATUS_BLOCK ioStatus;
  4632. FloppyDump(FLOPINFO,("Calling Fdc Device with %x\n", Ioctl));
  4633. KeInitializeEvent( &doneEvent,
  4634. NotificationEvent,
  4635. FALSE);
  4636. //
  4637. // Create an IRP for enabler
  4638. //
  4639. irp = IoBuildDeviceIoControlRequest( Ioctl,
  4640. DeviceObject,
  4641. NULL,
  4642. 0,
  4643. NULL,
  4644. 0,
  4645. TRUE,
  4646. &doneEvent,
  4647. &ioStatus );
  4648. if (irp == NULL) {
  4649. FloppyDump(FLOPDBGP,("FlFloppyDeviceIo: Can't allocate Irp\n"));
  4650. //
  4651. // If an Irp can't be allocated, then this call will
  4652. // simply return. This will leave the queue frozen for
  4653. // this device, which means it can no longer be accessed.
  4654. //
  4655. return STATUS_INSUFFICIENT_RESOURCES;
  4656. }
  4657. irpStack = IoGetNextIrpStackLocation(irp);
  4658. irpStack->Parameters.DeviceIoControl.Type3InputBuffer = Data;
  4659. //
  4660. // Call the driver and request the operation
  4661. //
  4662. ntStatus = IoCallDriver(DeviceObject, irp);
  4663. if ( ntStatus == STATUS_PENDING ) {
  4664. //
  4665. // Now wait for operation to complete (should already be done, but
  4666. // maybe not)
  4667. //
  4668. KeWaitForSingleObject( &doneEvent,
  4669. Executive,
  4670. KernelMode,
  4671. FALSE,
  4672. NULL );
  4673. ntStatus = ioStatus.Status;
  4674. }
  4675. return ntStatus;
  4676. }
  4677. NTSTATUS
  4678. FlHdbit(
  4679. IN OUT PDISKETTE_EXTENSION DisketteExtension
  4680. )
  4681. /*++
  4682. Routine Description:
  4683. Set a Hd bit or a FDD EXC bit.
  4684. Arguments:
  4685. DisketteExtension - a pointer to our data area for the device extension.
  4686. Return Value:
  4687. 0: Success
  4688. 1: Error
  4689. --*/
  4690. {
  4691. NTSTATUS ntStatus;
  4692. USHORT st; // State of HD bit
  4693. USHORT st2; // Set on/off HD bit
  4694. USHORT st3; // When set HD bit, then st3=1
  4695. USHORT st4; // 1.44MB bit for 1.44MB media
  4696. SHORT sel; // 1.44MB Selector No for 1.44MB media
  4697. SHORT st5=0; // 1.44MB on: wait for spin for 1.44MB media
  4698. LARGE_INTEGER motorOnDelay;
  4699. USHORT lpc;
  4700. UCHAR resultStatus0Save[4];
  4701. USHORT resultStatus0;
  4702. ULONG getStatusRetryCount;
  4703. ULONG rqmReadyRetryCount;
  4704. BOOLEAN media144MB;
  4705. BOOLEAN mediaUpTo120MB;
  4706. BOOLEAN supportDrive;
  4707. SET_HD_BIT_PARMS setHdBitParameter;
  4708. media144MB = FALSE;
  4709. mediaUpTo120MB = FALSE;
  4710. supportDrive = TRUE;
  4711. FloppyDump(
  4712. FLOPSTATUS,
  4713. ("Flpydisk : HdBit Media Type = %d \n", DisketteExtension->DriveMediaType)
  4714. );
  4715. switch(DisketteExtension->DriveMediaType){
  4716. //
  4717. // 1.44MB drive
  4718. //
  4719. case Drive144Media144Nec98: // 3.5" 1.44Mb drive; 1.44Mb media
  4720. media144MB = TRUE;
  4721. case Drive144Media120Nec98: // 3.5" 1.44Mb drive; 1.2Mb media
  4722. case Drive144Media123Nec98: // 3.5" 1.44Mb drive; 1.23Mb media
  4723. case Drive120Media120Nec98: // 5.25" 1.2Mb drive; 1.2Mb media
  4724. case Drive120Media123Nec98: // 5.25" 1.2Mb drive; 1.23Mb media
  4725. case Drive12EMedia120Nec98: // 5.25" 1.2Mb extension drive; 1.2Mb media
  4726. case Drive12EMedia123Nec98: // 5.25" 1.2Mb extension drive; 1.23Mb media
  4727. mediaUpTo120MB = TRUE;
  4728. case Drive360Media160Nec98: // 5.25" 360k drive; 160k media
  4729. case Drive360Media180Nec98: // 5.25" 360k drive; 180k media
  4730. case Drive360Media320Nec98: // 5.25" 360k drive; 320k media
  4731. case Drive360Media32XNec98: // 5.25" 360k drive; 320k 1k secs
  4732. case Drive360Media360Nec98: // 5.25" 360k drive; 360k media
  4733. case Drive120Media160Nec98: // 5.25" 720k drive; 160k media
  4734. case Drive120Media180Nec98: // 5.25" 720k drive; 180k media
  4735. case Drive120Media320Nec98: // 5.25" 720k drive; 320k media
  4736. case Drive120Media32XNec98: // 5.25" 720k drive; 320k 1k secs
  4737. case Drive120Media360Nec98: // 5.25" 720k drive; 360k media
  4738. case Drive120Media640Nec98: // 5.25" 720k drive; 640k media
  4739. case Drive120Media720Nec98: // 5.25" 720k drive; 720k media
  4740. case Drive144Media640Nec98: // 3.5" 1.44Mb drive; 640k media
  4741. case Drive144Media720Nec98: // 3.5" 1.44Mb drive; 720k media
  4742. break;
  4743. default:
  4744. //
  4745. // As 2HD
  4746. //
  4747. mediaUpTo120MB = TRUE;
  4748. break;
  4749. }
  4750. setHdBitParameter.Media144MB = media144MB;
  4751. setHdBitParameter.More120MB = mediaUpTo120MB;
  4752. setHdBitParameter.DeviceUnit = DisketteExtension->DeviceUnit;
  4753. setHdBitParameter.DriveType144MB = (DisketteExtension->DriveType == DRIVE_TYPE_1440) ? TRUE:FALSE;
  4754. ntStatus = FlFdcDeviceIo( DisketteExtension->TargetObject,
  4755. IOCTL_DISK_INTERNAL_SET_HD_BIT,
  4756. &setHdBitParameter );
  4757. if (!NT_SUCCESS(ntStatus)) {
  4758. return ntStatus;
  4759. }
  4760. if (setHdBitParameter.ChangedHdBit) {
  4761. ntStatus = FlDatarateSpecifyConfigure( DisketteExtension );
  4762. }
  4763. return ntStatus;
  4764. }
  4765. NTSTATUS
  4766. FloppyQueueRequest (
  4767. IN OUT PDISKETTE_EXTENSION DisketteExtension,
  4768. IN PIRP Irp
  4769. )
  4770. /*++
  4771. Routine Description:
  4772. Queues the Irp in the device queue. This routine will be called whenever
  4773. the device receives IRP_MN_QUERY_STOP_DEVICE or IRP_MN_QUERY_REMOVE_DEVICE
  4774. Arguments:
  4775. FdoData - pointer to the device's extension.
  4776. Irp - the request to be queued.
  4777. Return Value:
  4778. NT status code.
  4779. --*/
  4780. {
  4781. KIRQL oldIrql;
  4782. NTSTATUS ntStatus;
  4783. //
  4784. // Reset driver paging
  4785. //
  4786. FloppyResetDriverPaging();
  4787. //
  4788. // Check if we are allowed to queue requests.
  4789. //
  4790. ASSERT( DisketteExtension->HoldNewRequests );
  4791. //
  4792. // Preparing for dealing with cancelling stuff.
  4793. // We don't know how long the irp will be in the
  4794. // queue. So we need to handle cancel.
  4795. // Since we use out own queue, we don't need to use
  4796. // the cancel spin lock.
  4797. //
  4798. KeAcquireSpinLock(&DisketteExtension->FlCancelSpinLock,
  4799. &oldIrql);
  4800. IoSetCancelRoutine(Irp,
  4801. FloppyCancelQueuedRequest);
  4802. //
  4803. // Check if the irp was already canceled
  4804. //
  4805. if ( (Irp->Cancel ) &&
  4806. (IoSetCancelRoutine(Irp, NULL))) {
  4807. //
  4808. // Already canceled
  4809. //
  4810. Irp->IoStatus.Status = STATUS_CANCELLED;
  4811. Irp->IoStatus.Information = 0;
  4812. KeReleaseSpinLock(&DisketteExtension->FlCancelSpinLock,
  4813. oldIrql);
  4814. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  4815. FloppyPageEntireDriver();
  4816. ntStatus = STATUS_CANCELLED;
  4817. } else {
  4818. //
  4819. // Queue the Irp and set a cancel routine
  4820. //
  4821. Irp->IoStatus.Status = STATUS_PENDING;
  4822. IoMarkIrpPending(Irp);
  4823. ExInterlockedInsertTailList( &DisketteExtension->NewRequestQueue,
  4824. &Irp->Tail.Overlay.ListEntry,
  4825. &DisketteExtension->NewRequestQueueSpinLock);
  4826. KeReleaseSpinLock(&DisketteExtension->FlCancelSpinLock,
  4827. oldIrql);
  4828. ntStatus = STATUS_PENDING;
  4829. }
  4830. return ntStatus;
  4831. }
  4832. VOID
  4833. FloppyCancelQueuedRequest (
  4834. IN PDEVICE_OBJECT DeviceObject,
  4835. IN PIRP Irp
  4836. )
  4837. /*++
  4838. Routine Description:
  4839. The cancel routine. Will remove the IRP from the queue and will complete it.
  4840. The cancel spin lock is already acquired when this routine is called.
  4841. Arguments:
  4842. DeviceObject - pointer to the device object.
  4843. Irp - pointer to the IRP to be cancelled.
  4844. Return Value:
  4845. VOID.
  4846. --*/
  4847. {
  4848. PDISKETTE_EXTENSION disketteExtension = DeviceObject->DeviceExtension;
  4849. KIRQL oldIrql;
  4850. FloppyDump(FLOPDBGP,
  4851. ("Floppy Cancel called.\n"));
  4852. KeAcquireSpinLock(&disketteExtension->FlCancelSpinLock,
  4853. &oldIrql);
  4854. Irp->IoStatus.Status = STATUS_CANCELLED;
  4855. Irp->IoStatus.Information = 0;
  4856. //
  4857. // Make sure the IRP wasn't removed in Process routine.
  4858. //
  4859. if (Irp->Tail.Overlay.ListEntry.Flink) {
  4860. RemoveEntryList( &Irp->Tail.Overlay.ListEntry );
  4861. }
  4862. KeReleaseSpinLock(&disketteExtension->FlCancelSpinLock,
  4863. oldIrql);
  4864. IoReleaseCancelSpinLock( Irp->CancelIrql );
  4865. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  4866. FloppyPageEntireDriver();
  4867. return;
  4868. }
  4869. VOID
  4870. FloppyProcessQueuedRequests (
  4871. IN OUT PDISKETTE_EXTENSION DisketteExtension
  4872. )
  4873. /*++
  4874. Routine Description:
  4875. Removes an dprocesses the entries in the queue. If this routine is called
  4876. when processing IRP_MN_CANCEL_STOP_DEVICE, IRP_MN_CANCEL_REMOVE_DEVICE
  4877. or IRP_MN_START_DEVICE, the requests are passed to the next lower driver.
  4878. If the routine is called when IRP_MN_REMOVE_DEVICE is received, the IRPs
  4879. are completed with STATUS_DELETE_PENDING.
  4880. Arguments:
  4881. Return Value:
  4882. VOID.
  4883. --*/
  4884. {
  4885. KIRQL oldIrql;
  4886. PLIST_ENTRY headOfList;
  4887. PIRP currentIrp;
  4888. PIO_STACK_LOCATION irpSp;
  4889. //
  4890. // We need to dequeue all the entries in the queue, to reset the cancel
  4891. // routine for each of them and then to process then:
  4892. // - if the device is active, we will send them down
  4893. // - else we will complete them with STATUS_DELETE_PENDING
  4894. // (it is a surprise removal and we need to dispose the queue)
  4895. //
  4896. KeAcquireSpinLock(&DisketteExtension->FlCancelSpinLock,
  4897. &oldIrql);
  4898. while ((headOfList = ExInterlockedRemoveHeadList(
  4899. &DisketteExtension->NewRequestQueue,
  4900. &DisketteExtension->NewRequestQueueSpinLock)) != NULL) {
  4901. currentIrp = CONTAINING_RECORD( headOfList,
  4902. IRP,
  4903. Tail.Overlay.ListEntry);
  4904. if (IoSetCancelRoutine( currentIrp, NULL)) {
  4905. irpSp = IoGetCurrentIrpStackLocation( currentIrp );
  4906. } else {
  4907. //
  4908. // Cancel routine is already running for this IRP.
  4909. // Set the IRP field so that it won't be removed
  4910. // in the cancel routine again.
  4911. //
  4912. currentIrp->Tail.Overlay.ListEntry.Flink = NULL;
  4913. currentIrp = NULL;
  4914. }
  4915. KeReleaseSpinLock(&DisketteExtension->FlCancelSpinLock,
  4916. oldIrql);
  4917. if (currentIrp) {
  4918. if ( DisketteExtension->IsRemoved ) {
  4919. //
  4920. // The device was removed, we need to fail the request
  4921. //
  4922. currentIrp->IoStatus.Information = 0;
  4923. currentIrp->IoStatus.Status = STATUS_DELETE_PENDING;
  4924. IoCompleteRequest (currentIrp, IO_NO_INCREMENT);
  4925. } else {
  4926. switch ( irpSp->MajorFunction ) {
  4927. case IRP_MJ_READ:
  4928. case IRP_MJ_WRITE:
  4929. (VOID)FloppyReadWrite( DisketteExtension->DeviceObject, currentIrp );
  4930. break;
  4931. case IRP_MJ_DEVICE_CONTROL:
  4932. (VOID)FloppyDeviceControl( DisketteExtension->DeviceObject, currentIrp);
  4933. break;
  4934. default:
  4935. currentIrp->IoStatus.Information = 0;
  4936. currentIrp->IoStatus.Status = STATUS_UNSUCCESSFUL;
  4937. IoCompleteRequest (currentIrp, IO_NO_INCREMENT);
  4938. }
  4939. }
  4940. }
  4941. if (currentIrp) {
  4942. //
  4943. // Page out the driver if it's no more needed.
  4944. //
  4945. FloppyPageEntireDriver();
  4946. }
  4947. KeAcquireSpinLock(&DisketteExtension->FlCancelSpinLock,
  4948. &oldIrql);
  4949. }
  4950. KeReleaseSpinLock(&DisketteExtension->FlCancelSpinLock,
  4951. oldIrql);
  4952. return;
  4953. }