1 /* 2 * PROJECT: ReactOS Kernel 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: ntoskrnl/io/iomgr/irp.c 5 * PURPOSE: IRP Handling Functions 6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org) 7 * Gunnar Dalsnes 8 * Filip Navara (navaraf@reactos.org) 9 * Pierre Schweitzer (pierre@reactos.org) 10 */ 11 12 /* INCLUDES ****************************************************************/ 13 14 #include <ntoskrnl.h> 15 #define NDEBUG 16 #include <debug.h> 17 18 PIRP IopDeadIrp; 19 RESERVE_IRP_ALLOCATOR IopReserveIrpAllocator; 20 21 /* PRIVATE FUNCTIONS ********************************************************/ 22 23 VOID 24 NTAPI 25 IopFreeIrpKernelApc(IN PKAPC Apc, 26 IN PKNORMAL_ROUTINE *NormalRoutine, 27 IN PVOID *NormalContext, 28 IN PVOID *SystemArgument1, 29 IN PVOID *SystemArgument2) 30 { 31 /* Free the IRP */ 32 IoFreeIrp(CONTAINING_RECORD(Apc, IRP, Tail.Apc)); 33 } 34 35 VOID 36 NTAPI 37 IopAbortIrpKernelApc(IN PKAPC Apc) 38 { 39 /* Free the IRP */ 40 IoFreeIrp(CONTAINING_RECORD(Apc, IRP, Tail.Apc)); 41 } 42 43 NTSTATUS 44 NTAPI 45 IopCleanupFailedIrp(IN PFILE_OBJECT FileObject, 46 IN PKEVENT EventObject OPTIONAL, 47 IN PVOID Buffer OPTIONAL) 48 { 49 PAGED_CODE(); 50 51 /* Dereference the event */ 52 if (EventObject) ObDereferenceObject(EventObject); 53 54 /* Free a buffer, if any */ 55 if (Buffer) ExFreePool(Buffer); 56 57 /* If this was a file opened for synch I/O, then unlock it */ 58 if (FileObject->Flags & FO_SYNCHRONOUS_IO) IopUnlockFileObject(FileObject); 59 60 /* Now dereference it and return */ 61 ObDereferenceObject(FileObject); 62 return STATUS_INSUFFICIENT_RESOURCES; 63 } 64 65 VOID 66 NTAPI 67 IopAbortInterruptedIrp(IN PKEVENT EventObject, 68 IN PIRP Irp) 69 { 70 KIRQL OldIrql; 71 BOOLEAN CancelResult; 72 LARGE_INTEGER Wait; 73 PAGED_CODE(); 74 75 /* Raise IRQL to APC */ 76 KeRaiseIrql(APC_LEVEL, &OldIrql); 77 78 /* Check if nobody completed it yet */ 79 if (!KeReadStateEvent(EventObject)) 80 { 81 /* First, cancel it */ 82 CancelResult = IoCancelIrp(Irp); 83 KeLowerIrql(OldIrql); 84 85 /* Check if we cancelled it */ 86 if (CancelResult) 87 { 88 /* Wait for the IRP to be cancelled */ 89 Wait.QuadPart = -100000; 90 while (!KeReadStateEvent(EventObject)) 91 { 92 /* Delay indefintely */ 93 KeDelayExecutionThread(KernelMode, FALSE, &Wait); 94 } 95 } 96 else 97 { 98 /* No cancellation done, so wait for the I/O system to kill it */ 99 KeWaitForSingleObject(EventObject, 100 Executive, 101 KernelMode, 102 FALSE, 103 NULL); 104 } 105 } 106 else 107 { 108 /* We got preempted, so give up */ 109 KeLowerIrql(OldIrql); 110 } 111 } 112 113 VOID 114 NTAPI 115 IopDisassociateThreadIrp(VOID) 116 { 117 KIRQL OldIrql, LockIrql; 118 PETHREAD IrpThread; 119 PLIST_ENTRY IrpEntry; 120 PIO_ERROR_LOG_PACKET ErrorLogEntry; 121 PDEVICE_OBJECT DeviceObject = NULL; 122 PIO_STACK_LOCATION IoStackLocation; 123 124 /* First, raise to APC to protect IrpList */ 125 KeRaiseIrql(APC_LEVEL, &OldIrql); 126 127 /* Get the Thread and check the list */ 128 IrpThread = PsGetCurrentThread(); 129 if (IsListEmpty(&IrpThread->IrpList)) 130 { 131 /* It got completed now, so quit */ 132 KeLowerIrql(OldIrql); 133 return; 134 } 135 136 /* Ensure no one will come disturb */ 137 LockIrql = KeAcquireQueuedSpinLock(LockQueueIoCompletionLock); 138 139 /* Get the misbehaving IRP */ 140 IrpEntry = IrpThread->IrpList.Flink; 141 IopDeadIrp = CONTAINING_RECORD(IrpEntry, IRP, ThreadListEntry); 142 IOTRACE(IO_IRP_DEBUG, 143 "%s - Deassociating IRP %p for %p\n", 144 __FUNCTION__, 145 IopDeadIrp, 146 IrpThread); 147 148 /* Don't cancel the IRP if it's already been completed far */ 149 if (IopDeadIrp->CurrentLocation == (IopDeadIrp->StackCount + 2)) 150 { 151 /* Return */ 152 KeReleaseQueuedSpinLock(LockQueueIoCompletionLock, LockIrql); 153 KeLowerIrql(OldIrql); 154 return; 155 } 156 157 /* Disown the IRP! */ 158 IopDeadIrp->Tail.Overlay.Thread = NULL; 159 RemoveHeadList(&IrpThread->IrpList); 160 InitializeListHead(&IopDeadIrp->ThreadListEntry); 161 162 /* Get the stack location and check if it's valid */ 163 IoStackLocation = IoGetCurrentIrpStackLocation(IopDeadIrp); 164 if (IopDeadIrp->CurrentLocation <= IopDeadIrp->StackCount) 165 { 166 /* Get the device object */ 167 DeviceObject = IoStackLocation->DeviceObject; 168 } 169 170 KeReleaseQueuedSpinLock(LockQueueIoCompletionLock, LockIrql); 171 /* Lower IRQL now, since we have the pointers we need */ 172 KeLowerIrql(OldIrql); 173 174 /* Check if we can send an Error Log Entry*/ 175 if (DeviceObject) 176 { 177 /* Allocate an entry */ 178 ErrorLogEntry = IoAllocateErrorLogEntry(DeviceObject, 179 sizeof(IO_ERROR_LOG_PACKET)); 180 if (ErrorLogEntry) 181 { 182 /* Write the entry */ 183 ErrorLogEntry->ErrorCode = IO_DRIVER_CANCEL_TIMEOUT; 184 IoWriteErrorLogEntry(ErrorLogEntry); 185 } 186 } 187 } 188 189 VOID 190 NTAPI 191 IopCleanupIrp(IN PIRP Irp, 192 IN PFILE_OBJECT FileObject) 193 { 194 PMDL Mdl; 195 IOTRACE(IO_IRP_DEBUG, 196 "%s - Cleaning IRP %p for %p\n", 197 __FUNCTION__, 198 Irp, 199 FileObject); 200 201 /* Check if there's an MDL */ 202 while ((Mdl = Irp->MdlAddress)) 203 { 204 /* Clear all of them */ 205 Irp->MdlAddress = Mdl->Next; 206 IoFreeMdl(Mdl); 207 } 208 209 /* Check if the IRP has system buffer */ 210 if (Irp->Flags & IRP_DEALLOCATE_BUFFER) 211 { 212 /* Free the buffer */ 213 ExFreePoolWithTag(Irp->AssociatedIrp.SystemBuffer, TAG_SYS_BUF); 214 } 215 216 /* Check if this IRP has a user event, a file object, and is async */ 217 if ((Irp->UserEvent) && 218 !(Irp->Flags & IRP_SYNCHRONOUS_API) && 219 (FileObject)) 220 { 221 /* Dereference the User Event */ 222 ObDereferenceObject(Irp->UserEvent); 223 } 224 225 /* Check if we have a file object and this isn't a create operation */ 226 if ((FileObject) && !(Irp->Flags & IRP_CREATE_OPERATION)) 227 { 228 /* Dereference the file object */ 229 ObDereferenceObject(FileObject); 230 } 231 232 /* Free the IRP */ 233 IoFreeIrp(Irp); 234 } 235 236 VOID 237 NTAPI 238 IopCompleteRequest(IN PKAPC Apc, 239 IN PKNORMAL_ROUTINE* NormalRoutine, 240 IN PVOID* NormalContext, 241 IN PVOID* SystemArgument1, 242 IN PVOID* SystemArgument2) 243 { 244 PFILE_OBJECT FileObject; 245 PIRP Irp; 246 PMDL Mdl, NextMdl; 247 PVOID Port = NULL, Key = NULL; 248 BOOLEAN SignaledCreateRequest = FALSE; 249 250 /* Get data from the APC */ 251 FileObject = (PFILE_OBJECT)*SystemArgument1; 252 Irp = CONTAINING_RECORD(Apc, IRP, Tail.Apc); 253 IOTRACE(IO_IRP_DEBUG, 254 "%s - Completing IRP %p for %p\n", 255 __FUNCTION__, 256 Irp, 257 FileObject); 258 259 /* Sanity check */ 260 ASSERT(Irp->IoStatus.Status != (NTSTATUS)0xFFFFFFFF); 261 262 /* Check if we have a file object */ 263 if (*SystemArgument2) 264 { 265 /* Check if we're reparsing */ 266 if ((Irp->IoStatus.Status == STATUS_REPARSE) && 267 (Irp->IoStatus.Information == IO_REPARSE_TAG_MOUNT_POINT)) 268 { 269 PREPARSE_DATA_BUFFER ReparseData; 270 271 ReparseData = (PREPARSE_DATA_BUFFER)*SystemArgument2; 272 273 ASSERT(ReparseData->ReparseTag == IO_REPARSE_TAG_MOUNT_POINT); 274 ASSERT(ReparseData->ReparseDataLength < MAXIMUM_REPARSE_DATA_BUFFER_SIZE); 275 ASSERT(ReparseData->Reserved < MAXIMUM_REPARSE_DATA_BUFFER_SIZE); 276 277 IopDoNameTransmogrify(Irp, FileObject, ReparseData); 278 } 279 } 280 281 /* Handle Buffered case first */ 282 if (Irp->Flags & IRP_BUFFERED_IO) 283 { 284 /* Check if we have an input buffer and if we succeeded */ 285 if ((Irp->Flags & IRP_INPUT_OPERATION) && 286 (Irp->IoStatus.Status != STATUS_VERIFY_REQUIRED) && 287 !(NT_ERROR(Irp->IoStatus.Status))) 288 { 289 _SEH2_TRY 290 { 291 /* Copy the buffer back to the user */ 292 RtlCopyMemory(Irp->UserBuffer, 293 Irp->AssociatedIrp.SystemBuffer, 294 Irp->IoStatus.Information); 295 } 296 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 297 { 298 /* Fail the IRP */ 299 Irp->IoStatus.Status = _SEH2_GetExceptionCode(); 300 } 301 _SEH2_END; 302 } 303 304 /* Also check if we should de-allocate it */ 305 if (Irp->Flags & IRP_DEALLOCATE_BUFFER) 306 { 307 /* Deallocate it */ 308 ExFreePool(Irp->AssociatedIrp.SystemBuffer); 309 } 310 } 311 312 /* Now we got rid of these two... */ 313 Irp->Flags &= ~(IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER); 314 315 /* Check if there's an MDL */ 316 for (Mdl = Irp->MdlAddress; Mdl; Mdl = NextMdl) 317 { 318 /* Free it */ 319 NextMdl = Mdl->Next; 320 IoFreeMdl(Mdl); 321 } 322 323 /* No MDLs left */ 324 Irp->MdlAddress = NULL; 325 326 /* 327 * Check if either the request was completed without any errors 328 * (but warnings are OK!), or if it was completed with an error, but 329 * did return from a pending I/O Operation and is not synchronous. 330 */ 331 if (!NT_ERROR(Irp->IoStatus.Status) || 332 (Irp->PendingReturned && 333 !IsIrpSynchronous(Irp, FileObject))) 334 { 335 /* Get any information we need from the FO before we kill it */ 336 if ((FileObject) && (FileObject->CompletionContext)) 337 { 338 /* Save Completion Data */ 339 Port = FileObject->CompletionContext->Port; 340 Key = FileObject->CompletionContext->Key; 341 } 342 343 /* Check for UserIos */ 344 if (Irp->UserIosb != NULL) 345 { 346 /* Use SEH to make sure we don't write somewhere invalid */ 347 _SEH2_TRY 348 { 349 /* Save the IOSB Information */ 350 *Irp->UserIosb = Irp->IoStatus; 351 } 352 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 353 { 354 /* Ignore any error */ 355 } 356 _SEH2_END; 357 } 358 359 /* Check if we have an event or a file object */ 360 if (Irp->UserEvent) 361 { 362 /* At the very least, this is a PKEVENT, so signal it always */ 363 KeSetEvent(Irp->UserEvent, 0, FALSE); 364 365 /* Check if we also have a File Object */ 366 if (FileObject) 367 { 368 /* Check if this is an Asynch API */ 369 if (!(Irp->Flags & IRP_SYNCHRONOUS_API)) 370 { 371 /* Dereference the event */ 372 ObDereferenceObject(Irp->UserEvent); 373 } 374 375 /* 376 * Now, if this is a Synch I/O File Object, then this event is 377 * NOT an actual Executive Event, so we won't dereference it, 378 * and instead, we will signal the File Object 379 */ 380 if ((FileObject->Flags & FO_SYNCHRONOUS_IO) && 381 !(Irp->Flags & IRP_OB_QUERY_NAME)) 382 { 383 /* Signal the file object and set the status */ 384 KeSetEvent(&FileObject->Event, 0, FALSE); 385 FileObject->FinalStatus = Irp->IoStatus.Status; 386 } 387 388 /* 389 * This could also be a create operation, in which case we want 390 * to make sure there's no APC fired. 391 */ 392 if (Irp->Flags & IRP_CREATE_OPERATION) 393 { 394 /* Clear the APC Routine and remember this */ 395 Irp->Overlay.AsynchronousParameters.UserApcRoutine = NULL; 396 SignaledCreateRequest = TRUE; 397 } 398 } 399 } 400 else if (FileObject) 401 { 402 /* Signal the file object and set the status */ 403 KeSetEvent(&FileObject->Event, 0, FALSE); 404 FileObject->FinalStatus = Irp->IoStatus.Status; 405 406 /* 407 * This could also be a create operation, in which case we want 408 * to make sure there's no APC fired. 409 */ 410 if (Irp->Flags & IRP_CREATE_OPERATION) 411 { 412 /* Clear the APC Routine and remember this */ 413 Irp->Overlay.AsynchronousParameters.UserApcRoutine = NULL; 414 SignaledCreateRequest = TRUE; 415 } 416 } 417 418 /* Update transfer count for everything but create operation */ 419 if (!(Irp->Flags & IRP_CREATE_OPERATION)) 420 { 421 if (Irp->Flags & IRP_WRITE_OPERATION) 422 { 423 /* Update write transfer count */ 424 IopUpdateTransferCount(IopWriteTransfer, 425 (ULONG)Irp->IoStatus.Information); 426 } 427 else if (Irp->Flags & IRP_READ_OPERATION) 428 { 429 /* Update read transfer count */ 430 IopUpdateTransferCount(IopReadTransfer, 431 (ULONG)Irp->IoStatus.Information); 432 } 433 else 434 { 435 /* Update other transfer count */ 436 IopUpdateTransferCount(IopOtherTransfer, 437 (ULONG)Irp->IoStatus.Information); 438 } 439 } 440 441 /* Now that we've signaled the events, de-associate the IRP */ 442 IopUnQueueIrpFromThread(Irp); 443 444 /* Now check if a User APC Routine was requested */ 445 if (Irp->Overlay.AsynchronousParameters.UserApcRoutine) 446 { 447 /* Initialize it */ 448 KeInitializeApc(&Irp->Tail.Apc, 449 KeGetCurrentThread(), 450 CurrentApcEnvironment, 451 IopFreeIrpKernelApc, 452 IopAbortIrpKernelApc, 453 (PKNORMAL_ROUTINE)Irp-> 454 Overlay.AsynchronousParameters.UserApcRoutine, 455 Irp->RequestorMode, 456 Irp-> 457 Overlay.AsynchronousParameters.UserApcContext); 458 459 /* Queue it */ 460 KeInsertQueueApc(&Irp->Tail.Apc, Irp->UserIosb, NULL, 2); 461 } 462 else if ((Port) && 463 (Irp->Overlay.AsynchronousParameters.UserApcContext)) 464 { 465 /* We have an I/O Completion setup... create the special Overlay */ 466 Irp->Tail.CompletionKey = Key; 467 Irp->Tail.Overlay.PacketType = IopCompletionPacketIrp; 468 KeInsertQueue(Port, &Irp->Tail.Overlay.ListEntry); 469 } 470 else 471 { 472 /* Free the IRP since we don't need it anymore */ 473 IoFreeIrp(Irp); 474 } 475 476 /* Check if we have a file object that wasn't part of a create */ 477 if ((FileObject) && !(SignaledCreateRequest)) 478 { 479 /* Dereference it, since it's not needed anymore either */ 480 ObDereferenceObjectDeferDelete(FileObject); 481 } 482 } 483 else 484 { 485 /* 486 * Either we didn't return from the request, or we did return but this 487 * request was synchronous. 488 */ 489 if ((Irp->PendingReturned) && (FileObject)) 490 { 491 /* So we did return with a synch operation, was it the IRP? */ 492 if (Irp->Flags & IRP_SYNCHRONOUS_API) 493 { 494 /* Yes, this IRP was synchronous, so return the I/O Status */ 495 *Irp->UserIosb = Irp->IoStatus; 496 497 /* Now check if the user gave an event */ 498 if (Irp->UserEvent) 499 { 500 /* Signal it */ 501 KeSetEvent(Irp->UserEvent, 0, FALSE); 502 } 503 else 504 { 505 /* No event was given, so signal the FO instead */ 506 KeSetEvent(&FileObject->Event, 0, FALSE); 507 } 508 } 509 else 510 { 511 /* 512 * It's not the IRP that was synchronous, it was the FO 513 * that was opened this way. Signal its event. 514 */ 515 FileObject->FinalStatus = Irp->IoStatus.Status; 516 KeSetEvent(&FileObject->Event, 0, FALSE); 517 } 518 } 519 520 /* Now that we got here, we do this for incomplete I/Os as well */ 521 if ((FileObject) && !(Irp->Flags & IRP_CREATE_OPERATION)) 522 { 523 /* Dereference the File Object unless this was a create */ 524 ObDereferenceObjectDeferDelete(FileObject); 525 } 526 527 /* 528 * Check if this was an Executive Event (remember that we know this 529 * by checking if the IRP is synchronous) 530 */ 531 if ((Irp->UserEvent) && 532 (FileObject) && 533 !(Irp->Flags & IRP_SYNCHRONOUS_API)) 534 { 535 /* This isn't a PKEVENT, so dereference it */ 536 ObDereferenceObject(Irp->UserEvent); 537 } 538 539 /* Now that we've signaled the events, de-associate the IRP */ 540 IopUnQueueIrpFromThread(Irp); 541 542 /* Free the IRP as well */ 543 IoFreeIrp(Irp); 544 } 545 } 546 547 BOOLEAN 548 NTAPI 549 IopInitializeReserveIrp(IN PRESERVE_IRP_ALLOCATOR ReserveIrpAllocator) 550 { 551 /* Our allocated stack size */ 552 ReserveIrpAllocator->StackSize = 20; 553 554 /* Allocate the IRP now */ 555 ReserveIrpAllocator->ReserveIrp = IoAllocateIrp(ReserveIrpAllocator->StackSize, FALSE); 556 /* If we cannot, abort system boot */ 557 if (ReserveIrpAllocator->ReserveIrp == NULL) 558 { 559 return FALSE; 560 } 561 562 /* It's not in use */ 563 ReserveIrpAllocator->ReserveIrpInUse = 0; 564 /* And init the event */ 565 KeInitializeEvent(&ReserveIrpAllocator->WaitEvent, SynchronizationEvent, FALSE); 566 567 /* All good, keep booting */ 568 return TRUE; 569 } 570 571 PIRP 572 NTAPI 573 IopAllocateReserveIrp(IN CCHAR StackSize) 574 { 575 /* If we need a stack size higher than what was allocated, then fail */ 576 if (StackSize > IopReserveIrpAllocator.StackSize) 577 { 578 return NULL; 579 } 580 581 /* Now, wait until the IRP becomes available and reserve it immediately */ 582 while (InterlockedExchange(&IopReserveIrpAllocator.ReserveIrpInUse, 1) == 1) 583 { 584 KeWaitForSingleObject(&IopReserveIrpAllocator.WaitEvent, 585 Executive, 586 KernelMode, 587 FALSE, 588 NULL); 589 } 590 591 /* It's ours! Initialize it */ 592 IoInitializeIrp(IopReserveIrpAllocator.ReserveIrp, IoSizeOfIrp(StackSize), StackSize); 593 594 /* And return it to the caller */ 595 return IopReserveIrpAllocator.ReserveIrp; 596 } 597 598 VOID 599 IopFreeReserveIrp(IN CCHAR PriorityBoost) 600 { 601 /* Mark we don't use the IRP anymore */ 602 InterlockedExchange(&IopReserveIrpAllocator.ReserveIrpInUse, 0); 603 604 /* And set the event if someone is waiting on the IRP */ 605 KeSetEvent(&IopReserveIrpAllocator.WaitEvent, PriorityBoost, FALSE); 606 } 607 608 /* FUNCTIONS *****************************************************************/ 609 610 /* 611 * @implemented 612 */ 613 PIRP 614 NTAPI 615 IoAllocateIrp(IN CCHAR StackSize, 616 IN BOOLEAN ChargeQuota) 617 { 618 PIRP Irp = NULL; 619 USHORT Size = IoSizeOfIrp(StackSize); 620 PKPRCB Prcb; 621 UCHAR Flags = 0; 622 PNPAGED_LOOKASIDE_LIST List = NULL; 623 PP_NPAGED_LOOKASIDE_NUMBER ListType = LookasideSmallIrpList; 624 625 /* Set Charge Quota Flag */ 626 if (ChargeQuota) Flags |= IRP_QUOTA_CHARGED; 627 628 /* Get the PRCB */ 629 Prcb = KeGetCurrentPrcb(); 630 631 /* Figure out which Lookaside List to use */ 632 if ((StackSize <= 8) && (ChargeQuota == FALSE || Prcb->LookasideIrpFloat > 0)) 633 { 634 /* Set Fixed Size Flag */ 635 Flags |= IRP_ALLOCATED_FIXED_SIZE; 636 637 /* See if we should use big list */ 638 if (StackSize != 1) 639 { 640 Size = IoSizeOfIrp(8); 641 ListType = LookasideLargeIrpList; 642 } 643 644 /* Get the P List First */ 645 List = (PNPAGED_LOOKASIDE_LIST)Prcb->PPLookasideList[ListType].P; 646 647 /* Attempt allocation */ 648 List->L.TotalAllocates++; 649 Irp = (PIRP)InterlockedPopEntrySList(&List->L.ListHead); 650 651 /* Check if the P List failed */ 652 if (!Irp) 653 { 654 /* Let the balancer know */ 655 List->L.AllocateMisses++; 656 657 /* Try the L List */ 658 List = (PNPAGED_LOOKASIDE_LIST)Prcb->PPLookasideList[ListType].L; 659 List->L.TotalAllocates++; 660 Irp = (PIRP)InterlockedPopEntrySList(&List->L.ListHead); 661 } 662 } 663 664 /* Check if we have to use the pool */ 665 if (!Irp) 666 { 667 /* Did we try lookaside and fail? */ 668 if (Flags & IRP_ALLOCATED_FIXED_SIZE) List->L.AllocateMisses++; 669 670 /* Check if we should charge quota */ 671 if (ChargeQuota) 672 { 673 Irp = ExAllocatePoolWithQuotaTag(NonPagedPool | POOL_QUOTA_FAIL_INSTEAD_OF_RAISE, 674 Size, 675 TAG_IRP); 676 } 677 else 678 { 679 /* Allocate the IRP with no quota charge */ 680 Irp = ExAllocatePoolWithTag(NonPagedPool, Size, TAG_IRP); 681 } 682 683 /* Make sure it was sucessful */ 684 if (!Irp) return NULL; 685 } 686 else if (Flags & IRP_QUOTA_CHARGED) 687 { 688 /* Decrement lookaside float */ 689 InterlockedDecrement(&Prcb->LookasideIrpFloat); 690 Flags |= IRP_LOOKASIDE_ALLOCATION; 691 692 /* In this case there is no charge quota */ 693 Flags &= ~IRP_QUOTA_CHARGED; 694 } 695 696 /* Now Initialize it */ 697 IoInitializeIrp(Irp, Size, StackSize); 698 699 /* Set the Allocation Flags */ 700 Irp->AllocationFlags = Flags; 701 702 /* Return it */ 703 IOTRACE(IO_IRP_DEBUG, 704 "%s - Allocated IRP %p with allocation flags %lx\n", 705 __FUNCTION__, 706 Irp, 707 Flags); 708 return Irp; 709 } 710 711 /* 712 * @implemented 713 */ 714 PIRP 715 NTAPI 716 IopAllocateIrpMustSucceed(IN CCHAR StackSize) 717 { 718 LONG i; 719 PIRP Irp; 720 LARGE_INTEGER Sleep; 721 722 /* Try to get an IRP */ 723 Irp = IoAllocateIrp(StackSize, FALSE); 724 if (Irp) 725 return Irp; 726 727 /* If we fail, start looping till we may get one */ 728 i = LONG_MAX; 729 do { 730 i--; 731 732 /* First, sleep for 10ms */ 733 Sleep.QuadPart = -10 * 1000 * 10; 734 KeDelayExecutionThread(KernelMode, FALSE, &Sleep); 735 736 /* Then, retry allocation */ 737 Irp = IoAllocateIrp(StackSize, FALSE); 738 if (Irp) 739 return Irp; 740 } while (i > 0); 741 742 return Irp; 743 } 744 745 /* 746 * @implemented 747 */ 748 PIRP 749 NTAPI 750 IoBuildAsynchronousFsdRequest(IN ULONG MajorFunction, 751 IN PDEVICE_OBJECT DeviceObject, 752 IN PVOID Buffer, 753 IN ULONG Length, 754 IN PLARGE_INTEGER StartingOffset, 755 IN PIO_STATUS_BLOCK IoStatusBlock) 756 { 757 PIRP Irp; 758 PIO_STACK_LOCATION StackPtr; 759 760 /* Allocate IRP */ 761 Irp = IoAllocateIrp(DeviceObject->StackSize, FALSE); 762 if (!Irp) return NULL; 763 764 /* Get the Stack */ 765 StackPtr = IoGetNextIrpStackLocation(Irp); 766 767 /* Write the Major function and then deal with it */ 768 StackPtr->MajorFunction = (UCHAR)MajorFunction; 769 770 /* Do not handle the following here */ 771 if ((MajorFunction != IRP_MJ_FLUSH_BUFFERS) && 772 (MajorFunction != IRP_MJ_SHUTDOWN) && 773 (MajorFunction != IRP_MJ_PNP) && 774 (MajorFunction != IRP_MJ_POWER)) 775 { 776 /* Check if this is Buffered IO */ 777 if (DeviceObject->Flags & DO_BUFFERED_IO) 778 { 779 /* Allocate the System Buffer */ 780 Irp->AssociatedIrp.SystemBuffer = 781 ExAllocatePoolWithTag(NonPagedPool, Length, TAG_SYS_BUF); 782 if (!Irp->AssociatedIrp.SystemBuffer) 783 { 784 /* Free the IRP and fail */ 785 IoFreeIrp(Irp); 786 return NULL; 787 } 788 789 /* Set flags */ 790 Irp->Flags = IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER; 791 792 /* Handle special IRP_MJ_WRITE Case */ 793 if (MajorFunction == IRP_MJ_WRITE) 794 { 795 /* Copy the buffer data */ 796 RtlCopyMemory(Irp->AssociatedIrp.SystemBuffer, Buffer, Length); 797 } 798 else 799 { 800 /* Set the Input Operation flag and set this as a User Buffer */ 801 Irp->Flags |= IRP_INPUT_OPERATION; 802 Irp->UserBuffer = Buffer; 803 } 804 } 805 else if (DeviceObject->Flags & DO_DIRECT_IO) 806 { 807 /* Use an MDL for Direct I/O */ 808 Irp->MdlAddress = IoAllocateMdl(Buffer, 809 Length, 810 FALSE, 811 FALSE, 812 NULL); 813 if (!Irp->MdlAddress) 814 { 815 /* Free the IRP and fail */ 816 IoFreeIrp(Irp); 817 return NULL; 818 } 819 820 /* Probe and Lock */ 821 _SEH2_TRY 822 { 823 /* Do the probe */ 824 MmProbeAndLockPages(Irp->MdlAddress, 825 KernelMode, 826 MajorFunction == IRP_MJ_READ ? 827 IoWriteAccess : IoReadAccess); 828 } 829 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 830 { 831 /* Free the IRP and its MDL */ 832 IoFreeMdl(Irp->MdlAddress); 833 IoFreeIrp(Irp); 834 835 /* Fail */ 836 _SEH2_YIELD(return NULL); 837 } 838 _SEH2_END; 839 } 840 else 841 { 842 /* Neither, use the buffer */ 843 Irp->UserBuffer = Buffer; 844 } 845 846 /* Check if this is a read */ 847 if (MajorFunction == IRP_MJ_READ) 848 { 849 /* Set the parameters for a read */ 850 StackPtr->Parameters.Read.Length = Length; 851 StackPtr->Parameters.Read.ByteOffset = *StartingOffset; 852 } 853 else if (MajorFunction == IRP_MJ_WRITE) 854 { 855 /* Otherwise, set write parameters */ 856 StackPtr->Parameters.Write.Length = Length; 857 StackPtr->Parameters.Write.ByteOffset = *StartingOffset; 858 } 859 } 860 861 /* Set the Current Thread and IOSB */ 862 Irp->UserIosb = IoStatusBlock; 863 Irp->Tail.Overlay.Thread = PsGetCurrentThread(); 864 865 /* Return the IRP */ 866 IOTRACE(IO_IRP_DEBUG, 867 "%s - Built IRP %p with Major, Buffer, DO %lx %p %p\n", 868 __FUNCTION__, 869 Irp, 870 MajorFunction, 871 Buffer, 872 DeviceObject); 873 return Irp; 874 } 875 876 /* 877 * @implemented 878 */ 879 PIRP 880 NTAPI 881 IoBuildDeviceIoControlRequest(IN ULONG IoControlCode, 882 IN PDEVICE_OBJECT DeviceObject, 883 IN PVOID InputBuffer, 884 IN ULONG InputBufferLength, 885 IN PVOID OutputBuffer, 886 IN ULONG OutputBufferLength, 887 IN BOOLEAN InternalDeviceIoControl, 888 IN PKEVENT Event, 889 IN PIO_STATUS_BLOCK IoStatusBlock) 890 { 891 PIRP Irp; 892 PIO_STACK_LOCATION StackPtr; 893 ULONG BufferLength; 894 895 /* Allocate IRP */ 896 Irp = IoAllocateIrp(DeviceObject->StackSize, FALSE); 897 if (!Irp) return NULL; 898 899 /* Get the Stack */ 900 StackPtr = IoGetNextIrpStackLocation(Irp); 901 902 /* Set the DevCtl Type */ 903 StackPtr->MajorFunction = InternalDeviceIoControl ? 904 IRP_MJ_INTERNAL_DEVICE_CONTROL : 905 IRP_MJ_DEVICE_CONTROL; 906 907 /* Set the IOCTL Data */ 908 StackPtr->Parameters.DeviceIoControl.IoControlCode = IoControlCode; 909 StackPtr->Parameters.DeviceIoControl.InputBufferLength = InputBufferLength; 910 StackPtr->Parameters.DeviceIoControl.OutputBufferLength = 911 OutputBufferLength; 912 913 /* Handle the Methods */ 914 switch (IO_METHOD_FROM_CTL_CODE(IoControlCode)) 915 { 916 /* Buffered I/O */ 917 case METHOD_BUFFERED: 918 919 /* Select the right Buffer Length */ 920 BufferLength = InputBufferLength > OutputBufferLength ? 921 InputBufferLength : OutputBufferLength; 922 923 /* Make sure there is one */ 924 if (BufferLength) 925 { 926 /* Allocate the System Buffer */ 927 Irp->AssociatedIrp.SystemBuffer = 928 ExAllocatePoolWithTag(NonPagedPool, 929 BufferLength, 930 TAG_SYS_BUF); 931 if (!Irp->AssociatedIrp.SystemBuffer) 932 { 933 /* Free the IRP and fail */ 934 IoFreeIrp(Irp); 935 return NULL; 936 } 937 938 /* Check if we got a buffer */ 939 if (InputBuffer) 940 { 941 /* Copy into the System Buffer */ 942 RtlCopyMemory(Irp->AssociatedIrp.SystemBuffer, 943 InputBuffer, 944 InputBufferLength); 945 } 946 947 /* Write the flags */ 948 Irp->Flags = IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER; 949 if (OutputBuffer) Irp->Flags |= IRP_INPUT_OPERATION; 950 951 /* Save the Buffer */ 952 Irp->UserBuffer = OutputBuffer; 953 } 954 else 955 { 956 /* Clear the Flags and Buffer */ 957 Irp->Flags = 0; 958 Irp->UserBuffer = NULL; 959 } 960 break; 961 962 /* Direct I/O */ 963 case METHOD_IN_DIRECT: 964 case METHOD_OUT_DIRECT: 965 966 /* Check if we got an input buffer */ 967 if (InputBuffer) 968 { 969 /* Allocate the System Buffer */ 970 Irp->AssociatedIrp.SystemBuffer = 971 ExAllocatePoolWithTag(NonPagedPool, 972 InputBufferLength, 973 TAG_SYS_BUF); 974 if (!Irp->AssociatedIrp.SystemBuffer) 975 { 976 /* Free the IRP and fail */ 977 IoFreeIrp(Irp); 978 return NULL; 979 } 980 981 /* Copy into the System Buffer */ 982 RtlCopyMemory(Irp->AssociatedIrp.SystemBuffer, 983 InputBuffer, 984 InputBufferLength); 985 986 /* Write the flags */ 987 Irp->Flags = IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER; 988 } 989 else 990 { 991 /* Clear the flags */ 992 Irp->Flags = 0; 993 } 994 995 /* Check if we got an output buffer */ 996 if (OutputBuffer) 997 { 998 /* Allocate the System Buffer */ 999 Irp->MdlAddress = IoAllocateMdl(OutputBuffer, 1000 OutputBufferLength, 1001 FALSE, 1002 FALSE, 1003 Irp); 1004 if (!Irp->MdlAddress) 1005 { 1006 /* Free the IRP and fail */ 1007 IoFreeIrp(Irp); 1008 return NULL; 1009 } 1010 1011 /* Probe and Lock */ 1012 _SEH2_TRY 1013 { 1014 /* Do the probe */ 1015 MmProbeAndLockPages(Irp->MdlAddress, 1016 KernelMode, 1017 IO_METHOD_FROM_CTL_CODE(IoControlCode) == 1018 METHOD_IN_DIRECT ? 1019 IoReadAccess : IoWriteAccess); 1020 } 1021 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1022 { 1023 /* Free the MDL */ 1024 IoFreeMdl(Irp->MdlAddress); 1025 1026 /* Free the input buffer and IRP */ 1027 if (InputBuffer) ExFreePool(Irp->AssociatedIrp.SystemBuffer); 1028 IoFreeIrp(Irp); 1029 1030 /* Fail */ 1031 _SEH2_YIELD(return NULL); 1032 } 1033 _SEH2_END; 1034 } 1035 break; 1036 1037 case METHOD_NEITHER: 1038 1039 /* Just save the Buffer */ 1040 Irp->UserBuffer = OutputBuffer; 1041 StackPtr->Parameters.DeviceIoControl.Type3InputBuffer = InputBuffer; 1042 } 1043 1044 /* Now write the Event and IoSB */ 1045 Irp->UserIosb = IoStatusBlock; 1046 Irp->UserEvent = Event; 1047 1048 /* Sync IRPs are queued to requestor thread's irp cancel/cleanup list */ 1049 Irp->Tail.Overlay.Thread = PsGetCurrentThread(); 1050 IoQueueThreadIrp(Irp); 1051 1052 /* Return the IRP */ 1053 IOTRACE(IO_IRP_DEBUG, 1054 "%s - Built IRP %p with IOCTL, Buffers, DO %lx %p %p %p\n", 1055 __FUNCTION__, 1056 Irp, 1057 IoControlCode, 1058 InputBuffer, 1059 OutputBuffer, 1060 DeviceObject); 1061 return Irp; 1062 } 1063 1064 /* 1065 * @implemented 1066 */ 1067 PIRP 1068 NTAPI 1069 IoBuildSynchronousFsdRequest(IN ULONG MajorFunction, 1070 IN PDEVICE_OBJECT DeviceObject, 1071 IN PVOID Buffer, 1072 IN ULONG Length, 1073 IN PLARGE_INTEGER StartingOffset, 1074 IN PKEVENT Event, 1075 IN PIO_STATUS_BLOCK IoStatusBlock) 1076 { 1077 PIRP Irp; 1078 1079 /* Do the big work to set up the IRP */ 1080 Irp = IoBuildAsynchronousFsdRequest(MajorFunction, 1081 DeviceObject, 1082 Buffer, 1083 Length, 1084 StartingOffset, 1085 IoStatusBlock ); 1086 if (!Irp) return NULL; 1087 1088 /* Set the Event which makes it Syncronous */ 1089 Irp->UserEvent = Event; 1090 1091 /* Sync IRPs are queued to requestor thread's irp cancel/cleanup list */ 1092 IoQueueThreadIrp(Irp); 1093 return Irp; 1094 } 1095 1096 /* 1097 * @implemented 1098 */ 1099 BOOLEAN 1100 NTAPI 1101 IoCancelIrp(IN PIRP Irp) 1102 { 1103 KIRQL OldIrql; 1104 PDRIVER_CANCEL CancelRoutine; 1105 IOTRACE(IO_IRP_DEBUG, 1106 "%s - Canceling IRP %p\n", 1107 __FUNCTION__, 1108 Irp); 1109 ASSERT(Irp->Type == IO_TYPE_IRP); 1110 1111 /* Acquire the cancel lock and cancel the IRP */ 1112 IoAcquireCancelSpinLock(&OldIrql); 1113 Irp->Cancel = TRUE; 1114 1115 /* Clear the cancel routine and get the old one */ 1116 CancelRoutine = IoSetCancelRoutine(Irp, NULL); 1117 if (CancelRoutine) 1118 { 1119 /* We had a routine, make sure the IRP isn't completed */ 1120 if (Irp->CurrentLocation > (Irp->StackCount + 1)) 1121 { 1122 /* It is, bugcheck */ 1123 KeBugCheckEx(CANCEL_STATE_IN_COMPLETED_IRP, 1124 (ULONG_PTR)Irp, 1125 (ULONG_PTR)CancelRoutine, 1126 0, 1127 0); 1128 } 1129 1130 /* Set the cancel IRQL And call the routine */ 1131 Irp->CancelIrql = OldIrql; 1132 CancelRoutine(IoGetCurrentIrpStackLocation(Irp)->DeviceObject, Irp); 1133 return TRUE; 1134 } 1135 1136 /* Otherwise, release the cancel lock and fail */ 1137 IoReleaseCancelSpinLock(OldIrql); 1138 return FALSE; 1139 } 1140 1141 /* 1142 * @implemented 1143 */ 1144 VOID 1145 NTAPI 1146 IoCancelThreadIo(IN PETHREAD Thread) 1147 { 1148 KIRQL OldIrql; 1149 ULONG Retries = 3000; 1150 LARGE_INTEGER Interval; 1151 PLIST_ENTRY ListHead, NextEntry; 1152 PIRP Irp; 1153 PAGED_CODE(); 1154 1155 /* Windows isn't using given thread, but using current. */ 1156 Thread = PsGetCurrentThread(); 1157 1158 IOTRACE(IO_IRP_DEBUG, 1159 "%s - Canceling IRPs for Thread %p\n", 1160 __FUNCTION__, 1161 Thread); 1162 1163 /* Raise to APC to protect the IrpList */ 1164 KeRaiseIrql(APC_LEVEL, &OldIrql); 1165 1166 /* Start by cancelling all the IRPs in the current thread queue. */ 1167 ListHead = &Thread->IrpList; 1168 NextEntry = ListHead->Flink; 1169 while (ListHead != NextEntry) 1170 { 1171 /* Get the IRP */ 1172 Irp = CONTAINING_RECORD(NextEntry, IRP, ThreadListEntry); 1173 1174 /* Cancel it */ 1175 IoCancelIrp(Irp); 1176 1177 /* Move to the next entry */ 1178 NextEntry = NextEntry->Flink; 1179 } 1180 1181 /* Wait 100 milliseconds */ 1182 Interval.QuadPart = -1000000; 1183 1184 /* Wait till all the IRPs are completed or cancelled. */ 1185 while (!IsListEmpty(&Thread->IrpList)) 1186 { 1187 /* Now we can lower */ 1188 KeLowerIrql(OldIrql); 1189 1190 /* Wait a short while and then look if all our IRPs were completed. */ 1191 KeDelayExecutionThread(KernelMode, FALSE, &Interval); 1192 1193 /* 1194 * Don't stay here forever if some broken driver doesn't complete 1195 * the IRP. 1196 */ 1197 if (!(Retries--)) 1198 { 1199 /* Print out a message and remove the IRP */ 1200 DPRINT1("Broken driver did not complete!\n"); 1201 IopDisassociateThreadIrp(); 1202 } 1203 1204 /* Raise the IRQL Again */ 1205 KeRaiseIrql(APC_LEVEL, &OldIrql); 1206 } 1207 1208 /* We're done, lower the IRQL */ 1209 KeLowerIrql(OldIrql); 1210 } 1211 1212 /* 1213 * @implemented 1214 */ 1215 #undef IoCallDriver 1216 NTSTATUS 1217 NTAPI 1218 IoCallDriver(IN PDEVICE_OBJECT DeviceObject, 1219 IN PIRP Irp) 1220 { 1221 /* Call fastcall */ 1222 return IofCallDriver(DeviceObject, Irp); 1223 } 1224 1225 #define IoCallDriver IofCallDriver 1226 1227 /* 1228 * @implemented 1229 */ 1230 #undef IoCompleteRequest 1231 VOID 1232 NTAPI 1233 IoCompleteRequest(IN PIRP Irp, 1234 IN CCHAR PriorityBoost) 1235 { 1236 /* Call the fastcall */ 1237 IofCompleteRequest(Irp, PriorityBoost); 1238 } 1239 1240 #define IoCompleteRequest IofCompleteRequest 1241 1242 /* 1243 * @implemented 1244 */ 1245 VOID 1246 NTAPI 1247 IoEnqueueIrp(IN PIRP Irp) 1248 { 1249 /* This is the same as calling IoQueueThreadIrp */ 1250 IoQueueThreadIrp(Irp); 1251 } 1252 1253 /* 1254 * @implemented 1255 */ 1256 NTSTATUS 1257 FASTCALL 1258 IofCallDriver(IN PDEVICE_OBJECT DeviceObject, 1259 IN PIRP Irp) 1260 { 1261 PDRIVER_OBJECT DriverObject; 1262 PIO_STACK_LOCATION StackPtr; 1263 1264 /* Make sure this is a valid IRP */ 1265 ASSERT(Irp->Type == IO_TYPE_IRP); 1266 1267 /* Get the Driver Object */ 1268 DriverObject = DeviceObject->DriverObject; 1269 1270 /* Decrease the current location and check if */ 1271 Irp->CurrentLocation--; 1272 if (Irp->CurrentLocation <= 0) 1273 { 1274 /* This IRP ran out of stack, bugcheck */ 1275 KeBugCheckEx(NO_MORE_IRP_STACK_LOCATIONS, (ULONG_PTR)Irp, 0, 0, 0); 1276 } 1277 1278 /* Now update the stack location */ 1279 StackPtr = IoGetNextIrpStackLocation(Irp); 1280 Irp->Tail.Overlay.CurrentStackLocation = StackPtr; 1281 1282 /* Get the Device Object */ 1283 StackPtr->DeviceObject = DeviceObject; 1284 1285 /* Call it */ 1286 return DriverObject->MajorFunction[StackPtr->MajorFunction](DeviceObject, 1287 Irp); 1288 } 1289 1290 FORCEINLINE 1291 VOID 1292 IopClearStackLocation(IN PIO_STACK_LOCATION IoStackLocation) 1293 { 1294 IoStackLocation->MinorFunction = 0; 1295 IoStackLocation->Flags = 0; 1296 IoStackLocation->Control &= SL_ERROR_RETURNED; 1297 IoStackLocation->Parameters.Others.Argument1 = 0; 1298 IoStackLocation->Parameters.Others.Argument2 = 0; 1299 IoStackLocation->Parameters.Others.Argument3 = 0; 1300 IoStackLocation->FileObject = NULL; 1301 } 1302 1303 /* 1304 * @implemented 1305 */ 1306 VOID 1307 FASTCALL 1308 IofCompleteRequest(IN PIRP Irp, 1309 IN CCHAR PriorityBoost) 1310 { 1311 PIO_STACK_LOCATION StackPtr, LastStackPtr; 1312 PDEVICE_OBJECT DeviceObject; 1313 PFILE_OBJECT FileObject; 1314 PETHREAD Thread; 1315 NTSTATUS Status; 1316 PMDL Mdl, NextMdl; 1317 ULONG MasterCount; 1318 PIRP MasterIrp; 1319 ULONG Flags; 1320 NTSTATUS ErrorCode = STATUS_SUCCESS; 1321 PREPARSE_DATA_BUFFER DataBuffer = NULL; 1322 IOTRACE(IO_IRP_DEBUG, 1323 "%s - Completing IRP %p\n", 1324 __FUNCTION__, 1325 Irp); 1326 1327 /* Make sure this IRP isn't getting completed twice or is invalid */ 1328 if ((Irp->CurrentLocation) > (Irp->StackCount + 1)) 1329 { 1330 /* Bugcheck */ 1331 KeBugCheckEx(MULTIPLE_IRP_COMPLETE_REQUESTS, (ULONG_PTR)Irp, 0, 0, 0); 1332 } 1333 1334 /* Some sanity checks */ 1335 ASSERT(Irp->Type == IO_TYPE_IRP); 1336 ASSERT(!Irp->CancelRoutine); 1337 ASSERT(Irp->IoStatus.Status != STATUS_PENDING); 1338 ASSERT(Irp->IoStatus.Status != (NTSTATUS)0xFFFFFFFF); 1339 1340 /* Get the last stack */ 1341 LastStackPtr = (PIO_STACK_LOCATION)(Irp + 1); 1342 if (LastStackPtr->Control & SL_ERROR_RETURNED) 1343 { 1344 /* Get the error code */ 1345 ErrorCode = PtrToUlong(LastStackPtr->Parameters.Others.Argument4); 1346 } 1347 1348 /* 1349 * Start the loop with the current stack and point the IRP to the next stack 1350 * and then keep incrementing the stack as we loop through. The IRP should 1351 * always point to the next stack location w.r.t the one currently being 1352 * analyzed, so completion routine code will see the appropriate value. 1353 * Because of this, we must loop until the current stack location is +1 of 1354 * the stack count, because when StackPtr is at the end, CurrentLocation is +1. 1355 */ 1356 for (StackPtr = IoGetCurrentIrpStackLocation(Irp), 1357 Irp->CurrentLocation++, 1358 Irp->Tail.Overlay.CurrentStackLocation++; 1359 Irp->CurrentLocation <= (Irp->StackCount + 1); 1360 StackPtr++, 1361 Irp->CurrentLocation++, 1362 Irp->Tail.Overlay.CurrentStackLocation++) 1363 { 1364 /* Set Pending Returned */ 1365 Irp->PendingReturned = StackPtr->Control & SL_PENDING_RETURNED; 1366 1367 /* Check if we failed */ 1368 if (!NT_SUCCESS(Irp->IoStatus.Status)) 1369 { 1370 /* Check if it was changed by a completion routine */ 1371 if (Irp->IoStatus.Status != ErrorCode) 1372 { 1373 /* Update the error for the current stack */ 1374 ErrorCode = Irp->IoStatus.Status; 1375 StackPtr->Control |= SL_ERROR_RETURNED; 1376 LastStackPtr->Parameters.Others.Argument4 = UlongToPtr(ErrorCode); 1377 LastStackPtr->Control |= SL_ERROR_RETURNED; 1378 } 1379 } 1380 1381 /* Check if there is a Completion Routine to Call */ 1382 if ((NT_SUCCESS(Irp->IoStatus.Status) && 1383 (StackPtr->Control & SL_INVOKE_ON_SUCCESS)) || 1384 (!NT_SUCCESS(Irp->IoStatus.Status) && 1385 (StackPtr->Control & SL_INVOKE_ON_ERROR)) || 1386 (Irp->Cancel && 1387 (StackPtr->Control & SL_INVOKE_ON_CANCEL))) 1388 { 1389 /* Clear the stack location */ 1390 IopClearStackLocation(StackPtr); 1391 1392 /* Check for highest-level device completion routines */ 1393 if (Irp->CurrentLocation == (Irp->StackCount + 1)) 1394 { 1395 /* Clear the DO, since the current stack location is invalid */ 1396 DeviceObject = NULL; 1397 } 1398 else 1399 { 1400 /* Otherwise, return the real one */ 1401 DeviceObject = IoGetCurrentIrpStackLocation(Irp)->DeviceObject; 1402 } 1403 1404 /* Call the completion routine */ 1405 Status = StackPtr->CompletionRoutine(DeviceObject, 1406 Irp, 1407 StackPtr->Context); 1408 1409 /* Don't touch the Packet in this case, since it might be gone! */ 1410 if (Status == STATUS_MORE_PROCESSING_REQUIRED) return; 1411 } 1412 else 1413 { 1414 /* Otherwise, check if this is a completed IRP */ 1415 if ((Irp->CurrentLocation <= Irp->StackCount) && 1416 (Irp->PendingReturned)) 1417 { 1418 /* Mark it as pending */ 1419 IoMarkIrpPending(Irp); 1420 } 1421 1422 /* Clear the stack location */ 1423 IopClearStackLocation(StackPtr); 1424 } 1425 } 1426 1427 /* Check if the IRP is an associated IRP */ 1428 if (Irp->Flags & IRP_ASSOCIATED_IRP) 1429 { 1430 /* Get the master IRP and count */ 1431 MasterIrp = Irp->AssociatedIrp.MasterIrp; 1432 MasterCount = InterlockedDecrement(&MasterIrp->AssociatedIrp.IrpCount); 1433 1434 /* Free the MDLs */ 1435 for (Mdl = Irp->MdlAddress; Mdl; Mdl = NextMdl) 1436 { 1437 /* Go to the next one */ 1438 NextMdl = Mdl->Next; 1439 IoFreeMdl(Mdl); 1440 } 1441 1442 /* Free the IRP itself */ 1443 IoFreeIrp(Irp); 1444 1445 /* Complete the Master IRP */ 1446 if (!MasterCount) IofCompleteRequest(MasterIrp, PriorityBoost); 1447 return; 1448 } 1449 1450 /* Check whether we have to reparse */ 1451 if (Irp->IoStatus.Status == STATUS_REPARSE) 1452 { 1453 if (Irp->IoStatus.Information > IO_REMOUNT) 1454 { 1455 /* If that's a reparse tag we understand, save the buffer from deletion */ 1456 if (Irp->IoStatus.Information == IO_REPARSE_TAG_MOUNT_POINT) 1457 { 1458 ASSERT(Irp->Tail.Overlay.AuxiliaryBuffer != NULL); 1459 DataBuffer = (PREPARSE_DATA_BUFFER)Irp->Tail.Overlay.AuxiliaryBuffer; 1460 Irp->Tail.Overlay.AuxiliaryBuffer = NULL; 1461 } 1462 else 1463 { 1464 Irp->IoStatus.Status = STATUS_IO_REPARSE_TAG_NOT_HANDLED; 1465 } 1466 } 1467 } 1468 1469 /* Check if we have an auxiliary buffer */ 1470 if (Irp->Tail.Overlay.AuxiliaryBuffer) 1471 { 1472 /* Free it */ 1473 ExFreePool(Irp->Tail.Overlay.AuxiliaryBuffer); 1474 Irp->Tail.Overlay.AuxiliaryBuffer = NULL; 1475 } 1476 1477 /* Check if this is a Paging I/O or Close Operation */ 1478 if (Irp->Flags & (IRP_PAGING_IO | IRP_CLOSE_OPERATION)) 1479 { 1480 /* Handle a Close Operation or Sync Paging I/O */ 1481 if (Irp->Flags & (IRP_SYNCHRONOUS_PAGING_IO | IRP_CLOSE_OPERATION)) 1482 { 1483 /* Set the I/O Status and Signal the Event */ 1484 Flags = Irp->Flags & (IRP_SYNCHRONOUS_PAGING_IO | IRP_PAGING_IO); 1485 *Irp->UserIosb = Irp->IoStatus; 1486 KeSetEvent(Irp->UserEvent, PriorityBoost, FALSE); 1487 1488 /* Free the IRP for a Paging I/O Only, Close is handled by us */ 1489 if (Flags) 1490 { 1491 /* If we were using the reserve IRP, then call the appropriate 1492 * free function (to make the IRP available again) 1493 */ 1494 if (Irp == IopReserveIrpAllocator.ReserveIrp) 1495 { 1496 IopFreeReserveIrp(PriorityBoost); 1497 } 1498 /* Otherwise, free for real! */ 1499 else 1500 { 1501 IoFreeIrp(Irp); 1502 } 1503 } 1504 } 1505 else 1506 { 1507 #if 0 1508 /* Page 166 */ 1509 KeInitializeApc(&Irp->Tail.Apc 1510 &Irp->Tail.Overlay.Thread->Tcb, 1511 Irp->ApcEnvironment, 1512 IopCompletePageWrite, 1513 NULL, 1514 NULL, 1515 KernelMode, 1516 NULL); 1517 KeInsertQueueApc(&Irp->Tail.Apc, 1518 NULL, 1519 NULL, 1520 PriorityBoost); 1521 #else 1522 /* Not implemented yet. */ 1523 UNIMPLEMENTED_DBGBREAK("Not supported!\n"); 1524 #endif 1525 } 1526 1527 /* Get out of here */ 1528 return; 1529 } 1530 1531 /* Unlock MDL Pages, page 167. */ 1532 Mdl = Irp->MdlAddress; 1533 while (Mdl) 1534 { 1535 MmUnlockPages(Mdl); 1536 Mdl = Mdl->Next; 1537 } 1538 1539 /* Check if we should exit because of a Deferred I/O (page 168) */ 1540 if ((Irp->Flags & IRP_DEFER_IO_COMPLETION) && !(Irp->PendingReturned)) 1541 { 1542 /* Restore the saved reparse buffer for the caller */ 1543 if (Irp->IoStatus.Status == STATUS_REPARSE && 1544 Irp->IoStatus.Information == IO_REPARSE_TAG_MOUNT_POINT) 1545 { 1546 Irp->Tail.Overlay.AuxiliaryBuffer = (PCHAR)DataBuffer; 1547 } 1548 1549 /* 1550 * Return without queuing the completion APC, since the caller will 1551 * take care of doing its own optimized completion at PASSIVE_LEVEL. 1552 */ 1553 return; 1554 } 1555 1556 /* Get the thread and file object */ 1557 Thread = Irp->Tail.Overlay.Thread; 1558 FileObject = Irp->Tail.Overlay.OriginalFileObject; 1559 1560 /* Make sure the IRP isn't canceled */ 1561 if (!Irp->Cancel) 1562 { 1563 /* Initialize the APC */ 1564 KeInitializeApc(&Irp->Tail.Apc, 1565 &Thread->Tcb, 1566 Irp->ApcEnvironment, 1567 IopCompleteRequest, 1568 NULL, 1569 NULL, 1570 KernelMode, 1571 NULL); 1572 1573 /* Queue it */ 1574 KeInsertQueueApc(&Irp->Tail.Apc, 1575 FileObject, 1576 DataBuffer, 1577 PriorityBoost); 1578 } 1579 else 1580 { 1581 /* The IRP just got canceled... does a thread still own it? */ 1582 if (Thread) 1583 { 1584 /* Yes! There is still hope! Initialize the APC */ 1585 KeInitializeApc(&Irp->Tail.Apc, 1586 &Thread->Tcb, 1587 Irp->ApcEnvironment, 1588 IopCompleteRequest, 1589 NULL, 1590 NULL, 1591 KernelMode, 1592 NULL); 1593 1594 /* Queue it */ 1595 KeInsertQueueApc(&Irp->Tail.Apc, 1596 FileObject, 1597 DataBuffer, 1598 PriorityBoost); 1599 } 1600 else 1601 { 1602 /* Nothing left for us to do, kill it */ 1603 ASSERT(Irp->Cancel); 1604 IopCleanupIrp(Irp, FileObject); 1605 } 1606 } 1607 } 1608 1609 NTSTATUS 1610 NTAPI 1611 IopSynchronousCompletion(IN PDEVICE_OBJECT DeviceObject, 1612 IN PIRP Irp, 1613 IN PVOID Context) 1614 { 1615 if (Irp->PendingReturned) 1616 KeSetEvent((PKEVENT)Context, IO_NO_INCREMENT, FALSE); 1617 return STATUS_MORE_PROCESSING_REQUIRED; 1618 } 1619 1620 /* 1621 * @implemented 1622 */ 1623 BOOLEAN 1624 NTAPI 1625 IoForwardIrpSynchronously(IN PDEVICE_OBJECT DeviceObject, 1626 IN PIRP Irp) 1627 { 1628 KEVENT Event; 1629 NTSTATUS Status; 1630 1631 /* Check if next stack location is available */ 1632 if (Irp->CurrentLocation > Irp->StackCount || Irp->CurrentLocation <= 1) 1633 { 1634 /* No more stack location */ 1635 return FALSE; 1636 } 1637 1638 /* Initialize event */ 1639 KeInitializeEvent(&Event, NotificationEvent, FALSE); 1640 1641 /* Copy stack location for next driver */ 1642 IoCopyCurrentIrpStackLocationToNext(Irp); 1643 1644 /* Set a completion routine, which will signal the event */ 1645 IoSetCompletionRoutine(Irp, IopSynchronousCompletion, &Event, TRUE, TRUE, TRUE); 1646 1647 /* Call next driver */ 1648 Status = IoCallDriver(DeviceObject, Irp); 1649 1650 /* Check if irp is pending */ 1651 if (Status == STATUS_PENDING) 1652 { 1653 /* Yes, wait for its completion */ 1654 KeWaitForSingleObject(&Event, Suspended, KernelMode, FALSE, NULL); 1655 } 1656 1657 /* Return success */ 1658 return TRUE; 1659 } 1660 1661 /* 1662 * @implemented 1663 */ 1664 VOID 1665 NTAPI 1666 IoFreeIrp(IN PIRP Irp) 1667 { 1668 PNPAGED_LOOKASIDE_LIST List; 1669 PP_NPAGED_LOOKASIDE_NUMBER ListType = LookasideSmallIrpList; 1670 PKPRCB Prcb; 1671 IOTRACE(IO_IRP_DEBUG, 1672 "%s - Freeing IRPs %p\n", 1673 __FUNCTION__, 1674 Irp); 1675 1676 /* Make sure the Thread IRP list is empty and that it OK to free it */ 1677 ASSERT(Irp->Type == IO_TYPE_IRP); 1678 ASSERT(IsListEmpty(&Irp->ThreadListEntry)); 1679 ASSERT(Irp->CurrentLocation >= Irp->StackCount); 1680 1681 /* Get the PRCB */ 1682 Prcb = KeGetCurrentPrcb(); 1683 1684 /* If this was a lookaside alloc, increment lookaside float */ 1685 if (Irp->AllocationFlags & IRP_LOOKASIDE_ALLOCATION) 1686 { 1687 Irp->AllocationFlags &= ~IRP_LOOKASIDE_ALLOCATION; 1688 InterlockedIncrement(&Prcb->LookasideIrpFloat); 1689 } 1690 1691 /* If this was a pool alloc, free it with the pool */ 1692 if (!(Irp->AllocationFlags & IRP_ALLOCATED_FIXED_SIZE)) 1693 { 1694 /* Free it */ 1695 ExFreePoolWithTag(Irp, TAG_IRP); 1696 } 1697 else 1698 { 1699 /* Check if this was a Big IRP */ 1700 if (Irp->StackCount != 1) ListType = LookasideLargeIrpList; 1701 1702 /* Use the P List */ 1703 List = (PNPAGED_LOOKASIDE_LIST)Prcb->PPLookasideList[ListType].P; 1704 List->L.TotalFrees++; 1705 1706 /* Check if the Free was within the Depth or not */ 1707 if (ExQueryDepthSList(&List->L.ListHead) >= List->L.Depth) 1708 { 1709 /* Let the balancer know */ 1710 List->L.FreeMisses++; 1711 1712 /* Use the L List */ 1713 List = (PNPAGED_LOOKASIDE_LIST)Prcb->PPLookasideList[ListType].L; 1714 List->L.TotalFrees++; 1715 1716 /* Check if the Free was within the Depth or not */ 1717 if (ExQueryDepthSList(&List->L.ListHead) >= List->L.Depth) 1718 { 1719 /* All lists failed, use the pool */ 1720 List->L.FreeMisses++; 1721 ExFreePoolWithTag(Irp, TAG_IRP); 1722 Irp = NULL; 1723 } 1724 } 1725 1726 /* The free was within the Depth */ 1727 if (Irp) 1728 { 1729 /* Remove the association with the process */ 1730 if (Irp->AllocationFlags & IRP_QUOTA_CHARGED) 1731 { 1732 ExReturnPoolQuota(Irp); 1733 Irp->AllocationFlags &= ~IRP_QUOTA_CHARGED; 1734 } 1735 1736 /* Add it to the lookaside list */ 1737 InterlockedPushEntrySList(&List->L.ListHead, 1738 (PSLIST_ENTRY)Irp); 1739 } 1740 } 1741 } 1742 1743 /* 1744 * @implemented 1745 */ 1746 IO_PAGING_PRIORITY 1747 FASTCALL 1748 IoGetPagingIoPriority(IN PIRP Irp) 1749 { 1750 IO_PAGING_PRIORITY Priority; 1751 ULONG Flags; 1752 1753 /* Get the flags */ 1754 Flags = Irp->Flags; 1755 1756 /* Check what priority it has */ 1757 if (Flags & IRP_CLASS_CACHE_OPERATION) 1758 { 1759 /* High priority */ 1760 Priority = IoPagingPriorityHigh; 1761 } 1762 else if (Flags & IRP_PAGING_IO) 1763 { 1764 /* Normal priority */ 1765 Priority = IoPagingPriorityNormal; 1766 } 1767 else 1768 { 1769 /* Invalid -- not a paging IRP */ 1770 Priority = IoPagingPriorityInvalid; 1771 } 1772 1773 /* Return the priority */ 1774 return Priority; 1775 } 1776 1777 /* 1778 * @implemented 1779 */ 1780 PEPROCESS 1781 NTAPI 1782 IoGetRequestorProcess(IN PIRP Irp) 1783 { 1784 /* Return the requestor process */ 1785 if (Irp->Tail.Overlay.Thread) 1786 { 1787 if (Irp->ApcEnvironment == OriginalApcEnvironment) 1788 { 1789 return Irp->Tail.Overlay.Thread->ThreadsProcess; 1790 } 1791 else if (Irp->ApcEnvironment == AttachedApcEnvironment) 1792 { 1793 return (PEPROCESS)Irp->Tail.Overlay.Thread->Tcb.ApcState.Process; 1794 } 1795 } 1796 1797 return NULL; 1798 } 1799 1800 /* 1801 * @implemented 1802 */ 1803 ULONG 1804 NTAPI 1805 IoGetRequestorProcessId(IN PIRP Irp) 1806 { 1807 PEPROCESS Process; 1808 1809 /* Return the requestor process' id */ 1810 Process = IoGetRequestorProcess(Irp); 1811 if (Process) return PtrToUlong(Process->UniqueProcessId); 1812 1813 return 0; 1814 } 1815 1816 /* 1817 * @implemented 1818 */ 1819 NTSTATUS 1820 NTAPI 1821 IoGetRequestorSessionId(IN PIRP Irp, 1822 OUT PULONG pSessionId) 1823 { 1824 PEPROCESS Process; 1825 1826 /* Return the session */ 1827 if (Irp->Tail.Overlay.Thread) 1828 { 1829 Process = Irp->Tail.Overlay.Thread->ThreadsProcess; 1830 *pSessionId = MmGetSessionId(Process); 1831 return STATUS_SUCCESS; 1832 } 1833 1834 *pSessionId = (ULONG)-1; 1835 return STATUS_UNSUCCESSFUL; 1836 } 1837 1838 /* 1839 * @implemented 1840 */ 1841 PIRP 1842 NTAPI 1843 IoGetTopLevelIrp(VOID) 1844 { 1845 /* Return the IRP */ 1846 return (PIRP)PsGetCurrentThread()->TopLevelIrp; 1847 } 1848 1849 /* 1850 * @implemented 1851 */ 1852 VOID 1853 NTAPI 1854 IoInitializeIrp(IN PIRP Irp, 1855 IN USHORT PacketSize, 1856 IN CCHAR StackSize) 1857 { 1858 /* Clear it */ 1859 IOTRACE(IO_IRP_DEBUG, 1860 "%s - Initializing IRP %p\n", 1861 __FUNCTION__, 1862 Irp); 1863 RtlZeroMemory(Irp, PacketSize); 1864 1865 /* Set the Header and other data */ 1866 Irp->Type = IO_TYPE_IRP; 1867 Irp->Size = PacketSize; 1868 Irp->StackCount = StackSize; 1869 Irp->CurrentLocation = StackSize + 1; 1870 Irp->ApcEnvironment = KeGetCurrentThread()->ApcStateIndex; 1871 Irp->Tail.Overlay.CurrentStackLocation = (PIO_STACK_LOCATION)(Irp + 1) + StackSize; 1872 1873 /* Initialize the Thread List */ 1874 InitializeListHead(&Irp->ThreadListEntry); 1875 } 1876 1877 /* 1878 * @implemented 1879 */ 1880 BOOLEAN 1881 NTAPI 1882 IoIsOperationSynchronous(IN PIRP Irp) 1883 { 1884 BOOLEAN SynchIO; 1885 BOOLEAN ForceAsync; 1886 1887 /* If the IRP requests synchronous paging I/O, if the file object was opened 1888 for synchronous I/O, if the IRP_SYNCHRONOUS_API flag is set in the IRP 1889 the operation is synchronous */ 1890 SynchIO = (IoGetCurrentIrpStackLocation(Irp)->FileObject->Flags & FO_SYNCHRONOUS_IO) || 1891 (Irp->Flags & IRP_SYNCHRONOUS_API) || (Irp->Flags & IRP_SYNCHRONOUS_PAGING_IO); 1892 1893 /* If the IRP requests asynchronous paging I/O, the operation is asynchronous, 1894 even if one of the above conditions is true */ 1895 ForceAsync = (Irp->Flags & IRP_PAGING_IO) && !(Irp->Flags & IRP_SYNCHRONOUS_PAGING_IO); 1896 1897 /* Check the flags */ 1898 if (SynchIO && !ForceAsync) 1899 { 1900 /* Synch API or Paging I/O is OK, as is Sync File I/O */ 1901 return TRUE; 1902 } 1903 1904 /* Otherwise, it is an asynchronous operation. */ 1905 return FALSE; 1906 } 1907 1908 /* 1909 * @unimplemented 1910 */ 1911 BOOLEAN 1912 NTAPI 1913 IoIsValidNameGraftingBuffer(IN PIRP Irp, 1914 IN PREPARSE_DATA_BUFFER ReparseBuffer) 1915 { 1916 UNIMPLEMENTED; 1917 return FALSE; 1918 } 1919 1920 /* 1921 * @implemented 1922 */ 1923 PIRP 1924 NTAPI 1925 IoMakeAssociatedIrp(IN PIRP Irp, 1926 IN CCHAR StackSize) 1927 { 1928 PIRP AssocIrp; 1929 IOTRACE(IO_IRP_DEBUG, 1930 "%s - Associating IRP %p\n", 1931 __FUNCTION__, 1932 Irp); 1933 1934 /* Allocate the IRP */ 1935 AssocIrp = IoAllocateIrp(StackSize, FALSE); 1936 if (!AssocIrp) return NULL; 1937 1938 /* Set the Flags */ 1939 AssocIrp->Flags |= IRP_ASSOCIATED_IRP; 1940 1941 /* Set the Thread */ 1942 AssocIrp->Tail.Overlay.Thread = Irp->Tail.Overlay.Thread; 1943 1944 /* Associate them */ 1945 AssocIrp->AssociatedIrp.MasterIrp = Irp; 1946 return AssocIrp; 1947 } 1948 1949 /* 1950 * @implemented 1951 */ 1952 VOID 1953 NTAPI 1954 IoQueueThreadIrp(IN PIRP Irp) 1955 { 1956 IOTRACE(IO_IRP_DEBUG, 1957 "%s - Queueing IRP %p\n", 1958 __FUNCTION__, 1959 Irp); 1960 1961 /* Use our inlined routine */ 1962 IopQueueIrpToThread(Irp); 1963 } 1964 1965 /* 1966 * @implemented 1967 * Reference: Chris Cant's "Writing WDM Device Drivers" 1968 */ 1969 VOID 1970 NTAPI 1971 IoReuseIrp(IN OUT PIRP Irp, 1972 IN NTSTATUS Status) 1973 { 1974 UCHAR AllocationFlags; 1975 IOTRACE(IO_IRP_DEBUG, 1976 "%s - Reusing IRP %p\n", 1977 __FUNCTION__, 1978 Irp); 1979 1980 /* Make sure it's OK to reuse it */ 1981 ASSERT(!Irp->CancelRoutine); 1982 ASSERT(IsListEmpty(&Irp->ThreadListEntry)); 1983 1984 /* Get the old flags */ 1985 AllocationFlags = Irp->AllocationFlags; 1986 1987 /* Reinitialize the IRP */ 1988 IoInitializeIrp(Irp, Irp->Size, Irp->StackCount); 1989 1990 /* Duplicate the data */ 1991 Irp->IoStatus.Status = Status; 1992 Irp->AllocationFlags = AllocationFlags; 1993 } 1994 1995 /* 1996 * @implemented 1997 */ 1998 VOID 1999 NTAPI 2000 IoSetTopLevelIrp(IN PIRP Irp) 2001 { 2002 /* Set the IRP */ 2003 PsGetCurrentThread()->TopLevelIrp = (ULONG_PTR)Irp; 2004 } 2005 2006 #if defined (_WIN64) 2007 BOOLEAN 2008 NTAPI 2009 IoIs32bitProcess( 2010 IN PIRP Irp OPTIONAL) 2011 { 2012 UNIMPLEMENTED; 2013 return FALSE; 2014 } 2015 #endif 2016