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.

378 lines
11 KiB

  1. /*++
  2. Copyright (c) 1996-2000 Microsoft Corporation
  3. Module Name:
  4. UdfInit.c
  5. Abstract:
  6. This module implements the DRIVER_INITIALIZATION routine for Udfs
  7. // @@BEGIN_DDKSPLIT
  8. Author:
  9. Dan Lovinger [DanLo] 24-May-1996
  10. Tom Jolly [tomjolly] 21-Jan-2000
  11. Revision History:
  12. // @@END_DDKSPLIT
  13. --*/
  14. #include "UdfProcs.h"
  15. //
  16. // The Bug check file id for this module
  17. //
  18. #define BugCheckFileId (UDFS_BUG_CHECK_UDFINIT)
  19. //
  20. // The local debug trace level
  21. //
  22. #define Dbg (UDFS_DEBUG_LEVEL_UDFINIT)
  23. NTSTATUS
  24. DriverEntry(
  25. IN PDRIVER_OBJECT DriverObject,
  26. IN PUNICODE_STRING RegistryPath
  27. );
  28. VOID
  29. UdfInitializeGlobalData (
  30. IN PDRIVER_OBJECT DriverObject,
  31. IN PDEVICE_OBJECT *FileSystemDeviceObjects
  32. );
  33. #ifdef ALLOC_PRAGMA
  34. #pragma alloc_text(INIT, DriverEntry)
  35. #pragma alloc_text(INIT, UdfInitializeGlobalData)
  36. #endif
  37. //
  38. // Local support routine
  39. //
  40. NTSTATUS
  41. DriverEntry(
  42. IN PDRIVER_OBJECT DriverObject,
  43. IN PUNICODE_STRING RegistryPath
  44. )
  45. /*++
  46. Routine Description:
  47. This is the initialization routine for the UDF file system
  48. device driver. This routine creates the device object for the FileSystem
  49. device and performs all other driver initialization.
  50. Arguments:
  51. DriverObject - Pointer to driver object created by the system.
  52. Return Value:
  53. NTSTATUS - The function value is the final status from the initialization
  54. operation.
  55. --*/
  56. {
  57. NTSTATUS Status;
  58. UNICODE_STRING UnicodeString;
  59. PDEVICE_OBJECT UdfsFileSystemDeviceObjects[NUMBER_OF_FS_OBJECTS];
  60. PDEVICE_OBJECT UdfsDiskFileSystemDeviceObject;
  61. //
  62. // Create the device objects for both device "types". Since
  63. // UDF is a legitimate filesystem for media underlying device
  64. // drivers claiming both DVD/CDROMs and disks, we must register
  65. // this filesystem twice.
  66. //
  67. ASSERT( NUMBER_OF_FS_OBJECTS >= 2 );
  68. RtlZeroMemory( &UdfsFileSystemDeviceObjects, sizeof(PDEVICE_OBJECT) * NUMBER_OF_FS_OBJECTS );
  69. RtlInitUnicodeString( &UnicodeString, L"\\UdfsCdRom" );
  70. Status = IoCreateDevice( DriverObject,
  71. 0,
  72. &UnicodeString,
  73. FILE_DEVICE_CD_ROM_FILE_SYSTEM,
  74. 0,
  75. FALSE,
  76. &UdfsFileSystemDeviceObjects[0] );
  77. if (!NT_SUCCESS( Status )) {
  78. return Status;
  79. }
  80. RtlInitUnicodeString( &UnicodeString, L"\\UdfsDisk" );
  81. Status = IoCreateDevice( DriverObject,
  82. 0,
  83. &UnicodeString,
  84. FILE_DEVICE_DISK_FILE_SYSTEM,
  85. 0,
  86. FALSE,
  87. &UdfsFileSystemDeviceObjects[1] );
  88. if (!NT_SUCCESS( Status )) {
  89. ObDereferenceObject( UdfsFileSystemDeviceObjects[0] );
  90. return Status;
  91. }
  92. try {
  93. Status = STATUS_SUCCESS;
  94. //
  95. // Initialize the global data structures
  96. //
  97. UdfInitializeGlobalData( DriverObject, UdfsFileSystemDeviceObjects );
  98. //
  99. // Note that because of the way data caching is done, we set neither
  100. // the Direct I/O or Buffered I/O bit in DeviceObject->Flags. If
  101. // data is not in the cache, or the request is not buffered, we may,
  102. // set up for Direct I/O by hand.
  103. //
  104. //
  105. // Initialize the driver object with this driver's entry points.
  106. //
  107. // NOTE - Each entry in the dispatch table must have an entry in
  108. // the Fsp/Fsd dispatch switch statements.
  109. //
  110. DriverObject->MajorFunction[IRP_MJ_CREATE] =
  111. DriverObject->MajorFunction[IRP_MJ_CLOSE] =
  112. DriverObject->MajorFunction[IRP_MJ_READ] =
  113. DriverObject->MajorFunction[IRP_MJ_WRITE] =
  114. DriverObject->MajorFunction[IRP_MJ_QUERY_INFORMATION] =
  115. DriverObject->MajorFunction[IRP_MJ_SET_INFORMATION] =
  116. DriverObject->MajorFunction[IRP_MJ_QUERY_VOLUME_INFORMATION]=
  117. DriverObject->MajorFunction[IRP_MJ_DIRECTORY_CONTROL] =
  118. DriverObject->MajorFunction[IRP_MJ_FILE_SYSTEM_CONTROL] =
  119. DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] =
  120. DriverObject->MajorFunction[IRP_MJ_LOCK_CONTROL] =
  121. DriverObject->MajorFunction[IRP_MJ_CLEANUP] =
  122. DriverObject->MajorFunction[IRP_MJ_PNP] = (PDRIVER_DISPATCH) UdfFsdDispatch;
  123. DriverObject->FastIoDispatch = &UdfFastIoDispatch;
  124. //
  125. // Register the file system with the I/O system
  126. //
  127. IoRegisterFileSystem( UdfsFileSystemDeviceObjects[0] );
  128. IoRegisterFileSystem( UdfsFileSystemDeviceObjects[1] );
  129. }
  130. except (FsRtlIsNtstatusExpected(GetExceptionCode()) ?
  131. EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) {
  132. ObDereferenceObject( UdfsFileSystemDeviceObjects[0] );
  133. ObDereferenceObject( UdfsFileSystemDeviceObjects[1] );
  134. Status = GetExceptionCode();
  135. }
  136. //
  137. // And return to our caller
  138. //
  139. return Status;
  140. }
  141. //
  142. // Local support routine
  143. //
  144. VOID
  145. UdfInitializeGlobalData (
  146. IN PDRIVER_OBJECT DriverObject,
  147. IN PDEVICE_OBJECT *UdfsFileSystemDeviceObjects
  148. )
  149. /*++
  150. Routine Description:
  151. This routine initializes the global Udfs data structures.
  152. Arguments:
  153. DriverObject - Supplies the driver object for UDFS.
  154. FileSystemDeviceObjects - Supplies a vector of device objects for UDFS.
  155. Return Value:
  156. None.
  157. --*/
  158. {
  159. USHORT CcbMaxDepth;
  160. USHORT FcbDataMaxDepth;
  161. USHORT FcbIndexMaxDepth;
  162. USHORT FcbNonPagedMaxDepth;
  163. USHORT IrpContextMaxDepth;
  164. USHORT LcbMaxDepth;
  165. TIMESTAMP UdfTime;
  166. //
  167. // Initialize the CRC table. Per UDF 1.01, we use the seed 10041 octal (4129 dec).
  168. // We do this first because it can raise (allocates memory)
  169. //
  170. UdfInitializeCrc16( 4129 );
  171. //
  172. // Start by initializing the FastIoDispatch Table.
  173. //
  174. RtlZeroMemory( &UdfFastIoDispatch, sizeof( FAST_IO_DISPATCH ));
  175. UdfFastIoDispatch.SizeOfFastIoDispatch = sizeof(FAST_IO_DISPATCH);
  176. UdfFastIoDispatch.AcquireFileForNtCreateSection = UdfAcquireForCreateSection;
  177. UdfFastIoDispatch.ReleaseFileForNtCreateSection = UdfReleaseForCreateSection;
  178. UdfFastIoDispatch.FastIoCheckIfPossible = UdfFastIoCheckIfPossible; // CheckForFastIo
  179. UdfFastIoDispatch.FastIoRead = FsRtlCopyRead; // Read
  180. UdfFastIoDispatch.FastIoQueryBasicInfo = NULL; // QueryBasicInfo
  181. UdfFastIoDispatch.FastIoQueryStandardInfo = NULL; // QueryStandardInfo
  182. UdfFastIoDispatch.FastIoLock = NULL; // Lock
  183. UdfFastIoDispatch.FastIoUnlockSingle = NULL; // UnlockSingle
  184. UdfFastIoDispatch.FastIoUnlockAll = NULL; // UnlockAll
  185. UdfFastIoDispatch.FastIoUnlockAllByKey = NULL; // UnlockAllByKey
  186. UdfFastIoDispatch.FastIoQueryNetworkOpenInfo = NULL; // QueryNetworkInfo
  187. //
  188. // Initialize the UdfData structure.
  189. //
  190. RtlZeroMemory( &UdfData, sizeof( UDF_DATA ));
  191. UdfData.NodeTypeCode = UDFS_NTC_DATA_HEADER;
  192. UdfData.NodeByteSize = sizeof( UDF_DATA );
  193. UdfData.DriverObject = DriverObject;
  194. RtlCopyMemory( &UdfData.FileSystemDeviceObjects,
  195. UdfsFileSystemDeviceObjects,
  196. sizeof(PDEVICE_OBJECT) * NUMBER_OF_FS_OBJECTS );
  197. InitializeListHead( &UdfData.VcbQueue );
  198. ExInitializeResourceLite( &UdfData.DataResource );
  199. //
  200. // Initialize the cache manager callback routines
  201. //
  202. UdfData.CacheManagerCallbacks.AcquireForLazyWrite = &UdfAcquireForCache;
  203. UdfData.CacheManagerCallbacks.ReleaseFromLazyWrite = &UdfReleaseFromCache;
  204. UdfData.CacheManagerCallbacks.AcquireForReadAhead = &UdfAcquireForCache;
  205. UdfData.CacheManagerCallbacks.ReleaseFromReadAhead = &UdfReleaseFromCache;
  206. UdfData.CacheManagerVolumeCallbacks.AcquireForLazyWrite = &UdfNoopAcquire;
  207. UdfData.CacheManagerVolumeCallbacks.ReleaseFromLazyWrite = &UdfNoopRelease;
  208. UdfData.CacheManagerVolumeCallbacks.AcquireForReadAhead = &UdfNoopAcquire;
  209. UdfData.CacheManagerVolumeCallbacks.ReleaseFromReadAhead = &UdfNoopRelease;
  210. //
  211. // Initialize the lock mutex and the async and delay close queues.
  212. //
  213. ExInitializeFastMutex( &UdfData.UdfDataMutex );
  214. InitializeListHead( &UdfData.AsyncCloseQueue );
  215. InitializeListHead( &UdfData.DelayedCloseQueue );
  216. ExInitializeWorkItem( &UdfData.CloseItem,
  217. (PWORKER_THREAD_ROUTINE) UdfFspClose,
  218. NULL );
  219. //
  220. // Do the initialization based on the system size.
  221. //
  222. switch (MmQuerySystemSize()) {
  223. case MmSmallSystem:
  224. IrpContextMaxDepth = 4;
  225. UdfData.MaxDelayedCloseCount = 10;
  226. UdfData.MinDelayedCloseCount = 2;
  227. break;
  228. case MmLargeSystem:
  229. IrpContextMaxDepth = 24;
  230. UdfData.MaxDelayedCloseCount = 72;
  231. UdfData.MinDelayedCloseCount = 18;
  232. break;
  233. default:
  234. case MmMediumSystem:
  235. IrpContextMaxDepth = 8;
  236. UdfData.MaxDelayedCloseCount = 32;
  237. UdfData.MinDelayedCloseCount = 8;
  238. break;
  239. }
  240. //
  241. // Size lookasides to match what will commonly be dumped into them when we
  242. // run down the delayed close queues.
  243. //
  244. LcbMaxDepth =
  245. CcbMaxDepth =
  246. FcbDataMaxDepth =
  247. FcbNonPagedMaxDepth = (USHORT) (UdfData.MaxDelayedCloseCount - UdfData.MinDelayedCloseCount);
  248. //
  249. // We should tend to have fewer indices than files.
  250. //
  251. FcbIndexMaxDepth = FcbNonPagedMaxDepth / 2;
  252. #define NPagedInit(L,S,T,D) { ExInitializeNPagedLookasideList( (L), NULL, NULL, POOL_RAISE_IF_ALLOCATION_FAILURE, S, T, D); }
  253. #define PagedInit(L,S,T,D) { ExInitializePagedLookasideList( (L), NULL, NULL, POOL_RAISE_IF_ALLOCATION_FAILURE, S, T, D); }
  254. NPagedInit( &UdfIrpContextLookasideList, sizeof( IRP_CONTEXT ), TAG_IRP_CONTEXT, IrpContextMaxDepth );
  255. NPagedInit( &UdfFcbNonPagedLookasideList, sizeof( FCB_NONPAGED ), TAG_FCB_NONPAGED, FcbNonPagedMaxDepth );
  256. PagedInit( &UdfCcbLookasideList, sizeof( CCB ), TAG_CCB, CcbMaxDepth );
  257. PagedInit( &UdfFcbIndexLookasideList, SIZEOF_FCB_INDEX, TAG_FCB_INDEX, FcbIndexMaxDepth );
  258. PagedInit( &UdfFcbDataLookasideList, SIZEOF_FCB_DATA, TAG_FCB_DATA, FcbDataMaxDepth );
  259. PagedInit( &UdfLcbLookasideList, SIZEOF_LOOKASIDE_LCB, TAG_LCB, LcbMaxDepth );
  260. //
  261. // Initialize our default time which we use when enumerating FIDs whose
  262. // associated FEs are toast.
  263. //
  264. RtlZeroMemory( &UdfTime, sizeof( TIMESTAMP));
  265. UdfTime.Day = 1;
  266. UdfTime.Month = 7;
  267. UdfTime.Year = 1974;
  268. UdfConvertUdfTimeToNtTime( NULL,
  269. &UdfTime,
  270. &UdfCorruptFileTime);
  271. }