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.

277 lines
7.0 KiB

  1. #include <mpio.h>
  2. VOID
  3. MPIOInitQueue(
  4. IN PMP_QUEUE Queue,
  5. IN ULONG QueueTag
  6. )
  7. {
  8. //
  9. // Initialize this queue's spinlock.
  10. //
  11. KeInitializeSpinLock(&Queue->SpinLock);
  12. //
  13. // Init the list entry.
  14. //
  15. InitializeListHead(&Queue->ListEntry);
  16. //
  17. // Set the number of items.
  18. //
  19. Queue->QueuedItems = 0;
  20. //
  21. // Set the queue marker.
  22. //
  23. Queue->QueueIndicator = QueueTag;
  24. }
  25. VOID
  26. MPIOInsertQueue(
  27. IN PMP_QUEUE Queue,
  28. IN PMPQUEUE_ENTRY QueueEntry
  29. )
  30. {
  31. KIRQL irql;
  32. KeAcquireSpinLock(&Queue->SpinLock, &irql);
  33. //
  34. // Insert the entry on the queue.
  35. //
  36. InsertTailList(&Queue->ListEntry,
  37. &QueueEntry->ListEntry);
  38. //
  39. // Indicate the additonal item.
  40. //
  41. Queue->QueuedItems++;
  42. KeReleaseSpinLock(&Queue->SpinLock, irql);
  43. }
  44. PMPQUEUE_ENTRY
  45. MPIORemoveQueue(
  46. IN PMP_QUEUE Queue
  47. )
  48. {
  49. KIRQL irql;
  50. PLIST_ENTRY entry;
  51. KeAcquireSpinLock(&Queue->SpinLock, &irql);
  52. //
  53. // Get the next item on the list.
  54. //
  55. entry = RemoveHeadList(&Queue->ListEntry);
  56. //
  57. // Indicate one less.
  58. //
  59. Queue->QueuedItems--;
  60. KeReleaseSpinLock(&Queue->SpinLock, irql);
  61. return CONTAINING_RECORD(entry, MPQUEUE_ENTRY, ListEntry);
  62. }
  63. NTSTATUS
  64. MPIOIssueQueuedRequests(
  65. IN PREAL_DEV_INFO TargetInfo,
  66. IN PMP_QUEUE Queue,
  67. IN ULONG State,
  68. IN PULONG RequestCount
  69. )
  70. /*++
  71. Routine Description:
  72. This routine runs Queue, extracts the requests and issues them
  73. to DeviceObject.
  74. It is used to handle the various failure queue submissions.
  75. It assumes that the requests are ready to submit, context values
  76. updated, and all targetInfo,diskExtension, deviceExtension values to
  77. be correct.
  78. Arguments:
  79. DeviceObject - The MPDev Object to which the request is to be sent.
  80. Queue - The failure queue to be drained.
  81. CompletionRoutine - The appropriate completion routine to be set-up.
  82. Return Value:
  83. NTSTATUS
  84. --*/
  85. {
  86. PMPQUEUE_ENTRY queueEntry;
  87. PIRP irp;
  88. PIO_STACK_LOCATION irpStack;
  89. NTSTATUS status = STATUS_SUCCESS;
  90. PMPIO_CONTEXT context;
  91. ULONG issued = 0;
  92. ULONG initialCount = Queue->QueuedItems;
  93. //
  94. // See if anything is actually in the queue.
  95. //
  96. if (Queue->QueuedItems == 0) {
  97. MPDebugPrint((0,
  98. "MPIOIssueQueuedRequests: Asked to issue Zero requests\n"));
  99. DbgBreakPoint();
  100. *RequestCount = 0;
  101. return STATUS_SUCCESS;
  102. }
  103. MPDebugPrint((0,
  104. "IssueQueuedRequests: Handling (%x) in State (%x)\n",
  105. Queue,
  106. State));
  107. do {
  108. //
  109. // Get the next entry in the queue.
  110. // This also dec's QueuedItems.
  111. //
  112. queueEntry = MPIORemoveQueue(Queue);
  113. if (queueEntry) {
  114. //
  115. // Extract the irp, our stack location, and the
  116. // associated context.
  117. //
  118. irp = queueEntry->Irp;
  119. irpStack = IoGetCurrentIrpStackLocation(irp);
  120. context = irpStack->Parameters.Others.Argument4;
  121. //
  122. // Update the state, so that the completion routine
  123. // can figure out when this was sent.
  124. //
  125. context->CurrentState = State;
  126. context->ReIssued = TRUE;
  127. //
  128. // Hack out the Dsm Completion info.
  129. // Otherwise, we call them twice for this request.
  130. // BUGBUG: Or call set completion again.
  131. //
  132. context->DsmCompletion.DsmCompletionRoutine = NULL;
  133. context->DsmCompletion.DsmContext = NULL;
  134. context->TargetInfo = TargetInfo;
  135. InterlockedIncrement(&TargetInfo->Requests);
  136. //
  137. // Determine whether this contains an Srb or not.
  138. //
  139. if (irpStack->MajorFunction == IRP_MJ_SCSI) {
  140. //
  141. // Need to rebuild the Srb.
  142. //
  143. RtlCopyMemory(irpStack->Parameters.Scsi.Srb,
  144. &context->Srb,
  145. sizeof(SCSI_REQUEST_BLOCK));
  146. }
  147. //
  148. // Re-do status and information.
  149. //
  150. irp->IoStatus.Status = 0;
  151. irp->IoStatus.Information = 0;
  152. //
  153. // Rebuild port's stack location.
  154. //
  155. IoCopyCurrentIrpStackLocationToNext(irp);
  156. //
  157. // Set the appropriate completion routine.
  158. //
  159. IoSetCompletionRoutine(irp,
  160. MPPdoGlobalCompletion,
  161. context,
  162. TRUE,
  163. TRUE,
  164. TRUE);
  165. //
  166. // Send it to "DeviceObject".
  167. //
  168. status = IoCallDriver(TargetInfo->DevFilter, irp);
  169. issued++;
  170. //
  171. // Update the caller's number of requests.
  172. // TODO: This is updating the value that is already set.
  173. // Either remove the parameter and this count update, or
  174. // have callers use a different variable (not the diskExtension->XXXCount);
  175. //
  176. InterlockedIncrement(RequestCount);
  177. //
  178. // Status should be pending or success.
  179. //
  180. if ((status != STATUS_PENDING) &&
  181. (status != STATUS_SUCCESS)){
  182. //
  183. // Completion routine will handle this, but make a
  184. // note of the failure. TODO firgure out a better way to
  185. // do this.
  186. //
  187. MPDebugPrint((0,
  188. "MPIOIssueQueuedRequests: Irp (%x) sent to (%x) - Status (%x)\n",
  189. irp,
  190. TargetInfo->DevFilter,
  191. status));
  192. //
  193. // LOG
  194. //
  195. ASSERT(status == STATUS_SUCCESS);
  196. }
  197. //
  198. // Free the allocaiton.
  199. // NOTE: These queueEntries should be on a LookasideList.
  200. //
  201. ExFreePool(queueEntry);
  202. } else {
  203. //
  204. // Ensure that the queue is actually empty.
  205. //
  206. ASSERT(Queue->QueuedItems == 0);
  207. }
  208. //
  209. // BUGBUG: There is no throttling mechanism here. It would be easy
  210. // to overload the underlying adapter in some cases.
  211. //
  212. // NOTE: It may be better to re-categorize all the requests instead of
  213. // just sending all to the "newPath" that the DSM indicated.
  214. //
  215. // Need to note this behaviour to DSM authors!
  216. //
  217. } while (Queue->QueuedItems);
  218. MPDebugPrint((0,
  219. "IssueQueuedRequests: Issued (%x) out of (%x)\n",
  220. issued,
  221. initialCount));
  222. return status;
  223. }