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.

600 lines
16 KiB

  1. /*++
  2. Copyright (c) 1989-2000 Microsoft Corporation
  3. Module Name:
  4. Read.c
  5. Abstract:
  6. This module implements the File Read routine for Read called by the
  7. Fsd/Fsp dispatch drivers.
  8. // @@BEGIN_DDKSPLIT
  9. Author:
  10. Dan Lovinger [DanLo] 22-Sep-1996
  11. Tom Jolly [tomjolly] 21-Jan-2000
  12. Revision History:
  13. // @@END_DDKSPLIT
  14. --*/
  15. #include "UdfProcs.h"
  16. //
  17. // The Bug check file id for this module
  18. //
  19. #define BugCheckFileId (UDFS_BUG_CHECK_READ)
  20. //
  21. // The local debug trace level
  22. //
  23. #define Dbg (UDFS_DEBUG_LEVEL_READ)
  24. //
  25. // VOID
  26. // SafeZeroMemory (
  27. // IN PUCHAR At,
  28. // IN ULONG ByteCount
  29. // );
  30. //
  31. //
  32. // This macro just puts a nice little try-except around RtlZeroMemory
  33. //
  34. #define SafeZeroMemory(IC,AT,BYTE_COUNT) { \
  35. try { \
  36. RtlZeroMemory( (AT), (BYTE_COUNT) ); \
  37. } except( EXCEPTION_EXECUTE_HANDLER ) { \
  38. UdfRaiseStatus( IC, STATUS_INVALID_USER_BUFFER ); \
  39. } \
  40. }
  41. //
  42. // Read ahead amount used for normal data files
  43. //
  44. #define READ_AHEAD_GRANULARITY (0x10000)
  45. #ifdef ALLOC_PRAGMA
  46. #pragma alloc_text(PAGE, UdfCommonRead)
  47. #endif
  48. NTSTATUS
  49. UdfCommonRead (
  50. IN PIRP_CONTEXT IrpContext,
  51. IN PIRP Irp
  52. )
  53. /*++
  54. Routine Description:
  55. This is the common entry point for NtReadFile calls. For synchronous requests,
  56. CommonRead will complete the request in the current thread. If not
  57. synchronous the request will be passed to the Fsp if there is a need to
  58. block.
  59. Arguments:
  60. Irp - Supplies the Irp to process
  61. Return Value:
  62. NTSTATUS - The result of this operation.
  63. --*/
  64. {
  65. NTSTATUS Status;
  66. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  67. TYPE_OF_OPEN TypeOfOpen;
  68. PFCB Fcb;
  69. PCCB Ccb;
  70. PVCB Vcb;
  71. BOOLEAN Wait;
  72. ULONG PagingIo;
  73. ULONG SynchronousIo;
  74. ULONG NonCachedIo;
  75. LONGLONG StartingOffset;
  76. LONGLONG ByteRange;
  77. ULONG ByteCount;
  78. ULONG ReadByteCount;
  79. ULONG OriginalByteCount;
  80. PVOID SystemBuffer, UserBuffer;
  81. BOOLEAN ReleaseFile = TRUE;
  82. BOOLEAN ReleaseVmcbMap = FALSE;
  83. PFILE_OBJECT MappingFileObject;
  84. UDF_IO_CONTEXT LocalIoContext;
  85. PAGED_CODE();
  86. //
  87. // If this is a zero length read then return SUCCESS immediately.
  88. //
  89. if (IrpSp->Parameters.Read.Length == 0) {
  90. UdfCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
  91. return STATUS_SUCCESS;
  92. }
  93. //
  94. // Decode the file object and verify we support read on this. It
  95. // must be a user file, stream file or volume file (for a data disk).
  96. //
  97. TypeOfOpen = UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb );
  98. if ((TypeOfOpen == UnopenedFileObject) || (TypeOfOpen == UserDirectoryOpen)) {
  99. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_DEVICE_REQUEST );
  100. return STATUS_INVALID_DEVICE_REQUEST;
  101. }
  102. Vcb = Fcb->Vcb;
  103. //
  104. // Examine our input parameters to determine if this is noncached and/or
  105. // a paging io operation.
  106. //
  107. Wait = BooleanFlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
  108. PagingIo = FlagOn( Irp->Flags, IRP_PAGING_IO );
  109. NonCachedIo = FlagOn( Irp->Flags, IRP_NOCACHE );
  110. SynchronousIo = FlagOn( IrpSp->FileObject->Flags, FO_SYNCHRONOUS_IO );
  111. //
  112. // Extract the range of the Io.
  113. //
  114. StartingOffset = IrpSp->Parameters.Read.ByteOffset.QuadPart;
  115. OriginalByteCount = ByteCount = IrpSp->Parameters.Read.Length;
  116. ByteRange = StartingOffset + ByteCount;
  117. //
  118. // Make sure that Dasd access is always non-cached.
  119. //
  120. if (TypeOfOpen == UserVolumeOpen) {
  121. NonCachedIo = TRUE;
  122. }
  123. //
  124. // Acquire the file shared to perform the read. If we are doing paging IO,
  125. // it may be the case that we would have a deadlock imminent because we may
  126. // block on shared access, so starve out any exclusive waiters. This requires
  127. // a degree of caution - we believe that any paging IO bursts will recede and
  128. // allow the exclusive waiter in.
  129. //
  130. if (PagingIo) {
  131. UdfAcquireFileSharedStarveExclusive( IrpContext, Fcb );
  132. } else {
  133. UdfAcquireFileShared( IrpContext, Fcb );
  134. }
  135. //
  136. // Use a try-finally to facilitate cleanup.
  137. //
  138. try {
  139. //
  140. // Verify the Fcb. Allow reads if this is a DASD handle that is
  141. // dismounting the volume.
  142. //
  143. if ((TypeOfOpen != UserVolumeOpen) || (NULL == Ccb) ||
  144. !FlagOn( Ccb->Flags, CCB_FLAG_DISMOUNT_ON_CLOSE)) {
  145. UdfVerifyFcbOperation( IrpContext, Fcb );
  146. }
  147. //
  148. // If this is a user request then verify the oplock and filelock state.
  149. //
  150. if (TypeOfOpen == UserFileOpen) {
  151. //
  152. // We check whether we can proceed
  153. // based on the state of the file oplocks.
  154. //
  155. Status = FsRtlCheckOplock( &Fcb->Oplock,
  156. Irp,
  157. IrpContext,
  158. UdfOplockComplete,
  159. UdfPrePostIrp );
  160. //
  161. // If the result is not STATUS_SUCCESS then the Irp was completed
  162. // elsewhere.
  163. //
  164. if (Status != STATUS_SUCCESS) {
  165. Irp = NULL;
  166. IrpContext = NULL;
  167. try_leave( Status );
  168. }
  169. if (!PagingIo &&
  170. (Fcb->FileLock != NULL) &&
  171. !FsRtlCheckLockForReadAccess( Fcb->FileLock, Irp )) {
  172. try_leave( Status = STATUS_FILE_LOCK_CONFLICT );
  173. }
  174. }
  175. if ((TypeOfOpen != UserVolumeOpen) || !FlagOn( Ccb->Flags, CCB_FLAG_ALLOW_EXTENDED_DASD_IO )) {
  176. //
  177. // Complete the request if it begins beyond the end of file.
  178. //
  179. if (StartingOffset >= Fcb->FileSize.QuadPart) {
  180. try_leave( Status = STATUS_END_OF_FILE );
  181. }
  182. //
  183. // Truncate the read if it extends beyond the end of the file.
  184. //
  185. if (ByteRange > Fcb->FileSize.QuadPart) {
  186. ByteCount = (ULONG) (Fcb->FileSize.QuadPart - StartingOffset);
  187. ByteRange = Fcb->FileSize.QuadPart;
  188. }
  189. }
  190. //
  191. // Now if the data is embedded in the ICB, map through the metadata
  192. // stream to retrieve the bytes.
  193. //
  194. if (FlagOn( Fcb->FcbState, FCB_STATE_EMBEDDED_DATA )) {
  195. //
  196. // The metadata stream better be here by now.
  197. //
  198. ASSERT( Vcb->MetadataFcb->FileObject != NULL );
  199. //
  200. // Bias our starting offset by the offset of the ICB in the metadata
  201. // stream plus the offset of the data bytes in that ICB. Obviously,
  202. // we aren't doing non-cached IO here.
  203. //
  204. StartingOffset += (BytesFromSectors( Vcb, Fcb->EmbeddedVsn ) + Fcb->EmbeddedOffset);
  205. MappingFileObject = Vcb->MetadataFcb->FileObject;
  206. NonCachedIo = FALSE;
  207. //
  208. // Ensure that we're mapping within the range of the metadata stream
  209. //
  210. ASSERT( (StartingOffset + ByteCount) <= Vcb->MetadataFcb->FileSize.QuadPart);
  211. } else {
  212. //
  213. // We are mapping through the caller's fileobject
  214. //
  215. MappingFileObject = IrpSp->FileObject;
  216. }
  217. //
  218. // Handle the non-cached read first.
  219. //
  220. if (NonCachedIo) {
  221. //
  222. // If we have an unaligned transfer then post this request if
  223. // we can't wait. Unaligned means that the starting offset
  224. // is not on a sector boundary or the read is not integral
  225. // sectors.
  226. //
  227. ReadByteCount = SectorAlign( Vcb, ByteCount );
  228. if (SectorOffset( Vcb, StartingOffset ) ||
  229. (ReadByteCount > OriginalByteCount)) {
  230. if (!Wait) {
  231. UdfRaiseStatus( IrpContext, STATUS_CANT_WAIT );
  232. }
  233. //
  234. // Make sure we don't overwrite the buffer.
  235. //
  236. ReadByteCount = ByteCount;
  237. }
  238. //
  239. // Initialize the IoContext for the read.
  240. // If there is a context pointer, we need to make sure it was
  241. // allocated and not a stale stack pointer.
  242. //
  243. if (IrpContext->IoContext == NULL ||
  244. !FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO )) {
  245. //
  246. // If we can wait, use the context on the stack. Otherwise
  247. // we need to allocate one.
  248. //
  249. if (Wait) {
  250. IrpContext->IoContext = &LocalIoContext;
  251. ClearFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO );
  252. } else {
  253. IrpContext->IoContext = UdfAllocateIoContext();
  254. SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO );
  255. }
  256. }
  257. RtlZeroMemory( IrpContext->IoContext, sizeof( UDF_IO_CONTEXT ));
  258. //
  259. // Store whether we allocated this context structure in the structure
  260. // itself.
  261. //
  262. IrpContext->IoContext->AllocatedContext =
  263. BooleanFlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO );
  264. if (Wait) {
  265. KeInitializeEvent( &IrpContext->IoContext->SyncEvent,
  266. NotificationEvent,
  267. FALSE );
  268. } else {
  269. IrpContext->IoContext->ResourceThreadId = ExGetCurrentResourceThread();
  270. IrpContext->IoContext->Resource = Fcb->Resource;
  271. IrpContext->IoContext->RequestedByteCount = ByteCount;
  272. }
  273. Irp->IoStatus.Information = ReadByteCount;
  274. //
  275. // Call the NonCacheIo routine to perform the actual read.
  276. //
  277. Status = UdfNonCachedRead( IrpContext, Fcb, Ccb, StartingOffset, ReadByteCount );
  278. //
  279. // Don't complete this request now if STATUS_PENDING was returned.
  280. //
  281. if (Status == STATUS_PENDING) {
  282. Irp = NULL;
  283. ReleaseFile = FALSE;
  284. //
  285. // Test is we should zero part of the buffer or update the
  286. // synchronous file position.
  287. //
  288. } else {
  289. //
  290. // Convert any unknown error code to IO_ERROR.
  291. //
  292. if (!NT_SUCCESS( Status )) {
  293. //
  294. // Set the information field to zero.
  295. //
  296. Irp->IoStatus.Information = 0;
  297. //
  298. // Raise if this is a user induced error.
  299. //
  300. if (IoIsErrorUserInduced( Status )) {
  301. UdfRaiseStatus( IrpContext, Status );
  302. }
  303. Status = FsRtlNormalizeNtstatus( Status, STATUS_UNEXPECTED_IO_ERROR );
  304. //
  305. // Check if there is any portion of the user's buffer to zero.
  306. //
  307. } else if (ReadByteCount != ByteCount) {
  308. UdfMapUserBuffer( IrpContext, &UserBuffer );
  309. SafeZeroMemory( IrpContext,
  310. Add2Ptr( UserBuffer,
  311. ByteCount,
  312. PVOID ),
  313. ReadByteCount - ByteCount );
  314. Irp->IoStatus.Information = ByteCount;
  315. }
  316. //
  317. // Update the file position if this is a synchronous request.
  318. //
  319. if (SynchronousIo && !PagingIo && NT_SUCCESS( Status )) {
  320. IrpSp->FileObject->CurrentByteOffset.QuadPart = ByteRange;
  321. }
  322. }
  323. try_leave( NOTHING );
  324. }
  325. //
  326. // Handle the cached case. Start by initializing the private
  327. // cache map.
  328. //
  329. if (MappingFileObject->PrivateCacheMap == NULL) {
  330. //
  331. // The metadata Fcb stream was fired up before any data read. We should never
  332. // see it here.
  333. //
  334. ASSERT( MappingFileObject != Vcb->MetadataFcb->FileObject );
  335. //
  336. // Now initialize the cache map.
  337. //
  338. CcInitializeCacheMap( IrpSp->FileObject,
  339. (PCC_FILE_SIZES) &Fcb->AllocationSize,
  340. FALSE,
  341. &UdfData.CacheManagerCallbacks,
  342. Fcb );
  343. CcSetReadAheadGranularity( IrpSp->FileObject, READ_AHEAD_GRANULARITY );
  344. }
  345. //
  346. // Read from the cache if this is not an Mdl read.
  347. //
  348. if (!FlagOn( IrpContext->MinorFunction, IRP_MN_MDL )) {
  349. //
  350. // If we are in the Fsp now because we had to wait earlier,
  351. // we must map the user buffer, otherwise we can use the
  352. // user's buffer directly.
  353. //
  354. UdfMapUserBuffer( IrpContext, &SystemBuffer);
  355. //
  356. // Now try to do the copy.
  357. //
  358. if (MappingFileObject == Vcb->MetadataFcb->FileObject) {
  359. UdfAcquireVmcbForCcMap( IrpContext, Vcb);
  360. ReleaseVmcbMap = TRUE;
  361. }
  362. if (!CcCopyRead( MappingFileObject,
  363. (PLARGE_INTEGER) &StartingOffset,
  364. ByteCount,
  365. Wait,
  366. SystemBuffer,
  367. &Irp->IoStatus )) {
  368. try_leave( Status = STATUS_CANT_WAIT );
  369. }
  370. //
  371. // If the call didn't succeed, raise the error status
  372. //
  373. if (!NT_SUCCESS( Irp->IoStatus.Status )) {
  374. UdfNormalizeAndRaiseStatus( IrpContext, Irp->IoStatus.Status );
  375. }
  376. Status = Irp->IoStatus.Status;
  377. //
  378. // Otherwise perform the MdlRead operation.
  379. //
  380. } else {
  381. CcMdlRead( MappingFileObject,
  382. (PLARGE_INTEGER) &StartingOffset,
  383. ByteCount,
  384. &Irp->MdlAddress,
  385. &Irp->IoStatus );
  386. Status = Irp->IoStatus.Status;
  387. }
  388. //
  389. // Update the current file position in the user file object.
  390. //
  391. if (SynchronousIo && !PagingIo && NT_SUCCESS( Status )) {
  392. IrpSp->FileObject->CurrentByteOffset.QuadPart = ByteRange;
  393. }
  394. } finally {
  395. DebugUnwind( "UdfCommonRead" );
  396. //
  397. // Release the Fcb / Vmcb mapping resource
  398. //
  399. if (ReleaseFile) { UdfReleaseFile( IrpContext, Fcb ); }
  400. if (ReleaseVmcbMap) { UdfReleaseVmcb( IrpContext, Vcb); }
  401. }
  402. //
  403. // Post the request if we got CANT_WAIT.
  404. //
  405. if (Status == STATUS_CANT_WAIT) {
  406. Status = UdfFsdPostRequest( IrpContext, Irp );
  407. //
  408. // Otherwise complete the request.
  409. //
  410. } else {
  411. UdfCompleteRequest( IrpContext, Irp, Status );
  412. }
  413. return Status;
  414. }