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.

4626 lines
126 KiB

  1. /*++
  2. Copyright (c) 1996-2000 Microsoft Corporation
  3. Module Name:
  4. FsCtrl.c
  5. Abstract:
  6. This module implements the File System Control routines for Udfs called
  7. by the Fsd/Fsp dispatch drivers.
  8. // @@BEGIN_DDKSPLIT
  9. Author:
  10. Dan Lovinger [DanLo] 11-Jun-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_FSCTRL)
  20. //
  21. // The local debug trace level
  22. //
  23. #define Dbg (UDFS_DEBUG_LEVEL_FSCTRL)
  24. //
  25. // Local constants
  26. //
  27. BOOLEAN UdfDisable = FALSE;
  28. //
  29. // CRC of the PVD on Disney's Snow White title, so we can
  30. // ignore the volsetseqmax on that disc only.
  31. //
  32. #define UDF_SNOW_WHITE_PVD_CRC ((USHORT)0x1d05)
  33. #define UDF_SNOW_WHITE_PVD_CRC_VARIANT_2 ((USHORT)0x534e)
  34. //
  35. // Local macros
  36. //
  37. INLINE
  38. VOID
  39. UdfStoreFileSetDescriptorIfPrevailing (
  40. IN OUT PNSR_FSD *StoredFSD,
  41. IN OUT PNSR_FSD *NewFSD
  42. )
  43. {
  44. PNSR_FSD TempFSD;
  45. //
  46. // If we haven't stored a fileset descriptor or the fileset number
  47. // of the stored descriptor is less than the new descriptor, swap the
  48. // pointers around.
  49. //
  50. if (*StoredFSD == NULL || (*StoredFSD)->FileSet < (*NewFSD)->FileSet) {
  51. TempFSD = *StoredFSD;
  52. *StoredFSD = *NewFSD;
  53. *NewFSD = TempFSD;
  54. }
  55. }
  56. //
  57. // Local support routines
  58. //
  59. VOID
  60. UdfDetermineVolumeBounding (
  61. IN PIRP_CONTEXT IrpContext,
  62. IN PDEVICE_OBJECT TargetDeviceObject,
  63. IN PULONG S,
  64. IN PULONG N
  65. );
  66. NTSTATUS
  67. UdfDismountVolume (
  68. IN PIRP_CONTEXT IrpContext,
  69. IN PIRP Irp
  70. );
  71. NTSTATUS
  72. UdfFindAnchorVolumeDescriptor (
  73. IN PIRP_CONTEXT IrpContext,
  74. IN PVCB Vcb,
  75. IN OUT PNSR_ANCHOR *AnchorVolumeDescriptor
  76. );
  77. NTSTATUS
  78. UdfFindFileSetDescriptor (
  79. IN PIRP_CONTEXT IrpContext,
  80. IN PVCB Vcb,
  81. IN PLONGAD LongAd,
  82. IN OUT PNSR_FSD *FileSetDescriptor
  83. );
  84. NTSTATUS
  85. UdfFindVolumeDescriptors (
  86. IN PIRP_CONTEXT IrpContext,
  87. IN PVCB Vcb,
  88. IN PEXTENTAD Extent,
  89. IN OUT PPCB *Pcb,
  90. IN OUT PNSR_PVD *PrimaryVolumeDescriptor,
  91. IN OUT PNSR_LVOL *LogicalVolumeDescriptor
  92. );
  93. NTSTATUS
  94. UdfInvalidateVolumes (
  95. IN PIRP_CONTEXT IrpContext,
  96. IN PIRP Irp
  97. );
  98. NTSTATUS
  99. UdfIsPathnameValid (
  100. IN PIRP_CONTEXT IrpContext,
  101. IN PIRP Irp
  102. );
  103. BOOLEAN
  104. UdfIsRemount (
  105. IN PIRP_CONTEXT IrpContext,
  106. IN PVCB Vcb,
  107. OUT PVCB *OldVcb
  108. );
  109. UdfIsVolumeDirty (
  110. IN PIRP_CONTEXT IrpContext,
  111. IN PIRP Irp
  112. );
  113. NTSTATUS
  114. UdfIsVolumeMounted (
  115. IN PIRP_CONTEXT IrpContext,
  116. IN PIRP Irp
  117. );
  118. NTSTATUS
  119. UdfLockVolume (
  120. IN PIRP_CONTEXT IrpContext,
  121. IN PIRP Irp
  122. );
  123. VOID
  124. UdfRemountOldVcb(
  125. IN PIRP_CONTEXT IrpContext,
  126. IN PVCB OldVcb,
  127. IN PVCB NewVcb,
  128. IN PDEVICE_OBJECT DeviceObjectWeTalkTo
  129. );
  130. NTSTATUS
  131. UdfMountVolume(
  132. IN PIRP_CONTEXT IrpContext,
  133. IN PIRP Irp
  134. );
  135. NTSTATUS
  136. UdfOplockRequest (
  137. IN PIRP_CONTEXT IrpContext,
  138. IN PIRP Irp
  139. );
  140. BOOLEAN
  141. UdfRecognizeVolume (
  142. IN PIRP_CONTEXT IrpContext,
  143. IN PDEVICE_OBJECT DeviceObject,
  144. IN ULONG SectorSize,
  145. IN PULONG BoundS,
  146. IN OUT PBOOLEAN Bridge,
  147. OUT PUSHORT NSRVerFound
  148. );
  149. VOID
  150. UdfScanForDismountedVcb (
  151. IN PIRP_CONTEXT IrpContext
  152. );
  153. NTSTATUS
  154. UdfUnlockVolume (
  155. IN PIRP_CONTEXT IrpContext,
  156. IN PIRP Irp
  157. );
  158. VOID
  159. UdfUpdateVolumeLabel (
  160. IN PIRP_CONTEXT IrpContext,
  161. IN PWCHAR VolumeLabel,
  162. IN OUT PUSHORT VolumeLabelLength,
  163. IN PUCHAR Dstring,
  164. IN UCHAR FieldLength
  165. );
  166. VOID
  167. UdfUpdateVolumeSerialNumber (
  168. IN PIRP_CONTEXT IrpContext,
  169. IN OUT PULONG VolumeSerialNumber,
  170. IN PNSR_FSD Fsd
  171. );
  172. NTSTATUS
  173. UdfUserFsctl (
  174. IN PIRP_CONTEXT IrpContext,
  175. IN PIRP Irp
  176. );
  177. NTSTATUS
  178. UdfVerifyVolume (
  179. IN PIRP_CONTEXT IrpContext,
  180. IN PIRP Irp
  181. );
  182. NTSTATUS
  183. UdfAllowExtendedDasdIo(
  184. IN PIRP_CONTEXT IrpContext,
  185. IN PIRP Irp
  186. );
  187. #ifdef ALLOC_PRAGMA
  188. #pragma alloc_text(PAGE, UdfCommonFsControl)
  189. #pragma alloc_text(PAGE, UdfDetermineVolumeBounding)
  190. #pragma alloc_text(PAGE, UdfDismountVolume)
  191. #pragma alloc_text(PAGE, UdfFindAnchorVolumeDescriptor)
  192. #pragma alloc_text(PAGE, UdfFindFileSetDescriptor)
  193. #pragma alloc_text(PAGE, UdfFindVolumeDescriptors)
  194. #pragma alloc_text(PAGE, UdfIsPathnameValid)
  195. #pragma alloc_text(PAGE, UdfIsRemount)
  196. #pragma alloc_text(PAGE, UdfIsVolumeDirty)
  197. #pragma alloc_text(PAGE, UdfIsVolumeMounted)
  198. #pragma alloc_text(PAGE, UdfLockVolume)
  199. #pragma alloc_text(PAGE, UdfMountVolume)
  200. #pragma alloc_text(PAGE, UdfOplockRequest)
  201. #pragma alloc_text(PAGE, UdfRecognizeVolume)
  202. #pragma alloc_text(PAGE, UdfScanForDismountedVcb)
  203. #pragma alloc_text(PAGE, UdfStoreVolumeDescriptorIfPrevailing)
  204. #pragma alloc_text(PAGE, UdfUnlockVolume)
  205. #pragma alloc_text(PAGE, UdfUpdateVolumeLabel)
  206. #pragma alloc_text(PAGE, UdfUpdateVolumeSerialNumber)
  207. #pragma alloc_text(PAGE, UdfUserFsctl)
  208. #pragma alloc_text(PAGE, UdfVerifyVolume)
  209. #pragma alloc_text(PAGE, UdfAllowExtendedDasdIo)
  210. #endif
  211. VOID
  212. UdfStoreVolumeDescriptorIfPrevailing (
  213. IN OUT PNSR_VD_GENERIC *StoredVD,
  214. IN OUT PNSR_VD_GENERIC NewVD
  215. )
  216. /*++
  217. Routine Description:
  218. This routine updates Volume Descriptor if the new descriptor
  219. is more prevailing than the one currently stored.
  220. Arguments:
  221. StoredVD - pointer to a currently stored descriptor
  222. NewVD - pointer to a candidate descriptor
  223. Return Value:
  224. None.
  225. --*/
  226. {
  227. PNSR_VD_GENERIC TempVD;
  228. //
  229. // If we haven't stored a volume descriptor or the sequence number
  230. // of the stored descriptor is less than the new descriptor, make a copy
  231. // of it and store it.
  232. //
  233. if ((NULL == *StoredVD) || ((*StoredVD)->Sequence < NewVD->Sequence)) {
  234. if ( NULL == *StoredVD) {
  235. *StoredVD = (PNSR_VD_GENERIC) FsRtlAllocatePoolWithTag( UdfNonPagedPool,
  236. sizeof(NSR_VD_GENERIC),
  237. TAG_NSR_VDSD );
  238. }
  239. RtlCopyMemory( *StoredVD, NewVD, sizeof( NSR_VD_GENERIC));
  240. }
  241. }
  242. NTSTATUS
  243. UdfCommonFsControl (
  244. IN PIRP_CONTEXT IrpContext,
  245. IN PIRP Irp
  246. )
  247. /*++
  248. Routine Description:
  249. This is the common routine for doing FileSystem control operations called
  250. by both the fsd and fsp threads
  251. Arguments:
  252. Irp - Supplies the Irp to process
  253. Return Value:
  254. NTSTATUS - The return status for the operation
  255. --*/
  256. {
  257. NTSTATUS Status;
  258. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  259. PAGED_CODE();
  260. //
  261. // Check the input parameters
  262. //
  263. ASSERT_IRP_CONTEXT( IrpContext );
  264. ASSERT_IRP( Irp );
  265. //
  266. // Get a pointer to the current Irp stack location
  267. //
  268. IrpSp = IoGetCurrentIrpStackLocation( Irp );
  269. //
  270. // We know this is a file system control so we'll case on the
  271. // minor function, and call a internal worker routine to complete
  272. // the irp.
  273. //
  274. switch (IrpSp->MinorFunction) {
  275. case IRP_MN_MOUNT_VOLUME:
  276. Status = UdfMountVolume( IrpContext, Irp );
  277. break;
  278. case IRP_MN_VERIFY_VOLUME:
  279. Status = UdfVerifyVolume( IrpContext, Irp );
  280. break;
  281. case IRP_MN_USER_FS_REQUEST:
  282. Status = UdfUserFsctl( IrpContext, Irp );
  283. break;
  284. default:
  285. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_DEVICE_REQUEST );
  286. Status = STATUS_INVALID_DEVICE_REQUEST;
  287. break;
  288. }
  289. return Status;
  290. }
  291. //
  292. // Local support routine
  293. //
  294. NTSTATUS
  295. UdfUserFsctl (
  296. IN PIRP_CONTEXT IrpContext,
  297. IN PIRP Irp
  298. )
  299. /*++
  300. Routine Description:
  301. This is the common routine for implementing the user's requests made
  302. through NtFsControlFile.
  303. Arguments:
  304. Irp - Supplies the Irp being processed
  305. Return Value:
  306. NTSTATUS - The return status for the operation
  307. --*/
  308. {
  309. NTSTATUS Status;
  310. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  311. PAGED_CODE();
  312. //
  313. // Case on the control code.
  314. //
  315. switch ( IrpSp->Parameters.FileSystemControl.FsControlCode ) {
  316. case FSCTL_REQUEST_OPLOCK_LEVEL_1 :
  317. case FSCTL_REQUEST_OPLOCK_LEVEL_2 :
  318. case FSCTL_REQUEST_BATCH_OPLOCK :
  319. case FSCTL_OPLOCK_BREAK_ACKNOWLEDGE :
  320. case FSCTL_OPBATCH_ACK_CLOSE_PENDING :
  321. case FSCTL_OPLOCK_BREAK_NOTIFY :
  322. case FSCTL_OPLOCK_BREAK_ACK_NO_2 :
  323. case FSCTL_REQUEST_FILTER_OPLOCK :
  324. Status = UdfOplockRequest( IrpContext, Irp );
  325. break;
  326. case FSCTL_LOCK_VOLUME :
  327. Status = UdfLockVolume( IrpContext, Irp );
  328. break;
  329. case FSCTL_UNLOCK_VOLUME :
  330. Status = UdfUnlockVolume( IrpContext, Irp );
  331. break;
  332. case FSCTL_DISMOUNT_VOLUME :
  333. Status = UdfDismountVolume( IrpContext, Irp );
  334. break;
  335. case FSCTL_IS_VOLUME_DIRTY :
  336. Status = UdfIsVolumeDirty( IrpContext, Irp );
  337. break;
  338. case FSCTL_IS_VOLUME_MOUNTED :
  339. Status = UdfIsVolumeMounted( IrpContext, Irp );
  340. break;
  341. case FSCTL_IS_PATHNAME_VALID :
  342. Status = UdfIsPathnameValid( IrpContext, Irp );
  343. break;
  344. case FSCTL_INVALIDATE_VOLUMES :
  345. Status = UdfInvalidateVolumes( IrpContext, Irp );
  346. break;
  347. case FSCTL_ALLOW_EXTENDED_DASD_IO:
  348. Status = UdfAllowExtendedDasdIo( IrpContext, Irp );
  349. break;
  350. //
  351. // We don't support any of the known or unknown requests.
  352. //
  353. default:
  354. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_DEVICE_REQUEST );
  355. Status = STATUS_INVALID_DEVICE_REQUEST;
  356. break;
  357. }
  358. return Status;
  359. }
  360. //
  361. // Local support routine
  362. //
  363. NTSTATUS
  364. UdfOplockRequest (
  365. IN PIRP_CONTEXT IrpContext,
  366. IN PIRP Irp
  367. )
  368. /*++
  369. Routine Description:
  370. This is the common routine to handle oplock requests made via the
  371. NtFsControlFile call.
  372. Arguments:
  373. Irp - Supplies the Irp being processed
  374. Return Value:
  375. NTSTATUS - The return status for the operation
  376. --*/
  377. {
  378. NTSTATUS Status;
  379. PFCB Fcb;
  380. PCCB Ccb;
  381. ULONG OplockCount = 0;
  382. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  383. PAGED_CODE();
  384. //
  385. // We only permit oplock requests on files.
  386. //
  387. if (UdfDecodeFileObject( IrpSp->FileObject,
  388. &Fcb,
  389. &Ccb ) != UserFileOpen ) {
  390. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  391. return STATUS_INVALID_PARAMETER;
  392. }
  393. //
  394. // Make this a waitable Irpcontext so we don't fail to acquire
  395. // the resources.
  396. //
  397. SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
  398. ClearFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_FORCE_POST );
  399. //
  400. // Switch on the function control code. We grab the Fcb exclusively
  401. // for oplock requests, shared for oplock break acknowledgement.
  402. //
  403. switch (IrpSp->Parameters.FileSystemControl.FsControlCode) {
  404. case FSCTL_REQUEST_OPLOCK_LEVEL_1 :
  405. case FSCTL_REQUEST_OPLOCK_LEVEL_2 :
  406. case FSCTL_REQUEST_BATCH_OPLOCK :
  407. case FSCTL_REQUEST_FILTER_OPLOCK :
  408. UdfAcquireFcbExclusive( IrpContext, Fcb, FALSE );
  409. if (IrpSp->Parameters.FileSystemControl.FsControlCode == FSCTL_REQUEST_OPLOCK_LEVEL_2) {
  410. if (Fcb->FileLock != NULL) {
  411. OplockCount = (ULONG) FsRtlAreThereCurrentFileLocks( Fcb->FileLock );
  412. }
  413. } else {
  414. OplockCount = Fcb->FcbCleanup;
  415. }
  416. break;
  417. case FSCTL_OPLOCK_BREAK_ACKNOWLEDGE:
  418. case FSCTL_OPBATCH_ACK_CLOSE_PENDING:
  419. case FSCTL_OPLOCK_BREAK_NOTIFY:
  420. case FSCTL_OPLOCK_BREAK_ACK_NO_2:
  421. UdfAcquireFcbShared( IrpContext, Fcb, FALSE );
  422. break;
  423. default:
  424. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  425. return STATUS_INVALID_PARAMETER;
  426. }
  427. //
  428. // Use a try finally to free the Fcb.
  429. //
  430. try {
  431. //
  432. // Verify the Fcb.
  433. //
  434. UdfVerifyFcbOperation( IrpContext, Fcb );
  435. //
  436. // Call the FsRtl routine to grant/acknowledge oplock.
  437. //
  438. Status = FsRtlOplockFsctrl( &Fcb->Oplock,
  439. Irp,
  440. OplockCount );
  441. //
  442. // Set the flag indicating if Fast I/O is possible
  443. //
  444. UdfLockFcb( IrpContext, Fcb );
  445. Fcb->IsFastIoPossible = UdfIsFastIoPossible( Fcb );
  446. UdfUnlockFcb( IrpContext, Fcb );
  447. //
  448. // The oplock package will complete the Irp.
  449. //
  450. Irp = NULL;
  451. } finally {
  452. //
  453. // Release all of our resources
  454. //
  455. UdfReleaseFcb( IrpContext, Fcb );
  456. }
  457. //
  458. // Complete the request if there was no exception.
  459. //
  460. UdfCompleteRequest( IrpContext, Irp, Status );
  461. return Status;
  462. }
  463. //
  464. // Local support routine
  465. //
  466. NTSTATUS
  467. UdfLockVolumeInternal (
  468. IN PIRP_CONTEXT IrpContext,
  469. IN PVCB Vcb,
  470. IN PFILE_OBJECT FileObject OPTIONAL
  471. )
  472. /*++
  473. Routine Description:
  474. This routine performs the actual lock volume operation. It will be called
  475. by anyone wishing to try to protect the volume for a long duration. PNP
  476. operations are such a user.
  477. The volume must be held exclusive by the caller.
  478. Arguments:
  479. Vcb - The volume being locked.
  480. FileObject - File corresponding to the handle locking the volume. If this
  481. is not specified, a system lock is assumed.
  482. Return Value:
  483. NTSTATUS - The return status for the operation
  484. --*/
  485. {
  486. NTSTATUS Status;
  487. NTSTATUS FinalStatus = (FileObject? STATUS_ACCESS_DENIED: STATUS_DEVICE_BUSY);
  488. ULONG RemainingUserReferences = (FileObject? 1: 0);
  489. KIRQL SavedIrql;
  490. ASSERT_EXCLUSIVE_VCB( Vcb );
  491. //
  492. // The cleanup count for the volume only reflects the fileobject that
  493. // will lock the volume. Otherwise, we must fail the request.
  494. //
  495. // Since the only cleanup is for the provided fileobject, we will try
  496. // to get rid of all of the other user references. If there is only one
  497. // remaining after the purge then we can allow the volume to be locked.
  498. //
  499. UdfPurgeVolume( IrpContext, Vcb, FALSE );
  500. //
  501. // Now back out of our synchronization and wait for the lazy writer
  502. // to finish off any lazy closes that could have been outstanding.
  503. //
  504. // Since we purged, we know that the lazy writer will issue all
  505. // possible lazy closes in the next tick - if we hadn't, an otherwise
  506. // unopened file with a large amount of dirty data could have hung
  507. // around for a while as the data trickled out to the disk.
  508. //
  509. // This is even more important now since we send notification to
  510. // alert other folks that this style of check is about to happen so
  511. // that they can close their handles. We don't want to enter a fast
  512. // race with the lazy writer tearing down his references to the file.
  513. //
  514. UdfReleaseVcb( IrpContext, Vcb );
  515. Status = CcWaitForCurrentLazyWriterActivity();
  516. //
  517. // This is intentional. If we were able to get the Vcb before, just
  518. // wait for it and take advantage of knowing that it is OK to leave
  519. // the flag up.
  520. //
  521. SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
  522. UdfAcquireVcbExclusive( IrpContext, Vcb, FALSE );
  523. if (!NT_SUCCESS( Status )) {
  524. return Status;
  525. }
  526. UdfFspClose( Vcb );
  527. //
  528. // If the volume is already explicitly locked then fail. We use the
  529. // Vpb locked flag as an 'explicit lock' flag in the same way as Fat.
  530. //
  531. IoAcquireVpbSpinLock( &SavedIrql );
  532. if (!FlagOn( Vcb->Vpb->Flags, VPB_LOCKED ) &&
  533. (Vcb->VcbCleanup == RemainingUserReferences) &&
  534. (Vcb->VcbUserReference == Vcb->VcbResidualUserReference + RemainingUserReferences)) {
  535. SetFlag( Vcb->VcbState, VCB_STATE_LOCKED );
  536. SetFlag( Vcb->Vpb->Flags, VPB_LOCKED );
  537. Vcb->VolumeLockFileObject = FileObject;
  538. FinalStatus = STATUS_SUCCESS;
  539. }
  540. IoReleaseVpbSpinLock( SavedIrql );
  541. return FinalStatus;
  542. }
  543. NTSTATUS
  544. UdfUnlockVolumeInternal (
  545. IN PIRP_CONTEXT IrpContext,
  546. IN PVCB Vcb,
  547. IN PFILE_OBJECT FileObject OPTIONAL
  548. )
  549. /*++
  550. Routine Description:
  551. This routine performs the actual unlock volume operation.
  552. The volume must be held exclusive by the caller.
  553. Arguments:
  554. Vcb - The volume being locked.
  555. FileObject - File corresponding to the handle locking the volume. If this
  556. is not specified, a system lock is assumed.
  557. Return Value:
  558. NTSTATUS - The return status for the operation
  559. Attempting to remove a system lock that did not exist is OK.
  560. --*/
  561. {
  562. NTSTATUS Status = STATUS_NOT_LOCKED;
  563. KIRQL SavedIrql;
  564. IoAcquireVpbSpinLock( &SavedIrql );
  565. if (FlagOn(Vcb->Vpb->Flags, VPB_LOCKED) &&
  566. (FileObject == Vcb->VolumeLockFileObject)) {
  567. ClearFlag( Vcb->VcbState, VCB_STATE_LOCKED );
  568. ClearFlag( Vcb->Vpb->Flags, VPB_LOCKED);
  569. Vcb->VolumeLockFileObject = NULL;
  570. Status = STATUS_SUCCESS;
  571. }
  572. IoReleaseVpbSpinLock( SavedIrql );
  573. return Status;
  574. }
  575. //
  576. // Local support routine
  577. //
  578. NTSTATUS
  579. UdfLockVolume (
  580. IN PIRP_CONTEXT IrpContext,
  581. IN PIRP Irp
  582. )
  583. /*++
  584. Routine Description:
  585. This routine performs the lock volume operation. It is responsible for
  586. either completing of enqueuing the input Irp.
  587. Arguments:
  588. Irp - Supplies the Irp to process
  589. Return Value:
  590. NTSTATUS - The return status for the operation
  591. --*/
  592. {
  593. NTSTATUS Status;
  594. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  595. PVCB Vcb;
  596. PFCB Fcb;
  597. PCCB Ccb;
  598. PAGED_CODE();
  599. //
  600. // Decode the file object, the only type of opens we accept are
  601. // user volume opens.
  602. //
  603. if (UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb ) != UserVolumeOpen) {
  604. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  605. return STATUS_INVALID_PARAMETER;
  606. }
  607. DebugTrace(( +1, Dbg, "UdfLockVolume()\n"));
  608. //
  609. // Send our notification so that folks that like to hold handles on
  610. // volumes can get out of the way.
  611. //
  612. FsRtlNotifyVolumeEvent( IrpSp->FileObject, FSRTL_VOLUME_LOCK );
  613. //
  614. // Acquire exclusive access to the Vcb.
  615. //
  616. Vcb = Fcb->Vcb;
  617. UdfAcquireVcbExclusive( IrpContext, Vcb, FALSE );
  618. try {
  619. //
  620. // Verify the Vcb.
  621. //
  622. UdfVerifyVcb( IrpContext, Vcb );
  623. Status = UdfLockVolumeInternal( IrpContext, Vcb, IrpSp->FileObject );
  624. } finally {
  625. //
  626. // Release the Vcb.
  627. //
  628. UdfReleaseVcb( IrpContext, Vcb );
  629. if (AbnormalTermination() || !NT_SUCCESS( Status )) {
  630. FsRtlNotifyVolumeEvent( IrpSp->FileObject, FSRTL_VOLUME_LOCK_FAILED );
  631. }
  632. DebugTrace(( -1, Dbg, "UdfLockVolume() -> 0x%X\n", Status));
  633. }
  634. //
  635. // Complete the request if there haven't been any exceptions.
  636. //
  637. UdfCompleteRequest( IrpContext, Irp, Status );
  638. return Status;
  639. }
  640. //
  641. // Local support routine
  642. //
  643. NTSTATUS
  644. UdfUnlockVolume (
  645. IN PIRP_CONTEXT IrpContext,
  646. IN PIRP Irp
  647. )
  648. /*++
  649. Routine Description:
  650. This routine performs the unlock volume operation. It is responsible for
  651. either completing of enqueuing the input Irp.
  652. Arguments:
  653. Irp - Supplies the Irp to process
  654. Return Value:
  655. NTSTATUS - The return status for the operation
  656. --*/
  657. {
  658. NTSTATUS Status;
  659. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  660. PVCB Vcb;
  661. PFCB Fcb;
  662. PCCB Ccb;
  663. PAGED_CODE();
  664. //
  665. // Decode the file object, the only type of opens we accept are
  666. // user volume opens.
  667. //
  668. if (UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb ) != UserVolumeOpen ) {
  669. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  670. return STATUS_INVALID_PARAMETER;
  671. }
  672. //
  673. // Acquire exclusive access to the Vcb.
  674. //
  675. Vcb = Fcb->Vcb;
  676. UdfAcquireVcbExclusive( IrpContext, Vcb, FALSE );
  677. //
  678. // We won't check for a valid Vcb for this request. An unlock will always
  679. // succeed on a locked volume.
  680. //
  681. Status = UdfUnlockVolumeInternal( IrpContext, Vcb, IrpSp->FileObject );
  682. //
  683. // Release all of our resources
  684. //
  685. UdfReleaseVcb( IrpContext, Vcb );
  686. //
  687. // Send notification that the volume is avaliable.
  688. //
  689. if (NT_SUCCESS( Status )) {
  690. FsRtlNotifyVolumeEvent( IrpSp->FileObject, FSRTL_VOLUME_UNLOCK );
  691. }
  692. //
  693. // Complete the request if there haven't been any exceptions.
  694. //
  695. UdfCompleteRequest( IrpContext, Irp, Status );
  696. return Status;
  697. }
  698. //
  699. // Local support routine
  700. //
  701. NTSTATUS
  702. UdfDismountVolume (
  703. IN PIRP_CONTEXT IrpContext,
  704. IN PIRP Irp
  705. )
  706. /*++
  707. Routine Description:
  708. This routine performs the dismount volume operation. It is responsible for
  709. either completing of enqueuing the input Irp. We only dismount a volume which
  710. has been locked. The intent here is that someone has locked the volume (they are the
  711. only remaining handle). We set the volume state to invalid so that it will be torn
  712. down quickly.
  713. Arguments:
  714. Irp - Supplies the Irp to process
  715. Return Value:
  716. NTSTATUS - The return status for the operation
  717. --*/
  718. {
  719. NTSTATUS Status;
  720. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  721. PVCB Vcb;
  722. PFCB Fcb;
  723. PCCB Ccb;
  724. PAGED_CODE();
  725. if (UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb ) != UserVolumeOpen ) {
  726. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  727. return STATUS_INVALID_PARAMETER;
  728. }
  729. DebugTrace(( +1, Dbg, "UdfDismountVolume()\n"));
  730. Vcb = Fcb->Vcb;
  731. //
  732. // Acquire exclusive access to the Vcb, and take the global resource to
  733. // sync. against mounts, verifies etc.
  734. //
  735. UdfAcquireUdfData( IrpContext);
  736. UdfAcquireVcbExclusive( IrpContext, Vcb, FALSE );
  737. //
  738. // Mark the volume as invalid, but only do it if the vcb is locked
  739. // by this handle and the volume is currently mounted. No more
  740. // operations will occur on this vcb except cleanup/close.
  741. //
  742. if (Vcb->VcbCondition != VcbMounted) {
  743. Status = STATUS_VOLUME_DISMOUNTED;
  744. }
  745. else {
  746. //
  747. // Invalidate the volume right now.
  748. //
  749. // The intent here is to make every subsequent operation
  750. // on the volume fail and grease the rails toward dismount.
  751. //
  752. UdfLockVcb( IrpContext, Vcb );
  753. if (Vcb->VcbCondition != VcbDismountInProgress) {
  754. Vcb->VcbCondition = VcbInvalid;
  755. }
  756. UdfUnlockVcb( IrpContext, Vcb );
  757. //
  758. // Set flag to tell the close path that we want to force dismount
  759. // the volume when this handle is closed.
  760. //
  761. SetFlag( Ccb->Flags, CCB_FLAG_DISMOUNT_ON_CLOSE);
  762. Status = STATUS_SUCCESS;
  763. }
  764. //
  765. // Release all of our resources
  766. //
  767. UdfReleaseVcb( IrpContext, Vcb );
  768. UdfReleaseUdfData( IrpContext);
  769. DebugTrace(( -1, Dbg, "UdfDismountVolume() -> 0x%x\n", Status));
  770. //
  771. // Complete the request if there haven't been any exceptions.
  772. //
  773. UdfCompleteRequest( IrpContext, Irp, Status );
  774. return Status;
  775. }
  776. NTSTATUS
  777. UdfAllowExtendedDasdIo(
  778. IN PIRP_CONTEXT IrpContext,
  779. IN PIRP Irp
  780. )
  781. /*++
  782. Routine Description:
  783. This routine marks the CCB to indicate that the handle
  784. may be used to read past the end of the volume file. The
  785. handle must be a dasd handle.
  786. Arguments:
  787. Irp - Supplies the Irp being processed.
  788. Return Value:
  789. NTSTATUS - The return status for the operation.
  790. --*/
  791. {
  792. PIO_STACK_LOCATION IrpSp;
  793. NTSTATUS Status = STATUS_SUCCESS;
  794. PFCB Fcb;
  795. PCCB Ccb;
  796. //
  797. // Get the current Irp stack location and save some references.
  798. //
  799. IrpSp = IoGetCurrentIrpStackLocation( Irp );
  800. //
  801. // Extract and decode the file object and check for type of open.
  802. //
  803. if (UserVolumeOpen != UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb )) {
  804. Status = STATUS_INVALID_PARAMETER;
  805. }
  806. else {
  807. SetFlag( Ccb->Flags, CCB_FLAG_ALLOW_EXTENDED_DASD_IO );
  808. }
  809. UdfCompleteRequest( IrpContext, Irp, Status );
  810. return Status;
  811. }
  812. //
  813. // Local support routine
  814. //
  815. UdfIsVolumeDirty (
  816. IN PIRP_CONTEXT IrpContext,
  817. IN PIRP Irp
  818. )
  819. /*++
  820. Routine Description:
  821. This routine determines if a volume is currently dirty.
  822. Arguments:
  823. Irp - Supplies the Irp to process
  824. Return Value:
  825. NTSTATUS - The return status for the operation
  826. --*/
  827. {
  828. PIO_STACK_LOCATION IrpSp;
  829. TYPE_OF_OPEN TypeOfOpen;
  830. PVCB Vcb;
  831. PFCB Fcb;
  832. PCCB Ccb;
  833. PULONG VolumeState;
  834. //
  835. // Get the current stack location and extract the output
  836. // buffer information.
  837. //
  838. IrpSp = IoGetCurrentIrpStackLocation( Irp );
  839. //
  840. // Get a pointer to the output buffer.
  841. //
  842. if (Irp->AssociatedIrp.SystemBuffer != NULL) {
  843. VolumeState = Irp->AssociatedIrp.SystemBuffer;
  844. } else {
  845. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_USER_BUFFER );
  846. return STATUS_INVALID_USER_BUFFER;
  847. }
  848. //
  849. // Make sure the output buffer is large enough and then initialize
  850. // the answer to be that the volume isn't dirty.
  851. //
  852. if (IrpSp->Parameters.FileSystemControl.OutputBufferLength < sizeof(ULONG)) {
  853. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  854. return STATUS_INVALID_PARAMETER;
  855. }
  856. *VolumeState = 0;
  857. //
  858. // Decode the file object
  859. //
  860. TypeOfOpen = UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb );
  861. if (TypeOfOpen != UserVolumeOpen) {
  862. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  863. return STATUS_INVALID_PARAMETER;
  864. }
  865. if (Fcb->Vcb->VcbCondition != VcbMounted) {
  866. UdfCompleteRequest( IrpContext, Irp, STATUS_VOLUME_DISMOUNTED );
  867. return STATUS_VOLUME_DISMOUNTED;
  868. }
  869. //
  870. // Now set up to return the clean state. If we paid attention to the dirty
  871. // state of the media we could be more accurate, but since this is a readonly
  872. // implementation at the moment we think it is clean all of the time.
  873. //
  874. Irp->IoStatus.Information = sizeof( ULONG );
  875. UdfCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
  876. return STATUS_SUCCESS;
  877. }
  878. //
  879. // Local support routine
  880. //
  881. NTSTATUS
  882. UdfIsVolumeMounted (
  883. IN PIRP_CONTEXT IrpContext,
  884. IN PIRP Irp
  885. )
  886. /*++
  887. Routine Description:
  888. This routine determines if a volume is currently mounted.
  889. Arguments:
  890. Irp - Supplies the Irp to process
  891. Return Value:
  892. NTSTATUS - The return status for the operation
  893. --*/
  894. {
  895. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  896. PFCB Fcb;
  897. PCCB Ccb;
  898. PAGED_CODE();
  899. //
  900. // Decode the file object.
  901. //
  902. UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb );
  903. if (Fcb != NULL) {
  904. //
  905. // Disable PopUps, we want to return any error.
  906. //
  907. SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_DISABLE_POPUPS );
  908. //
  909. // Verify the Vcb. This will raise in the error condition.
  910. //
  911. UdfVerifyVcb( IrpContext, Fcb->Vcb );
  912. }
  913. UdfCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
  914. return STATUS_SUCCESS;
  915. }
  916. //
  917. // Local support routine
  918. //
  919. NTSTATUS
  920. UdfIsPathnameValid (
  921. IN PIRP_CONTEXT IrpContext,
  922. IN PIRP Irp
  923. )
  924. /*++
  925. Routine Description:
  926. This routine determines if pathname is a valid UDFS pathname.
  927. We always succeed this request.
  928. Arguments:
  929. Irp - Supplies the Irp to process.
  930. Return Value:
  931. None
  932. --*/
  933. {
  934. PAGED_CODE();
  935. UdfCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
  936. return STATUS_SUCCESS;
  937. }
  938. //
  939. // Local support routine
  940. //
  941. NTSTATUS
  942. UdfInvalidateVolumes (
  943. IN PIRP_CONTEXT IrpContext,
  944. IN PIRP Irp
  945. )
  946. /*++
  947. Routine Description:
  948. This routine searches for all the volumes mounted on the same real device
  949. of the current DASD handle, and marks them all bad. The only operation
  950. that can be done on such handles is cleanup and close.
  951. Arguments:
  952. Irp - Supplies the Irp to process
  953. Return Value:
  954. NTSTATUS - The return status for the operation
  955. --*/
  956. {
  957. NTSTATUS Status;
  958. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  959. KIRQL SavedIrql;
  960. LUID TcbPrivilege = {SE_TCB_PRIVILEGE, 0};
  961. HANDLE Handle;
  962. PVCB Vcb;
  963. PLIST_ENTRY Links;
  964. PFILE_OBJECT FileToMarkBad;
  965. PDEVICE_OBJECT DeviceToMarkBad;
  966. //
  967. // Check for the correct security access.
  968. // The caller must have the SeTcbPrivilege.
  969. //
  970. if (!SeSinglePrivilegeCheck( TcbPrivilege, Irp->RequestorMode )) {
  971. UdfCompleteRequest( IrpContext, Irp, STATUS_PRIVILEGE_NOT_HELD );
  972. return STATUS_PRIVILEGE_NOT_HELD;
  973. }
  974. //
  975. // Try to get a pointer to the device object from the handle passed in.
  976. //
  977. #if defined(_WIN64)
  978. if (IoIs32bitProcess( Irp )) {
  979. if (IrpSp->Parameters.FileSystemControl.InputBufferLength != sizeof( UINT32 )) {
  980. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  981. return STATUS_INVALID_PARAMETER;
  982. }
  983. Handle = (HANDLE) LongToHandle( *((PUINT32) Irp->AssociatedIrp.SystemBuffer) );
  984. } else {
  985. #endif
  986. if (IrpSp->Parameters.FileSystemControl.InputBufferLength != sizeof( HANDLE )) {
  987. UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_PARAMETER );
  988. return STATUS_INVALID_PARAMETER;
  989. }
  990. Handle = *((PHANDLE) Irp->AssociatedIrp.SystemBuffer);
  991. #if defined(_WIN64)
  992. }
  993. #endif
  994. Status = ObReferenceObjectByHandle( Handle,
  995. 0,
  996. *IoFileObjectType,
  997. KernelMode,
  998. &FileToMarkBad,
  999. NULL );
  1000. if (!NT_SUCCESS(Status)) {
  1001. UdfCompleteRequest( IrpContext, Irp, Status );
  1002. return Status;
  1003. }
  1004. //
  1005. // Grab the DeviceObject from the FileObject.
  1006. //
  1007. DeviceToMarkBad = FileToMarkBad->DeviceObject;
  1008. //
  1009. // We only needed the device object involved, not a reference to the file.
  1010. //
  1011. ObDereferenceObject( FileToMarkBad );
  1012. //
  1013. // Make sure this request can wait.
  1014. //
  1015. SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
  1016. ClearFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_FORCE_POST );
  1017. UdfAcquireUdfData( IrpContext );
  1018. //
  1019. // Nothing can go wrong now.
  1020. //
  1021. //
  1022. // Now walk through all the mounted Vcb's looking for candidates to
  1023. // mark invalid.
  1024. //
  1025. // On volumes we mark invalid, check for dismount possibility (which is
  1026. // why we have to get the next link so early).
  1027. //
  1028. Links = UdfData.VcbQueue.Flink;
  1029. while (Links != &UdfData.VcbQueue) {
  1030. Vcb = CONTAINING_RECORD( Links, VCB, VcbLinks);
  1031. Links = Links->Flink;
  1032. //
  1033. // If we get a match, mark the volume Bad, and also check to
  1034. // see if the volume should go away.
  1035. //
  1036. UdfLockVcb( IrpContext, Vcb );
  1037. if (Vcb->Vpb->RealDevice == DeviceToMarkBad) {
  1038. //
  1039. // Take the VPB spinlock, and look to see if this volume is the
  1040. // one currently mounted on the actual device. If it is, pull it
  1041. // off immediately.
  1042. //
  1043. IoAcquireVpbSpinLock( &SavedIrql );
  1044. if (DeviceToMarkBad->Vpb == Vcb->Vpb) {
  1045. PVPB NewVpb = Vcb->SwapVpb;
  1046. ASSERT( FlagOn( Vcb->Vpb->Flags, VPB_MOUNTED));
  1047. ASSERT( NULL != NewVpb);
  1048. RtlZeroMemory( NewVpb, sizeof( VPB ) );
  1049. NewVpb->Type = IO_TYPE_VPB;
  1050. NewVpb->Size = sizeof( VPB );
  1051. NewVpb->RealDevice = DeviceToMarkBad;
  1052. NewVpb->Flags = FlagOn( DeviceToMarkBad->Vpb->Flags, VPB_REMOVE_PENDING );
  1053. DeviceToMarkBad->Vpb = NewVpb;
  1054. Vcb->SwapVpb = NULL;
  1055. }
  1056. IoReleaseVpbSpinLock( SavedIrql );
  1057. if (Vcb->VcbCondition != VcbDismountInProgress) {
  1058. UdfSetVcbCondition( Vcb, VcbInvalid);
  1059. }
  1060. UdfUnlockVcb( IrpContext, Vcb );
  1061. UdfPurgeVolume( IrpContext, Vcb, FALSE );
  1062. UdfCheckForDismount( IrpContext, Vcb, FALSE );
  1063. } else {
  1064. UdfUnlockVcb( IrpContext, Vcb );
  1065. }
  1066. }
  1067. UdfReleaseUdfData( IrpContext );
  1068. UdfCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
  1069. return STATUS_SUCCESS;
  1070. }
  1071. VOID
  1072. UdfRemountOldVcb(
  1073. IN PIRP_CONTEXT IrpContext,
  1074. IN PVCB OldVcb,
  1075. IN PVCB NewVcb,
  1076. IN PDEVICE_OBJECT DeviceObjectWeTalkTo
  1077. )
  1078. {
  1079. KIRQL SavedIrql;
  1080. ObDereferenceObject( OldVcb->TargetDeviceObject );
  1081. IoAcquireVpbSpinLock( &SavedIrql);
  1082. NewVcb->Vpb->RealDevice->Vpb = OldVcb->Vpb;
  1083. OldVcb->Vpb->RealDevice = NewVcb->Vpb->RealDevice;
  1084. OldVcb->TargetDeviceObject = DeviceObjectWeTalkTo;
  1085. UdfSetVcbCondition( OldVcb, VcbMounted);
  1086. UdfSetMediaChangeCount( OldVcb, NewVcb->MediaChangeCount);
  1087. ClearFlag( OldVcb->VcbState, VCB_STATE_VPB_NOT_ON_DEVICE);
  1088. IoReleaseVpbSpinLock( SavedIrql);
  1089. }
  1090. //
  1091. // Local support routine
  1092. //
  1093. NTSTATUS
  1094. UdfMountVolume (
  1095. IN PIRP_CONTEXT IrpContext,
  1096. IN PIRP Irp
  1097. )
  1098. /*++
  1099. Routine Description:
  1100. This routine performs the mount volume operation. It is responsible for
  1101. either completing of enqueuing the input Irp.
  1102. Its job is to verify that the volume denoted in the IRP is a UDF volume,
  1103. and create the VCB and root directory FCB structures. The algorithm it
  1104. uses is essentially as follows:
  1105. 1. Create a new Vcb Structure, and initialize it enough to do I/O
  1106. through the on-disk volume descriptors.
  1107. 2. Read the disk and check if it is a UDF volume.
  1108. 3. If it is not a UDF volume then delete the Vcb and
  1109. complete the IRP with STATUS_UNRECOGNIZED_VOLUME
  1110. 4. Check if the volume was previously mounted and if it was then do a
  1111. remount operation. This involves deleting the VCB, hook in the
  1112. old VCB, and complete the IRP.
  1113. 5. Otherwise create a Vcb and root directory FCB
  1114. Arguments:
  1115. Irp - Supplies the Irp to process
  1116. Return Value:
  1117. NTSTATUS - The return status for the operation
  1118. --*/
  1119. {
  1120. NTSTATUS Status;
  1121. PVOLUME_DEVICE_OBJECT VolDo = NULL;
  1122. PVCB Vcb = NULL;
  1123. PVCB OldVcb = NULL;
  1124. PPCB Pcb = NULL;
  1125. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  1126. PDEVICE_OBJECT DeviceObjectWeTalkTo = IrpSp->Parameters.MountVolume.DeviceObject;
  1127. PVPB Vpb = IrpSp->Parameters.MountVolume.Vpb;
  1128. PFILE_OBJECT FileObjectToNotify = NULL;
  1129. ULONG MediaChangeCount = 0;
  1130. DISK_GEOMETRY DiskGeometry;
  1131. PNSR_ANCHOR AnchorVolumeDescriptor = NULL;
  1132. PNSR_PVD PrimaryVolumeDescriptor = NULL;
  1133. PNSR_LVOL LogicalVolumeDescriptor = NULL;
  1134. PNSR_FSD FileSetDescriptor = NULL;
  1135. BOOLEAN BridgeMedia;
  1136. BOOLEAN SetDoVerifyOnFail;
  1137. USHORT NSRVerFound;
  1138. ULONG BoundS;
  1139. ULONG BoundN;
  1140. PAGED_CODE();
  1141. //
  1142. // Check the input parameters
  1143. //
  1144. ASSERT_IRP_CONTEXT( IrpContext );
  1145. ASSERT_IRP( Irp );
  1146. //
  1147. // Check that we are talking to a Cdrom or Disk device. This request should
  1148. // always be waitable.
  1149. //
  1150. ASSERT( Vpb->RealDevice->DeviceType == FILE_DEVICE_CD_ROM ||
  1151. Vpb->RealDevice->DeviceType == FILE_DEVICE_DISK );
  1152. ASSERT( FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT ));
  1153. DebugTrace(( +1, Dbg | UDFS_DEBUG_LEVEL_VERFYSUP, "UdfMountVolume (Vpb %p, Dev %p)\n",
  1154. Vpb, Vpb->RealDevice));
  1155. //
  1156. // Update the real device in the IrpContext from the Vpb. There was no available
  1157. // file object when the IrpContext was created.
  1158. //
  1159. IrpContext->RealDevice = Vpb->RealDevice;
  1160. SetDoVerifyOnFail = UdfRealDevNeedsVerify( IrpContext->RealDevice);
  1161. //
  1162. // Check if we have disabled the mount process.
  1163. //
  1164. if (UdfDisable) {
  1165. UdfCompleteRequest( IrpContext, Irp, STATUS_UNRECOGNIZED_VOLUME );
  1166. DebugTrace(( 0, Dbg, "UdfMountVolume, disabled\n" ));
  1167. DebugTrace(( -1, Dbg, "UdfMountVolume -> STATUS_UNRECOGNIZED_VOLUME\n" ));
  1168. return STATUS_UNRECOGNIZED_VOLUME;
  1169. }
  1170. //
  1171. // Don't even attempt to mount floppy discs
  1172. //
  1173. if (FlagOn( Vpb->RealDevice->Characteristics, FILE_FLOPPY_DISKETTE)) {
  1174. UdfCompleteRequest( IrpContext, Irp, STATUS_UNRECOGNIZED_VOLUME );
  1175. return STATUS_UNRECOGNIZED_VOLUME;
  1176. }
  1177. //
  1178. // Do a CheckVerify here to lift the MediaChange ticker from the driver
  1179. //
  1180. Status = UdfPerformDevIoCtrl( IrpContext,
  1181. ( Vpb->RealDevice->DeviceType == FILE_DEVICE_CD_ROM ?
  1182. IOCTL_CDROM_CHECK_VERIFY :
  1183. IOCTL_DISK_CHECK_VERIFY ),
  1184. DeviceObjectWeTalkTo,
  1185. NULL,
  1186. 0,
  1187. &MediaChangeCount,
  1188. sizeof(ULONG),
  1189. FALSE,
  1190. TRUE,
  1191. NULL );
  1192. if (!NT_SUCCESS( Status )) {
  1193. UdfCompleteRequest( IrpContext, Irp, Status );
  1194. DebugTrace(( 0, Dbg,
  1195. "UdfMountVolume, CHECK_VERIFY handed back status %08x (so don't continue)\n",
  1196. Status ));
  1197. DebugTrace(( -1, Dbg,
  1198. "UdfMountVolume -> %08x\n",
  1199. Status ));
  1200. return Status;
  1201. }
  1202. //
  1203. // Now let's make Jeff delirious and call to get the disk geometry. This
  1204. // will fix the case where the first change line is swallowed.
  1205. //
  1206. // This IOCTL does not have a generic STORAGE equivalent, so we must figure
  1207. // our which variant to pass down from the real underlying device object (as
  1208. // opposed to the top of the driver filter stack we will really be attaching
  1209. // on top of).
  1210. //
  1211. Status = UdfPerformDevIoCtrl( IrpContext,
  1212. ( Vpb->RealDevice->DeviceType == FILE_DEVICE_CD_ROM ?
  1213. IOCTL_CDROM_GET_DRIVE_GEOMETRY :
  1214. IOCTL_DISK_GET_DRIVE_GEOMETRY ),
  1215. DeviceObjectWeTalkTo,
  1216. NULL,
  1217. 0,
  1218. &DiskGeometry,
  1219. sizeof( DISK_GEOMETRY ),
  1220. FALSE,
  1221. TRUE,
  1222. NULL );
  1223. //
  1224. // If this call failed, we might be able to get away with a heuristic guess as to
  1225. // what the sector size is (per CDFS), but that is playing with fire. Nearly every
  1226. // failure here will be a permanent problem of some form.
  1227. //
  1228. if (!NT_SUCCESS( Status )) {
  1229. UdfCompleteRequest( IrpContext, Irp, Status );
  1230. DebugTrace(( 0, Dbg, "UdfMountVolume, GET_DRIVE_GEOMETRY failed\n" ));
  1231. DebugTrace(( -1, Dbg,
  1232. "UdfMountVolume -> %08x\n",
  1233. Status ));
  1234. return Status;
  1235. }
  1236. //
  1237. // Acquire the global resource to do mount operations.
  1238. //
  1239. UdfAcquireUdfData( IrpContext );
  1240. //
  1241. // Use a try-finally to facilitate cleanup.
  1242. //
  1243. try {
  1244. //
  1245. // Do a quick check to see if there any Vcb's which can be removed.
  1246. //
  1247. UdfScanForDismountedVcb( IrpContext );
  1248. //
  1249. // Make sure that the driver/drive is not screwing up underneath of us by
  1250. // feeding us garbage for the sector size.
  1251. //
  1252. if (DiskGeometry.BytesPerSector == 0 ||
  1253. (DiskGeometry.BytesPerSector & ~( 1 << UdfHighBit( DiskGeometry.BytesPerSector ))) != 0) {
  1254. DebugTrace(( 0, 0,
  1255. "UdfMountVolume, bad DiskGeometry (%08x) .BytesPerSector == %08x\n",
  1256. &DiskGeometry,
  1257. DiskGeometry.BytesPerSector ));
  1258. ASSERT( FALSE );
  1259. try_leave( Status = STATUS_DRIVER_INTERNAL_ERROR );
  1260. }
  1261. //
  1262. // Now find the multi-session bounds on this media.
  1263. //
  1264. UdfDetermineVolumeBounding( IrpContext,
  1265. DeviceObjectWeTalkTo,
  1266. &BoundS,
  1267. &BoundN );
  1268. //
  1269. // Now go confirm that this volume may be a UDF image by looking for a
  1270. // valid ISO 13346 Volume Recognition Sequence in the last and first
  1271. // sessions.
  1272. //
  1273. if (!UdfRecognizeVolume( IrpContext,
  1274. DeviceObjectWeTalkTo,
  1275. DiskGeometry.BytesPerSector,
  1276. &BoundS,
  1277. &BridgeMedia,
  1278. &NSRVerFound)) {
  1279. DebugTrace(( 0, Dbg, "UdfMountVolume, recognition failed so not mounting\n" ));
  1280. try_leave( Status = STATUS_UNRECOGNIZED_VOLUME );
  1281. }
  1282. //
  1283. // Create the DeviceObject for this mount attempt
  1284. //
  1285. Status = IoCreateDevice( UdfData.DriverObject,
  1286. sizeof( VOLUME_DEVICE_OBJECT ) - sizeof( DEVICE_OBJECT ),
  1287. NULL,
  1288. FILE_DEVICE_CD_ROM_FILE_SYSTEM,
  1289. 0,
  1290. FALSE,
  1291. (PDEVICE_OBJECT *) &VolDo );
  1292. if (!NT_SUCCESS( Status )) {
  1293. DebugTrace(( 0, Dbg, "UdfMountVolume, couldn't get voldo! (%08x)\n", Status ));
  1294. try_leave( Status );
  1295. }
  1296. //
  1297. // Our alignment requirement is the larger of the processor alignment requirement
  1298. // already in the volume device object and that in the DeviceObjectWeTalkTo
  1299. //
  1300. if (DeviceObjectWeTalkTo->AlignmentRequirement > VolDo->DeviceObject.AlignmentRequirement) {
  1301. VolDo->DeviceObject.AlignmentRequirement = DeviceObjectWeTalkTo->AlignmentRequirement;
  1302. }
  1303. //
  1304. // Initialize the overflow queue for the volume
  1305. //
  1306. VolDo->OverflowQueueCount = 0;
  1307. InitializeListHead( &VolDo->OverflowQueue );
  1308. VolDo->PostedRequestCount = 0;
  1309. KeInitializeSpinLock( &VolDo->OverflowQueueSpinLock );
  1310. //
  1311. // Now before we can initialize the Vcb we need to set up the
  1312. // device object field in the VPB to point to our new volume device
  1313. // object.
  1314. //
  1315. Vpb->DeviceObject = (PDEVICE_OBJECT) VolDo;
  1316. //
  1317. // Initialize the Vcb. This routine will raise on an allocation
  1318. // failure.
  1319. //
  1320. UdfInitializeVcb( IrpContext,
  1321. &VolDo->Vcb,
  1322. DeviceObjectWeTalkTo,
  1323. Vpb,
  1324. &DiskGeometry,
  1325. MediaChangeCount );
  1326. //
  1327. // We must initialize the stack size in our device object before
  1328. // the following reads, because the I/O system has not done it yet.
  1329. //
  1330. ((PDEVICE_OBJECT) VolDo)->StackSize = (CCHAR) (DeviceObjectWeTalkTo->StackSize + 1);
  1331. //
  1332. // Set the correct sector size. IO defaults to 512b for DISK_FS and 2k for
  1333. // CDROM_FS....
  1334. //
  1335. ((PDEVICE_OBJECT) VolDo)->SectorSize = (USHORT) DiskGeometry.BytesPerSector;
  1336. //
  1337. // Pick up a local pointer to the new Vcb. Here is where we start
  1338. // thinking about cleanup of structures if the mount is failed.
  1339. //
  1340. ClearFlag( VolDo->DeviceObject.Flags, DO_DEVICE_INITIALIZING );
  1341. Vcb = &VolDo->Vcb;
  1342. Vpb = NULL;
  1343. VolDo = NULL;
  1344. //
  1345. // Store the session bounds we determined earlier.
  1346. //
  1347. Vcb->BoundS = BoundS;
  1348. Vcb->BoundN = BoundN;
  1349. //
  1350. // Store the Vcb in the IrpContext as we didn't have one before.
  1351. //
  1352. IrpContext->Vcb = Vcb;
  1353. UdfAcquireVcbExclusive( IrpContext, Vcb, FALSE );
  1354. //
  1355. // Store the NSR version that we found
  1356. //
  1357. Vcb->NsrVersion = NSRVerFound;
  1358. //
  1359. // Let's reference the Vpb to make sure we are the one to
  1360. // have the last dereference.
  1361. //
  1362. Vcb->Vpb->ReferenceCount += 1;
  1363. //
  1364. // Clear the verify bit for the start of mount.
  1365. //
  1366. UdfMarkRealDevVerifyOk( Vcb->Vpb->RealDevice);
  1367. //
  1368. // Now find the Anchor Volume Descriptor so we can discover the Volume Set
  1369. // Descriptor Sequence extent.
  1370. //
  1371. Status = UdfFindAnchorVolumeDescriptor( IrpContext,
  1372. Vcb,
  1373. &AnchorVolumeDescriptor );
  1374. if (!NT_SUCCESS(Status)) {
  1375. DebugTrace(( 0, Dbg, "UdfMountVolume, couldn't find anchor descriptors\n" ));
  1376. try_leave( Status );
  1377. }
  1378. //
  1379. // Now search for the prevailing copies of the PVD, LVD, and related PD in the
  1380. // extents indicated by the AVD.
  1381. //
  1382. Status = UdfFindVolumeDescriptors( IrpContext,
  1383. Vcb,
  1384. &AnchorVolumeDescriptor->Main,
  1385. &Pcb,
  1386. &PrimaryVolumeDescriptor,
  1387. &LogicalVolumeDescriptor );
  1388. //
  1389. // If we discovered invalid structures on the main extent, we may still
  1390. // be able to use the reserve extent. By definition the two extents
  1391. // must be logically equal, so just plow into it on any error.
  1392. //
  1393. if (!NT_SUCCESS( Status )) {
  1394. Status = UdfFindVolumeDescriptors( IrpContext,
  1395. Vcb,
  1396. &AnchorVolumeDescriptor->Reserve,
  1397. &Pcb,
  1398. &PrimaryVolumeDescriptor,
  1399. &LogicalVolumeDescriptor );
  1400. }
  1401. if (!NT_SUCCESS(Status)) {
  1402. DebugTrace(( 0, Dbg, "UdfMountVolume, couldn't find good VSD descriptors (PVD/LVD/PD) status %X\n", Status ));
  1403. try_leave( Status );
  1404. }
  1405. //
  1406. // Now go complete initialization of the Pcb. After this point, we can perform
  1407. // physical partition mappings and know that the partition table is good.
  1408. //
  1409. Status = UdfCompletePcb( IrpContext,
  1410. Vcb,
  1411. Pcb );
  1412. if (!NT_SUCCESS(Status)) {
  1413. DebugTrace(( 0, Dbg, "UdfMountVolume, Pcb completion failed\n" ));
  1414. try_leave( Status );
  1415. }
  1416. Vcb->Pcb = Pcb;
  1417. Pcb = NULL;
  1418. //
  1419. // Set up all the support we need to do reads into the volume.
  1420. //
  1421. UdfUpdateVcbPhase0( IrpContext, Vcb );
  1422. //
  1423. // Now go get the fileset descriptor that will finally reveal the location
  1424. // of the root directory on this volume.
  1425. //
  1426. Status = UdfFindFileSetDescriptor( IrpContext,
  1427. Vcb,
  1428. &LogicalVolumeDescriptor->FSD,
  1429. &FileSetDescriptor );
  1430. if (!NT_SUCCESS(Status)) {
  1431. try_leave( NOTHING );
  1432. }
  1433. //
  1434. // Now that we have everything together, update the Vpb with identification
  1435. // of this volume.
  1436. //
  1437. UdfUpdateVolumeLabel( IrpContext,
  1438. Vcb->Vpb->VolumeLabel,
  1439. &Vcb->Vpb->VolumeLabelLength,
  1440. LogicalVolumeDescriptor->VolumeID,
  1441. sizeof( LogicalVolumeDescriptor->VolumeID ));
  1442. UdfUpdateVolumeSerialNumber( IrpContext,
  1443. &Vcb->Vpb->SerialNumber,
  1444. FileSetDescriptor );
  1445. //
  1446. // Check if this is a remount operation. If so then clean up
  1447. // the data structures passed in and created here.
  1448. //
  1449. if (UdfIsRemount( IrpContext, Vcb, &OldVcb )) {
  1450. KIRQL SavedIrql;
  1451. DebugTrace((0, Dbg | UDFS_DEBUG_LEVEL_VERFYSUP, "Remounting Vcb %p (Vpb %p)\n",
  1452. OldVcb , OldVcb->Vpb));
  1453. //
  1454. // Link the old Vcb to point to the new device object that we
  1455. // should be talking to, dereferencing the previous. Call a nonpaged
  1456. // routine to do this since we take the Vpb spinlock.
  1457. //
  1458. UdfRemountOldVcb( IrpContext,
  1459. OldVcb,
  1460. Vcb,
  1461. DeviceObjectWeTalkTo);
  1462. //
  1463. // Push the state of the method 2 bit across. In changing the device,
  1464. // we may now be on one with a different requirement.
  1465. //
  1466. ClearFlag( OldVcb->VcbState, VCB_STATE_METHOD_2_FIXUP );
  1467. SetFlag( OldVcb->VcbState, FlagOn( Vcb->VcbState, VCB_STATE_METHOD_2_FIXUP ));
  1468. //
  1469. // See if we will need to provide notification of the remount. This is the readonly
  1470. // filesystem's form of dismount/mount notification - we promise that whenever a
  1471. // volume is "dismounted", that a mount notification will occur when it is revalidated.
  1472. // Note that we do not send mount on normal remounts - that would duplicate the media
  1473. // arrival notification of the device driver.
  1474. //
  1475. if (FlagOn( OldVcb->VcbState, VCB_STATE_NOTIFY_REMOUNT )) {
  1476. ClearFlag( OldVcb->VcbState, VCB_STATE_NOTIFY_REMOUNT );
  1477. FileObjectToNotify = OldVcb->RootIndexFcb->FileObject;
  1478. ObReferenceObject( FileObjectToNotify );
  1479. }
  1480. DebugTrace(( 0, Dbg, "UdfMountVolume, remounted old Vcb %08x\n", OldVcb ));
  1481. try_leave( Status = STATUS_SUCCESS );
  1482. }
  1483. //
  1484. // Initialize the Vcb and associated structures from our volume descriptors
  1485. //
  1486. UdfUpdateVcbPhase1( IrpContext,
  1487. Vcb,
  1488. FileSetDescriptor );
  1489. //
  1490. // Drop an extra reference on the root dir file so we'll be able to send
  1491. // notification.
  1492. //
  1493. if (Vcb->RootIndexFcb) {
  1494. FileObjectToNotify = Vcb->RootIndexFcb->FileObject;
  1495. ObReferenceObject( FileObjectToNotify );
  1496. }
  1497. //
  1498. // The new mount is complete. Remove the additional references on this
  1499. // Vcb since, at this point, we have added the real references this volume
  1500. // will have during its lifetime. We also need to drop the additional
  1501. // reference on the device we mounted.
  1502. //
  1503. Vcb->VcbReference -= Vcb->VcbResidualReference;
  1504. ASSERT( Vcb->VcbReference == Vcb->VcbResidualReference );
  1505. ObDereferenceObject( Vcb->TargetDeviceObject );
  1506. UdfSetVcbCondition( Vcb, VcbMounted);
  1507. UdfReleaseVcb( IrpContext, Vcb );
  1508. Vcb = NULL;
  1509. Status = STATUS_SUCCESS;
  1510. } finally {
  1511. DebugUnwind( "UdfMountVolume" );
  1512. //
  1513. // If we are not mounting the device, then set the verify bit again.
  1514. //
  1515. if ((AbnormalTermination() || (Status != STATUS_SUCCESS)) &&
  1516. SetDoVerifyOnFail) {
  1517. UdfMarkRealDevForVerify( IrpContext->RealDevice);
  1518. }
  1519. //
  1520. // If we didn't complete the mount then cleanup any remaining structures.
  1521. //
  1522. if (Vpb != NULL) { Vpb->DeviceObject = NULL; }
  1523. if (Pcb != NULL) {
  1524. UdfDeletePcb( Pcb );
  1525. }
  1526. if (Vcb != NULL) {
  1527. //
  1528. // Make sure there is no Vcb in the IrpContext since it could go away
  1529. //
  1530. IrpContext->Vcb = NULL;
  1531. Vcb->VcbReference -= Vcb->VcbResidualReference;
  1532. if (UdfDismountVcb( IrpContext, Vcb )) {
  1533. UdfReleaseVcb( IrpContext, Vcb );
  1534. }
  1535. } else if (VolDo != NULL) {
  1536. IoDeleteDevice( (PDEVICE_OBJECT)VolDo );
  1537. Vpb->DeviceObject = NULL;
  1538. }
  1539. //
  1540. // Release the global resource.
  1541. //
  1542. UdfReleaseUdfData( IrpContext );
  1543. //
  1544. // Free any structures we may have been allocated
  1545. //
  1546. UdfFreePool( &AnchorVolumeDescriptor );
  1547. UdfFreePool( &PrimaryVolumeDescriptor );
  1548. UdfFreePool( &LogicalVolumeDescriptor );
  1549. UdfFreePool( &FileSetDescriptor );
  1550. }
  1551. //
  1552. // Now send mount notification.
  1553. //
  1554. if (FileObjectToNotify) {
  1555. FsRtlNotifyVolumeEvent( FileObjectToNotify, FSRTL_VOLUME_MOUNT );
  1556. ObDereferenceObject( FileObjectToNotify );
  1557. }
  1558. //
  1559. // Complete the request if no exception.
  1560. //
  1561. UdfCompleteRequest( IrpContext, Irp, Status );
  1562. DebugTrace(( -1, Dbg, "UdfMountVolume -> %08x\n", Status ));
  1563. return Status;
  1564. }
  1565. //
  1566. // Local support routine
  1567. //
  1568. NTSTATUS
  1569. UdfVerifyVolume (
  1570. IN PIRP_CONTEXT IrpContext,
  1571. IN PIRP Irp
  1572. )
  1573. /*++
  1574. Routine Description:
  1575. This routine performs the verify volume operation. It is responsible for
  1576. either completing of enqueuing the input Irp.
  1577. Arguments:
  1578. Irp - Supplies the Irp to process
  1579. Return Value:
  1580. NTSTATUS - The return status for the operation
  1581. --*/
  1582. {
  1583. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  1584. PVPB Vpb = IrpSp->Parameters.VerifyVolume.Vpb;
  1585. PVCB Vcb = &((PVOLUME_DEVICE_OBJECT) IrpSp->Parameters.VerifyVolume.DeviceObject)->Vcb;
  1586. PPCB Pcb = NULL;
  1587. PNSR_ANCHOR AnchorVolumeDescriptor = NULL;
  1588. PNSR_PVD PrimaryVolumeDescriptor = NULL;
  1589. PNSR_LVOL LogicalVolumeDescriptor = NULL;
  1590. PNSR_FSD FileSetDescriptor = NULL;
  1591. ULONG MediaChangeCount = Vcb->MediaChangeCount;
  1592. ULONG Index;
  1593. PFILE_OBJECT FileObjectToNotify = NULL;
  1594. BOOLEAN ReturnError;
  1595. BOOLEAN ReleaseVcb;
  1596. IO_STATUS_BLOCK Iosb;
  1597. WCHAR VolumeLabel[ MAXIMUM_VOLUME_LABEL_LENGTH / sizeof( WCHAR )];
  1598. USHORT VolumeLabelLength;
  1599. ULONG VolumeSerialNumber;
  1600. NTSTATUS Status;
  1601. PAGED_CODE();
  1602. //
  1603. // Check input.
  1604. //
  1605. ASSERT_IRP_CONTEXT( IrpContext );
  1606. //
  1607. // Check that we are talking to a Cdrom or Disk device. This request should
  1608. // always be waitable.
  1609. //
  1610. ASSERT( Vpb->RealDevice->DeviceType == FILE_DEVICE_CD_ROM ||
  1611. Vpb->RealDevice->DeviceType == FILE_DEVICE_DISK );
  1612. ASSERT_VCB( Vcb );
  1613. //
  1614. // Update the real device in the IrpContext from the Vpb. There was no available
  1615. // file object when the IrpContext was created.
  1616. //
  1617. IrpContext->RealDevice = Vpb->RealDevice;
  1618. //
  1619. // Acquire the global to synchronise against mounts and teardown.
  1620. //
  1621. UdfAcquireUdfData( IrpContext );
  1622. UdfAcquireVcbExclusive( IrpContext, Vcb, FALSE );
  1623. ReleaseVcb = TRUE;
  1624. DebugTrace(( +1, Dbg, "UdfVerifyVolume, Vcb %08x\n", Vcb ));
  1625. try {
  1626. //
  1627. // Verify that there is a disk here.
  1628. //
  1629. Status = UdfPerformDevIoCtrl( IrpContext,
  1630. ( Vpb->RealDevice->DeviceType == FILE_DEVICE_CD_ROM ?
  1631. IOCTL_CDROM_CHECK_VERIFY :
  1632. IOCTL_DISK_CHECK_VERIFY ),
  1633. Vcb->TargetDeviceObject,
  1634. NULL,
  1635. 0,
  1636. &MediaChangeCount,
  1637. sizeof(ULONG),
  1638. FALSE,
  1639. TRUE,
  1640. &Iosb );
  1641. if (!NT_SUCCESS( Status )) {
  1642. DebugTrace(( 0, Dbg, "UdfVerifyVolume, CHECK_VERIFY failed\n" ));
  1643. //
  1644. // If we will allow a raw mount then return WRONG_VOLUME to
  1645. // allow the volume to be mounted by raw.
  1646. //
  1647. if (FlagOn( IrpSp->Flags, SL_ALLOW_RAW_MOUNT )) {
  1648. DebugTrace(( 0, Dbg, "UdfVerifyVolume, ... allowing raw mount\n" ));
  1649. Status = STATUS_WRONG_VOLUME;
  1650. }
  1651. try_leave( Status );
  1652. }
  1653. if (Iosb.Information != sizeof(ULONG)) {
  1654. //
  1655. // Be safe about the count in case the driver didn't fill it in
  1656. //
  1657. MediaChangeCount = 0;
  1658. }
  1659. //
  1660. // Verify that the device actually saw a change. If the driver does not
  1661. // support the MCC, then we must verify the volume in any case.
  1662. //
  1663. if (MediaChangeCount == 0 || (Vcb->MediaChangeCount != MediaChangeCount)) {
  1664. //
  1665. // Now we need to navigate the disc to find the relavent decriptors. This is
  1666. // much the same as the mount process.
  1667. //
  1668. //
  1669. // Find the AVD.
  1670. //
  1671. Status = UdfFindAnchorVolumeDescriptor( IrpContext,
  1672. Vcb,
  1673. &AnchorVolumeDescriptor );
  1674. if (!NT_SUCCESS(Status)) {
  1675. DebugTrace(( 0, Dbg, "UdfVerifyVolume, No AVD visible\n" ));
  1676. try_leave( Status = STATUS_WRONG_VOLUME );
  1677. }
  1678. //
  1679. // Get the prevailing descriptors out of the VDS, building a fresh Pcb.
  1680. //
  1681. Status = UdfFindVolumeDescriptors( IrpContext,
  1682. Vcb,
  1683. &AnchorVolumeDescriptor->Main,
  1684. &Pcb,
  1685. &PrimaryVolumeDescriptor,
  1686. &LogicalVolumeDescriptor );
  1687. //
  1688. // Try the reserve sequence in case of error.
  1689. //
  1690. if (Status == STATUS_DISK_CORRUPT_ERROR) {
  1691. Status = UdfFindVolumeDescriptors( IrpContext,
  1692. Vcb,
  1693. &AnchorVolumeDescriptor->Reserve,
  1694. &Pcb,
  1695. &PrimaryVolumeDescriptor,
  1696. &LogicalVolumeDescriptor );
  1697. }
  1698. //
  1699. // If we're totally unable to find a VDS, give up.
  1700. //
  1701. if (!NT_SUCCESS(Status)) {
  1702. DebugTrace(( 0, Dbg, "UdfVerifyVolume, PVD/LVD/PD pickup failed\n" ));
  1703. try_leave( Status = STATUS_WRONG_VOLUME );
  1704. }
  1705. //
  1706. // Now go complete initialization of the Pcb so we can compare it.
  1707. //
  1708. Status = UdfCompletePcb( IrpContext,
  1709. Vcb,
  1710. Pcb );
  1711. if (!NT_SUCCESS(Status)) {
  1712. DebugTrace(( 0, Dbg, "UdfVerifyVolume, Pcb completion failed\n" ));
  1713. try_leave( Status = STATUS_WRONG_VOLUME );
  1714. }
  1715. //
  1716. // Now let's compare this new Pcb to the previous Vcb's Pcb to see if they
  1717. // appear to be equivalent.
  1718. //
  1719. if (!UdfEquivalentPcb( IrpContext,
  1720. Pcb,
  1721. Vcb->Pcb)) {
  1722. DebugTrace(( 0, Dbg, "UdfVerifyVolume, Pcbs are not equivalent\n" ));
  1723. try_leave( Status = STATUS_WRONG_VOLUME );
  1724. }
  1725. //
  1726. // At this point we know that the Vcb's Pcb is OK for mapping to find the fileset
  1727. // descriptor, so we can drop the new one we built for comparison purposes.
  1728. //
  1729. UdfDeletePcb( Pcb );
  1730. Pcb = NULL;
  1731. //
  1732. // Go pick up the fileset descriptor.
  1733. //
  1734. Status = UdfFindFileSetDescriptor( IrpContext,
  1735. Vcb,
  1736. &LogicalVolumeDescriptor->FSD,
  1737. &FileSetDescriptor );
  1738. if (!NT_SUCCESS(Status)) {
  1739. try_leave( Status = STATUS_WRONG_VOLUME );
  1740. }
  1741. //
  1742. // Now that everything is in place, build a volume label and serial number from these
  1743. // descriptors and perform the final check that this Vcb is (or is not) the right one
  1744. // for the media now in the drive.
  1745. //
  1746. UdfUpdateVolumeLabel( IrpContext,
  1747. VolumeLabel,
  1748. &VolumeLabelLength,
  1749. LogicalVolumeDescriptor->VolumeID,
  1750. sizeof( LogicalVolumeDescriptor->VolumeID ));
  1751. UdfUpdateVolumeSerialNumber( IrpContext,
  1752. &VolumeSerialNumber,
  1753. FileSetDescriptor );
  1754. if ((Vcb->Vpb->SerialNumber != VolumeSerialNumber) ||
  1755. (Vcb->Vpb->VolumeLabelLength != VolumeLabelLength) ||
  1756. (VolumeLabelLength != RtlCompareMemory( Vcb->Vpb->VolumeLabel,
  1757. VolumeLabel,
  1758. VolumeLabelLength))) {
  1759. DebugTrace(( 0, Dbg, "UdfVerifyVolume, volume label/sn mismatch\n" ));
  1760. try_leave( Status = STATUS_WRONG_VOLUME );
  1761. }
  1762. }
  1763. //
  1764. // The volume is OK, clear the verify bit.
  1765. //
  1766. DebugTrace(( 0, Dbg, "UdfVerifyVolume, looks like the same volume\n" ));
  1767. UdfSetVcbCondition( Vcb, VcbMounted);
  1768. UdfMarkRealDevVerifyOk( Vpb->RealDevice);
  1769. //
  1770. // See if we will need to provide notification of the remount. This is the readonly
  1771. // filesystem's form of dismount/mount notification.
  1772. //
  1773. if (FlagOn( Vcb->VcbState, VCB_STATE_NOTIFY_REMOUNT )) {
  1774. ClearFlag( Vcb->VcbState, VCB_STATE_NOTIFY_REMOUNT );
  1775. FileObjectToNotify = Vcb->RootIndexFcb->FileObject;
  1776. ObReferenceObject( FileObjectToNotify );
  1777. }
  1778. } finally {
  1779. //
  1780. // If we did not raise an exception, update the current Vcb.
  1781. //
  1782. if (!AbnormalTermination()) {
  1783. //
  1784. // Update the media change count to note that we have verified the volume
  1785. // at this value
  1786. //
  1787. UdfSetMediaChangeCount( Vcb, MediaChangeCount);
  1788. //
  1789. // Mark the Vcb as not mounted.
  1790. //
  1791. if (Status == STATUS_WRONG_VOLUME) {
  1792. UdfSetVcbCondition( Vcb, VcbNotMounted);
  1793. //
  1794. // Now, if there are no user handles to the volume, try to spark
  1795. // teardown by purging the volume.
  1796. //
  1797. if (Vcb->VcbCleanup == 0) {
  1798. if (NT_SUCCESS( UdfPurgeVolume( IrpContext, Vcb, FALSE ))) {
  1799. ReleaseVcb = UdfCheckForDismount( IrpContext, Vcb, FALSE );
  1800. }
  1801. }
  1802. }
  1803. }
  1804. DebugTrace(( -1, Dbg, "UdfVerifyVolume -> %08x\n", Status ));
  1805. if (ReleaseVcb) {
  1806. UdfReleaseVcb( IrpContext, Vcb );
  1807. }
  1808. UdfReleaseUdfData( IrpContext );
  1809. //
  1810. // Delete the Pcb if built.
  1811. //
  1812. if (Pcb != NULL) {
  1813. UdfDeletePcb( Pcb );
  1814. }
  1815. UdfFreePool( &AnchorVolumeDescriptor );
  1816. UdfFreePool( &PrimaryVolumeDescriptor );
  1817. UdfFreePool( &LogicalVolumeDescriptor );
  1818. UdfFreePool( &FileSetDescriptor );
  1819. }
  1820. //
  1821. // Now send mount notification.
  1822. //
  1823. if (FileObjectToNotify) {
  1824. FsRtlNotifyVolumeEvent( FileObjectToNotify, FSRTL_VOLUME_MOUNT );
  1825. ObDereferenceObject( FileObjectToNotify );
  1826. }
  1827. //
  1828. // Complete the request if no exception.
  1829. //
  1830. UdfCompleteRequest( IrpContext, Irp, Status );
  1831. return Status;
  1832. }
  1833. //
  1834. // Local support routine
  1835. //
  1836. BOOLEAN
  1837. UdfIsRemount (
  1838. IN PIRP_CONTEXT IrpContext,
  1839. IN PVCB Vcb,
  1840. OUT PVCB *OldVcb
  1841. )
  1842. /*++
  1843. Routine Description:
  1844. This routine walks through the links of the Vcb chain in the global
  1845. data structure. The remount condition is met when the following
  1846. conditions are all met:
  1847. 1 - The 32 serial for this VPB matches that in a previous
  1848. VPB.
  1849. 2 - The volume label for this VPB matches that in the previous
  1850. VPB.
  1851. 3 - The system pointer to the real device object in the current
  1852. VPB matches that in the same previous VPB.
  1853. 4 - Finally the previous Vcb cannot be invalid or have a dismount
  1854. underway.
  1855. If a VPB is found which matches these conditions, then the address of
  1856. the Vcb for that VPB is returned via the pointer OldVcb.
  1857. Skip over the current Vcb.
  1858. Arguments:
  1859. Vcb - This is the Vcb we are checking for a remount.
  1860. OldVcb - A pointer to the address to store the address for the Vcb
  1861. for the volume if this is a remount. (This is a pointer to
  1862. a pointer)
  1863. Return Value:
  1864. BOOLEAN - TRUE if this is in fact a remount, FALSE otherwise.
  1865. --*/
  1866. {
  1867. PLIST_ENTRY Link;
  1868. PVPB Vpb = Vcb->Vpb;
  1869. PVPB OldVpb;
  1870. BOOLEAN Remount = FALSE;
  1871. PAGED_CODE();
  1872. //
  1873. // Check input.
  1874. //
  1875. ASSERT_IRP_CONTEXT( IrpContext );
  1876. ASSERT_VCB( Vcb );
  1877. DebugTrace(( +1, Dbg, "UdfIsRemount, Vcb %08x\n", Vcb ));
  1878. for (Link = UdfData.VcbQueue.Flink;
  1879. Link != &UdfData.VcbQueue;
  1880. Link = Link->Flink) {
  1881. *OldVcb = CONTAINING_RECORD( Link, VCB, VcbLinks );
  1882. //
  1883. // Skip ourselves.
  1884. //
  1885. if (Vcb == *OldVcb) { continue; }
  1886. //
  1887. // Look at the Vpb and state of the previous Vcb.
  1888. //
  1889. OldVpb = (*OldVcb)->Vpb;
  1890. if ((OldVpb != Vpb) &&
  1891. (OldVpb->RealDevice == Vpb->RealDevice) &&
  1892. ((*OldVcb)->VcbCondition == VcbNotMounted)) {
  1893. //
  1894. // Go ahead and compare serial numbers and volume label.
  1895. //
  1896. if ((OldVpb->SerialNumber == Vpb->SerialNumber) &&
  1897. (Vpb->VolumeLabelLength == OldVpb->VolumeLabelLength) &&
  1898. (RtlEqualMemory( OldVpb->VolumeLabel,
  1899. Vpb->VolumeLabel,
  1900. Vpb->VolumeLabelLength ))) {
  1901. //
  1902. // Got it.
  1903. //
  1904. DebugTrace(( 0, Dbg, "UdfIsRemount, matched OldVcb %08x\n", *OldVcb ));
  1905. Remount = TRUE;
  1906. break;
  1907. }
  1908. }
  1909. }
  1910. DebugTrace(( -1, Dbg, "UdfIsRemount -> %c\n", (Remount? 'T' : 'F' )));
  1911. return Remount;
  1912. }
  1913. //
  1914. // Local support routine
  1915. //
  1916. NTSTATUS
  1917. UdfFindFileSetDescriptor (
  1918. IN PIRP_CONTEXT IrpContext,
  1919. IN PVCB Vcb,
  1920. IN PLONGAD LongAd,
  1921. IN OUT PNSR_FSD *FileSetDescriptor
  1922. )
  1923. /*++
  1924. Routine Description:
  1925. This routine walks a Fileset Descriptor Sequence looking for the default
  1926. descriptor. This will reveal the location of the root directory on the
  1927. volume.
  1928. Arguments:
  1929. Vcb - Vcb of volume to search
  1930. LongAd - Long allocation descriptor describing the start of the sequence
  1931. FileSetDescriptor - Address of caller's pointer to an FSD
  1932. Return Value:
  1933. STATUS_SUCCESS if all descriptors are found, read, and are valid.
  1934. STATUS_DISK_CORRUPT_ERROR if corrupt/bad descriptors are found (may be raised)
  1935. --*/
  1936. {
  1937. PNSR_FSD FSD = NULL;
  1938. ULONGLONG Offset;
  1939. ULONG Lbn, Len;
  1940. NTSTATUS Status = STATUS_SUCCESS;
  1941. PAGED_CODE();
  1942. //
  1943. // Check inputs
  1944. //
  1945. ASSERT_IRP_CONTEXT( IrpContext );
  1946. ASSERT( *FileSetDescriptor == NULL );
  1947. DebugTrace(( +1, Dbg,
  1948. "UdfFindFileSetDescriptor, Vcb %08x, LongAd %08x %x/%08x +%08x (type %x)\n",
  1949. Vcb,
  1950. LongAd,
  1951. LongAd->Start.Partition,
  1952. LongAd->Start.Lbn,
  1953. LongAd->Length.Length,
  1954. LongAd->Length.Type ));
  1955. //
  1956. // If the extent we begin from is not a whole number of recorded logical blocks,
  1957. // we can't continue.
  1958. //
  1959. if (LongAd->Length.Length == 0 ||
  1960. LongAd->Length.Type != NSRLENGTH_TYPE_RECORDED ||
  1961. BlockOffset( Vcb, LongAd->Length.Length )) {
  1962. DebugTrace(( +0, Dbg,
  1963. "UdfFindFileSetDescriptor, bad longad length\n" ));
  1964. DebugTrace(( -1, Dbg,
  1965. "UdfFindFileSetDescriptor -> STATUS_DISK_CORRUPT_ERROR\n" ));
  1966. return STATUS_DISK_CORRUPT_ERROR;
  1967. }
  1968. //
  1969. // Use a try-finally for cleanup
  1970. //
  1971. try {
  1972. try {
  1973. for ( //
  1974. // Home ourselves in the search and make a pass through the sequence.
  1975. //
  1976. Len = LongAd->Length.Length,
  1977. Lbn = LongAd->Start.Lbn;
  1978. Len;
  1979. //
  1980. // Advance to the next descriptor offset in the sequence.
  1981. //
  1982. Len -= BlockSize( Vcb ),
  1983. Lbn++) {
  1984. //
  1985. // Allocate a buffer to read fileset descriptors.
  1986. //
  1987. if (FSD == NULL) {
  1988. FSD = FsRtlAllocatePoolWithTag( UdfNonPagedPool,
  1989. UdfRawBufferSize( Vcb, sizeof(NSR_FSD) ),
  1990. TAG_NSR_FSD );
  1991. }
  1992. //
  1993. // Lookup the physical offset for this block. We could be mapping
  1994. // through a VAT here so we can't just assume that all the
  1995. // blocks in the extent are physically contiguous. The FSD seems to
  1996. // be the exception here - there's nothing that says it must be in
  1997. // physical partition, and it can have a terminator, => 2 blocks
  1998. // minumum. There is no single block virtual extent limitation in UDF 1.50.
  1999. //
  2000. Offset = LlBytesFromSectors( Vcb, UdfLookupPsnOfExtent( IrpContext,
  2001. Vcb,
  2002. LongAd->Start.Partition,
  2003. Lbn,
  2004. BlockSize( Vcb)));
  2005. Status = UdfReadSectors( IrpContext,
  2006. Offset,
  2007. UdfRawReadSize( Vcb, sizeof(NSR_FSD) ),
  2008. TRUE,
  2009. FSD,
  2010. Vcb->TargetDeviceObject );
  2011. if (!NT_SUCCESS( Status ) ||
  2012. FSD->Destag.Ident == DESTAG_ID_NOTSPEC) {
  2013. //
  2014. // These are both an excellent sign that this is an unrecorded sector, which
  2015. // is defined to terminate the sequence. (3/8.4.2)
  2016. //
  2017. break;
  2018. }
  2019. if ((FSD->Destag.Ident != DESTAG_ID_NSR_FSD &&
  2020. FSD->Destag.Ident != DESTAG_ID_NSR_TERM) ||
  2021. !UdfVerifyDescriptor( IrpContext,
  2022. &FSD->Destag,
  2023. FSD->Destag.Ident,
  2024. sizeof(NSR_FSD),
  2025. Lbn,
  2026. TRUE)) {
  2027. //
  2028. // If we spot an illegal descriptor type in the stream, there is no reasonable
  2029. // way to guess that we can continue (the disc may be trash beyond this point).
  2030. // Clearly, we also cannot trust the next extent pointed to by a corrupt
  2031. // descriptor.
  2032. //
  2033. try_leave( Status = STATUS_DISK_CORRUPT_ERROR );
  2034. }
  2035. if (FSD->Destag.Ident == DESTAG_ID_NSR_TERM) {
  2036. //
  2037. // This is a way to terminate the sequence.
  2038. //
  2039. break;
  2040. }
  2041. //
  2042. // Reset the pointers to the possible next extent
  2043. //
  2044. LongAd = &FSD->NextExtent;
  2045. if (LongAd->Length.Length) {
  2046. //
  2047. // A fileset descriptor containing a nonzero next extent pointer also
  2048. // terminates this extent of the FSD sequence. (4/8.3.1)
  2049. //
  2050. // If the extent referred to is not fully recorded, this will
  2051. // terminate the sequence.
  2052. //
  2053. if (LongAd->Length.Type != NSRLENGTH_TYPE_RECORDED) {
  2054. break;
  2055. }
  2056. Len = LongAd->Length.Length;
  2057. //
  2058. // The extent must be a multiple of a block size.
  2059. //
  2060. if (BlockOffset( Vcb, Len )) {
  2061. DebugTrace(( +0, Dbg,
  2062. "UdfFindFileSetDescriptor, interior extent not blocksize in length\n" ));
  2063. try_leave ( Status = STATUS_DISK_CORRUPT_ERROR );
  2064. }
  2065. Lbn = LongAd->Start.Lbn;
  2066. //
  2067. // Note that we must correct the values to take into account
  2068. // the changes that will be made next time through the for loop.
  2069. //
  2070. Len += BlockSize( Vcb );
  2071. Lbn -= 1;
  2072. }
  2073. UdfStoreFileSetDescriptorIfPrevailing( FileSetDescriptor, &FSD );
  2074. }
  2075. }
  2076. finally {
  2077. DebugUnwind( "UdfFindFileSetDescriptor");
  2078. //
  2079. // Free up the buffer space we may have allocated
  2080. //
  2081. UdfFreePool( &FSD );
  2082. }
  2083. } except( UdfExceptionFilter( IrpContext, GetExceptionInformation() )) {
  2084. //
  2085. // Transmute raised apparent file corruption to disk corruption - we are not
  2086. // yet touching the visible filesystem.
  2087. //
  2088. Status = IrpContext->ExceptionStatus;
  2089. DebugTrace(( +0, Dbg,
  2090. "UdfFindFileSetDescriptor, exception %08x thrown\n", Status ));
  2091. if (Status == STATUS_FILE_CORRUPT_ERROR) {
  2092. DebugTrace(( +0, Dbg,
  2093. "UdfFindFileSetDescriptor, translating file corrupt to disk corrupt\n" ));
  2094. Status = STATUS_DISK_CORRUPT_ERROR;
  2095. }
  2096. }
  2097. //
  2098. // Success is when we've really found something. If we failed to find the
  2099. // descriptor, commute whatever intermediate status was involved and clean up.
  2100. //
  2101. if (*FileSetDescriptor == NULL) {
  2102. Status = STATUS_UNRECOGNIZED_VOLUME;
  2103. }
  2104. if (!NT_SUCCESS( Status )) {
  2105. UdfFreePool( FileSetDescriptor );
  2106. }
  2107. DebugTrace(( -1, Dbg,
  2108. "UdfFindFileSetDescriptor -> %08x\n", Status ));
  2109. return Status;
  2110. }
  2111. //
  2112. // Local support routine
  2113. //
  2114. NTSTATUS
  2115. UdfFindVolumeDescriptors (
  2116. IN PIRP_CONTEXT IrpContext,
  2117. IN PVCB Vcb,
  2118. IN PEXTENTAD Extent,
  2119. IN OUT PPCB *Pcb,
  2120. IN OUT PNSR_PVD *PrimaryVolumeDescriptor,
  2121. IN OUT PNSR_LVOL *LogicalVolumeDescriptor
  2122. )
  2123. /*++
  2124. Routine Description:
  2125. This routine walks the indicated Volume Descriptor Sequence searching for the
  2126. active descriptors for this volume and generates an initializing Pcb from the
  2127. referenced partitions. No updating of the Vcb occurs.
  2128. Arguments:
  2129. Vcb - Vcb of volume to search
  2130. Extent - Extent to search
  2131. Pcb - Address of a caller's pointer to a Pcb
  2132. PrimaryVolumeDescriptor - Address of caller's pointer to a PVD
  2133. LogicalVolumeDescriptor - Address of caller's pointer to an LVD
  2134. Return Value:
  2135. STATUS_SUCCESS if all descriptors are found, read, and are valid.
  2136. STATUS_DISK_CORRUPT_ERROR if corrupt descriptors are found.
  2137. STATUS_UNRECOGNIZED_VOLUME if noncompliant descriptors are found.
  2138. Descriptors are only returned on success.
  2139. --*/
  2140. {
  2141. PNSR_VD_GENERIC GenericVD = NULL;
  2142. ULONGLONG Offset;
  2143. ULONG Len;
  2144. ULONG MaxSize;
  2145. ULONG UnitSize = UdfRawReadSize( Vcb, sizeof(NSR_VD_GENERIC) );
  2146. NTSTATUS Status = STATUS_SUCCESS;
  2147. ULONG ThisPass = 1;
  2148. ULONG MaxVdpExtents;
  2149. PAGED_CODE();
  2150. //
  2151. // Check the input parameters
  2152. //
  2153. ASSERT_IRP_CONTEXT( IrpContext);
  2154. ASSERT_VCB( Vcb );
  2155. ASSERT_OPTIONAL_PCB( *Pcb );
  2156. DebugTrace(( +1, Dbg,
  2157. "UdfFindVolumeDescriptors, Vcb %08x, Extent %08x +%08x\n",
  2158. Vcb,
  2159. Extent->Lsn,
  2160. Extent->Len ));
  2161. //
  2162. // If the extent we begin from is not at least the size of an aligned descriptor
  2163. // or is sized in base units other than aligned descriptors, we can't continue.
  2164. //
  2165. if (Extent->Len < UnitSize ||
  2166. Extent->Len % UnitSize) {
  2167. DebugTrace(( 0, Dbg,
  2168. "UdfFindVolumeDescriptors, Base extent length %08x is mismatched with read size %08x\n",
  2169. Extent->Len,
  2170. UnitSize ));
  2171. DebugTrace(( -1, Dbg,
  2172. "UdfFindVolumeDescriptors -> STATUS_DISK_CORRUPT_ERROR\n" ));
  2173. return STATUS_DISK_CORRUPT_ERROR;
  2174. }
  2175. //
  2176. // Use a try-finally to facilitate cleanup.
  2177. //
  2178. try {
  2179. DebugTrace(( 0, Dbg,
  2180. "UdfFindVolumeDescriptors, starting pass 1, find LVD/PVD\n" ));
  2181. //
  2182. // We will make at least one pass through the Volume Descriptor Sequence to find
  2183. // the prevailing versions of the two controlling descriptors - the PVD and LVD.
  2184. // In order to avoid picking up partition descriptors that aren't actually going
  2185. // to be referenced by the LVD, we will pick them up in a second pass if we find
  2186. // a PVD and LVD that look reasonable and then stick them in a Pcb.
  2187. //
  2188. for (ThisPass = 1; ThisPass <= 2; ThisPass++) {
  2189. MaxVdpExtents = 16;
  2190. for ( //
  2191. // Home ourselves in the search and make a pass through the sequence.
  2192. //
  2193. Offset = LlBytesFromSectors( Vcb, Extent->Lsn ),
  2194. Len = Extent->Len;
  2195. //
  2196. // If we have reached the end of the extent's indicated valid
  2197. // length, we are done. This usually will not happen.
  2198. //
  2199. Len;
  2200. //
  2201. // Advance to the next descriptor offset in the sequence.
  2202. //
  2203. Offset += UnitSize,
  2204. Len -= UnitSize
  2205. ) {
  2206. //
  2207. // Allocate a buffer to read generic volume descriptors.
  2208. //
  2209. if (GenericVD == NULL) {
  2210. GenericVD = (PNSR_VD_GENERIC) FsRtlAllocatePoolWithTag( UdfNonPagedPool,
  2211. UdfRawBufferSize( Vcb, sizeof(NSR_VD_GENERIC) ),
  2212. TAG_NSR_VDSD );
  2213. }
  2214. Status = UdfReadSectors( IrpContext,
  2215. Offset,
  2216. UnitSize,
  2217. TRUE,
  2218. GenericVD,
  2219. Vcb->TargetDeviceObject );
  2220. //
  2221. // Thise is a decent sign that this is an unrecorded sector and is
  2222. // defined to terminate the sequence.
  2223. //
  2224. if (!NT_SUCCESS( Status )) {
  2225. break;
  2226. }
  2227. //
  2228. // Calculate the maximum size we expect this descriptor to be. For LVDs
  2229. // the descriptor can be followed by upto 2 partition maps, pushing it
  2230. // over the 512 byte ECMA desc. limit which we were assuming was the max.
  2231. //
  2232. MaxSize = sizeof( NSR_VD_GENERIC);
  2233. if (DESTAG_ID_NSR_LVOL == GenericVD->Destag.Ident) {
  2234. MaxSize += 2 * sizeof( PARTMAP_UDF_GENERIC);
  2235. ASSERT( BlockSize( Vcb) >= 1024);
  2236. }
  2237. if (GenericVD->Destag.Ident > DESTAG_ID_MAXIMUM_PART3 ||
  2238. !UdfVerifyDescriptor( IrpContext,
  2239. &GenericVD->Destag,
  2240. GenericVD->Destag.Ident,
  2241. MaxSize,
  2242. (ULONG) SectorsFromBytes( Vcb, Offset ),
  2243. TRUE)) {
  2244. //
  2245. // If we spot an illegal descriptor type in the stream, there is no reasonable
  2246. // way to guess that we can continue (the disc may be trash beyond this point).
  2247. // Likewise, even if we have a single corrupt descriptor we cannot continue because
  2248. // this may be corruption of a descriptor we may have otherwise required for operation
  2249. // (i.e., one of the prevailing descriptors).
  2250. //
  2251. DebugTrace(( 0, Dbg,
  2252. "UdfFindVolumeDescriptors, descriptor didn't verify\n" ));
  2253. try_leave( Status = STATUS_DISK_CORRUPT_ERROR );
  2254. }
  2255. if (GenericVD->Destag.Ident == DESTAG_ID_NSR_TERM) {
  2256. //
  2257. // The Terminating Descriptor (3/10.9) is the usual way to stop a search.
  2258. //
  2259. break;
  2260. }
  2261. if (GenericVD->Destag.Ident == DESTAG_ID_NSR_VDP) {
  2262. //
  2263. // Follow a Volume Desciptor Pointer (3/10.3) to the next extent of the sequence.
  2264. // We will only follow a maximum of 16 extents, to guard against loops.
  2265. //
  2266. if (0 == --MaxVdpExtents) {
  2267. try_leave( Status = STATUS_DISK_CORRUPT_ERROR );
  2268. }
  2269. //
  2270. // Bias the values by UnitSize, so that the next loop iteration will change them
  2271. // to the correct values.
  2272. //
  2273. Offset = LlBytesFromSectors( Vcb, ((PNSR_VDP) GenericVD)->Next.Lsn ) - UnitSize;
  2274. Len = ((PNSR_VDP) GenericVD)->Next.Len;
  2275. //
  2276. // We cannot do anything if the extent is invalid
  2277. //
  2278. if (Len < UnitSize ||
  2279. Len % UnitSize) {
  2280. DebugTrace(( 0, Dbg,
  2281. "UdfFindVolumeDescriptors, following extent length %08x is mismatched with read size %08x\n",
  2282. Extent->Len,
  2283. UnitSize ));
  2284. try_leave( Status = STATUS_DISK_CORRUPT_ERROR );
  2285. }
  2286. Len += UnitSize;
  2287. continue;
  2288. }
  2289. DebugTrace(( 0, Dbg,
  2290. "UdfFindVolumeDescriptors, descriptor tag %08x\n",
  2291. GenericVD->Destag.Ident ));
  2292. if (ThisPass == 1) {
  2293. //
  2294. // Our first pass is to find prevailing LVD and PVD.
  2295. //
  2296. switch (GenericVD->Destag.Ident) {
  2297. case DESTAG_ID_NSR_PVD:
  2298. UdfStoreVolumeDescriptorIfPrevailing( (PNSR_VD_GENERIC *) PrimaryVolumeDescriptor,
  2299. GenericVD );
  2300. break;
  2301. case DESTAG_ID_NSR_LVOL:
  2302. UdfStoreVolumeDescriptorIfPrevailing( (PNSR_VD_GENERIC *) LogicalVolumeDescriptor,
  2303. GenericVD );
  2304. break;
  2305. default:
  2306. break;
  2307. }
  2308. } else {
  2309. PNSR_PART PartitionDescriptor = (PNSR_PART) GenericVD;
  2310. USHORT ExpectedNsrVer;
  2311. //
  2312. // Our second pass is to pick up all relevant NSR02/3 PD
  2313. //
  2314. if (PartitionDescriptor->Destag.Ident != DESTAG_ID_NSR_PART) {
  2315. continue;
  2316. }
  2317. //
  2318. // Look at the NSR standard revision
  2319. //
  2320. if (UdfEqualEntityId( &PartitionDescriptor->ContentsID, &UdfNSR02Identifier, NULL )) {
  2321. ExpectedNsrVer = VsdIdentNSR02;
  2322. }
  2323. else if (UdfEqualEntityId( &PartitionDescriptor->ContentsID, &UdfNSR03Identifier, NULL )) {
  2324. ExpectedNsrVer = VsdIdentNSR03;
  2325. }
  2326. else {
  2327. //
  2328. // Unknown NSR revision
  2329. //
  2330. ExpectedNsrVer = VsdIdentBad;
  2331. }
  2332. //
  2333. // Check that the NSR version in this PD matches what we found in the VRS earlier.
  2334. //
  2335. if (ExpectedNsrVer != Vcb->NsrVersion) {
  2336. DebugTrace(( 0, Dbg, "UdfFindVolumeDescriptors: NSR version in PartitionDescriptor (%d) != NSR found in VRS (%d)\n", ExpectedNsrVer, Vcb->NsrVersion));
  2337. try_leave( Status = STATUS_UNRECOGNIZED_VOLUME );
  2338. }
  2339. UdfAddToPcb( *Pcb, (PNSR_PART) GenericVD );
  2340. }
  2341. } // inner descriptor loop.
  2342. //
  2343. // Now that a pass through the VDS has been completed, analyze the results.
  2344. //
  2345. if (ThisPass == 1) {
  2346. PNSR_PVD PVD;
  2347. PNSR_LVOL LVD;
  2348. USHORT MaxVerBasedOnNSR;
  2349. //
  2350. // Reference the descriptors for ease of use
  2351. //
  2352. PVD = *PrimaryVolumeDescriptor;
  2353. LVD = *LogicalVolumeDescriptor;
  2354. //
  2355. // Check that the descriptors indicate a logical volume which appears to
  2356. // be a valid UDF volume.
  2357. //
  2358. if ((PVD == NULL &&
  2359. DebugTrace(( 0, Dbg,
  2360. "UdfFindVolumeDescriptors, don't have a PVD\n" ))) ||
  2361. (LVD == NULL &&
  2362. DebugTrace(( 0, Dbg,
  2363. "UdfFindVolumeDescriptors, don't have an LVD\n" )))) {
  2364. try_leave( Status = STATUS_UNRECOGNIZED_VOLUME );
  2365. }
  2366. //
  2367. // Store away the UDF revision in the VCB for future reference, and clamp the
  2368. // maximum acceptable revision based on the previously encountered NSR version.
  2369. //
  2370. Vcb->UdfRevision = ((PUDF_SUFFIX_DOMAIN)&(LVD->DomainID.Suffix))->UdfRevision;
  2371. MaxVerBasedOnNSR = (VsdIdentNSR03 > Vcb->NsrVersion) ? UDF_VERSION_150 : UDF_VERSION_RECOGNIZED;
  2372. DebugTrace((0,Dbg,"UdfFindVolumeDescriptors() Pass 1: Found LVD specifying DomainID %x\n", ((PUDF_SUFFIX_DOMAIN)&(LVD->DomainID.Suffix))->UdfRevision));
  2373. if (
  2374. //
  2375. // Now check the PVD
  2376. //
  2377. //
  2378. // The Volume Set Sequence fields indicates how many volumes form
  2379. // the volume set and what number this volume is in that sequence.
  2380. // We are a level 2 implementation, meaning that the volumes we read
  2381. // consist of a single volume. (3/11)
  2382. //
  2383. (PVD->VolSetSeq > 1 &&
  2384. DebugTrace(( 0, Dbg,
  2385. "UdfFindVolumeDescriptors, PVD VolSetSeq %08x - not volume 1 of a volume set\n",
  2386. PVD->VolSetSeq ))) ||
  2387. (((PVD->VolSetSeqMax > 1) && (PVD->Destag.CRC != UDF_SNOW_WHITE_PVD_CRC) &&
  2388. (PVD->Destag.CRC != UDF_SNOW_WHITE_PVD_CRC_VARIANT_2)) &&
  2389. DebugTrace(( 0, Dbg,
  2390. "UdfFindVolumeDescriptors, PVD VolSetSeqMax %08x - volume in a non-unit volume set\n",
  2391. PVD->VolSetSeqMax ))) ||
  2392. (PVD->CharSetList != UDF_CHARSETLIST &&
  2393. DebugTrace(( 0, Dbg,
  2394. "UdfFindVolumeDescriptors, PVD CharSetList %08x != CS0 only\n",
  2395. PVD->CharSetList ))) ||
  2396. (PVD->CharSetListMax != UDF_CHARSETLIST &&
  2397. DebugTrace(( 0, Dbg,
  2398. "UdfFindVolumeDescriptors, PVD CharSetListMax %08x != CS0 only\n",
  2399. PVD->CharSetListMax ))) ||
  2400. //
  2401. // The two character sets must be UDF CS0. CS0 is a "by convention"
  2402. // character set in ISO 13346, which UDF specifies for our domain.
  2403. //
  2404. (!UdfEqualCharspec( &PVD->CharsetDesc, &UdfCS0Identifier, CHARSPEC_T_CS0 ) &&
  2405. DebugTrace(( 0, Dbg,
  2406. "UdfFindVolumeDescriptors, PVD CharsetDesc != CS0 only\n" ))) ||
  2407. (!UdfEqualCharspec( &PVD->CharsetExplan, &UdfCS0Identifier, CHARSPEC_T_CS0 ) &&
  2408. DebugTrace(( 0, Dbg,
  2409. "UdfFindVolumeDescriptors, PVD CharsetExplan != CS0 only\n" ))) ||
  2410. //
  2411. // Now check the LVD
  2412. //
  2413. //
  2414. // The LVD is a variant sized structure. Check that the claimed size fits in a single
  2415. // logical sector. Although an LVD may legally exceed a single sector, we will never
  2416. // want to deal with such a volume.
  2417. //
  2418. (ISONsrLvolSize( LVD ) > SectorSize( Vcb ) &&
  2419. DebugTrace(( 0, Dbg,
  2420. "UdfFindVolumeDescriptors, LVD is bigger than a sector\n" ))) ||
  2421. //
  2422. // The character set used in the LVD must be UDF CS0 as well.
  2423. //
  2424. (!UdfEqualCharspec( &LVD->Charset, &UdfCS0Identifier, CHARSPEC_T_CS0 ) &&
  2425. DebugTrace(( 0, Dbg,
  2426. "UdfFindVolumeDescriptors, LVD Charset != CS0 only\n" ))) ||
  2427. //
  2428. // The specified block size must equal the physical sector size.
  2429. //
  2430. (LVD->BlockSize != SectorSize( Vcb ) &&
  2431. DebugTrace(( 0, Dbg,
  2432. "UdfFindVolumeDescriptors, LVD BlockSize %08x != SectorSize %08x\n" ))) ||
  2433. //
  2434. // The domain must be within the version we read
  2435. //
  2436. (!UdfDomainIdentifierContained( &LVD->DomainID,
  2437. &UdfDomainIdentifier,
  2438. UDF_VERSION_MINIMUM,
  2439. MaxVerBasedOnNSR ) &&
  2440. DebugTrace(( 0, Dbg,
  2441. "UdfFindVolumeDescriptors, domain ID indicates unreadable volume\n" ))) ||
  2442. //
  2443. // Although we can handle any number of partitions, UDF only specifies
  2444. // a single partition or special dual partition formats.
  2445. //
  2446. (LVD->MapTableCount > 2 &&
  2447. DebugTrace(( 0, Dbg,
  2448. "UdfFindVolumeDescriptors, LVD MapTableCount %08x greater than allowed (2)\n",
  2449. LVD->MapTableCount )))
  2450. ) {
  2451. DebugTrace(( 0, Dbg,
  2452. "UdfFindVolumeDescriptors, ... so returning STATUS_UNRECOGNIZED_VOLUME\n" ));
  2453. try_leave( Status = STATUS_UNRECOGNIZED_VOLUME );
  2454. }
  2455. //
  2456. // Now that we have performed the simple field checks, build a Pcb.
  2457. //
  2458. Status = UdfInitializePcb( IrpContext, Vcb, Pcb, LVD );
  2459. if (!NT_SUCCESS(Status)) {
  2460. DebugTrace(( 0, Dbg,
  2461. "UdfFindVolumeDescriptors, Pcb intialization failed (!)\n" ));
  2462. try_leave( Status );
  2463. }
  2464. }
  2465. //
  2466. // Go onto Pass 2 to find the Partition Descriptors
  2467. //
  2468. DebugTrace(( 0, Dbg,
  2469. "UdfFindVolumeDescriptors, starting pass 2, find associated PD\n" ));
  2470. }
  2471. } finally {
  2472. DebugUnwind( "UdfFindVolumeDescriptors" );
  2473. //
  2474. // Free up the buffer space we may have allocated
  2475. //
  2476. UdfFreePool( &GenericVD );
  2477. }
  2478. DebugTrace(( -1, Dbg,
  2479. "UdfFindVolumeDescriptors -> %08x\n", Status ));
  2480. //
  2481. // Success is when we've really found something. If we failed to find both
  2482. // descriptors, commute whatever intermediate status was involved and clean up.
  2483. //
  2484. if (*PrimaryVolumeDescriptor == NULL || *LogicalVolumeDescriptor == NULL) {
  2485. Status = STATUS_UNRECOGNIZED_VOLUME;
  2486. }
  2487. if (!NT_SUCCESS( Status )) {
  2488. UdfFreePool(PrimaryVolumeDescriptor);
  2489. UdfFreePool(LogicalVolumeDescriptor);
  2490. }
  2491. return Status;
  2492. }
  2493. //
  2494. // Local support routine
  2495. //
  2496. NTSTATUS
  2497. UdfFindAnchorVolumeDescriptor (
  2498. IN PIRP_CONTEXT IrpContext,
  2499. IN PVCB Vcb,
  2500. IN OUT PNSR_ANCHOR *AnchorVolumeDescriptor
  2501. )
  2502. /*++
  2503. Routine Description:
  2504. This routine will find the Anchor Volume Descriptor for a piece of media
  2505. Arguments:
  2506. Vcb - Vcb of volume to search
  2507. AnchorVolumeDescriptor - Caller's pointer to an AVD
  2508. Return Value:
  2509. Boolean TRUE if AVD is discovered, FALSE otherwise.
  2510. --*/
  2511. {
  2512. ULONG ThisPass;
  2513. ULONG ReadLsn;
  2514. ULONG Lsn;
  2515. BOOLEAN Found = FALSE;
  2516. NTSTATUS Status;
  2517. PAGED_CODE();
  2518. //
  2519. // Check the input parameters
  2520. //
  2521. ASSERT_IRP_CONTEXT( IrpContext);
  2522. ASSERT_VCB( Vcb );
  2523. ASSERT(*AnchorVolumeDescriptor == NULL);
  2524. DebugTrace(( +1, Dbg, "UdfFindAnchorVolumeDescriptors()\n"));
  2525. //
  2526. // Discover the Anchor Volume Descriptor, which will point towards the
  2527. // Volume Set Descriptor Sequence. The AVD may exist at sector 256 or
  2528. // in the last sector of the volume.
  2529. //
  2530. *AnchorVolumeDescriptor = (PNSR_ANCHOR) FsRtlAllocatePoolWithTag( UdfNonPagedPool,
  2531. UdfRawBufferSize( Vcb, sizeof(NSR_ANCHOR) ),
  2532. TAG_NSR_VDSD );
  2533. //
  2534. // Search the three possible locations for an AVD to exist on the volume,
  2535. // plus check for the possibility of a method 2 fixup requirement.
  2536. //
  2537. for ( ThisPass = 1; ThisPass < 11; ThisPass++ ) {
  2538. if (ThisPass == 1) {
  2539. ReadLsn = Lsn = ANCHOR_SECTOR + Vcb->BoundS;
  2540. } else if (ThisPass == 2) {
  2541. //
  2542. // It is so unlikely that we will get a disk that doesn't have
  2543. // an anchor at 256 that this is a pretty good indication we
  2544. // have a CD-RW here and the drive is method 2 goofy. Take
  2545. // a shot.
  2546. //
  2547. ReadLsn = UdfMethod2TransformSector( Vcb, ANCHOR_SECTOR );
  2548. Lsn = ANCHOR_SECTOR;
  2549. } else if (ThisPass >= 3) {
  2550. ULONG SubPass = (ThisPass > 6) ? (ThisPass - 4) : ThisPass;
  2551. //
  2552. // Our remaining two chances depend on being able to determine
  2553. // the last recorded sector for the volume. If we were unable
  2554. // to do this, stop.
  2555. //
  2556. if (!Vcb->BoundN) {
  2557. break;
  2558. }
  2559. //
  2560. // Note that although we're only looking at 2 sectors (N, N-256),
  2561. // because of the fuzziness of N on CD media (can include runout
  2562. // of 2 sectors) and method 2 addressing bugs in some drives, we
  2563. // potentially have to look at 8 locations... We work fowards to
  2564. // try and avoid reading invalid sectors (which can take some time).
  2565. //
  2566. ReadLsn = Lsn = Vcb->BoundN - ( SubPass == 3? (ANCHOR_SECTOR + 2): // 3,7
  2567. ( SubPass == 4? ANCHOR_SECTOR: // 4,8
  2568. ( SubPass == 5? 2 : 0 ))); // 5,9 6,10
  2569. //
  2570. // Also try the method 2 transformed version of each address (pass 7..10)
  2571. // If we get this far, it might take a while...
  2572. //
  2573. if (6 < ThisPass) {
  2574. ReadLsn = UdfMethod2TransformSector( Vcb, Lsn);
  2575. }
  2576. }
  2577. DebugTrace(( 0, Dbg, "Pass: %d Trying Lsn/ReadLsn %X / %X\n", ThisPass, Lsn, ReadLsn));
  2578. //
  2579. // We may have more chances to succeed if failure occurs.
  2580. //
  2581. Status = UdfReadSectors( IrpContext,
  2582. LlBytesFromSectors( Vcb, ReadLsn ),
  2583. UdfRawReadSize( Vcb, sizeof(NSR_ANCHOR) ),
  2584. TRUE,
  2585. *AnchorVolumeDescriptor,
  2586. Vcb->TargetDeviceObject );
  2587. if ( NT_SUCCESS( Status ) &&
  2588. UdfVerifyDescriptor( IrpContext,
  2589. &(*AnchorVolumeDescriptor)->Destag,
  2590. DESTAG_ID_NSR_ANCHOR,
  2591. sizeof(NSR_ANCHOR),
  2592. Lsn,
  2593. TRUE)
  2594. ) {
  2595. //
  2596. // Got one! Set the method 2 fixup appropriately.
  2597. //
  2598. if (ReadLsn != Lsn) {
  2599. DebugTrace(( 0, Dbg, "************************************************\n"));
  2600. DebugTrace(( 0, Dbg, "METHOD 2 FIXUPS ACTIVATED FOR Vcb @ %08x\n", Vcb ));
  2601. DebugTrace(( 0, Dbg, "************************************************\n"));
  2602. SetFlag( Vcb->VcbState, VCB_STATE_METHOD_2_FIXUP );
  2603. } else {
  2604. ClearFlag( Vcb->VcbState, VCB_STATE_METHOD_2_FIXUP );
  2605. }
  2606. Status = STATUS_SUCCESS;
  2607. break;
  2608. }
  2609. }
  2610. if (11 == ThisPass) {
  2611. Status = STATUS_UNRECOGNIZED_VOLUME;
  2612. }
  2613. DebugTrace(( -1, Dbg, "UdfFindAnchorVolumeDescriptors() -> %X\n", Status));
  2614. return Status;
  2615. }
  2616. //
  2617. // Local support routine
  2618. //
  2619. BOOLEAN
  2620. UdfRecognizeVolume (
  2621. IN PIRP_CONTEXT IrpContext,
  2622. IN PDEVICE_OBJECT DeviceObject,
  2623. IN ULONG SectorSize,
  2624. IN OUT PULONG BoundS,
  2625. IN OUT PBOOLEAN Bridge,
  2626. OUT PUSHORT NSRVerFound
  2627. )
  2628. /*++
  2629. Routine Description:
  2630. This routine walks the Volume Recognition Sequence to determine
  2631. whether this volume contains an NSR02 (ISO 13346 Section 4) image.
  2632. Arguments:
  2633. DeviceObject - device we are checking
  2634. SectorSize - size of a physical sector on this device
  2635. Bridge - will return whether there appear to be ISO 9660 structures
  2636. on the media
  2637. NSRVerFound - returns either VsdIdentNSR02 or VsdIdentNSR03 if successful
  2638. Return Value:
  2639. Boolean TRUE if we found NSR02/3, FALSE otherwise.
  2640. --*/
  2641. {
  2642. NTSTATUS Status;
  2643. BOOLEAN FoundBEA;
  2644. BOOLEAN FoundNSR;
  2645. BOOLEAN Resolved;
  2646. USHORT ThisRecordType;
  2647. PVSD_GENERIC VolumeStructureDescriptor;
  2648. PVSD_GENERIC VolumeStructureDescriptorBuffer;
  2649. ULONGLONG Offset;
  2650. PAGED_CODE();
  2651. //
  2652. // Check the input parameters
  2653. //
  2654. ASSERT_IRP_CONTEXT( IrpContext);
  2655. VolumeStructureDescriptorBuffer = (PVSD_GENERIC) FsRtlAllocatePoolWithTag( UdfNonPagedPool,
  2656. UdfRawBufferSizeN( SectorSize,
  2657. sizeof(VSD_GENERIC) ),
  2658. TAG_NSR_VSD );
  2659. DebugTrace(( +1, Dbg,
  2660. "UdfRecognizeVolume, DevObj %08x SectorSize %08x\n",
  2661. DeviceObject,
  2662. SectorSize ));
  2663. //
  2664. // Use try-finally to facilitate cleanup
  2665. //
  2666. try {
  2667. Retry:
  2668. FoundBEA =
  2669. FoundNSR =
  2670. Resolved = FALSE;
  2671. Offset = (SectorSize * (*BoundS)) + SectorAlignN( SectorSize, VRA_BOUNDARY_LOCATION );
  2672. while (!Resolved) {
  2673. //
  2674. // It's possible that the sector size is > 2k which is the descriptor
  2675. // size. Only read if we've processed all 2k blocks in the prev. sector
  2676. //
  2677. if (0 == (Offset & (SectorSize - 1))) {
  2678. VolumeStructureDescriptor = VolumeStructureDescriptorBuffer;
  2679. Status = UdfReadSectors( IrpContext,
  2680. Offset,
  2681. UdfRawReadSizeN( SectorSize,
  2682. sizeof(VSD_GENERIC) ),
  2683. TRUE,
  2684. VolumeStructureDescriptor,
  2685. DeviceObject );
  2686. if (!NT_SUCCESS( Status )) {
  2687. break;
  2688. }
  2689. }
  2690. //
  2691. // Now check the type of the descriptor. All ISO 13346 VSDs are
  2692. // of Type 0, 9660 PVDs are Type 1, 9660 SVDs are Type 2, and 9660
  2693. // terminating descriptors are Type 255.
  2694. //
  2695. if (VolumeStructureDescriptor->Type == 0) {
  2696. //
  2697. // In order to properly recognize the volume, we must know all of the
  2698. // Structure identifiers in ISO 13346 so that we can terminate if a
  2699. // badly formatted (or, shockingly, non 13346) volume is presented to us.
  2700. //
  2701. ThisRecordType = (USHORT)UdfFindInParseTable( VsdIdentParseTable,
  2702. VolumeStructureDescriptor->Ident,
  2703. VSD_LENGTH_IDENT );
  2704. switch ( ThisRecordType ) {
  2705. case VsdIdentBEA01:
  2706. //
  2707. // Only one BEA may exist and its version must be 1 (2/9.2.3)
  2708. //
  2709. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got a BEA01\n" ));
  2710. if ((FoundBEA &&
  2711. DebugTrace(( 0, Dbg,
  2712. "UdfRecognizeVolume, ... but it is a duplicate!\n" ))) ||
  2713. (VolumeStructureDescriptor->Version != 1 &&
  2714. DebugTrace(( 0, Dbg,
  2715. "UdfRecognizeVolume, ... but it has a wacky version number %02x != 1!\n",
  2716. VolumeStructureDescriptor->Version )))) {
  2717. Resolved = TRUE;
  2718. break;
  2719. }
  2720. FoundBEA = TRUE;
  2721. break;
  2722. case VsdIdentTEA01:
  2723. //
  2724. // If we reach the TEA it must be the case that we don't recognize
  2725. //
  2726. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got a TEA01\n" ));
  2727. Resolved = TRUE;
  2728. break;
  2729. case VsdIdentNSR02:
  2730. case VsdIdentNSR03:
  2731. //
  2732. // We recognize NSR02/3 version 1 embedded after a BEA (3/9.1.3). For
  2733. // simplicity we will not bother being a complete nitpick and check
  2734. // for a bounding TEA, although we will be optimistic in the case where
  2735. // we fail to match the version.
  2736. //
  2737. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got an NSR0%c\n", ((VsdIdentNSR02 == ThisRecordType) ? '2' : '3')));
  2738. if ((FoundBEA ||
  2739. !DebugTrace(( 0, Dbg, "UdfRecognizeVolume, ... but we haven't seen a BEA01 yet!\n" ))) &&
  2740. (VolumeStructureDescriptor->Version == 1 ||
  2741. !DebugTrace(( 0, Dbg, "UdfRecognizeVolume, ... but it has a wacky version number %02x != 1\n",
  2742. VolumeStructureDescriptor->Version )))
  2743. ) {
  2744. FoundNSR = Resolved = TRUE;
  2745. *NSRVerFound = ThisRecordType; // Report the NSR version we found here
  2746. break;
  2747. }
  2748. break;
  2749. case VsdIdentCD001:
  2750. case VsdIdentCDW01:
  2751. case VsdIdentNSR01:
  2752. case VsdIdentCDW02:
  2753. case VsdIdentBOOT2:
  2754. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got a valid but uninteresting 13346 descriptor (%d)\n", ThisRecordType ));
  2755. //
  2756. // Valid but uninteresting (to us) descriptors
  2757. //
  2758. break;
  2759. default:
  2760. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got an invalid 13346 descriptor (%d)\n", ThisRecordType ));
  2761. //
  2762. // Stumbling across something we don't know, it must be that this
  2763. // is not a valid 13346 image
  2764. //
  2765. Resolved = TRUE;
  2766. break;
  2767. }
  2768. }
  2769. else if (!FoundBEA && (VolumeStructureDescriptor->Type < 3 ||
  2770. VolumeStructureDescriptor->Type == 255)) {
  2771. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got a 9660 descriptor\n" ));
  2772. //
  2773. // Only HSG (CDROM) and 9660 (CD001) are possible, and they are only legal
  2774. // before the ISO 13346 BEA/TEA extent. By design, an ISO 13346 VSD precisely
  2775. // overlaps a 9660 PVD/SVD in the appropriate fields.
  2776. //
  2777. // Note that we aren't being strict about the structure of the 9660 descriptors
  2778. // since that really isn't very interesting. We care more about the 13346.
  2779. //
  2780. //
  2781. switch (UdfFindInParseTable( VsdIdentParseTable,
  2782. VolumeStructureDescriptor->Ident,
  2783. VSD_LENGTH_IDENT )) {
  2784. case VsdIdentCDROM:
  2785. case VsdIdentCD001:
  2786. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, ... seems we have 9660 here\n" ));
  2787. //
  2788. // Note to our caller that we seem to have ISO 9660 here
  2789. //
  2790. *Bridge = TRUE;
  2791. break;
  2792. default:
  2793. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, ... but it looks wacky\n" ));
  2794. //
  2795. // This probably was a false alert, but in any case there is nothing
  2796. // on this volume for us.
  2797. //
  2798. Resolved = TRUE;
  2799. break;
  2800. }
  2801. } else {
  2802. //
  2803. // Something else must be recorded on this volume.
  2804. //
  2805. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, got an unrecognizeable descriptor, probably not 13346/9660\n" ));
  2806. break;
  2807. }
  2808. //
  2809. // Unfortunately the VRS is specified as 2k descriptors regardless of sector
  2810. // size, 4k sectors iterate through the 2k blocks within a sector.
  2811. //
  2812. Offset += sizeof(VSD_GENERIC);
  2813. VolumeStructureDescriptor = Add2Ptr( VolumeStructureDescriptor,
  2814. sizeof( VSD_GENERIC),
  2815. PVSD_GENERIC);
  2816. }
  2817. //
  2818. // If this was the first pass, and we weren't looking at the start
  2819. // of the disc (i.e. later session), and we didn't find anything,
  2820. // then try the first track in the first session.
  2821. //
  2822. if (!FoundNSR && (0 != *BoundS)) {
  2823. DebugTrace(( 0, Dbg, "UdfRecognizeVolume, failed to find VRS in last session, trying first\n" ));
  2824. *BoundS = 0;
  2825. goto Retry;
  2826. }
  2827. }
  2828. finally {
  2829. DebugUnwind( "UdfRecognizeVolume" );
  2830. //
  2831. // Free up our temporary buffer
  2832. //
  2833. UdfFreePool( &VolumeStructureDescriptorBuffer );
  2834. if (AbnormalTermination()) {
  2835. //
  2836. // Commute a status we raised for empty devices so that other filesystems
  2837. // can have a crack at this.
  2838. //
  2839. if (UdfIsRawDevice(IrpContext, IrpContext->ExceptionStatus)) {
  2840. IrpContext->ExceptionStatus = STATUS_UNRECOGNIZED_VOLUME;
  2841. }
  2842. }
  2843. }
  2844. DebugTrace(( -1, Dbg, "UdfRecognizeVolume -> %u\n", FoundNSR ));
  2845. return FoundNSR;
  2846. }
  2847. //
  2848. // Local support routine
  2849. //
  2850. VOID
  2851. UdfScanForDismountedVcb (
  2852. IN PIRP_CONTEXT IrpContext
  2853. )
  2854. /*++
  2855. Routine Description:
  2856. This routine walks through the list of Vcb's looking for any which may
  2857. now be deleted. They may have been left on the list because there were
  2858. outstanding references.
  2859. Arguments:
  2860. Return Value:
  2861. None
  2862. --*/
  2863. {
  2864. PVCB Vcb;
  2865. PLIST_ENTRY Links;
  2866. PAGED_CODE();
  2867. //
  2868. // Check input.
  2869. //
  2870. ASSERT_IRP_CONTEXT( IrpContext );
  2871. ASSERT_EXCLUSIVE_UDFDATA;
  2872. //
  2873. // Walk through all of the Vcb's attached to the global data.
  2874. //
  2875. Links = UdfData.VcbQueue.Flink;
  2876. while (Links != &UdfData.VcbQueue) {
  2877. Vcb = CONTAINING_RECORD( Links, VCB, VcbLinks );
  2878. //
  2879. // Move to the next link now since the current Vcb may be deleted.
  2880. //
  2881. Links = Links->Flink;
  2882. //
  2883. // If dismount is already underway then check if this Vcb can
  2884. // go away.
  2885. //
  2886. if ((Vcb->VcbCondition == VcbDismountInProgress) ||
  2887. (Vcb->VcbCondition == VcbInvalid) ||
  2888. ((Vcb->VcbCondition == VcbNotMounted) && (Vcb->VcbReference <= Vcb->VcbResidualReference))) {
  2889. UdfCheckForDismount( IrpContext, Vcb, FALSE );
  2890. }
  2891. }
  2892. return;
  2893. }
  2894. VOID
  2895. UdfDetermineVolumeBounding (
  2896. IN PIRP_CONTEXT IrpContext,
  2897. IN PDEVICE_OBJECT TargetDeviceObject,
  2898. IN PULONG S,
  2899. IN PULONG N
  2900. )
  2901. /*++
  2902. Routine Description:
  2903. This routine will figure out where the base offset to discover volume descriptors
  2904. lies and where the end of the disc is. In the case where this is a non-CD media,
  2905. this will tend to not to set the end bound since there is no uniform way to figure
  2906. that piece of information out.
  2907. The bounding information is used to start the hunt for CD-UDF (UDF 1.5) volumes.
  2908. Anyone who puts CD-UDF on non-CD media deserves what they get.
  2909. Arguments:
  2910. Vcb - the volume we are operating on
  2911. S - an address to store the start of the volume for the purposes of finding descriptors
  2912. N - an address to store the end of the volume for the purposes of finding descriptors
  2913. Return Value:
  2914. None.
  2915. Benign inability find the S/N information will result in 0/0 being returned.
  2916. --*/
  2917. {
  2918. NTSTATUS Status;
  2919. PCDROM_TOC CdromToc;
  2920. PTRACK_DATA TrackData;
  2921. CDROM_READ_TOC_EX Command;
  2922. PAGED_CODE();
  2923. //
  2924. // Check input.
  2925. //
  2926. ASSERT_IRP_CONTEXT( IrpContext );
  2927. //
  2928. // Whack the inputs to the benign state.
  2929. //
  2930. *S = *N = 0;
  2931. //
  2932. // Currently we do nothing here for non CD class devices. This does
  2933. // mean that we can't mount (e.g.) WORM/RAM/MO media which has been
  2934. // recorded with sequential UDF and a VAT.
  2935. //
  2936. if (TargetDeviceObject->DeviceType != FILE_DEVICE_CD_ROM) {
  2937. DebugTrace(( 0, Dbg, "Not determining volume bounds / session info - not CDROM class device\n"));
  2938. return;
  2939. }
  2940. //
  2941. // Allocate a buffer for the last session information.
  2942. //
  2943. CdromToc = FsRtlAllocatePoolWithTag( UdfPagedPool,
  2944. sizeof( CDROM_TOC ),
  2945. TAG_CDROM_TOC );
  2946. RtlZeroMemory( CdromToc, sizeof( CDROM_TOC ));
  2947. DebugTrace(( +1, Dbg,
  2948. "UdfDetermineVolumeBounding, S %08x N %08x\n",
  2949. S,
  2950. N ));
  2951. //
  2952. // Zero the command block. This conveniently corresponds to an
  2953. // LBA mode READ_TOC request.
  2954. //
  2955. RtlZeroMemory( &Command, sizeof( Command));
  2956. //
  2957. // Try to retrieve the CDROM last session information.
  2958. //
  2959. try {
  2960. //
  2961. // Pull up the TOC. The information for track AA (start of leadout)
  2962. // will get us the end of disc within some tolerance dependent on how
  2963. // much the device manufacturer paid attention to specifications.
  2964. // (-152, -150, -2, and 0 are possible offsets to the real end).
  2965. //
  2966. Status = UdfPerformDevIoCtrl( IrpContext,
  2967. IOCTL_CDROM_READ_TOC_EX,
  2968. TargetDeviceObject,
  2969. &Command,
  2970. sizeof( Command),
  2971. CdromToc,
  2972. sizeof( CDROM_TOC ),
  2973. FALSE,
  2974. TRUE,
  2975. NULL );
  2976. //
  2977. // If this failed, try again with the MSF variant of the command
  2978. //
  2979. if (!NT_SUCCESS(Status) &&
  2980. (STATUS_INSUFFICIENT_RESOURCES != Status)) {
  2981. Command.Msf = 1;
  2982. Status = UdfPerformDevIoCtrl( IrpContext,
  2983. IOCTL_CDROM_READ_TOC_EX,
  2984. TargetDeviceObject,
  2985. &Command,
  2986. sizeof( Command),
  2987. CdromToc,
  2988. sizeof( CDROM_TOC ),
  2989. FALSE,
  2990. TRUE,
  2991. NULL );
  2992. }
  2993. //
  2994. // Raise an exception if there was an allocation failure.
  2995. //
  2996. if (Status == STATUS_INSUFFICIENT_RESOURCES) {
  2997. DebugTrace(( 0, Dbg, "UdfDetermineVolumeBounding, READ_TOC failed INSUFFICIENT_RESOURCES\n" ));
  2998. UdfRaiseStatus( IrpContext, Status );
  2999. }
  3000. //
  3001. // For other errors, just fail. Perhaps this will turn out to be benign, in any case
  3002. // the mount will rapidly and correctly fail if it really was dependant on this work.
  3003. //
  3004. if (!NT_SUCCESS( Status )) {
  3005. try_leave( NOTHING );
  3006. }
  3007. //
  3008. // Sanity chck that the TOC is well-bounded.
  3009. //
  3010. if (CdromToc->LastTrack - CdromToc->FirstTrack >= MAXIMUM_NUMBER_TRACKS) {
  3011. DebugTrace(( 0, Dbg, "UdfDetermineVolumeBounding, TOC malf (too many tracks)\n" ));
  3012. try_leave( NOTHING );
  3013. }
  3014. #if DBG
  3015. {
  3016. ULONG Track;
  3017. for ( Track = 0; Track <= (ULONG)(CdromToc->LastTrack - CdromToc->FirstTrack + 1); Track++) {
  3018. DebugTrace(( 0, Dbg, " TOC[%02x]: Num: %x Ctrl/Adr: %x/%x Addr: %08x\n", Track, CdromToc->TrackData[Track].TrackNumber, CdromToc->TrackData[Track].Control, CdromToc->TrackData[Track].Adr, *(PULONG)(CdromToc->TrackData[Track].Address)));
  3019. }
  3020. }
  3021. #endif
  3022. TrackData = &CdromToc->TrackData[(CdromToc->LastTrack - CdromToc->FirstTrack + 1)];
  3023. //
  3024. // Last track better have number 0xAA ...
  3025. //
  3026. if (TrackData->TrackNumber != 0xaa) {
  3027. DebugTrace(( 0, Dbg, "UdfDetermineVolumeBounding, TOC malf (aa not last)\n" ));
  3028. //
  3029. // Some drives do this wrong, apparently, so we won't enforce it.
  3030. //
  3031. // try_leave( NOTHING );
  3032. }
  3033. //
  3034. // Now, find the AA (leadout 'track') info
  3035. //
  3036. if (Command.Msf) {
  3037. //
  3038. // Convert MSF to a logical block address. 75 frames/sectors
  3039. // per second, 60 seconds per minute. The MSF address is stored LSB (the F byte) high
  3040. // in the word.
  3041. //
  3042. // NOTE: MSF is only capable of representing 256*(256+256*60)*75 = 0x11ce20 sectors.
  3043. // This is 2.3gb, much less than the size of DVD media, which will respond to CDROM_TOC.
  3044. // Caveat user. And actually the maximum 'legal' value is 63/59/74.
  3045. //
  3046. *N = (TrackData->Address[3] + (TrackData->Address[2] + TrackData->Address[1] * 60) * 75) - 1;
  3047. //
  3048. // We must bias back by 0/2/0 MSF since that is the defined location of sector 0. This
  3049. // works out to 150 sectors.
  3050. //
  3051. if (*N <= 150) {
  3052. *N = 0;
  3053. try_leave( NOTHING );
  3054. }
  3055. *N -= 150;
  3056. }
  3057. else {
  3058. //
  3059. // The non-MSF (LBA) request succeeded, so just fix the endianness.
  3060. //
  3061. SwapCopyUchar4( N, &TrackData->Address);
  3062. if (0 != *N) {
  3063. *N -= 1;
  3064. }
  3065. }
  3066. //
  3067. // Seems that some DVD drives always return AA start 0x6dd39 (which is the max legally
  3068. // representable MSF value 99/59/74) to TOC queries, even in LBA mode. If this
  3069. // is what we have for the leadout address, then lets see what READ_CAPACITY says.
  3070. // We'll also issue read capacity if the address is > than this, since we must
  3071. // be dealing with DVD or DDCD media, so the drive must support the command and
  3072. // it should give a definitive answer.
  3073. //
  3074. if (0x6dd38 <= *N) {
  3075. PDISK_GEOMETRY_EX Geometry = (PVOID)CdromToc;
  3076. ULONG Blocks;
  3077. Status = UdfPerformDevIoCtrl( IrpContext,
  3078. IOCTL_CDROM_GET_DRIVE_GEOMETRY_EX,
  3079. TargetDeviceObject,
  3080. NULL,
  3081. 0,
  3082. CdromToc,
  3083. sizeof( CDROM_TOC ),
  3084. FALSE,
  3085. TRUE,
  3086. NULL );
  3087. if (NT_SUCCESS( Status)) {
  3088. Blocks = (ULONG)(Geometry->DiskSize.QuadPart / Geometry->Geometry.BytesPerSector) - 1;
  3089. if (Blocks > *N) {
  3090. DebugTrace((0, Dbg, "Using READ_CAPACITY media size of 0x%X in place of maxed out READ_TOC value\n", Blocks));
  3091. *N = Blocks;
  3092. }
  3093. }
  3094. else {
  3095. DebugTrace(( 0, Dbg, "GET_DRIVE_GEO failed, %x\n", Status));
  3096. }
  3097. }
  3098. //
  3099. // Query the last session information from the driver. Not that this
  3100. // actually issues an LBA mode READ_TOC_EX and pulls the address from
  3101. // there.
  3102. //
  3103. Status = UdfPerformDevIoCtrl( IrpContext,
  3104. IOCTL_CDROM_GET_LAST_SESSION,
  3105. TargetDeviceObject,
  3106. NULL,
  3107. 0,
  3108. CdromToc,
  3109. sizeof( CDROM_TOC ),
  3110. FALSE,
  3111. TRUE,
  3112. NULL );
  3113. //
  3114. // Raise an exception if there was an allocation failure.
  3115. //
  3116. if (Status == STATUS_INSUFFICIENT_RESOURCES) {
  3117. DebugTrace(( 0, Dbg, "UdfDetermineVolumeBounding, GET_LAST_SESSION failed INSUFFICIENT_RESOURCES\n" ));
  3118. UdfRaiseStatus( IrpContext, Status );
  3119. }
  3120. //
  3121. // Now, if we got anything interesting out of this try, return it. If this
  3122. // failed for any other reason, we don't really care - it just means that
  3123. // if this was CDUDF media, we're gonna fail to figure it out pretty quickly.
  3124. //
  3125. // Life is tough.
  3126. //
  3127. if (NT_SUCCESS( Status ) &&
  3128. CdromToc->FirstTrack != CdromToc->LastTrack) {
  3129. //
  3130. // The 0 entry in TrackData tells us about the first track in the last
  3131. // session as a logical block address.
  3132. //
  3133. SwapCopyUchar4( S, &CdromToc->TrackData[0].Address );
  3134. //
  3135. // Save grief if the session info is messed up.
  3136. //
  3137. if (*N <= *S) {
  3138. DebugTrace(( 0, Dbg, "UdfDetermineVolumeBounding, N (0x%x) before S (0x%x), whacking both back!\n", *N, *S ));
  3139. *S = *N = 0;
  3140. }
  3141. }
  3142. DebugTrace(( 0, Dbg, "UdfDetermineVolumeBounding, S 0x%08x, N (== AA start - 150) 0x%08x\n", *S, *N));
  3143. }
  3144. finally {
  3145. DebugUnwind( "UdfDetermineVolumeBounding" );
  3146. if (CdromToc != NULL) {
  3147. UdfFreePool( &CdromToc );
  3148. }
  3149. }
  3150. DebugTrace(( -1, Dbg, "UdfDetermineVolumeBounding -> VOID\n" ));
  3151. return;
  3152. }
  3153. //
  3154. // Local support routine
  3155. //
  3156. VOID
  3157. UdfUpdateVolumeLabel (
  3158. IN PIRP_CONTEXT IrpContext,
  3159. IN PWCHAR VolumeLabel,
  3160. IN OUT PUSHORT VolumeLabelLength,
  3161. IN PUCHAR Dstring,
  3162. IN UCHAR FieldLength
  3163. )
  3164. /*++
  3165. Routine Description:
  3166. This routine will retrieve an NT volume label from a logical volume descriptor.
  3167. Arguments:
  3168. VolumeLabel - a volume label to fill in.
  3169. VolumeLabelLength - returns the length of the returned volume label.
  3170. Dstring - the dstring field containing the volume id.
  3171. FieldLength - the length of the dstring field.
  3172. Return Value:
  3173. None.
  3174. --*/
  3175. {
  3176. BOOLEAN Result;
  3177. PAGED_CODE();
  3178. //
  3179. // Check inputs.
  3180. //
  3181. ASSERT_IRP_CONTEXT( IrpContext );
  3182. DebugTrace(( +1, Dbg,
  3183. "UdfUpdateVolumeLabel, Label %08x, Dstring %08x FieldLength %02x\n",
  3184. VolumeLabel,
  3185. Dstring,
  3186. FieldLength ));
  3187. //
  3188. // Check that the dstring is usable as a volume identification.
  3189. //
  3190. Result = UdfCheckLegalCS0Dstring( IrpContext,
  3191. Dstring,
  3192. 0,
  3193. FieldLength,
  3194. TRUE );
  3195. //
  3196. // Update the label directly if the dstring is good.
  3197. //
  3198. if (Result) {
  3199. UNICODE_STRING TemporaryUnicodeString;
  3200. TemporaryUnicodeString.Buffer = VolumeLabel;
  3201. TemporaryUnicodeString.MaximumLength = MAXIMUM_VOLUME_LABEL_LENGTH;
  3202. TemporaryUnicodeString.Length = 0;
  3203. UdfConvertCS0DstringToUnicode( IrpContext,
  3204. Dstring,
  3205. 0,
  3206. FieldLength,
  3207. &TemporaryUnicodeString );
  3208. //
  3209. // Now retrieve the name for return to the caller.
  3210. //
  3211. RtlCopyMemory( VolumeLabel, TemporaryUnicodeString.Buffer, TemporaryUnicodeString.Length );
  3212. *VolumeLabelLength = TemporaryUnicodeString.Length;
  3213. DebugTrace(( 0, Dbg,
  3214. "UdfUpdateVolumeLabel, Labeled as \"%wZ\"\n",
  3215. &TemporaryUnicodeString ));
  3216. //
  3217. // Treat as label.
  3218. //
  3219. } else {
  3220. *VolumeLabelLength = 0;
  3221. DebugTrace(( 0, Dbg,
  3222. "UdfUpdateVolumeLabel, invalid label.\n" ));
  3223. }
  3224. DebugTrace(( -1, Dbg,
  3225. "UdfUpdateVolumeLabel -> VOID\n" ));
  3226. }
  3227. //
  3228. // Local support routine
  3229. //
  3230. VOID
  3231. UdfUpdateVolumeSerialNumber (
  3232. IN PIRP_CONTEXT IrpContext,
  3233. IN OUT PULONG VolumeSerialNumber,
  3234. IN PNSR_FSD Fsd
  3235. )
  3236. /*++
  3237. Routine Description:
  3238. This routine will compute the volume serial number for a set of descriptors.
  3239. Arguments:
  3240. VolumeSerialNumber - returns the volume serial number corresponding to these descriptors.
  3241. Fsd - the fileset descriptor to examine.
  3242. Return Value:
  3243. None.
  3244. --*/
  3245. {
  3246. ULONG VsnLe;
  3247. PAGED_CODE();
  3248. //
  3249. // Check input.
  3250. //
  3251. ASSERT_IRP_CONTEXT( IrpContext );
  3252. //
  3253. // The serial number is just off of the FSD. This matches Win9x.
  3254. //
  3255. VsnLe = UdfSerial32( (PCHAR) Fsd, sizeof( NSR_FSD ));
  3256. SwapCopyUchar4( VolumeSerialNumber, &VsnLe );
  3257. }