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.

2099 lines
60 KiB

  1. /*++
  2. Copyright (c) 1989-2000 Microsoft Corporation
  3. Module Name:
  4. DirCtrl.c
  5. Abstract:
  6. This module implements the File Directory Control routines for Udfs called
  7. by the Fsd/Fsp dispatch drivers.
  8. // @@BEGIN_DDKSPLIT
  9. Author:
  10. Dan Lovinger [DanLo] 27-Nov-1996
  11. Revision History:
  12. Tom Jolly [TomJolly] 1-March-2000 UDF 2.01 support
  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_DIRCTRL)
  20. //
  21. // The local debug trace level
  22. //
  23. #define Dbg (UDFS_DEBUG_LEVEL_DIRCTRL)
  24. //
  25. // Local structures
  26. //
  27. //
  28. // The following is used for the more complete enumeration required in the DirectoryControl path
  29. // and encapsulates the structures for enumerating both directories and ICBs, as well as converted
  30. // data from the ICB.
  31. //
  32. typedef struct _COMPOUND_DIR_ENUM_CONTEXT {
  33. //
  34. // Standard enumeration contexts. For this enumeration we walk the directory and lift the
  35. // associated ICB for each entry.
  36. //
  37. DIR_ENUM_CONTEXT DirContext;
  38. ICB_SEARCH_CONTEXT IcbContext;
  39. //
  40. // Timestamps converted from the ICB into NT-native form.
  41. //
  42. TIMESTAMP_BUNDLE Timestamps;
  43. //
  44. // File index corresponding to the current position in the enumeration.
  45. //
  46. LARGE_INTEGER FileIndex;
  47. } COMPOUND_DIR_ENUM_CONTEXT, *PCOMPOUND_DIR_ENUM_CONTEXT;
  48. //
  49. // Local macros
  50. //
  51. //
  52. // Constants defining the space of FileIndices for directory enumeration.
  53. //
  54. //
  55. // The virtual (synthesized) file indices
  56. //
  57. #define UDF_FILE_INDEX_VIRTUAL_SELF 0
  58. //
  59. // The file index where the physical directory entries begin
  60. //
  61. #define UDF_FILE_INDEX_PHYSICAL 1
  62. //
  63. // Provide initialization and cleanup for compound enumeration contexts.
  64. //
  65. INLINE
  66. VOID
  67. UdfInitializeCompoundDirContext (
  68. IN PIRP_CONTEXT IrpContext,
  69. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext
  70. )
  71. {
  72. UdfInitializeDirContext( IrpContext, &CompoundDirContext->DirContext );
  73. UdfFastInitializeIcbContext( IrpContext, &CompoundDirContext->IcbContext );
  74. RtlZeroMemory( &CompoundDirContext->Timestamps, sizeof( TIMESTAMP_BUNDLE ));
  75. CompoundDirContext->FileIndex.QuadPart = 0;
  76. }
  77. INLINE
  78. VOID
  79. UdfCleanupCompoundDirContext (
  80. IN PIRP_CONTEXT IrpContext,
  81. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext
  82. )
  83. {
  84. UdfCleanupDirContext( IrpContext, &CompoundDirContext->DirContext );
  85. UdfCleanupIcbContext( IrpContext, &CompoundDirContext->IcbContext );
  86. }
  87. //
  88. // UDF directories are unsorted (UDF 1.0.1 2.3.5.3) and do not contain self
  89. // entries. For directory enumeration we must provide a way for a restart to
  90. // occur at a random entry (SL_INDEX_SPECIFIED), but the key used is only
  91. // 32bits. Since the directory is unsorted, the filename is unsuitable for
  92. // quickly finding a restart point (even assuming that it was sorted,
  93. // discovering a directory entry is still not fast). Additionally, we must
  94. // synthesize the self-entry. So, here is how we map the space of file
  95. // indices to directory entries:
  96. //
  97. // File Index Directory Entry
  98. //
  99. // 0 self ('.')
  100. // 1 at byte offset 0 in the stream
  101. // N at byte offset N-1 in the stream
  102. //
  103. // The highest 32bit FileIndex returned will be stashed in the Ccb.
  104. //
  105. // For FileIndex > 2^32, we will return FileIndex 0 in the query structure.
  106. // On a restart, we will notice a FileIndex of zero and use the saved high
  107. // 32bit FileIndex as the starting point for a linear scan to find the named
  108. // directory entry in the restart request. In this way we only penalize the
  109. // improbable case of a directory stream > 2^32 bytes.
  110. //
  111. // The following inline routines assist with this mapping.
  112. //
  113. INLINE
  114. LONGLONG
  115. UdfFileIndexToPhysicalOffset(
  116. LONGLONG FileIndex
  117. )
  118. {
  119. return FileIndex - UDF_FILE_INDEX_PHYSICAL;
  120. }
  121. INLINE
  122. LONGLONG
  123. UdfPhysicalOffsetToFileIndex(
  124. LONGLONG PhysicalOffset
  125. )
  126. {
  127. return PhysicalOffset + UDF_FILE_INDEX_PHYSICAL;
  128. }
  129. INLINE
  130. BOOLEAN
  131. UdfIsFileIndexVirtual(
  132. LONGLONG FileIndex
  133. )
  134. {
  135. return FileIndex < UDF_FILE_INDEX_PHYSICAL;
  136. }
  137. //
  138. // Local support routines
  139. //
  140. NTSTATUS
  141. UdfQueryDirectory (
  142. IN PIRP_CONTEXT IrpContext,
  143. IN PIRP Irp,
  144. IN PIO_STACK_LOCATION IrpSp,
  145. IN PFCB Fcb,
  146. IN PCCB Ccb
  147. );
  148. NTSTATUS
  149. UdfNotifyChangeDirectory (
  150. IN PIRP_CONTEXT IrpContext,
  151. IN PIRP Irp,
  152. IN PIO_STACK_LOCATION IrpSp,
  153. IN PCCB Ccb
  154. );
  155. NTSTATUS
  156. UdfInitializeEnumeration (
  157. IN PIRP_CONTEXT IrpContext,
  158. IN PIO_STACK_LOCATION IrpSp,
  159. IN PFCB Fcb,
  160. IN OUT PCCB Ccb,
  161. IN OUT PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext,
  162. OUT PBOOLEAN ReturnNextEntry,
  163. OUT PBOOLEAN ReturnSingleEntry,
  164. OUT PBOOLEAN InitialQuery
  165. );
  166. BOOLEAN
  167. UdfEnumerateIndex (
  168. IN PIRP_CONTEXT IrpContext,
  169. IN PCCB Ccb,
  170. IN OUT PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext,
  171. IN BOOLEAN ReturnNextEntry
  172. );
  173. VOID
  174. UdfLookupFileEntryInEnumeration (
  175. IN PIRP_CONTEXT IrpContext,
  176. IN PFCB Fcb,
  177. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext
  178. );
  179. BOOLEAN
  180. UdfLookupInitialFileIndex (
  181. IN PIRP_CONTEXT IrpContext,
  182. IN PFCB Fcb,
  183. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext,
  184. IN PLONGLONG InitialIndex
  185. );
  186. BOOLEAN
  187. UdfLookupNextFileIndex (
  188. IN PIRP_CONTEXT IrpContext,
  189. IN PFCB Fcb,
  190. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext
  191. );
  192. #ifdef ALLOC_PRAGMA
  193. #pragma alloc_text(PAGE, UdfCommonDirControl)
  194. #pragma alloc_text(PAGE, UdfEnumerateIndex)
  195. #pragma alloc_text(PAGE, UdfInitializeEnumeration)
  196. #pragma alloc_text(PAGE, UdfLookupFileEntryInEnumeration)
  197. #pragma alloc_text(PAGE, UdfLookupInitialFileIndex)
  198. #pragma alloc_text(PAGE, UdfLookupNextFileIndex)
  199. #pragma alloc_text(PAGE, UdfNotifyChangeDirectory)
  200. #pragma alloc_text(PAGE, UdfQueryDirectory)
  201. #endif
  202. NTSTATUS
  203. UdfCommonDirControl (
  204. IN PIRP_CONTEXT IrpContext,
  205. IN PIRP Irp
  206. )
  207. /*++
  208. Routine Description:
  209. This routine is the entry point for the directory control operations. These
  210. are directory enumerations and directory notify calls. We verify the
  211. user's handle is for a directory and then call the appropriate routine.
  212. Arguments:
  213. Irp - Irp for this request.
  214. Return Value:
  215. NTSTATUS - Status returned from the lower level routines.
  216. --*/
  217. {
  218. NTSTATUS Status;
  219. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  220. PFCB Fcb;
  221. PCCB Ccb;
  222. PAGED_CODE();
  223. //
  224. // Decode the user file object and fail this request if it is not
  225. // a user directory.
  226. //
  227. if (UdfDecodeFileObject( IrpSp->FileObject,
  228. &Fcb,
  229. &Ccb ) != UserDirectoryOpen) {
  230. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  231. return STATUS_INVALID_PARAMETER;
  232. }
  233. //
  234. // We know this is a directory control so we'll case on the
  235. // minor function, and call a internal worker routine to complete
  236. // the irp.
  237. //
  238. switch (IrpSp->MinorFunction) {
  239. case IRP_MN_QUERY_DIRECTORY:
  240. Status = UdfQueryDirectory( IrpContext, Irp, IrpSp, Fcb, Ccb );
  241. break;
  242. case IRP_MN_NOTIFY_CHANGE_DIRECTORY:
  243. Status = UdfNotifyChangeDirectory( IrpContext, Irp, IrpSp, Ccb );
  244. break;
  245. default:
  246. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_DEVICE_REQUEST );
  247. Status = STATUS_INVALID_DEVICE_REQUEST;
  248. break;
  249. }
  250. return Status;
  251. }
  252. //
  253. // Local support routines
  254. //
  255. NTSTATUS
  256. UdfQueryDirectory (
  257. IN PIRP_CONTEXT IrpContext,
  258. IN PIRP Irp,
  259. IN PIO_STACK_LOCATION IrpSp,
  260. IN PFCB Fcb,
  261. IN PCCB Ccb
  262. )
  263. /*++
  264. Routine Description:
  265. This routine performs the query directory operation. It is responsible
  266. for either completing of enqueuing the input Irp. We store the state of the
  267. search in the Ccb.
  268. Arguments:
  269. Irp - Supplies the Irp to process
  270. IrpSp - Stack location for this Irp.
  271. Fcb - Fcb for this directory.
  272. Ccb - Ccb for this directory open.
  273. Return Value:
  274. NTSTATUS - The return status for the operation
  275. --*/
  276. {
  277. NTSTATUS Status = STATUS_SUCCESS;
  278. ULONG Information = 0;
  279. ULONG LastEntry = 0;
  280. ULONG NextEntry = 0;
  281. ULONG FileNameBytes;
  282. ULONG BytesConverted;
  283. LARGE_INTEGER PreviousFileIndex;
  284. COMPOUND_DIR_ENUM_CONTEXT CompoundDirContext;
  285. PNSR_FID ThisFid;
  286. PICBFILE ThisFe;
  287. BOOLEAN InitialQuery;
  288. BOOLEAN ReturnNextEntry;
  289. BOOLEAN ReturnSingleEntry;
  290. BOOLEAN Found;
  291. BOOLEAN EasCorrupt;
  292. PCHAR UserBuffer;
  293. ULONG BytesRemainingInBuffer;
  294. ULONG BaseLength;
  295. PFILE_BOTH_DIR_INFORMATION DirInfo;
  296. PFILE_NAMES_INFORMATION NamesInfo;
  297. PFILE_ID_FULL_DIR_INFORMATION IdFullDirInfo;
  298. PFILE_ID_BOTH_DIR_INFORMATION IdBothDirInfo;
  299. PAGED_CODE();
  300. DebugTrace(( 0, Dbg, "UdfQueryDirectory\n" ));
  301. //
  302. // Check if we support this search mode. Also remember the size of the base part of
  303. // each of these structures.
  304. //
  305. switch (IrpSp->Parameters.QueryDirectory.FileInformationClass) {
  306. case FileDirectoryInformation:
  307. BaseLength = FIELD_OFFSET( FILE_DIRECTORY_INFORMATION,
  308. FileName[0] );
  309. break;
  310. case FileFullDirectoryInformation:
  311. BaseLength = FIELD_OFFSET( FILE_FULL_DIR_INFORMATION,
  312. FileName[0] );
  313. break;
  314. case FileIdFullDirectoryInformation:
  315. BaseLength = FIELD_OFFSET( FILE_ID_FULL_DIR_INFORMATION,
  316. FileName[0] );
  317. break;
  318. case FileNamesInformation:
  319. BaseLength = FIELD_OFFSET( FILE_NAMES_INFORMATION,
  320. FileName[0] );
  321. break;
  322. case FileBothDirectoryInformation:
  323. BaseLength = FIELD_OFFSET( FILE_BOTH_DIR_INFORMATION,
  324. FileName[0] );
  325. break;
  326. case FileIdBothDirectoryInformation:
  327. BaseLength = FIELD_OFFSET( FILE_ID_BOTH_DIR_INFORMATION,
  328. FileName[0] );
  329. break;
  330. default:
  331. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_INFO_CLASS );
  332. return STATUS_INVALID_INFO_CLASS;
  333. }
  334. //
  335. // Get the user buffer.
  336. //
  337. UdfMapUserBuffer( IrpContext, &UserBuffer);
  338. //
  339. // Initialize our search context.
  340. //
  341. UdfInitializeCompoundDirContext( IrpContext, &CompoundDirContext );
  342. //
  343. // Acquire the directory.
  344. //
  345. UdfAcquireFileShared( IrpContext, Fcb );
  346. //
  347. // Use a try-finally to facilitate cleanup.
  348. //
  349. try {
  350. //
  351. // Verify the Fcb is still good.
  352. //
  353. UdfVerifyFcbOperation( IrpContext, Fcb );
  354. //
  355. // Start by getting the initial state for the enumeration. This will set up the Ccb with
  356. // the initial search parameters and let us know the starting offset in the directory
  357. // to search.
  358. //
  359. Status = UdfInitializeEnumeration( IrpContext,
  360. IrpSp,
  361. Fcb,
  362. Ccb,
  363. &CompoundDirContext,
  364. &ReturnNextEntry,
  365. &ReturnSingleEntry,
  366. &InitialQuery );
  367. if (!NT_SUCCESS( Status )) {
  368. try_leave( Status );
  369. }
  370. //
  371. // At this point we are about to enter our query loop. We have
  372. // determined the index into the directory file to begin the
  373. // search. LastEntry and NextEntry are used to index into the user
  374. // buffer. LastEntry is the last entry we've added, NextEntry is
  375. // current one we're working on. If NextEntry is non-zero, then
  376. // at least one entry was added.
  377. //
  378. while (TRUE) {
  379. //
  380. // If the user had requested only a single match and we have
  381. // returned that, then we stop at this point. We update the Ccb with
  382. // the status based on the last entry returned.
  383. //
  384. if ((NextEntry != 0) && ReturnSingleEntry) {
  385. try_leave( Status );
  386. }
  387. //
  388. // We try to locate the next matching dirent. Our search if based on a starting
  389. // dirent offset, whether we should return the current or next entry, whether
  390. // we should be doing a short name search and finally whether we should be
  391. // checking for a version match.
  392. //
  393. PreviousFileIndex = CompoundDirContext.FileIndex;
  394. try {
  395. Found = UdfEnumerateIndex( IrpContext, Ccb, &CompoundDirContext, ReturnNextEntry );
  396. }
  397. except (((0 != NextEntry) &&
  398. ((GetExceptionCode() == STATUS_FILE_CORRUPT_ERROR) ||
  399. (GetExceptionCode() == STATUS_CRC_ERROR)))
  400. ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) {
  401. DebugTrace((0, Dbg, "UdfQueryDirectory - Corrupt. Returning buffer so far, setting back enumeration\n"));
  402. //
  403. // We encountered corruption in the directory. We will swallow this
  404. // error since we have already placed some previous entries in the user
  405. // buffer, and raise it when we're called again. This is
  406. // to return as much as possible in the case of corrupt directories,
  407. // particularly discs which are padded wrong at the end of the dir.
  408. //
  409. ReturnNextEntry = TRUE;
  410. //
  411. // Point to the previous Fid, so that we will advance again to the point
  412. // of corruption on next call (we only do bounds checking on advance, not
  413. // when restarting *at* a particular entry).
  414. //
  415. CompoundDirContext.FileIndex = PreviousFileIndex;
  416. try_leave( Status = STATUS_SUCCESS);
  417. }
  418. //
  419. // Initialize the value for the next search.
  420. //
  421. ReturnNextEntry = TRUE;
  422. //
  423. // If we didn't receive a dirent, then we are at the end of the
  424. // directory. If we have returned any files, we exit with
  425. // success, otherwise we return STATUS_NO_MORE_FILES.
  426. //
  427. if (!Found) {
  428. if (NextEntry == 0) {
  429. Status = STATUS_NO_MORE_FILES;
  430. if (InitialQuery) {
  431. Status = STATUS_NO_SUCH_FILE;
  432. }
  433. }
  434. try_leave( Status );
  435. }
  436. //
  437. // Remember the dirent/file entry for the file we just found.
  438. //
  439. ThisFid = CompoundDirContext.DirContext.Fid;
  440. //
  441. // Here are the rules concerning filling up the buffer:
  442. //
  443. // 1. The Io system garentees that there will always be
  444. // enough room for at least one base record.
  445. //
  446. // 2. If the full first record (including file name) cannot
  447. // fit, as much of the name as possible is copied and
  448. // STATUS_BUFFER_OVERFLOW is returned.
  449. //
  450. // 3. If a subsequent record cannot completely fit into the
  451. // buffer, none of it (as in 0 bytes) is copied, and
  452. // STATUS_SUCCESS is returned. A subsequent query will
  453. // pick up with this record.
  454. //
  455. //
  456. // We can look directly at the dirent that we found.
  457. //
  458. FileNameBytes = CompoundDirContext.DirContext.CaseObjectName.Length;
  459. //
  460. // If the slot for the next entry would be beyond the length of the
  461. // user's buffer just exit (we know we've returned at least one entry
  462. // already). This will happen when we align the pointer past the end.
  463. //
  464. if (NextEntry > IrpSp->Parameters.QueryDirectory.Length) {
  465. ReturnNextEntry = FALSE;
  466. try_leave( Status = STATUS_SUCCESS );
  467. }
  468. //
  469. // Compute the number of bytes remaining in the buffer. Round this
  470. // down to a WCHAR boundary so we can copy full characters.
  471. //
  472. BytesRemainingInBuffer = IrpSp->Parameters.QueryDirectory.Length - NextEntry;
  473. ClearFlag( BytesRemainingInBuffer, 1 );
  474. //
  475. // If this won't fit and we have returned a previous entry then just
  476. // return STATUS_SUCCESS.
  477. //
  478. if ((BaseLength + FileNameBytes) > BytesRemainingInBuffer) {
  479. //
  480. // If we already found an entry then just exit.
  481. //
  482. if (NextEntry != 0) {
  483. ReturnNextEntry = FALSE;
  484. try_leave( Status = STATUS_SUCCESS );
  485. }
  486. //
  487. // Reduce the FileNameBytes to just fit in the buffer.
  488. //
  489. FileNameBytes = BytesRemainingInBuffer - BaseLength;
  490. //
  491. // Use a status code of STATUS_BUFFER_OVERFLOW. Also set
  492. // ReturnSingleEntry so that we will exit the loop at the top.
  493. //
  494. Status = STATUS_BUFFER_OVERFLOW;
  495. ReturnSingleEntry = TRUE;
  496. }
  497. //
  498. // Protect access to the user buffer with an exception handler.
  499. // Since (at our request) IO doesn't buffer these requests, we have
  500. // to guard against a user messing with the page protection and other
  501. // such trickery.
  502. //
  503. try {
  504. //
  505. // Zero and initialize the base part of the current entry.
  506. //
  507. RtlZeroMemory( Add2Ptr( UserBuffer, NextEntry, PVOID ),
  508. BaseLength );
  509. //
  510. // Now we have an entry to return to our caller.
  511. // We'll case on the type of information requested and fill up
  512. // the user buffer if everything fits.
  513. //
  514. switch (IrpSp->Parameters.QueryDirectory.FileInformationClass) {
  515. case FileBothDirectoryInformation:
  516. case FileFullDirectoryInformation:
  517. case FileIdBothDirectoryInformation:
  518. case FileIdFullDirectoryInformation:
  519. case FileDirectoryInformation:
  520. DirInfo = Add2Ptr( UserBuffer, NextEntry, PFILE_BOTH_DIR_INFORMATION );
  521. //
  522. // These information types require we look up the file entry.
  523. // We will swallow certain corruption errors here in the interest of
  524. // allowing users to access other objects in the directory. The
  525. // errors will be reported later if the corrupt object is opened.
  526. //
  527. EasCorrupt = FALSE;
  528. ThisFe = NULL;
  529. try {
  530. UdfLookupFileEntryInEnumeration( IrpContext,
  531. Fcb,
  532. &CompoundDirContext );
  533. //
  534. // Directly reference the file entry we just looked up.
  535. //
  536. ThisFe = (PICBFILE) CompoundDirContext.IcbContext.Active.View;
  537. //
  538. // Now go gather all of the timestamps for this guy.
  539. //
  540. UdfUpdateTimestampsFromIcbContext ( IrpContext,
  541. &CompoundDirContext.IcbContext,
  542. &CompoundDirContext.Timestamps );
  543. }
  544. except (UdfQueryDirExceptionFilter( GetExceptionInformation())) {
  545. //
  546. // The currently mapped ICB will have been left in IcbContext->Current,
  547. // and we could look at it and pull out timestamps / filesizes,
  548. // but it could be complete trash, so we'll just zero these fields
  549. // in this dir record.
  550. //
  551. DebugTrace(( 0, Dbg, "Ignoring corrupt FE (referenced by FID in dir FCB 0x%p) during dir enum\n", Fcb));
  552. //
  553. // We either failed verification of the core FE fields, or looking
  554. // up the EAs for timestamps. Either way, the EAs are definitely dead
  555. // which means the create time is invalid
  556. //
  557. EasCorrupt = TRUE;
  558. IrpContext->ExceptionStatus = STATUS_SUCCESS;
  559. }
  560. //
  561. // If we actually have an FE here that means that the core content verified
  562. // ok, so pull out the relevant information.
  563. //
  564. if (NULL != ThisFe) {
  565. DirInfo->LastWriteTime =
  566. DirInfo->ChangeTime = CompoundDirContext.Timestamps.ModificationTime;
  567. DirInfo->LastAccessTime = CompoundDirContext.Timestamps.AccessTime;
  568. if (!EasCorrupt) {
  569. DirInfo->CreationTime = CompoundDirContext.Timestamps.CreationTime;
  570. }
  571. else {
  572. DirInfo->CreationTime = UdfCorruptFileTime;
  573. }
  574. //
  575. // Set the attributes and sizes separately for directories and
  576. // files.
  577. //
  578. if (ThisFe->Icbtag.FileType == ICBTAG_FILE_T_DIRECTORY) {
  579. DirInfo->EndOfFile.QuadPart = DirInfo->AllocationSize.QuadPart = 0;
  580. SetFlag( DirInfo->FileAttributes, FILE_ATTRIBUTE_DIRECTORY );
  581. } else {
  582. DirInfo->EndOfFile.QuadPart = ThisFe->InfoLength;
  583. DirInfo->AllocationSize.QuadPart = LlBlockAlign( Fcb->Vcb, ThisFe->InfoLength );
  584. }
  585. }
  586. else {
  587. //
  588. // FE is corrupt. Fill in (irrelevant) valid times.
  589. //
  590. DirInfo->CreationTime =
  591. DirInfo->ChangeTime =
  592. DirInfo->LastWriteTime =
  593. DirInfo->LastAccessTime = UdfCorruptFileTime;
  594. }
  595. //
  596. // All Cdrom files are readonly. We also copy the existence
  597. // bit to the hidden attribute, assuming that synthesized FIDs
  598. // are never hidden.
  599. //
  600. SetFlag( DirInfo->FileAttributes, FILE_ATTRIBUTE_READONLY );
  601. if (ThisFid && FlagOn( ThisFid->Flags, NSR_FID_F_HIDDEN )) {
  602. SetFlag( DirInfo->FileAttributes, FILE_ATTRIBUTE_HIDDEN );
  603. }
  604. //
  605. // The file index for real file indices > 2^32 is zero. When asked to
  606. // restart at an index of zero, we will know to use a stashed starting
  607. // point to beging to search, by name, for the correct restart point.
  608. //
  609. if (CompoundDirContext.FileIndex.HighPart == 0) {
  610. DirInfo->FileIndex = CompoundDirContext.FileIndex.LowPart;
  611. } else {
  612. DirInfo->FileIndex = 0;
  613. }
  614. DirInfo->FileNameLength = FileNameBytes;
  615. break;
  616. case FileNamesInformation:
  617. NamesInfo = Add2Ptr( UserBuffer, NextEntry, PFILE_NAMES_INFORMATION );
  618. if (CompoundDirContext.FileIndex.HighPart == 0) {
  619. NamesInfo->FileIndex = CompoundDirContext.FileIndex.LowPart;
  620. } else {
  621. NamesInfo->FileIndex = 0;
  622. }
  623. NamesInfo->FileNameLength = FileNameBytes;
  624. break;
  625. }
  626. //
  627. // Fill in the FileId
  628. //
  629. switch (IrpSp->Parameters.QueryDirectory.FileInformationClass) {
  630. case FileIdBothDirectoryInformation:
  631. IdBothDirInfo = Add2Ptr( UserBuffer, NextEntry, PFILE_ID_BOTH_DIR_INFORMATION );
  632. UdfSetFidFromFidAndFe( IdBothDirInfo->FileId, ThisFid, ThisFe );
  633. break;
  634. case FileIdFullDirectoryInformation:
  635. IdFullDirInfo = Add2Ptr( UserBuffer, NextEntry, PFILE_ID_FULL_DIR_INFORMATION );
  636. UdfSetFidFromFidAndFe( IdFullDirInfo->FileId, ThisFid, ThisFe );
  637. break;
  638. default:
  639. break;
  640. }
  641. //
  642. // Now copy as much of the name as possible.
  643. //
  644. if (FileNameBytes != 0) {
  645. //
  646. // This is a Unicode name, we can copy the bytes directly.
  647. //
  648. RtlCopyMemory( Add2Ptr( UserBuffer, NextEntry + BaseLength, PVOID ),
  649. CompoundDirContext.DirContext.ObjectName.Buffer,
  650. FileNameBytes );
  651. }
  652. //
  653. // Fill in the short name if we got STATUS_SUCCESS. The short name
  654. // may already be in the file context, otherwise we will check
  655. // whether the long name is 8.3. Special case the self and parent
  656. // directory names.
  657. //
  658. if ((Status == STATUS_SUCCESS) &&
  659. (IrpSp->Parameters.QueryDirectory.FileInformationClass == FileBothDirectoryInformation ||
  660. IrpSp->Parameters.QueryDirectory.FileInformationClass == FileIdBothDirectoryInformation) &&
  661. FlagOn( CompoundDirContext.DirContext.Flags, DIR_CONTEXT_FLAG_SEEN_NONCONSTANT )) {
  662. //
  663. // If we already have the short name then copy into the user's buffer.
  664. //
  665. if (CompoundDirContext.DirContext.ShortObjectName.Length != 0) {
  666. RtlCopyMemory( DirInfo->ShortName,
  667. CompoundDirContext.DirContext.ShortObjectName.Buffer,
  668. CompoundDirContext.DirContext.ShortObjectName.Length );
  669. DirInfo->ShortNameLength = (CCHAR) CompoundDirContext.DirContext.ShortObjectName.Length;
  670. //
  671. // If the short name length is currently zero then check if
  672. // the long name is not 8.3. We can copy the short name in
  673. // unicode form directly into the caller's buffer.
  674. //
  675. } else {
  676. if (!UdfIs8dot3Name( IrpContext,
  677. CompoundDirContext.DirContext.ObjectName )) {
  678. UNICODE_STRING ShortName;
  679. ShortName.Buffer = DirInfo->ShortName;
  680. ShortName.MaximumLength = BYTE_COUNT_8_DOT_3;
  681. UdfGenerate8dot3Name( IrpContext,
  682. &CompoundDirContext.DirContext.PureObjectName,
  683. &ShortName );
  684. DirInfo->ShortNameLength = (CCHAR) ShortName.Length;
  685. }
  686. }
  687. }
  688. //
  689. // Update the information with the number of bytes stored in the
  690. // buffer. We quad-align the existing buffer to add any necessary
  691. // pad bytes.
  692. //
  693. Information = NextEntry + BaseLength + FileNameBytes;
  694. //
  695. // Go back to the previous entry and fill in the update to this entry.
  696. //
  697. *(Add2Ptr( UserBuffer, LastEntry, PULONG )) = NextEntry - LastEntry;
  698. //
  699. // Set up our variables for the next dirent.
  700. //
  701. InitialQuery = FALSE;
  702. LastEntry = NextEntry;
  703. NextEntry = QuadAlign( Information );
  704. }
  705. except (!FsRtlIsNtstatusExpected(GetExceptionCode()) ?
  706. EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) {
  707. //
  708. // We must have had a problem filling in the user's buffer, so stop
  709. // and fail this request.
  710. //
  711. Information = 0;
  712. try_leave( Status = GetExceptionCode());
  713. }
  714. }
  715. }
  716. finally {
  717. if (!AbnormalTermination() && !NT_ERROR( Status )) {
  718. //
  719. // Update the Ccb to show the current state of the enumeration.
  720. //
  721. UdfLockFcb( IrpContext, Fcb );
  722. Ccb->CurrentFileIndex = CompoundDirContext.FileIndex.QuadPart;
  723. //
  724. // Update our notion of a high 32bit file index. We only do this once to avoid the hit
  725. // of thrashing the Fcb mutex to do this for every entry. If it is ever neccesary to use
  726. // this information, the difference of a few dozen entries from the optimal pick-up point
  727. // will be trivial.
  728. //
  729. if (CompoundDirContext.FileIndex.HighPart == 0 &&
  730. CompoundDirContext.FileIndex.LowPart > Ccb->HighestReturnableFileIndex) {
  731. Ccb->HighestReturnableFileIndex = CompoundDirContext.FileIndex.LowPart;
  732. }
  733. //
  734. // Mark in the CCB whether or not to skip the current entry on next call
  735. // (if we returned it in the current buffer).
  736. //
  737. ClearFlag( Ccb->Flags, CCB_FLAG_ENUM_RETURN_NEXT );
  738. if (ReturnNextEntry) {
  739. SetFlag( Ccb->Flags, CCB_FLAG_ENUM_RETURN_NEXT );
  740. }
  741. UdfUnlockFcb( IrpContext, Fcb );
  742. }
  743. //
  744. // Cleanup our search context.
  745. //
  746. UdfCleanupCompoundDirContext( IrpContext, &CompoundDirContext );
  747. //
  748. // Release the Fcb.
  749. //
  750. UdfReleaseFile( IrpContext, Fcb );
  751. }
  752. DebugTrace(( 0, Dbg, "UdfQueryDirectory -> %x\n", Status ));
  753. //
  754. // Complete the request here.
  755. //
  756. Irp->IoStatus.Information = Information;
  757. UdfCompleteRequest( IrpContext, Irp, Status );
  758. return Status;
  759. }
  760. //
  761. // Local support routines
  762. //
  763. NTSTATUS
  764. UdfNotifyChangeDirectory (
  765. IN PIRP_CONTEXT IrpContext,
  766. IN PIRP Irp,
  767. IN PIO_STACK_LOCATION IrpSp,
  768. IN PCCB Ccb
  769. )
  770. /*++
  771. Routine Description:
  772. This routine performs the notify change directory operation. It is
  773. responsible for either completing of enqueuing the input Irp. Although there
  774. will never be a notify signalled on a readonly disk we still support this call.
  775. We have already checked that this is not an OpenById handle.
  776. Arguments:
  777. Irp - Supplies the Irp to process
  778. IrpSp - Io stack location for this request.
  779. Ccb - Handle to the directory being watched.
  780. Return Value:
  781. NTSTATUS - STATUS_PENDING, any other error will raise.
  782. --*/
  783. {
  784. PAGED_CODE();
  785. //
  786. // Always set the wait bit in the IrpContext so the initial wait can't fail.
  787. //
  788. SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
  789. //
  790. // Acquire the Vcb shared.
  791. //
  792. UdfAcquireVcbShared( IrpContext, IrpContext->Vcb, FALSE );
  793. //
  794. // Use a try-finally to facilitate cleanup.
  795. //
  796. try {
  797. //
  798. // Verify the Vcb.
  799. //
  800. UdfVerifyVcb( IrpContext, IrpContext->Vcb );
  801. //
  802. // Call the Fsrtl package to process the request. We cast the
  803. // unicode strings to ansi strings as the dir notify package
  804. // only deals with memory matching.
  805. //
  806. FsRtlNotifyFullChangeDirectory( IrpContext->Vcb->NotifySync,
  807. &IrpContext->Vcb->DirNotifyList,
  808. Ccb,
  809. (PSTRING) &IrpSp->FileObject->FileName,
  810. BooleanFlagOn( IrpSp->Flags, SL_WATCH_TREE ),
  811. FALSE,
  812. IrpSp->Parameters.NotifyDirectory.CompletionFilter,
  813. Irp,
  814. NULL,
  815. NULL );
  816. } finally {
  817. //
  818. // Release the Vcb.
  819. //
  820. UdfReleaseVcb( IrpContext, IrpContext->Vcb );
  821. }
  822. //
  823. // Cleanup the IrpContext.
  824. //
  825. UdfCompleteRequest( IrpContext, NULL, STATUS_SUCCESS );
  826. return STATUS_PENDING;
  827. }
  828. //
  829. // Local support routine
  830. //
  831. NTSTATUS
  832. UdfInitializeEnumeration (
  833. IN PIRP_CONTEXT IrpContext,
  834. IN PIO_STACK_LOCATION IrpSp,
  835. IN PFCB Fcb,
  836. IN OUT PCCB Ccb,
  837. IN OUT PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext,
  838. OUT PBOOLEAN ReturnNextEntry,
  839. OUT PBOOLEAN ReturnSingleEntry,
  840. OUT PBOOLEAN InitialQuery
  841. )
  842. /*++
  843. Routine Description:
  844. This routine is called to initialize the enumeration variables and structures.
  845. We look at the state of a previous enumeration from the Ccb as well as any
  846. input values from the user. On exit we will position the DirContext at
  847. a file in the directory and let the caller know whether this entry or the
  848. next entry should be returned.
  849. Arguments:
  850. IrpSp - Irp stack location for this request.
  851. Fcb - Fcb for this directory.
  852. Ccb - Ccb for the directory handle.
  853. CompoundDirContext - Context to use for this enumeration.
  854. ReturnNextEntry - Address to store whether we should return the entry at
  855. the context position or the next entry.
  856. ReturnSingleEntry - Address to store whether we should only return
  857. a single entry.
  858. InitialQuery - Address to store whether this is the first enumeration
  859. query on this handle.
  860. Return Value:
  861. None.
  862. --*/
  863. {
  864. NTSTATUS Status;
  865. PUNICODE_STRING FileName;
  866. UNICODE_STRING SearchExpression;
  867. PUNICODE_STRING RestartName = NULL;
  868. ULONG CcbFlags;
  869. LONGLONG FileIndex;
  870. ULONG HighFileIndex;
  871. BOOLEAN KnownIndex;
  872. BOOLEAN Found;
  873. PAGED_CODE();
  874. //
  875. // Check inputs.
  876. //
  877. ASSERT_IRP_CONTEXT( IrpContext );
  878. ASSERT_FCB_INDEX( Fcb );
  879. ASSERT_CCB( Ccb );
  880. //
  881. // If this is the initial query then build a search expression from the input
  882. // file name.
  883. //
  884. if (!FlagOn( Ccb->Flags, CCB_FLAG_ENUM_INITIALIZED )) {
  885. FileName = (PUNICODE_STRING) IrpSp->Parameters.QueryDirectory.FileName;
  886. CcbFlags = 0;
  887. //
  888. // If the filename is not specified or is a single '*' then we will
  889. // match all names.
  890. //
  891. if ((FileName == NULL) ||
  892. (FileName->Buffer == NULL) ||
  893. (FileName->Length == 0) ||
  894. ((FileName->Length == sizeof( WCHAR )) &&
  895. (FileName->Buffer[0] == L'*'))) {
  896. SetFlag( CcbFlags, CCB_FLAG_ENUM_MATCH_ALL );
  897. SearchExpression.Length =
  898. SearchExpression.MaximumLength = 0;
  899. SearchExpression.Buffer = NULL;
  900. //
  901. // Otherwise build the name from the name in the stack location.
  902. // This involves checking for wild card characters and upcasing the
  903. // string if this is a case-insensitive search.
  904. //
  905. } else {
  906. //
  907. // The name better have at least one character.
  908. //
  909. if (FileName->Length == 0) {
  910. UdfRaiseStatus( IrpContext, STATUS_INVALID_PARAMETER );
  911. }
  912. //
  913. // Check for wildcards in the separate components.
  914. //
  915. if (FsRtlDoesNameContainWildCards( FileName)) {
  916. SetFlag( CcbFlags, CCB_FLAG_ENUM_NAME_EXP_HAS_WILD );
  917. }
  918. //
  919. // Now create the search expression to store in the Ccb.
  920. //
  921. SearchExpression.Buffer = FsRtlAllocatePoolWithTag( UdfPagedPool,
  922. FileName->Length,
  923. TAG_ENUM_EXPRESSION );
  924. SearchExpression.MaximumLength = FileName->Length;
  925. //
  926. // Either copy the name directly or perform the upcase.
  927. //
  928. if (FlagOn( Ccb->Flags, CCB_FLAG_IGNORE_CASE )) {
  929. Status = RtlUpcaseUnicodeString( &SearchExpression,
  930. FileName,
  931. FALSE );
  932. //
  933. // This should never fail.
  934. //
  935. ASSERT( Status == STATUS_SUCCESS );
  936. } else {
  937. RtlCopyMemory( SearchExpression.Buffer,
  938. FileName->Buffer,
  939. FileName->Length );
  940. }
  941. SearchExpression.Length = FileName->Length;
  942. }
  943. //
  944. // But we do not want to return the constant "." and ".." entries for
  945. // the root directory, for consistency with the rest of Microsoft's
  946. // filesystems.
  947. //
  948. if (Fcb == Fcb->Vcb->RootIndexFcb) {
  949. SetFlag( CcbFlags, CCB_FLAG_ENUM_NOMATCH_CONSTANT_ENTRY );
  950. }
  951. //
  952. // Now lock the Fcb in order to update the Ccb with the inital
  953. // enumeration values.
  954. //
  955. UdfLockFcb( IrpContext, Fcb );
  956. //
  957. // Check again that this is the initial search.
  958. //
  959. if (!FlagOn( Ccb->Flags, CCB_FLAG_ENUM_INITIALIZED )) {
  960. //
  961. // Update the values in the Ccb.
  962. //
  963. Ccb->CurrentFileIndex = 0;
  964. Ccb->SearchExpression = SearchExpression;
  965. //
  966. // Set the appropriate flags in the Ccb.
  967. //
  968. SetFlag( Ccb->Flags, CcbFlags | CCB_FLAG_ENUM_INITIALIZED );
  969. //
  970. // Otherwise cleanup any buffer allocated here.
  971. //
  972. } else {
  973. if (!FlagOn( CcbFlags, CCB_FLAG_ENUM_MATCH_ALL )) {
  974. UdfFreePool( &SearchExpression.Buffer );
  975. }
  976. }
  977. //
  978. // Otherwise lock the Fcb so we can read the current enumeration values.
  979. //
  980. } else {
  981. UdfLockFcb( IrpContext, Fcb );
  982. }
  983. //
  984. // Capture the current state of the enumeration.
  985. //
  986. // If the user specified an index then use his offset. We always
  987. // return the next entry in this case. If no name is specified,
  988. // then we can't perform the restart.
  989. //
  990. if (FlagOn( IrpSp->Flags, SL_INDEX_SPECIFIED ) &&
  991. IrpSp->Parameters.QueryDirectory.FileName != NULL) {
  992. KnownIndex = FALSE;
  993. FileIndex = IrpSp->Parameters.QueryDirectory.FileIndex;
  994. RestartName = (PUNICODE_STRING) IrpSp->Parameters.QueryDirectory.FileName;
  995. *ReturnNextEntry = TRUE;
  996. //
  997. // We will use the highest file index reportable to the caller as a
  998. // starting point as required if we cannot directly land at the
  999. // specified location.
  1000. //
  1001. HighFileIndex = Ccb->HighestReturnableFileIndex;
  1002. //
  1003. // If we are restarting the scan then go from the self entry.
  1004. //
  1005. } else if (FlagOn( IrpSp->Flags, SL_RESTART_SCAN )) {
  1006. KnownIndex = TRUE;
  1007. FileIndex = 0;
  1008. *ReturnNextEntry = FALSE;
  1009. //
  1010. // Otherwise use the values from the Ccb.
  1011. //
  1012. } else {
  1013. KnownIndex = TRUE;
  1014. FileIndex = Ccb->CurrentFileIndex;
  1015. *ReturnNextEntry = BooleanFlagOn( Ccb->Flags, CCB_FLAG_ENUM_RETURN_NEXT );
  1016. }
  1017. //
  1018. // Unlock the Fcb.
  1019. //
  1020. UdfUnlockFcb( IrpContext, Fcb );
  1021. //
  1022. // We have the starting offset in the directory and whether to return
  1023. // that entry or the next. If we are at the beginning of the directory
  1024. // and are returning that entry, then tell our caller this is the
  1025. // initial query.
  1026. //
  1027. *InitialQuery = FALSE;
  1028. if ((FileIndex == 0) &&
  1029. !(*ReturnNextEntry)) {
  1030. *InitialQuery = TRUE;
  1031. }
  1032. //
  1033. // Determine the offset in the stream to position the context and
  1034. // whether this offset is known to be a file offset.
  1035. //
  1036. // If this offset is known to be safe then go ahead and position the
  1037. // context. This handles the cases where the offset is the beginning
  1038. // of the stream, the offset is from a previous search or this is the
  1039. // initial query.
  1040. //
  1041. if (KnownIndex) {
  1042. Found = UdfLookupInitialFileIndex( IrpContext, Fcb, CompoundDirContext, &FileIndex );
  1043. ASSERT( Found );
  1044. //
  1045. // Avoid a raise in UdfUpdateDirNames if we're re-starting from a CCB index
  1046. // after the parent entry, but in a new call to querydirectory (new DirContext)
  1047. //
  1048. if (1 <= FileIndex) {
  1049. SetFlag( CompoundDirContext->DirContext.Flags, DIR_CONTEXT_FLAG_SEEN_PARENT );
  1050. }
  1051. //
  1052. // Try to directly jump to the specified file index. Otherwise we walk through
  1053. // the directory from the beginning (or the saved highest known offset if that is
  1054. // useful) until we reach the entry which contains this offset.
  1055. //
  1056. } else {
  1057. //
  1058. // We need to handle the special case of a restart from a synthesized
  1059. // entry - this is the one time where the restart index can be zero
  1060. // without requiring us to search above the 2^32 byte mark.
  1061. //
  1062. if (UdfFullCompareNames( IrpContext,
  1063. RestartName,
  1064. &UdfUnicodeDirectoryNames[SELF_ENTRY] ) == EqualTo) {
  1065. FileIndex = UDF_FILE_INDEX_VIRTUAL_SELF;
  1066. Found = UdfLookupInitialFileIndex( IrpContext, Fcb, CompoundDirContext, &FileIndex );
  1067. ASSERT( Found );
  1068. //
  1069. // We are restarting from a physical entry. If the restart index is zero, we were
  1070. // unable to inform the caller as to the "real" file index due to the dispartity
  1071. // between the ULONG FileIndex in the return structures and the LONGLONG offsets
  1072. // possible in directory streams. In this case, we will go as high as we were able
  1073. // to inform the caller of and search linearly from that point forward.
  1074. //
  1075. // It is also possible (realistic? unknown) that the restart index is somewhere in the
  1076. // middle of an entry and we won't find anything useable. In this case we try to find
  1077. // the entry which contains this index, using it as the real restart point.
  1078. //
  1079. } else {
  1080. //
  1081. // See if we need the high water mark.
  1082. //
  1083. if (FileIndex == 0) {
  1084. //
  1085. // We know that this is good.
  1086. //
  1087. FileIndex = Max( Ccb->HighestReturnableFileIndex, UDF_FILE_INDEX_PHYSICAL );;
  1088. KnownIndex = TRUE;
  1089. }
  1090. //
  1091. // The file index is now useful, falling into two cases
  1092. //
  1093. // 1) KnownIndex == FALSE - searching by index
  1094. // 2) KnownIndex == TRUE - searching by name
  1095. //
  1096. // Go set up our inquiry.
  1097. //
  1098. Found = UdfLookupInitialFileIndex( IrpContext, Fcb, CompoundDirContext, &FileIndex );
  1099. if (KnownIndex) {
  1100. //
  1101. // Walk forward to discover an entry named per the caller's expectation.
  1102. //
  1103. do {
  1104. UdfUpdateDirNames( IrpContext,
  1105. &CompoundDirContext->DirContext,
  1106. BooleanFlagOn( Ccb->Flags, CCB_FLAG_IGNORE_CASE ));
  1107. if (UdfFullCompareNames( IrpContext,
  1108. &CompoundDirContext->DirContext.CaseObjectName,
  1109. RestartName ) == EqualTo) {
  1110. break;
  1111. }
  1112. Found = UdfLookupNextFileIndex( IrpContext, Fcb, CompoundDirContext );
  1113. } while (Found);
  1114. } else if (!Found) {
  1115. LONGLONG LastFileIndex;
  1116. //
  1117. // Perform the search for the entry by index from the beginning of the physical directory.
  1118. //
  1119. LastFileIndex = UDF_FILE_INDEX_PHYSICAL;
  1120. Found = UdfLookupInitialFileIndex( IrpContext, Fcb, CompoundDirContext, &LastFileIndex );
  1121. ASSERT( Found );
  1122. //
  1123. // Keep walking through the directory until we run out of
  1124. // entries or we find an entry which ends beyond the input
  1125. // index value (index search case) or corresponds to the
  1126. // name we are looking for (name search case).
  1127. //
  1128. do {
  1129. //
  1130. // If we have passed the index value then exit.
  1131. //
  1132. if (CompoundDirContext->FileIndex.QuadPart > FileIndex) {
  1133. Found = FALSE;
  1134. break;
  1135. }
  1136. //
  1137. // Remember the current position in case we need to go back.
  1138. //
  1139. LastFileIndex = CompoundDirContext->FileIndex.QuadPart;
  1140. //
  1141. // Exit if the next entry is beyond the desired index value.
  1142. //
  1143. if (LastFileIndex + ISONsrFidSize( CompoundDirContext->DirContext.Fid ) > FileIndex) {
  1144. break;
  1145. }
  1146. Found = UdfLookupNextFileIndex( IrpContext, Fcb, CompoundDirContext );
  1147. } while (Found);
  1148. //
  1149. // If we didn't find the entry then go back to the last known entry.
  1150. //
  1151. if (!Found) {
  1152. UdfCleanupDirContext( IrpContext, &CompoundDirContext->DirContext );
  1153. UdfInitializeDirContext( IrpContext, &CompoundDirContext->DirContext );
  1154. Found = UdfLookupInitialFileIndex( IrpContext, Fcb, CompoundDirContext, &LastFileIndex );
  1155. ASSERT( Found );
  1156. }
  1157. }
  1158. }
  1159. }
  1160. //
  1161. // Only update the dirent name if we will need it for some reason.
  1162. // Don't update this name if we are returning the next entry, and
  1163. // don't update it if it was already done.
  1164. //
  1165. if (!(*ReturnNextEntry) &&
  1166. CompoundDirContext->DirContext.PureObjectName.Buffer == NULL) {
  1167. //
  1168. // If the caller specified an index that corresponds to a
  1169. // deleted file, they are trying to be tricky. Don't let them.
  1170. //
  1171. if (CompoundDirContext->DirContext.Fid &&
  1172. FlagOn( CompoundDirContext->DirContext.Fid->Flags, NSR_FID_F_DELETED )) {
  1173. return STATUS_INVALID_PARAMETER;
  1174. }
  1175. //
  1176. // Update the names in the dirent.
  1177. //
  1178. UdfUpdateDirNames( IrpContext,
  1179. &CompoundDirContext->DirContext,
  1180. BooleanFlagOn( Ccb->Flags, CCB_FLAG_IGNORE_CASE ));
  1181. }
  1182. //
  1183. // Look at the flag in the IrpSp indicating whether to return just
  1184. // one entry.
  1185. //
  1186. *ReturnSingleEntry = FALSE;
  1187. if (FlagOn( IrpSp->Flags, SL_RETURN_SINGLE_ENTRY )) {
  1188. *ReturnSingleEntry = TRUE;
  1189. }
  1190. return STATUS_SUCCESS;
  1191. }
  1192. //
  1193. // Local support routine
  1194. //
  1195. BOOLEAN
  1196. UdfEnumerateIndex (
  1197. IN PIRP_CONTEXT IrpContext,
  1198. IN PCCB Ccb,
  1199. IN OUT PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext,
  1200. IN BOOLEAN ReturnNextEntry
  1201. )
  1202. /*++
  1203. Routine Description:
  1204. This routine is the worker routine for index enumeration. We are positioned
  1205. at some dirent in the directory and will either return the first match
  1206. at that point or look to the next entry. The Ccb contains details about
  1207. the type of matching to do.
  1208. Arguments:
  1209. Ccb - Ccb for this directory handle.
  1210. CompoundDirContext - context already positioned at some entry in the directory.
  1211. ReturnNextEntry - Indicates if we are returning this entry or should start
  1212. with the next entry.
  1213. Return Value:
  1214. BOOLEAN - TRUE if next entry is found, FALSE otherwise.
  1215. --*/
  1216. {
  1217. BOOLEAN Found = FALSE;
  1218. PDIR_ENUM_CONTEXT DirContext;
  1219. PAGED_CODE();
  1220. //
  1221. // Check inputs.
  1222. //
  1223. ASSERT_IRP_CONTEXT( IrpContext );
  1224. ASSERT_CCB( Ccb );
  1225. //
  1226. // Directly reference the directory enumeration context for convenience.
  1227. //
  1228. DirContext = &CompoundDirContext->DirContext;
  1229. //
  1230. // Loop until we find a match or exaust the directory.
  1231. //
  1232. while (TRUE) {
  1233. //
  1234. // Move to the next entry unless we want to consider the current
  1235. // entry.
  1236. //
  1237. if (ReturnNextEntry) {
  1238. if (!UdfLookupNextFileIndex( IrpContext, Ccb->Fcb, CompoundDirContext )) {
  1239. break;
  1240. }
  1241. if (FlagOn( DirContext->Fid->Flags, NSR_FID_F_DELETED )) {
  1242. continue;
  1243. }
  1244. UdfUpdateDirNames( IrpContext,
  1245. DirContext,
  1246. BooleanFlagOn( Ccb->Flags, CCB_FLAG_IGNORE_CASE ));
  1247. } else {
  1248. ReturnNextEntry = TRUE;
  1249. }
  1250. //
  1251. // Don't bother if we have a constant entry and are ignoring them.
  1252. //
  1253. if (!FlagOn( DirContext->Flags, DIR_CONTEXT_FLAG_SEEN_NONCONSTANT ) &&
  1254. FlagOn( Ccb->Flags, CCB_FLAG_ENUM_NOMATCH_CONSTANT_ENTRY )) {
  1255. continue;
  1256. }
  1257. //
  1258. // If we match all names then return to our caller.
  1259. //
  1260. if (FlagOn( Ccb->Flags, CCB_FLAG_ENUM_MATCH_ALL )) {
  1261. DirContext->ShortObjectName.Length = 0;
  1262. Found = TRUE;
  1263. break;
  1264. }
  1265. //
  1266. // Check if the long name matches the search expression.
  1267. //
  1268. if (UdfIsNameInExpression( IrpContext,
  1269. &DirContext->CaseObjectName,
  1270. &Ccb->SearchExpression,
  1271. BooleanFlagOn( Ccb->Flags, CCB_FLAG_ENUM_NAME_EXP_HAS_WILD ))) {
  1272. //
  1273. // Let our caller know we found an entry.
  1274. //
  1275. DirContext->ShortObjectName.Length = 0;
  1276. Found = TRUE;
  1277. break;
  1278. }
  1279. //
  1280. // The long name didn't match so we need to check for a
  1281. // possible short name match. There is no match if the
  1282. // long name is one of the constant entries or already
  1283. // is 8dot3.
  1284. //
  1285. if (!(!FlagOn( DirContext->Flags, DIR_CONTEXT_FLAG_SEEN_NONCONSTANT ) ||
  1286. UdfIs8dot3Name( IrpContext,
  1287. DirContext->CaseObjectName ))) {
  1288. //
  1289. // Allocate the shortname if it isn't already done.
  1290. //
  1291. if (DirContext->ShortObjectName.Buffer == NULL) {
  1292. DirContext->ShortObjectName.Buffer = FsRtlAllocatePoolWithTag( UdfPagedPool,
  1293. BYTE_COUNT_8_DOT_3,
  1294. TAG_SHORT_FILE_NAME );
  1295. DirContext->ShortObjectName.MaximumLength = BYTE_COUNT_8_DOT_3;
  1296. }
  1297. UdfGenerate8dot3Name( IrpContext,
  1298. &DirContext->PureObjectName,
  1299. &DirContext->ShortObjectName );
  1300. //
  1301. // Check if this name matches.
  1302. //
  1303. if (UdfIsNameInExpression( IrpContext,
  1304. &DirContext->ShortObjectName,
  1305. &Ccb->SearchExpression,
  1306. BooleanFlagOn( Ccb->Flags, CCB_FLAG_ENUM_NAME_EXP_HAS_WILD ))) {
  1307. //
  1308. // Let our caller know we found an entry.
  1309. //
  1310. Found = TRUE;
  1311. break;
  1312. }
  1313. }
  1314. }
  1315. return Found;
  1316. }
  1317. //
  1318. // Local support routine
  1319. //
  1320. VOID
  1321. UdfLookupFileEntryInEnumeration (
  1322. IN PIRP_CONTEXT IrpContext,
  1323. IN PFCB Fcb,
  1324. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext
  1325. )
  1326. /*++
  1327. Routine Description:
  1328. This routine retrieves the file entry associated with the current location in
  1329. the enumeration of a compound directory context.
  1330. Arguments:
  1331. Fcb - the directory being enumerated.
  1332. CompoundDirContext - a corresponding context for the enumeration.
  1333. Return Value:
  1334. None. Status may be raised on discovery of corruption.
  1335. --*/
  1336. {
  1337. PNSR_FID Fid;
  1338. PICBFILE Fe;
  1339. ULONG Length;
  1340. Fid = CompoundDirContext->DirContext.Fid;
  1341. //
  1342. // Figure out where the ICB we want is.
  1343. //
  1344. if (UdfIsFileIndexVirtual( CompoundDirContext->FileIndex.QuadPart )) {
  1345. //
  1346. // Synthesize! We only have to synthesize the self entry. The name is already done,
  1347. // so the remaining work is trivial.
  1348. //
  1349. ASSERT( Fid == NULL );
  1350. //
  1351. // Lift the FE corresponding to this directory
  1352. //
  1353. UdfCleanupIcbContext( IrpContext, &CompoundDirContext->IcbContext );
  1354. UdfInitializeIcbContextFromFcb( IrpContext,
  1355. &CompoundDirContext->IcbContext,
  1356. Fcb );
  1357. Length = Fcb->RootExtentLength;
  1358. } else {
  1359. //
  1360. // Lift the FE corresponding to this FID.
  1361. //
  1362. ASSERT( Fid != NULL );
  1363. UdfCleanupIcbContext( IrpContext, &CompoundDirContext->IcbContext );
  1364. UdfInitializeIcbContext( IrpContext,
  1365. &CompoundDirContext->IcbContext,
  1366. Fcb->Vcb,
  1367. DESTAG_ID_NSR_FILE,
  1368. Fid->Icb.Start.Partition,
  1369. Fid->Icb.Start.Lbn,
  1370. BlockSize( IrpContext->Vcb) );
  1371. Length = Fid->Icb.Length.Length;
  1372. }
  1373. //
  1374. // Retrieve the ICB for inspection.
  1375. //
  1376. UdfLookupActiveIcb( IrpContext,
  1377. &CompoundDirContext->IcbContext,
  1378. Length);
  1379. Fe = (PICBFILE) CompoundDirContext->IcbContext.Active.View;
  1380. //
  1381. // Perform some basic verification that the FE is of the proper type and that
  1382. // FID and FE agree as to the type of the object. We explicitly check that
  1383. // a legal filesystem-level FE type is discovered, even though we don't support
  1384. // them in other paths. Note that we leave the IcbContext->IcbType as FILE even
  1385. // though we may have picked up an extended file entry.
  1386. //
  1387. if (((Fe->Destag.Ident != DESTAG_ID_NSR_FILE) &&
  1388. ((Fe->Destag.Ident != DESTAG_ID_NSR_EXT_FILE) || (!UdfExtendedFEAllowed( IrpContext->Vcb)))) ||
  1389. (((Fid && FlagOn( Fid->Flags, NSR_FID_F_DIRECTORY )) ||
  1390. Fid == NULL) &&
  1391. Fe->Icbtag.FileType != ICBTAG_FILE_T_DIRECTORY) ||
  1392. (Fe->Icbtag.FileType != ICBTAG_FILE_T_FILE &&
  1393. Fe->Icbtag.FileType != ICBTAG_FILE_T_DIRECTORY &&
  1394. Fe->Icbtag.FileType != ICBTAG_FILE_T_BLOCK_DEV &&
  1395. Fe->Icbtag.FileType != ICBTAG_FILE_T_CHAR_DEV &&
  1396. Fe->Icbtag.FileType != ICBTAG_FILE_T_FIFO &&
  1397. Fe->Icbtag.FileType != ICBTAG_FILE_T_C_ISSOCK &&
  1398. Fe->Icbtag.FileType != ICBTAG_FILE_T_PATHLINK &&
  1399. Fe->Icbtag.FileType != ICBTAG_FILE_T_REALTIME)
  1400. ) {
  1401. UdfRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
  1402. }
  1403. }
  1404. //
  1405. // Local support routine
  1406. //
  1407. BOOLEAN
  1408. UdfLookupInitialFileIndex (
  1409. IN PIRP_CONTEXT IrpContext,
  1410. IN PFCB Fcb,
  1411. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext,
  1412. IN PLONGLONG InitialIndex
  1413. )
  1414. /*++
  1415. Routine Description:
  1416. This routine begins the enumeration of a directory by setting the context
  1417. at the first avaliable virtual directory entry.
  1418. Arguments:
  1419. Fcb - the directory being enumerated.
  1420. CompoundDirContext - a corresponding context for the enumeration.
  1421. InitialIndex - an optional starting file index to base the enumeration.
  1422. Return Value:
  1423. TRUE will be returned if a valid entry is found at this offset, FALSE otherwise.
  1424. --*/
  1425. {
  1426. LONGLONG DirOffset;
  1427. if (UdfIsFileIndexVirtual( *InitialIndex )) {
  1428. //
  1429. // We only synthesize a single virtual directory entry. Position the context
  1430. // at the virtual self entry.
  1431. //
  1432. CompoundDirContext->FileIndex.QuadPart = UDF_FILE_INDEX_VIRTUAL_SELF;
  1433. return TRUE;
  1434. }
  1435. CompoundDirContext->FileIndex.QuadPart = *InitialIndex;
  1436. //
  1437. // Find the base offset in the directory and look it up.
  1438. //
  1439. DirOffset = UdfFileIndexToPhysicalOffset( *InitialIndex );
  1440. return UdfLookupInitialDirEntry( IrpContext,
  1441. Fcb,
  1442. &CompoundDirContext->DirContext,
  1443. &DirOffset );
  1444. }
  1445. //
  1446. // Local support routine
  1447. //
  1448. BOOLEAN
  1449. UdfLookupNextFileIndex (
  1450. IN PIRP_CONTEXT IrpContext,
  1451. IN PFCB Fcb,
  1452. IN PCOMPOUND_DIR_ENUM_CONTEXT CompoundDirContext
  1453. )
  1454. /*++
  1455. Routine Description:
  1456. This routine advances the enumeration of a virtual directory by one entry.
  1457. Arguments:
  1458. Fcb - the directory being enumerated.
  1459. CompoundDirContext - a corresponding context for the enumeration.
  1460. Return Value:
  1461. BOOLEAN True if another Fid is avaliable, False if we are at the end.
  1462. --*/
  1463. {
  1464. ULONG Advance;
  1465. BOOLEAN Result;
  1466. //
  1467. // Advance from the synthesized to the physical directory.
  1468. //
  1469. if (UdfIsFileIndexVirtual( CompoundDirContext->FileIndex.QuadPart )) {
  1470. Result = UdfLookupInitialDirEntry( IrpContext,
  1471. Fcb,
  1472. &CompoundDirContext->DirContext,
  1473. NULL );
  1474. if (Result) {
  1475. CompoundDirContext->FileIndex.QuadPart = UDF_FILE_INDEX_PHYSICAL;
  1476. }
  1477. return Result;
  1478. }
  1479. Advance = ISONsrFidSize( CompoundDirContext->DirContext.Fid );
  1480. //
  1481. // Advance to the next entry in this directory.
  1482. //
  1483. Result = UdfLookupNextDirEntry( IrpContext, Fcb, &CompoundDirContext->DirContext );
  1484. if (Result) {
  1485. CompoundDirContext->FileIndex.QuadPart += Advance;
  1486. }
  1487. return Result;
  1488. }