1 /* 2 * PROJECT: ReactOS Kernel 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: ntoskrnl/ex/handle.c 5 * PURPOSE: Generic Executive Handle Tables 6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org) 7 * Thomas Weidenmueller <w3seek@reactos.com> 8 */ 9 10 /* INCLUDES ******************************************************************/ 11 12 #include <ntoskrnl.h> 13 #define NDEBUG 14 #include <debug.h> 15 16 /* GLOBALS *******************************************************************/ 17 18 LIST_ENTRY HandleTableListHead; 19 EX_PUSH_LOCK HandleTableListLock; 20 #define SizeOfHandle(x) (sizeof(HANDLE) * (x)) 21 #define INDEX_TO_HANDLE_VALUE(x) ((x) << HANDLE_TAG_BITS) 22 23 /* PRIVATE FUNCTIONS *********************************************************/ 24 25 VOID 26 NTAPI 27 INIT_FUNCTION 28 ExpInitializeHandleTables(VOID) 29 { 30 /* Initialize the list of handle tables and the lock */ 31 InitializeListHead(&HandleTableListHead); 32 ExInitializePushLock(&HandleTableListLock); 33 } 34 35 PHANDLE_TABLE_ENTRY 36 NTAPI 37 ExpLookupHandleTableEntry(IN PHANDLE_TABLE HandleTable, 38 IN EXHANDLE Handle) 39 { 40 ULONG TableLevel; 41 ULONG_PTR TableBase; 42 PHANDLE_TABLE_ENTRY HandleArray, Entry; 43 PVOID *PointerArray; 44 45 /* Clear the tag bits */ 46 Handle.TagBits = 0; 47 48 /* Check if the handle is in the allocated range */ 49 if (Handle.Value >= HandleTable->NextHandleNeedingPool) 50 { 51 return NULL; 52 } 53 54 /* Get the table code */ 55 TableBase = HandleTable->TableCode; 56 57 /* Extract the table level and actual table base */ 58 TableLevel = (ULONG)(TableBase & 3); 59 TableBase &= ~3; 60 61 PointerArray = (PVOID*)TableBase; 62 HandleArray = (PHANDLE_TABLE_ENTRY)TableBase; 63 64 /* Check what level we're running at */ 65 switch (TableLevel) 66 { 67 case 2: 68 69 /* Get the mid level pointer array */ 70 PointerArray = PointerArray[Handle.HighIndex]; 71 ASSERT(PointerArray != NULL); 72 73 /* Fall through */ 74 case 1: 75 76 /* Get the handle array */ 77 HandleArray = PointerArray[Handle.MidIndex]; 78 ASSERT(HandleArray != NULL); 79 80 /* Fall through */ 81 case 0: 82 83 /* Get the entry using the low index */ 84 Entry = &HandleArray[Handle.LowIndex]; 85 86 /* All done */ 87 break; 88 89 default: 90 91 ASSERT(FALSE); 92 Entry = NULL; 93 } 94 95 /* Return the handle entry */ 96 return Entry; 97 } 98 99 PVOID 100 NTAPI 101 ExpAllocateTablePagedPool(IN PEPROCESS Process OPTIONAL, 102 IN SIZE_T Size) 103 { 104 PVOID Buffer; 105 106 /* Do the allocation */ 107 Buffer = ExAllocatePoolWithTag(PagedPool, Size, TAG_OBJECT_TABLE); 108 if (Buffer) 109 { 110 /* Clear the memory */ 111 RtlZeroMemory(Buffer, Size); 112 113 /* Check if we have a process to charge quota */ 114 if (Process) 115 { 116 /* FIXME: Charge quota */ 117 } 118 } 119 120 /* Return the allocated memory */ 121 return Buffer; 122 } 123 124 PVOID 125 NTAPI 126 ExpAllocateTablePagedPoolNoZero(IN PEPROCESS Process OPTIONAL, 127 IN SIZE_T Size) 128 { 129 PVOID Buffer; 130 131 /* Do the allocation */ 132 Buffer = ExAllocatePoolWithTag(PagedPool, Size, TAG_OBJECT_TABLE); 133 if (Buffer) 134 { 135 /* Check if we have a process to charge quota */ 136 if (Process) 137 { 138 /* FIXME: Charge quota */ 139 } 140 } 141 142 /* Return the allocated memory */ 143 return Buffer; 144 } 145 146 VOID 147 NTAPI 148 ExpFreeTablePagedPool(IN PEPROCESS Process OPTIONAL, 149 IN PVOID Buffer, 150 IN SIZE_T Size) 151 { 152 /* Free the buffer */ 153 ExFreePoolWithTag(Buffer, TAG_OBJECT_TABLE); 154 if (Process) 155 { 156 /* FIXME: Release quota */ 157 } 158 } 159 160 VOID 161 NTAPI 162 ExpFreeLowLevelTable(IN PEPROCESS Process, 163 IN PHANDLE_TABLE_ENTRY TableEntry) 164 { 165 /* Check if we have an entry */ 166 if (TableEntry[0].Object) 167 { 168 /* Free the entry */ 169 ExpFreeTablePagedPool(Process, 170 TableEntry[0].Object, 171 LOW_LEVEL_ENTRIES * 172 sizeof(HANDLE_TABLE_ENTRY_INFO)); 173 } 174 175 /* Free the table */ 176 ExpFreeTablePagedPool(Process, TableEntry, PAGE_SIZE); 177 } 178 179 VOID 180 NTAPI 181 ExpFreeHandleTable(IN PHANDLE_TABLE HandleTable) 182 { 183 PEPROCESS Process = HandleTable->QuotaProcess; 184 ULONG i, j; 185 ULONG_PTR TableCode = HandleTable->TableCode; 186 ULONG_PTR TableBase = TableCode & ~3; 187 ULONG TableLevel = (ULONG)(TableCode & 3); 188 PHANDLE_TABLE_ENTRY Level1, *Level2, **Level3; 189 PAGED_CODE(); 190 191 /* Check which level we're at */ 192 if (TableLevel == 0) 193 { 194 /* Select the first level table base and just free it */ 195 Level1 = (PVOID)TableBase; 196 ExpFreeLowLevelTable(Process, Level1); 197 } 198 else if (TableLevel == 1) 199 { 200 /* Select the second level table base */ 201 Level2 = (PVOID)TableBase; 202 203 /* Loop each mid level entry */ 204 for (i = 0; i < MID_LEVEL_ENTRIES; i++) 205 { 206 /* Leave if we've reached the last entry */ 207 if (!Level2[i]) break; 208 209 /* Free the second level table */ 210 ExpFreeLowLevelTable(Process, Level2[i]); 211 } 212 213 /* Free the second level table */ 214 ExpFreeTablePagedPool(Process, Level2, PAGE_SIZE); 215 } 216 else 217 { 218 /* Select the third level table base */ 219 Level3 = (PVOID)TableBase; 220 221 /* Loop each high level entry */ 222 for (i = 0; i < HIGH_LEVEL_ENTRIES; i++) 223 { 224 /* Leave if we've reached the last entry */ 225 if (!Level3[i]) break; 226 227 /* Loop each mid level entry */ 228 for (j = 0; j < MID_LEVEL_ENTRIES; j++) 229 { 230 /* Leave if we've reached the last entry */ 231 if (!Level3[i][j]) break; 232 233 /* Free the second level table */ 234 ExpFreeLowLevelTable(Process, Level3[i][j]); 235 } 236 237 /* Free the third level table entry */ 238 ExpFreeTablePagedPool(Process, Level3[i], PAGE_SIZE); 239 } 240 241 /* Free the third level table */ 242 ExpFreeTablePagedPool(Process, 243 Level3, 244 SizeOfHandle(HIGH_LEVEL_ENTRIES)); 245 } 246 247 /* Free the actual table and check if we need to release quota */ 248 ExFreePoolWithTag(HandleTable, TAG_OBJECT_TABLE); 249 if (Process) 250 { 251 /* FIXME: TODO */ 252 } 253 } 254 255 VOID 256 NTAPI 257 ExpFreeHandleTableEntry(IN PHANDLE_TABLE HandleTable, 258 IN EXHANDLE Handle, 259 IN PHANDLE_TABLE_ENTRY HandleTableEntry) 260 { 261 ULONG OldValue, *Free; 262 ULONG LockIndex; 263 PAGED_CODE(); 264 265 /* Sanity checks */ 266 ASSERT(HandleTableEntry->Object == NULL); 267 ASSERT(HandleTableEntry == ExpLookupHandleTableEntry(HandleTable, Handle)); 268 269 /* Decrement the handle count */ 270 InterlockedDecrement(&HandleTable->HandleCount); 271 272 /* Mark the handle as free */ 273 Handle.TagBits = 0; 274 275 /* Check if we're FIFO */ 276 if (!HandleTable->StrictFIFO) 277 { 278 /* Select a lock index */ 279 LockIndex = Handle.Index % 4; 280 281 /* Select which entry to use */ 282 Free = (HandleTable->HandleTableLock[LockIndex].Locked) ? 283 &HandleTable->FirstFree : &HandleTable->LastFree; 284 } 285 else 286 { 287 /* No need to worry about locking, take the last entry */ 288 Free = &HandleTable->LastFree; 289 } 290 291 /* Start value change loop */ 292 for (;;) 293 { 294 /* Get the current value and write */ 295 OldValue = *Free; 296 HandleTableEntry->NextFreeTableEntry = OldValue; 297 if (InterlockedCompareExchange((PLONG)Free, Handle.AsULONG, OldValue) == OldValue) 298 { 299 /* Break out, we're done. Make sure the handle value makes sense */ 300 ASSERT((OldValue & FREE_HANDLE_MASK) < 301 HandleTable->NextHandleNeedingPool); 302 break; 303 } 304 } 305 } 306 307 PHANDLE_TABLE 308 NTAPI 309 ExpAllocateHandleTable(IN PEPROCESS Process OPTIONAL, 310 IN BOOLEAN NewTable) 311 { 312 PHANDLE_TABLE HandleTable; 313 PHANDLE_TABLE_ENTRY HandleTableTable, HandleEntry; 314 ULONG i; 315 PAGED_CODE(); 316 317 /* Allocate the table */ 318 HandleTable = ExAllocatePoolWithTag(PagedPool, 319 sizeof(HANDLE_TABLE), 320 TAG_OBJECT_TABLE); 321 if (!HandleTable) return NULL; 322 323 /* Check if we have a process */ 324 if (Process) 325 { 326 /* FIXME: Charge quota */ 327 } 328 329 /* Clear the table */ 330 RtlZeroMemory(HandleTable, sizeof(HANDLE_TABLE)); 331 332 /* Now allocate the first level structures */ 333 HandleTableTable = ExpAllocateTablePagedPoolNoZero(Process, PAGE_SIZE); 334 if (!HandleTableTable) 335 { 336 /* Failed, free the table */ 337 ExFreePoolWithTag(HandleTable, TAG_OBJECT_TABLE); 338 return NULL; 339 } 340 341 /* Write the pointer to our first level structures */ 342 HandleTable->TableCode = (ULONG_PTR)HandleTableTable; 343 344 /* Initialize the first entry */ 345 HandleEntry = &HandleTableTable[0]; 346 HandleEntry->NextFreeTableEntry = -2; 347 HandleEntry->Value = 0; 348 349 /* Check if this is a new table */ 350 if (NewTable) 351 { 352 /* Go past the root entry */ 353 HandleEntry++; 354 355 /* Loop every low level entry */ 356 for (i = 1; i < (LOW_LEVEL_ENTRIES - 1); i++) 357 { 358 /* Set up the free data */ 359 HandleEntry->Value = 0; 360 HandleEntry->NextFreeTableEntry = INDEX_TO_HANDLE_VALUE(i + 1); 361 362 /* Move to the next entry */ 363 HandleEntry++; 364 } 365 366 /* Terminate the last entry */ 367 HandleEntry->Value = 0; 368 HandleEntry->NextFreeTableEntry = 0; 369 HandleTable->FirstFree = INDEX_TO_HANDLE_VALUE(1); 370 } 371 372 /* Set the next handle needing pool after our allocated page from above */ 373 HandleTable->NextHandleNeedingPool = INDEX_TO_HANDLE_VALUE(LOW_LEVEL_ENTRIES); 374 375 /* Setup the rest of the handle table data */ 376 HandleTable->QuotaProcess = Process; 377 HandleTable->UniqueProcessId = PsGetCurrentProcess()->UniqueProcessId; 378 HandleTable->Flags = 0; 379 380 /* Loop all the handle table locks */ 381 for (i = 0; i < 4; i++) 382 { 383 /* Initialize the handle table lock */ 384 ExInitializePushLock(&HandleTable->HandleTableLock[i]); 385 } 386 387 /* Initialize the contention event lock and return the lock */ 388 ExInitializePushLock(&HandleTable->HandleContentionEvent); 389 return HandleTable; 390 } 391 392 PHANDLE_TABLE_ENTRY 393 NTAPI 394 ExpAllocateLowLevelTable(IN PHANDLE_TABLE HandleTable, 395 IN BOOLEAN DoInit) 396 { 397 ULONG i, Base; 398 PHANDLE_TABLE_ENTRY Low, HandleEntry; 399 400 /* Allocate the low level table */ 401 Low = ExpAllocateTablePagedPoolNoZero(HandleTable->QuotaProcess, 402 PAGE_SIZE); 403 if (!Low) return NULL; 404 405 /* Setup the initial entry */ 406 HandleEntry = &Low[0]; 407 HandleEntry->NextFreeTableEntry = -2; 408 HandleEntry->Value = 0; 409 410 /* Check if we're initializing */ 411 if (DoInit) 412 { 413 /* Go to the next entry and the base entry */ 414 HandleEntry++; 415 Base = HandleTable->NextHandleNeedingPool + INDEX_TO_HANDLE_VALUE(2); 416 417 /* Loop each entry */ 418 for (i = Base; 419 i < Base + INDEX_TO_HANDLE_VALUE(LOW_LEVEL_ENTRIES - 2); 420 i += INDEX_TO_HANDLE_VALUE(1)) 421 { 422 /* Free this entry and move on to the next one */ 423 HandleEntry->NextFreeTableEntry = i; 424 HandleEntry->Value = 0; 425 HandleEntry++; 426 } 427 428 /* Terminate the last entry */ 429 HandleEntry->NextFreeTableEntry = 0; 430 HandleEntry->Value = 0; 431 } 432 433 /* Return the low level table */ 434 return Low; 435 } 436 437 PHANDLE_TABLE_ENTRY* 438 NTAPI 439 ExpAllocateMidLevelTable(IN PHANDLE_TABLE HandleTable, 440 IN BOOLEAN DoInit, 441 OUT PHANDLE_TABLE_ENTRY *LowTableEntry) 442 { 443 PHANDLE_TABLE_ENTRY *Mid, Low; 444 445 /* Allocate the mid level table */ 446 Mid = ExpAllocateTablePagedPool(HandleTable->QuotaProcess, PAGE_SIZE); 447 if (!Mid) return NULL; 448 449 /* Allocate a new low level for it */ 450 Low = ExpAllocateLowLevelTable(HandleTable, DoInit); 451 if (!Low) 452 { 453 /* We failed, free the mid table */ 454 ExpFreeTablePagedPool(HandleTable->QuotaProcess, Mid, PAGE_SIZE); 455 return NULL; 456 } 457 458 /* Link the tables and return the pointer */ 459 Mid[0] = Low; 460 *LowTableEntry = Low; 461 return Mid; 462 } 463 464 BOOLEAN 465 NTAPI 466 ExpAllocateHandleTableEntrySlow(IN PHANDLE_TABLE HandleTable, 467 IN BOOLEAN DoInit) 468 { 469 ULONG i, j, Index; 470 PHANDLE_TABLE_ENTRY Low = NULL, *Mid, **High, *SecondLevel, **ThirdLevel; 471 ULONG NewFree, FirstFree; 472 PVOID Value; 473 ULONG_PTR TableCode = HandleTable->TableCode; 474 ULONG_PTR TableBase = TableCode & ~3; 475 ULONG TableLevel = (ULONG)(TableCode & 3); 476 PAGED_CODE(); 477 478 /* Check how many levels we already have */ 479 if (TableLevel == 0) 480 { 481 /* Allocate a mid level, since we only have a low level */ 482 Mid = ExpAllocateMidLevelTable(HandleTable, DoInit, &Low); 483 if (!Mid) return FALSE; 484 485 /* Link up the tables */ 486 Mid[1] = Mid[0]; 487 Mid[0] = (PVOID)TableBase; 488 489 /* Write the new level and attempt to change the table code */ 490 TableBase = ((ULONG_PTR)Mid) | 1; 491 Value = InterlockedExchangePointer((PVOID*)&HandleTable->TableCode, (PVOID)TableBase); 492 } 493 else if (TableLevel == 1) 494 { 495 /* Setup the 2nd level table */ 496 SecondLevel = (PVOID)TableBase; 497 498 /* Get if the next index can fit in the table */ 499 i = HandleTable->NextHandleNeedingPool / 500 INDEX_TO_HANDLE_VALUE(LOW_LEVEL_ENTRIES); 501 if (i < MID_LEVEL_ENTRIES) 502 { 503 /* We need to allocate a new table */ 504 Low = ExpAllocateLowLevelTable(HandleTable, DoInit); 505 if (!Low) return FALSE; 506 507 /* Update the table */ 508 Value = InterlockedExchangePointer((PVOID*)&SecondLevel[i], Low); 509 ASSERT(Value == NULL); 510 } 511 else 512 { 513 /* We need a new high level table */ 514 High = ExpAllocateTablePagedPool(HandleTable->QuotaProcess, 515 SizeOfHandle(HIGH_LEVEL_ENTRIES)); 516 if (!High) return FALSE; 517 518 /* Allocate a new mid level table as well */ 519 Mid = ExpAllocateMidLevelTable(HandleTable, DoInit, &Low); 520 if (!Mid) 521 { 522 /* We failed, free the high level table as well */ 523 ExpFreeTablePagedPool(HandleTable->QuotaProcess, 524 High, 525 SizeOfHandle(HIGH_LEVEL_ENTRIES)); 526 return FALSE; 527 } 528 529 /* Link up the tables */ 530 High[0] = (PVOID)TableBase; 531 High[1] = Mid; 532 533 /* Write the new table and change the table code */ 534 TableBase = ((ULONG_PTR)High) | 2; 535 Value = InterlockedExchangePointer((PVOID*)&HandleTable->TableCode, 536 (PVOID)TableBase); 537 } 538 } 539 else if (TableLevel == 2) 540 { 541 /* Setup the 3rd level table */ 542 ThirdLevel = (PVOID)TableBase; 543 544 /* Get the index and check if it can fit */ 545 i = HandleTable->NextHandleNeedingPool / INDEX_TO_HANDLE_VALUE(MAX_MID_INDEX); 546 if (i >= HIGH_LEVEL_ENTRIES) return FALSE; 547 548 /* Check if there's no mid-level table */ 549 if (!ThirdLevel[i]) 550 { 551 /* Allocate a new mid level table */ 552 Mid = ExpAllocateMidLevelTable(HandleTable, DoInit, &Low); 553 if (!Mid) return FALSE; 554 555 /* Update the table pointer */ 556 Value = InterlockedExchangePointer((PVOID*)&ThirdLevel[i], Mid); 557 ASSERT(Value == NULL); 558 } 559 else 560 { 561 /* We have one, check at which index we should insert our entry */ 562 Index = (HandleTable->NextHandleNeedingPool / INDEX_TO_HANDLE_VALUE(1)) - 563 i * MAX_MID_INDEX; 564 j = Index / LOW_LEVEL_ENTRIES; 565 566 /* Allocate a new low level */ 567 Low = ExpAllocateLowLevelTable(HandleTable, DoInit); 568 if (!Low) return FALSE; 569 570 /* Update the table pointer */ 571 Value = InterlockedExchangePointer((PVOID*)&ThirdLevel[i][j], Low); 572 ASSERT(Value == NULL); 573 } 574 } 575 else 576 { 577 /* Something is really broken */ 578 ASSERT(FALSE); 579 } 580 581 /* Update the index of the next handle */ 582 Index = InterlockedExchangeAdd((PLONG) &HandleTable->NextHandleNeedingPool, 583 INDEX_TO_HANDLE_VALUE(LOW_LEVEL_ENTRIES)); 584 585 /* Check if need to initialize the table */ 586 if (DoInit) 587 { 588 /* Create a new index number */ 589 Index += INDEX_TO_HANDLE_VALUE(1); 590 591 /* Start free index change loop */ 592 for (;;) 593 { 594 /* Setup the first free index */ 595 FirstFree = HandleTable->FirstFree; 596 Low[LOW_LEVEL_ENTRIES - 1].NextFreeTableEntry = FirstFree; 597 598 /* Change the index */ 599 NewFree = InterlockedCompareExchange((PLONG) &HandleTable->FirstFree, 600 Index, 601 FirstFree); 602 if (NewFree == FirstFree) break; 603 } 604 } 605 606 /* All done */ 607 return TRUE; 608 } 609 610 ULONG 611 NTAPI 612 ExpMoveFreeHandles(IN PHANDLE_TABLE HandleTable) 613 { 614 ULONG LastFree, i; 615 616 /* Clear the last free index */ 617 LastFree = InterlockedExchange((PLONG) &HandleTable->LastFree, 0); 618 619 /* Check if we had no index */ 620 if (!LastFree) return LastFree; 621 622 /* Acquire the locks we need */ 623 for (i = 1; i < 4; i++) 624 { 625 /* Acquire this lock exclusively */ 626 ExWaitOnPushLock(&HandleTable->HandleTableLock[i]); 627 } 628 629 /* Check if we're not strict FIFO */ 630 if (!HandleTable->StrictFIFO) 631 { 632 /* Update the first free index */ 633 if (!InterlockedCompareExchange((PLONG) &HandleTable->FirstFree, LastFree, 0)) 634 { 635 /* We're done, exit */ 636 return LastFree; 637 } 638 } 639 640 /* We are strict FIFO, we need to reverse the entries */ 641 ASSERT(FALSE); 642 return LastFree; 643 } 644 645 PHANDLE_TABLE_ENTRY 646 NTAPI 647 ExpAllocateHandleTableEntry(IN PHANDLE_TABLE HandleTable, 648 OUT PEXHANDLE NewHandle) 649 { 650 ULONG OldValue, NewValue, NewValue1; 651 PHANDLE_TABLE_ENTRY Entry; 652 EXHANDLE Handle, OldHandle; 653 BOOLEAN Result; 654 ULONG i; 655 656 /* Start allocation loop */ 657 for (;;) 658 { 659 /* Get the current link */ 660 OldValue = HandleTable->FirstFree; 661 while (!OldValue) 662 { 663 /* No free entries remain, lock the handle table */ 664 KeEnterCriticalRegion(); 665 ExAcquirePushLockExclusive(&HandleTable->HandleTableLock[0]); 666 667 /* Check the value again */ 668 OldValue = HandleTable->FirstFree; 669 if (OldValue) 670 { 671 /* Another thread has already created a new level, bail out */ 672 ExReleasePushLockExclusive(&HandleTable->HandleTableLock[0]); 673 KeLeaveCriticalRegion(); 674 break; 675 } 676 677 /* Now move any free handles */ 678 OldValue = ExpMoveFreeHandles(HandleTable); 679 if (OldValue) 680 { 681 /* Another thread has already moved them, bail out */ 682 ExReleasePushLockExclusive(&HandleTable->HandleTableLock[0]); 683 KeLeaveCriticalRegion(); 684 break; 685 } 686 687 /* We're the first one through, so do the actual allocation */ 688 Result = ExpAllocateHandleTableEntrySlow(HandleTable, TRUE); 689 690 /* Unlock the table and get the value now */ 691 ExReleasePushLockExclusive(&HandleTable->HandleTableLock[0]); 692 KeLeaveCriticalRegion(); 693 OldValue = HandleTable->FirstFree; 694 695 /* Check if allocation failed */ 696 if (!Result) 697 { 698 /* Check if nobody else went through here */ 699 if (!OldValue) 700 { 701 /* We're still the only thread around, so fail */ 702 NewHandle->GenericHandleOverlay = NULL; 703 return NULL; 704 } 705 } 706 } 707 708 /* We made it, write the current value */ 709 Handle.Value = (OldValue & FREE_HANDLE_MASK); 710 711 /* Lookup the entry for this handle */ 712 Entry = ExpLookupHandleTableEntry(HandleTable, Handle); 713 714 /* Get an available lock and acquire it */ 715 OldHandle.Value = OldValue; 716 i = OldHandle.Index % 4; 717 KeEnterCriticalRegion(); 718 ExAcquirePushLockShared(&HandleTable->HandleTableLock[i]); 719 720 /* Check if the value changed after acquiring the lock */ 721 if (OldValue != *(volatile ULONG*)&HandleTable->FirstFree) 722 { 723 /* It did, so try again */ 724 ExReleasePushLockShared(&HandleTable->HandleTableLock[i]); 725 KeLeaveCriticalRegion(); 726 continue; 727 } 728 729 /* Now get the next value and do the compare */ 730 NewValue = *(volatile ULONG*)&Entry->NextFreeTableEntry; 731 NewValue1 = InterlockedCompareExchange((PLONG) &HandleTable->FirstFree, 732 NewValue, 733 OldValue); 734 735 /* The change was done, so release the lock */ 736 ExReleasePushLockShared(&HandleTable->HandleTableLock[i]); 737 KeLeaveCriticalRegion(); 738 739 /* Check if the compare was successful */ 740 if (NewValue1 == OldValue) 741 { 742 /* Make sure that the new handle is in range, and break out */ 743 ASSERT((NewValue & FREE_HANDLE_MASK) < 744 HandleTable->NextHandleNeedingPool); 745 break; 746 } 747 else 748 { 749 /* The compare failed, make sure we expected it */ 750 ASSERT((NewValue1 & FREE_HANDLE_MASK) != 751 (OldValue & FREE_HANDLE_MASK)); 752 } 753 } 754 755 /* Increase the number of handles */ 756 InterlockedIncrement(&HandleTable->HandleCount); 757 758 /* Return the handle and the entry */ 759 *NewHandle = Handle; 760 return Entry; 761 } 762 763 PHANDLE_TABLE 764 NTAPI 765 ExCreateHandleTable(IN PEPROCESS Process OPTIONAL) 766 { 767 PHANDLE_TABLE HandleTable; 768 PAGED_CODE(); 769 770 /* Allocate the handle table */ 771 HandleTable = ExpAllocateHandleTable(Process, TRUE); 772 if (!HandleTable) return NULL; 773 774 /* Acquire the handle table lock */ 775 KeEnterCriticalRegion(); 776 ExAcquirePushLockExclusive(&HandleTableListLock); 777 778 /* Insert it into the list */ 779 InsertTailList(&HandleTableListHead, &HandleTable->HandleTableList); 780 781 /* Release the lock */ 782 ExReleasePushLockExclusive(&HandleTableListLock); 783 KeLeaveCriticalRegion(); 784 785 /* Return the handle table */ 786 return HandleTable; 787 } 788 789 HANDLE 790 NTAPI 791 ExCreateHandle(IN PHANDLE_TABLE HandleTable, 792 IN PHANDLE_TABLE_ENTRY HandleTableEntry) 793 { 794 EXHANDLE Handle; 795 PHANDLE_TABLE_ENTRY NewEntry; 796 PAGED_CODE(); 797 798 /* Start with a clean handle */ 799 Handle.GenericHandleOverlay = NULL; 800 801 /* Allocate a new entry */ 802 NewEntry = ExpAllocateHandleTableEntry(HandleTable, &Handle); 803 if (NewEntry) 804 { 805 /* Enter a critical region */ 806 KeEnterCriticalRegion(); 807 808 /* Write the entry */ 809 *NewEntry = *HandleTableEntry; 810 811 /* Unlock it and leave the critical region */ 812 ExUnlockHandleTableEntry(HandleTable, NewEntry); 813 KeLeaveCriticalRegion(); 814 } 815 816 /* Return the handle value */ 817 return Handle.GenericHandleOverlay; 818 } 819 820 VOID 821 NTAPI 822 ExpBlockOnLockedHandleEntry(IN PHANDLE_TABLE HandleTable, 823 IN PHANDLE_TABLE_ENTRY HandleTableEntry) 824 { 825 LONG_PTR OldValue; 826 EX_PUSH_LOCK_WAIT_BLOCK WaitBlock; 827 828 /* Block on the pushlock */ 829 ExBlockPushLock(&HandleTable->HandleContentionEvent, &WaitBlock); 830 831 /* Get the current value and check if it's been unlocked */ 832 OldValue = HandleTableEntry->Value; 833 if (!(OldValue) || (OldValue & EXHANDLE_TABLE_ENTRY_LOCK_BIT)) 834 { 835 /* Unblock the pushlock and return */ 836 ExfUnblockPushLock(&HandleTable->HandleContentionEvent, &WaitBlock); 837 } 838 else 839 { 840 /* Wait for it to be unblocked */ 841 ExWaitForUnblockPushLock(&HandleTable->HandleContentionEvent, 842 &WaitBlock); 843 } 844 } 845 846 BOOLEAN 847 NTAPI 848 ExpLockHandleTableEntry(IN PHANDLE_TABLE HandleTable, 849 IN PHANDLE_TABLE_ENTRY HandleTableEntry) 850 { 851 LONG_PTR NewValue, OldValue; 852 853 /* Sanity check */ 854 ASSERT((KeGetCurrentThread()->CombinedApcDisable != 0) || 855 (KeGetCurrentIrql() == APC_LEVEL)); 856 857 /* Start lock loop */ 858 for (;;) 859 { 860 /* Get the current value and check if it's locked */ 861 OldValue = *(volatile LONG_PTR *)&HandleTableEntry->Object; 862 if (OldValue & EXHANDLE_TABLE_ENTRY_LOCK_BIT) 863 { 864 /* It's not locked, remove the lock bit to lock it */ 865 NewValue = OldValue & ~EXHANDLE_TABLE_ENTRY_LOCK_BIT; 866 if (InterlockedCompareExchangePointer(&HandleTableEntry->Object, 867 (PVOID)NewValue, 868 (PVOID)OldValue) == (PVOID)OldValue) 869 { 870 /* We locked it, get out */ 871 return TRUE; 872 } 873 } 874 else 875 { 876 /* We couldn't lock it, bail out if it's been freed */ 877 if (!OldValue) return FALSE; 878 } 879 880 /* It's locked, wait for it to be unlocked */ 881 ExpBlockOnLockedHandleEntry(HandleTable, HandleTableEntry); 882 } 883 } 884 885 VOID 886 NTAPI 887 ExUnlockHandleTableEntry(IN PHANDLE_TABLE HandleTable, 888 IN PHANDLE_TABLE_ENTRY HandleTableEntry) 889 { 890 LONG_PTR OldValue; 891 PAGED_CODE(); 892 893 /* Sanity check */ 894 ASSERT((KeGetCurrentThread()->CombinedApcDisable != 0) || 895 (KeGetCurrentIrql() == APC_LEVEL)); 896 897 /* Set the lock bit and make sure it wasn't earlier */ 898 OldValue = InterlockedOr((PLONG) &HandleTableEntry->Value, 899 EXHANDLE_TABLE_ENTRY_LOCK_BIT); 900 ASSERT((OldValue & EXHANDLE_TABLE_ENTRY_LOCK_BIT) == 0); 901 902 /* Unblock any waiters */ 903 ExfUnblockPushLock(&HandleTable->HandleContentionEvent, NULL); 904 } 905 906 VOID 907 NTAPI 908 ExRemoveHandleTable(IN PHANDLE_TABLE HandleTable) 909 { 910 PAGED_CODE(); 911 912 /* Acquire the table lock */ 913 KeEnterCriticalRegion(); 914 ExAcquirePushLockExclusive(&HandleTableListLock); 915 916 /* Remove the table and reset the list */ 917 RemoveEntryList(&HandleTable->HandleTableList); 918 InitializeListHead(&HandleTable->HandleTableList); 919 920 /* Release the lock */ 921 ExReleasePushLockExclusive(&HandleTableListLock); 922 KeLeaveCriticalRegion(); 923 } 924 925 VOID 926 NTAPI 927 ExDestroyHandleTable(IN PHANDLE_TABLE HandleTable, 928 IN PVOID DestroyHandleProcedure OPTIONAL) 929 { 930 PAGED_CODE(); 931 932 /* Remove the handle from the list */ 933 ExRemoveHandleTable(HandleTable); 934 935 /* Check if we have a destroy callback */ 936 if (DestroyHandleProcedure) 937 { 938 /* FIXME: */ 939 ASSERT(FALSE); 940 } 941 942 /* Free the handle table */ 943 ExpFreeHandleTable(HandleTable); 944 } 945 946 BOOLEAN 947 NTAPI 948 ExDestroyHandle(IN PHANDLE_TABLE HandleTable, 949 IN HANDLE Handle, 950 IN PHANDLE_TABLE_ENTRY HandleTableEntry OPTIONAL) 951 { 952 EXHANDLE ExHandle; 953 PVOID Object; 954 PAGED_CODE(); 955 956 /* Setup the actual handle value */ 957 ExHandle.GenericHandleOverlay = Handle; 958 959 /* Enter a critical region and check if we have to lookup the handle */ 960 KeEnterCriticalRegion(); 961 if (!HandleTableEntry) 962 { 963 /* Lookup the entry */ 964 HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, ExHandle); 965 966 /* Make sure that we found an entry, and that it's valid */ 967 if (!(HandleTableEntry) || 968 !(HandleTableEntry->Object) || 969 (HandleTableEntry->NextFreeTableEntry == -2)) 970 { 971 /* Invalid handle, fail */ 972 KeLeaveCriticalRegion(); 973 return FALSE; 974 } 975 976 /* Lock the entry */ 977 if (!ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 978 { 979 /* Couldn't lock, fail */ 980 KeLeaveCriticalRegion(); 981 return FALSE; 982 } 983 } 984 else 985 { 986 /* Make sure the handle is locked */ 987 ASSERT((HandleTableEntry->Value & EXHANDLE_TABLE_ENTRY_LOCK_BIT) == 0); 988 } 989 990 /* Clear the handle */ 991 Object = InterlockedExchangePointer((PVOID*)&HandleTableEntry->Object, NULL); 992 993 /* Sanity checks */ 994 ASSERT(Object != NULL); 995 ASSERT((((ULONG_PTR)Object) & EXHANDLE_TABLE_ENTRY_LOCK_BIT) == 0); 996 997 /* Unblock the pushlock */ 998 ExfUnblockPushLock(&HandleTable->HandleContentionEvent, NULL); 999 1000 /* Free the actual entry */ 1001 ExpFreeHandleTableEntry(HandleTable, ExHandle, HandleTableEntry); 1002 1003 /* If we got here, return success */ 1004 KeLeaveCriticalRegion(); 1005 return TRUE; 1006 } 1007 1008 PHANDLE_TABLE_ENTRY 1009 NTAPI 1010 ExMapHandleToPointer(IN PHANDLE_TABLE HandleTable, 1011 IN HANDLE Handle) 1012 { 1013 EXHANDLE ExHandle; 1014 PHANDLE_TABLE_ENTRY HandleTableEntry; 1015 PAGED_CODE(); 1016 1017 /* Set the handle value */ 1018 ExHandle.GenericHandleOverlay = Handle; 1019 1020 /* Fail if we got an invalid index */ 1021 if (!(ExHandle.Index & (LOW_LEVEL_ENTRIES - 1))) return NULL; 1022 1023 /* Do the lookup */ 1024 HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, ExHandle); 1025 if (!HandleTableEntry) return NULL; 1026 1027 /* Lock it */ 1028 if (!ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) return NULL; 1029 1030 /* Return the entry */ 1031 return HandleTableEntry; 1032 } 1033 1034 PHANDLE_TABLE 1035 NTAPI 1036 ExDupHandleTable(IN PEPROCESS Process, 1037 IN PHANDLE_TABLE HandleTable, 1038 IN PEX_DUPLICATE_HANDLE_CALLBACK DupHandleProcedure, 1039 IN ULONG_PTR Mask) 1040 { 1041 PHANDLE_TABLE NewTable; 1042 EXHANDLE Handle; 1043 PHANDLE_TABLE_ENTRY HandleTableEntry, NewEntry; 1044 BOOLEAN Failed = FALSE; 1045 PAGED_CODE(); 1046 1047 /* Allocate the duplicated copy */ 1048 NewTable = ExpAllocateHandleTable(Process, FALSE); 1049 if (!NewTable) return NULL; 1050 1051 /* Loop each entry */ 1052 while (NewTable->NextHandleNeedingPool < 1053 HandleTable->NextHandleNeedingPool) 1054 { 1055 /* Insert it into the duplicated copy */ 1056 if (!ExpAllocateHandleTableEntrySlow(NewTable, FALSE)) 1057 { 1058 /* Insert failed, free the new copy and return */ 1059 ExpFreeHandleTable(NewTable); 1060 return NULL; 1061 } 1062 } 1063 1064 /* Setup the initial handle table data */ 1065 NewTable->HandleCount = 0; 1066 NewTable->ExtraInfoPages = 0; 1067 NewTable->FirstFree = 0; 1068 1069 /* Setup the first handle value */ 1070 Handle.Value = INDEX_TO_HANDLE_VALUE(1); 1071 1072 /* Enter a critical region and lookup the new entry */ 1073 KeEnterCriticalRegion(); 1074 while ((NewEntry = ExpLookupHandleTableEntry(NewTable, Handle))) 1075 { 1076 /* Lookup the old entry */ 1077 HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, Handle); 1078 1079 /* Loop each entry */ 1080 do 1081 { 1082 /* Check if it doesn't match the audit mask */ 1083 if (!(HandleTableEntry->Value & Mask)) 1084 { 1085 /* Free it since we won't use it */ 1086 Failed = TRUE; 1087 } 1088 else 1089 { 1090 /* Lock the entry */ 1091 if (!ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 1092 { 1093 /* Free it since we can't lock it, so we won't use it */ 1094 Failed = TRUE; 1095 } 1096 else 1097 { 1098 /* Copy the handle value */ 1099 *NewEntry = *HandleTableEntry; 1100 1101 /* Call the duplicate callback */ 1102 if (DupHandleProcedure(Process, 1103 HandleTable, 1104 HandleTableEntry, 1105 NewEntry)) 1106 { 1107 /* Clear failure flag */ 1108 Failed = FALSE; 1109 1110 /* Lock the entry, increase the handle count */ 1111 NewEntry->Value |= EXHANDLE_TABLE_ENTRY_LOCK_BIT; 1112 NewTable->HandleCount++; 1113 } 1114 else 1115 { 1116 /* Duplication callback refused, fail */ 1117 Failed = TRUE; 1118 } 1119 } 1120 } 1121 1122 /* Check if we failed earlier and need to free */ 1123 if (Failed) 1124 { 1125 /* Free this entry */ 1126 NewEntry->Object = NULL; 1127 NewEntry->NextFreeTableEntry = NewTable->FirstFree; 1128 NewTable->FirstFree = (ULONG)Handle.Value; 1129 } 1130 1131 /* Increase the handle value and move to the next entry */ 1132 Handle.Value += INDEX_TO_HANDLE_VALUE(1); 1133 NewEntry++; 1134 HandleTableEntry++; 1135 } while (Handle.Value % INDEX_TO_HANDLE_VALUE(LOW_LEVEL_ENTRIES)); 1136 1137 /* We're done, skip the last entry */ 1138 Handle.Value += INDEX_TO_HANDLE_VALUE(1); 1139 } 1140 1141 /* Acquire the table lock and insert this new table into the list */ 1142 ExAcquirePushLockExclusive(&HandleTableListLock); 1143 InsertTailList(&HandleTableListHead, &NewTable->HandleTableList); 1144 ExReleasePushLockExclusive(&HandleTableListLock); 1145 1146 /* Leave the critical region we entered previously and return the table */ 1147 KeLeaveCriticalRegion(); 1148 return NewTable; 1149 } 1150 1151 BOOLEAN 1152 NTAPI 1153 ExChangeHandle(IN PHANDLE_TABLE HandleTable, 1154 IN HANDLE Handle, 1155 IN PEX_CHANGE_HANDLE_CALLBACK ChangeRoutine, 1156 IN ULONG_PTR Context) 1157 { 1158 EXHANDLE ExHandle; 1159 PHANDLE_TABLE_ENTRY HandleTableEntry; 1160 BOOLEAN Result = FALSE; 1161 PAGED_CODE(); 1162 1163 /* Set the handle value */ 1164 ExHandle.GenericHandleOverlay = Handle; 1165 1166 /* Find the entry for this handle */ 1167 HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, ExHandle); 1168 1169 /* Make sure that we found an entry, and that it's valid */ 1170 if (!(HandleTableEntry) || 1171 !(HandleTableEntry->Object) || 1172 (HandleTableEntry->NextFreeTableEntry == -2)) 1173 { 1174 /* It isn't, fail */ 1175 return FALSE; 1176 } 1177 1178 /* Enter a critical region */ 1179 KeEnterCriticalRegion(); 1180 1181 /* Try locking the handle entry */ 1182 if (ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 1183 { 1184 /* Call the change routine and unlock the entry */ 1185 Result = ChangeRoutine(HandleTableEntry, Context); 1186 ExUnlockHandleTableEntry(HandleTable, HandleTableEntry); 1187 } 1188 1189 /* Leave the critical region and return the callback result */ 1190 KeLeaveCriticalRegion(); 1191 return Result; 1192 } 1193 1194 VOID 1195 NTAPI 1196 ExSweepHandleTable(IN PHANDLE_TABLE HandleTable, 1197 IN PEX_SWEEP_HANDLE_CALLBACK EnumHandleProcedure, 1198 IN PVOID Context) 1199 { 1200 EXHANDLE Handle; 1201 PHANDLE_TABLE_ENTRY HandleTableEntry; 1202 PAGED_CODE(); 1203 1204 /* Set the initial value and loop the entries */ 1205 Handle.Value = INDEX_TO_HANDLE_VALUE(1); 1206 while ((HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, Handle))) 1207 { 1208 /* Loop each handle */ 1209 do 1210 { 1211 /* Lock the entry */ 1212 if (ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 1213 { 1214 /* Notify the callback routine */ 1215 EnumHandleProcedure(HandleTableEntry, 1216 Handle.GenericHandleOverlay, 1217 Context); 1218 } 1219 1220 /* Go to the next handle and entry */ 1221 Handle.Value += INDEX_TO_HANDLE_VALUE(1); 1222 HandleTableEntry++; 1223 } while (Handle.Value % INDEX_TO_HANDLE_VALUE(LOW_LEVEL_ENTRIES)); 1224 1225 /* Skip past the last entry */ 1226 Handle.Value += INDEX_TO_HANDLE_VALUE(1); 1227 } 1228 } 1229 1230 /* 1231 * @implemented 1232 */ 1233 BOOLEAN 1234 NTAPI 1235 ExEnumHandleTable(IN PHANDLE_TABLE HandleTable, 1236 IN PEX_ENUM_HANDLE_CALLBACK EnumHandleProcedure, 1237 IN OUT PVOID Context, 1238 OUT PHANDLE EnumHandle OPTIONAL) 1239 { 1240 EXHANDLE Handle; 1241 PHANDLE_TABLE_ENTRY HandleTableEntry; 1242 BOOLEAN Result = FALSE; 1243 PAGED_CODE(); 1244 1245 /* Enter a critical region */ 1246 KeEnterCriticalRegion(); 1247 1248 /* Set the initial value and loop the entries */ 1249 Handle.Value = 0; 1250 while ((HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, Handle))) 1251 { 1252 /* Validate the entry */ 1253 if ((HandleTableEntry->Object) && 1254 (HandleTableEntry->NextFreeTableEntry != -2)) 1255 { 1256 /* Lock the entry */ 1257 if (ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 1258 { 1259 /* Notify the callback routine */ 1260 Result = EnumHandleProcedure(HandleTableEntry, 1261 Handle.GenericHandleOverlay, 1262 Context); 1263 1264 /* Unlock it */ 1265 ExUnlockHandleTableEntry(HandleTable, HandleTableEntry); 1266 1267 /* Was this the one looked for? */ 1268 if (Result) 1269 { 1270 /* If so, return it if requested */ 1271 if (EnumHandle) *EnumHandle = Handle.GenericHandleOverlay; 1272 break; 1273 } 1274 } 1275 } 1276 1277 /* Go to the next entry */ 1278 Handle.Value += INDEX_TO_HANDLE_VALUE(1); 1279 } 1280 1281 /* Leave the critical region and return callback result */ 1282 KeLeaveCriticalRegion(); 1283 return Result; 1284 } 1285