1 /* 2 * COPYRIGHT: See COPYRIGHT.TXT 3 * PROJECT: Ext2 File System Driver for WinNT/2K/XP 4 * FILE: dispatch.c 5 * PROGRAMMER: Matt Wu <mattwu@163.com> 6 * HOMEPAGE: http://www.ext2fsd.com 7 * UPDATE HISTORY: 8 */ 9 10 /* INCLUDES *****************************************************************/ 11 12 #include "ext2fs.h" 13 14 /* GLOBALS ***************************************************************/ 15 16 extern PEXT2_GLOBAL Ext2Global; 17 18 /* DEFINITIONS *************************************************************/ 19 20 #ifdef ALLOC_PRAGMA 21 #pragma alloc_text(PAGE, Ext2QueueRequest) 22 #pragma alloc_text(PAGE, Ext2DeQueueRequest) 23 #endif 24 25 /* 26 * Ext2OplockComplete 27 * 28 * callback routine of FsRtlCheckOplock when an oplock break has 29 * completed, allowing an Irp to resume execution. 30 * 31 * Arguments: 32 * 33 * Context: the IrpContext to be queued 34 * Irp: the I/O request packet 35 * 36 * Return Value: 37 * N/A 38 */ 39 40 VOID NTAPI 41 Ext2OplockComplete ( 42 IN PVOID Context, 43 IN PIRP Irp 44 ) 45 { 46 // 47 // Check on the return value in the Irp. 48 // 49 50 if (Irp->IoStatus.Status == STATUS_SUCCESS) { 51 52 // 53 // queue the Irp context in the workqueue. 54 // 55 56 Ext2QueueRequest((PEXT2_IRP_CONTEXT)Context); 57 58 } else { 59 60 // 61 // complete the request in case of failure 62 // 63 64 Ext2CompleteIrpContext( (PEXT2_IRP_CONTEXT) Context, 65 Irp->IoStatus.Status ); 66 } 67 68 return; 69 } 70 71 72 /* 73 * Ext2LockIrp 74 * 75 * performs buffer locking if we need pend the process of the Irp 76 * 77 * Arguments: 78 * Context: the irp context 79 * Irp: the I/O request packet. 80 * 81 * Return Value: 82 * N/A 83 */ 84 85 VOID NTAPI 86 Ext2LockIrp ( 87 IN PVOID Context, 88 IN PIRP Irp 89 ) 90 { 91 PIO_STACK_LOCATION IrpSp; 92 PEXT2_IRP_CONTEXT IrpContext; 93 94 if (Irp == NULL) { 95 return; 96 } 97 98 IrpSp = IoGetCurrentIrpStackLocation(Irp); 99 100 IrpContext = (PEXT2_IRP_CONTEXT) Context; 101 102 if ( IrpContext->MajorFunction == IRP_MJ_READ || 103 IrpContext->MajorFunction == IRP_MJ_WRITE ) { 104 105 // 106 // lock the user's buffer to MDL, if the I/O is bufferred 107 // 108 109 if (!IsFlagOn(IrpContext->MinorFunction, IRP_MN_MDL)) { 110 111 Ext2LockUserBuffer( Irp, IrpSp->Parameters.Write.Length, 112 (IrpContext->MajorFunction == IRP_MJ_READ) ? 113 IoWriteAccess : IoReadAccess ); 114 } 115 116 } else if (IrpContext->MajorFunction == IRP_MJ_DIRECTORY_CONTROL 117 && IrpContext->MinorFunction == IRP_MN_QUERY_DIRECTORY) { 118 119 ULONG Length = ((PEXTENDED_IO_STACK_LOCATION) IrpSp)->Parameters.QueryDirectory.Length; 120 Ext2LockUserBuffer(Irp, Length, IoWriteAccess); 121 122 } else if (IrpContext->MajorFunction == IRP_MJ_QUERY_EA) { 123 124 ULONG Length = ((PEXTENDED_IO_STACK_LOCATION) IrpSp)->Parameters.QueryEa.Length; 125 Ext2LockUserBuffer(Irp, Length, IoWriteAccess); 126 127 } else if (IrpContext->MajorFunction == IRP_MJ_SET_EA) { 128 ULONG Length = ((PEXTENDED_IO_STACK_LOCATION) IrpSp)->Parameters.SetEa.Length; 129 Ext2LockUserBuffer(Irp, Length, IoReadAccess); 130 131 } else if ( (IrpContext->MajorFunction == IRP_MJ_FILE_SYSTEM_CONTROL) && 132 (IrpContext->MinorFunction == IRP_MN_USER_FS_REQUEST) ) { 133 PEXTENDED_IO_STACK_LOCATION EIrpSp = (PEXTENDED_IO_STACK_LOCATION)IrpSp; 134 if ( (EIrpSp->Parameters.FileSystemControl.FsControlCode == FSCTL_GET_VOLUME_BITMAP) || 135 (EIrpSp->Parameters.FileSystemControl.FsControlCode == FSCTL_GET_RETRIEVAL_POINTERS) || 136 (EIrpSp->Parameters.FileSystemControl.FsControlCode == FSCTL_GET_RETRIEVAL_POINTER_BASE) ) { 137 ULONG Length = EIrpSp->Parameters.FileSystemControl.OutputBufferLength; 138 Ext2LockUserBuffer(Irp, Length, IoWriteAccess); 139 } 140 } 141 142 // Mark the request as pending status 143 144 IoMarkIrpPending( Irp ); 145 146 return; 147 } 148 149 NTSTATUS 150 Ext2QueueRequest (IN PEXT2_IRP_CONTEXT IrpContext) 151 { 152 ASSERT(IrpContext); 153 154 ASSERT((IrpContext->Identifier.Type == EXT2ICX) && 155 (IrpContext->Identifier.Size == sizeof(EXT2_IRP_CONTEXT))); 156 157 /* set the flags of "can wait" and "queued" */ 158 SetFlag(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT); 159 SetFlag(IrpContext->Flags, IRP_CONTEXT_FLAG_REQUEUED); 160 161 /* make sure the buffer is kept valid in system context */ 162 Ext2LockIrp(IrpContext, IrpContext->Irp); 163 164 /* initialize workite*/ 165 ExInitializeWorkItem( 166 &IrpContext->WorkQueueItem, 167 Ext2DeQueueRequest, 168 IrpContext ); 169 170 /* dispatch it */ 171 ExQueueWorkItem(&IrpContext->WorkQueueItem, CriticalWorkQueue); 172 173 return STATUS_PENDING; 174 } 175 176 177 VOID NTAPI 178 Ext2DeQueueRequest (IN PVOID Context) 179 { 180 PEXT2_IRP_CONTEXT IrpContext; 181 182 IrpContext = (PEXT2_IRP_CONTEXT) Context; 183 184 ASSERT(IrpContext); 185 186 ASSERT((IrpContext->Identifier.Type == EXT2ICX) && 187 (IrpContext->Identifier.Size == sizeof(EXT2_IRP_CONTEXT))); 188 189 _SEH2_TRY { 190 191 _SEH2_TRY { 192 193 FsRtlEnterFileSystem(); 194 195 if (!IrpContext->IsTopLevel) { 196 IoSetTopLevelIrp((PIRP) FSRTL_FSP_TOP_LEVEL_IRP); 197 } 198 199 Ext2DispatchRequest(IrpContext); 200 201 } _SEH2_EXCEPT (Ext2ExceptionFilter(IrpContext, _SEH2_GetExceptionInformation())) { 202 203 Ext2ExceptionHandler(IrpContext); 204 } _SEH2_END; 205 206 } _SEH2_FINALLY { 207 208 IoSetTopLevelIrp(NULL); 209 210 FsRtlExitFileSystem(); 211 } _SEH2_END; 212 } 213 214 215 NTSTATUS 216 Ext2DispatchRequest (IN PEXT2_IRP_CONTEXT IrpContext) 217 { 218 ASSERT(IrpContext); 219 220 ASSERT((IrpContext->Identifier.Type == EXT2ICX) && 221 (IrpContext->Identifier.Size == sizeof(EXT2_IRP_CONTEXT))); 222 223 switch (IrpContext->MajorFunction) { 224 225 case IRP_MJ_CREATE: 226 return Ext2Create(IrpContext); 227 228 case IRP_MJ_CLOSE: 229 return Ext2Close(IrpContext); 230 231 case IRP_MJ_READ: 232 return Ext2Read(IrpContext); 233 234 case IRP_MJ_WRITE: 235 return Ext2Write(IrpContext); 236 237 case IRP_MJ_FLUSH_BUFFERS: 238 return Ext2Flush(IrpContext); 239 240 case IRP_MJ_QUERY_INFORMATION: 241 return Ext2QueryFileInformation(IrpContext); 242 243 case IRP_MJ_SET_INFORMATION: 244 return Ext2SetFileInformation(IrpContext); 245 246 case IRP_MJ_QUERY_VOLUME_INFORMATION: 247 return Ext2QueryVolumeInformation(IrpContext); 248 249 case IRP_MJ_SET_VOLUME_INFORMATION: 250 return Ext2SetVolumeInformation(IrpContext); 251 252 case IRP_MJ_DIRECTORY_CONTROL: 253 return Ext2DirectoryControl(IrpContext); 254 255 case IRP_MJ_FILE_SYSTEM_CONTROL: 256 return Ext2FileSystemControl(IrpContext); 257 258 case IRP_MJ_DEVICE_CONTROL: 259 return Ext2DeviceControl(IrpContext); 260 261 case IRP_MJ_LOCK_CONTROL: 262 return Ext2LockControl(IrpContext); 263 264 case IRP_MJ_CLEANUP: 265 return Ext2Cleanup(IrpContext); 266 267 case IRP_MJ_SHUTDOWN: 268 return Ext2ShutDown(IrpContext); 269 270 #if (_WIN32_WINNT >= 0x0500) 271 case IRP_MJ_PNP: 272 return Ext2Pnp(IrpContext); 273 #endif //(_WIN32_WINNT >= 0x0500) 274 default: 275 DEBUG(DL_ERR, ( "Ext2DispatchRequest: Unexpected major function: %xh\n", 276 IrpContext->MajorFunction)); 277 278 Ext2CompleteIrpContext(IrpContext, STATUS_DRIVER_INTERNAL_ERROR); 279 280 return STATUS_DRIVER_INTERNAL_ERROR; 281 } 282 } 283 284 285 NTSTATUS NTAPI 286 Ext2BuildRequest (PDEVICE_OBJECT DeviceObject, PIRP Irp) 287 { 288 BOOLEAN AtIrqlPassiveLevel = FALSE; 289 BOOLEAN IsTopLevelIrp = FALSE; 290 PEXT2_IRP_CONTEXT IrpContext = NULL; 291 NTSTATUS Status = STATUS_UNSUCCESSFUL; 292 293 _SEH2_TRY { 294 295 _SEH2_TRY { 296 297 #if EXT2_DEBUG 298 Ext2DbgPrintCall(DeviceObject, Irp); 299 #endif 300 301 AtIrqlPassiveLevel = (KeGetCurrentIrql() == PASSIVE_LEVEL); 302 303 if (AtIrqlPassiveLevel) { 304 FsRtlEnterFileSystem(); 305 } 306 307 if (!IoGetTopLevelIrp()) { 308 IsTopLevelIrp = TRUE; 309 IoSetTopLevelIrp(Irp); 310 } 311 312 IrpContext = Ext2AllocateIrpContext(DeviceObject, Irp); 313 314 if (!IrpContext) { 315 316 Status = STATUS_INSUFFICIENT_RESOURCES; 317 Irp->IoStatus.Status = Status; 318 319 Ext2CompleteRequest(Irp, TRUE, IO_NO_INCREMENT); 320 321 } else { 322 323 if ((IrpContext->MajorFunction == IRP_MJ_CREATE) && 324 !AtIrqlPassiveLevel) { 325 326 DbgBreak(); 327 } 328 329 Status = Ext2DispatchRequest(IrpContext); 330 } 331 } _SEH2_EXCEPT (Ext2ExceptionFilter(IrpContext, _SEH2_GetExceptionInformation())) { 332 333 Status = Ext2ExceptionHandler(IrpContext); 334 } _SEH2_END; 335 336 } _SEH2_FINALLY { 337 338 if (IsTopLevelIrp) { 339 IoSetTopLevelIrp(NULL); 340 } 341 342 if (AtIrqlPassiveLevel) { 343 FsRtlExitFileSystem(); 344 } 345 } _SEH2_END; 346 347 return Status; 348 } 349