1 /* 2 * PROJECT: ReactOS Kernel 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: ntoskrnl/fsrtl/filelock.c 5 * PURPOSE: File Locking implementation for File System Drivers 6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org) 7 */ 8 9 /* INCLUDES ******************************************************************/ 10 11 #include <ntoskrnl.h> 12 #define NDEBUG 13 #include <debug.h> 14 15 /* GLOBALS *******************************************************************/ 16 17 PAGED_LOOKASIDE_LIST FsRtlFileLockLookasideList; 18 19 /* Note: this aligns the two types of lock entry structs so we can access the 20 FILE_LOCK_INFO part in common. Add elements after Shared if new stuff is needed. 21 */ 22 typedef union _COMBINED_LOCK_ELEMENT 23 { 24 struct 25 { 26 LIST_ENTRY dummy; 27 FILE_SHARED_LOCK_ENTRY Shared; 28 }; 29 FILE_EXCLUSIVE_LOCK_ENTRY Exclusive; 30 } 31 COMBINED_LOCK_ELEMENT, *PCOMBINED_LOCK_ELEMENT; 32 33 typedef struct _LOCK_INFORMATION 34 { 35 RTL_GENERIC_TABLE RangeTable; 36 IO_CSQ Csq; 37 KSPIN_LOCK CsqLock; 38 LIST_ENTRY CsqList; 39 PFILE_LOCK BelongsTo; 40 LIST_ENTRY SharedLocks; 41 ULONG Generation; 42 } 43 LOCK_INFORMATION, *PLOCK_INFORMATION; 44 45 typedef struct _LOCK_SHARED_RANGE 46 { 47 LIST_ENTRY Entry; 48 LARGE_INTEGER Start, End; 49 ULONG Key; 50 PVOID ProcessId; 51 } 52 LOCK_SHARED_RANGE, *PLOCK_SHARED_RANGE; 53 54 #define TAG_TABLE 'BATL' 55 #define TAG_RANGE 'ARSF' 56 #define TAG_FLOCK 'KCLF' 57 58 /* PRIVATE FUNCTIONS *********************************************************/ 59 60 VOID 61 NTAPI 62 FsRtlCompleteLockIrpReal(IN PCOMPLETE_LOCK_IRP_ROUTINE CompleteRoutine, 63 IN PVOID Context, 64 IN PIRP Irp, 65 IN NTSTATUS Status, 66 OUT PNTSTATUS NewStatus, 67 IN PFILE_OBJECT FileObject OPTIONAL); 68 69 /* Generic table methods */ 70 71 static PVOID NTAPI LockAllocate(PRTL_GENERIC_TABLE Table, CLONG Bytes) 72 { 73 PVOID Result; 74 Result = ExAllocatePoolWithTag(NonPagedPool, Bytes, TAG_TABLE); 75 DPRINT("LockAllocate(%lu) => %p\n", Bytes, Result); 76 return Result; 77 } 78 79 static VOID NTAPI LockFree(PRTL_GENERIC_TABLE Table, PVOID Buffer) 80 { 81 DPRINT("LockFree(%p)\n", Buffer); 82 ExFreePoolWithTag(Buffer, TAG_TABLE); 83 } 84 85 static RTL_GENERIC_COMPARE_RESULTS NTAPI LockCompare 86 (PRTL_GENERIC_TABLE Table, PVOID PtrA, PVOID PtrB) 87 { 88 PCOMBINED_LOCK_ELEMENT A = PtrA, B = PtrB; 89 RTL_GENERIC_COMPARE_RESULTS Result; 90 #if 0 91 DPRINT("Starting to compare element %x to element %x\n", PtrA, PtrB); 92 #endif 93 /* Match if we overlap */ 94 if (((A->Exclusive.FileLock.StartingByte.QuadPart < 95 B->Exclusive.FileLock.EndingByte.QuadPart) && 96 (A->Exclusive.FileLock.StartingByte.QuadPart >= 97 B->Exclusive.FileLock.StartingByte.QuadPart)) || 98 ((B->Exclusive.FileLock.StartingByte.QuadPart < 99 A->Exclusive.FileLock.EndingByte.QuadPart) && 100 (B->Exclusive.FileLock.StartingByte.QuadPart >= 101 A->Exclusive.FileLock.StartingByte.QuadPart))) 102 return GenericEqual; 103 /* Otherwise, key on the starting byte */ 104 Result = 105 (A->Exclusive.FileLock.StartingByte.QuadPart < 106 B->Exclusive.FileLock.StartingByte.QuadPart) ? GenericLessThan : 107 (A->Exclusive.FileLock.StartingByte.QuadPart > 108 B->Exclusive.FileLock.StartingByte.QuadPart) ? GenericGreaterThan : 109 GenericEqual; 110 #if 0 111 DPRINT("Compare(%x:%x) %x-%x to %x-%x => %d\n", 112 A,B, 113 A->Exclusive.FileLock.StartingByte.LowPart, 114 A->Exclusive.FileLock.EndingByte.LowPart, 115 B->Exclusive.FileLock.StartingByte.LowPart, 116 B->Exclusive.FileLock.EndingByte.LowPart, 117 Result); 118 #endif 119 return Result; 120 } 121 122 /* CSQ methods */ 123 124 static NTSTATUS NTAPI LockInsertIrpEx 125 (PIO_CSQ Csq, 126 PIRP Irp, 127 PVOID InsertContext) 128 { 129 PLOCK_INFORMATION LockInfo = CONTAINING_RECORD(Csq, LOCK_INFORMATION, Csq); 130 InsertTailList(&LockInfo->CsqList, &Irp->Tail.Overlay.ListEntry); 131 return STATUS_SUCCESS; 132 } 133 134 static VOID NTAPI LockRemoveIrp(PIO_CSQ Csq, PIRP Irp) 135 { 136 RemoveEntryList(&Irp->Tail.Overlay.ListEntry); 137 } 138 139 static PIRP NTAPI LockPeekNextIrp(PIO_CSQ Csq, PIRP Irp, PVOID PeekContext) 140 { 141 // Context will be a COMBINED_LOCK_ELEMENT. We're looking for a 142 // lock that can be acquired, now that the lock matching PeekContext 143 // has been removed. 144 COMBINED_LOCK_ELEMENT LockElement; 145 PCOMBINED_LOCK_ELEMENT WhereUnlock = PeekContext; 146 PLOCK_INFORMATION LockInfo = CONTAINING_RECORD(Csq, LOCK_INFORMATION, Csq); 147 PLIST_ENTRY Following; 148 DPRINT("PeekNextIrp(IRP %p, Context %p)\n", Irp, PeekContext); 149 if (!Irp) 150 { 151 Following = LockInfo->CsqList.Flink; 152 } 153 else 154 Following = Irp->Tail.Overlay.ListEntry.Flink; 155 156 DPRINT("ListEntry %p Head %p\n", Following, &LockInfo->CsqList); 157 for (; 158 Following != &LockInfo->CsqList; 159 Following = Following->Flink) 160 { 161 PIO_STACK_LOCATION IoStack; 162 BOOLEAN Matching; 163 Irp = CONTAINING_RECORD(Following, IRP, Tail.Overlay.ListEntry); 164 DPRINT("Irp %p\n", Irp); 165 IoStack = IoGetCurrentIrpStackLocation(Irp); 166 LockElement.Exclusive.FileLock.StartingByte = 167 IoStack->Parameters.LockControl.ByteOffset; 168 LockElement.Exclusive.FileLock.EndingByte.QuadPart = 169 LockElement.Exclusive.FileLock.StartingByte.QuadPart + 170 IoStack->Parameters.LockControl.Length->QuadPart; 171 /* If a context was specified, it's a range to check to unlock */ 172 if (WhereUnlock) 173 { 174 Matching = LockCompare 175 (&LockInfo->RangeTable, &LockElement, WhereUnlock) != GenericEqual; 176 } 177 /* Else get any completable IRP */ 178 else 179 { 180 Matching = FALSE; 181 } 182 if (!Matching) 183 { 184 // This IRP is fine... 185 DPRINT("Returning the IRP %p\n", Irp); 186 return Irp; 187 } 188 } 189 DPRINT("Return NULL\n"); 190 return NULL; 191 } 192 193 static VOID NTAPI 194 LockAcquireQueueLock(PIO_CSQ Csq, PKIRQL Irql) 195 { 196 PLOCK_INFORMATION LockInfo = CONTAINING_RECORD(Csq, LOCK_INFORMATION, Csq); 197 KeAcquireSpinLock(&LockInfo->CsqLock, Irql); 198 } 199 200 static VOID NTAPI 201 LockReleaseQueueLock(PIO_CSQ Csq, KIRQL Irql) 202 { 203 PLOCK_INFORMATION LockInfo = CONTAINING_RECORD(Csq, LOCK_INFORMATION, Csq); 204 KeReleaseSpinLock(&LockInfo->CsqLock, Irql); 205 } 206 207 static VOID NTAPI 208 LockCompleteCanceledIrp(PIO_CSQ Csq, PIRP Irp) 209 { 210 NTSTATUS Status; 211 PLOCK_INFORMATION LockInfo = CONTAINING_RECORD(Csq, LOCK_INFORMATION, Csq); 212 DPRINT("Complete cancelled IRP %p Status %x\n", Irp, STATUS_CANCELLED); 213 FsRtlCompleteLockIrpReal 214 (LockInfo->BelongsTo->CompleteLockIrpRoutine, 215 NULL, 216 Irp, 217 STATUS_CANCELLED, 218 &Status, 219 NULL); 220 } 221 222 VOID 223 NTAPI 224 FsRtlCompleteLockIrpReal(IN PCOMPLETE_LOCK_IRP_ROUTINE CompleteRoutine, 225 IN PVOID Context, 226 IN PIRP Irp, 227 IN NTSTATUS Status, 228 OUT PNTSTATUS NewStatus, 229 IN PFILE_OBJECT FileObject OPTIONAL) 230 { 231 /* Check if we have a complete routine */ 232 Irp->IoStatus.Information = 0; 233 if (CompleteRoutine) 234 { 235 /* Check if we have a file object */ 236 if (FileObject) FileObject->LastLock = NULL; 237 238 /* Set the I/O Status and do completion */ 239 Irp->IoStatus.Status = Status; 240 DPRINT("Calling completion routine %p Status %x\n", Irp, Status); 241 *NewStatus = CompleteRoutine(Context, Irp); 242 } 243 else 244 { 245 /* Otherwise do a normal I/O complete request */ 246 DPRINT("Completing IRP %p Status %x\n", Irp, Status); 247 FsRtlCompleteRequest(Irp, Status); 248 *NewStatus = Status; 249 } 250 } 251 252 /* PUBLIC FUNCTIONS **********************************************************/ 253 254 /* 255 * @implemented 256 */ 257 PFILE_LOCK_INFO 258 NTAPI 259 FsRtlGetNextFileLock(IN PFILE_LOCK FileLock, 260 IN BOOLEAN Restart) 261 { 262 PCOMBINED_LOCK_ELEMENT Entry; 263 if (!FileLock->LockInformation) return NULL; 264 Entry = RtlEnumerateGenericTable(FileLock->LockInformation, Restart); 265 if (!Entry) return NULL; 266 else return &Entry->Exclusive.FileLock; 267 } 268 269 VOID 270 NTAPI 271 FsRtlpExpandLockElement 272 (PCOMBINED_LOCK_ELEMENT ToExpand, 273 PCOMBINED_LOCK_ELEMENT Conflict) 274 { 275 if (ToExpand->Exclusive.FileLock.StartingByte.QuadPart > 276 Conflict->Exclusive.FileLock.StartingByte.QuadPart) 277 { 278 ToExpand->Exclusive.FileLock.StartingByte = 279 Conflict->Exclusive.FileLock.StartingByte; 280 } 281 if (ToExpand->Exclusive.FileLock.EndingByte.QuadPart < 282 Conflict->Exclusive.FileLock.EndingByte.QuadPart) 283 { 284 ToExpand->Exclusive.FileLock.EndingByte = 285 Conflict->Exclusive.FileLock.EndingByte; 286 } 287 } 288 289 /* This function expands the conflicting range Conflict by removing and reinserting it, 290 then adds a shared range of the same size */ 291 PCOMBINED_LOCK_ELEMENT 292 NTAPI 293 FsRtlpRebuildSharedLockRange 294 (PFILE_LOCK FileLock, 295 PLOCK_INFORMATION LockInfo, 296 PCOMBINED_LOCK_ELEMENT Conflict) 297 { 298 /* Starting at Conflict->StartingByte and going to Conflict->EndingByte 299 * capture and expand a shared range from the shared range list. 300 * Finish when we've incorporated all overlapping shared regions. 301 */ 302 BOOLEAN InsertedNew = FALSE, RemovedOld; 303 COMBINED_LOCK_ELEMENT NewElement = *Conflict; 304 PCOMBINED_LOCK_ELEMENT Entry; 305 while ((Entry = RtlLookupElementGenericTable 306 (FileLock->LockInformation, &NewElement))) 307 { 308 FsRtlpExpandLockElement(&NewElement, Entry); 309 RemovedOld = RtlDeleteElementGenericTable 310 (&LockInfo->RangeTable, 311 Entry); 312 ASSERT(RemovedOld); 313 } 314 Conflict = RtlInsertElementGenericTable 315 (&LockInfo->RangeTable, 316 &NewElement, 317 sizeof(NewElement), 318 &InsertedNew); 319 ASSERT(InsertedNew); 320 return Conflict; 321 } 322 323 /* 324 * @implemented 325 */ 326 BOOLEAN 327 NTAPI 328 FsRtlPrivateLock(IN PFILE_LOCK FileLock, 329 IN PFILE_OBJECT FileObject, 330 IN PLARGE_INTEGER FileOffset, 331 IN PLARGE_INTEGER Length, 332 IN PEPROCESS Process, 333 IN ULONG Key, 334 IN BOOLEAN FailImmediately, 335 IN BOOLEAN ExclusiveLock, 336 OUT PIO_STATUS_BLOCK IoStatus, 337 IN PIRP Irp OPTIONAL, 338 IN PVOID Context OPTIONAL, 339 IN BOOLEAN AlreadySynchronized) 340 { 341 NTSTATUS Status; 342 COMBINED_LOCK_ELEMENT ToInsert; 343 PCOMBINED_LOCK_ELEMENT Conflict; 344 PLOCK_INFORMATION LockInfo; 345 PLOCK_SHARED_RANGE NewSharedRange; 346 BOOLEAN InsertedNew; 347 ULARGE_INTEGER UnsignedStart; 348 ULARGE_INTEGER UnsignedEnd; 349 350 DPRINT("FsRtlPrivateLock(%wZ, Offset %08x%08x (%d), Length %08x%08x (%d), Key %x, FailImmediately %u, Exclusive %u)\n", 351 &FileObject->FileName, 352 FileOffset->HighPart, 353 FileOffset->LowPart, 354 (int)FileOffset->QuadPart, 355 Length->HighPart, 356 Length->LowPart, 357 (int)Length->QuadPart, 358 Key, 359 FailImmediately, 360 ExclusiveLock); 361 362 UnsignedStart.QuadPart = FileOffset->QuadPart; 363 UnsignedEnd.QuadPart = FileOffset->QuadPart + Length->QuadPart; 364 365 if (UnsignedEnd.QuadPart < UnsignedStart.QuadPart) 366 { 367 DPRINT("File offset out of range\n"); 368 IoStatus->Status = STATUS_INVALID_PARAMETER; 369 if (Irp) 370 { 371 DPRINT("Complete lock %p Status %x\n", Irp, IoStatus->Status); 372 FsRtlCompleteLockIrpReal 373 (FileLock->CompleteLockIrpRoutine, 374 Context, 375 Irp, 376 IoStatus->Status, 377 &Status, 378 FileObject); 379 } 380 return FALSE; 381 } 382 383 /* Initialize the lock, if necessary */ 384 if (!FileLock->LockInformation) 385 { 386 LockInfo = ExAllocatePoolWithTag(NonPagedPool, sizeof(LOCK_INFORMATION), TAG_FLOCK); 387 if (!LockInfo) 388 { 389 IoStatus->Status = STATUS_NO_MEMORY; 390 return FALSE; 391 } 392 FileLock->LockInformation = LockInfo; 393 394 LockInfo->BelongsTo = FileLock; 395 InitializeListHead(&LockInfo->SharedLocks); 396 397 RtlInitializeGenericTable 398 (&LockInfo->RangeTable, 399 LockCompare, 400 LockAllocate, 401 LockFree, 402 NULL); 403 404 KeInitializeSpinLock(&LockInfo->CsqLock); 405 InitializeListHead(&LockInfo->CsqList); 406 407 IoCsqInitializeEx 408 (&LockInfo->Csq, 409 LockInsertIrpEx, 410 LockRemoveIrp, 411 LockPeekNextIrp, 412 LockAcquireQueueLock, 413 LockReleaseQueueLock, 414 LockCompleteCanceledIrp); 415 } 416 417 LockInfo = FileLock->LockInformation; 418 ToInsert.Exclusive.FileLock.FileObject = FileObject; 419 ToInsert.Exclusive.FileLock.StartingByte = *FileOffset; 420 ToInsert.Exclusive.FileLock.EndingByte.QuadPart = FileOffset->QuadPart + Length->QuadPart; 421 ToInsert.Exclusive.FileLock.ProcessId = Process; 422 ToInsert.Exclusive.FileLock.Key = Key; 423 ToInsert.Exclusive.FileLock.ExclusiveLock = ExclusiveLock; 424 425 Conflict = RtlInsertElementGenericTable 426 (FileLock->LockInformation, 427 &ToInsert, 428 sizeof(ToInsert), 429 &InsertedNew); 430 431 if (Conflict && !InsertedNew) 432 { 433 if (Conflict->Exclusive.FileLock.ExclusiveLock || ExclusiveLock) 434 { 435 DPRINT("Conflict %08x%08x:%08x%08x Exc %u (Want Exc %u)\n", 436 Conflict->Exclusive.FileLock.StartingByte.HighPart, 437 Conflict->Exclusive.FileLock.StartingByte.LowPart, 438 Conflict->Exclusive.FileLock.EndingByte.HighPart, 439 Conflict->Exclusive.FileLock.EndingByte.LowPart, 440 Conflict->Exclusive.FileLock.ExclusiveLock, 441 ExclusiveLock); 442 if (FailImmediately) 443 { 444 DPRINT("STATUS_FILE_LOCK_CONFLICT\n"); 445 IoStatus->Status = STATUS_FILE_LOCK_CONFLICT; 446 if (Irp) 447 { 448 DPRINT("STATUS_FILE_LOCK_CONFLICT: Complete\n"); 449 FsRtlCompleteLockIrpReal 450 (FileLock->CompleteLockIrpRoutine, 451 Context, 452 Irp, 453 IoStatus->Status, 454 &Status, 455 FileObject); 456 } 457 return FALSE; 458 } 459 else 460 { 461 IoStatus->Status = STATUS_PENDING; 462 if (Irp) 463 { 464 Irp->IoStatus.Information = LockInfo->Generation; 465 IoMarkIrpPending(Irp); 466 IoCsqInsertIrpEx 467 (&LockInfo->Csq, 468 Irp, 469 NULL, 470 NULL); 471 } 472 } 473 return FALSE; 474 } 475 else 476 { 477 ULONG i; 478 /* We know of at least one lock in range that's shared. We need to 479 * find out if any more exist and any are exclusive. */ 480 for (i = 0; i < RtlNumberGenericTableElements(&LockInfo->RangeTable); i++) 481 { 482 Conflict = RtlGetElementGenericTable(&LockInfo->RangeTable, i); 483 484 /* The first argument will be inserted as a shared range */ 485 if (Conflict && (LockCompare(&LockInfo->RangeTable, Conflict, &ToInsert) == GenericEqual)) 486 { 487 if (Conflict->Exclusive.FileLock.ExclusiveLock) 488 { 489 /* Found an exclusive match */ 490 if (FailImmediately) 491 { 492 IoStatus->Status = STATUS_FILE_LOCK_CONFLICT; 493 DPRINT("STATUS_FILE_LOCK_CONFLICT\n"); 494 if (Irp) 495 { 496 DPRINT("STATUS_FILE_LOCK_CONFLICT: Complete\n"); 497 FsRtlCompleteLockIrpReal 498 (FileLock->CompleteLockIrpRoutine, 499 Context, 500 Irp, 501 IoStatus->Status, 502 &Status, 503 FileObject); 504 } 505 } 506 else 507 { 508 IoStatus->Status = STATUS_PENDING; 509 if (Irp) 510 { 511 IoMarkIrpPending(Irp); 512 IoCsqInsertIrpEx 513 (&LockInfo->Csq, 514 Irp, 515 NULL, 516 NULL); 517 } 518 } 519 return FALSE; 520 } 521 } 522 } 523 524 DPRINT("Overlapping shared lock %wZ %08x%08x %08x%08x\n", 525 &FileObject->FileName, 526 Conflict->Exclusive.FileLock.StartingByte.HighPart, 527 Conflict->Exclusive.FileLock.StartingByte.LowPart, 528 Conflict->Exclusive.FileLock.EndingByte.HighPart, 529 Conflict->Exclusive.FileLock.EndingByte.LowPart); 530 Conflict = FsRtlpRebuildSharedLockRange(FileLock, 531 LockInfo, 532 &ToInsert); 533 if (!Conflict) 534 { 535 IoStatus->Status = STATUS_NO_MEMORY; 536 if (Irp) 537 { 538 FsRtlCompleteLockIrpReal 539 (FileLock->CompleteLockIrpRoutine, 540 Context, 541 Irp, 542 IoStatus->Status, 543 &Status, 544 FileObject); 545 } 546 } 547 548 /* We got here because there were only overlapping shared locks */ 549 /* A shared lock is both a range *and* a list entry. Insert the 550 entry here. */ 551 552 DPRINT("Adding shared lock %wZ\n", &FileObject->FileName); 553 NewSharedRange = 554 ExAllocatePoolWithTag(NonPagedPool, sizeof(*NewSharedRange), TAG_RANGE); 555 if (!NewSharedRange) 556 { 557 IoStatus->Status = STATUS_NO_MEMORY; 558 if (Irp) 559 { 560 FsRtlCompleteLockIrpReal 561 (FileLock->CompleteLockIrpRoutine, 562 Context, 563 Irp, 564 IoStatus->Status, 565 &Status, 566 FileObject); 567 } 568 return FALSE; 569 } 570 DPRINT("Adding shared lock %wZ\n", &FileObject->FileName); 571 NewSharedRange->Start = *FileOffset; 572 NewSharedRange->End.QuadPart = FileOffset->QuadPart + Length->QuadPart; 573 NewSharedRange->Key = Key; 574 NewSharedRange->ProcessId = ToInsert.Exclusive.FileLock.ProcessId; 575 InsertTailList(&LockInfo->SharedLocks, &NewSharedRange->Entry); 576 577 DPRINT("Acquired shared lock %wZ %08x%08x %08x%08x\n", 578 &FileObject->FileName, 579 Conflict->Exclusive.FileLock.StartingByte.HighPart, 580 Conflict->Exclusive.FileLock.StartingByte.LowPart, 581 Conflict->Exclusive.FileLock.EndingByte.HighPart, 582 Conflict->Exclusive.FileLock.EndingByte.LowPart); 583 IoStatus->Status = STATUS_SUCCESS; 584 if (Irp) 585 { 586 FsRtlCompleteLockIrpReal 587 (FileLock->CompleteLockIrpRoutine, 588 Context, 589 Irp, 590 IoStatus->Status, 591 &Status, 592 FileObject); 593 } 594 return TRUE; 595 } 596 } 597 else if (!Conflict) 598 { 599 /* Conflict here is (or would be) the newly inserted element, but we ran 600 * out of space probably. */ 601 IoStatus->Status = STATUS_NO_MEMORY; 602 if (Irp) 603 { 604 FsRtlCompleteLockIrpReal 605 (FileLock->CompleteLockIrpRoutine, 606 Context, 607 Irp, 608 IoStatus->Status, 609 &Status, 610 FileObject); 611 } 612 return FALSE; 613 } 614 else 615 { 616 DPRINT("Inserted new lock %wZ %08x%08x %08x%08x exclusive %u\n", 617 &FileObject->FileName, 618 Conflict->Exclusive.FileLock.StartingByte.HighPart, 619 Conflict->Exclusive.FileLock.StartingByte.LowPart, 620 Conflict->Exclusive.FileLock.EndingByte.HighPart, 621 Conflict->Exclusive.FileLock.EndingByte.LowPart, 622 Conflict->Exclusive.FileLock.ExclusiveLock); 623 if (!ExclusiveLock) 624 { 625 NewSharedRange = 626 ExAllocatePoolWithTag(NonPagedPool, sizeof(*NewSharedRange), TAG_RANGE); 627 if (!NewSharedRange) 628 { 629 IoStatus->Status = STATUS_NO_MEMORY; 630 if (Irp) 631 { 632 FsRtlCompleteLockIrpReal 633 (FileLock->CompleteLockIrpRoutine, 634 Context, 635 Irp, 636 IoStatus->Status, 637 &Status, 638 FileObject); 639 } 640 return FALSE; 641 } 642 DPRINT("Adding shared lock %wZ\n", &FileObject->FileName); 643 NewSharedRange->Start = *FileOffset; 644 NewSharedRange->End.QuadPart = FileOffset->QuadPart + Length->QuadPart; 645 NewSharedRange->Key = Key; 646 NewSharedRange->ProcessId = Process; 647 InsertTailList(&LockInfo->SharedLocks, &NewSharedRange->Entry); 648 } 649 650 /* Assume all is cool, and lock is set */ 651 IoStatus->Status = STATUS_SUCCESS; 652 653 if (Irp) 654 { 655 /* Complete the request */ 656 FsRtlCompleteLockIrpReal(FileLock->CompleteLockIrpRoutine, 657 Context, 658 Irp, 659 IoStatus->Status, 660 &Status, 661 FileObject); 662 663 /* Update the status */ 664 IoStatus->Status = Status; 665 } 666 } 667 668 return TRUE; 669 } 670 671 /* 672 * @implemented 673 */ 674 BOOLEAN 675 NTAPI 676 FsRtlCheckLockForReadAccess(IN PFILE_LOCK FileLock, 677 IN PIRP Irp) 678 { 679 BOOLEAN Result; 680 PIO_STACK_LOCATION IoStack = IoGetCurrentIrpStackLocation(Irp); 681 COMBINED_LOCK_ELEMENT ToFind; 682 PCOMBINED_LOCK_ELEMENT Found; 683 DPRINT("CheckLockForReadAccess(%wZ, Offset %08x%08x, Length %x)\n", 684 &IoStack->FileObject->FileName, 685 IoStack->Parameters.Read.ByteOffset.HighPart, 686 IoStack->Parameters.Read.ByteOffset.LowPart, 687 IoStack->Parameters.Read.Length); 688 if (!FileLock->LockInformation) { 689 DPRINT("CheckLockForReadAccess(%wZ) => TRUE\n", &IoStack->FileObject->FileName); 690 return TRUE; 691 } 692 ToFind.Exclusive.FileLock.StartingByte = IoStack->Parameters.Read.ByteOffset; 693 ToFind.Exclusive.FileLock.EndingByte.QuadPart = 694 ToFind.Exclusive.FileLock.StartingByte.QuadPart + 695 IoStack->Parameters.Read.Length; 696 Found = RtlLookupElementGenericTable 697 (FileLock->LockInformation, 698 &ToFind); 699 if (!Found) { 700 DPRINT("CheckLockForReadAccess(%wZ) => TRUE\n", &IoStack->FileObject->FileName); 701 return TRUE; 702 } 703 Result = !Found->Exclusive.FileLock.ExclusiveLock || 704 IoStack->Parameters.Read.Key == Found->Exclusive.FileLock.Key; 705 DPRINT("CheckLockForReadAccess(%wZ) => %s\n", &IoStack->FileObject->FileName, Result ? "TRUE" : "FALSE"); 706 return Result; 707 } 708 709 /* 710 * @implemented 711 */ 712 BOOLEAN 713 NTAPI 714 FsRtlCheckLockForWriteAccess(IN PFILE_LOCK FileLock, 715 IN PIRP Irp) 716 { 717 BOOLEAN Result; 718 PIO_STACK_LOCATION IoStack = IoGetCurrentIrpStackLocation(Irp); 719 COMBINED_LOCK_ELEMENT ToFind; 720 PCOMBINED_LOCK_ELEMENT Found; 721 PEPROCESS Process = Irp->Tail.Overlay.Thread->ThreadsProcess; 722 DPRINT("CheckLockForWriteAccess(%wZ, Offset %08x%08x, Length %x)\n", 723 &IoStack->FileObject->FileName, 724 IoStack->Parameters.Write.ByteOffset.HighPart, 725 IoStack->Parameters.Write.ByteOffset.LowPart, 726 IoStack->Parameters.Write.Length); 727 if (!FileLock->LockInformation) { 728 DPRINT("CheckLockForWriteAccess(%wZ) => TRUE\n", &IoStack->FileObject->FileName); 729 return TRUE; 730 } 731 ToFind.Exclusive.FileLock.StartingByte = IoStack->Parameters.Write.ByteOffset; 732 ToFind.Exclusive.FileLock.EndingByte.QuadPart = 733 ToFind.Exclusive.FileLock.StartingByte.QuadPart + 734 IoStack->Parameters.Write.Length; 735 Found = RtlLookupElementGenericTable 736 (FileLock->LockInformation, 737 &ToFind); 738 if (!Found) { 739 DPRINT("CheckLockForWriteAccess(%wZ) => TRUE\n", &IoStack->FileObject->FileName); 740 return TRUE; 741 } 742 Result = Process == Found->Exclusive.FileLock.ProcessId; 743 DPRINT("CheckLockForWriteAccess(%wZ) => %s\n", &IoStack->FileObject->FileName, Result ? "TRUE" : "FALSE"); 744 return Result; 745 } 746 747 /* 748 * @implemented 749 */ 750 BOOLEAN 751 NTAPI 752 FsRtlFastCheckLockForRead(IN PFILE_LOCK FileLock, 753 IN PLARGE_INTEGER FileOffset, 754 IN PLARGE_INTEGER Length, 755 IN ULONG Key, 756 IN PFILE_OBJECT FileObject, 757 IN PVOID Process) 758 { 759 PEPROCESS EProcess = Process; 760 COMBINED_LOCK_ELEMENT ToFind; 761 PCOMBINED_LOCK_ELEMENT Found; 762 DPRINT("FsRtlFastCheckLockForRead(%wZ, Offset %08x%08x, Length %08x%08x, Key %x)\n", 763 &FileObject->FileName, 764 FileOffset->HighPart, 765 FileOffset->LowPart, 766 Length->HighPart, 767 Length->LowPart, 768 Key); 769 ToFind.Exclusive.FileLock.StartingByte = *FileOffset; 770 ToFind.Exclusive.FileLock.EndingByte.QuadPart = 771 FileOffset->QuadPart + Length->QuadPart; 772 if (!FileLock->LockInformation) return TRUE; 773 Found = RtlLookupElementGenericTable 774 (FileLock->LockInformation, 775 &ToFind); 776 if (!Found || !Found->Exclusive.FileLock.ExclusiveLock) return TRUE; 777 return Found->Exclusive.FileLock.Key == Key && 778 Found->Exclusive.FileLock.ProcessId == EProcess; 779 } 780 781 /* 782 * @implemented 783 */ 784 BOOLEAN 785 NTAPI 786 FsRtlFastCheckLockForWrite(IN PFILE_LOCK FileLock, 787 IN PLARGE_INTEGER FileOffset, 788 IN PLARGE_INTEGER Length, 789 IN ULONG Key, 790 IN PFILE_OBJECT FileObject, 791 IN PVOID Process) 792 { 793 BOOLEAN Result; 794 PEPROCESS EProcess = Process; 795 COMBINED_LOCK_ELEMENT ToFind; 796 PCOMBINED_LOCK_ELEMENT Found; 797 DPRINT("FsRtlFastCheckLockForWrite(%wZ, Offset %08x%08x, Length %08x%08x, Key %x)\n", 798 &FileObject->FileName, 799 FileOffset->HighPart, 800 FileOffset->LowPart, 801 Length->HighPart, 802 Length->LowPart, 803 Key); 804 ToFind.Exclusive.FileLock.StartingByte = *FileOffset; 805 ToFind.Exclusive.FileLock.EndingByte.QuadPart = 806 FileOffset->QuadPart + Length->QuadPart; 807 if (!FileLock->LockInformation) { 808 DPRINT("CheckForWrite(%wZ) => TRUE\n", &FileObject->FileName); 809 return TRUE; 810 } 811 Found = RtlLookupElementGenericTable 812 (FileLock->LockInformation, 813 &ToFind); 814 if (!Found) { 815 DPRINT("CheckForWrite(%wZ) => TRUE\n", &FileObject->FileName); 816 return TRUE; 817 } 818 Result = Found->Exclusive.FileLock.Key == Key && 819 Found->Exclusive.FileLock.ProcessId == EProcess; 820 DPRINT("CheckForWrite(%wZ) => %s\n", &FileObject->FileName, Result ? "TRUE" : "FALSE"); 821 return Result; 822 } 823 824 /* 825 * @implemented 826 */ 827 NTSTATUS 828 NTAPI 829 FsRtlFastUnlockSingle(IN PFILE_LOCK FileLock, 830 IN PFILE_OBJECT FileObject, 831 IN PLARGE_INTEGER FileOffset, 832 IN PLARGE_INTEGER Length, 833 IN PEPROCESS Process, 834 IN ULONG Key, 835 IN PVOID Context OPTIONAL, 836 IN BOOLEAN AlreadySynchronized) 837 { 838 BOOLEAN FoundShared = FALSE; 839 PLIST_ENTRY SharedEntry; 840 PLOCK_SHARED_RANGE SharedRange = NULL; 841 COMBINED_LOCK_ELEMENT Find; 842 PCOMBINED_LOCK_ELEMENT Entry; 843 PIRP NextMatchingLockIrp; 844 PLOCK_INFORMATION InternalInfo = FileLock->LockInformation; 845 DPRINT("FsRtlFastUnlockSingle(%wZ, Offset %08x%08x (%d), Length %08x%08x (%d), Key %x)\n", 846 &FileObject->FileName, 847 FileOffset->HighPart, 848 FileOffset->LowPart, 849 (int)FileOffset->QuadPart, 850 Length->HighPart, 851 Length->LowPart, 852 (int)Length->QuadPart, 853 Key); 854 // The region to unlock must correspond exactly to a previously locked region 855 // -- msdn 856 // But Windows 2003 doesn't assert on it and simply ignores that parameter 857 // ASSERT(AlreadySynchronized); 858 Find.Exclusive.FileLock.StartingByte = *FileOffset; 859 Find.Exclusive.FileLock.EndingByte.QuadPart = 860 FileOffset->QuadPart + Length->QuadPart; 861 if (!InternalInfo) { 862 DPRINT("File not previously locked (ever)\n"); 863 return STATUS_RANGE_NOT_LOCKED; 864 } 865 Entry = RtlLookupElementGenericTable(&InternalInfo->RangeTable, &Find); 866 if (!Entry) { 867 DPRINT("Range not locked %wZ\n", &FileObject->FileName); 868 return STATUS_RANGE_NOT_LOCKED; 869 } 870 871 DPRINT("Found lock entry: Exclusive %u %08x%08x:%08x%08x %wZ\n", 872 Entry->Exclusive.FileLock.ExclusiveLock, 873 Entry->Exclusive.FileLock.StartingByte.HighPart, 874 Entry->Exclusive.FileLock.StartingByte.LowPart, 875 Entry->Exclusive.FileLock.EndingByte.HighPart, 876 Entry->Exclusive.FileLock.EndingByte.LowPart, 877 &FileObject->FileName); 878 879 if (Entry->Exclusive.FileLock.ExclusiveLock) 880 { 881 if (Entry->Exclusive.FileLock.Key != Key || 882 Entry->Exclusive.FileLock.ProcessId != Process || 883 Entry->Exclusive.FileLock.StartingByte.QuadPart != FileOffset->QuadPart || 884 Entry->Exclusive.FileLock.EndingByte.QuadPart != 885 FileOffset->QuadPart + Length->QuadPart) 886 { 887 DPRINT("Range not locked %wZ\n", &FileObject->FileName); 888 return STATUS_RANGE_NOT_LOCKED; 889 } 890 RtlCopyMemory(&Find, Entry, sizeof(Find)); 891 // Remove the old exclusive lock region 892 RtlDeleteElementGenericTable(&InternalInfo->RangeTable, Entry); 893 } 894 else 895 { 896 DPRINT("Shared lock %wZ Start %08x%08x End %08x%08x\n", 897 &FileObject->FileName, 898 Entry->Exclusive.FileLock.StartingByte.HighPart, 899 Entry->Exclusive.FileLock.StartingByte.LowPart, 900 Entry->Exclusive.FileLock.EndingByte.HighPart, 901 Entry->Exclusive.FileLock.EndingByte.LowPart); 902 for (SharedEntry = InternalInfo->SharedLocks.Flink; 903 SharedEntry != &InternalInfo->SharedLocks; 904 SharedEntry = SharedEntry->Flink) 905 { 906 SharedRange = CONTAINING_RECORD(SharedEntry, LOCK_SHARED_RANGE, Entry); 907 if (SharedRange->Start.QuadPart == FileOffset->QuadPart && 908 SharedRange->End.QuadPart == FileOffset->QuadPart + Length->QuadPart && 909 SharedRange->Key == Key && 910 SharedRange->ProcessId == Process) 911 { 912 FoundShared = TRUE; 913 DPRINT("Found shared element to delete %wZ Start %08x%08x End %08x%08x Key %x\n", 914 &FileObject->FileName, 915 SharedRange->Start.HighPart, 916 SharedRange->Start.LowPart, 917 SharedRange->End.HighPart, 918 SharedRange->End.LowPart, 919 SharedRange->Key); 920 break; 921 } 922 } 923 if (FoundShared) 924 { 925 /* Remove the found range from the shared range lists */ 926 RemoveEntryList(&SharedRange->Entry); 927 ExFreePoolWithTag(SharedRange, TAG_RANGE); 928 /* We need to rebuild the list of shared ranges. */ 929 DPRINT("Removing the lock entry %wZ (%08x%08x:%08x%08x)\n", 930 &FileObject->FileName, 931 Entry->Exclusive.FileLock.StartingByte.HighPart, 932 Entry->Exclusive.FileLock.StartingByte.LowPart, 933 Entry->Exclusive.FileLock.EndingByte.HighPart, 934 Entry->Exclusive.FileLock.EndingByte.LowPart); 935 936 /* Remember what was in there and remove it from the table */ 937 Find = *Entry; 938 RtlDeleteElementGenericTable(&InternalInfo->RangeTable, &Find); 939 /* Put shared locks back in place */ 940 for (SharedEntry = InternalInfo->SharedLocks.Flink; 941 SharedEntry != &InternalInfo->SharedLocks; 942 SharedEntry = SharedEntry->Flink) 943 { 944 COMBINED_LOCK_ELEMENT LockElement; 945 SharedRange = CONTAINING_RECORD(SharedEntry, LOCK_SHARED_RANGE, Entry); 946 LockElement.Exclusive.FileLock.FileObject = FileObject; 947 LockElement.Exclusive.FileLock.StartingByte = SharedRange->Start; 948 LockElement.Exclusive.FileLock.EndingByte = SharedRange->End; 949 LockElement.Exclusive.FileLock.ProcessId = SharedRange->ProcessId; 950 LockElement.Exclusive.FileLock.Key = SharedRange->Key; 951 LockElement.Exclusive.FileLock.ExclusiveLock = FALSE; 952 953 if (LockCompare(&InternalInfo->RangeTable, &Find, &LockElement) != GenericEqual) 954 { 955 DPRINT("Skipping range %08x%08x:%08x%08x\n", 956 LockElement.Exclusive.FileLock.StartingByte.HighPart, 957 LockElement.Exclusive.FileLock.StartingByte.LowPart, 958 LockElement.Exclusive.FileLock.EndingByte.HighPart, 959 LockElement.Exclusive.FileLock.EndingByte.LowPart); 960 continue; 961 } 962 DPRINT("Re-creating range %08x%08x:%08x%08x\n", 963 LockElement.Exclusive.FileLock.StartingByte.HighPart, 964 LockElement.Exclusive.FileLock.StartingByte.LowPart, 965 LockElement.Exclusive.FileLock.EndingByte.HighPart, 966 LockElement.Exclusive.FileLock.EndingByte.LowPart); 967 FsRtlpRebuildSharedLockRange(FileLock, InternalInfo, &LockElement); 968 } 969 } 970 else 971 { 972 return STATUS_RANGE_NOT_LOCKED; 973 } 974 } 975 976 #ifndef NDEBUG 977 DPRINT("Lock still has:\n"); 978 for (SharedEntry = InternalInfo->SharedLocks.Flink; 979 SharedEntry != &InternalInfo->SharedLocks; 980 SharedEntry = SharedEntry->Flink) 981 { 982 SharedRange = CONTAINING_RECORD(SharedEntry, LOCK_SHARED_RANGE, Entry); 983 DPRINT("Shared element %wZ Offset %08x%08x Length %08x%08x Key %x\n", 984 &FileObject->FileName, 985 SharedRange->Start.HighPart, 986 SharedRange->Start.LowPart, 987 SharedRange->End.HighPart, 988 SharedRange->End.LowPart, 989 SharedRange->Key); 990 } 991 #endif 992 993 // this is definitely the thing we want 994 InternalInfo->Generation++; 995 while ((NextMatchingLockIrp = IoCsqRemoveNextIrp(&InternalInfo->Csq, &Find))) 996 { 997 if (NextMatchingLockIrp->IoStatus.Information == InternalInfo->Generation) 998 { 999 // We've already looked at this one, meaning that we looped. 1000 // Put it back and exit. 1001 IoCsqInsertIrpEx 1002 (&InternalInfo->Csq, 1003 NextMatchingLockIrp, 1004 NULL, 1005 NULL); 1006 break; 1007 } 1008 // Got a new lock irp... try to do the new lock operation 1009 // Note that we pick an operation that would succeed at the time 1010 // we looked, but can't guarantee that it won't just be re-queued 1011 // because somebody else snatched part of the range in a new thread. 1012 DPRINT("Locking another IRP %p for %p %wZ\n", 1013 NextMatchingLockIrp, FileLock, &FileObject->FileName); 1014 FsRtlProcessFileLock(InternalInfo->BelongsTo, NextMatchingLockIrp, NULL); 1015 } 1016 1017 DPRINT("Success %wZ\n", &FileObject->FileName); 1018 return STATUS_SUCCESS; 1019 } 1020 1021 /* 1022 * @implemented 1023 */ 1024 NTSTATUS 1025 NTAPI 1026 FsRtlFastUnlockAll(IN PFILE_LOCK FileLock, 1027 IN PFILE_OBJECT FileObject, 1028 IN PEPROCESS Process, 1029 IN PVOID Context OPTIONAL) 1030 { 1031 PLIST_ENTRY ListEntry; 1032 PCOMBINED_LOCK_ELEMENT Entry; 1033 PLOCK_INFORMATION InternalInfo = FileLock->LockInformation; 1034 DPRINT("FsRtlFastUnlockAll(%wZ)\n", &FileObject->FileName); 1035 // XXX Synchronize somehow 1036 if (!FileLock->LockInformation) { 1037 DPRINT("Not locked %wZ\n", &FileObject->FileName); 1038 return STATUS_RANGE_NOT_LOCKED; // no locks 1039 } 1040 for (ListEntry = InternalInfo->SharedLocks.Flink; 1041 ListEntry != &InternalInfo->SharedLocks;) 1042 { 1043 LARGE_INTEGER Length; 1044 PLOCK_SHARED_RANGE Range = CONTAINING_RECORD(ListEntry, LOCK_SHARED_RANGE, Entry); 1045 Length.QuadPart = Range->End.QuadPart - Range->Start.QuadPart; 1046 ListEntry = ListEntry->Flink; 1047 if (Range->ProcessId != Process) 1048 continue; 1049 FsRtlFastUnlockSingle 1050 (FileLock, 1051 FileObject, 1052 &Range->Start, 1053 &Length, 1054 Range->ProcessId, 1055 Range->Key, 1056 Context, 1057 TRUE); 1058 } 1059 for (Entry = RtlEnumerateGenericTable(&InternalInfo->RangeTable, TRUE); 1060 Entry; 1061 Entry = RtlEnumerateGenericTable(&InternalInfo->RangeTable, FALSE)) 1062 { 1063 LARGE_INTEGER Length; 1064 // We'll take the first one to be the list head, and free the others first... 1065 Length.QuadPart = 1066 Entry->Exclusive.FileLock.EndingByte.QuadPart - 1067 Entry->Exclusive.FileLock.StartingByte.QuadPart; 1068 FsRtlFastUnlockSingle 1069 (FileLock, 1070 Entry->Exclusive.FileLock.FileObject, 1071 &Entry->Exclusive.FileLock.StartingByte, 1072 &Length, 1073 Entry->Exclusive.FileLock.ProcessId, 1074 Entry->Exclusive.FileLock.Key, 1075 Context, 1076 TRUE); 1077 } 1078 DPRINT("Done %wZ\n", &FileObject->FileName); 1079 return STATUS_SUCCESS; 1080 } 1081 1082 /* 1083 * @implemented 1084 */ 1085 NTSTATUS 1086 NTAPI 1087 FsRtlFastUnlockAllByKey(IN PFILE_LOCK FileLock, 1088 IN PFILE_OBJECT FileObject, 1089 IN PEPROCESS Process, 1090 IN ULONG Key, 1091 IN PVOID Context OPTIONAL) 1092 { 1093 PLIST_ENTRY ListEntry; 1094 PCOMBINED_LOCK_ELEMENT Entry; 1095 PLOCK_INFORMATION InternalInfo = FileLock->LockInformation; 1096 1097 DPRINT("FsRtlFastUnlockAllByKey(%wZ,Key %x)\n", &FileObject->FileName, Key); 1098 1099 // XXX Synchronize somehow 1100 if (!FileLock->LockInformation) return STATUS_RANGE_NOT_LOCKED; // no locks 1101 for (ListEntry = InternalInfo->SharedLocks.Flink; 1102 ListEntry != &InternalInfo->SharedLocks;) 1103 { 1104 PLOCK_SHARED_RANGE Range = CONTAINING_RECORD(ListEntry, LOCK_SHARED_RANGE, Entry); 1105 LARGE_INTEGER Length; 1106 Length.QuadPart = Range->End.QuadPart - Range->Start.QuadPart; 1107 ListEntry = ListEntry->Flink; 1108 if (Range->ProcessId != Process || 1109 Range->Key != Key) 1110 continue; 1111 FsRtlFastUnlockSingle 1112 (FileLock, 1113 FileObject, 1114 &Range->Start, 1115 &Length, 1116 Range->ProcessId, 1117 Range->Key, 1118 Context, 1119 TRUE); 1120 } 1121 for (Entry = RtlEnumerateGenericTable(&InternalInfo->RangeTable, TRUE); 1122 Entry; 1123 Entry = RtlEnumerateGenericTable(&InternalInfo->RangeTable, FALSE)) 1124 { 1125 LARGE_INTEGER Length; 1126 // We'll take the first one to be the list head, and free the others first... 1127 Length.QuadPart = 1128 Entry->Exclusive.FileLock.EndingByte.QuadPart - 1129 Entry->Exclusive.FileLock.StartingByte.QuadPart; 1130 if (Entry->Exclusive.FileLock.Key == Key && 1131 Entry->Exclusive.FileLock.ProcessId == Process) 1132 { 1133 FsRtlFastUnlockSingle 1134 (FileLock, 1135 Entry->Exclusive.FileLock.FileObject, 1136 &Entry->Exclusive.FileLock.StartingByte, 1137 &Length, 1138 Entry->Exclusive.FileLock.ProcessId, 1139 Entry->Exclusive.FileLock.Key, 1140 Context, 1141 TRUE); 1142 } 1143 } 1144 1145 return STATUS_SUCCESS; 1146 } 1147 1148 /* 1149 * @implemented 1150 */ 1151 NTSTATUS 1152 NTAPI 1153 FsRtlProcessFileLock(IN PFILE_LOCK FileLock, 1154 IN PIRP Irp, 1155 IN PVOID Context OPTIONAL) 1156 { 1157 PIO_STACK_LOCATION IoStackLocation; 1158 NTSTATUS Status; 1159 IO_STATUS_BLOCK IoStatusBlock; 1160 1161 /* Get the I/O Stack location */ 1162 IoStackLocation = IoGetCurrentIrpStackLocation(Irp); 1163 ASSERT(IoStackLocation->MajorFunction == IRP_MJ_LOCK_CONTROL); 1164 1165 /* Clear the I/O status block and check what function this is */ 1166 IoStatusBlock.Information = 0; 1167 1168 DPRINT("FsRtlProcessFileLock(%wZ, MinorFunction %x)\n", 1169 &IoStackLocation->FileObject->FileName, 1170 IoStackLocation->MinorFunction); 1171 1172 switch(IoStackLocation->MinorFunction) 1173 { 1174 /* A lock */ 1175 case IRP_MN_LOCK: 1176 1177 /* Call the private lock routine */ 1178 FsRtlPrivateLock(FileLock, 1179 IoStackLocation->FileObject, 1180 &IoStackLocation-> 1181 Parameters.LockControl.ByteOffset, 1182 IoStackLocation->Parameters.LockControl.Length, 1183 IoGetRequestorProcess(Irp), 1184 IoStackLocation->Parameters.LockControl.Key, 1185 IoStackLocation->Flags & SL_FAIL_IMMEDIATELY, 1186 IoStackLocation->Flags & SL_EXCLUSIVE_LOCK, 1187 &IoStatusBlock, 1188 Irp, 1189 Context, 1190 FALSE); 1191 return IoStatusBlock.Status; 1192 1193 /* A single unlock */ 1194 case IRP_MN_UNLOCK_SINGLE: 1195 1196 /* Call fast unlock */ 1197 IoStatusBlock.Status = 1198 FsRtlFastUnlockSingle(FileLock, 1199 IoStackLocation->FileObject, 1200 &IoStackLocation->Parameters.LockControl. 1201 ByteOffset, 1202 IoStackLocation->Parameters.LockControl. 1203 Length, 1204 IoGetRequestorProcess(Irp), 1205 IoStackLocation->Parameters.LockControl. 1206 Key, 1207 Context, 1208 FALSE); 1209 break; 1210 1211 /* Total unlock */ 1212 case IRP_MN_UNLOCK_ALL: 1213 1214 /* Do a fast unlock */ 1215 IoStatusBlock.Status = FsRtlFastUnlockAll(FileLock, 1216 IoStackLocation-> 1217 FileObject, 1218 IoGetRequestorProcess(Irp), 1219 Context); 1220 break; 1221 1222 /* Unlock by key */ 1223 case IRP_MN_UNLOCK_ALL_BY_KEY: 1224 1225 /* Do it */ 1226 IoStatusBlock.Status = 1227 FsRtlFastUnlockAllByKey(FileLock, 1228 IoStackLocation->FileObject, 1229 IoGetRequestorProcess(Irp), 1230 IoStackLocation->Parameters. 1231 LockControl.Key, 1232 Context); 1233 break; 1234 1235 /* Invalid request */ 1236 default: 1237 1238 /* Complete it */ 1239 FsRtlCompleteRequest(Irp, STATUS_INVALID_DEVICE_REQUEST); 1240 IoStatusBlock.Status = STATUS_INVALID_DEVICE_REQUEST; 1241 return STATUS_INVALID_DEVICE_REQUEST; 1242 } 1243 1244 /* Return the status */ 1245 DPRINT("Lock IRP %p %x\n", Irp, IoStatusBlock.Status); 1246 FsRtlCompleteLockIrpReal 1247 (FileLock->CompleteLockIrpRoutine, 1248 Context, 1249 Irp, 1250 IoStatusBlock.Status, 1251 &Status, 1252 NULL); 1253 return IoStatusBlock.Status; 1254 } 1255 1256 /* 1257 * @implemented 1258 */ 1259 VOID 1260 NTAPI 1261 FsRtlInitializeFileLock (IN PFILE_LOCK FileLock, 1262 IN PCOMPLETE_LOCK_IRP_ROUTINE CompleteLockIrpRoutine OPTIONAL, 1263 IN PUNLOCK_ROUTINE UnlockRoutine OPTIONAL) 1264 { 1265 /* Setup the lock */ 1266 RtlZeroMemory(FileLock, sizeof(*FileLock)); 1267 FileLock->FastIoIsQuestionable = FALSE; 1268 FileLock->CompleteLockIrpRoutine = CompleteLockIrpRoutine; 1269 FileLock->UnlockRoutine = UnlockRoutine; 1270 FileLock->LockInformation = NULL; 1271 } 1272 1273 /* 1274 * @implemented 1275 */ 1276 VOID 1277 NTAPI 1278 FsRtlUninitializeFileLock(IN PFILE_LOCK FileLock) 1279 { 1280 if (FileLock->LockInformation) 1281 { 1282 PIRP Irp; 1283 PLOCK_INFORMATION InternalInfo = FileLock->LockInformation; 1284 PCOMBINED_LOCK_ELEMENT Entry; 1285 PLIST_ENTRY SharedEntry; 1286 PLOCK_SHARED_RANGE SharedRange; 1287 // MSDN: this completes any remaining lock IRPs 1288 for (SharedEntry = InternalInfo->SharedLocks.Flink; 1289 SharedEntry != &InternalInfo->SharedLocks;) 1290 { 1291 SharedRange = CONTAINING_RECORD(SharedEntry, LOCK_SHARED_RANGE, Entry); 1292 SharedEntry = SharedEntry->Flink; 1293 RemoveEntryList(&SharedRange->Entry); 1294 ExFreePoolWithTag(SharedRange, TAG_RANGE); 1295 } 1296 while ((Entry = RtlGetElementGenericTable(&InternalInfo->RangeTable, 0)) != NULL) 1297 { 1298 RtlDeleteElementGenericTable(&InternalInfo->RangeTable, Entry); 1299 } 1300 while ((Irp = IoCsqRemoveNextIrp(&InternalInfo->Csq, NULL)) != NULL) 1301 { 1302 FsRtlProcessFileLock(FileLock, Irp, NULL); 1303 } 1304 ExFreePoolWithTag(InternalInfo, TAG_FLOCK); 1305 FileLock->LockInformation = NULL; 1306 } 1307 } 1308 1309 /* 1310 * @implemented 1311 */ 1312 PFILE_LOCK 1313 NTAPI 1314 FsRtlAllocateFileLock(IN PCOMPLETE_LOCK_IRP_ROUTINE CompleteLockIrpRoutine OPTIONAL, 1315 IN PUNLOCK_ROUTINE UnlockRoutine OPTIONAL) 1316 { 1317 PFILE_LOCK FileLock; 1318 1319 /* Try to allocate it */ 1320 FileLock = ExAllocateFromPagedLookasideList(&FsRtlFileLockLookasideList); 1321 if (FileLock) 1322 { 1323 /* Initialize it */ 1324 FsRtlInitializeFileLock(FileLock, 1325 CompleteLockIrpRoutine, 1326 UnlockRoutine); 1327 } 1328 1329 /* Return the lock */ 1330 return FileLock; 1331 } 1332 1333 /* 1334 * @implemented 1335 */ 1336 VOID 1337 NTAPI 1338 FsRtlFreeFileLock(IN PFILE_LOCK FileLock) 1339 { 1340 /* Uninitialize and free the lock */ 1341 FsRtlUninitializeFileLock(FileLock); 1342 ExFreeToPagedLookasideList(&FsRtlFileLockLookasideList, FileLock); 1343 } 1344