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.

464 lines
11 KiB

  1. /*++
  2. Copyright (c) 1989-2000 Microsoft Corporation
  3. Module Name:
  4. FspDisp.c
  5. Abstract:
  6. This module implements the main dispatch procedure/thread for the Fat
  7. Fsp
  8. // @@BEGIN_DDKSPLIT
  9. Author:
  10. Gary Kimura [GaryKi] 28-Dec-1989
  11. Revision History:
  12. // @@END_DDKSPLIT
  13. --*/
  14. #include "FatProcs.h"
  15. //
  16. // Internal support routine, spinlock wrapper.
  17. //
  18. PVOID
  19. FatRemoveOverflowEntry (
  20. IN PVOLUME_DEVICE_OBJECT VolDo
  21. );
  22. //
  23. // Define our local debug trace level
  24. //
  25. #define Dbg (DEBUG_TRACE_FSP_DISPATCHER)
  26. #ifdef ALLOC_PRAGMA
  27. #pragma alloc_text(PAGE, FatFspDispatch)
  28. #endif
  29. VOID
  30. FatFspDispatch (
  31. IN PVOID Context
  32. )
  33. /*++
  34. Routine Description:
  35. This is the main FSP thread routine that is executed to receive
  36. and dispatch IRP requests. Each FSP thread begins its execution here.
  37. There is one thread created at system initialization time and subsequent
  38. threads created as needed.
  39. Arguments:
  40. Context - Supplies the thread id.
  41. Return Value:
  42. None - This routine never exits
  43. --*/
  44. {
  45. NTSTATUS Status;
  46. PIRP Irp;
  47. PIRP_CONTEXT IrpContext;
  48. PIO_STACK_LOCATION IrpSp;
  49. PVOLUME_DEVICE_OBJECT VolDo;
  50. IrpContext = (PIRP_CONTEXT)Context;
  51. Irp = IrpContext->OriginatingIrp;
  52. IrpSp = IoGetCurrentIrpStackLocation( Irp );
  53. //
  54. // Now because we are the Fsp we will force the IrpContext to
  55. // indicate true on Wait.
  56. //
  57. SetFlag(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT | IRP_CONTEXT_FLAG_IN_FSP);
  58. //
  59. // If this request has an associated volume device object, remember it.
  60. //
  61. if ( IrpSp->FileObject != NULL ) {
  62. VolDo = CONTAINING_RECORD( IrpSp->DeviceObject,
  63. VOLUME_DEVICE_OBJECT,
  64. DeviceObject );
  65. } else {
  66. VolDo = NULL;
  67. }
  68. //
  69. // Now case on the function code. For each major function code,
  70. // either call the appropriate FSP routine or case on the minor
  71. // function and then call the FSP routine. The FSP routine that
  72. // we call is responsible for completing the IRP, and not us.
  73. // That way the routine can complete the IRP and then continue
  74. // post processing as required. For example, a read can be
  75. // satisfied right away and then read can be done.
  76. //
  77. // We'll do all of the work within an exception handler that
  78. // will be invoked if ever some underlying operation gets into
  79. // trouble (e.g., if FatReadSectorsSync has trouble).
  80. //
  81. while ( TRUE ) {
  82. DebugTrace(0, Dbg, "FatFspDispatch: Irp = 0x%08lx\n", Irp);
  83. //
  84. // If this Irp was top level, note it in our thread local storage.
  85. //
  86. FsRtlEnterFileSystem();
  87. if ( FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_RECURSIVE_CALL) ) {
  88. IoSetTopLevelIrp( (PIRP)FSRTL_FSP_TOP_LEVEL_IRP );
  89. } else {
  90. IoSetTopLevelIrp( Irp );
  91. }
  92. try {
  93. switch ( IrpContext->MajorFunction ) {
  94. //
  95. // For Create Operation,
  96. //
  97. case IRP_MJ_CREATE:
  98. (VOID) FatCommonCreate( IrpContext, Irp );
  99. break;
  100. //
  101. // For close operations. We do a little kludge here in case
  102. // this close causes a volume to go away. It will NULL the
  103. // VolDo local variable so that we will not try to look at
  104. // the overflow queue.
  105. //
  106. case IRP_MJ_CLOSE:
  107. {
  108. PVCB Vcb;
  109. PFCB Fcb;
  110. PCCB Ccb;
  111. TYPE_OF_OPEN TypeOfOpen;
  112. //
  113. // Extract and decode the file object
  114. //
  115. TypeOfOpen = FatDecodeFileObject( IrpSp->FileObject, &Vcb, &Fcb, &Ccb );
  116. //
  117. // Do the close. We have a slightly different format
  118. // for this call because of the async closes.
  119. //
  120. Status = FatCommonClose( Vcb, Fcb, Ccb, TypeOfOpen, TRUE, &VolDo);
  121. ASSERT(Status == STATUS_SUCCESS);
  122. FatCompleteRequest( IrpContext, Irp, Status );
  123. break;
  124. }
  125. //
  126. // For read operations
  127. //
  128. case IRP_MJ_READ:
  129. (VOID) FatCommonRead( IrpContext, Irp );
  130. break;
  131. //
  132. // For write operations,
  133. //
  134. case IRP_MJ_WRITE:
  135. (VOID) FatCommonWrite( IrpContext, Irp );
  136. break;
  137. //
  138. // For Query Information operations,
  139. //
  140. case IRP_MJ_QUERY_INFORMATION:
  141. (VOID) FatCommonQueryInformation( IrpContext, Irp );
  142. break;
  143. //
  144. // For Set Information operations,
  145. //
  146. case IRP_MJ_SET_INFORMATION:
  147. (VOID) FatCommonSetInformation( IrpContext, Irp );
  148. break;
  149. //
  150. // For Query EA operations,
  151. //
  152. case IRP_MJ_QUERY_EA:
  153. (VOID) FatCommonQueryEa( IrpContext, Irp );
  154. break;
  155. //
  156. // For Set EA operations,
  157. //
  158. case IRP_MJ_SET_EA:
  159. (VOID) FatCommonSetEa( IrpContext, Irp );
  160. break;
  161. //
  162. // For Flush buffers operations,
  163. //
  164. case IRP_MJ_FLUSH_BUFFERS:
  165. (VOID) FatCommonFlushBuffers( IrpContext, Irp );
  166. break;
  167. //
  168. // For Query Volume Information operations,
  169. //
  170. case IRP_MJ_QUERY_VOLUME_INFORMATION:
  171. (VOID) FatCommonQueryVolumeInfo( IrpContext, Irp );
  172. break;
  173. //
  174. // For Set Volume Information operations,
  175. //
  176. case IRP_MJ_SET_VOLUME_INFORMATION:
  177. (VOID) FatCommonSetVolumeInfo( IrpContext, Irp );
  178. break;
  179. //
  180. // For File Cleanup operations,
  181. //
  182. case IRP_MJ_CLEANUP:
  183. (VOID) FatCommonCleanup( IrpContext, Irp );
  184. break;
  185. //
  186. // For Directory Control operations,
  187. //
  188. case IRP_MJ_DIRECTORY_CONTROL:
  189. (VOID) FatCommonDirectoryControl( IrpContext, Irp );
  190. break;
  191. //
  192. // For File System Control operations,
  193. //
  194. case IRP_MJ_FILE_SYSTEM_CONTROL:
  195. (VOID) FatCommonFileSystemControl( IrpContext, Irp );
  196. break;
  197. //
  198. // For Lock Control operations,
  199. //
  200. case IRP_MJ_LOCK_CONTROL:
  201. (VOID) FatCommonLockControl( IrpContext, Irp );
  202. break;
  203. //
  204. // For Device Control operations,
  205. //
  206. case IRP_MJ_DEVICE_CONTROL:
  207. (VOID) FatCommonDeviceControl( IrpContext, Irp );
  208. break;
  209. //
  210. // For the Shutdown operation,
  211. //
  212. case IRP_MJ_SHUTDOWN:
  213. (VOID) FatCommonShutdown( IrpContext, Irp );
  214. break;
  215. //
  216. // For plug and play operations.
  217. //
  218. case IRP_MJ_PNP:
  219. //
  220. // I don't believe this should ever occur, but allow for the unexpected.
  221. //
  222. (VOID) FatCommonPnp( IrpContext, Irp );
  223. break;
  224. //
  225. // For any other major operations, return an invalid
  226. // request.
  227. //
  228. default:
  229. FatCompleteRequest( IrpContext, Irp, STATUS_INVALID_DEVICE_REQUEST );
  230. break;
  231. }
  232. } except(FatExceptionFilter( IrpContext, GetExceptionInformation() )) {
  233. //
  234. // We had some trouble trying to perform the requested
  235. // operation, so we'll abort the I/O request with
  236. // the error status that we get back from the
  237. // execption code.
  238. //
  239. (VOID) FatProcessException( IrpContext, Irp, GetExceptionCode() );
  240. }
  241. IoSetTopLevelIrp( NULL );
  242. FsRtlExitFileSystem();
  243. //
  244. // If there are any entries on this volume's overflow queue, service
  245. // them.
  246. //
  247. if ( VolDo != NULL ) {
  248. PVOID Entry;
  249. //
  250. // We have a volume device object so see if there is any work
  251. // left to do in its overflow queue.
  252. //
  253. Entry = FatRemoveOverflowEntry( VolDo );
  254. //
  255. // There wasn't an entry, break out of the loop and return to
  256. // the Ex Worker thread.
  257. //
  258. if ( Entry == NULL ) {
  259. break;
  260. }
  261. //
  262. // Extract the IrpContext, Irp, and IrpSp, and loop.
  263. //
  264. IrpContext = CONTAINING_RECORD( Entry,
  265. IRP_CONTEXT,
  266. WorkQueueItem.List );
  267. SetFlag(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT | IRP_CONTEXT_FLAG_IN_FSP);
  268. Irp = IrpContext->OriginatingIrp;
  269. IrpSp = IoGetCurrentIrpStackLocation( Irp );
  270. continue;
  271. } else {
  272. break;
  273. }
  274. }
  275. //
  276. // Decrement the PostedRequestCount.
  277. //
  278. if ( VolDo ) {
  279. ExInterlockedAddUlong( &VolDo->PostedRequestCount,
  280. 0xffffffff,
  281. &VolDo->OverflowQueueSpinLock );
  282. }
  283. return;
  284. }
  285. //
  286. // Internal support routine, spinlock wrapper.
  287. //
  288. PVOID
  289. FatRemoveOverflowEntry (
  290. IN PVOLUME_DEVICE_OBJECT VolDo
  291. )
  292. {
  293. PVOID Entry;
  294. KIRQL SavedIrql;
  295. KeAcquireSpinLock( &VolDo->OverflowQueueSpinLock, &SavedIrql );
  296. if (VolDo->OverflowQueueCount > 0) {
  297. //
  298. // There is overflow work to do in this volume so we'll
  299. // decrement the Overflow count, dequeue the IRP, and release
  300. // the Event
  301. //
  302. VolDo->OverflowQueueCount -= 1;
  303. Entry = RemoveHeadList( &VolDo->OverflowQueue );
  304. } else {
  305. Entry = NULL;
  306. }
  307. KeReleaseSpinLock( &VolDo->OverflowQueueSpinLock, SavedIrql );
  308. return Entry;
  309. }