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.

1782 lines
45 KiB

  1. /*++
  2. Copyright (c) 1990-2000 Microsoft Corporation
  3. Module Name:
  4. cache.c
  5. Abstract:
  6. This module implements the cache management routines for the Fat
  7. FSD and FSP, by calling the Common Cache Manager.
  8. // @@BEGIN_DDKSPLIT
  9. Author:
  10. Tom Miller [TomM] 26-Jan-1990
  11. Revision History:
  12. // @@END_DDKSPLIT
  13. --*/
  14. #include "FatProcs.h"
  15. //
  16. // The Bug check file id for this module
  17. //
  18. #define BugCheckFileId (FAT_BUG_CHECK_CACHESUP)
  19. //
  20. // Local debug trace level
  21. //
  22. #define Dbg (DEBUG_TRACE_CACHESUP)
  23. #if DBG
  24. BOOLEAN
  25. FatIsCurrentOperationSynchedForDcbTeardown (
  26. IN PIRP_CONTEXT IrpContext,
  27. IN PDCB Dcb
  28. );
  29. #endif
  30. #ifdef ALLOC_PRAGMA
  31. #pragma alloc_text(PAGE, FatCloseEaFile)
  32. #pragma alloc_text(PAGE, FatCompleteMdl)
  33. #pragma alloc_text(PAGE, FatOpenDirectoryFile)
  34. #pragma alloc_text(PAGE, FatOpenEaFile)
  35. #pragma alloc_text(PAGE, FatPinMappedData)
  36. #pragma alloc_text(PAGE, FatPrepareWriteDirectoryFile)
  37. #pragma alloc_text(PAGE, FatPrepareWriteVolumeFile)
  38. #pragma alloc_text(PAGE, FatReadDirectoryFile)
  39. #pragma alloc_text(PAGE, FatReadVolumeFile)
  40. #pragma alloc_text(PAGE, FatRepinBcb)
  41. #pragma alloc_text(PAGE, FatSyncUninitializeCacheMap)
  42. #pragma alloc_text(PAGE, FatUnpinRepinnedBcbs)
  43. #pragma alloc_text(PAGE, FatZeroData)
  44. #if DBG
  45. #pragma alloc_text(PAGE, FatIsCurrentOperationSynchedForDcbTeardown)
  46. #endif
  47. #endif
  48. VOID
  49. FatReadVolumeFile (
  50. IN PIRP_CONTEXT IrpContext,
  51. IN PVCB Vcb,
  52. IN VBO StartingVbo,
  53. IN ULONG ByteCount,
  54. OUT PBCB *Bcb,
  55. OUT PVOID *Buffer
  56. )
  57. /*++
  58. Routine Description:
  59. This routine is called when the specified range of sectors is to be
  60. read into the cache. In fat, the volume file only contains the boot
  61. sector, reserved sectors, and the "fat(s)." Thus the volume file is
  62. of fixed size and only extends up to (but not not including) the root
  63. directory entry, and will never move or change size.
  64. The fat volume file is also peculiar in that, since it starts at the
  65. logical beginning of the disk, Vbo == Lbo.
  66. Arguments:
  67. Vcb - Pointer to the VCB for the volume
  68. StartingVbo - The virtual offset of the first desired byte
  69. ByteCount - Number of bytes desired
  70. Bcb - Returns a pointer to the BCB which is valid until unpinned
  71. Buffer - Returns a pointer to the sectors, which is valid until unpinned
  72. --*/
  73. {
  74. LARGE_INTEGER Vbo;
  75. PAGED_CODE();
  76. //
  77. // Check to see that all references are within the Bios Parameter Block
  78. // or the fat(s). A special case is made when StartingVbo == 0 at
  79. // mounting time since we do not know how big the fat is.
  80. //
  81. ASSERT( ((StartingVbo == 0) || ((StartingVbo + ByteCount) <= (ULONG)
  82. (FatRootDirectoryLbo( &Vcb->Bpb ) + PAGE_SIZE))));
  83. DebugTrace(+1, Dbg, "FatReadVolumeFile\n", 0);
  84. DebugTrace( 0, Dbg, "Vcb = %08lx\n", Vcb);
  85. DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", StartingVbo);
  86. DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
  87. //
  88. // Call the Cache manager to attempt the transfer.
  89. //
  90. Vbo.QuadPart = StartingVbo;
  91. if (!CcMapData( Vcb->VirtualVolumeFile,
  92. &Vbo,
  93. ByteCount,
  94. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT),
  95. Bcb,
  96. Buffer )) {
  97. ASSERT( !FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT) );
  98. //
  99. // Could not read the data without waiting (cache miss).
  100. //
  101. FatRaiseStatus( IrpContext, STATUS_CANT_WAIT );
  102. }
  103. DbgDoit( IrpContext->PinCount += 1 )
  104. DebugTrace(-1, Dbg, "FatReadVolumeFile -> VOID, *BCB = %08lx\n", *Bcb);
  105. return;
  106. }
  107. VOID
  108. FatPrepareWriteVolumeFile (
  109. IN PIRP_CONTEXT IrpContext,
  110. IN PVCB Vcb,
  111. IN VBO StartingVbo,
  112. IN ULONG ByteCount,
  113. OUT PBCB *Bcb,
  114. OUT PVOID *Buffer,
  115. IN BOOLEAN Reversible,
  116. IN BOOLEAN Zero
  117. )
  118. /*++
  119. Routine Description:
  120. This routine first looks to see if the specified range of sectors,
  121. is already in the cache. If so, it increments the BCB PinCount,
  122. sets the BCB dirty, and returns with the location of the sectors.
  123. If the sectors are not in the cache and Wait is TRUE, it finds a
  124. free BCB (potentially causing a flush), and clears out the entire
  125. buffer. Once this is done, it increments the BCB PinCount, sets the
  126. BCB dirty, and returns with the location of the sectors.
  127. If the sectors are not in the cache and Wait is FALSE, this routine
  128. raises STATUS_CANT_WAIT.
  129. Arguments:
  130. Vcb - Pointer to the VCB for the volume
  131. StartingVbo - The virtual offset of the first byte to be written
  132. ByteCount - Number of bytes to be written
  133. Bcb - Returns a pointer to the BCB which is valid until unpinned
  134. Buffer - Returns a pointer to the sectors, which is valid until unpinned
  135. Reversible - Supplies TRUE if the specified range of modification should
  136. be repinned so that the operation can be reversed in a controlled
  137. fashion if errors are encountered.
  138. Zero - Supplies TRUE if the specified range of bytes should be zeroed
  139. --*/
  140. {
  141. LARGE_INTEGER Vbo;
  142. PAGED_CODE();
  143. //
  144. // Check to see that all references are within the Bios Parameter Block
  145. // or the fat(s).
  146. //
  147. ASSERT( ((StartingVbo + ByteCount) <= (ULONG)
  148. (FatRootDirectoryLbo( &Vcb->Bpb ))));
  149. DebugTrace(+1, Dbg, "FatPrepareWriteVolumeFile\n", 0);
  150. DebugTrace( 0, Dbg, "Vcb = %08lx\n", Vcb);
  151. DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", (ULONG)StartingVbo);
  152. DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
  153. DebugTrace( 0, Dbg, "Zero = %08lx\n", Zero);
  154. //
  155. // Call the Cache manager to attempt the transfer.
  156. //
  157. Vbo.QuadPart = StartingVbo;
  158. if (!CcPinRead( Vcb->VirtualVolumeFile,
  159. &Vbo,
  160. ByteCount,
  161. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT),
  162. Bcb,
  163. Buffer )) {
  164. ASSERT( !FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT) );
  165. //
  166. // Could not read the data without waiting (cache miss).
  167. //
  168. FatRaiseStatus( IrpContext, STATUS_CANT_WAIT );
  169. }
  170. //
  171. // This keeps the data pinned until we complete the request
  172. // and writes the dirty bit through to the disk.
  173. //
  174. DbgDoit( IrpContext->PinCount += 1 )
  175. try {
  176. if (Zero) {
  177. RtlZeroMemory( *Buffer, ByteCount );
  178. }
  179. FatSetDirtyBcb( IrpContext, *Bcb, Vcb, Reversible );
  180. } finally {
  181. if (AbnormalTermination()) {
  182. FatUnpinBcb(IrpContext, *Bcb);
  183. }
  184. }
  185. DebugTrace(-1, Dbg, "FatPrepareWriteVolumeFile -> VOID, *Bcb = %08lx\n", *Bcb);
  186. return;
  187. }
  188. VOID
  189. FatReadDirectoryFile (
  190. IN PIRP_CONTEXT IrpContext,
  191. IN PDCB Dcb,
  192. IN VBO StartingVbo,
  193. IN ULONG ByteCount,
  194. IN BOOLEAN Pin,
  195. OUT PBCB *Bcb,
  196. OUT PVOID *Buffer,
  197. OUT PNTSTATUS Status
  198. )
  199. /*++
  200. Routine Description:
  201. This routine is called when the specified range of sectors is to be
  202. read into the cache. If the desired range falls beyond the current
  203. cache mapping, the fat will be searched, and if the desired range can
  204. be satisfied, the cache mapping will be extended and the MCB updated
  205. accordingly.
  206. Arguments:
  207. Dcb - Pointer to the DCB for the directory
  208. StartingVbo - The virtual offset of the first desired byte
  209. ByteCount - Number of bytes desired
  210. Pin - Tells us if we should pin instead of just mapping.
  211. Bcb - Returns a pointer to the BCB which is valid until unpinned
  212. Buffer - Returns a pointer to the sectors, which is valid until unpinned
  213. Status - Returns the status of the operation.
  214. --*/
  215. {
  216. LARGE_INTEGER Vbo;
  217. PAGED_CODE();
  218. DebugTrace(+1, Dbg, "FatReadDirectoryFile\n", 0);
  219. DebugTrace( 0, Dbg, "Dcb = %08lx\n", Dcb);
  220. DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", StartingVbo);
  221. DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
  222. //
  223. // Check for the zero case
  224. //
  225. if (ByteCount == 0) {
  226. DebugTrace(0, Dbg, "Nothing to read\n", 0);
  227. *Bcb = NULL;
  228. *Buffer = NULL;
  229. *Status = STATUS_SUCCESS;
  230. DebugTrace(-1, Dbg, "FatReadDirectoryFile -> VOID\n", 0);
  231. return;
  232. }
  233. //
  234. // If we need to create a directory file and initialize the
  235. // cachemap, do so.
  236. //
  237. FatOpenDirectoryFile( IrpContext, Dcb );
  238. //
  239. // Now if the transfer is beyond the allocation size return EOF.
  240. //
  241. if (StartingVbo >= Dcb->Header.AllocationSize.LowPart) {
  242. DebugTrace(0, Dbg, "End of file read for directory\n", 0);
  243. *Bcb = NULL;
  244. *Buffer = NULL;
  245. *Status = STATUS_END_OF_FILE;
  246. DebugTrace(-1, Dbg, "FatReadDirectoryFile -> VOID\n", 0);
  247. return;
  248. }
  249. //
  250. // If the caller is trying to read past the EOF, truncate the
  251. // read.
  252. //
  253. ByteCount = (Dcb->Header.AllocationSize.LowPart - StartingVbo < ByteCount) ?
  254. Dcb->Header.AllocationSize.LowPart - StartingVbo : ByteCount;
  255. ASSERT( ByteCount != 0 );
  256. //
  257. // Call the Cache manager to attempt the transfer.
  258. //
  259. Vbo.QuadPart = StartingVbo;
  260. if (Pin ?
  261. !CcPinRead( Dcb->Specific.Dcb.DirectoryFile,
  262. &Vbo,
  263. ByteCount,
  264. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT),
  265. Bcb,
  266. Buffer )
  267. :
  268. !CcMapData( Dcb->Specific.Dcb.DirectoryFile,
  269. &Vbo,
  270. ByteCount,
  271. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT),
  272. Bcb,
  273. Buffer ) ) {
  274. //
  275. // Could not read the data without waiting (cache miss).
  276. //
  277. *Bcb = NULL;
  278. *Buffer = NULL;
  279. FatRaiseStatus( IrpContext, STATUS_CANT_WAIT );
  280. }
  281. DbgDoit( IrpContext->PinCount += 1 )
  282. *Status = STATUS_SUCCESS;
  283. DebugTrace(-1, Dbg, "FatReadDirectoryFile -> VOID, *BCB = %08lx\n", *Bcb);
  284. return;
  285. }
  286. VOID
  287. FatPrepareWriteDirectoryFile (
  288. IN PIRP_CONTEXT IrpContext,
  289. IN PDCB Dcb,
  290. IN VBO StartingVbo,
  291. IN ULONG ByteCount,
  292. OUT PBCB *Bcb,
  293. OUT PVOID *Buffer,
  294. IN BOOLEAN Zero,
  295. IN BOOLEAN Reversible,
  296. OUT PNTSTATUS Status
  297. )
  298. /*++
  299. Routine Description:
  300. This routine first looks to see if the specified range of sectors
  301. is already in the cache. If so, it increments the BCB PinCount,
  302. sets the BCB dirty, and returns TRUE with the location of the sectors.
  303. The IrpContext->Flags .. Wait == TRUE/FALSE actions of this routine are identical to
  304. FatPrepareWriteVolumeFile() above.
  305. Arguments:
  306. Dcb - Pointer to the DCB for the directory
  307. StartingVbo - The virtual offset of the first byte to be written
  308. ByteCount - Number of bytes to be written
  309. Bcb - Returns a pointer to the BCB which is valid until unpinned
  310. Buffer - Returns a pointer to the sectors, which is valid until unpinned
  311. Zero - Supplies TRUE if the specified range of bytes should be zeroed
  312. Reversible - Supplies TRUE if the specified range of modification should
  313. be repinned so that the operation can be reversed in a controlled
  314. fashion if errors are encountered.
  315. Status - Returns the status of the operation.
  316. --*/
  317. {
  318. LARGE_INTEGER Vbo;
  319. ULONG InitialAllocation;
  320. BOOLEAN UnwindWeAllocatedDiskSpace = FALSE;
  321. ULONG ClusterSize;
  322. PVOID LocalBuffer;
  323. PAGED_CODE();
  324. DebugTrace(+1, Dbg, "FatPrepareWriteDirectoryFile\n", 0);
  325. DebugTrace( 0, Dbg, "Dcb = %08lx\n", Dcb);
  326. DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", (ULONG)StartingVbo);
  327. DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
  328. DebugTrace( 0, Dbg, "Zero = %08lx\n", Zero);
  329. *Bcb = NULL;
  330. *Buffer = NULL;
  331. //
  332. // If we need to create a directory file and initialize the
  333. // cachemap, do so.
  334. //
  335. FatOpenDirectoryFile( IrpContext, Dcb );
  336. //
  337. // If the transfer is beyond the allocation size we need to
  338. // extend the directory's allocation. The call to
  339. // AddFileAllocation will raise a condition if
  340. // it runs out of disk space. Note that the root directory
  341. // cannot be extended.
  342. //
  343. Vbo.QuadPart = StartingVbo;
  344. try {
  345. if (StartingVbo + ByteCount > Dcb->Header.AllocationSize.LowPart) {
  346. if (NodeType(Dcb) == FAT_NTC_ROOT_DCB &&
  347. !FatIsFat32(Dcb->Vcb)) {
  348. FatRaiseStatus( IrpContext, STATUS_DISK_FULL );
  349. }
  350. DebugTrace(0, Dbg, "Try extending normal directory\n", 0);
  351. InitialAllocation = Dcb->Header.AllocationSize.LowPart;
  352. FatAddFileAllocation( IrpContext,
  353. Dcb,
  354. Dcb->Specific.Dcb.DirectoryFile,
  355. StartingVbo + ByteCount );
  356. UnwindWeAllocatedDiskSpace = TRUE;
  357. //
  358. // Inform the cache manager of the new allocation
  359. //
  360. Dcb->Header.FileSize.LowPart =
  361. Dcb->Header.AllocationSize.LowPart;
  362. CcSetFileSizes( Dcb->Specific.Dcb.DirectoryFile,
  363. (PCC_FILE_SIZES)&Dcb->Header.AllocationSize );
  364. //
  365. // Set up the Bitmap buffer if it is not big enough already
  366. //
  367. FatCheckFreeDirentBitmap( IrpContext, Dcb );
  368. //
  369. // The newly allocated clusters should be zeroed starting at
  370. // the previous allocation size
  371. //
  372. Zero = TRUE;
  373. Vbo.QuadPart = InitialAllocation;
  374. ByteCount = Dcb->Header.AllocationSize.LowPart - InitialAllocation;
  375. }
  376. //
  377. // Call the Cache Manager to attempt the transfer, going one cluster
  378. // at a time to avoid pinning across a page boundary.
  379. //
  380. ClusterSize =
  381. 1 << Dcb->Vcb->AllocationSupport.LogOfBytesPerCluster;
  382. while (ByteCount > 0) {
  383. ULONG BytesToPin;
  384. *Bcb = NULL;
  385. if (ByteCount > ClusterSize) {
  386. BytesToPin = ClusterSize;
  387. } else {
  388. BytesToPin = ByteCount;
  389. }
  390. ASSERT( (Vbo.QuadPart / ClusterSize) ==
  391. (Vbo.QuadPart + BytesToPin - 1)/ClusterSize );
  392. if (!CcPinRead( Dcb->Specific.Dcb.DirectoryFile,
  393. &Vbo,
  394. BytesToPin,
  395. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT),
  396. Bcb,
  397. &LocalBuffer )) {
  398. //
  399. // Could not read the data without waiting (cache miss).
  400. //
  401. FatRaiseStatus( IrpContext, STATUS_CANT_WAIT );
  402. }
  403. //
  404. // Update our caller with the beginning of their request.
  405. //
  406. if (*Buffer == NULL) {
  407. *Buffer = LocalBuffer;
  408. }
  409. DbgDoit( IrpContext->PinCount += 1 )
  410. if (Zero) {
  411. //
  412. // We set this guy dirty right now so that we can raise CANT_WAIT when
  413. // it needs to be done. It'd be beautiful if we could noop the read IO
  414. // since we know we don't care about it.
  415. //
  416. RtlZeroMemory( LocalBuffer, BytesToPin );
  417. CcSetDirtyPinnedData( *Bcb, NULL );
  418. }
  419. ByteCount -= BytesToPin;
  420. Vbo.QuadPart += BytesToPin;
  421. if (ByteCount > 0) {
  422. FatUnpinBcb( IrpContext, *Bcb );
  423. }
  424. }
  425. //
  426. // This lets us get the data pinned until we complete the request
  427. // and writes the dirty bit through to the disk.
  428. //
  429. FatSetDirtyBcb( IrpContext, *Bcb, Dcb->Vcb, Reversible );
  430. *Status = STATUS_SUCCESS;
  431. } finally {
  432. DebugUnwind( FatPrepareWriteDirectoryFile );
  433. if (AbnormalTermination()) {
  434. //
  435. // These steps are carefully arranged - FatTruncateFileAllocation can raise.
  436. // Make sure we unpin the buffer. If FTFA raises, the effect should be benign.
  437. //
  438. FatUnpinBcb(IrpContext, *Bcb);
  439. if (UnwindWeAllocatedDiskSpace == TRUE) {
  440. //
  441. // Inform the cache manager of the change.
  442. //
  443. FatTruncateFileAllocation( IrpContext, Dcb, InitialAllocation );
  444. Dcb->Header.FileSize.LowPart =
  445. Dcb->Header.AllocationSize.LowPart;
  446. CcSetFileSizes( Dcb->Specific.Dcb.DirectoryFile,
  447. (PCC_FILE_SIZES)&Dcb->Header.AllocationSize );
  448. }
  449. }
  450. DebugTrace(-1, Dbg, "FatPrepareWriteDirectoryFile -> (VOID), *Bcb = %08lx\n", *Bcb);
  451. }
  452. return;
  453. }
  454. #if DBG
  455. BOOLEAN FatDisableParentCheck = 0;
  456. BOOLEAN
  457. FatIsCurrentOperationSynchedForDcbTeardown (
  458. IN PIRP_CONTEXT IrpContext,
  459. IN PDCB Dcb
  460. )
  461. {
  462. PIRP Irp = IrpContext->OriginatingIrp;
  463. PIO_STACK_LOCATION Stack = IoGetCurrentIrpStackLocation( Irp ) ;
  464. PFILE_OBJECT FileObject = Stack->FileObject;
  465. PVCB Vcb;
  466. PFCB Fcb;
  467. PCCB Ccb;
  468. PFILE_OBJECT ToCheck[3];
  469. ULONG Index = 0;
  470. PAGED_CODE();
  471. //
  472. // While mounting, we're OK without having to own anything.
  473. //
  474. if (Stack->MajorFunction == IRP_MJ_FILE_SYSTEM_CONTROL &&
  475. Stack->MinorFunction == IRP_MN_MOUNT_VOLUME) {
  476. return TRUE;
  477. }
  478. //
  479. // With the Vcb held, the close path is blocked out.
  480. //
  481. if (ExIsResourceAcquiredSharedLite( &Dcb->Vcb->Resource ) ||
  482. ExIsResourceAcquiredExclusiveLite( &Dcb->Vcb->Resource )) {
  483. return TRUE;
  484. }
  485. //
  486. // Accept this assertion at face value. It comes from GetDirentForFcbOrDcb,
  487. // and is reliable.
  488. //
  489. if (FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_PARENT_BY_CHILD )) {
  490. return TRUE;
  491. }
  492. //
  493. // Determine which fileobjects are around on this operation.
  494. //
  495. if (Stack->MajorFunction == IRP_MJ_SET_INFORMATION &&
  496. Stack->Parameters.SetFile.FileObject) {
  497. ToCheck[Index++] = Stack->Parameters.SetFile.FileObject;
  498. }
  499. if (Stack->FileObject) {
  500. ToCheck[Index++] = Stack->FileObject;
  501. }
  502. ToCheck[Index] = NULL;
  503. //
  504. // If the fileobjects we have are for this dcb or a child of it, we are
  505. // also guaranteed that this dcb isn't going anywhere (even without
  506. // the Vcb).
  507. //
  508. for (Index = 0; ToCheck[Index] != NULL; Index++) {
  509. (VOID) FatDecodeFileObject( ToCheck[Index], &Vcb, &Fcb, &Ccb );
  510. while ( Fcb ) {
  511. if (Fcb == Dcb) {
  512. return TRUE;
  513. }
  514. Fcb = Fcb->ParentDcb;
  515. }
  516. }
  517. return FatDisableParentCheck;
  518. }
  519. #endif // DBG
  520. VOID
  521. FatOpenDirectoryFile (
  522. IN PIRP_CONTEXT IrpContext,
  523. IN PDCB Dcb
  524. )
  525. /*++
  526. Routine Description:
  527. This routine opens a new directory file if one is not already open.
  528. Arguments:
  529. Dcb - Pointer to the DCB for the directory
  530. Return Value:
  531. None.
  532. --*/
  533. {
  534. PAGED_CODE();
  535. DebugTrace(+1, Dbg, "FatOpenDirectoryFile\n", 0);
  536. DebugTrace( 0, Dbg, "Dcb = %08lx\n", Dcb);
  537. //
  538. // If we don't have some hold on this Dcb (there are several ways), there is nothing
  539. // to prevent child files from closing and tearing this branch of the tree down in the
  540. // midst of our slapping this reference onto it.
  541. //
  542. // I really wish we had a proper Fcb synchronization model (like CDFS/UDFS/NTFS).
  543. //
  544. ASSERT( FatIsCurrentOperationSynchedForDcbTeardown( IrpContext, Dcb ));
  545. //
  546. // If we haven't yet set the correct AllocationSize, do so.
  547. //
  548. if (Dcb->Header.AllocationSize.QuadPart == FCB_LOOKUP_ALLOCATIONSIZE_HINT) {
  549. FatLookupFileAllocationSize( IrpContext, Dcb );
  550. Dcb->Header.FileSize.LowPart =
  551. Dcb->Header.AllocationSize.LowPart;
  552. }
  553. //
  554. // Setup the Bitmap buffer if it is not big enough already
  555. //
  556. FatCheckFreeDirentBitmap( IrpContext, Dcb );
  557. //
  558. // Check if we need to create a directory file.
  559. //
  560. // We first do a spot check and then synchronize and check again.
  561. //
  562. if (Dcb->Specific.Dcb.DirectoryFile == NULL) {
  563. PFILE_OBJECT DirectoryFileObject = NULL;
  564. FatAcquireDirectoryFileMutex( Dcb->Vcb );
  565. try {
  566. if (Dcb->Specific.Dcb.DirectoryFile == NULL) {
  567. PDEVICE_OBJECT RealDevice;
  568. //
  569. // Create the special file object for the directory file, and set
  570. // up its pointers back to the Dcb and the section object pointer.
  571. // Note that setting the DirectoryFile pointer in the Dcb has
  572. // to be the last thing done.
  573. //
  574. // Preallocate a close context since we have no Ccb for this object.
  575. //
  576. RealDevice = Dcb->Vcb->CurrentDevice;
  577. DirectoryFileObject = IoCreateStreamFileObject( NULL, RealDevice );
  578. FatPreallocateCloseContext();
  579. FatSetFileObject( DirectoryFileObject,
  580. DirectoryFile,
  581. Dcb,
  582. NULL );
  583. DirectoryFileObject->SectionObjectPointer = &Dcb->NonPaged->SectionObjectPointers;
  584. DirectoryFileObject->ReadAccess = TRUE;
  585. DirectoryFileObject->WriteAccess = TRUE;
  586. DirectoryFileObject->DeleteAccess = TRUE;
  587. InterlockedIncrement( &Dcb->Specific.Dcb.DirectoryFileOpenCount );
  588. Dcb->Specific.Dcb.DirectoryFile = DirectoryFileObject;
  589. //
  590. // Indicate we're happy with the fileobject now.
  591. //
  592. DirectoryFileObject = NULL;
  593. }
  594. } finally {
  595. FatReleaseDirectoryFileMutex( Dcb->Vcb );
  596. //
  597. // Rip the object up if we couldn't get the close context.
  598. //
  599. if (DirectoryFileObject) {
  600. ObDereferenceObject( DirectoryFileObject );
  601. }
  602. }
  603. }
  604. //
  605. // Finally check if we need to initialize the Cache Map for the
  606. // directory file. The size of the section we are going to map
  607. // the current allocation size for the directory. Note that the
  608. // cache manager will provide syncronization for us.
  609. //
  610. if ( Dcb->Specific.Dcb.DirectoryFile->PrivateCacheMap == NULL ) {
  611. Dcb->Header.ValidDataLength = FatMaxLarge;
  612. Dcb->ValidDataToDisk = MAXULONG;
  613. CcInitializeCacheMap( Dcb->Specific.Dcb.DirectoryFile,
  614. (PCC_FILE_SIZES)&Dcb->Header.AllocationSize,
  615. TRUE,
  616. &FatData.CacheManagerNoOpCallbacks,
  617. Dcb );
  618. }
  619. DebugTrace(-1, Dbg, "FatOpenDirectoryFile -> VOID\n", 0);
  620. return;
  621. }
  622. PFILE_OBJECT
  623. FatOpenEaFile (
  624. IN PIRP_CONTEXT IrpContext,
  625. IN PFCB EaFcb
  626. )
  627. /*++
  628. Routine Description:
  629. This routine opens the Ea file.
  630. Arguments:
  631. EaFcb - Pointer to the Fcb for the Ea file.
  632. Return Value:
  633. Pointer to the new file object.
  634. --*/
  635. {
  636. PFILE_OBJECT EaFileObject = NULL;
  637. PDEVICE_OBJECT RealDevice;
  638. PAGED_CODE();
  639. DebugTrace(+1, Dbg, "FatOpenEaFile\n", 0);
  640. DebugTrace( 0, Dbg, "EaFcb = %08lx\n", EaFcb);
  641. //
  642. // Create the special file object for the ea file, and set
  643. // up its pointers back to the Fcb and the section object pointer
  644. //
  645. RealDevice = EaFcb->Vcb->CurrentDevice;
  646. EaFileObject = IoCreateStreamFileObject( NULL, RealDevice );
  647. try {
  648. FatPreallocateCloseContext();
  649. FatSetFileObject( EaFileObject,
  650. EaFile,
  651. EaFcb,
  652. NULL );
  653. EaFileObject->SectionObjectPointer = &EaFcb->NonPaged->SectionObjectPointers;
  654. EaFileObject->ReadAccess = TRUE;
  655. EaFileObject->WriteAccess = TRUE;
  656. //
  657. // Finally check if we need to initialize the Cache Map for the
  658. // ea file. The size of the section we are going to map
  659. // the current allocation size for the Fcb.
  660. //
  661. EaFcb->Header.ValidDataLength = FatMaxLarge;
  662. CcInitializeCacheMap( EaFileObject,
  663. (PCC_FILE_SIZES)&EaFcb->Header.AllocationSize,
  664. TRUE,
  665. &FatData.CacheManagerCallbacks,
  666. EaFcb );
  667. CcSetAdditionalCacheAttributes( EaFileObject, TRUE, TRUE );
  668. } finally {
  669. //
  670. // Drop the fileobject if we're raising. Two cases: couldn't get
  671. // the close context, and it is still an UnopenedFileObject, or
  672. // we lost trying to build the cache map - in which case we're
  673. // OK for the close context if we have to.
  674. //
  675. if (AbnormalTermination()) {
  676. ObDereferenceObject( EaFileObject );
  677. }
  678. }
  679. DebugTrace(-1, Dbg, "FatOpenEaFile -> %08lx\n", EaFileObject);
  680. UNREFERENCED_PARAMETER( IrpContext );
  681. return EaFileObject;
  682. }
  683. VOID
  684. FatCloseEaFile (
  685. IN PIRP_CONTEXT IrpContext,
  686. IN PVCB Vcb,
  687. IN BOOLEAN FlushFirst
  688. )
  689. /*++
  690. Routine Description:
  691. This routine shuts down the ea file. Usually this is required when the volume
  692. begins to leave the system: after verify, dismount, deletion, pnp.
  693. Arguments:
  694. Vcb - the volume to close the ea file on
  695. FlushFirst - whether the file should be flushed
  696. Return Value:
  697. None. As a side effect, the EA fileobject in the Vcb is cleared.
  698. Caller must have the Vcb exclusive.
  699. --*/
  700. {
  701. PFILE_OBJECT EaFileObject = Vcb->VirtualEaFile;
  702. PAGED_CODE();
  703. DebugTrace(+1, Dbg, "FatCloseEaFile\n", 0);
  704. DebugTrace( 0, Dbg, "Vcb = %08lx\n", Vcb);
  705. ASSERT( FatVcbAcquiredExclusive(IrpContext, Vcb) );
  706. if (EaFileObject != NULL) {
  707. EaFileObject = Vcb->VirtualEaFile;
  708. if (FlushFirst) {
  709. CcFlushCache( Vcb->VirtualEaFile->SectionObjectPointer, NULL, 0, NULL );
  710. }
  711. Vcb->VirtualEaFile = NULL;
  712. //
  713. // Empty the Mcb for the Ea file.
  714. //
  715. FatRemoveMcbEntry( Vcb, &Vcb->EaFcb->Mcb, 0, 0xFFFFFFFF );
  716. //
  717. // Set the file object type to unopened file object
  718. // and dereference it.
  719. //
  720. FatSetFileObject( EaFileObject,
  721. UnopenedFileObject,
  722. NULL,
  723. NULL );
  724. FatSyncUninitializeCacheMap( IrpContext, EaFileObject );
  725. ObDereferenceObject( EaFileObject );
  726. }
  727. DebugTrace(-1, Dbg, "FatCloseEaFile -> %08lx\n", EaFileObject);
  728. }
  729. VOID
  730. FatSetDirtyBcb (
  731. IN PIRP_CONTEXT IrpContext,
  732. IN PBCB Bcb,
  733. IN PVCB Vcb OPTIONAL,
  734. IN BOOLEAN Reversible
  735. )
  736. /*++
  737. Routine Description:
  738. This routine saves a reference to the bcb in the irp context and
  739. sets the bcb dirty. This will have the affect of keeping the page in
  740. memory until we complete the request
  741. In addition, a DPC is set to fire in 5 seconds (or if one is pending,
  742. pushed back 5 seconds) to mark the volume clean.
  743. Arguments:
  744. Bcb - Supplies the Bcb being set dirty
  745. Vcb - Supplies the volume being marked dirty
  746. Reversible - Supplies TRUE if the specified range of bcb should be repinned
  747. so that the changes can be reversed in a controlled fashion if errors
  748. are encountered.
  749. Return Value:
  750. None.
  751. --*/
  752. {
  753. DebugTrace(+1, Dbg, "FatSetDirtyBcb\n", 0 );
  754. DebugTrace( 0, Dbg, "IrpContext = %08lx\n", IrpContext );
  755. DebugTrace( 0, Dbg, "Bcb = %08lx\n", Bcb );
  756. DebugTrace( 0, Dbg, "Vcb = %08lx\n", Vcb );
  757. //
  758. // Repin the bcb as required
  759. //
  760. if (Reversible) {
  761. FatRepinBcb( IrpContext, Bcb );
  762. }
  763. //
  764. // Set the bcb dirty
  765. //
  766. CcSetDirtyPinnedData( Bcb, NULL );
  767. //
  768. // If volume dirtying isn't disabled for this operation (for
  769. // instance, when we're changing the dirty state), set the
  770. // volume dirty if we were given a Vcb that we want to perform
  771. // clean volume processing on, and return.
  772. //
  773. // As a historical note, we used to key off of the old floppy
  774. // (now deferred flush) bit to disable dirtying behavior. Since
  775. // hotpluggable media can still be yanked while operations are
  776. // in flight, recognize that its really the case that FAT12
  777. // doesn't have the dirty bit.
  778. //
  779. if ( !FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_DISABLE_DIRTY) &&
  780. ARGUMENT_PRESENT(Vcb) &&
  781. !FatIsFat12(Vcb)) {
  782. KIRQL SavedIrql;
  783. BOOLEAN SetTimer;
  784. LARGE_INTEGER TimeSincePreviousCall;
  785. LARGE_INTEGER CurrentTime;
  786. //
  787. // "Borrow" the irp context spinlock.
  788. //
  789. KeQuerySystemTime( &CurrentTime );
  790. KeAcquireSpinLock( &FatData.GeneralSpinLock, &SavedIrql );
  791. TimeSincePreviousCall.QuadPart =
  792. CurrentTime.QuadPart - Vcb->LastFatMarkVolumeDirtyCall.QuadPart;
  793. //
  794. // If more than one second has elapsed since the prior call
  795. // to here, bump the timer up again and see if we need to
  796. // physically mark the volume dirty.
  797. //
  798. if ( (TimeSincePreviousCall.HighPart != 0) ||
  799. (TimeSincePreviousCall.LowPart > (1000 * 1000 * 10)) ) {
  800. SetTimer = TRUE;
  801. } else {
  802. SetTimer = FALSE;
  803. }
  804. KeReleaseSpinLock( &FatData.GeneralSpinLock, SavedIrql );
  805. if ( SetTimer ) {
  806. LARGE_INTEGER CleanVolumeTimer;
  807. //
  808. // We use a shorter volume clean timer for hot plug volumes.
  809. //
  810. CleanVolumeTimer.QuadPart = FlagOn( Vcb->VcbState, VCB_STATE_FLAG_DEFERRED_FLUSH)
  811. ? (LONG)-1500*1000*10
  812. : (LONG)-8*1000*1000*10;
  813. (VOID)KeCancelTimer( &Vcb->CleanVolumeTimer );
  814. (VOID)KeRemoveQueueDpc( &Vcb->CleanVolumeDpc );
  815. //
  816. // We have now synchronized with anybody clearing the dirty
  817. // flag, so we can now see if we really have to actually write
  818. // out the physical bit.
  819. //
  820. if ( !FlagOn(Vcb->VcbState, VCB_STATE_FLAG_VOLUME_DIRTY) ) {
  821. //
  822. // We want to really mark the volume dirty now.
  823. //
  824. if (!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_MOUNTED_DIRTY)) {
  825. FatMarkVolume( IrpContext, Vcb, VolumeDirty );
  826. }
  827. SetFlag( Vcb->VcbState, VCB_STATE_FLAG_VOLUME_DIRTY );
  828. //
  829. // Lock the volume if it is removable.
  830. //
  831. if (FlagOn( Vcb->VcbState, VCB_STATE_FLAG_REMOVABLE_MEDIA)) {
  832. FatToggleMediaEjectDisable( IrpContext, Vcb, TRUE );
  833. }
  834. }
  835. KeAcquireSpinLock( &FatData.GeneralSpinLock, &SavedIrql );
  836. KeQuerySystemTime( &Vcb->LastFatMarkVolumeDirtyCall );
  837. KeReleaseSpinLock( &FatData.GeneralSpinLock, SavedIrql );
  838. KeSetTimer( &Vcb->CleanVolumeTimer,
  839. CleanVolumeTimer,
  840. &Vcb->CleanVolumeDpc );
  841. }
  842. }
  843. DebugTrace(-1, Dbg, "FatSetDirtyBcb -> VOID\n", 0 );
  844. }
  845. VOID
  846. FatRepinBcb (
  847. IN PIRP_CONTEXT IrpContext,
  848. IN PBCB Bcb
  849. )
  850. /*++
  851. Routine Description:
  852. This routine saves a reference to the bcb in the irp context. This will
  853. have the affect of keeping the page in memory until we complete the
  854. request
  855. Arguments:
  856. Bcb - Supplies the Bcb being referenced
  857. Return Value:
  858. None.
  859. --*/
  860. {
  861. PREPINNED_BCBS Repinned;
  862. ULONG i;
  863. PAGED_CODE();
  864. DebugTrace(+1, Dbg, "FatRepinBcb\n", 0 );
  865. DebugTrace( 0, Dbg, "IrpContext = %08lx\n", IrpContext );
  866. DebugTrace( 0, Dbg, "Bcb = %08lx\n", Bcb );
  867. //
  868. // The algorithm is to search the list of repinned records until
  869. // we either find a match for the bcb or we find a null slot.
  870. //
  871. Repinned = &IrpContext->Repinned;
  872. while (TRUE) {
  873. //
  874. // For every entry in the repinned record check if the bcb's
  875. // match or if the entry is null. If the bcb's match then
  876. // we've done because we've already repinned this bcb, if
  877. // the entry is null then we know, because it's densely packed,
  878. // that the bcb is not in the list so add it to the repinned
  879. // record and repin it.
  880. //
  881. for (i = 0; i < REPINNED_BCBS_ARRAY_SIZE; i += 1) {
  882. if (Repinned->Bcb[i] == Bcb) {
  883. DebugTrace(-1, Dbg, "FatRepinBcb -> VOID\n", 0 );
  884. return;
  885. }
  886. if (Repinned->Bcb[i] == NULL) {
  887. Repinned->Bcb[i] = Bcb;
  888. CcRepinBcb( Bcb );
  889. DebugTrace(-1, Dbg, "FatRepinBcb -> VOID\n", 0 );
  890. return;
  891. }
  892. }
  893. //
  894. // We finished checking one repinned record so now locate the next
  895. // repinned record, If there isn't one then allocate and zero out
  896. // a new one.
  897. //
  898. if (Repinned->Next == NULL) {
  899. Repinned->Next = FsRtlAllocatePoolWithTag( PagedPool,
  900. sizeof(REPINNED_BCBS),
  901. TAG_REPINNED_BCB );
  902. RtlZeroMemory( Repinned->Next, sizeof(REPINNED_BCBS) );
  903. }
  904. Repinned = Repinned->Next;
  905. }
  906. }
  907. VOID
  908. FatUnpinRepinnedBcbs (
  909. IN PIRP_CONTEXT IrpContext
  910. )
  911. /*++
  912. Routine Description:
  913. This routine frees all of the repinned bcbs, stored in an IRP context.
  914. Arguments:
  915. Return Value:
  916. None.
  917. --*/
  918. {
  919. IO_STATUS_BLOCK RaiseIosb;
  920. PREPINNED_BCBS Repinned;
  921. BOOLEAN WriteThroughToDisk;
  922. PFILE_OBJECT FileObject = NULL;
  923. BOOLEAN ForceVerify = FALSE;
  924. ULONG i;
  925. PAGED_CODE();
  926. DebugTrace(+1, Dbg, "FatUnpinRepinnedBcbs\n", 0 );
  927. DebugTrace( 0, Dbg, "IrpContext = %08lx\n", IrpContext );
  928. //
  929. // The algorithm for this procedure is to scan the entire list of
  930. // repinned records unpinning any repinned bcbs. We start off
  931. // with the first record in the irp context, and while there is a
  932. // record to scan we do the following loop.
  933. //
  934. Repinned = &IrpContext->Repinned;
  935. RaiseIosb.Status = STATUS_SUCCESS;
  936. //
  937. // If the request is write through or the media is deferred flush,
  938. // unpin the bcb's write through.
  939. //
  940. WriteThroughToDisk = (BOOLEAN) (!FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_DISABLE_WRITE_THROUGH) &&
  941. IrpContext->Vcb != NULL &&
  942. (FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WRITE_THROUGH) ||
  943. FlagOn(IrpContext->Vcb->VcbState, VCB_STATE_FLAG_DEFERRED_FLUSH)));
  944. while (Repinned != NULL) {
  945. //
  946. // For every non-null entry in the repinned record unpin the
  947. // repinned entry.
  948. //
  949. // If the this is removable media (therefore all requests write-
  950. // through) and the write fails, purge the cache so that we throw
  951. // away the modifications as we will be returning an error to the
  952. // user.
  953. //
  954. for (i = 0; i < REPINNED_BCBS_ARRAY_SIZE; i += 1) {
  955. if (Repinned->Bcb[i] != NULL) {
  956. IO_STATUS_BLOCK Iosb;
  957. if (WriteThroughToDisk &&
  958. FlagOn(IrpContext->Vcb->VcbState, VCB_STATE_FLAG_DEFERRED_FLUSH)) {
  959. FileObject = CcGetFileObjectFromBcb( Repinned->Bcb[i] );
  960. }
  961. CcUnpinRepinnedBcb( Repinned->Bcb[i],
  962. WriteThroughToDisk,
  963. &Iosb );
  964. if ( !NT_SUCCESS(Iosb.Status) ) {
  965. if (RaiseIosb.Status == STATUS_SUCCESS) {
  966. RaiseIosb = Iosb;
  967. }
  968. //
  969. // If this was a writethrough device, purge the cache,
  970. // except for Irp major codes that either don't handle
  971. // the error paths correctly or are simple victims like
  972. // cleanup.c.
  973. //
  974. if (FileObject &&
  975. (IrpContext->MajorFunction != IRP_MJ_CLEANUP) &&
  976. (IrpContext->MajorFunction != IRP_MJ_FLUSH_BUFFERS) &&
  977. (IrpContext->MajorFunction != IRP_MJ_SET_INFORMATION)) {
  978. //
  979. // The call to CcPurgeCacheSection() below will
  980. // purge the entire file from memory. It will also
  981. // block until all the file's BCB's are pinned.
  982. //
  983. // We end up in a deadlock situation of there
  984. // are any other pinned BCB's in this IRP context
  985. // so the first thing we do is search the list
  986. // for BCB's pinned in the same file and unpin
  987. // them.
  988. //
  989. // We are probably not going to lose data because
  990. // it's safe to assume that all flushes will
  991. // fail after the first one fails.
  992. //
  993. ULONG j;
  994. for (j = i + 1; j < REPINNED_BCBS_ARRAY_SIZE; j++) {
  995. if (Repinned->Bcb[j] != NULL) {
  996. if (CcGetFileObjectFromBcb( Repinned->Bcb[j] ) == FileObject) {
  997. CcUnpinRepinnedBcb( Repinned->Bcb[j],
  998. FALSE,
  999. &Iosb );
  1000. Repinned->Bcb[j] = NULL;
  1001. }
  1002. }
  1003. }
  1004. CcPurgeCacheSection( FileObject->SectionObjectPointer,
  1005. NULL,
  1006. 0,
  1007. FALSE );
  1008. //
  1009. // Force a verify operation here since who knows
  1010. // what state things are in.
  1011. //
  1012. ForceVerify = TRUE;
  1013. }
  1014. }
  1015. Repinned->Bcb[i] = NULL;
  1016. }
  1017. }
  1018. //
  1019. // Now find the next repinned record in the list, and possibly
  1020. // delete the one we've just processed.
  1021. //
  1022. if (Repinned != &IrpContext->Repinned) {
  1023. PREPINNED_BCBS Saved;
  1024. Saved = Repinned->Next;
  1025. ExFreePool( Repinned );
  1026. Repinned = Saved;
  1027. } else {
  1028. Repinned = Repinned->Next;
  1029. IrpContext->Repinned.Next = NULL;
  1030. }
  1031. }
  1032. //
  1033. // Now if we weren't completely successful in the our unpin
  1034. // then raise the iosb we got
  1035. //
  1036. if (!NT_SUCCESS(RaiseIosb.Status)) {
  1037. if (ForceVerify && FileObject) {
  1038. SetFlag(FileObject->DeviceObject->Flags, DO_VERIFY_VOLUME);
  1039. IoSetHardErrorOrVerifyDevice( IrpContext->OriginatingIrp,
  1040. FileObject->DeviceObject );
  1041. }
  1042. if (!FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_DISABLE_RAISE )) {
  1043. IrpContext->OriginatingIrp->IoStatus = RaiseIosb;
  1044. FatNormalizeAndRaiseStatus( IrpContext, RaiseIosb.Status );
  1045. }
  1046. }
  1047. DebugTrace(-1, Dbg, "FatUnpinRepinnedBcbs -> VOID\n", 0 );
  1048. return;
  1049. }
  1050. FINISHED
  1051. FatZeroData (
  1052. IN PIRP_CONTEXT IrpContext,
  1053. IN PVCB Vcb,
  1054. IN PFILE_OBJECT FileObject,
  1055. IN ULONG StartingZero,
  1056. IN ULONG ByteCount
  1057. )
  1058. /*++
  1059. **** Temporary function - Remove when CcZeroData is capable of handling
  1060. non sector aligned requests.
  1061. --*/
  1062. {
  1063. LARGE_INTEGER ZeroStart = {0,0};
  1064. LARGE_INTEGER BeyondZeroEnd = {0,0};
  1065. ULONG SectorSize;
  1066. BOOLEAN Finished;
  1067. PAGED_CODE();
  1068. SectorSize = (ULONG)Vcb->Bpb.BytesPerSector;
  1069. ZeroStart.LowPart = (StartingZero + (SectorSize - 1)) & ~(SectorSize - 1);
  1070. //
  1071. // Detect overflow if we were asked to zero in the last sector of the file,
  1072. // which must be "zeroed" already (or we're in trouble).
  1073. //
  1074. if (StartingZero != 0 && ZeroStart.LowPart == 0) {
  1075. return TRUE;
  1076. }
  1077. //
  1078. // Note that BeyondZeroEnd can take the value 4gb.
  1079. //
  1080. BeyondZeroEnd.QuadPart = ((ULONGLONG) StartingZero + ByteCount + (SectorSize - 1))
  1081. & (~((LONGLONG) SectorSize - 1));
  1082. //
  1083. // If we were called to just zero part of a sector we are in trouble.
  1084. //
  1085. if ( ZeroStart.QuadPart == BeyondZeroEnd.QuadPart ) {
  1086. return TRUE;
  1087. }
  1088. Finished = CcZeroData( FileObject,
  1089. &ZeroStart,
  1090. &BeyondZeroEnd,
  1091. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT) );
  1092. return Finished;
  1093. }
  1094. NTSTATUS
  1095. FatCompleteMdl (
  1096. IN PIRP_CONTEXT IrpContext,
  1097. IN PIRP Irp
  1098. )
  1099. /*++
  1100. Routine Description:
  1101. This routine performs the function of completing Mdl read and write
  1102. requests. It should be called only from FatFsdRead and FatFsdWrite.
  1103. Arguments:
  1104. Irp - Supplies the originating Irp.
  1105. Return Value:
  1106. NTSTATUS - Will always be STATUS_PENDING or STATUS_SUCCESS.
  1107. --*/
  1108. {
  1109. PFILE_OBJECT FileObject;
  1110. PIO_STACK_LOCATION IrpSp;
  1111. PAGED_CODE();
  1112. DebugTrace(+1, Dbg, "FatCompleteMdl\n", 0 );
  1113. DebugTrace( 0, Dbg, "IrpContext = %08lx\n", IrpContext );
  1114. DebugTrace( 0, Dbg, "Irp = %08lx\n", Irp );
  1115. //
  1116. // Do completion processing.
  1117. //
  1118. FileObject = IoGetCurrentIrpStackLocation( Irp )->FileObject;
  1119. switch( IrpContext->MajorFunction ) {
  1120. case IRP_MJ_READ:
  1121. CcMdlReadComplete( FileObject, Irp->MdlAddress );
  1122. break;
  1123. case IRP_MJ_WRITE:
  1124. IrpSp = IoGetCurrentIrpStackLocation( Irp );
  1125. ASSERT( FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT ));
  1126. CcMdlWriteComplete( FileObject, &IrpSp->Parameters.Write.ByteOffset, Irp->MdlAddress );
  1127. Irp->IoStatus.Status = STATUS_SUCCESS;
  1128. break;
  1129. default:
  1130. DebugTrace( DEBUG_TRACE_ERROR, 0, "Illegal Mdl Complete.\n", 0);
  1131. FatBugCheck( IrpContext->MajorFunction, 0, 0 );
  1132. }
  1133. //
  1134. // Mdl is now deallocated.
  1135. //
  1136. Irp->MdlAddress = NULL;
  1137. //
  1138. // Complete the request and exit right away.
  1139. //
  1140. FatCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
  1141. DebugTrace(-1, Dbg, "FatCompleteMdl -> STATUS_SUCCESS\n", 0 );
  1142. return STATUS_SUCCESS;
  1143. }
  1144. VOID
  1145. FatSyncUninitializeCacheMap (
  1146. IN PIRP_CONTEXT IrpContext,
  1147. IN PFILE_OBJECT FileObject
  1148. )
  1149. /*++
  1150. Routine Description:
  1151. The routine performs a CcUnitializeCacheMap to LargeZero synchronously. That
  1152. is it waits on the Cc event. This call is useful when we want to be certain
  1153. when a close will actually some in.
  1154. Return Value:
  1155. None.
  1156. --*/
  1157. {
  1158. CACHE_UNINITIALIZE_EVENT UninitializeCompleteEvent;
  1159. NTSTATUS WaitStatus;
  1160. PAGED_CODE();
  1161. KeInitializeEvent( &UninitializeCompleteEvent.Event,
  1162. SynchronizationEvent,
  1163. FALSE);
  1164. CcUninitializeCacheMap( FileObject,
  1165. &FatLargeZero,
  1166. &UninitializeCompleteEvent );
  1167. //
  1168. // Now wait for the cache manager to finish purging the file.
  1169. // This will garentee that Mm gets the purge before we
  1170. // delete the Vcb.
  1171. //
  1172. WaitStatus = KeWaitForSingleObject( &UninitializeCompleteEvent.Event,
  1173. Executive,
  1174. KernelMode,
  1175. FALSE,
  1176. NULL);
  1177. ASSERT(WaitStatus == STATUS_SUCCESS);
  1178. }
  1179. VOID
  1180. FatPinMappedData (
  1181. IN PIRP_CONTEXT IrpContext,
  1182. IN PDCB Dcb,
  1183. IN VBO StartingVbo,
  1184. IN ULONG ByteCount,
  1185. OUT PBCB *Bcb
  1186. )
  1187. /*++
  1188. Routine Description:
  1189. This routine pins data that was previously mapped before setting it dirty.
  1190. Arguments:
  1191. Dcb - Pointer to the DCB for the directory
  1192. StartingVbo - The virtual offset of the first desired byte
  1193. ByteCount - Number of bytes desired
  1194. Bcb - Returns a pointer to the BCB which is valid until unpinned
  1195. --*/
  1196. {
  1197. LARGE_INTEGER Vbo;
  1198. PAGED_CODE();
  1199. DebugTrace(+1, Dbg, "FatPinMappedData\n", 0);
  1200. DebugTrace( 0, Dbg, "Dcb = %08lx\n", Dcb);
  1201. DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", StartingVbo);
  1202. DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
  1203. //
  1204. // Call the Cache manager to perform the operation.
  1205. //
  1206. Vbo.QuadPart = StartingVbo;
  1207. if (!CcPinMappedData( Dcb->Specific.Dcb.DirectoryFile,
  1208. &Vbo,
  1209. ByteCount,
  1210. BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT),
  1211. Bcb )) {
  1212. //
  1213. // Could not pin the data without waiting (cache miss).
  1214. //
  1215. FatRaiseStatus( IrpContext, STATUS_CANT_WAIT );
  1216. }
  1217. DebugTrace(-1, Dbg, "FatReadDirectoryFile -> VOID, *BCB = %08lx\n", *Bcb);
  1218. return;
  1219. }