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.

3559 lines
92 KiB

  1. /*++
  2. Copyright (C) Microsoft Corporation, 1994 - 1999
  3. Module Name:
  4. scsitape.c
  5. Abstract:
  6. This is the tape class driver.
  7. Environment:
  8. kernel mode only
  9. Revision History:
  10. --*/
  11. #include "tape.h"
  12. //
  13. // Define the maximum inquiry data length.
  14. //
  15. #define MAXIMUM_TAPE_INQUIRY_DATA 252
  16. #define UNDEFINED_BLOCK_SIZE ((ULONG) -1)
  17. #define TAPE_SRB_LIST_SIZE 4
  18. NTSTATUS
  19. DriverEntry(
  20. IN PDRIVER_OBJECT DriverObject,
  21. IN PUNICODE_STRING RegistryPath
  22. );
  23. VOID
  24. TapeUnload(
  25. IN PDRIVER_OBJECT DriverObject
  26. );
  27. NTSTATUS
  28. TapeAddDevice(
  29. IN PDRIVER_OBJECT DriverObject,
  30. IN PDEVICE_OBJECT Pdo
  31. );
  32. NTSTATUS
  33. TapeStartDevice(
  34. IN PDEVICE_OBJECT Fdo
  35. );
  36. NTSTATUS
  37. CreateTapeDeviceObject(
  38. IN PDRIVER_OBJECT DriverObject,
  39. IN PDEVICE_OBJECT PhysicalDeviceObject,
  40. IN PTAPE_INIT_DATA_EX TapeInitData
  41. );
  42. VOID
  43. TapeError(
  44. IN PDEVICE_OBJECT DeviceObject,
  45. IN PSCSI_REQUEST_BLOCK Srb,
  46. IN OUT PNTSTATUS Status,
  47. IN OUT PBOOLEAN Retry
  48. );
  49. NTSTATUS
  50. TapeReadWriteVerification(
  51. IN PDEVICE_OBJECT DeviceObject,
  52. IN PIRP Irp
  53. );
  54. VOID
  55. TapeReadWrite(
  56. IN PDEVICE_OBJECT Fdo,
  57. IN PIRP Irp
  58. );
  59. VOID
  60. SplitTapeRequest(
  61. IN PDEVICE_OBJECT Fdo,
  62. IN PIRP Irp,
  63. IN ULONG MaximumBytes
  64. );
  65. NTSTATUS
  66. TapeIoCompleteAssociated(
  67. IN PDEVICE_OBJECT Fdo,
  68. IN PIRP Irp,
  69. IN PVOID Context
  70. );
  71. NTSTATUS
  72. TapeDeviceControl(
  73. IN PDEVICE_OBJECT DeviceObject,
  74. IN PIRP Irp
  75. );
  76. NTSTATUS
  77. TapeInitDevice(
  78. IN PDEVICE_OBJECT Fdo
  79. );
  80. NTSTATUS
  81. TapeRemoveDevice(
  82. IN PDEVICE_OBJECT DeviceObject,
  83. IN UCHAR Type
  84. );
  85. NTSTATUS
  86. TapeStopDevice(
  87. IN PDEVICE_OBJECT DeviceObject,
  88. IN UCHAR Type
  89. );
  90. #ifdef ALLOC_PRAGMA
  91. #pragma alloc_text(INIT, DriverEntry)
  92. #pragma alloc_text(PAGE, TapeUnload)
  93. #pragma alloc_text(PAGE, TapeClassInitialize)
  94. #pragma alloc_text(PAGE, TapeAddDevice)
  95. #pragma alloc_text(PAGE, CreateTapeDeviceObject)
  96. #pragma alloc_text(PAGE, TapeStartDevice)
  97. #pragma alloc_text(PAGE, TapeInitDevice)
  98. #pragma alloc_text(PAGE, TapeRemoveDevice)
  99. #pragma alloc_text(PAGE, TapeStopDevice)
  100. #pragma alloc_text(PAGE, TapeDeviceControl)
  101. #pragma alloc_text(PAGE, TapeReadWriteVerification)
  102. #pragma alloc_text(PAGE, TapeReadWrite)
  103. #pragma alloc_text(PAGE, SplitTapeRequest)
  104. #pragma alloc_text(PAGE, ScsiTapeFreeSrbBuffer)
  105. #pragma alloc_text(PAGE, TapeClassZeroMemory)
  106. #pragma alloc_text(PAGE, TapeClassCompareMemory)
  107. #pragma alloc_text(PAGE, TapeClassLiDiv)
  108. #pragma alloc_text(PAGE, GetTimeoutDeltaFromRegistry)
  109. #endif
  110. NTSTATUS
  111. DriverEntry(
  112. IN PDRIVER_OBJECT DriverObject,
  113. IN PUNICODE_STRING RegistryPath
  114. )
  115. /*++
  116. Routine Description:
  117. This is the entry point for this EXPORT DRIVER. It does nothing.
  118. --*/
  119. {
  120. return STATUS_SUCCESS;
  121. }
  122. ULONG
  123. TapeClassInitialize(
  124. IN PVOID Argument1,
  125. IN PVOID Argument2,
  126. IN PTAPE_INIT_DATA_EX TapeInitData
  127. )
  128. /*++
  129. Routine Description:
  130. This routine is called by a tape mini-class driver during its
  131. DriverEntry routine to initialize the driver.
  132. Arguments:
  133. Argument1 - Supplies the first argument to DriverEntry.
  134. Argument2 - Supplies the second argument to DriverEntry.
  135. TapeInitData - Supplies the tape initialization data.
  136. Return Value:
  137. A valid return code for a DriverEntry routine.
  138. --*/
  139. {
  140. PDRIVER_OBJECT driverObject = Argument1;
  141. PUNICODE_STRING registryPath = Argument2;
  142. PTAPE_INIT_DATA_EX driverExtension;
  143. NTSTATUS status;
  144. CLASS_INIT_DATA initializationData;
  145. TAPE_INIT_DATA_EX tmpInitData;
  146. PAGED_CODE();
  147. DebugPrint((1,"\n\nSCSI Tape Class Driver\n"));
  148. //
  149. // Zero InitData
  150. //
  151. RtlZeroMemory (&tmpInitData, sizeof(TAPE_INIT_DATA_EX));
  152. //
  153. // Save the tape init data passed in from the miniclass driver. When AddDevice gets called, it will be used.
  154. // First a check for 4.0 vs. later miniclass drivers.
  155. //
  156. if (TapeInitData->InitDataSize != sizeof(TAPE_INIT_DATA_EX)) {
  157. //
  158. // Earlier rev. Copy the bits around so that the EX structure is correct.
  159. //
  160. RtlCopyMemory(&tmpInitData.VerifyInquiry, TapeInitData, sizeof(TAPE_INIT_DATA));
  161. //
  162. // Mark it as an earlier rev.
  163. //
  164. tmpInitData.InitDataSize = sizeof(TAPE_INIT_DATA);
  165. } else {
  166. RtlCopyMemory(&tmpInitData, TapeInitData, sizeof(TAPE_INIT_DATA_EX));
  167. }
  168. //
  169. // Get the driverExtension
  170. status = IoAllocateDriverObjectExtension(driverObject,
  171. TapeClassInitialize,
  172. sizeof(TAPE_INIT_DATA_EX),
  173. &driverExtension);
  174. if (!NT_SUCCESS(status)) {
  175. if(status == STATUS_OBJECT_NAME_COLLISION) {
  176. //
  177. // An extension already exists for this key. Get a pointer to it
  178. //
  179. driverExtension = IoGetDriverObjectExtension(driverObject,
  180. TapeClassInitialize);
  181. if (driverExtension == NULL) {
  182. DebugPrint((1, "TapeClassInitialize : driverExtension NULL\n"));
  183. return STATUS_INSUFFICIENT_RESOURCES;
  184. }
  185. } else {
  186. //
  187. // As this failed, the tape init data won't be able to be stored.
  188. //
  189. DebugPrint((1, "TapeClassInitialize: Error %x allocating driver extension.\n",
  190. status));
  191. return status;
  192. }
  193. }
  194. RtlCopyMemory(driverExtension, &tmpInitData, sizeof(TAPE_INIT_DATA_EX));
  195. RtlZeroMemory (&initializationData, sizeof(CLASS_INIT_DATA));
  196. //
  197. // Set sizes
  198. //
  199. initializationData.InitializationDataSize = sizeof(CLASS_INIT_DATA);
  200. initializationData.FdoData.DeviceExtensionSize = sizeof(FUNCTIONAL_DEVICE_EXTENSION) +
  201. sizeof(TAPE_DATA) + tmpInitData.MinitapeExtensionSize;
  202. initializationData.FdoData.DeviceType = FILE_DEVICE_TAPE;
  203. initializationData.FdoData.DeviceCharacteristics = FILE_REMOVABLE_MEDIA |
  204. FILE_DEVICE_SECURE_OPEN;
  205. //
  206. // Set entry points
  207. //
  208. initializationData.FdoData.ClassStartDevice = TapeStartDevice;
  209. initializationData.FdoData.ClassStopDevice = TapeStopDevice;
  210. initializationData.FdoData.ClassInitDevice = TapeInitDevice;
  211. initializationData.FdoData.ClassRemoveDevice = TapeRemoveDevice;
  212. initializationData.ClassAddDevice = TapeAddDevice;
  213. initializationData.FdoData.ClassError = TapeError;
  214. initializationData.FdoData.ClassReadWriteVerification = TapeReadWriteVerification;
  215. initializationData.FdoData.ClassDeviceControl = TapeDeviceControl;
  216. initializationData.FdoData.ClassShutdownFlush = NULL;
  217. initializationData.FdoData.ClassCreateClose = NULL;
  218. //
  219. // Routines for WMI support
  220. //
  221. initializationData.FdoData.ClassWmiInfo.GuidCount = 6;
  222. initializationData.FdoData.ClassWmiInfo.GuidRegInfo = TapeWmiGuidList;
  223. initializationData.FdoData.ClassWmiInfo.ClassQueryWmiRegInfo = TapeQueryWmiRegInfo;
  224. initializationData.FdoData.ClassWmiInfo.ClassQueryWmiDataBlock = TapeQueryWmiDataBlock;
  225. initializationData.FdoData.ClassWmiInfo.ClassSetWmiDataBlock = TapeSetWmiDataBlock;
  226. initializationData.FdoData.ClassWmiInfo.ClassSetWmiDataItem = TapeSetWmiDataItem;
  227. initializationData.FdoData.ClassWmiInfo.ClassExecuteWmiMethod = TapeExecuteWmiMethod;
  228. initializationData.FdoData.ClassWmiInfo.ClassWmiFunctionControl = TapeWmiFunctionControl;
  229. initializationData.ClassUnload = TapeUnload;
  230. //
  231. // Call the class init routine last, so can cleanup if it fails
  232. //
  233. status = ClassInitialize( driverObject, registryPath, &initializationData);
  234. if (!NT_SUCCESS(status)) {
  235. DebugPrint((1, "TapeClassInitialize: Error %x from classinit\n", status));
  236. TapeUnload(driverObject);
  237. }
  238. return status;
  239. }
  240. VOID
  241. TapeUnload(
  242. IN PDRIVER_OBJECT DriverObject
  243. )
  244. {
  245. PAGED_CODE();
  246. UNREFERENCED_PARAMETER(DriverObject);
  247. return;
  248. }
  249. NTSTATUS
  250. TapeAddDevice(
  251. IN PDRIVER_OBJECT DriverObject,
  252. IN PDEVICE_OBJECT PhysicalDeviceObject
  253. )
  254. /*++
  255. Routine Description:
  256. This routine creates and initializes a new FDO for the corresponding
  257. PDO. It may perform property queries on the FDO but cannot do any
  258. media access operations.
  259. Arguments:
  260. DriverObject - Tape class driver object.
  261. Pdo - the physical device object we are being added to
  262. Return Value:
  263. status
  264. --*/
  265. {
  266. PTAPE_INIT_DATA_EX tapeInitData;
  267. NTSTATUS status;
  268. PULONG tapeCount;
  269. PAGED_CODE();
  270. //
  271. // Get the saved-off tape init data.
  272. //
  273. tapeInitData = IoGetDriverObjectExtension(DriverObject, TapeClassInitialize);
  274. ASSERT(tapeInitData);
  275. //
  276. // Get the address of the count of the number of tape devices already initialized.
  277. //
  278. tapeCount = &IoGetConfigurationInformation()->TapeCount;
  279. status = CreateTapeDeviceObject(DriverObject,
  280. PhysicalDeviceObject,
  281. tapeInitData);
  282. if(NT_SUCCESS(status)) {
  283. (*tapeCount)++;
  284. }
  285. return status;
  286. }
  287. NTSTATUS
  288. CreateTapeDeviceObject(
  289. IN PDRIVER_OBJECT DriverObject,
  290. IN PDEVICE_OBJECT PhysicalDeviceObject,
  291. IN PTAPE_INIT_DATA_EX TapeInitData
  292. )
  293. /*++
  294. Routine Description:
  295. This routine creates an object for the device.
  296. Arguments:
  297. DriverObject - Pointer to driver object created by system.
  298. PhysicalDeviceObject - DeviceObject of the attached to device.
  299. TapeInitData - Supplies the tape initialization data.
  300. Return Value:
  301. NTSTATUS
  302. --*/
  303. {
  304. UCHAR deviceNameBuffer[64];
  305. NTSTATUS status;
  306. PDEVICE_OBJECT deviceObject;
  307. PTAPE_INIT_DATA_EX tapeInitData;
  308. PDEVICE_OBJECT lowerDevice;
  309. PTAPE_DATA tapeData;
  310. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = NULL;
  311. WCHAR dosNameBuffer[64];
  312. WCHAR wideNameBuffer[64];
  313. UNICODE_STRING dosUnicodeString;
  314. UNICODE_STRING deviceUnicodeString;
  315. ULONG tapeCount;
  316. PAGED_CODE();
  317. DebugPrint((3,"CreateDeviceObject: Enter routine\n"));
  318. lowerDevice = IoGetAttachedDeviceReference(PhysicalDeviceObject);
  319. //
  320. // Claim the device. Note that any errors after this
  321. // will goto the generic handler, where the device will
  322. // be released.
  323. //
  324. status = ClassClaimDevice(lowerDevice, FALSE);
  325. if(!NT_SUCCESS(status)) {
  326. //
  327. // Someone already had this device - we're in trouble
  328. //
  329. ObDereferenceObject(lowerDevice);
  330. return status;
  331. }
  332. //
  333. // Create device object for this device.
  334. //
  335. tapeCount = 0;
  336. do {
  337. sprintf(deviceNameBuffer,
  338. "\\Device\\Tape%d",
  339. tapeCount);
  340. status = ClassCreateDeviceObject(DriverObject,
  341. deviceNameBuffer,
  342. PhysicalDeviceObject,
  343. TRUE,
  344. &deviceObject);
  345. tapeCount++;
  346. } while (status == STATUS_OBJECT_NAME_COLLISION);
  347. if (!NT_SUCCESS(status)) {
  348. DebugPrint((1,"CreateTapeDeviceObjects: Can not create device %s\n",
  349. deviceNameBuffer));
  350. goto CreateTapeDeviceObjectExit;
  351. }
  352. //
  353. // Indicate that IRPs should include MDLs.
  354. //
  355. deviceObject->Flags |= DO_DIRECT_IO;
  356. fdoExtension = deviceObject->DeviceExtension;
  357. //
  358. // Back pointer to device object.
  359. //
  360. fdoExtension->CommonExtension.DeviceObject = deviceObject;
  361. //
  362. // This is the physical device.
  363. //
  364. fdoExtension->CommonExtension.PartitionZeroExtension = fdoExtension;
  365. //
  366. // Initialize lock count to zero. The lock count is used to
  367. // disable the ejection mechanism when media is mounted.
  368. //
  369. fdoExtension->LockCount = 0;
  370. //
  371. // Save system tape number
  372. //
  373. fdoExtension->DeviceNumber = tapeCount - 1;
  374. //
  375. // Set the alignment requirements for the device based on the
  376. // host adapter requirements
  377. //
  378. if (lowerDevice->AlignmentRequirement > deviceObject->AlignmentRequirement) {
  379. deviceObject->AlignmentRequirement = lowerDevice->AlignmentRequirement;
  380. }
  381. //
  382. // Save the device descriptors
  383. //
  384. fdoExtension->AdapterDescriptor = NULL;
  385. fdoExtension->DeviceDescriptor = NULL;
  386. //
  387. // Attach to the PDO
  388. //
  389. fdoExtension->LowerPdo = PhysicalDeviceObject;
  390. fdoExtension->CommonExtension.LowerDeviceObject =
  391. IoAttachDeviceToDeviceStack(deviceObject, PhysicalDeviceObject);
  392. if(fdoExtension->CommonExtension.LowerDeviceObject == NULL) {
  393. //
  394. // The attach failed. Cleanup and return.
  395. //
  396. status = STATUS_UNSUCCESSFUL;
  397. goto CreateTapeDeviceObjectExit;
  398. }
  399. //
  400. // Save the tape initialization data.
  401. //
  402. RtlCopyMemory(fdoExtension->CommonExtension.DriverData, TapeInitData,sizeof(TAPE_INIT_DATA_EX));
  403. //
  404. // Initialize the splitrequest spinlock.
  405. //
  406. tapeData = (PTAPE_DATA)fdoExtension->CommonExtension.DriverData;
  407. KeInitializeSpinLock(&tapeData->SplitRequestSpinLock);
  408. //
  409. // Create the dos port driver name.
  410. //
  411. swprintf(dosNameBuffer,
  412. L"\\DosDevices\\TAPE%d",
  413. fdoExtension->DeviceNumber);
  414. RtlInitUnicodeString(&dosUnicodeString, dosNameBuffer);
  415. //
  416. // Recreate the deviceName
  417. //
  418. swprintf(wideNameBuffer,
  419. L"\\Device\\Tape%d",
  420. fdoExtension->DeviceNumber);
  421. RtlInitUnicodeString(&deviceUnicodeString,
  422. wideNameBuffer);
  423. status = IoAssignArcName(&dosUnicodeString,
  424. &deviceUnicodeString);
  425. if (NT_SUCCESS(status)) {
  426. tapeData->DosNameCreated = TRUE;
  427. } else {
  428. tapeData->DosNameCreated = FALSE;
  429. }
  430. //
  431. // The device is initialized properly - mark it as such.
  432. //
  433. deviceObject->Flags &= ~DO_DEVICE_INITIALIZING;
  434. ObDereferenceObject(lowerDevice);
  435. return(STATUS_SUCCESS);
  436. CreateTapeDeviceObjectExit:
  437. //
  438. // Release the device since an error occured.
  439. //
  440. // ClassClaimDevice(PortDeviceObject,
  441. // LunInfo,
  442. // TRUE,
  443. // NULL);
  444. ObDereferenceObject(lowerDevice);
  445. if (deviceObject != NULL) {
  446. IoDeleteDevice(deviceObject);
  447. }
  448. return status;
  449. } // end CreateTapeDeviceObject()
  450. NTSTATUS
  451. TapeStartDevice(
  452. IN PDEVICE_OBJECT Fdo
  453. )
  454. /*++
  455. Routine Description:
  456. This routine is called after InitDevice, and creates the symbolic link,
  457. and sets up information in the registry.
  458. The routine could be called multiple times, in the event of a StopDevice.
  459. Arguments:
  460. Fdo - a pointer to the functional device object for this device
  461. Return Value:
  462. status
  463. --*/
  464. {
  465. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension;
  466. PTAPE_DATA tapeData;
  467. PTAPE_INIT_DATA_EX tapeInitData;
  468. PINQUIRYDATA inquiryData;
  469. ULONG inquiryLength;
  470. SCSI_REQUEST_BLOCK srb;
  471. PCDB cdb;
  472. NTSTATUS status;
  473. PVOID minitapeExtension;
  474. PMODE_CAP_PAGE capPage = NULL ;
  475. PMODE_CAPABILITIES_PAGE capabilitiesPage;
  476. ULONG pageLength;
  477. PAGED_CODE();
  478. //
  479. // Build and send request to get inquiry data.
  480. //
  481. inquiryData = ExAllocatePool(NonPagedPoolCacheAligned, MAXIMUM_TAPE_INQUIRY_DATA);
  482. if (!inquiryData) {
  483. //
  484. // The buffer cannot be allocated.
  485. //
  486. return STATUS_INSUFFICIENT_RESOURCES;
  487. }
  488. //
  489. // Get the tape init data again.
  490. //
  491. tapeData = (PTAPE_DATA)(fdoExtension->CommonExtension.DriverData);
  492. tapeInitData = &tapeData->TapeInitData;
  493. RtlZeroMemory(&srb, SCSI_REQUEST_BLOCK_SIZE);
  494. //
  495. // Set timeout value.
  496. //
  497. srb.TimeOutValue = 2;
  498. srb.CdbLength = 6;
  499. cdb = (PCDB)srb.Cdb;
  500. //
  501. // Set CDB operation code.
  502. //
  503. cdb->CDB6INQUIRY.OperationCode = SCSIOP_INQUIRY;
  504. //
  505. // Set allocation length to inquiry data buffer size.
  506. //
  507. cdb->CDB6INQUIRY.AllocationLength = MAXIMUM_TAPE_INQUIRY_DATA;
  508. status = ClassSendSrbSynchronous(Fdo,
  509. &srb,
  510. inquiryData,
  511. MAXIMUM_TAPE_INQUIRY_DATA,
  512. FALSE);
  513. if (SRB_STATUS(srb.SrbStatus) == SRB_STATUS_SUCCESS ||
  514. SRB_STATUS(srb.SrbStatus) == SRB_STATUS_DATA_OVERRUN) {
  515. srb.SrbStatus = SRB_STATUS_SUCCESS;
  516. }
  517. if (srb.SrbStatus == SRB_STATUS_SUCCESS) {
  518. inquiryLength = inquiryData->AdditionalLength + FIELD_OFFSET(INQUIRYDATA, Reserved);
  519. if (inquiryLength > srb.DataTransferLength) {
  520. inquiryLength = srb.DataTransferLength;
  521. }
  522. //
  523. // Verify that we really want this device.
  524. //
  525. if (tapeInitData->QueryModeCapabilitiesPage ) {
  526. capPage = ExAllocatePool(NonPagedPoolCacheAligned,
  527. sizeof(MODE_CAP_PAGE));
  528. }
  529. if (capPage) {
  530. pageLength = ClassModeSense(Fdo,
  531. (PCHAR) capPage,
  532. sizeof(MODE_CAP_PAGE),
  533. MODE_PAGE_CAPABILITIES);
  534. if (pageLength == 0) {
  535. pageLength = ClassModeSense(Fdo,
  536. (PCHAR) capPage,
  537. sizeof(MODE_CAP_PAGE),
  538. MODE_PAGE_CAPABILITIES);
  539. }
  540. if (pageLength < (sizeof(MODE_CAP_PAGE) - 1)) {
  541. ExFreePool(capPage);
  542. capPage = NULL;
  543. }
  544. }
  545. if (capPage) {
  546. capabilitiesPage = &(capPage->CapabilitiesPage);
  547. } else {
  548. capabilitiesPage = NULL;
  549. }
  550. //
  551. // Initialize the minitape extension.
  552. //
  553. if (tapeInitData->ExtensionInit) {
  554. minitapeExtension = tapeData + 1;
  555. tapeInitData->ExtensionInit(minitapeExtension,
  556. inquiryData,
  557. capabilitiesPage);
  558. }
  559. if (capPage) {
  560. ExFreePool(capPage);
  561. }
  562. } else {
  563. inquiryLength = 0;
  564. }
  565. //
  566. // Add tape device number to registry
  567. //
  568. ClassUpdateInformationInRegistry(Fdo,
  569. "Tape",
  570. fdoExtension->DeviceNumber,
  571. inquiryData,
  572. inquiryLength);
  573. ExFreePool(inquiryData);
  574. status = IoSetDeviceInterfaceState(&(tapeData->TapeInterfaceString),
  575. TRUE);
  576. if(!NT_SUCCESS(status)) {
  577. DebugPrint((1,
  578. "TapeStartDevice: Unable to register Tape%x interface name - %x.\n",
  579. fdoExtension->DeviceNumber,
  580. status));
  581. }
  582. return STATUS_SUCCESS;
  583. }
  584. NTSTATUS
  585. TapeInitDevice(
  586. IN PDEVICE_OBJECT Fdo
  587. )
  588. /*++
  589. Routine Description:
  590. This routine will complete the tape miniclass initialization. This includes
  591. allocating sense info buffers and srb s-lists.
  592. This routine will not clean up allocate resources if it fails - that
  593. is left for device stop/removal
  594. Arguments:
  595. Fdo - a pointer to the functional device object for this device
  596. Return Value:
  597. status
  598. --*/
  599. {
  600. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension;
  601. PVOID senseData = NULL;
  602. PTAPE_DATA tapeData;
  603. PTAPE_INIT_DATA_EX tapeInitData;
  604. NTSTATUS status;
  605. PVOID minitapeExtension;
  606. STORAGE_PROPERTY_ID propertyId;
  607. UNICODE_STRING interfaceName;
  608. PAGED_CODE();
  609. //
  610. // Allocate request sense buffer.
  611. //
  612. senseData = ExAllocatePool(NonPagedPoolCacheAligned,
  613. SENSE_BUFFER_SIZE);
  614. if (senseData == NULL) {
  615. //
  616. // The buffer cannot be allocated.
  617. //
  618. return STATUS_INSUFFICIENT_RESOURCES;
  619. }
  620. //
  621. // Build the lookaside list for srb's for the physical disk. Should only
  622. // need a couple.
  623. //
  624. ClassInitializeSrbLookasideList(&(fdoExtension->CommonExtension),
  625. TAPE_SRB_LIST_SIZE);
  626. //
  627. // Set the sense data pointer in the device extension.
  628. //
  629. fdoExtension->SenseData = senseData;
  630. fdoExtension->DiskGeometry.BytesPerSector = UNDEFINED_BLOCK_SIZE;
  631. //
  632. // Get the tape init data again.
  633. //
  634. tapeData = (PTAPE_DATA)(fdoExtension->CommonExtension.DriverData);
  635. tapeInitData = &tapeData->TapeInitData;
  636. //
  637. // Set timeout value in seconds.
  638. //
  639. if (tapeInitData->DefaultTimeOutValue) {
  640. fdoExtension->TimeOutValue = tapeInitData->DefaultTimeOutValue;
  641. } else {
  642. fdoExtension->TimeOutValue = 180;
  643. }
  644. //
  645. // Used to keep track of the last time a drive clean
  646. // notification was sent by the driver
  647. //
  648. tapeData->LastDriveCleanRequestTime.QuadPart = 0;
  649. //
  650. // SRB Timeout delta is used to increase the timeout for certain
  651. // commands - typically, commands such as SET_POSITION, ERASE, etc.
  652. //
  653. tapeData->SrbTimeoutDelta = GetTimeoutDeltaFromRegistry(fdoExtension->LowerPdo);
  654. if ((tapeData->SrbTimeoutDelta) == 0) {
  655. tapeData->SrbTimeoutDelta = fdoExtension->TimeOutValue;
  656. }
  657. //
  658. // Call port driver to get adapter capabilities.
  659. //
  660. propertyId = StorageAdapterProperty;
  661. status = ClassGetDescriptor(fdoExtension->CommonExtension.LowerDeviceObject,
  662. &propertyId,
  663. &(fdoExtension->AdapterDescriptor));
  664. if(!NT_SUCCESS(status)) {
  665. DebugPrint((1,
  666. "TapeStartDevice: Unable to get adapter descriptor. Status %x\n",
  667. status));
  668. ExFreePool(senseData);
  669. return status;
  670. }
  671. //
  672. // Register for media change notification
  673. //
  674. ClassInitializeMediaChangeDetection(fdoExtension,
  675. "Tape");
  676. //
  677. // Register interfaces for this device.
  678. //
  679. RtlInitUnicodeString(&tapeData->TapeInterfaceString, NULL);
  680. status = IoRegisterDeviceInterface(fdoExtension->LowerPdo,
  681. (LPGUID) &TapeClassGuid,
  682. NULL,
  683. &(tapeData->TapeInterfaceString));
  684. if(!NT_SUCCESS(status)) {
  685. DebugPrint((1,
  686. "TapeInitDevice: Unable to register Tape%x interface name - %x.\n",
  687. fdoExtension->DeviceNumber,
  688. status));
  689. status = STATUS_SUCCESS;
  690. }
  691. return STATUS_SUCCESS;
  692. } // End TapeStartDevice
  693. NTSTATUS
  694. TapeRemoveDevice(
  695. IN PDEVICE_OBJECT DeviceObject,
  696. IN UCHAR Type
  697. )
  698. /*++
  699. Routine Description:
  700. This routine is responsible for releasing any resources in use by the
  701. tape driver.
  702. Arguments:
  703. DeviceObject - the device object being removed
  704. Return Value:
  705. none - this routine may not fail
  706. --*/
  707. {
  708. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
  709. PTAPE_DATA tapeData = (PTAPE_DATA)fdoExtension->CommonExtension.DriverData;
  710. WCHAR dosNameBuffer[64];
  711. UNICODE_STRING dosUnicodeString;
  712. NTSTATUS status;
  713. PAGED_CODE();
  714. if((Type == IRP_MN_QUERY_REMOVE_DEVICE) ||
  715. (Type == IRP_MN_CANCEL_REMOVE_DEVICE)) {
  716. return STATUS_SUCCESS;
  717. }
  718. //
  719. // Free all allocated memory.
  720. //
  721. if (fdoExtension->DeviceDescriptor) {
  722. ExFreePool(fdoExtension->DeviceDescriptor);
  723. fdoExtension->DeviceDescriptor = NULL;
  724. }
  725. if (fdoExtension->AdapterDescriptor) {
  726. ExFreePool(fdoExtension->AdapterDescriptor);
  727. fdoExtension->AdapterDescriptor = NULL;
  728. }
  729. if (fdoExtension->SenseData) {
  730. ExFreePool(fdoExtension->SenseData);
  731. fdoExtension->SenseData = NULL;
  732. }
  733. //
  734. // Remove the lookaside list.
  735. //
  736. ClassDeleteSrbLookasideList(&fdoExtension->CommonExtension);
  737. if(tapeData->TapeInterfaceString.Buffer != NULL) {
  738. IoSetDeviceInterfaceState(&(tapeData->TapeInterfaceString),
  739. FALSE);
  740. RtlFreeUnicodeString(&(tapeData->TapeInterfaceString));
  741. //
  742. // Clear it.
  743. //
  744. RtlInitUnicodeString(&(tapeData->TapeInterfaceString), NULL);
  745. }
  746. if(tapeData->DosNameCreated) {
  747. //
  748. // Delete the symbolic link "tapeN".
  749. //
  750. swprintf(dosNameBuffer,
  751. L"\\DosDevices\\TAPE%d",
  752. fdoExtension->DeviceNumber);
  753. RtlInitUnicodeString(&dosUnicodeString, dosNameBuffer);
  754. IoDeleteSymbolicLink(&dosUnicodeString);
  755. tapeData->DosNameCreated = FALSE;
  756. }
  757. IoGetConfigurationInformation()->TapeCount--;
  758. return STATUS_SUCCESS;
  759. }
  760. NTSTATUS
  761. TapeStopDevice(
  762. IN PDEVICE_OBJECT DeviceObject,
  763. IN UCHAR Type
  764. )
  765. {
  766. PAGED_CODE();
  767. return STATUS_SUCCESS;
  768. }
  769. BOOLEAN
  770. ScsiTapeNtStatusToTapeStatus(
  771. IN NTSTATUS NtStatus,
  772. OUT PTAPE_STATUS TapeStatus
  773. )
  774. /*++
  775. Routine Description:
  776. This routine translates an NT status code to a TAPE status code.
  777. Arguments:
  778. NtStatus - Supplies the NT status code.
  779. TapeStatus - Returns the tape status code.
  780. Return Value:
  781. FALSE - No tranlation was possible.
  782. TRUE - Success.
  783. --*/
  784. {
  785. switch (NtStatus) {
  786. case STATUS_SUCCESS:
  787. *TapeStatus = TAPE_STATUS_SUCCESS;
  788. break;
  789. case STATUS_INSUFFICIENT_RESOURCES:
  790. *TapeStatus = TAPE_STATUS_INSUFFICIENT_RESOURCES;
  791. break;
  792. case STATUS_NOT_IMPLEMENTED:
  793. *TapeStatus = TAPE_STATUS_NOT_IMPLEMENTED;
  794. break;
  795. case STATUS_INVALID_DEVICE_REQUEST:
  796. *TapeStatus = TAPE_STATUS_INVALID_DEVICE_REQUEST;
  797. break;
  798. case STATUS_INVALID_PARAMETER:
  799. *TapeStatus = TAPE_STATUS_INVALID_PARAMETER;
  800. break;
  801. case STATUS_VERIFY_REQUIRED:
  802. case STATUS_MEDIA_CHANGED:
  803. *TapeStatus = TAPE_STATUS_MEDIA_CHANGED;
  804. break;
  805. case STATUS_BUS_RESET:
  806. *TapeStatus = TAPE_STATUS_BUS_RESET;
  807. break;
  808. case STATUS_SETMARK_DETECTED:
  809. *TapeStatus = TAPE_STATUS_SETMARK_DETECTED;
  810. break;
  811. case STATUS_FILEMARK_DETECTED:
  812. *TapeStatus = TAPE_STATUS_FILEMARK_DETECTED;
  813. break;
  814. case STATUS_BEGINNING_OF_MEDIA:
  815. *TapeStatus = TAPE_STATUS_BEGINNING_OF_MEDIA;
  816. break;
  817. case STATUS_END_OF_MEDIA:
  818. *TapeStatus = TAPE_STATUS_END_OF_MEDIA;
  819. break;
  820. case STATUS_BUFFER_OVERFLOW:
  821. *TapeStatus = TAPE_STATUS_BUFFER_OVERFLOW;
  822. break;
  823. case STATUS_NO_DATA_DETECTED:
  824. *TapeStatus = TAPE_STATUS_NO_DATA_DETECTED;
  825. break;
  826. case STATUS_EOM_OVERFLOW:
  827. *TapeStatus = TAPE_STATUS_EOM_OVERFLOW;
  828. break;
  829. case STATUS_NO_MEDIA:
  830. case STATUS_NO_MEDIA_IN_DEVICE:
  831. *TapeStatus = TAPE_STATUS_NO_MEDIA;
  832. break;
  833. case STATUS_IO_DEVICE_ERROR:
  834. case STATUS_NONEXISTENT_SECTOR:
  835. *TapeStatus = TAPE_STATUS_IO_DEVICE_ERROR;
  836. break;
  837. case STATUS_UNRECOGNIZED_MEDIA:
  838. *TapeStatus = TAPE_STATUS_UNRECOGNIZED_MEDIA;
  839. break;
  840. case STATUS_DEVICE_NOT_READY:
  841. *TapeStatus = TAPE_STATUS_DEVICE_NOT_READY;
  842. break;
  843. case STATUS_MEDIA_WRITE_PROTECTED:
  844. *TapeStatus = TAPE_STATUS_MEDIA_WRITE_PROTECTED;
  845. break;
  846. case STATUS_DEVICE_DATA_ERROR:
  847. *TapeStatus = TAPE_STATUS_DEVICE_DATA_ERROR;
  848. break;
  849. case STATUS_NO_SUCH_DEVICE:
  850. *TapeStatus = TAPE_STATUS_NO_SUCH_DEVICE;
  851. break;
  852. case STATUS_INVALID_BLOCK_LENGTH:
  853. *TapeStatus = TAPE_STATUS_INVALID_BLOCK_LENGTH;
  854. break;
  855. case STATUS_IO_TIMEOUT:
  856. *TapeStatus = TAPE_STATUS_IO_TIMEOUT;
  857. break;
  858. case STATUS_DEVICE_NOT_CONNECTED:
  859. *TapeStatus = TAPE_STATUS_DEVICE_NOT_CONNECTED;
  860. break;
  861. case STATUS_DATA_OVERRUN:
  862. *TapeStatus = TAPE_STATUS_DATA_OVERRUN;
  863. break;
  864. case STATUS_DEVICE_BUSY:
  865. *TapeStatus = TAPE_STATUS_DEVICE_BUSY;
  866. break;
  867. case STATUS_CLEANER_CARTRIDGE_INSTALLED:
  868. *TapeStatus = TAPE_STATUS_CLEANER_CARTRIDGE_INSTALLED;
  869. break;
  870. default:
  871. return FALSE;
  872. }
  873. return TRUE;
  874. }
  875. BOOLEAN
  876. ScsiTapeTapeStatusToNtStatus(
  877. IN TAPE_STATUS TapeStatus,
  878. OUT PNTSTATUS NtStatus
  879. )
  880. /*++
  881. Routine Description:
  882. This routine translates a TAPE status code to an NT status code.
  883. Arguments:
  884. TapeStatus - Supplies the tape status code.
  885. NtStatus - Returns the NT status code.
  886. Return Value:
  887. FALSE - No tranlation was possible.
  888. TRUE - Success.
  889. --*/
  890. {
  891. switch (TapeStatus) {
  892. case TAPE_STATUS_SUCCESS:
  893. *NtStatus = STATUS_SUCCESS;
  894. break;
  895. case TAPE_STATUS_INSUFFICIENT_RESOURCES:
  896. *NtStatus = STATUS_INSUFFICIENT_RESOURCES;
  897. break;
  898. case TAPE_STATUS_NOT_IMPLEMENTED:
  899. *NtStatus = STATUS_NOT_IMPLEMENTED;
  900. break;
  901. case TAPE_STATUS_INVALID_DEVICE_REQUEST:
  902. *NtStatus = STATUS_INVALID_DEVICE_REQUEST;
  903. break;
  904. case TAPE_STATUS_INVALID_PARAMETER:
  905. *NtStatus = STATUS_INVALID_PARAMETER;
  906. break;
  907. case TAPE_STATUS_MEDIA_CHANGED:
  908. *NtStatus = STATUS_VERIFY_REQUIRED;
  909. break;
  910. case TAPE_STATUS_BUS_RESET:
  911. *NtStatus = STATUS_BUS_RESET;
  912. break;
  913. case TAPE_STATUS_SETMARK_DETECTED:
  914. *NtStatus = STATUS_SETMARK_DETECTED;
  915. break;
  916. case TAPE_STATUS_FILEMARK_DETECTED:
  917. *NtStatus = STATUS_FILEMARK_DETECTED;
  918. break;
  919. case TAPE_STATUS_BEGINNING_OF_MEDIA:
  920. *NtStatus = STATUS_BEGINNING_OF_MEDIA;
  921. break;
  922. case TAPE_STATUS_END_OF_MEDIA:
  923. *NtStatus = STATUS_END_OF_MEDIA;
  924. break;
  925. case TAPE_STATUS_BUFFER_OVERFLOW:
  926. *NtStatus = STATUS_BUFFER_OVERFLOW;
  927. break;
  928. case TAPE_STATUS_NO_DATA_DETECTED:
  929. *NtStatus = STATUS_NO_DATA_DETECTED;
  930. break;
  931. case TAPE_STATUS_EOM_OVERFLOW:
  932. *NtStatus = STATUS_EOM_OVERFLOW;
  933. break;
  934. case TAPE_STATUS_NO_MEDIA:
  935. *NtStatus = STATUS_NO_MEDIA;
  936. break;
  937. case TAPE_STATUS_IO_DEVICE_ERROR:
  938. *NtStatus = STATUS_IO_DEVICE_ERROR;
  939. break;
  940. case TAPE_STATUS_UNRECOGNIZED_MEDIA:
  941. *NtStatus = STATUS_UNRECOGNIZED_MEDIA;
  942. break;
  943. case TAPE_STATUS_DEVICE_NOT_READY:
  944. *NtStatus = STATUS_DEVICE_NOT_READY;
  945. break;
  946. case TAPE_STATUS_MEDIA_WRITE_PROTECTED:
  947. *NtStatus = STATUS_MEDIA_WRITE_PROTECTED;
  948. break;
  949. case TAPE_STATUS_DEVICE_DATA_ERROR:
  950. *NtStatus = STATUS_DEVICE_DATA_ERROR;
  951. break;
  952. case TAPE_STATUS_NO_SUCH_DEVICE:
  953. *NtStatus = STATUS_NO_SUCH_DEVICE;
  954. break;
  955. case TAPE_STATUS_INVALID_BLOCK_LENGTH:
  956. *NtStatus = STATUS_INVALID_BLOCK_LENGTH;
  957. break;
  958. case TAPE_STATUS_IO_TIMEOUT:
  959. *NtStatus = STATUS_IO_TIMEOUT;
  960. break;
  961. case TAPE_STATUS_DEVICE_NOT_CONNECTED:
  962. *NtStatus = STATUS_DEVICE_NOT_CONNECTED;
  963. break;
  964. case TAPE_STATUS_DATA_OVERRUN:
  965. *NtStatus = STATUS_DATA_OVERRUN;
  966. break;
  967. case TAPE_STATUS_DEVICE_BUSY:
  968. *NtStatus = STATUS_DEVICE_BUSY;
  969. break;
  970. case TAPE_STATUS_REQUIRES_CLEANING:
  971. *NtStatus = STATUS_DEVICE_REQUIRES_CLEANING;
  972. break;
  973. case TAPE_STATUS_CLEANER_CARTRIDGE_INSTALLED:
  974. *NtStatus = STATUS_CLEANER_CARTRIDGE_INSTALLED;
  975. break;
  976. default:
  977. return FALSE;
  978. }
  979. return TRUE;
  980. }
  981. VOID
  982. TapeError(
  983. IN PDEVICE_OBJECT FDO,
  984. IN PSCSI_REQUEST_BLOCK Srb,
  985. IN OUT PNTSTATUS Status,
  986. IN OUT PBOOLEAN Retry
  987. )
  988. /*++
  989. Routine Description:
  990. When a request completes with error, the routine ScsiClassInterpretSenseInfo is
  991. called to determine from the sense data whether the request should be
  992. retried and what NT status to set in the IRP. Then this routine is called
  993. for tape requests to handle tape-specific errors and update the nt status
  994. and retry boolean.
  995. Arguments:
  996. DeviceObject - Supplies a pointer to the device object.
  997. Srb - Supplies a pointer to the failing Srb.
  998. Status - NT Status used to set the IRP's completion status.
  999. Retry - Indicates that this request should be retried.
  1000. Return Value:
  1001. None.
  1002. --*/
  1003. {
  1004. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = FDO->DeviceExtension;
  1005. PTAPE_DATA tapeData = (PTAPE_DATA)(fdoExtension->CommonExtension.DriverData);
  1006. PTAPE_INIT_DATA_EX tapeInitData = &tapeData->TapeInitData;
  1007. PVOID minitapeExtension = (tapeData + 1);
  1008. PSENSE_DATA senseBuffer = Srb->SenseInfoBuffer;
  1009. PIRP irp = Srb->OriginalRequest;
  1010. LONG residualBlocks;
  1011. LONG length;
  1012. TAPE_STATUS tapeStatus, oldTapeStatus;
  1013. TARGET_DEVICE_CUSTOM_NOTIFICATION NotificationStructure[2];
  1014. //
  1015. // Never retry tape requests.
  1016. //
  1017. *Retry = FALSE;
  1018. //
  1019. // Check that request sense buffer is valid.
  1020. //
  1021. if (Srb->SrbStatus & SRB_STATUS_AUTOSENSE_VALID) {
  1022. DebugPrint((1,
  1023. "Sense Code : %x, Ad Sense : %x, Ad Sense Qual : %x\n",
  1024. ((senseBuffer->SenseKey) & 0xf),
  1025. (senseBuffer->AdditionalSenseCode),
  1026. (senseBuffer->AdditionalSenseCodeQualifier)));
  1027. switch (senseBuffer->SenseKey & 0xf) {
  1028. case SCSI_SENSE_UNIT_ATTENTION:
  1029. switch (senseBuffer->AdditionalSenseCode) {
  1030. case SCSI_ADSENSE_MEDIUM_CHANGED:
  1031. DebugPrint((1,
  1032. "InterpretSenseInfo: Media changed\n"));
  1033. *Status = STATUS_MEDIA_CHANGED;
  1034. break;
  1035. default:
  1036. DebugPrint((1,
  1037. "InterpretSenseInfo: Bus reset\n"));
  1038. *Status = STATUS_BUS_RESET;
  1039. break;
  1040. }
  1041. break;
  1042. case SCSI_SENSE_RECOVERED_ERROR:
  1043. //
  1044. // Check other indicators
  1045. //
  1046. if (senseBuffer->FileMark) {
  1047. switch (senseBuffer->AdditionalSenseCodeQualifier) {
  1048. case SCSI_SENSEQ_SETMARK_DETECTED :
  1049. DebugPrint((1,
  1050. "InterpretSenseInfo: Setmark detected\n"));
  1051. *Status = STATUS_SETMARK_DETECTED;
  1052. break ;
  1053. case SCSI_SENSEQ_FILEMARK_DETECTED :
  1054. default:
  1055. DebugPrint((1,
  1056. "InterpretSenseInfo: Filemark detected\n"));
  1057. *Status = STATUS_FILEMARK_DETECTED;
  1058. break ;
  1059. }
  1060. } else if ( senseBuffer->EndOfMedia ) {
  1061. switch( senseBuffer->AdditionalSenseCodeQualifier ) {
  1062. case SCSI_SENSEQ_BEGINNING_OF_MEDIA_DETECTED :
  1063. DebugPrint((1,
  1064. "InterpretSenseInfo: Beginning of media detected\n"));
  1065. *Status = STATUS_BEGINNING_OF_MEDIA;
  1066. break ;
  1067. case SCSI_SENSEQ_END_OF_MEDIA_DETECTED :
  1068. default:
  1069. DebugPrint((1,
  1070. "InterpretSenseInfo: End of media detected\n"));
  1071. *Status = STATUS_END_OF_MEDIA;
  1072. break ;
  1073. }
  1074. }
  1075. break;
  1076. case SCSI_SENSE_NO_SENSE:
  1077. //
  1078. // Check other indicators
  1079. //
  1080. if (senseBuffer->FileMark) {
  1081. switch( senseBuffer->AdditionalSenseCodeQualifier ) {
  1082. case SCSI_SENSEQ_SETMARK_DETECTED :
  1083. DebugPrint((1,
  1084. "InterpretSenseInfo: Setmark detected\n"));
  1085. *Status = STATUS_SETMARK_DETECTED;
  1086. break ;
  1087. case SCSI_SENSEQ_FILEMARK_DETECTED :
  1088. default:
  1089. DebugPrint((1,
  1090. "InterpretSenseInfo: Filemark detected\n"));
  1091. *Status = STATUS_FILEMARK_DETECTED;
  1092. break ;
  1093. }
  1094. } else if (senseBuffer->EndOfMedia) {
  1095. switch(senseBuffer->AdditionalSenseCodeQualifier) {
  1096. case SCSI_SENSEQ_BEGINNING_OF_MEDIA_DETECTED :
  1097. DebugPrint((1,
  1098. "InterpretSenseInfo: Beginning of media detected\n"));
  1099. *Status = STATUS_BEGINNING_OF_MEDIA;
  1100. break ;
  1101. case SCSI_SENSEQ_END_OF_MEDIA_DETECTED :
  1102. default:
  1103. DebugPrint((1,
  1104. "InterpretSenseInfo: End of media detected\n"));
  1105. *Status = STATUS_END_OF_MEDIA;
  1106. break;
  1107. }
  1108. } else if (senseBuffer->IncorrectLength) {
  1109. //
  1110. // If we're in variable block mode then ignore
  1111. // incorrect length.
  1112. //
  1113. if (fdoExtension->DiskGeometry.BytesPerSector == 0 &&
  1114. Srb->Cdb[0] == SCSIOP_READ6) {
  1115. REVERSE_BYTES((FOUR_BYTE UNALIGNED *)&residualBlocks,
  1116. (FOUR_BYTE UNALIGNED *)(senseBuffer->Information));
  1117. if (residualBlocks >= 0) {
  1118. DebugPrint((1,"InterpretSenseInfo: In variable block mode :We read less than specified\n"));
  1119. *Status = STATUS_SUCCESS;
  1120. } else {
  1121. DebugPrint((1,"InterpretSenseInfo: In variable block mode :Data left in block\n"));
  1122. *Status = STATUS_BUFFER_OVERFLOW;
  1123. }
  1124. }
  1125. }
  1126. break;
  1127. case SCSI_SENSE_BLANK_CHECK:
  1128. DebugPrint((1,
  1129. "InterpretSenseInfo: Media blank check\n"));
  1130. *Status = STATUS_NO_DATA_DETECTED;
  1131. break;
  1132. case SCSI_SENSE_VOL_OVERFLOW:
  1133. DebugPrint((1,
  1134. "InterpretSenseInfo: End of Media Overflow\n"));
  1135. *Status = STATUS_EOM_OVERFLOW;
  1136. break;
  1137. case SCSI_SENSE_NOT_READY:
  1138. switch (senseBuffer->AdditionalSenseCode) {
  1139. case SCSI_ADSENSE_LUN_NOT_READY:
  1140. switch (senseBuffer->AdditionalSenseCodeQualifier) {
  1141. case SCSI_SENSEQ_MANUAL_INTERVENTION_REQUIRED:
  1142. *Status = STATUS_NO_MEDIA;
  1143. break;
  1144. case SCSI_SENSEQ_FORMAT_IN_PROGRESS:
  1145. break;
  1146. case SCSI_SENSEQ_INIT_COMMAND_REQUIRED:
  1147. default:
  1148. //
  1149. // Allow retries, if the drive isn't ready.
  1150. //
  1151. *Retry = TRUE;
  1152. break;
  1153. }
  1154. break;
  1155. case SCSI_ADSENSE_NO_MEDIA_IN_DEVICE:
  1156. DebugPrint((1,
  1157. "InterpretSenseInfo:"
  1158. " No Media in device.\n"));
  1159. *Status = STATUS_NO_MEDIA;
  1160. break;
  1161. }
  1162. break;
  1163. } // end switch
  1164. //
  1165. // Check if a filemark or setmark was encountered,
  1166. // or an end-of-media or no-data condition exists.
  1167. //
  1168. if ((NT_WARNING(*Status) || NT_SUCCESS( *Status)) &&
  1169. (Srb->Cdb[0] == SCSIOP_WRITE6 || Srb->Cdb[0] == SCSIOP_READ6)) {
  1170. LONG actualLength;
  1171. //
  1172. // Not all bytes were transfered. Update information field with
  1173. // number of bytes transfered from sense buffer.
  1174. //
  1175. if (senseBuffer->Valid) {
  1176. REVERSE_BYTES((FOUR_BYTE UNALIGNED *)&residualBlocks,
  1177. (FOUR_BYTE UNALIGNED *)(senseBuffer->Information));
  1178. } else {
  1179. residualBlocks = 0;
  1180. }
  1181. length = ((PCDB) Srb->Cdb)->CDB6READWRITETAPE.TransferLenLSB;
  1182. length |= ((PCDB) Srb->Cdb)->CDB6READWRITETAPE.TransferLen << 8;
  1183. length |= ((PCDB) Srb->Cdb)->CDB6READWRITETAPE.TransferLenMSB << 16;
  1184. actualLength = length;
  1185. length -= residualBlocks;
  1186. if (length < 0) {
  1187. length = 0;
  1188. *Status = STATUS_IO_DEVICE_ERROR;
  1189. }
  1190. if (fdoExtension->DiskGeometry.BytesPerSector) {
  1191. actualLength *= fdoExtension->DiskGeometry.BytesPerSector;
  1192. length *= fdoExtension->DiskGeometry.BytesPerSector;
  1193. }
  1194. if (length > actualLength) {
  1195. length = actualLength;
  1196. }
  1197. irp->IoStatus.Information = length;
  1198. DebugPrint((1,"ScsiTapeError: Transfer Count: %lx\n", Srb->DataTransferLength));
  1199. DebugPrint((1," Residual Blocks: %lx\n", residualBlocks));
  1200. DebugPrint((1," Irp IoStatus Information = %lx\n", irp->IoStatus.Information));
  1201. }
  1202. } else {
  1203. DebugPrint((1, "SRB Status : %x, SCSI Status : %x\n",
  1204. SRB_STATUS(Srb->SrbStatus),
  1205. (Srb->ScsiStatus)));
  1206. }
  1207. //
  1208. // Call tape device specific error handler.
  1209. //
  1210. if (tapeInitData->TapeError &&
  1211. ScsiTapeNtStatusToTapeStatus(*Status, &tapeStatus)) {
  1212. oldTapeStatus = tapeStatus;
  1213. tapeInitData->TapeError(minitapeExtension, Srb, &tapeStatus);
  1214. if (tapeStatus != oldTapeStatus) {
  1215. ScsiTapeTapeStatusToNtStatus(tapeStatus, Status);
  1216. }
  1217. }
  1218. //
  1219. // Notify the system that this tape drive requires cleaning
  1220. //
  1221. if ((*Status) == STATUS_DEVICE_REQUIRES_CLEANING) {
  1222. LARGE_INTEGER currentTime;
  1223. LARGE_INTEGER driveCleanInterval;
  1224. KeQuerySystemTime(&currentTime);
  1225. driveCleanInterval.QuadPart = ONE_SECOND;
  1226. driveCleanInterval.QuadPart *= TAPE_DRIVE_CLEAN_NOTIFICATION_INTERVAL;
  1227. if ((currentTime.QuadPart) >
  1228. ((tapeData->LastDriveCleanRequestTime.QuadPart) +
  1229. (driveCleanInterval.QuadPart))) {
  1230. NotificationStructure[0].Event = GUID_IO_DRIVE_REQUIRES_CLEANING;
  1231. NotificationStructure[0].Version = 1;
  1232. NotificationStructure[0].Size = sizeof(TARGET_DEVICE_CUSTOM_NOTIFICATION) +
  1233. sizeof(ULONG) - sizeof(UCHAR);
  1234. NotificationStructure[0].FileObject = NULL;
  1235. NotificationStructure[0].NameBufferOffset = -1;
  1236. //
  1237. // Increasing Index for this event
  1238. //
  1239. *((PULONG) (&(NotificationStructure[0].CustomDataBuffer[0]))) = 0;
  1240. IoReportTargetDeviceChangeAsynchronous(fdoExtension->LowerPdo,
  1241. &NotificationStructure[0],
  1242. NULL,
  1243. NULL);
  1244. tapeData->LastDriveCleanRequestTime.QuadPart = currentTime.QuadPart;
  1245. }
  1246. }
  1247. return;
  1248. } // end ScsiTapeError()
  1249. NTSTATUS
  1250. TapeReadWriteVerification(
  1251. IN PDEVICE_OBJECT DeviceObject,
  1252. IN PIRP Irp
  1253. )
  1254. /*++
  1255. Routine Description:
  1256. This routine builds up the given irp for a read or write request.
  1257. Arguments:
  1258. DeviceObject - Supplies the device object.
  1259. Irp - Supplies the I/O request packet.
  1260. Return Value:
  1261. None.
  1262. --*/
  1263. {
  1264. PCOMMON_DEVICE_EXTENSION commonExtension = DeviceObject->DeviceExtension;
  1265. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = commonExtension->PartitionZeroExtension;
  1266. PSTORAGE_ADAPTER_DESCRIPTOR adapterDescriptor = fdoExtension->CommonExtension.PartitionZeroExtension->AdapterDescriptor;
  1267. PIO_STACK_LOCATION currentIrpStack = IoGetCurrentIrpStackLocation(Irp);
  1268. ULONG transferPages;
  1269. ULONG transferByteCount = currentIrpStack->Parameters.Read.Length;
  1270. LARGE_INTEGER startingOffset = currentIrpStack->Parameters.Read.ByteOffset;
  1271. ULONG maximumTransferLength = adapterDescriptor->MaximumTransferLength;
  1272. ULONG bytesPerSector = fdoExtension->DiskGeometry.BytesPerSector;
  1273. PAGED_CODE();
  1274. //
  1275. // Since most tape devices don't support 10-byte read/write, the entire request must be dealt with here.
  1276. // STATUS_PENDING will be returned to the classpnp driver, so that it does nothing.
  1277. //
  1278. //
  1279. // Ensure that the request is for something valid - ie. not 0.
  1280. //
  1281. if (currentIrpStack->Parameters.Read.Length == 0) {
  1282. //
  1283. // Class code will handle this.
  1284. //
  1285. return STATUS_SUCCESS;
  1286. }
  1287. //
  1288. // Check that blocksize has been established.
  1289. //
  1290. if (bytesPerSector == UNDEFINED_BLOCK_SIZE) {
  1291. DebugPrint((1,
  1292. "TapeReadWriteVerification: Invalid block size - UNDEFINED\n"));
  1293. Irp->IoStatus.Status = STATUS_INVALID_PARAMETER;
  1294. Irp->IoStatus.Information = 0;
  1295. //
  1296. // ClassPnp will handle completing the request.
  1297. //
  1298. return STATUS_INVALID_PARAMETER;
  1299. }
  1300. if (bytesPerSector) {
  1301. if (transferByteCount % bytesPerSector) {
  1302. DebugPrint((1,
  1303. "TapeReadWriteVerification: Invalid block size\n"));
  1304. Irp->IoStatus.Status = STATUS_INVALID_PARAMETER;
  1305. Irp->IoStatus.Information = 0;
  1306. //
  1307. // ClassPnp will handle completing the request.
  1308. //
  1309. return STATUS_INVALID_PARAMETER;
  1310. }
  1311. }
  1312. //
  1313. // Calculate number of pages in this transfer.
  1314. //
  1315. transferPages = ADDRESS_AND_SIZE_TO_SPAN_PAGES(MmGetMdlVirtualAddress(Irp->MdlAddress),
  1316. currentIrpStack->Parameters.Read.Length);
  1317. //
  1318. // Check if request length is greater than the maximum number of
  1319. // bytes that the hardware can transfer.
  1320. //
  1321. //
  1322. // Calculate number of pages in this transfer.
  1323. //
  1324. if (currentIrpStack->Parameters.Read.Length > maximumTransferLength ||
  1325. transferPages > adapterDescriptor->MaximumPhysicalPages) {
  1326. DebugPrint((2,
  1327. "TapeReadWriteVerification: Request greater than maximum\n"));
  1328. DebugPrint((2,
  1329. "TapeReadWriteVerification: Maximum is %lx\n",
  1330. maximumTransferLength));
  1331. DebugPrint((2,
  1332. "TapeReadWriteVerification: Byte count is %lx\n",
  1333. currentIrpStack->Parameters.Read.Length));
  1334. transferPages = adapterDescriptor->MaximumPhysicalPages - 1;
  1335. if (maximumTransferLength > transferPages << PAGE_SHIFT ) {
  1336. maximumTransferLength = transferPages << PAGE_SHIFT;
  1337. }
  1338. //
  1339. // Check that maximum transfer size is not zero.
  1340. //
  1341. if (maximumTransferLength == 0) {
  1342. maximumTransferLength = PAGE_SIZE;
  1343. }
  1344. //
  1345. // Ensure that this is reasonable, according to the current block size.
  1346. //
  1347. if (bytesPerSector) {
  1348. if (maximumTransferLength % bytesPerSector) {
  1349. ULONG tmpLength;
  1350. tmpLength = maximumTransferLength % bytesPerSector;
  1351. maximumTransferLength = maximumTransferLength - tmpLength;
  1352. }
  1353. }
  1354. //
  1355. // Mark IRP with status pending.
  1356. //
  1357. IoMarkIrpPending(Irp);
  1358. //
  1359. // Request greater than port driver maximum.
  1360. // Break up into smaller routines.
  1361. //
  1362. SplitTapeRequest(DeviceObject, Irp, maximumTransferLength);
  1363. return STATUS_PENDING;
  1364. }
  1365. //
  1366. // Build SRB and CDB for this IRP.
  1367. //
  1368. TapeReadWrite(DeviceObject, Irp);
  1369. IoMarkIrpPending(Irp);
  1370. IoCallDriver(commonExtension->LowerDeviceObject, Irp);
  1371. return STATUS_PENDING;
  1372. }
  1373. VOID
  1374. SplitTapeRequest(
  1375. IN PDEVICE_OBJECT Fdo,
  1376. IN PIRP Irp,
  1377. IN ULONG MaximumBytes
  1378. )
  1379. /*++
  1380. Routine Description:
  1381. Break request into smaller requests.
  1382. Each new request will be the maximum transfer
  1383. size that the port driver can handle or if it
  1384. is the final request, it may be the residual
  1385. size.
  1386. The number of IRPs required to process this
  1387. request is written in the current stack of
  1388. the original IRP. Then as each new IRP completes
  1389. the count in the original IRP is decremented.
  1390. When the count goes to zero, the original IRP
  1391. is completed.
  1392. Arguments:
  1393. DeviceObject - Pointer to the device object
  1394. Irp - Pointer to Irp
  1395. Return Value:
  1396. None.
  1397. --*/
  1398. {
  1399. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension;
  1400. PIO_STACK_LOCATION currentIrpStack = IoGetCurrentIrpStackLocation(Irp);
  1401. PIO_STACK_LOCATION nextIrpStack = IoGetNextIrpStackLocation(Irp);
  1402. ULONG irpCount;
  1403. ULONG transferByteCount = currentIrpStack->Parameters.Read.Length;
  1404. PSCSI_REQUEST_BLOCK srb;
  1405. LARGE_INTEGER startingOffset = currentIrpStack->Parameters.Read.ByteOffset;
  1406. ULONG dataLength = MaximumBytes;
  1407. PVOID dataBuffer = MmGetMdlVirtualAddress(Irp->MdlAddress);
  1408. LONG remainingIrps;
  1409. BOOLEAN completeOriginalIrp = FALSE;
  1410. NTSTATUS status;
  1411. ULONG i;
  1412. PAGED_CODE();
  1413. //
  1414. // Caluculate number of requests to break this IRP into.
  1415. //
  1416. irpCount = (transferByteCount + MaximumBytes - 1) / MaximumBytes;
  1417. DebugPrint((2,
  1418. "SplitTapeRequest: Requires %d IRPs\n", irpCount));
  1419. DebugPrint((2,
  1420. "SplitTapeRequest: Original IRP %p\n", Irp));
  1421. //
  1422. // If all partial transfers complete successfully then
  1423. // the status is already set up.
  1424. // Failing partial transfer IRP will set status to
  1425. // error and bytes transferred to 0 during IoCompletion.
  1426. //
  1427. Irp->IoStatus.Status = STATUS_SUCCESS;
  1428. //CEP Irp->IoStatus.Information = transferByteCount;
  1429. Irp->IoStatus.Information = 0;
  1430. //
  1431. // Save number of IRPs to complete count on current stack
  1432. // of original IRP.
  1433. //
  1434. nextIrpStack->Parameters.Others.Argument1 = ULongToPtr( irpCount );
  1435. for (i = 0; i < irpCount; i++) {
  1436. PIRP newIrp;
  1437. PIO_STACK_LOCATION newIrpStack;
  1438. //
  1439. // Allocate new IRP.
  1440. //
  1441. newIrp = IoAllocateIrp(Fdo->StackSize, FALSE);
  1442. if (newIrp == NULL) {
  1443. DebugPrint((1,
  1444. "SplitTapeRequest: Can't allocate Irp\n"));
  1445. //
  1446. // Decrement count of outstanding partial requests.
  1447. //
  1448. remainingIrps = InterlockedDecrement((PLONG)&nextIrpStack->Parameters.Others.Argument1);
  1449. //
  1450. // Check if any outstanding IRPs.
  1451. //
  1452. if (remainingIrps == 0) {
  1453. completeOriginalIrp = TRUE;
  1454. }
  1455. //
  1456. // Update original IRP with failing status.
  1457. //
  1458. Irp->IoStatus.Status = STATUS_INSUFFICIENT_RESOURCES;
  1459. Irp->IoStatus.Information = 0;
  1460. //
  1461. // Keep going with this request as outstanding partials
  1462. // may be in progress.
  1463. //
  1464. goto KeepGoing;
  1465. }
  1466. DebugPrint((2,
  1467. "SplitTapeRequest: New IRP %p\n", newIrp));
  1468. //
  1469. // Write MDL address to new IRP.
  1470. // In the port driver the SRB data length
  1471. // field is used as an offset into the MDL,
  1472. // so the same MDL can be used for each partial
  1473. // transfer. This saves having to build a new
  1474. // MDL for each partial transfer.
  1475. //
  1476. newIrp->MdlAddress = Irp->MdlAddress;
  1477. //
  1478. // At this point there is no current stack.
  1479. // IoSetNextIrpStackLocation will make the
  1480. // first stack location the current stack
  1481. // so that the SRB address can be written
  1482. // there.
  1483. //
  1484. IoSetNextIrpStackLocation(newIrp);
  1485. newIrpStack = IoGetCurrentIrpStackLocation(newIrp);
  1486. newIrpStack->MajorFunction = currentIrpStack->MajorFunction;
  1487. newIrpStack->Parameters.Read.Length = dataLength;
  1488. newIrpStack->Parameters.Read.ByteOffset = startingOffset;
  1489. newIrpStack->DeviceObject = Fdo;
  1490. //
  1491. // Build SRB and CDB.
  1492. //
  1493. TapeReadWrite(Fdo, newIrp);
  1494. //
  1495. // Adjust SRB for this partial transfer.
  1496. //
  1497. newIrpStack = IoGetNextIrpStackLocation(newIrp);
  1498. srb = newIrpStack->Parameters.Others.Argument1;
  1499. srb->DataBuffer = dataBuffer;
  1500. //
  1501. // Write original IRP address to new IRP.
  1502. //
  1503. newIrp->AssociatedIrp.MasterIrp = Irp;
  1504. //
  1505. // Set the completion routine to TapeIoCompleteAssociated.
  1506. //
  1507. IoSetCompletionRoutine(newIrp,
  1508. TapeIoCompleteAssociated,
  1509. srb,
  1510. TRUE,
  1511. TRUE,
  1512. TRUE);
  1513. //
  1514. // Call port driver with new request.
  1515. //
  1516. status = IoCallDriver(fdoExtension->CommonExtension.LowerDeviceObject, newIrp);
  1517. if (!NT_SUCCESS(status)) {
  1518. DebugPrint((1,
  1519. "SplitTapeRequest: IoCallDriver returned error\n"));
  1520. //
  1521. // Decrement count of outstanding partial requests.
  1522. //
  1523. remainingIrps = InterlockedDecrement((PLONG)&nextIrpStack->Parameters.Others.Argument1);
  1524. //
  1525. // Check if any outstanding IRPs.
  1526. //
  1527. if (remainingIrps == 0) {
  1528. completeOriginalIrp = TRUE;
  1529. }
  1530. //
  1531. // Update original IRP with failing status.
  1532. //
  1533. Irp->IoStatus.Status = status;
  1534. Irp->IoStatus.Information = 0;
  1535. //
  1536. // Deallocate this partial IRP.
  1537. //
  1538. IoFreeIrp(newIrp);
  1539. }
  1540. KeepGoing:
  1541. //
  1542. // Set up for next request.
  1543. //
  1544. dataBuffer = (PCHAR)dataBuffer + MaximumBytes;
  1545. transferByteCount -= MaximumBytes;
  1546. if (transferByteCount > MaximumBytes) {
  1547. dataLength = MaximumBytes;
  1548. } else {
  1549. dataLength = transferByteCount;
  1550. }
  1551. //
  1552. // Adjust disk byte offset.
  1553. //
  1554. startingOffset.QuadPart += MaximumBytes;
  1555. }
  1556. //
  1557. // Check if original IRP should be completed.
  1558. //
  1559. if (completeOriginalIrp) {
  1560. ClassReleaseRemoveLock(Fdo, Irp);
  1561. ClassCompleteRequest(Fdo, Irp, 0);
  1562. }
  1563. return;
  1564. } // end SplitTapeRequest()
  1565. VOID
  1566. TapeReadWrite(
  1567. IN PDEVICE_OBJECT Fdo,
  1568. IN PIRP Irp
  1569. )
  1570. /*++
  1571. Routine Description:
  1572. This routine builds up the given irp for a read or write request.
  1573. Arguments:
  1574. DeviceObject - Supplies the device object.
  1575. Irp - Supplies the I/O request packet.
  1576. Return Value:
  1577. None.
  1578. --*/
  1579. {
  1580. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension;
  1581. PTAPE_DATA tapeData = (PTAPE_DATA)(fdoExtension->CommonExtension.DriverData);
  1582. PTAPE_INIT_DATA_EX tapeInitData = &tapeData->TapeInitData;
  1583. PVOID minitapeExtension = (tapeData + 1);
  1584. PIO_STACK_LOCATION irpSp, nextSp;
  1585. PSCSI_REQUEST_BLOCK srb;
  1586. PCDB cdb;
  1587. ULONG transferBlocks;
  1588. PAGED_CODE();
  1589. //
  1590. // Allocate an Srb.
  1591. //
  1592. srb = ExAllocateFromNPagedLookasideList(&(fdoExtension->CommonExtension.SrbLookasideList));
  1593. srb->SrbFlags = 0;
  1594. irpSp = IoGetCurrentIrpStackLocation(Irp);
  1595. if (irpSp->MajorFunction == IRP_MJ_READ) {
  1596. srb->SrbFlags |= SRB_FLAGS_DATA_IN;
  1597. } else {
  1598. srb->SrbFlags |= SRB_FLAGS_DATA_OUT;
  1599. }
  1600. srb->Length = SCSI_REQUEST_BLOCK_SIZE;
  1601. srb->Function = SRB_FUNCTION_EXECUTE_SCSI;
  1602. srb->SrbStatus = 0;
  1603. srb->ScsiStatus = 0;
  1604. srb->QueueAction = SRB_SIMPLE_TAG_REQUEST;
  1605. srb->SrbFlags |= fdoExtension->SrbFlags;
  1606. srb->DataTransferLength = irpSp->Parameters.Read.Length;
  1607. srb->TimeOutValue = fdoExtension->TimeOutValue;
  1608. srb->DataBuffer = MmGetMdlVirtualAddress(Irp->MdlAddress);
  1609. srb->SenseInfoBuffer = fdoExtension->SenseData;
  1610. srb->SenseInfoBufferLength = SENSE_BUFFER_SIZE;
  1611. srb->NextSrb = NULL;
  1612. srb->OriginalRequest = Irp;
  1613. srb->SrbExtension = NULL;
  1614. srb->QueueSortKey = 0;
  1615. //
  1616. // Indicate that 6-byte CDB's will be used.
  1617. //
  1618. srb->CdbLength = CDB6GENERIC_LENGTH;
  1619. //
  1620. // Fill in CDB fields.
  1621. //
  1622. cdb = (PCDB)srb->Cdb;
  1623. //
  1624. // Zero CDB in SRB.
  1625. //
  1626. RtlZeroMemory(cdb, MAXIMUM_CDB_SIZE);
  1627. if (fdoExtension->DiskGeometry.BytesPerSector) {
  1628. //
  1629. // Since we are writing fixed block mode, normalize transfer count
  1630. // to number of blocks.
  1631. //
  1632. transferBlocks = irpSp->Parameters.Read.Length / fdoExtension->DiskGeometry.BytesPerSector;
  1633. //
  1634. // Tell the device that we are in fixed block mode.
  1635. //
  1636. cdb->CDB6READWRITETAPE.VendorSpecific = 1;
  1637. } else {
  1638. //
  1639. // Variable block mode transfer.
  1640. //
  1641. transferBlocks = irpSp->Parameters.Read.Length;
  1642. cdb->CDB6READWRITETAPE.VendorSpecific = 0;
  1643. }
  1644. //
  1645. // Set up transfer length
  1646. //
  1647. cdb->CDB6READWRITETAPE.TransferLenMSB = (UCHAR)((transferBlocks >> 16) & 0xff);
  1648. cdb->CDB6READWRITETAPE.TransferLen = (UCHAR)((transferBlocks >> 8) & 0xff);
  1649. cdb->CDB6READWRITETAPE.TransferLenLSB = (UCHAR)(transferBlocks & 0xff);
  1650. //
  1651. // Set transfer direction.
  1652. //
  1653. if (srb->SrbFlags & SRB_FLAGS_DATA_IN) {
  1654. DebugPrint((3,
  1655. "TapeReadWrite: Read Command\n"));
  1656. cdb->CDB6READWRITETAPE.OperationCode = SCSIOP_READ6;
  1657. } else {
  1658. DebugPrint((3,
  1659. "TapeReadWrite: Write Command\n"));
  1660. cdb->CDB6READWRITETAPE.OperationCode = SCSIOP_WRITE6;
  1661. }
  1662. nextSp = IoGetNextIrpStackLocation(Irp);
  1663. nextSp->MajorFunction = IRP_MJ_SCSI;
  1664. nextSp->Parameters.Scsi.Srb = srb;
  1665. irpSp->Parameters.Others.Argument4 = (PVOID) MAXIMUM_RETRIES;
  1666. IoSetCompletionRoutine(Irp,
  1667. ClassIoComplete,
  1668. srb,
  1669. TRUE,
  1670. TRUE,
  1671. FALSE);
  1672. if (tapeInitData->PreProcessReadWrite) {
  1673. //
  1674. // If the routine exists, call it. The miniclass driver will
  1675. // do whatever it needs to.
  1676. //
  1677. tapeInitData->PreProcessReadWrite(minitapeExtension,
  1678. NULL,
  1679. NULL,
  1680. srb,
  1681. 0,
  1682. 0,
  1683. NULL);
  1684. }
  1685. }
  1686. NTSTATUS
  1687. TapeIoCompleteAssociated(
  1688. IN PDEVICE_OBJECT Fdo,
  1689. IN PIRP Irp,
  1690. IN PVOID Context
  1691. )
  1692. /*++
  1693. Routine Description:
  1694. This routine executes when the port driver has completed a request.
  1695. It looks at the SRB status in the completing SRB and if not success
  1696. it checks for valid request sense buffer information. If valid, the
  1697. info is used to update status with more precise message of type of
  1698. error. This routine deallocates the SRB. This routine is used for
  1699. requests which were build by split request. After it has processed
  1700. the request it decrements the Irp count in the master Irp. If the
  1701. count goes to zero then the master Irp is completed.
  1702. Arguments:
  1703. DeviceObject - Supplies the device object which represents the logical
  1704. unit.
  1705. Irp - Supplies the Irp which has completed.
  1706. Context - Supplies a pointer to the SRB.
  1707. Return Value:
  1708. NT status
  1709. --*/
  1710. {
  1711. PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
  1712. PSCSI_REQUEST_BLOCK srb = Context;
  1713. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = Fdo->DeviceExtension;
  1714. PTAPE_DATA tapeData = (PTAPE_DATA)(fdoExtension->CommonExtension.DriverData);
  1715. LONG irpCount;
  1716. PIRP originalIrp = Irp->AssociatedIrp.MasterIrp;
  1717. NTSTATUS status;
  1718. //
  1719. // Check SRB status for success of completing request.
  1720. //
  1721. if (SRB_STATUS(srb->SrbStatus) != SRB_STATUS_SUCCESS) {
  1722. DebugPrint((2,
  1723. "TapeIoCompleteAssociated: IRP %p, SRB %p", Irp, srb));
  1724. //
  1725. // Release the queue if it is frozen.
  1726. //
  1727. if (srb->SrbStatus & SRB_STATUS_QUEUE_FROZEN) {
  1728. ClassReleaseQueue(Fdo);
  1729. }
  1730. ClassInterpretSenseInfo(Fdo,
  1731. srb,
  1732. irpStack->MajorFunction,
  1733. irpStack->MajorFunction == IRP_MJ_DEVICE_CONTROL ?
  1734. irpStack->Parameters.DeviceIoControl.IoControlCode : 0,
  1735. MAXIMUM_RETRIES - ((ULONG)(ULONG_PTR)irpStack->Parameters.Others.Argument4),
  1736. &status,
  1737. NULL);
  1738. //
  1739. // Return the highest error that occurs. This way warning take precedence
  1740. // over success and errors take precedence over warnings.
  1741. //
  1742. if ((ULONG) status > (ULONG) originalIrp->IoStatus.Status) {
  1743. //
  1744. // Ignore any requests which were flushed.
  1745. //
  1746. if (SRB_STATUS(srb->SrbStatus) != SRB_STATUS_REQUEST_FLUSHED) {
  1747. originalIrp->IoStatus.Status = status;
  1748. }
  1749. }
  1750. } // end if (SRB_STATUS(srb->SrbStatus) ...
  1751. ExInterlockedAddUlong((PULONG)&originalIrp->IoStatus.Information,
  1752. (ULONG)Irp->IoStatus.Information,
  1753. &tapeData->SplitRequestSpinLock );
  1754. //
  1755. // Return SRB to the slist
  1756. //
  1757. ExFreeToNPagedLookasideList((&fdoExtension->CommonExtension.SrbLookasideList), srb);
  1758. DebugPrint((2,
  1759. "TapeIoCompleteAssociated: Partial xfer IRP %p\n", Irp));
  1760. //
  1761. // Get next stack location. This original request is unused
  1762. // except to keep track of the completing partial IRPs so the
  1763. // stack location is valid.
  1764. //
  1765. irpStack = IoGetNextIrpStackLocation(originalIrp);
  1766. //
  1767. //
  1768. // If any of the asynchronous partial transfer IRPs fail with an error
  1769. // with an error then the original IRP will return 0 bytes transfered.
  1770. // This is an optimization for successful transfers.
  1771. //
  1772. if (NT_ERROR(originalIrp->IoStatus.Status)) {
  1773. originalIrp->IoStatus.Information = 0;
  1774. //
  1775. // Set the hard error if necessary.
  1776. //
  1777. if (IoIsErrorUserInduced(originalIrp->IoStatus.Status)) {
  1778. //
  1779. // Store DeviceObject for filesystem.
  1780. //
  1781. IoSetHardErrorOrVerifyDevice(originalIrp, Fdo);
  1782. }
  1783. }
  1784. //
  1785. // Decrement and get the count of remaining IRPs.
  1786. //
  1787. irpCount = InterlockedDecrement((PLONG)&irpStack->Parameters.Others.Argument1);
  1788. DebugPrint((2,
  1789. "TapeIoCompleteAssociated: Partial IRPs left %d\n",
  1790. irpCount));
  1791. if (irpCount == 0) {
  1792. #if DBG
  1793. irpStack = IoGetCurrentIrpStackLocation(originalIrp);
  1794. if (originalIrp->IoStatus.Information != irpStack->Parameters.Read.Length) {
  1795. DebugPrint((1,
  1796. "TapeIoCompleteAssociated: Short transfer. Request length: %lx, Return length: %lx, Status: %lx\n",
  1797. irpStack->Parameters.Read.Length,
  1798. originalIrp->IoStatus.Information,
  1799. originalIrp->IoStatus.Status));
  1800. }
  1801. #endif
  1802. //
  1803. // All partial IRPs have completed.
  1804. //
  1805. DebugPrint((2,
  1806. "TapeIoCompleteAssociated: All partial IRPs complete %p\n",
  1807. originalIrp));
  1808. //
  1809. // Release the lock and complete the original request.
  1810. //
  1811. ClassReleaseRemoveLock(Fdo, originalIrp);
  1812. ClassCompleteRequest(Fdo,originalIrp, IO_DISK_INCREMENT);
  1813. }
  1814. //
  1815. // Deallocate IRP and indicate the I/O system should not attempt any more
  1816. // processing.
  1817. //
  1818. IoFreeIrp(Irp);
  1819. return STATUS_MORE_PROCESSING_REQUIRED;
  1820. } // end TapeIoCompleteAssociated()
  1821. VOID
  1822. ScsiTapeFreeSrbBuffer(
  1823. IN OUT PSCSI_REQUEST_BLOCK Srb
  1824. )
  1825. /*++
  1826. Routine Description:
  1827. This routine frees an SRB buffer that was previously allocated with
  1828. 'TapeClassAllocateSrbBuffer'.
  1829. Arguments:
  1830. Srb - Supplies the SCSI request block.
  1831. Return Value:
  1832. None.
  1833. --*/
  1834. {
  1835. PAGED_CODE();
  1836. if (Srb->DataBuffer) {
  1837. ExFreePool(Srb->DataBuffer);
  1838. Srb->DataBuffer = NULL;
  1839. }
  1840. Srb->DataTransferLength = 0;
  1841. }
  1842. #define IOCTL_TAPE_OLD_SET_MEDIA_PARAMS CTL_CODE(IOCTL_TAPE_BASE, 0x0008, METHOD_BUFFERED, FILE_READ_ACCESS | FILE_WRITE_ACCESS)
  1843. NTSTATUS
  1844. TapeDeviceControl(
  1845. IN PDEVICE_OBJECT DeviceObject,
  1846. IN PIRP Irp
  1847. )
  1848. /*++
  1849. Routine Description:
  1850. This routine is the dispatcher for device control requests. It
  1851. looks at the IOCTL code and calls the appropriate tape device
  1852. routine.
  1853. Arguments:
  1854. DeviceObject
  1855. Irp - Request packet
  1856. Return Value:
  1857. --*/
  1858. {
  1859. PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation(Irp);
  1860. PFUNCTIONAL_DEVICE_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
  1861. PTAPE_DATA tapeData= (PTAPE_DATA) (fdoExtension->CommonExtension.DriverData);
  1862. PTAPE_INIT_DATA_EX tapeInitData = &tapeData->TapeInitData;
  1863. PVOID minitapeExtension = tapeData + 1;
  1864. NTSTATUS status = STATUS_SUCCESS;
  1865. TAPE_PROCESS_COMMAND_ROUTINE commandRoutine;
  1866. ULONG i;
  1867. PVOID commandExtension;
  1868. SCSI_REQUEST_BLOCK srb;
  1869. BOOLEAN writeToDevice;
  1870. TAPE_STATUS tStatus;
  1871. TAPE_STATUS LastError ;
  1872. ULONG retryFlags, numRetries;
  1873. TAPE_WMI_OPERATIONS WMIOperations;
  1874. TAPE_DRIVE_PROBLEM_TYPE DriveProblemType;
  1875. PVOID commandParameters;
  1876. ULONG ioControlCode;
  1877. PWMI_TAPE_PROBLEM_WARNING TapeDriveProblem = NULL;
  1878. ULONG timeoutDelta = 0;
  1879. ULONG dataTransferLength = 0;
  1880. PAGED_CODE();
  1881. DebugPrint((3,"ScsiTapeDeviceControl: Enter routine\n"));
  1882. Irp->IoStatus.Information = 0;
  1883. ioControlCode = irpStack->Parameters.DeviceIoControl.IoControlCode;
  1884. switch (ioControlCode) {
  1885. case IOCTL_STORAGE_GET_MEDIA_TYPES_EX: {
  1886. ULONG tmpSize;
  1887. //
  1888. // Validate version. Don't send this to a 4.0 miniclass driver.
  1889. //
  1890. if (tapeInitData->InitDataSize == sizeof(TAPE_INIT_DATA_EX)) {
  1891. //
  1892. // Validate buffer length.
  1893. //
  1894. tmpSize = (tapeInitData->MediaTypesSupported - 1) * sizeof(DEVICE_MEDIA_INFO);
  1895. if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
  1896. sizeof(GET_MEDIA_TYPES) + tmpSize) {
  1897. status = STATUS_INFO_LENGTH_MISMATCH;
  1898. break;
  1899. }
  1900. //
  1901. // Validate that the buffer is large enough for all media types.
  1902. //
  1903. commandRoutine = tapeInitData->TapeGetMediaTypes;
  1904. } else {
  1905. status = STATUS_NOT_IMPLEMENTED;
  1906. }
  1907. break;
  1908. }
  1909. case IOCTL_TAPE_GET_DRIVE_PARAMS:
  1910. //
  1911. // Validate buffer length.
  1912. //
  1913. if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
  1914. sizeof(TAPE_GET_DRIVE_PARAMETERS)) {
  1915. status = STATUS_INFO_LENGTH_MISMATCH;
  1916. break;
  1917. }
  1918. commandRoutine = tapeInitData->GetDriveParameters;
  1919. Irp->IoStatus.Information = sizeof(TAPE_GET_DRIVE_PARAMETERS);
  1920. break;
  1921. case IOCTL_TAPE_SET_DRIVE_PARAMS:
  1922. //
  1923. // Validate buffer length.
  1924. //
  1925. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  1926. sizeof(TAPE_SET_DRIVE_PARAMETERS)) {
  1927. status = STATUS_INFO_LENGTH_MISMATCH;
  1928. break;
  1929. }
  1930. commandRoutine = tapeInitData->SetDriveParameters;
  1931. break;
  1932. case IOCTL_TAPE_GET_MEDIA_PARAMS:
  1933. //
  1934. // Validate buffer length.
  1935. //
  1936. if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
  1937. sizeof(TAPE_GET_MEDIA_PARAMETERS)) {
  1938. status = STATUS_INFO_LENGTH_MISMATCH;
  1939. break;
  1940. }
  1941. commandRoutine = tapeInitData->GetMediaParameters;
  1942. Irp->IoStatus.Information = sizeof(TAPE_GET_MEDIA_PARAMETERS);
  1943. break;
  1944. //
  1945. // OLD_SET_XXX is here for legacy apps (defined READ/WRITE)
  1946. //
  1947. case IOCTL_TAPE_OLD_SET_MEDIA_PARAMS:
  1948. case IOCTL_TAPE_SET_MEDIA_PARAMS: {
  1949. PTAPE_SET_MEDIA_PARAMETERS tapeSetMediaParams = Irp->AssociatedIrp.SystemBuffer;
  1950. ULONG maxBytes1,maxBytes2,maxSize;
  1951. //
  1952. // Validate buffer length.
  1953. //
  1954. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  1955. sizeof(TAPE_SET_MEDIA_PARAMETERS)) {
  1956. status = STATUS_INFO_LENGTH_MISMATCH;
  1957. break;
  1958. }
  1959. //
  1960. // Ensure that Max. block size is less than the miniports
  1961. // reported MaximumTransferLength.
  1962. //
  1963. maxBytes1 = PAGE_SIZE * (fdoExtension->AdapterDescriptor->MaximumPhysicalPages - 1);
  1964. maxBytes2 = fdoExtension->AdapterDescriptor->MaximumTransferLength;
  1965. maxSize = (maxBytes1 > maxBytes2) ? maxBytes2 : maxBytes1;
  1966. if (tapeSetMediaParams->BlockSize > maxSize) {
  1967. DebugPrint((1,
  1968. "ScsiTapeDeviceControl: Attempted to set blocksize greater than miniport capabilities\n"));
  1969. DebugPrint((1,"BlockSize %x, Miniport Maximum %x\n",
  1970. tapeSetMediaParams->BlockSize,
  1971. maxSize));
  1972. status = STATUS_INVALID_PARAMETER;
  1973. break;
  1974. }
  1975. commandRoutine = tapeInitData->SetMediaParameters;
  1976. break;
  1977. }
  1978. case IOCTL_TAPE_CREATE_PARTITION:
  1979. //
  1980. // Validate buffer length.
  1981. //
  1982. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  1983. sizeof(TAPE_CREATE_PARTITION)) {
  1984. status = STATUS_INFO_LENGTH_MISMATCH;
  1985. break;
  1986. }
  1987. commandRoutine = tapeInitData->CreatePartition;
  1988. timeoutDelta = tapeData->SrbTimeoutDelta;
  1989. break;
  1990. case IOCTL_TAPE_ERASE:
  1991. //
  1992. // Validate buffer length.
  1993. //
  1994. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  1995. sizeof(TAPE_ERASE)) {
  1996. status = STATUS_INFO_LENGTH_MISMATCH;
  1997. break;
  1998. }
  1999. commandRoutine = tapeInitData->Erase;
  2000. timeoutDelta = tapeData->SrbTimeoutDelta;
  2001. break;
  2002. case IOCTL_TAPE_PREPARE:
  2003. //
  2004. // Validate buffer length.
  2005. //
  2006. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  2007. sizeof(TAPE_PREPARE)) {
  2008. status = STATUS_INFO_LENGTH_MISMATCH;
  2009. break;
  2010. }
  2011. commandRoutine = tapeInitData->Prepare;
  2012. timeoutDelta = tapeData->SrbTimeoutDelta;
  2013. break;
  2014. case IOCTL_TAPE_WRITE_MARKS:
  2015. //
  2016. // Validate buffer length.
  2017. //
  2018. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  2019. sizeof(TAPE_WRITE_MARKS)) {
  2020. status = STATUS_INFO_LENGTH_MISMATCH;
  2021. break;
  2022. }
  2023. commandRoutine = tapeInitData->WriteMarks;
  2024. timeoutDelta = tapeData->SrbTimeoutDelta;
  2025. break;
  2026. case IOCTL_TAPE_GET_POSITION:
  2027. //
  2028. // Validate buffer length.
  2029. //
  2030. if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
  2031. sizeof(TAPE_GET_POSITION)) {
  2032. status = STATUS_INFO_LENGTH_MISMATCH;
  2033. break;
  2034. }
  2035. commandRoutine = tapeInitData->GetPosition;
  2036. Irp->IoStatus.Information = sizeof(TAPE_GET_POSITION);
  2037. break;
  2038. case IOCTL_TAPE_SET_POSITION:
  2039. //
  2040. // Validate buffer length.
  2041. //
  2042. if (irpStack->Parameters.DeviceIoControl.InputBufferLength <
  2043. sizeof(TAPE_SET_POSITION)) {
  2044. status = STATUS_INFO_LENGTH_MISMATCH;
  2045. break;
  2046. }
  2047. commandRoutine = tapeInitData->SetPosition;
  2048. timeoutDelta = tapeData->SrbTimeoutDelta;
  2049. break;
  2050. case IOCTL_TAPE_GET_STATUS:
  2051. commandRoutine = tapeInitData->GetStatus;
  2052. break;
  2053. case IOCTL_STORAGE_PREDICT_FAILURE : {
  2054. //
  2055. // This IOCTL is for checking the tape drive
  2056. // to see if the device is having any problem.
  2057. //
  2058. PSTORAGE_PREDICT_FAILURE checkFailure;
  2059. checkFailure = (PSTORAGE_PREDICT_FAILURE)Irp->AssociatedIrp.SystemBuffer;
  2060. if (irpStack->Parameters.DeviceIoControl.OutputBufferLength <
  2061. sizeof(STORAGE_PREDICT_FAILURE)) {
  2062. status = STATUS_BUFFER_TOO_SMALL;
  2063. break;
  2064. }
  2065. //
  2066. // WMI routine to check for drive problems.
  2067. //
  2068. commandRoutine = tapeInitData->TapeWMIOperations;
  2069. if (commandRoutine == NULL) {
  2070. //
  2071. // WMI not supported by minidriver.
  2072. //
  2073. status = STATUS_WMI_NOT_SUPPORTED;
  2074. break;
  2075. }
  2076. TapeDriveProblem = ExAllocatePool(NonPagedPool,
  2077. sizeof(WMI_TAPE_PROBLEM_WARNING));
  2078. if (TapeDriveProblem == NULL) {
  2079. status = STATUS_NO_MEMORY;
  2080. break;
  2081. }
  2082. //
  2083. // Call the WMI method to check for drive problem.
  2084. //
  2085. RtlZeroMemory(TapeDriveProblem, sizeof(WMI_TAPE_PROBLEM_WARNING));
  2086. TapeDriveProblem->DriveProblemType = TapeDriveProblemNone;
  2087. WMIOperations.Method = TAPE_CHECK_FOR_DRIVE_PROBLEM;
  2088. WMIOperations.DataBufferSize = sizeof(WMI_TAPE_PROBLEM_WARNING);
  2089. WMIOperations.DataBuffer = (PVOID)TapeDriveProblem;
  2090. break;
  2091. }
  2092. default:
  2093. //
  2094. // Pass the request to the common device control routine.
  2095. //
  2096. return ClassDeviceControl(DeviceObject, Irp);
  2097. } // end switch()
  2098. if (!NT_SUCCESS(status)) {
  2099. Irp->IoStatus.Information = 0;
  2100. Irp->IoStatus.Status = status;
  2101. ClassReleaseRemoveLock(DeviceObject, Irp);
  2102. ClassCompleteRequest(DeviceObject,Irp, IO_NO_INCREMENT);
  2103. return status;
  2104. }
  2105. if (tapeInitData->CommandExtensionSize) {
  2106. commandExtension = ExAllocatePool(NonPagedPool,
  2107. tapeInitData->CommandExtensionSize);
  2108. } else {
  2109. commandExtension = NULL;
  2110. }
  2111. if (ioControlCode == IOCTL_STORAGE_PREDICT_FAILURE) {
  2112. commandParameters = (PVOID)&WMIOperations;
  2113. } else {
  2114. commandParameters = Irp->AssociatedIrp.SystemBuffer;
  2115. }
  2116. RtlZeroMemory(&srb, sizeof(SCSI_REQUEST_BLOCK));
  2117. LastError = TAPE_STATUS_SUCCESS ;
  2118. for (i = 0; ; i++) {
  2119. srb.TimeOutValue = fdoExtension->TimeOutValue;
  2120. srb.SrbFlags = 0;
  2121. retryFlags = 0;
  2122. tStatus = commandRoutine(minitapeExtension, commandExtension,
  2123. commandParameters, &srb, i,
  2124. LastError, &retryFlags);
  2125. //
  2126. // Add Srb Timeout delta to the current timeout value
  2127. // set in the SRB.
  2128. //
  2129. srb.TimeOutValue += timeoutDelta;
  2130. LastError = TAPE_STATUS_SUCCESS ;
  2131. numRetries = retryFlags&TAPE_RETRY_MASK;
  2132. if (tStatus == TAPE_STATUS_CHECK_TEST_UNIT_READY) {
  2133. PCDB cdb = (PCDB)srb.Cdb;
  2134. //
  2135. // Prepare SCSI command (CDB)
  2136. //
  2137. TapeClassZeroMemory(srb.Cdb, MAXIMUM_CDB_SIZE);
  2138. srb.CdbLength = CDB6GENERIC_LENGTH;
  2139. cdb->CDB6GENERIC.OperationCode = SCSIOP_TEST_UNIT_READY;
  2140. srb.DataTransferLength = 0 ;
  2141. DebugPrint((3,"Test Unit Ready\n"));
  2142. } else if (tStatus == TAPE_STATUS_CALLBACK) {
  2143. LastError = TAPE_STATUS_CALLBACK ;
  2144. continue;
  2145. } else if (tStatus != TAPE_STATUS_SEND_SRB_AND_CALLBACK) {
  2146. break;
  2147. }
  2148. if (srb.DataBuffer && !srb.DataTransferLength) {
  2149. ScsiTapeFreeSrbBuffer(&srb);
  2150. }
  2151. if (srb.DataBuffer && (srb.SrbFlags&SRB_FLAGS_DATA_OUT)) {
  2152. writeToDevice = TRUE;
  2153. } else {
  2154. writeToDevice = FALSE;
  2155. }
  2156. dataTransferLength = srb.DataTransferLength;
  2157. for (;;) {
  2158. status = ClassSendSrbSynchronous(DeviceObject, &srb,
  2159. srb.DataBuffer,
  2160. srb.DataTransferLength,
  2161. writeToDevice);
  2162. if (NT_SUCCESS(status) ||
  2163. (status == STATUS_DATA_OVERRUN)) {
  2164. if (status == STATUS_DATA_OVERRUN) {
  2165. if ((srb.DataTransferLength) <= dataTransferLength) {
  2166. DebugPrint((1, "DataUnderRun reported as overrun\n"));
  2167. status = STATUS_SUCCESS;
  2168. break;
  2169. }
  2170. } else {
  2171. break;
  2172. }
  2173. }
  2174. if ((status == STATUS_BUS_RESET) ||
  2175. (status == STATUS_IO_TIMEOUT)) {
  2176. //
  2177. // Timeout value for the command probably wasn't sufficient.
  2178. // Update timeout delta from the registry
  2179. //
  2180. tapeData->SrbTimeoutDelta = GetTimeoutDeltaFromRegistry(fdoExtension->LowerPdo);
  2181. if ((tapeData->SrbTimeoutDelta) == 0) {
  2182. tapeData->SrbTimeoutDelta = fdoExtension->TimeOutValue;
  2183. timeoutDelta = tapeData->SrbTimeoutDelta;
  2184. srb.TimeOutValue += timeoutDelta;
  2185. }
  2186. }
  2187. if (numRetries == 0) {
  2188. if (retryFlags&RETURN_ERRORS) {
  2189. ScsiTapeNtStatusToTapeStatus(status, &LastError) ;
  2190. break ;
  2191. }
  2192. if (retryFlags&IGNORE_ERRORS) {
  2193. break;
  2194. }
  2195. if (commandExtension) {
  2196. ExFreePool(commandExtension);
  2197. }
  2198. ScsiTapeFreeSrbBuffer(&srb);
  2199. if (TapeDriveProblem) {
  2200. ExFreePool(TapeDriveProblem);
  2201. }
  2202. Irp->IoStatus.Information = 0;
  2203. Irp->IoStatus.Status = status;
  2204. ClassReleaseRemoveLock(DeviceObject, Irp);
  2205. ClassCompleteRequest(DeviceObject, Irp, IO_NO_INCREMENT);
  2206. return status;
  2207. }
  2208. numRetries--;
  2209. }
  2210. }
  2211. ScsiTapeFreeSrbBuffer(&srb);
  2212. if (commandExtension) {
  2213. ExFreePool(commandExtension);
  2214. }
  2215. if (!ScsiTapeTapeStatusToNtStatus(tStatus, &status)) {
  2216. status = STATUS_IO_DEVICE_ERROR;
  2217. }
  2218. if (NT_SUCCESS(status)) {
  2219. PTAPE_GET_MEDIA_PARAMETERS tapeGetMediaParams;
  2220. PTAPE_SET_MEDIA_PARAMETERS tapeSetMediaParams;
  2221. PTAPE_GET_DRIVE_PARAMETERS tapeGetDriveParams;
  2222. PGET_MEDIA_TYPES tapeGetMediaTypes;
  2223. ULONG maxBytes1,maxBytes2,maxSize;
  2224. switch (irpStack->Parameters.DeviceIoControl.IoControlCode) {
  2225. case IOCTL_STORAGE_GET_MEDIA_TYPES_EX:
  2226. tapeGetMediaTypes = Irp->AssociatedIrp.SystemBuffer;
  2227. //
  2228. // Set information field based on the returned number of mediaTypes
  2229. //
  2230. Irp->IoStatus.Information = sizeof(GET_MEDIA_TYPES);
  2231. Irp->IoStatus.Information += ((tapeGetMediaTypes->MediaInfoCount - 1) * sizeof(DEVICE_MEDIA_INFO));
  2232. DebugPrint((1,"Tape: GET_MEDIA_TYPES - Information %x\n", Irp->IoStatus.Information));
  2233. break;
  2234. case IOCTL_TAPE_GET_MEDIA_PARAMS:
  2235. tapeGetMediaParams = Irp->AssociatedIrp.SystemBuffer;
  2236. //
  2237. // Check if block size has been initialized.
  2238. //
  2239. if (fdoExtension->DiskGeometry.BytesPerSector ==
  2240. UNDEFINED_BLOCK_SIZE) {
  2241. //
  2242. // Set the block size in the device object.
  2243. //
  2244. fdoExtension->DiskGeometry.BytesPerSector =
  2245. tapeGetMediaParams->BlockSize;
  2246. }
  2247. break;
  2248. case IOCTL_TAPE_OLD_SET_MEDIA_PARAMS:
  2249. case IOCTL_TAPE_SET_MEDIA_PARAMS:
  2250. tapeSetMediaParams = Irp->AssociatedIrp.SystemBuffer;
  2251. //
  2252. // Set the block size in the device object.
  2253. //
  2254. fdoExtension->DiskGeometry.BytesPerSector =
  2255. tapeSetMediaParams->BlockSize;
  2256. break;
  2257. case IOCTL_TAPE_GET_DRIVE_PARAMS:
  2258. tapeGetDriveParams = Irp->AssociatedIrp.SystemBuffer;
  2259. //
  2260. // Ensure that Max. block size is less than the miniports
  2261. // reported MaximumTransferLength.
  2262. //
  2263. maxBytes1 = PAGE_SIZE * (fdoExtension->AdapterDescriptor->MaximumPhysicalPages - 1);
  2264. maxBytes2 = fdoExtension->AdapterDescriptor->MaximumTransferLength;
  2265. maxSize = (maxBytes1 > maxBytes2) ? maxBytes2 : maxBytes1;
  2266. if (tapeGetDriveParams->MaximumBlockSize > maxSize) {
  2267. tapeGetDriveParams->MaximumBlockSize = maxSize;
  2268. DebugPrint((1,
  2269. "ScsiTapeDeviceControl: Resetting max. tape block size to %x\n",
  2270. tapeGetDriveParams->MaximumBlockSize));
  2271. }
  2272. //
  2273. // Ensure that the default block size is less than or equal
  2274. // to maximum block size.
  2275. //
  2276. if ((tapeGetDriveParams->DefaultBlockSize) >
  2277. (tapeGetDriveParams->MaximumBlockSize)) {
  2278. tapeGetDriveParams->DefaultBlockSize =
  2279. tapeGetDriveParams->MaximumBlockSize;
  2280. }
  2281. //
  2282. // Ensure the blocksize we return are power of 2
  2283. //
  2284. UPDATE_BLOCK_SIZE(tapeGetDriveParams->DefaultBlockSize);
  2285. UPDATE_BLOCK_SIZE(tapeGetDriveParams->MaximumBlockSize);
  2286. UPDATE_BLOCK_SIZE(tapeGetDriveParams->MinimumBlockSize);
  2287. break;
  2288. case IOCTL_STORAGE_PREDICT_FAILURE: {
  2289. PSTORAGE_PREDICT_FAILURE checkFailure;
  2290. WMI_TAPE_PROBLEM_WARNING TapeProblemWarning;
  2291. GUID TapeProblemWarningGuid = WMI_TAPE_PROBLEM_WARNING_GUID;
  2292. checkFailure = (PSTORAGE_PREDICT_FAILURE)Irp->AssociatedIrp.SystemBuffer;
  2293. //
  2294. // We don't want classpnp to notify WMI if the drive is having
  2295. // problems or not. We'll handle that here. So, set
  2296. // PredictFailure to 0. Then, classpnp will not process
  2297. // it further.
  2298. //
  2299. checkFailure->PredictFailure = 0;
  2300. //
  2301. // If the drive is reporting problem, we'll notify WMI
  2302. //
  2303. if (TapeDriveProblem->DriveProblemType !=
  2304. TapeDriveProblemNone) {
  2305. DebugPrint((1,
  2306. "IOCTL_STORAGE_PREDICT_FAILURE : Tape drive %p",
  2307. " is experiencing problem %d\n",
  2308. DeviceObject,
  2309. TapeDriveProblem->DriveProblemType));
  2310. ClassWmiFireEvent(DeviceObject,
  2311. &TapeProblemWarningGuid,
  2312. 0,
  2313. sizeof(WMI_TAPE_PROBLEM_WARNING),
  2314. (PUCHAR)TapeDriveProblem);
  2315. //
  2316. // ISSUE 02/28/2000 - nramas : We should decide whether
  2317. // or not we need to log an event in addition to
  2318. // firing a WMI event.
  2319. //
  2320. }
  2321. Irp->IoStatus.Information = sizeof(STORAGE_PREDICT_FAILURE);
  2322. //
  2323. // Free the buffer allocated for tape problem
  2324. // warning data
  2325. //
  2326. ExFreePool(TapeDriveProblem);
  2327. break;
  2328. }
  2329. case IOCTL_TAPE_ERASE: {
  2330. //
  2331. // Notify that the media has been successfully erased
  2332. //
  2333. TARGET_DEVICE_CUSTOM_NOTIFICATION NotificationStructure[2];
  2334. NotificationStructure[0].Event = GUID_IO_TAPE_ERASE;
  2335. NotificationStructure[0].Version = 1;
  2336. NotificationStructure[0].Size = sizeof(TARGET_DEVICE_CUSTOM_NOTIFICATION) +
  2337. sizeof(ULONG) - sizeof(UCHAR);
  2338. NotificationStructure[0].FileObject = NULL;
  2339. NotificationStructure[0].NameBufferOffset = -1;
  2340. //
  2341. // Increasing Index for this event
  2342. //
  2343. *((PULONG) (&(NotificationStructure[0].CustomDataBuffer[0]))) = 0;
  2344. IoReportTargetDeviceChangeAsynchronous(fdoExtension->LowerPdo,
  2345. &NotificationStructure[0],
  2346. NULL,
  2347. NULL);
  2348. break;
  2349. }
  2350. }
  2351. } else {
  2352. Irp->IoStatus.Information = 0;
  2353. if (TapeDriveProblem) {
  2354. ExFreePool(TapeDriveProblem);
  2355. }
  2356. }
  2357. Irp->IoStatus.Status = status;
  2358. ClassReleaseRemoveLock(DeviceObject, Irp);
  2359. ClassCompleteRequest(DeviceObject,Irp, 2);
  2360. return status;
  2361. } // end ScsiScsiTapeDeviceControl()
  2362. BOOLEAN
  2363. TapeClassAllocateSrbBuffer(
  2364. IN OUT PSCSI_REQUEST_BLOCK Srb,
  2365. IN ULONG SrbBufferSize
  2366. )
  2367. /*++
  2368. Routine Description:
  2369. This routine allocates a 'DataBuffer' for the given SRB of the given
  2370. size.
  2371. Arguments:
  2372. Srb - Supplies the SCSI request block.
  2373. SrbBufferSize - Supplies the desired 'DataBuffer' size.
  2374. Return Value:
  2375. FALSE - The allocation failed.
  2376. TRUE - The allocation succeeded.
  2377. --*/
  2378. {
  2379. PVOID p;
  2380. PAGED_CODE();
  2381. if (Srb->DataBuffer) {
  2382. ExFreePool(Srb->DataBuffer);
  2383. }
  2384. p = ExAllocatePool(NonPagedPoolCacheAligned, SrbBufferSize);
  2385. if (!p) {
  2386. Srb->DataBuffer = NULL;
  2387. Srb->DataTransferLength = 0;
  2388. return FALSE;
  2389. }
  2390. Srb->DataBuffer = p;
  2391. Srb->DataTransferLength = SrbBufferSize;
  2392. RtlZeroMemory(p, SrbBufferSize);
  2393. return TRUE;
  2394. }
  2395. VOID
  2396. TapeClassZeroMemory(
  2397. IN OUT PVOID Buffer,
  2398. IN ULONG BufferSize
  2399. )
  2400. /*++
  2401. Routine Description:
  2402. This routine zeroes the given memory.
  2403. Arguments:
  2404. Buffer - Supplies the buffer.
  2405. BufferSize - Supplies the buffer size.
  2406. Return Value:
  2407. None.
  2408. --*/
  2409. {
  2410. PAGED_CODE();
  2411. RtlZeroMemory(Buffer, BufferSize);
  2412. }
  2413. ULONG
  2414. TapeClassCompareMemory(
  2415. IN OUT PVOID Source1,
  2416. IN OUT PVOID Source2,
  2417. IN ULONG Length
  2418. )
  2419. /*++
  2420. Routine Description:
  2421. This routine compares the two memory buffers and returns the number
  2422. of bytes that are equivalent.
  2423. Arguments:
  2424. Source1 - Supplies the first memory buffer.
  2425. Source2 - Supplies the second memory buffer.
  2426. Length - Supplies the number of bytes to be compared.
  2427. Return Value:
  2428. The number of bytes that compared as equal.
  2429. --*/
  2430. {
  2431. PAGED_CODE();
  2432. return (ULONG)RtlCompareMemory(Source1, Source2, Length);
  2433. }
  2434. LARGE_INTEGER
  2435. TapeClassLiDiv(
  2436. IN LARGE_INTEGER Dividend,
  2437. IN LARGE_INTEGER Divisor
  2438. )
  2439. {
  2440. LARGE_INTEGER li;
  2441. PAGED_CODE();
  2442. li.QuadPart = Dividend.QuadPart / Divisor.QuadPart;
  2443. return li;
  2444. }
  2445. ULONG
  2446. GetTimeoutDeltaFromRegistry(
  2447. IN PDEVICE_OBJECT LowerPdo
  2448. )
  2449. {
  2450. ULONG srbTimeoutDelta = 0;
  2451. HANDLE deviceKey;
  2452. NTSTATUS status;
  2453. RTL_QUERY_REGISTRY_TABLE queryTable[2];
  2454. OBJECT_ATTRIBUTES objectAttributes;
  2455. PAGED_CODE();
  2456. #define SRB_TIMEOUT_DELTA (L"SrbTimeoutDelta")
  2457. ASSERT(LowerPdo != NULL);
  2458. //
  2459. // Open a handle to the device node
  2460. //
  2461. status = IoOpenDeviceRegistryKey(LowerPdo,
  2462. PLUGPLAY_REGKEY_DEVICE,
  2463. KEY_QUERY_VALUE,
  2464. &deviceKey);
  2465. if (!NT_SUCCESS(status)) {
  2466. DebugPrint((1,
  2467. "IoOpenDeviceRegistryKey Failed in GetTimeoutDeltaFromRegistry : %x\n",
  2468. status));
  2469. return 0;
  2470. }
  2471. RtlZeroMemory(&queryTable[0], sizeof(queryTable));
  2472. queryTable[0].Name = SRB_TIMEOUT_DELTA;
  2473. queryTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
  2474. queryTable[0].EntryContext = &srbTimeoutDelta;
  2475. queryTable[0].DefaultType = REG_DWORD;
  2476. queryTable[0].DefaultData = NULL;
  2477. queryTable[0].DefaultLength = 0;
  2478. status = RtlQueryRegistryValues(RTL_REGISTRY_HANDLE,
  2479. (PWSTR)deviceKey,
  2480. queryTable,
  2481. NULL,
  2482. NULL);
  2483. if (!NT_SUCCESS(status)) {
  2484. DebugPrint((1,
  2485. "RtlQueryRegistryValue failed for SrbTimeoutDelta : %x\n",
  2486. status));
  2487. srbTimeoutDelta = 0;
  2488. }
  2489. ZwClose(deviceKey);
  2490. DebugPrint((3, "SrbTimeoutDelta read from registry %x\n",
  2491. srbTimeoutDelta));
  2492. return srbTimeoutDelta;
  2493. }
  2494. #if DBG
  2495. #define TAPE_DEBUG_PRINT_BUFF_LEN 128
  2496. ULONG TapeClassDebug = 0;
  2497. UCHAR TapeClassBuffer[TAPE_DEBUG_PRINT_BUFF_LEN];
  2498. VOID
  2499. TapeDebugPrint(
  2500. ULONG DebugPrintLevel,
  2501. PCCHAR DebugMessage,
  2502. ...
  2503. )
  2504. /*++
  2505. Routine Description:
  2506. Debug print for all Tape minidrivers
  2507. Arguments:
  2508. Debug print level between 0 and 3, with 3 being the most verbose.
  2509. Return Value:
  2510. None
  2511. --*/
  2512. {
  2513. va_list ap;
  2514. va_start(ap, DebugMessage);
  2515. if ((DebugPrintLevel <= (TapeClassDebug & 0x0000ffff)) ||
  2516. ((1 << (DebugPrintLevel + 15)) & TapeClassDebug)) {
  2517. _vsnprintf(TapeClassBuffer, TAPE_DEBUG_PRINT_BUFF_LEN,
  2518. DebugMessage, ap);
  2519. DbgPrintEx(DPFLTR_TAPE_ID, DPFLTR_INFO_LEVEL, TapeClassBuffer);
  2520. }
  2521. va_end(ap);
  2522. } // end TapeDebugPrint()
  2523. #else
  2524. //
  2525. // TapeDebugPrint stub
  2526. //
  2527. VOID
  2528. TapeDebugPrint(
  2529. ULONG DebugPrintLevel,
  2530. PCCHAR DebugMessage,
  2531. ...
  2532. )
  2533. {
  2534. }
  2535. #endif