1 /* 2 * PROJECT: ReactOS Kernel 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: ntoskrnl/config/cmapi.c 5 * PURPOSE: Configuration Manager - Internal Registry APIs 6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org) 7 * Eric Kohl 8 * Aleksandar Andrejevic <theflash AT sdf DOT lonestar DOT org> 9 */ 10 11 /* INCLUDES ******************************************************************/ 12 13 #include "ntoskrnl.h" 14 #define NDEBUG 15 #include "debug.h" 16 17 /* FUNCTIONS *****************************************************************/ 18 19 BOOLEAN 20 NTAPI 21 CmpIsHiveAlreadyLoaded(IN HANDLE KeyHandle, 22 IN POBJECT_ATTRIBUTES SourceFile, 23 OUT PCMHIVE *CmHive) 24 { 25 NTSTATUS Status; 26 PCM_KEY_BODY KeyBody; 27 PCMHIVE Hive; 28 BOOLEAN Loaded = FALSE; 29 PAGED_CODE(); 30 31 /* Sanity check */ 32 CMP_ASSERT_EXCLUSIVE_REGISTRY_LOCK(); 33 34 /* Reference the handle */ 35 Status = ObReferenceObjectByHandle(KeyHandle, 36 0, 37 CmpKeyObjectType, 38 KernelMode, 39 (PVOID)&KeyBody, 40 NULL); 41 if (!NT_SUCCESS(Status)) return Loaded; 42 43 /* Don't touch deleted KCBs */ 44 if (KeyBody->KeyControlBlock->Delete) return Loaded; 45 46 Hive = CONTAINING_RECORD(KeyBody->KeyControlBlock->KeyHive, CMHIVE, Hive); 47 48 /* Must be the root key */ 49 if (!(KeyBody->KeyControlBlock->Flags & KEY_HIVE_ENTRY) || 50 !(Hive->FileUserName.Buffer)) 51 { 52 /* It isn't */ 53 ObDereferenceObject(KeyBody); 54 return Loaded; 55 } 56 57 /* Now compare the name of the file */ 58 if (!RtlCompareUnicodeString(&Hive->FileUserName, 59 SourceFile->ObjectName, 60 TRUE)) 61 { 62 /* Same file found */ 63 Loaded = TRUE; 64 *CmHive = Hive; 65 66 /* If the hive is frozen, not sure what to do */ 67 if (Hive->Frozen) 68 { 69 /* FIXME: TODO */ 70 DPRINT1("ERROR: Hive is frozen\n"); 71 while (TRUE); 72 } 73 } 74 75 /* Dereference and return result */ 76 ObDereferenceObject(KeyBody); 77 return Loaded; 78 } 79 80 BOOLEAN 81 NTAPI 82 CmpDoFlushAll(IN BOOLEAN ForceFlush) 83 { 84 PLIST_ENTRY NextEntry; 85 PCMHIVE Hive; 86 NTSTATUS Status; 87 BOOLEAN Result = TRUE; 88 89 /* Make sure that the registry isn't read-only now */ 90 if (CmpNoWrite) return TRUE; 91 92 /* Otherwise, acquire the hive list lock and disable force flush */ 93 CmpForceForceFlush = FALSE; 94 ExAcquirePushLockShared(&CmpHiveListHeadLock); 95 96 /* Loop the hive list */ 97 NextEntry = CmpHiveListHead.Flink; 98 while (NextEntry != &CmpHiveListHead) 99 { 100 /* Get the hive */ 101 Hive = CONTAINING_RECORD(NextEntry, CMHIVE, HiveList); 102 if (!(Hive->Hive.HiveFlags & HIVE_NOLAZYFLUSH)) 103 { 104 /* Acquire the flusher lock */ 105 CmpLockHiveFlusherExclusive(Hive); 106 107 /* Check for illegal state */ 108 if ((ForceFlush) && (Hive->UseCount)) 109 { 110 /* Registry needs to be locked down */ 111 CMP_ASSERT_EXCLUSIVE_REGISTRY_LOCK(); 112 DPRINT1("FIXME: Hive is damaged and needs fixup\n"); 113 while (TRUE); 114 } 115 116 /* Only sync if we are forced to or if it won't cause a hive shrink */ 117 if ((ForceFlush) || (!HvHiveWillShrink(&Hive->Hive))) 118 { 119 /* Do the sync */ 120 Status = HvSyncHive(&Hive->Hive); 121 122 /* If something failed - set the flag and continue looping */ 123 if (!NT_SUCCESS(Status)) Result = FALSE; 124 } 125 else 126 { 127 /* We won't flush if the hive might shrink */ 128 Result = FALSE; 129 CmpForceForceFlush = TRUE; 130 } 131 132 /* Release the flusher lock */ 133 CmpUnlockHiveFlusher(Hive); 134 } 135 136 /* Try the next entry */ 137 NextEntry = NextEntry->Flink; 138 } 139 140 /* Release lock and return */ 141 ExReleasePushLock(&CmpHiveListHeadLock); 142 return Result; 143 } 144 145 NTSTATUS 146 NTAPI 147 CmpSetValueKeyNew(IN PHHIVE Hive, 148 IN PCM_KEY_NODE Parent, 149 IN PUNICODE_STRING ValueName, 150 IN ULONG Index, 151 IN ULONG Type, 152 IN PVOID Data, 153 IN ULONG DataSize, 154 IN ULONG StorageType, 155 IN ULONG SmallData) 156 { 157 PCELL_DATA CellData; 158 HCELL_INDEX ValueCell; 159 NTSTATUS Status; 160 161 /* Check if we already have a value list */ 162 if (Parent->ValueList.Count) 163 { 164 /* Then make sure it's valid and dirty it */ 165 ASSERT(Parent->ValueList.List != HCELL_NIL); 166 if (!HvMarkCellDirty(Hive, Parent->ValueList.List, FALSE)) 167 { 168 /* Fail if we're out of space for log changes */ 169 return STATUS_NO_LOG_SPACE; 170 } 171 } 172 173 /* Allocate a value cell */ 174 ValueCell = HvAllocateCell(Hive, 175 FIELD_OFFSET(CM_KEY_VALUE, Name) + 176 CmpNameSize(Hive, ValueName), 177 StorageType, 178 HCELL_NIL); 179 if (ValueCell == HCELL_NIL) return STATUS_INSUFFICIENT_RESOURCES; 180 181 /* Get the actual data for it */ 182 CellData = HvGetCell(Hive, ValueCell); 183 if (!CellData) ASSERT(FALSE); 184 185 /* Now we can release it, make sure it's also dirty */ 186 HvReleaseCell(Hive, ValueCell); 187 ASSERT(HvIsCellDirty(Hive, ValueCell)); 188 189 /* Set it up and copy the name */ 190 CellData->u.KeyValue.Signature = CM_KEY_VALUE_SIGNATURE; 191 _SEH2_TRY 192 { 193 /* This can crash since the name is coming from user-mode */ 194 CellData->u.KeyValue.NameLength = CmpCopyName(Hive, 195 CellData->u.KeyValue.Name, 196 ValueName); 197 } 198 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 199 { 200 /* Fail */ 201 DPRINT1("Invalid user data!\n"); 202 HvFreeCell(Hive, ValueCell); 203 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 204 } 205 _SEH2_END; 206 207 /* Check for compressed name */ 208 if (CellData->u.KeyValue.NameLength < ValueName->Length) 209 { 210 /* This is a compressed name */ 211 CellData->u.KeyValue.Flags = VALUE_COMP_NAME; 212 } 213 else 214 { 215 /* No flags to set */ 216 CellData->u.KeyValue.Flags = 0; 217 } 218 219 /* Check if this is a normal key */ 220 if (DataSize > CM_KEY_VALUE_SMALL) 221 { 222 /* Build a data cell for it */ 223 Status = CmpSetValueDataNew(Hive, 224 Data, 225 DataSize, 226 StorageType, 227 ValueCell, 228 &CellData->u.KeyValue.Data); 229 if (!NT_SUCCESS(Status)) 230 { 231 /* We failed, free the cell */ 232 HvFreeCell(Hive, ValueCell); 233 return Status; 234 } 235 236 /* Otherwise, set the data length, and make sure the data is dirty */ 237 CellData->u.KeyValue.DataLength = DataSize; 238 ASSERT(HvIsCellDirty(Hive, CellData->u.KeyValue.Data)); 239 } 240 else 241 { 242 /* This is a small key, set the data directly inside */ 243 CellData->u.KeyValue.DataLength = DataSize + CM_KEY_VALUE_SPECIAL_SIZE; 244 CellData->u.KeyValue.Data = SmallData; 245 } 246 247 /* Set the type now */ 248 CellData->u.KeyValue.Type = Type; 249 250 /* Add this value cell to the child list */ 251 Status = CmpAddValueToList(Hive, 252 ValueCell, 253 Index, 254 StorageType, 255 &Parent->ValueList); 256 257 /* If we failed, free the entire cell, including the data */ 258 if (!NT_SUCCESS(Status)) 259 { 260 /* Overwrite the status with a known one */ 261 CmpFreeValue(Hive, ValueCell); 262 Status = STATUS_INSUFFICIENT_RESOURCES; 263 } 264 265 /* Return Status */ 266 return Status; 267 } 268 269 NTSTATUS 270 NTAPI 271 CmpSetValueKeyExisting(IN PHHIVE Hive, 272 IN HCELL_INDEX OldChild, 273 IN PCM_KEY_VALUE Value, 274 IN ULONG Type, 275 IN PVOID Data, 276 IN ULONG DataSize, 277 IN ULONG StorageType, 278 IN ULONG TempData) 279 { 280 HCELL_INDEX DataCell, NewCell; 281 PCELL_DATA CellData; 282 ULONG Length; 283 BOOLEAN WasSmall, IsSmall; 284 285 /* Registry writes must be blocked */ 286 CMP_ASSERT_FLUSH_LOCK(Hive); 287 288 /* Mark the old child cell dirty */ 289 if (!HvMarkCellDirty(Hive, OldChild, FALSE)) return STATUS_NO_LOG_SPACE; 290 291 /* See if this is a small or normal key */ 292 WasSmall = CmpIsKeyValueSmall(&Length, Value->DataLength); 293 294 /* See if our new data can fit in a small key */ 295 IsSmall = (DataSize <= CM_KEY_VALUE_SMALL) ? TRUE: FALSE; 296 297 /* Big keys are unsupported */ 298 ASSERT_VALUE_BIG(Hive, Length); 299 ASSERT_VALUE_BIG(Hive, DataSize); 300 301 /* Mark the old value dirty */ 302 if (!CmpMarkValueDataDirty(Hive, Value)) return STATUS_NO_LOG_SPACE; 303 304 /* Check if we have a small key */ 305 if (IsSmall) 306 { 307 /* Check if we had a normal key with some data in it */ 308 if (!(WasSmall) && (Length > 0)) 309 { 310 /* Free the previous data */ 311 CmpFreeValueData(Hive, Value->Data, Length); 312 } 313 314 /* Write our data directly */ 315 Value->DataLength = DataSize + CM_KEY_VALUE_SPECIAL_SIZE; 316 Value->Data = TempData; 317 Value->Type = Type; 318 return STATUS_SUCCESS; 319 } 320 321 /* We have a normal key. Was the old cell also normal and had data? */ 322 if (!(WasSmall) && (Length > 0)) 323 { 324 /* Get the current data cell and actual data inside it */ 325 DataCell = Value->Data; 326 ASSERT(DataCell != HCELL_NIL); 327 CellData = HvGetCell(Hive, DataCell); 328 if (!CellData) return STATUS_INSUFFICIENT_RESOURCES; 329 330 /* Immediately release the cell */ 331 HvReleaseCell(Hive, DataCell); 332 333 /* Make sure that the data cell actually has a size */ 334 ASSERT(HvGetCellSize(Hive, CellData) > 0); 335 336 /* Check if the previous data cell could fit our new data */ 337 if (DataSize <= (ULONG)(HvGetCellSize(Hive, CellData))) 338 { 339 /* Re-use it then */ 340 NewCell = DataCell; 341 } 342 else 343 { 344 /* Otherwise, re-allocate the current data cell */ 345 NewCell = HvReallocateCell(Hive, DataCell, DataSize); 346 if (NewCell == HCELL_NIL) return STATUS_INSUFFICIENT_RESOURCES; 347 } 348 } 349 else 350 { 351 /* This was a small key, or a key with no data, allocate a cell */ 352 NewCell = HvAllocateCell(Hive, DataSize, StorageType, HCELL_NIL); 353 if (NewCell == HCELL_NIL) return STATUS_INSUFFICIENT_RESOURCES; 354 } 355 356 /* Now get the actual data for our data cell */ 357 CellData = HvGetCell(Hive, NewCell); 358 if (!CellData) ASSERT(FALSE); 359 360 /* Release it immediately */ 361 HvReleaseCell(Hive, NewCell); 362 363 /* Copy our data into the data cell's buffer, and set up the value */ 364 RtlCopyMemory(CellData, Data, DataSize); 365 Value->Data = NewCell; 366 Value->DataLength = DataSize; 367 Value->Type = Type; 368 369 /* Return success */ 370 ASSERT(HvIsCellDirty(Hive, NewCell)); 371 return STATUS_SUCCESS; 372 } 373 374 NTSTATUS 375 NTAPI 376 CmpQueryKeyData(IN PHHIVE Hive, 377 IN PCM_KEY_NODE Node, 378 IN KEY_INFORMATION_CLASS KeyInformationClass, 379 IN OUT PVOID KeyInformation, 380 IN ULONG Length, 381 IN OUT PULONG ResultLength) 382 { 383 NTSTATUS Status; 384 ULONG Size, SizeLeft, MinimumSize, Offset; 385 PKEY_INFORMATION Info = (PKEY_INFORMATION)KeyInformation; 386 USHORT NameLength; 387 PVOID ClassData; 388 389 /* Check if the value is compressed */ 390 if (Node->Flags & KEY_COMP_NAME) 391 { 392 /* Get the compressed name size */ 393 NameLength = CmpCompressedNameSize(Node->Name, Node->NameLength); 394 } 395 else 396 { 397 /* Get the real size */ 398 NameLength = Node->NameLength; 399 } 400 401 /* Check what kind of information is being requested */ 402 switch (KeyInformationClass) 403 { 404 /* Basic information */ 405 case KeyBasicInformation: 406 407 /* This is the size we need */ 408 Size = FIELD_OFFSET(KEY_BASIC_INFORMATION, Name) + NameLength; 409 410 /* And this is the minimum we can work with */ 411 MinimumSize = FIELD_OFFSET(KEY_BASIC_INFORMATION, Name); 412 413 /* Let the caller know and assume success */ 414 *ResultLength = Size; 415 Status = STATUS_SUCCESS; 416 417 /* Check if the bufer we got is too small */ 418 if (Length < MinimumSize) 419 { 420 /* Let the caller know and fail */ 421 Status = STATUS_BUFFER_TOO_SMALL; 422 break; 423 } 424 425 /* Copy the basic information */ 426 Info->KeyBasicInformation.LastWriteTime = Node->LastWriteTime; 427 Info->KeyBasicInformation.TitleIndex = 0; 428 Info->KeyBasicInformation.NameLength = NameLength; 429 430 /* Only the name is left */ 431 SizeLeft = Length - MinimumSize; 432 Size = NameLength; 433 434 /* Check if we don't have enough space for the name */ 435 if (SizeLeft < Size) 436 { 437 /* Truncate the name we'll return, and tell the caller */ 438 Size = SizeLeft; 439 Status = STATUS_BUFFER_OVERFLOW; 440 } 441 442 /* Check if this is a compressed key */ 443 if (Node->Flags & KEY_COMP_NAME) 444 { 445 /* Copy the compressed name */ 446 CmpCopyCompressedName(Info->KeyBasicInformation.Name, 447 SizeLeft, 448 Node->Name, 449 Node->NameLength); 450 } 451 else 452 { 453 /* Otherwise, copy the raw name */ 454 RtlCopyMemory(Info->KeyBasicInformation.Name, 455 Node->Name, 456 Size); 457 } 458 break; 459 460 /* Node information */ 461 case KeyNodeInformation: 462 463 /* Calculate the size we need */ 464 Size = FIELD_OFFSET(KEY_NODE_INFORMATION, Name) + 465 NameLength + 466 Node->ClassLength; 467 468 /* And the minimum size we can support */ 469 MinimumSize = FIELD_OFFSET(KEY_NODE_INFORMATION, Name); 470 471 /* Return the size to the caller and assume succes */ 472 *ResultLength = Size; 473 Status = STATUS_SUCCESS; 474 475 /* Check if the caller's buffer is too small */ 476 if (Length < MinimumSize) 477 { 478 /* Let them know, and fail */ 479 Status = STATUS_BUFFER_TOO_SMALL; 480 break; 481 } 482 483 /* Copy the basic information */ 484 Info->KeyNodeInformation.LastWriteTime = Node->LastWriteTime; 485 Info->KeyNodeInformation.TitleIndex = 0; 486 Info->KeyNodeInformation.ClassLength = Node->ClassLength; 487 Info->KeyNodeInformation.NameLength = NameLength; 488 489 /* Now the name is left */ 490 SizeLeft = Length - MinimumSize; 491 Size = NameLength; 492 493 /* Check if the name can fit entirely */ 494 if (SizeLeft < Size) 495 { 496 /* It can't, we'll have to truncate. Tell the caller */ 497 Size = SizeLeft; 498 Status = STATUS_BUFFER_OVERFLOW; 499 } 500 501 /* Check if the key node name is compressed */ 502 if (Node->Flags & KEY_COMP_NAME) 503 { 504 /* Copy the compressed name */ 505 CmpCopyCompressedName(Info->KeyNodeInformation.Name, 506 SizeLeft, 507 Node->Name, 508 Node->NameLength); 509 } 510 else 511 { 512 /* It isn't, so copy the raw name */ 513 RtlCopyMemory(Info->KeyNodeInformation.Name, 514 Node->Name, 515 Size); 516 } 517 518 /* Check if the node has a class */ 519 if (Node->ClassLength > 0) 520 { 521 /* Set the class offset */ 522 Offset = FIELD_OFFSET(KEY_NODE_INFORMATION, Name) + NameLength; 523 Offset = ALIGN_UP_BY(Offset, sizeof(ULONG)); 524 Info->KeyNodeInformation.ClassOffset = Offset; 525 526 /* Get the class data */ 527 ClassData = HvGetCell(Hive, Node->Class); 528 if (ClassData == NULL) 529 { 530 Status = STATUS_INSUFFICIENT_RESOURCES; 531 break; 532 } 533 534 /* Check if we can copy anything */ 535 if (Length > Offset) 536 { 537 /* Copy the class data */ 538 RtlCopyMemory((PUCHAR)Info + Offset, 539 ClassData, 540 min(Node->ClassLength, Length - Offset)); 541 } 542 543 /* Check if the buffer was large enough */ 544 if (Length < Offset + Node->ClassLength) 545 { 546 Status = STATUS_BUFFER_OVERFLOW; 547 } 548 549 /* Release the class cell */ 550 HvReleaseCell(Hive, Node->Class); 551 } 552 else 553 { 554 /* It doesn't, so set offset to -1, not 0! */ 555 Info->KeyNodeInformation.ClassOffset = 0xFFFFFFFF; 556 } 557 break; 558 559 /* Full information requsted */ 560 case KeyFullInformation: 561 562 /* This is the size we need */ 563 Size = FIELD_OFFSET(KEY_FULL_INFORMATION, Class) + 564 Node->ClassLength; 565 566 /* This is what we can work with */ 567 MinimumSize = FIELD_OFFSET(KEY_FULL_INFORMATION, Class); 568 569 /* Return it to caller and assume success */ 570 *ResultLength = Size; 571 Status = STATUS_SUCCESS; 572 573 /* Check if the caller's buffer is to small */ 574 if (Length < MinimumSize) 575 { 576 /* Let them know and fail */ 577 Status = STATUS_BUFFER_TOO_SMALL; 578 break; 579 } 580 581 /* Now copy all the basic information */ 582 Info->KeyFullInformation.LastWriteTime = Node->LastWriteTime; 583 Info->KeyFullInformation.TitleIndex = 0; 584 Info->KeyFullInformation.ClassLength = Node->ClassLength; 585 Info->KeyFullInformation.SubKeys = Node->SubKeyCounts[Stable] + 586 Node->SubKeyCounts[Volatile]; 587 Info->KeyFullInformation.Values = Node->ValueList.Count; 588 Info->KeyFullInformation.MaxNameLen = Node->MaxNameLen; 589 Info->KeyFullInformation.MaxClassLen = Node->MaxClassLen; 590 Info->KeyFullInformation.MaxValueNameLen = Node->MaxValueNameLen; 591 Info->KeyFullInformation.MaxValueDataLen = Node->MaxValueDataLen; 592 593 /* Check if we have a class */ 594 if (Node->ClassLength > 0) 595 { 596 /* Set the class offset */ 597 Offset = FIELD_OFFSET(KEY_FULL_INFORMATION, Class); 598 Info->KeyFullInformation.ClassOffset = Offset; 599 600 /* Get the class data */ 601 ClassData = HvGetCell(Hive, Node->Class); 602 if (ClassData == NULL) 603 { 604 Status = STATUS_INSUFFICIENT_RESOURCES; 605 break; 606 } 607 608 /* Copy the class data */ 609 ASSERT(Length >= Offset); 610 RtlCopyMemory(Info->KeyFullInformation.Class, 611 ClassData, 612 min(Node->ClassLength, Length - Offset)); 613 614 /* Check if the buffer was large enough */ 615 if (Length < Offset + Node->ClassLength) 616 { 617 Status = STATUS_BUFFER_OVERFLOW; 618 } 619 620 /* Release the class cell */ 621 HvReleaseCell(Hive, Node->Class); 622 } 623 else 624 { 625 /* We don't have a class, so set offset to -1, not 0! */ 626 Info->KeyFullInformation.ClassOffset = 0xFFFFFFFF; 627 } 628 break; 629 630 /* Any other class that got sent here is invalid! */ 631 default: 632 633 /* Set failure code */ 634 Status = STATUS_INVALID_PARAMETER; 635 break; 636 } 637 638 /* Return status */ 639 return Status; 640 } 641 642 NTSTATUS 643 NTAPI 644 CmSetValueKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 645 IN PUNICODE_STRING ValueName, 646 IN ULONG Type, 647 IN PVOID Data, 648 IN ULONG DataLength) 649 { 650 PHHIVE Hive = NULL; 651 PCM_KEY_NODE Parent; 652 PCM_KEY_VALUE Value = NULL; 653 HCELL_INDEX CurrentChild, Cell; 654 NTSTATUS Status; 655 BOOLEAN Found, Result; 656 ULONG Count, ChildIndex, SmallData, Storage; 657 VALUE_SEARCH_RETURN_TYPE SearchResult; 658 BOOLEAN FirstTry = TRUE, FlusherLocked = FALSE; 659 HCELL_INDEX ParentCell = HCELL_NIL, ChildCell = HCELL_NIL; 660 661 /* Acquire hive and KCB lock */ 662 CmpLockRegistry(); 663 CmpAcquireKcbLockShared(Kcb); 664 665 /* Sanity check */ 666 ASSERT(sizeof(ULONG) == CM_KEY_VALUE_SMALL); 667 668 /* Don't touch deleted KCBs */ 669 DoAgain: 670 if (Kcb->Delete) 671 { 672 /* Fail */ 673 Status = STATUS_KEY_DELETED; 674 goto Quickie; 675 } 676 677 /* Don't let anyone mess with symlinks */ 678 if ((Kcb->Flags & KEY_SYM_LINK) && 679 ((Type != REG_LINK) || 680 !(ValueName) || 681 !(RtlEqualUnicodeString(&CmSymbolicLinkValueName, ValueName, TRUE)))) 682 { 683 /* Invalid modification of a symlink key */ 684 Status = STATUS_ACCESS_DENIED; 685 goto Quickie; 686 } 687 688 /* Check if this is the first attempt */ 689 if (FirstTry) 690 { 691 /* Search for the value in the cache */ 692 SearchResult = CmpCompareNewValueDataAgainstKCBCache(Kcb, 693 ValueName, 694 Type, 695 Data, 696 DataLength); 697 if (SearchResult == SearchNeedExclusiveLock) 698 { 699 /* Try again with the exclusive lock */ 700 CmpConvertKcbSharedToExclusive(Kcb); 701 goto DoAgain; 702 } 703 else if (SearchResult == SearchSuccess) 704 { 705 /* We don't actually need to do anything! */ 706 Status = STATUS_SUCCESS; 707 goto Quickie; 708 } 709 710 /* We need the exclusive KCB lock now */ 711 if (!(CmpIsKcbLockedExclusive(Kcb)) && 712 !(CmpTryToConvertKcbSharedToExclusive(Kcb))) 713 { 714 /* Acquire exclusive lock */ 715 CmpConvertKcbSharedToExclusive(Kcb); 716 } 717 718 /* Cache lookup failed, so don't try it next time */ 719 FirstTry = FALSE; 720 721 /* Now grab the flush lock since the key will be modified */ 722 ASSERT(FlusherLocked == FALSE); 723 CmpLockHiveFlusherShared((PCMHIVE)Kcb->KeyHive); 724 FlusherLocked = TRUE; 725 goto DoAgain; 726 } 727 else 728 { 729 /* Get pointer to key cell */ 730 Hive = Kcb->KeyHive; 731 Cell = Kcb->KeyCell; 732 733 /* Get the parent */ 734 Parent = (PCM_KEY_NODE)HvGetCell(Hive, Cell); 735 ASSERT(Parent); 736 ParentCell = Cell; 737 738 /* Prepare to scan the key node */ 739 Count = Parent->ValueList.Count; 740 Found = FALSE; 741 if (Count > 0) 742 { 743 /* Try to find the existing name */ 744 Result = CmpFindNameInList(Hive, 745 &Parent->ValueList, 746 ValueName, 747 &ChildIndex, 748 &CurrentChild); 749 if (!Result) 750 { 751 /* Fail */ 752 Status = STATUS_INSUFFICIENT_RESOURCES; 753 goto Quickie; 754 } 755 756 /* Check if we found something */ 757 if (CurrentChild != HCELL_NIL) 758 { 759 /* Release existing child */ 760 if (ChildCell != HCELL_NIL) 761 { 762 HvReleaseCell(Hive, ChildCell); 763 ChildCell = HCELL_NIL; 764 } 765 766 /* Get its value */ 767 Value = (PCM_KEY_VALUE)HvGetCell(Hive, CurrentChild); 768 if (!Value) 769 { 770 /* Fail */ 771 Status = STATUS_INSUFFICIENT_RESOURCES; 772 goto Quickie; 773 } 774 775 /* Remember that we found it */ 776 ChildCell = CurrentChild; 777 Found = TRUE; 778 } 779 } 780 else 781 { 782 /* No child list, we'll need to add it */ 783 ChildIndex = 0; 784 } 785 } 786 787 /* Should only get here on the second pass */ 788 ASSERT(FirstTry == FALSE); 789 790 /* The KCB must be locked exclusive at this point */ 791 CMP_ASSERT_KCB_LOCK(Kcb); 792 793 /* Mark the cell dirty */ 794 if (!HvMarkCellDirty(Hive, Cell, FALSE)) 795 { 796 /* Not enough log space, fail */ 797 Status = STATUS_NO_LOG_SPACE; 798 goto Quickie; 799 } 800 801 /* Get the storage type */ 802 Storage = HvGetCellType(Cell); 803 804 /* Check if this is small data */ 805 SmallData = 0; 806 if ((DataLength <= CM_KEY_VALUE_SMALL) && (DataLength > 0)) 807 { 808 /* Need SEH because user data may be invalid */ 809 _SEH2_TRY 810 { 811 /* Copy it */ 812 RtlCopyMemory(&SmallData, Data, DataLength); 813 } 814 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 815 { 816 /* Return failure code */ 817 Status = _SEH2_GetExceptionCode(); 818 _SEH2_YIELD(goto Quickie); 819 } 820 _SEH2_END; 821 } 822 823 /* Check if we didn't find a matching key */ 824 if (!Found) 825 { 826 /* Call the internal routine */ 827 Status = CmpSetValueKeyNew(Hive, 828 Parent, 829 ValueName, 830 ChildIndex, 831 Type, 832 Data, 833 DataLength, 834 Storage, 835 SmallData); 836 } 837 else 838 { 839 /* Call the internal routine */ 840 Status = CmpSetValueKeyExisting(Hive, 841 CurrentChild, 842 Value, 843 Type, 844 Data, 845 DataLength, 846 Storage, 847 SmallData); 848 } 849 850 /* Check for success */ 851 if (NT_SUCCESS(Status)) 852 { 853 /* Check if the maximum value name length changed */ 854 ASSERT(Parent->MaxValueNameLen == Kcb->KcbMaxValueNameLen); 855 if (Parent->MaxValueNameLen < ValueName->Length) 856 { 857 /* Set the new values */ 858 Parent->MaxValueNameLen = ValueName->Length; 859 Kcb->KcbMaxValueNameLen = ValueName->Length; 860 } 861 862 /* Check if the maximum data length changed */ 863 ASSERT(Parent->MaxValueDataLen == Kcb->KcbMaxValueDataLen); 864 if (Parent->MaxValueDataLen < DataLength) 865 { 866 /* Update it */ 867 Parent->MaxValueDataLen = DataLength; 868 Kcb->KcbMaxValueDataLen = Parent->MaxValueDataLen; 869 } 870 871 /* Save the write time */ 872 KeQuerySystemTime(&Parent->LastWriteTime); 873 Kcb->KcbLastWriteTime = Parent->LastWriteTime; 874 875 /* Check if the cell is cached */ 876 if ((Found) && (CMP_IS_CELL_CACHED(Kcb->ValueCache.ValueList))) 877 { 878 /* Shouldn't happen */ 879 ASSERT(FALSE); 880 } 881 else 882 { 883 /* Cleanup the value cache */ 884 CmpCleanUpKcbValueCache(Kcb); 885 886 /* Sanity checks */ 887 ASSERT(!(CMP_IS_CELL_CACHED(Kcb->ValueCache.ValueList))); 888 ASSERT(!(Kcb->ExtFlags & CM_KCB_SYM_LINK_FOUND)); 889 890 /* Set the value cache */ 891 Kcb->ValueCache.Count = Parent->ValueList.Count; 892 Kcb->ValueCache.ValueList = Parent->ValueList.List; 893 } 894 895 /* Notify registered callbacks */ 896 CmpReportNotify(Kcb, 897 Hive, 898 Kcb->KeyCell, 899 REG_NOTIFY_CHANGE_LAST_SET); 900 } 901 902 /* Release the cells */ 903 Quickie: 904 if ((ParentCell != HCELL_NIL) && (Hive)) HvReleaseCell(Hive, ParentCell); 905 if ((ChildCell != HCELL_NIL) && (Hive)) HvReleaseCell(Hive, ChildCell); 906 907 /* Release the locks */ 908 if (FlusherLocked) CmpUnlockHiveFlusher((PCMHIVE)Hive); 909 CmpReleaseKcbLock(Kcb); 910 CmpUnlockRegistry(); 911 return Status; 912 } 913 914 NTSTATUS 915 NTAPI 916 CmDeleteValueKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 917 IN UNICODE_STRING ValueName) 918 { 919 NTSTATUS Status = STATUS_OBJECT_NAME_NOT_FOUND; 920 PHHIVE Hive; 921 PCM_KEY_NODE Parent; 922 HCELL_INDEX ChildCell, Cell; 923 PCHILD_LIST ChildList; 924 PCM_KEY_VALUE Value = NULL; 925 ULONG ChildIndex; 926 BOOLEAN Result; 927 928 /* Acquire hive lock */ 929 CmpLockRegistry(); 930 931 /* Lock KCB exclusively */ 932 CmpAcquireKcbLockExclusive(Kcb); 933 934 /* Don't touch deleted keys */ 935 if (Kcb->Delete) 936 { 937 /* Undo everything */ 938 CmpReleaseKcbLock(Kcb); 939 CmpUnlockRegistry(); 940 return STATUS_KEY_DELETED; 941 } 942 943 /* Get the hive and the cell index */ 944 Hive = Kcb->KeyHive; 945 Cell = Kcb->KeyCell; 946 947 /* Lock flushes */ 948 CmpLockHiveFlusherShared((PCMHIVE)Hive); 949 950 /* Get the parent key node */ 951 Parent = (PCM_KEY_NODE)HvGetCell(Hive, Cell); 952 ASSERT(Parent); 953 954 /* Get the value list and check if it has any entries */ 955 ChildList = &Parent->ValueList; 956 if (ChildList->Count) 957 { 958 /* Try to find this value */ 959 Result = CmpFindNameInList(Hive, 960 ChildList, 961 &ValueName, 962 &ChildIndex, 963 &ChildCell); 964 if (!Result) 965 { 966 /* Fail */ 967 Status = STATUS_INSUFFICIENT_RESOURCES; 968 goto Quickie; 969 } 970 971 /* Value not found, return error */ 972 if (ChildCell == HCELL_NIL) goto Quickie; 973 974 /* We found the value, mark all relevant cells dirty */ 975 if (!((HvMarkCellDirty(Hive, Cell, FALSE)) && 976 (HvMarkCellDirty(Hive, Parent->ValueList.List, FALSE)) && 977 (HvMarkCellDirty(Hive, ChildCell, FALSE)))) 978 { 979 /* Not enough log space, fail */ 980 Status = STATUS_NO_LOG_SPACE; 981 goto Quickie; 982 } 983 984 /* Get the key value */ 985 Value = (PCM_KEY_VALUE)HvGetCell(Hive, ChildCell); 986 ASSERT(Value); 987 988 /* Mark it and all related data as dirty */ 989 if (!CmpMarkValueDataDirty(Hive, Value)) 990 { 991 /* Not enough log space, fail */ 992 Status = STATUS_NO_LOG_SPACE; 993 goto Quickie; 994 } 995 996 /* Sanity checks */ 997 ASSERT(HvIsCellDirty(Hive, Parent->ValueList.List)); 998 ASSERT(HvIsCellDirty(Hive, ChildCell)); 999 1000 /* Remove the value from the child list */ 1001 Status = CmpRemoveValueFromList(Hive, ChildIndex, ChildList); 1002 if (!NT_SUCCESS(Status)) 1003 { 1004 /* Set known error */ 1005 Status = STATUS_INSUFFICIENT_RESOURCES; 1006 goto Quickie; 1007 } 1008 1009 /* Remove the value and its data itself */ 1010 if (!CmpFreeValue(Hive, ChildCell)) 1011 { 1012 /* Failed to free the value, fail */ 1013 Status = STATUS_INSUFFICIENT_RESOURCES; 1014 goto Quickie; 1015 } 1016 1017 /* Set the last write time */ 1018 KeQuerySystemTime(&Parent->LastWriteTime); 1019 Kcb->KcbLastWriteTime = Parent->LastWriteTime; 1020 1021 /* Sanity check */ 1022 ASSERT(Parent->MaxValueNameLen == Kcb->KcbMaxValueNameLen); 1023 ASSERT(Parent->MaxValueDataLen == Kcb->KcbMaxValueDataLen); 1024 ASSERT(HvIsCellDirty(Hive, Cell)); 1025 1026 /* Check if the value list is empty now */ 1027 if (!Parent->ValueList.Count) 1028 { 1029 /* Then clear key node data */ 1030 Parent->MaxValueNameLen = 0; 1031 Parent->MaxValueDataLen = 0; 1032 Kcb->KcbMaxValueNameLen = 0; 1033 Kcb->KcbMaxValueDataLen = 0; 1034 } 1035 1036 /* Cleanup the value cache */ 1037 CmpCleanUpKcbValueCache(Kcb); 1038 1039 /* Sanity checks */ 1040 ASSERT(!(CMP_IS_CELL_CACHED(Kcb->ValueCache.ValueList))); 1041 ASSERT(!(Kcb->ExtFlags & CM_KCB_SYM_LINK_FOUND)); 1042 1043 /* Set the value cache */ 1044 Kcb->ValueCache.Count = ChildList->Count; 1045 Kcb->ValueCache.ValueList = ChildList->List; 1046 1047 /* Notify registered callbacks */ 1048 CmpReportNotify(Kcb, Hive, Cell, REG_NOTIFY_CHANGE_LAST_SET); 1049 1050 /* Change default Status to success */ 1051 Status = STATUS_SUCCESS; 1052 } 1053 1054 Quickie: 1055 /* Release the parent cell, if any */ 1056 if (Parent) HvReleaseCell(Hive, Cell); 1057 1058 /* Check if we had a value */ 1059 if (Value) 1060 { 1061 /* Release the child cell */ 1062 ASSERT(ChildCell != HCELL_NIL); 1063 HvReleaseCell(Hive, ChildCell); 1064 } 1065 1066 /* Release locks */ 1067 CmpUnlockHiveFlusher((PCMHIVE)Hive); 1068 CmpReleaseKcbLock(Kcb); 1069 CmpUnlockRegistry(); 1070 return Status; 1071 } 1072 1073 NTSTATUS 1074 NTAPI 1075 CmQueryValueKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 1076 IN UNICODE_STRING ValueName, 1077 IN KEY_VALUE_INFORMATION_CLASS KeyValueInformationClass, 1078 IN PVOID KeyValueInformation, 1079 IN ULONG Length, 1080 IN PULONG ResultLength) 1081 { 1082 NTSTATUS Status; 1083 PCM_KEY_VALUE ValueData; 1084 ULONG Index; 1085 BOOLEAN ValueCached = FALSE; 1086 PCM_CACHED_VALUE *CachedValue; 1087 HCELL_INDEX CellToRelease; 1088 VALUE_SEARCH_RETURN_TYPE Result; 1089 PHHIVE Hive; 1090 PAGED_CODE(); 1091 1092 /* Acquire hive lock */ 1093 CmpLockRegistry(); 1094 1095 /* Lock the KCB shared */ 1096 CmpAcquireKcbLockShared(Kcb); 1097 1098 /* Don't touch deleted keys */ 1099 DoAgain: 1100 if (Kcb->Delete) 1101 { 1102 /* Undo everything */ 1103 CmpReleaseKcbLock(Kcb); 1104 CmpUnlockRegistry(); 1105 return STATUS_KEY_DELETED; 1106 } 1107 1108 /* We don't deal with this yet */ 1109 if (Kcb->ExtFlags & CM_KCB_SYM_LINK_FOUND) 1110 { 1111 /* Shouldn't happen */ 1112 ASSERT(FALSE); 1113 } 1114 1115 /* Get the hive */ 1116 Hive = Kcb->KeyHive; 1117 1118 /* Find the key value */ 1119 Result = CmpFindValueByNameFromCache(Kcb, 1120 &ValueName, 1121 &CachedValue, 1122 &Index, 1123 &ValueData, 1124 &ValueCached, 1125 &CellToRelease); 1126 if (Result == SearchNeedExclusiveLock) 1127 { 1128 /* Check if we need an exclusive lock */ 1129 ASSERT(CellToRelease == HCELL_NIL); 1130 ASSERT(ValueData == NULL); 1131 1132 /* Try with exclusive KCB lock */ 1133 CmpConvertKcbSharedToExclusive(Kcb); 1134 goto DoAgain; 1135 } 1136 1137 if (Result == SearchSuccess) 1138 { 1139 /* Sanity check */ 1140 ASSERT(ValueData != NULL); 1141 1142 /* User data, protect against exceptions */ 1143 _SEH2_TRY 1144 { 1145 /* Query the information requested */ 1146 Result = CmpQueryKeyValueData(Kcb, 1147 CachedValue, 1148 ValueData, 1149 ValueCached, 1150 KeyValueInformationClass, 1151 KeyValueInformation, 1152 Length, 1153 ResultLength, 1154 &Status); 1155 if (Result == SearchNeedExclusiveLock) 1156 { 1157 /* Release the value cell */ 1158 if (CellToRelease != HCELL_NIL) 1159 { 1160 HvReleaseCell(Hive, CellToRelease); 1161 CellToRelease = HCELL_NIL; 1162 } 1163 1164 /* Try with exclusive KCB lock */ 1165 CmpConvertKcbSharedToExclusive(Kcb); 1166 _SEH2_YIELD(goto DoAgain); 1167 } 1168 } 1169 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1170 { 1171 Status = _SEH2_GetExceptionCode(); 1172 } 1173 _SEH2_END; 1174 } 1175 else 1176 { 1177 /* Failed to find the value */ 1178 Status = STATUS_OBJECT_NAME_NOT_FOUND; 1179 } 1180 1181 /* If we have a cell to release, do so */ 1182 if (CellToRelease != HCELL_NIL) HvReleaseCell(Hive, CellToRelease); 1183 1184 /* Release locks */ 1185 CmpReleaseKcbLock(Kcb); 1186 CmpUnlockRegistry(); 1187 return Status; 1188 } 1189 1190 NTSTATUS 1191 NTAPI 1192 CmEnumerateValueKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 1193 IN ULONG Index, 1194 IN KEY_VALUE_INFORMATION_CLASS KeyValueInformationClass, 1195 IN PVOID KeyValueInformation, 1196 IN ULONG Length, 1197 IN PULONG ResultLength) 1198 { 1199 NTSTATUS Status; 1200 PHHIVE Hive; 1201 PCM_KEY_NODE Parent; 1202 HCELL_INDEX CellToRelease = HCELL_NIL, CellToRelease2 = HCELL_NIL; 1203 VALUE_SEARCH_RETURN_TYPE Result; 1204 BOOLEAN IndexIsCached, ValueIsCached = FALSE; 1205 PCELL_DATA CellData; 1206 PCM_CACHED_VALUE *CachedValue; 1207 PCM_KEY_VALUE ValueData = NULL; 1208 PAGED_CODE(); 1209 1210 /* Acquire hive lock */ 1211 CmpLockRegistry(); 1212 1213 /* Lock the KCB shared */ 1214 CmpAcquireKcbLockShared(Kcb); 1215 1216 /* Don't touch deleted keys */ 1217 DoAgain: 1218 if (Kcb->Delete) 1219 { 1220 /* Undo everything */ 1221 CmpReleaseKcbLock(Kcb); 1222 CmpUnlockRegistry(); 1223 return STATUS_KEY_DELETED; 1224 } 1225 1226 /* Get the hive and parent */ 1227 Hive = Kcb->KeyHive; 1228 Parent = (PCM_KEY_NODE)HvGetCell(Hive, Kcb->KeyCell); 1229 ASSERT(Parent); 1230 1231 /* FIXME: Lack of cache? */ 1232 if (Kcb->ValueCache.Count != Parent->ValueList.Count) 1233 { 1234 DPRINT1("HACK: Overriding value cache count\n"); 1235 Kcb->ValueCache.Count = Parent->ValueList.Count; 1236 } 1237 1238 /* Make sure the index is valid */ 1239 if (Index >= Kcb->ValueCache.Count) 1240 { 1241 /* Release the cell and fail */ 1242 HvReleaseCell(Hive, Kcb->KeyCell); 1243 Status = STATUS_NO_MORE_ENTRIES; 1244 goto Quickie; 1245 } 1246 1247 /* We don't deal with this yet */ 1248 if (Kcb->ExtFlags & CM_KCB_SYM_LINK_FOUND) 1249 { 1250 /* Shouldn't happen */ 1251 ASSERT(FALSE); 1252 } 1253 1254 /* Find the value list */ 1255 Result = CmpGetValueListFromCache(Kcb, 1256 &CellData, 1257 &IndexIsCached, 1258 &CellToRelease); 1259 if (Result == SearchNeedExclusiveLock) 1260 { 1261 /* Check if we need an exclusive lock */ 1262 ASSERT(CellToRelease == HCELL_NIL); 1263 HvReleaseCell(Hive, Kcb->KeyCell); 1264 1265 /* Try with exclusive KCB lock */ 1266 CmpConvertKcbSharedToExclusive(Kcb); 1267 goto DoAgain; 1268 } 1269 else if (Result != SearchSuccess) 1270 { 1271 /* Sanity check */ 1272 ASSERT(CellData == NULL); 1273 1274 /* Release the cell and fail */ 1275 Status = STATUS_INSUFFICIENT_RESOURCES; 1276 goto Quickie; 1277 } 1278 1279 /* Now get the key value */ 1280 Result = CmpGetValueKeyFromCache(Kcb, 1281 CellData, 1282 Index, 1283 &CachedValue, 1284 &ValueData, 1285 IndexIsCached, 1286 &ValueIsCached, 1287 &CellToRelease2); 1288 if (Result == SearchNeedExclusiveLock) 1289 { 1290 /* Cleanup state */ 1291 ASSERT(CellToRelease2 == HCELL_NIL); 1292 if (CellToRelease) 1293 { 1294 HvReleaseCell(Hive, CellToRelease); 1295 CellToRelease = HCELL_NIL; 1296 } 1297 HvReleaseCell(Hive, Kcb->KeyCell); 1298 1299 /* Try with exclusive KCB lock */ 1300 CmpConvertKcbSharedToExclusive(Kcb); 1301 goto DoAgain; 1302 } 1303 else if (Result != SearchSuccess) 1304 { 1305 /* Sanity check */ 1306 ASSERT(ValueData == NULL); 1307 1308 /* Release the cells and fail */ 1309 Status = STATUS_INSUFFICIENT_RESOURCES; 1310 goto Quickie; 1311 } 1312 1313 /* User data, need SEH */ 1314 _SEH2_TRY 1315 { 1316 /* Query the information requested */ 1317 Result = CmpQueryKeyValueData(Kcb, 1318 CachedValue, 1319 ValueData, 1320 ValueIsCached, 1321 KeyValueInformationClass, 1322 KeyValueInformation, 1323 Length, 1324 ResultLength, 1325 &Status); 1326 if (Result == SearchNeedExclusiveLock) 1327 { 1328 /* Cleanup state */ 1329 if (CellToRelease2) HvReleaseCell(Hive, CellToRelease2); 1330 HvReleaseCell(Hive, Kcb->KeyCell); 1331 if (CellToRelease) HvReleaseCell(Hive, CellToRelease); 1332 1333 /* Try with exclusive KCB lock */ 1334 CmpConvertKcbSharedToExclusive(Kcb); 1335 _SEH2_YIELD(goto DoAgain); 1336 } 1337 } 1338 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1339 { 1340 /* Get exception code */ 1341 Status = _SEH2_GetExceptionCode(); 1342 } 1343 _SEH2_END; 1344 1345 Quickie: 1346 /* If we have a cell to release, do so */ 1347 if (CellToRelease != HCELL_NIL) HvReleaseCell(Hive, CellToRelease); 1348 1349 /* Release the parent cell */ 1350 HvReleaseCell(Hive, Kcb->KeyCell); 1351 1352 /* If we have a cell to release, do so */ 1353 if (CellToRelease2 != HCELL_NIL) HvReleaseCell(Hive, CellToRelease2); 1354 1355 /* Release locks */ 1356 CmpReleaseKcbLock(Kcb); 1357 CmpUnlockRegistry(); 1358 return Status; 1359 } 1360 1361 static 1362 NTSTATUS 1363 CmpQueryKeyDataFromCache( 1364 _In_ PCM_KEY_CONTROL_BLOCK Kcb, 1365 _Out_ PKEY_CACHED_INFORMATION KeyCachedInfo, 1366 _In_ ULONG Length, 1367 _Out_ PULONG ResultLength) 1368 { 1369 PCM_KEY_NODE Node; 1370 PHHIVE KeyHive; 1371 HCELL_INDEX KeyCell; 1372 USHORT NameLength; 1373 PAGED_CODE(); 1374 1375 /* Get the hive and cell index */ 1376 KeyHive = Kcb->KeyHash.KeyHive; 1377 KeyCell = Kcb->KeyHash.KeyCell; 1378 1379 #if DBG 1380 /* Get the cell node */ 1381 Node = HvGetCell(KeyHive, KeyCell); 1382 if (Node != NULL) 1383 { 1384 ULONG SubKeyCount; 1385 ASSERT(Node->ValueList.Count == Kcb->ValueCache.Count); 1386 1387 if (!(Kcb->ExtFlags & CM_KCB_INVALID_CACHED_INFO)) 1388 { 1389 SubKeyCount = Node->SubKeyCounts[0] + Node->SubKeyCounts[1]; 1390 if (Kcb->ExtFlags & CM_KCB_NO_SUBKEY) 1391 { 1392 ASSERT(SubKeyCount == 0); 1393 } 1394 else if (Kcb->ExtFlags & CM_KCB_SUBKEY_ONE) 1395 { 1396 ASSERT(SubKeyCount == 1); 1397 } 1398 else if (Kcb->ExtFlags & CM_KCB_SUBKEY_HINT) 1399 { 1400 ASSERT(SubKeyCount == Kcb->IndexHint->Count); 1401 } 1402 else 1403 { 1404 ASSERT(SubKeyCount == Kcb->SubKeyCount); 1405 } 1406 } 1407 1408 ASSERT(Node->LastWriteTime.QuadPart == Kcb->KcbLastWriteTime.QuadPart); 1409 ASSERT(Node->MaxNameLen == Kcb->KcbMaxNameLen); 1410 ASSERT(Node->MaxValueNameLen == Kcb->KcbMaxValueNameLen); 1411 ASSERT(Node->MaxValueDataLen == Kcb->KcbMaxValueDataLen); 1412 1413 /* Release the cell */ 1414 HvReleaseCell(KeyHive, KeyCell); 1415 } 1416 #endif // DBG 1417 1418 /* Make sure we have a name block */ 1419 if (Kcb->NameBlock == NULL) 1420 { 1421 return STATUS_INSUFFICIENT_RESOURCES; 1422 } 1423 1424 /* Check for compressed name */ 1425 if (Kcb->NameBlock->Compressed) 1426 { 1427 /* Calculate the name size */ 1428 NameLength = CmpCompressedNameSize(Kcb->NameBlock->NameHash.Name, 1429 Kcb->NameBlock->NameHash.NameLength); 1430 } 1431 else 1432 { 1433 /* Use the stored name size */ 1434 NameLength = Kcb->NameBlock->NameHash.NameLength; 1435 } 1436 1437 /* Validate buffer length (we do not copy the name!) */ 1438 *ResultLength = sizeof(*KeyCachedInfo); 1439 if (Length < *ResultLength) 1440 { 1441 return STATUS_BUFFER_TOO_SMALL; 1442 } 1443 1444 /* Fill the structure */ 1445 KeyCachedInfo->LastWriteTime = Kcb->KcbLastWriteTime; 1446 KeyCachedInfo->TitleIndex = 0; 1447 KeyCachedInfo->NameLength = NameLength; 1448 KeyCachedInfo->Values = Kcb->ValueCache.Count; 1449 KeyCachedInfo->MaxNameLen = Kcb->KcbMaxNameLen; 1450 KeyCachedInfo->MaxValueNameLen = Kcb->KcbMaxValueNameLen; 1451 KeyCachedInfo->MaxValueDataLen = Kcb->KcbMaxValueDataLen; 1452 1453 /* Check the ExtFlags for what we have */ 1454 if (Kcb->ExtFlags & CM_KCB_INVALID_CACHED_INFO) 1455 { 1456 /* Cache is not valid, do a full lookup */ 1457 DPRINT1("Kcb cache incoherency detected, kcb = %p\n", Kcb); 1458 1459 /* Get the cell node */ 1460 Node = HvGetCell(KeyHive, KeyCell); 1461 if (Node == NULL) 1462 { 1463 return STATUS_INSUFFICIENT_RESOURCES; 1464 } 1465 1466 /* Calculate number of subkeys */ 1467 KeyCachedInfo->SubKeys = Node->SubKeyCounts[0] + Node->SubKeyCounts[1]; 1468 1469 /* Release the cell */ 1470 HvReleaseCell(KeyHive, KeyCell); 1471 } 1472 else if (Kcb->ExtFlags & CM_KCB_NO_SUBKEY) 1473 { 1474 /* There are no subkeys */ 1475 KeyCachedInfo->SubKeys = 0; 1476 } 1477 else if (Kcb->ExtFlags & CM_KCB_SUBKEY_ONE) 1478 { 1479 /* There is exactly one subley */ 1480 KeyCachedInfo->SubKeys = 1; 1481 } 1482 else if (Kcb->ExtFlags & CM_KCB_SUBKEY_HINT) 1483 { 1484 /* Get the number of subkeys from the subkey hint */ 1485 KeyCachedInfo->SubKeys = Kcb->IndexHint->Count; 1486 } 1487 else 1488 { 1489 /* No subkey hint, use the key count field */ 1490 KeyCachedInfo->SubKeys = Kcb->SubKeyCount; 1491 } 1492 1493 return STATUS_SUCCESS; 1494 } 1495 1496 static 1497 NTSTATUS 1498 CmpQueryFlagsInformation( 1499 _In_ PCM_KEY_CONTROL_BLOCK Kcb, 1500 _Out_ PKEY_USER_FLAGS_INFORMATION KeyFlagsInfo, 1501 _In_ ULONG Length, 1502 _In_ PULONG ResultLength) 1503 { 1504 /* Validate the buffer size */ 1505 *ResultLength = sizeof(*KeyFlagsInfo); 1506 if (Length < *ResultLength) 1507 { 1508 return STATUS_BUFFER_TOO_SMALL; 1509 } 1510 1511 /* Copy the user flags */ 1512 KeyFlagsInfo->UserFlags = Kcb->KcbUserFlags; 1513 1514 return STATUS_SUCCESS; 1515 } 1516 1517 static 1518 NTSTATUS 1519 CmpQueryNameInformation( 1520 _In_ PCM_KEY_CONTROL_BLOCK Kcb, 1521 _Out_opt_ PKEY_NAME_INFORMATION KeyNameInfo, 1522 _In_ ULONG Length, 1523 _Out_ PULONG ResultLength) 1524 { 1525 ULONG NeededLength; 1526 PCM_KEY_CONTROL_BLOCK CurrentKcb; 1527 1528 NeededLength = 0; 1529 CurrentKcb = Kcb; 1530 1531 /* Count the needed buffer size */ 1532 while (CurrentKcb) 1533 { 1534 if (CurrentKcb->NameBlock->Compressed) 1535 NeededLength += CmpCompressedNameSize(CurrentKcb->NameBlock->Name, CurrentKcb->NameBlock->NameLength); 1536 else 1537 NeededLength += CurrentKcb->NameBlock->NameLength; 1538 1539 NeededLength += sizeof(OBJ_NAME_PATH_SEPARATOR); 1540 1541 CurrentKcb = CurrentKcb->ParentKcb; 1542 } 1543 1544 _SEH2_TRY 1545 { 1546 *ResultLength = FIELD_OFFSET(KEY_NAME_INFORMATION, Name) + NeededLength; 1547 if (Length < RTL_SIZEOF_THROUGH_FIELD(KEY_NAME_INFORMATION, NameLength)) 1548 _SEH2_YIELD(return STATUS_BUFFER_TOO_SMALL); 1549 if (Length < *ResultLength) 1550 { 1551 KeyNameInfo->NameLength = NeededLength; 1552 _SEH2_YIELD(return STATUS_BUFFER_OVERFLOW); 1553 } 1554 } 1555 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1556 { 1557 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 1558 } 1559 _SEH2_END; 1560 1561 /* Do the real copy */ 1562 KeyNameInfo->NameLength = 0; 1563 CurrentKcb = Kcb; 1564 1565 _SEH2_TRY 1566 { 1567 while (CurrentKcb) 1568 { 1569 ULONG NameLength; 1570 1571 if (CurrentKcb->NameBlock->Compressed) 1572 { 1573 NameLength = CmpCompressedNameSize(CurrentKcb->NameBlock->Name, CurrentKcb->NameBlock->NameLength); 1574 /* Copy the compressed name */ 1575 CmpCopyCompressedName(&KeyNameInfo->Name[(NeededLength - NameLength)/sizeof(WCHAR)], 1576 NameLength, 1577 CurrentKcb->NameBlock->Name, 1578 CurrentKcb->NameBlock->NameLength); 1579 } 1580 else 1581 { 1582 NameLength = CurrentKcb->NameBlock->NameLength; 1583 /* Otherwise, copy the raw name */ 1584 RtlCopyMemory(&KeyNameInfo->Name[(NeededLength - NameLength)/sizeof(WCHAR)], 1585 CurrentKcb->NameBlock->Name, 1586 NameLength); 1587 } 1588 1589 NeededLength -= NameLength; 1590 NeededLength -= sizeof(OBJ_NAME_PATH_SEPARATOR); 1591 /* Add path separator */ 1592 KeyNameInfo->Name[NeededLength/sizeof(WCHAR)] = OBJ_NAME_PATH_SEPARATOR; 1593 KeyNameInfo->NameLength += NameLength + sizeof(OBJ_NAME_PATH_SEPARATOR); 1594 1595 CurrentKcb = CurrentKcb->ParentKcb; 1596 } 1597 } 1598 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1599 { 1600 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 1601 } 1602 _SEH2_END; 1603 1604 /* Make sure we copied everything */ 1605 ASSERT(NeededLength == 0); 1606 ASSERT(KeyNameInfo->Name[0] == OBJ_NAME_PATH_SEPARATOR); 1607 1608 /* We're done */ 1609 return STATUS_SUCCESS; 1610 } 1611 1612 1613 NTSTATUS 1614 NTAPI 1615 CmQueryKey(_In_ PCM_KEY_CONTROL_BLOCK Kcb, 1616 _In_ KEY_INFORMATION_CLASS KeyInformationClass, 1617 _Out_opt_ PVOID KeyInformation, 1618 _In_ ULONG Length, 1619 _Out_ PULONG ResultLength) 1620 { 1621 NTSTATUS Status; 1622 PHHIVE Hive; 1623 PCM_KEY_NODE Parent; 1624 HV_TRACK_CELL_REF CellReferences = {0}; 1625 1626 /* Acquire hive lock */ 1627 CmpLockRegistry(); 1628 1629 /* Lock KCB shared */ 1630 CmpAcquireKcbLockShared(Kcb); 1631 1632 /* Don't touch deleted keys */ 1633 if (Kcb->Delete) 1634 { 1635 /* Fail */ 1636 Status = STATUS_KEY_DELETED; 1637 goto Quickie; 1638 } 1639 1640 /* Data can be user-mode, use SEH */ 1641 _SEH2_TRY 1642 { 1643 /* Check what class we got */ 1644 switch (KeyInformationClass) 1645 { 1646 /* Typical information */ 1647 case KeyFullInformation: 1648 case KeyBasicInformation: 1649 case KeyNodeInformation: 1650 { 1651 /* Get the hive and parent */ 1652 Hive = Kcb->KeyHive; 1653 Parent = (PCM_KEY_NODE)HvGetCell(Hive, Kcb->KeyCell); 1654 ASSERT(Parent); 1655 1656 /* Track cell references */ 1657 if (!HvTrackCellRef(&CellReferences, Hive, Kcb->KeyCell)) 1658 { 1659 /* Not enough memory to track references */ 1660 Status = STATUS_INSUFFICIENT_RESOURCES; 1661 } 1662 else 1663 { 1664 /* Call the internal API */ 1665 Status = CmpQueryKeyData(Hive, 1666 Parent, 1667 KeyInformationClass, 1668 KeyInformation, 1669 Length, 1670 ResultLength); 1671 } 1672 break; 1673 } 1674 1675 case KeyCachedInformation: 1676 { 1677 /* Call the internal API */ 1678 Status = CmpQueryKeyDataFromCache(Kcb, 1679 KeyInformation, 1680 Length, 1681 ResultLength); 1682 break; 1683 } 1684 1685 case KeyFlagsInformation: 1686 { 1687 /* Call the internal API */ 1688 Status = CmpQueryFlagsInformation(Kcb, 1689 KeyInformation, 1690 Length, 1691 ResultLength); 1692 break; 1693 } 1694 1695 case KeyNameInformation: 1696 { 1697 /* Call the internal API */ 1698 Status = CmpQueryNameInformation(Kcb, 1699 KeyInformation, 1700 Length, 1701 ResultLength); 1702 break; 1703 } 1704 1705 /* Illegal classes */ 1706 default: 1707 { 1708 /* Print message and fail */ 1709 DPRINT1("Unsupported class: %d!\n", KeyInformationClass); 1710 Status = STATUS_INVALID_INFO_CLASS; 1711 break; 1712 } 1713 } 1714 } 1715 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1716 { 1717 /* Fail with exception code */ 1718 Status = _SEH2_GetExceptionCode(); 1719 _SEH2_YIELD(goto Quickie); 1720 } 1721 _SEH2_END; 1722 1723 Quickie: 1724 /* Release references */ 1725 HvReleaseFreeCellRefArray(&CellReferences); 1726 1727 /* Release locks */ 1728 CmpReleaseKcbLock(Kcb); 1729 CmpUnlockRegistry(); 1730 return Status; 1731 } 1732 1733 NTSTATUS 1734 NTAPI 1735 CmEnumerateKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 1736 IN ULONG Index, 1737 IN KEY_INFORMATION_CLASS KeyInformationClass, 1738 IN PVOID KeyInformation, 1739 IN ULONG Length, 1740 IN PULONG ResultLength) 1741 { 1742 NTSTATUS Status; 1743 PHHIVE Hive; 1744 PCM_KEY_NODE Parent, Child; 1745 HCELL_INDEX ChildCell; 1746 HV_TRACK_CELL_REF CellReferences = {0}; 1747 1748 /* Acquire hive lock */ 1749 CmpLockRegistry(); 1750 1751 /* Lock the KCB shared */ 1752 CmpAcquireKcbLockShared(Kcb); 1753 1754 /* Don't touch deleted keys */ 1755 if (Kcb->Delete) 1756 { 1757 /* Undo everything */ 1758 Status = STATUS_KEY_DELETED; 1759 goto Quickie; 1760 } 1761 1762 /* Get the hive and parent */ 1763 Hive = Kcb->KeyHive; 1764 Parent = (PCM_KEY_NODE)HvGetCell(Hive, Kcb->KeyCell); 1765 ASSERT(Parent); 1766 1767 /* Get the child cell */ 1768 ChildCell = CmpFindSubKeyByNumber(Hive, Parent, Index); 1769 1770 /* Release the parent cell */ 1771 HvReleaseCell(Hive, Kcb->KeyCell); 1772 1773 /* Check if we found the child */ 1774 if (ChildCell == HCELL_NIL) 1775 { 1776 /* We didn't, fail */ 1777 Status = STATUS_NO_MORE_ENTRIES; 1778 goto Quickie; 1779 } 1780 1781 /* Now get the actual child node */ 1782 Child = (PCM_KEY_NODE)HvGetCell(Hive, ChildCell); 1783 ASSERT(Child); 1784 1785 /* Track references */ 1786 if (!HvTrackCellRef(&CellReferences, Hive, ChildCell)) 1787 { 1788 /* Can't allocate memory for tracking */ 1789 Status = STATUS_INSUFFICIENT_RESOURCES; 1790 goto Quickie; 1791 } 1792 1793 /* Data can be user-mode, use SEH */ 1794 _SEH2_TRY 1795 { 1796 /* Query the data requested */ 1797 Status = CmpQueryKeyData(Hive, 1798 Child, 1799 KeyInformationClass, 1800 KeyInformation, 1801 Length, 1802 ResultLength); 1803 } 1804 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1805 { 1806 /* Fail with exception code */ 1807 Status = _SEH2_GetExceptionCode(); 1808 _SEH2_YIELD(goto Quickie); 1809 } 1810 _SEH2_END; 1811 1812 Quickie: 1813 /* Release references */ 1814 HvReleaseFreeCellRefArray(&CellReferences); 1815 1816 /* Release locks */ 1817 CmpReleaseKcbLock(Kcb); 1818 CmpUnlockRegistry(); 1819 return Status; 1820 } 1821 1822 NTSTATUS 1823 NTAPI 1824 CmDeleteKey(IN PCM_KEY_BODY KeyBody) 1825 { 1826 NTSTATUS Status; 1827 PHHIVE Hive; 1828 PCM_KEY_NODE Node, Parent; 1829 HCELL_INDEX Cell, ParentCell; 1830 PCM_KEY_CONTROL_BLOCK Kcb; 1831 1832 /* Acquire hive lock */ 1833 CmpLockRegistry(); 1834 1835 /* Get the kcb */ 1836 Kcb = KeyBody->KeyControlBlock; 1837 1838 /* Don't allow deleting the root */ 1839 if (!Kcb->ParentKcb) 1840 { 1841 /* Fail */ 1842 CmpUnlockRegistry(); 1843 return STATUS_CANNOT_DELETE; 1844 } 1845 1846 /* Lock parent and child */ 1847 CmpAcquireTwoKcbLocksExclusiveByKey(Kcb->ConvKey, Kcb->ParentKcb->ConvKey); 1848 1849 /* Check if we're already being deleted */ 1850 if (Kcb->Delete) 1851 { 1852 /* Don't do it twice */ 1853 Status = STATUS_SUCCESS; 1854 goto Quickie; 1855 } 1856 1857 /* Get the hive and node */ 1858 Hive = Kcb->KeyHive; 1859 Cell = Kcb->KeyCell; 1860 1861 /* Lock flushes */ 1862 CmpLockHiveFlusherShared((PCMHIVE)Hive); 1863 1864 /* Get the key node */ 1865 Node = (PCM_KEY_NODE)HvGetCell(Hive, Cell); 1866 ASSERT(Node); 1867 1868 /* Sanity check */ 1869 ASSERT(Node->Flags == Kcb->Flags); 1870 1871 /* Check if we don't have any children */ 1872 if (!(Node->SubKeyCounts[Stable] + Node->SubKeyCounts[Volatile]) && 1873 !(Node->Flags & KEY_NO_DELETE)) 1874 { 1875 /* Send notification to registered callbacks */ 1876 CmpReportNotify(Kcb, Hive, Cell, REG_NOTIFY_CHANGE_NAME); 1877 1878 /* Get the parent and free the cell */ 1879 ParentCell = Node->Parent; 1880 Status = CmpFreeKeyByCell(Hive, Cell, TRUE); 1881 if (NT_SUCCESS(Status)) 1882 { 1883 /* Flush any notifications */ 1884 CmpFlushNotifiesOnKeyBodyList(Kcb, FALSE); 1885 1886 /* Clean up information we have on the subkey */ 1887 CmpCleanUpSubKeyInfo(Kcb->ParentKcb); 1888 1889 /* Get the parent node */ 1890 Parent = (PCM_KEY_NODE)HvGetCell(Hive, ParentCell); 1891 if (Parent) 1892 { 1893 /* Update the maximum name length */ 1894 Kcb->ParentKcb->KcbMaxNameLen = (USHORT)Parent->MaxNameLen; 1895 1896 /* Make sure we're dirty */ 1897 ASSERT(HvIsCellDirty(Hive, ParentCell)); 1898 1899 /* Update the write time */ 1900 KeQuerySystemTime(&Parent->LastWriteTime); 1901 Kcb->ParentKcb->KcbLastWriteTime = Parent->LastWriteTime; 1902 1903 /* Release the cell */ 1904 HvReleaseCell(Hive, ParentCell); 1905 } 1906 1907 /* Set the KCB in delete mode and remove it */ 1908 Kcb->Delete = TRUE; 1909 CmpRemoveKeyControlBlock(Kcb); 1910 1911 /* Clear the cell */ 1912 Kcb->KeyCell = HCELL_NIL; 1913 } 1914 } 1915 else 1916 { 1917 /* Fail */ 1918 Status = STATUS_CANNOT_DELETE; 1919 } 1920 1921 /* Release the cell */ 1922 HvReleaseCell(Hive, Cell); 1923 1924 /* Release flush lock */ 1925 CmpUnlockHiveFlusher((PCMHIVE)Hive); 1926 1927 /* Release the KCB locks */ 1928 Quickie: 1929 CmpReleaseTwoKcbLockByKey(Kcb->ConvKey, Kcb->ParentKcb->ConvKey); 1930 1931 /* Release hive lock */ 1932 CmpUnlockRegistry(); 1933 return Status; 1934 } 1935 1936 NTSTATUS 1937 NTAPI 1938 CmFlushKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 1939 IN BOOLEAN ExclusiveLock) 1940 { 1941 PCMHIVE CmHive; 1942 NTSTATUS Status = STATUS_SUCCESS; 1943 PHHIVE Hive; 1944 1945 /* Ignore flushes until we're ready */ 1946 if (CmpNoWrite) return STATUS_SUCCESS; 1947 1948 /* Get the hives */ 1949 Hive = Kcb->KeyHive; 1950 CmHive = (PCMHIVE)Hive; 1951 1952 /* Check if this is the master hive */ 1953 if (CmHive == CmiVolatileHive) 1954 { 1955 /* Flush all the hives instead */ 1956 CmpDoFlushAll(FALSE); 1957 } 1958 else 1959 { 1960 /* Don't touch the hive */ 1961 CmpLockHiveFlusherExclusive(CmHive); 1962 ASSERT(CmHive->ViewLock); 1963 KeAcquireGuardedMutex(CmHive->ViewLock); 1964 CmHive->ViewLockOwner = KeGetCurrentThread(); 1965 1966 /* Will the hive shrink? */ 1967 if (HvHiveWillShrink(Hive)) 1968 { 1969 /* I don't believe the current Hv does shrinking */ 1970 ASSERT(FALSE); 1971 } 1972 else 1973 { 1974 /* Now we can release views */ 1975 ASSERT(CmHive->ViewLock); 1976 CMP_ASSERT_EXCLUSIVE_REGISTRY_LOCK_OR_LOADING(CmHive); 1977 ASSERT(KeGetCurrentThread() == CmHive->ViewLockOwner); 1978 KeReleaseGuardedMutex(CmHive->ViewLock); 1979 } 1980 1981 /* Flush only this hive */ 1982 if (!HvSyncHive(Hive)) 1983 { 1984 /* Fail */ 1985 Status = STATUS_REGISTRY_IO_FAILED; 1986 } 1987 1988 /* Release the flush lock */ 1989 CmpUnlockHiveFlusher(CmHive); 1990 } 1991 1992 /* Return the status */ 1993 return Status; 1994 } 1995 1996 NTSTATUS 1997 NTAPI 1998 CmLoadKey(IN POBJECT_ATTRIBUTES TargetKey, 1999 IN POBJECT_ATTRIBUTES SourceFile, 2000 IN ULONG Flags, 2001 IN PCM_KEY_BODY KeyBody) 2002 { 2003 SECURITY_QUALITY_OF_SERVICE ServiceQos; 2004 SECURITY_CLIENT_CONTEXT ClientSecurityContext; 2005 HANDLE KeyHandle; 2006 BOOLEAN Allocate = TRUE; 2007 PCMHIVE CmHive, LoadedHive; 2008 NTSTATUS Status; 2009 CM_PARSE_CONTEXT ParseContext; 2010 2011 /* Check if we have a trust key */ 2012 if (KeyBody) 2013 { 2014 /* Fail */ 2015 DPRINT("Trusted classes not yet supported\n"); 2016 } 2017 2018 /* Build a service QoS for a security context */ 2019 ServiceQos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE); 2020 ServiceQos.ImpersonationLevel = SecurityImpersonation; 2021 ServiceQos.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING; 2022 ServiceQos.EffectiveOnly = TRUE; 2023 Status = SeCreateClientSecurity(PsGetCurrentThread(), 2024 &ServiceQos, 2025 FALSE, 2026 &ClientSecurityContext); 2027 if (!NT_SUCCESS(Status)) 2028 { 2029 /* Fail */ 2030 DPRINT1("Security context failed\n"); 2031 return Status; 2032 } 2033 2034 /* Open the target key */ 2035 RtlZeroMemory(&ParseContext, sizeof(ParseContext)); 2036 ParseContext.CreateOperation = FALSE; 2037 Status = ObOpenObjectByName(TargetKey, 2038 CmpKeyObjectType, 2039 KernelMode, 2040 NULL, 2041 KEY_READ, 2042 &ParseContext, 2043 &KeyHandle); 2044 if (!NT_SUCCESS(Status)) KeyHandle = NULL; 2045 2046 /* Open the hive */ 2047 Status = CmpCmdHiveOpen(SourceFile, 2048 &ClientSecurityContext, 2049 &Allocate, 2050 &CmHive, 2051 0); 2052 2053 /* Get rid of the security context */ 2054 SeDeleteClientSecurity(&ClientSecurityContext); 2055 2056 /* See if we failed */ 2057 if (!NT_SUCCESS(Status)) 2058 { 2059 /* See if the target already existed */ 2060 if (KeyHandle) 2061 { 2062 /* Lock the registry */ 2063 CmpLockRegistryExclusive(); 2064 2065 /* Check if we are already loaded */ 2066 if (CmpIsHiveAlreadyLoaded(KeyHandle, SourceFile, &LoadedHive)) 2067 { 2068 /* That's okay then */ 2069 ASSERT(LoadedHive); 2070 Status = STATUS_SUCCESS; 2071 } 2072 2073 /* Release the registry */ 2074 CmpUnlockRegistry(); 2075 } 2076 2077 /* Close the key handle if we had one */ 2078 if (KeyHandle) ZwClose(KeyHandle); 2079 return Status; 2080 } 2081 2082 /* Lock the registry shared */ 2083 CmpLockRegistry(); 2084 2085 /* Lock loading */ 2086 ExAcquirePushLockExclusive(&CmpLoadHiveLock); 2087 2088 /* Lock the hive to this thread */ 2089 CmHive->Hive.HiveFlags |= HIVE_IS_UNLOADING; 2090 CmHive->CreatorOwner = KeGetCurrentThread(); 2091 2092 /* Set flag */ 2093 if (Flags & REG_NO_LAZY_FLUSH) CmHive->Hive.HiveFlags |= HIVE_NOLAZYFLUSH; 2094 2095 /* Link the hive */ 2096 Status = CmpLinkHiveToMaster(TargetKey->ObjectName, 2097 TargetKey->RootDirectory, 2098 CmHive, 2099 Allocate, 2100 TargetKey->SecurityDescriptor); 2101 if (NT_SUCCESS(Status)) 2102 { 2103 /* Add to HiveList key */ 2104 CmpAddToHiveFileList(CmHive); 2105 2106 /* Sync the hive if necessary */ 2107 if (Allocate) 2108 { 2109 /* Sync it under the flusher lock */ 2110 CmpLockHiveFlusherExclusive(CmHive); 2111 HvSyncHive(&CmHive->Hive); 2112 CmpUnlockHiveFlusher(CmHive); 2113 } 2114 2115 /* Release the hive */ 2116 CmHive->Hive.HiveFlags &= ~HIVE_IS_UNLOADING; 2117 CmHive->CreatorOwner = NULL; 2118 2119 /* Allow loads */ 2120 ExReleasePushLock(&CmpLoadHiveLock); 2121 } 2122 else 2123 { 2124 DPRINT1("CmpLinkHiveToMaster failed, Status %lx\n", Status); 2125 /* FIXME: TODO */ 2126 ASSERT(FALSE); 2127 } 2128 2129 /* Is this first profile load? */ 2130 if (!CmpProfileLoaded && !CmpWasSetupBoot) 2131 { 2132 /* User is now logged on, set quotas */ 2133 CmpProfileLoaded = TRUE; 2134 CmpSetGlobalQuotaAllowed(); 2135 } 2136 2137 /* Unlock the registry */ 2138 CmpUnlockRegistry(); 2139 2140 /* Close handle and return */ 2141 if (KeyHandle) ZwClose(KeyHandle); 2142 return Status; 2143 } 2144 2145 static 2146 BOOLEAN 2147 NTAPI 2148 CmpUnlinkHiveFromMaster(IN PCMHIVE CmHive, 2149 IN HCELL_INDEX Cell) 2150 { 2151 PCELL_DATA CellData; 2152 HCELL_INDEX LinkCell; 2153 NTSTATUS Status; 2154 2155 DPRINT("CmpUnlinkHiveFromMaster()\n"); 2156 2157 /* Get the cell data */ 2158 CellData = HvGetCell(&CmHive->Hive, Cell); 2159 if (CellData == NULL) 2160 return FALSE; 2161 2162 /* Get the link cell and release the current cell */ 2163 LinkCell = CellData->u.KeyNode.Parent; 2164 HvReleaseCell(&CmHive->Hive, Cell); 2165 2166 /* Remove the link cell from the master hive */ 2167 CmpLockHiveFlusherExclusive(CmiVolatileHive); 2168 Status = CmpFreeKeyByCell((PHHIVE)CmiVolatileHive, 2169 LinkCell, 2170 TRUE); 2171 CmpUnlockHiveFlusher(CmiVolatileHive); 2172 if (!NT_SUCCESS(Status)) 2173 { 2174 DPRINT1("CmpFreeKeyByCell() failed (Status 0x%08lx)\n", Status); 2175 return FALSE; 2176 } 2177 2178 /* Remove the hive from the list */ 2179 ExAcquirePushLockExclusive(&CmpHiveListHeadLock); 2180 RemoveEntryList(&CmHive->HiveList); 2181 ExReleasePushLock(&CmpHiveListHeadLock); 2182 2183 return TRUE; 2184 } 2185 2186 NTSTATUS 2187 NTAPI 2188 CmUnloadKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 2189 IN ULONG Flags) 2190 { 2191 PHHIVE Hive; 2192 PCMHIVE CmHive; 2193 HCELL_INDEX Cell; 2194 2195 DPRINT("CmUnloadKey(%p, %lx)\n", Kcb, Flags); 2196 2197 /* Get the hive */ 2198 Hive = Kcb->KeyHive; 2199 Cell = Kcb->KeyCell; 2200 CmHive = (PCMHIVE)Hive; 2201 2202 /* Fail if the key is not a hive root key */ 2203 if (Cell != Hive->BaseBlock->RootCell) 2204 { 2205 DPRINT1("Key is not a hive root key!\n"); 2206 return STATUS_INVALID_PARAMETER; 2207 } 2208 2209 /* Fail if we try to unload the master hive */ 2210 if (CmHive == CmiVolatileHive) 2211 { 2212 DPRINT1("Do not try to unload the master hive!\n"); 2213 return STATUS_INVALID_PARAMETER; 2214 } 2215 2216 /* Mark this hive as being unloaded */ 2217 Hive->HiveFlags |= HIVE_IS_UNLOADING; 2218 2219 /* Search for any opened keys in this hive, and take an appropriate action */ 2220 if (Kcb->RefCount > 1) 2221 { 2222 if (Flags != REG_FORCE_UNLOAD) 2223 { 2224 if (CmCountOpenSubKeys(Kcb, FALSE) != 0) 2225 { 2226 /* There are open subkeys but we don't force hive unloading, fail */ 2227 Hive->HiveFlags &= ~HIVE_IS_UNLOADING; 2228 return STATUS_CANNOT_DELETE; 2229 } 2230 } 2231 else 2232 { 2233 DPRINT1("CmUnloadKey: Force unloading is UNIMPLEMENTED, expect dangling KCBs problems!\n"); 2234 } 2235 } 2236 2237 /* Flush the hive */ 2238 CmFlushKey(Kcb, TRUE); 2239 2240 /* Unlink the hive from the master hive */ 2241 if (!CmpUnlinkHiveFromMaster(CmHive, Cell)) 2242 { 2243 DPRINT("CmpUnlinkHiveFromMaster() failed!\n"); 2244 2245 /* Remove the unloading flag and return failure */ 2246 Hive->HiveFlags &= ~HIVE_IS_UNLOADING; 2247 return STATUS_INSUFFICIENT_RESOURCES; 2248 } 2249 2250 /* Clean up information we have on the subkey */ 2251 CmpCleanUpSubKeyInfo(Kcb->ParentKcb); 2252 2253 /* Set the KCB in delete mode and remove it */ 2254 Kcb->Delete = TRUE; 2255 CmpRemoveKeyControlBlock(Kcb); 2256 2257 if (Flags != REG_FORCE_UNLOAD) 2258 { 2259 /* Release the KCB locks */ 2260 CmpReleaseTwoKcbLockByKey(Kcb->ConvKey, Kcb->ParentKcb->ConvKey); 2261 2262 /* Release the hive loading lock */ 2263 ExReleasePushLockExclusive(&CmpLoadHiveLock); 2264 } 2265 2266 /* Release hive lock */ 2267 CmpUnlockRegistry(); 2268 2269 /* Close file handles */ 2270 CmpCloseHiveFiles(CmHive); 2271 2272 /* Remove the hive from the hive file list */ 2273 CmpRemoveFromHiveFileList(CmHive); 2274 2275 /* Destroy the security descriptor cache */ 2276 CmpDestroySecurityCache(CmHive); 2277 2278 /* Destroy the view list */ 2279 CmpDestroyHiveViewList(CmHive); 2280 2281 /* Delete the flusher lock */ 2282 ExDeleteResourceLite(CmHive->FlusherLock); 2283 ExFreePoolWithTag(CmHive->FlusherLock, TAG_CMHIVE); 2284 2285 /* Delete the view lock */ 2286 ExFreePoolWithTag(CmHive->ViewLock, TAG_CMHIVE); 2287 2288 /* Free the hive storage */ 2289 HvFree(Hive); 2290 2291 /* Free the hive */ 2292 CmpFree(CmHive, TAG_CM); 2293 2294 return STATUS_SUCCESS; 2295 } 2296 2297 ULONG 2298 NTAPI 2299 CmCountOpenSubKeys(IN PCM_KEY_CONTROL_BLOCK RootKcb, 2300 IN BOOLEAN RemoveEmptyCacheEntries) 2301 { 2302 PCM_KEY_HASH Entry; 2303 PCM_KEY_CONTROL_BLOCK CachedKcb; 2304 PCM_KEY_CONTROL_BLOCK ParentKcb; 2305 ULONG ParentKeyCount; 2306 ULONG i, j; 2307 ULONG SubKeys = 0; 2308 2309 DPRINT("CmCountOpenSubKeys() called\n"); 2310 2311 /* The root key is the only referenced key. There are no refereced sub keys. */ 2312 if (RootKcb->RefCount == 1) 2313 { 2314 DPRINT("Open sub keys: 0\n"); 2315 return 0; 2316 } 2317 2318 /* Enumerate all hash lists */ 2319 for (i = 0; i < CmpHashTableSize; i++) 2320 { 2321 /* Get the first cache entry */ 2322 Entry = CmpCacheTable[i].Entry; 2323 2324 /* Enumerate all cache entries */ 2325 while (Entry) 2326 { 2327 /* Get the KCB of the current cache entry */ 2328 CachedKcb = CONTAINING_RECORD(Entry, CM_KEY_CONTROL_BLOCK, KeyHash); 2329 2330 /* Check keys only that are subkeys to our root key */ 2331 if (CachedKcb->TotalLevels > RootKcb->TotalLevels) 2332 { 2333 /* Calculate the number of parent keys to the root key */ 2334 ParentKeyCount = CachedKcb->TotalLevels - RootKcb->TotalLevels; 2335 2336 /* Find a parent key that could be the root key */ 2337 ParentKcb = CachedKcb; 2338 for (j = 0; j < ParentKeyCount; j++) 2339 { 2340 ParentKcb = ParentKcb->ParentKcb; 2341 } 2342 2343 /* Check whether the parent is the root key */ 2344 if (ParentKcb == RootKcb) 2345 { 2346 DPRINT("Found a sub key, RefCount = %u\n", CachedKcb->RefCount); 2347 2348 if (CachedKcb->RefCount > 0) 2349 { 2350 /* Count the current hash entry if it is in use */ 2351 SubKeys++; 2352 } 2353 else if ((CachedKcb->RefCount == 0) && (RemoveEmptyCacheEntries != FALSE)) 2354 { 2355 /* Remove the current key from the delayed close list */ 2356 CmpRemoveFromDelayedClose(CachedKcb); 2357 2358 /* Remove the current cache entry */ 2359 CmpCleanUpKcbCacheWithLock(CachedKcb, TRUE); 2360 2361 /* Restart, because the hash list has changed */ 2362 Entry = CmpCacheTable[i].Entry; 2363 continue; 2364 } 2365 } 2366 } 2367 2368 /* Get the next cache entry */ 2369 Entry = Entry->NextHash; 2370 } 2371 } 2372 2373 DPRINT("Open sub keys: %u\n", SubKeys); 2374 return SubKeys; 2375 } 2376 2377 static 2378 NTSTATUS 2379 CmpDeepCopyKeyInternal(IN PHHIVE SourceHive, 2380 IN HCELL_INDEX SrcKeyCell, 2381 IN PHHIVE DestinationHive, 2382 IN HCELL_INDEX Parent, 2383 IN HSTORAGE_TYPE StorageType, 2384 OUT PHCELL_INDEX DestKeyCell OPTIONAL) 2385 { 2386 NTSTATUS Status; 2387 PCM_KEY_NODE SrcNode; 2388 PCM_KEY_NODE DestNode = NULL; 2389 HCELL_INDEX NewKeyCell = HCELL_NIL; 2390 HCELL_INDEX NewClassCell = HCELL_NIL, NewSecCell = HCELL_NIL; 2391 HCELL_INDEX SubKey, NewSubKey; 2392 ULONG Index, SubKeyCount; 2393 2394 PAGED_CODE(); 2395 2396 DPRINT("CmpDeepCopyKeyInternal(0x%08X, 0x%08X, 0x%08X, 0x%08X, 0x%08X, 0x%08X)\n", 2397 SourceHive, 2398 SrcKeyCell, 2399 DestinationHive, 2400 Parent, 2401 StorageType, 2402 DestKeyCell); 2403 2404 /* Get the source cell node */ 2405 SrcNode = HvGetCell(SourceHive, SrcKeyCell); 2406 ASSERT(SrcNode); 2407 2408 /* Sanity check */ 2409 ASSERT(SrcNode->Signature == CM_KEY_NODE_SIGNATURE); 2410 2411 /* Create a simple copy of the source key */ 2412 NewKeyCell = CmpCopyCell(SourceHive, 2413 SrcKeyCell, 2414 DestinationHive, 2415 StorageType); 2416 if (NewKeyCell == HCELL_NIL) 2417 { 2418 /* Not enough storage space */ 2419 Status = STATUS_INSUFFICIENT_RESOURCES; 2420 goto Cleanup; 2421 } 2422 2423 /* Get the destination cell node */ 2424 DestNode = HvGetCell(DestinationHive, NewKeyCell); 2425 ASSERT(DestNode); 2426 2427 /* Set the parent and copy the flags */ 2428 DestNode->Parent = Parent; 2429 DestNode->Flags = (SrcNode->Flags & KEY_COMP_NAME); // Keep only the single permanent flag 2430 if (Parent == HCELL_NIL) 2431 { 2432 /* This is the new root node */ 2433 DestNode->Flags |= KEY_HIVE_ENTRY | KEY_NO_DELETE; 2434 } 2435 2436 /* Copy the class cell */ 2437 if (SrcNode->ClassLength > 0) 2438 { 2439 NewClassCell = CmpCopyCell(SourceHive, 2440 SrcNode->Class, 2441 DestinationHive, 2442 StorageType); 2443 if (NewClassCell == HCELL_NIL) 2444 { 2445 /* Not enough storage space */ 2446 Status = STATUS_INSUFFICIENT_RESOURCES; 2447 goto Cleanup; 2448 } 2449 2450 DestNode->Class = NewClassCell; 2451 DestNode->ClassLength = SrcNode->ClassLength; 2452 } 2453 else 2454 { 2455 DestNode->Class = HCELL_NIL; 2456 DestNode->ClassLength = 0; 2457 } 2458 2459 /* Copy the security cell (FIXME: HACKish poor-man version) */ 2460 if (SrcNode->Security != HCELL_NIL) 2461 { 2462 NewSecCell = CmpCopyCell(SourceHive, 2463 SrcNode->Security, 2464 DestinationHive, 2465 StorageType); 2466 if (NewSecCell == HCELL_NIL) 2467 { 2468 /* Not enough storage space */ 2469 Status = STATUS_INSUFFICIENT_RESOURCES; 2470 goto Cleanup; 2471 } 2472 } 2473 DestNode->Security = NewSecCell; 2474 2475 /* Copy the value list */ 2476 Status = CmpCopyKeyValueList(SourceHive, 2477 &SrcNode->ValueList, 2478 DestinationHive, 2479 &DestNode->ValueList, 2480 StorageType); 2481 if (!NT_SUCCESS(Status)) 2482 goto Cleanup; 2483 2484 /* Clear the invalid subkey index */ 2485 DestNode->SubKeyCounts[Stable] = DestNode->SubKeyCounts[Volatile] = 0; 2486 DestNode->SubKeyLists[Stable] = DestNode->SubKeyLists[Volatile] = HCELL_NIL; 2487 2488 /* Calculate the total number of subkeys */ 2489 SubKeyCount = SrcNode->SubKeyCounts[Stable] + SrcNode->SubKeyCounts[Volatile]; 2490 2491 /* Loop through all the subkeys */ 2492 for (Index = 0; Index < SubKeyCount; Index++) 2493 { 2494 /* Get the subkey */ 2495 SubKey = CmpFindSubKeyByNumber(SourceHive, SrcNode, Index); 2496 ASSERT(SubKey != HCELL_NIL); 2497 2498 /* Call the function recursively for the subkey */ 2499 // 2500 // FIXME: Danger!! Kernel stack exhaustion!! 2501 // 2502 Status = CmpDeepCopyKeyInternal(SourceHive, 2503 SubKey, 2504 DestinationHive, 2505 NewKeyCell, 2506 StorageType, 2507 &NewSubKey); 2508 if (!NT_SUCCESS(Status)) 2509 goto Cleanup; 2510 2511 /* Add the copy of the subkey to the new key */ 2512 if (!CmpAddSubKey(DestinationHive, 2513 NewKeyCell, 2514 NewSubKey)) 2515 { 2516 /* Cleanup allocated cell */ 2517 HvFreeCell(DestinationHive, NewSubKey); 2518 2519 Status = STATUS_INSUFFICIENT_RESOURCES; 2520 goto Cleanup; 2521 } 2522 } 2523 2524 /* Set success */ 2525 Status = STATUS_SUCCESS; 2526 2527 Cleanup: 2528 2529 /* Release the cells */ 2530 if (DestNode) HvReleaseCell(DestinationHive, NewKeyCell); 2531 if (SrcNode) HvReleaseCell(SourceHive, SrcKeyCell); 2532 2533 /* Cleanup allocated cells in case of failure */ 2534 if (!NT_SUCCESS(Status)) 2535 { 2536 if (NewSecCell != HCELL_NIL) 2537 HvFreeCell(DestinationHive, NewSecCell); 2538 2539 if (NewClassCell != HCELL_NIL) 2540 HvFreeCell(DestinationHive, NewClassCell); 2541 2542 if (NewKeyCell != HCELL_NIL) 2543 HvFreeCell(DestinationHive, NewKeyCell); 2544 2545 NewKeyCell = HCELL_NIL; 2546 } 2547 2548 /* Set the cell index if requested and return status */ 2549 if (DestKeyCell) *DestKeyCell = NewKeyCell; 2550 return Status; 2551 } 2552 2553 NTSTATUS 2554 NTAPI 2555 CmpDeepCopyKey(IN PHHIVE SourceHive, 2556 IN HCELL_INDEX SrcKeyCell, 2557 IN PHHIVE DestinationHive, 2558 IN HSTORAGE_TYPE StorageType, 2559 OUT PHCELL_INDEX DestKeyCell OPTIONAL) 2560 { 2561 /* Call the internal function */ 2562 return CmpDeepCopyKeyInternal(SourceHive, 2563 SrcKeyCell, 2564 DestinationHive, 2565 HCELL_NIL, 2566 StorageType, 2567 DestKeyCell); 2568 } 2569 2570 NTSTATUS 2571 NTAPI 2572 CmSaveKey(IN PCM_KEY_CONTROL_BLOCK Kcb, 2573 IN HANDLE FileHandle, 2574 IN ULONG Flags) 2575 { 2576 NTSTATUS Status = STATUS_SUCCESS; 2577 PCMHIVE KeyHive = NULL; 2578 PAGED_CODE(); 2579 2580 DPRINT("CmSaveKey(0x%08X, 0x%08X, %lu)\n", Kcb, FileHandle, Flags); 2581 2582 /* Lock the registry and KCB */ 2583 CmpLockRegistry(); 2584 CmpAcquireKcbLockShared(Kcb); 2585 2586 if (Kcb->Delete) 2587 { 2588 /* The source key has been deleted, do nothing */ 2589 Status = STATUS_KEY_DELETED; 2590 goto Cleanup; 2591 } 2592 2593 if (Kcb->KeyHive == &CmiVolatileHive->Hive) 2594 { 2595 /* Keys that are directly in the master hive can't be saved */ 2596 Status = STATUS_ACCESS_DENIED; 2597 goto Cleanup; 2598 } 2599 2600 /* Create a new hive that will hold the key */ 2601 Status = CmpInitializeHive(&KeyHive, 2602 HINIT_CREATE, 2603 HIVE_VOLATILE, 2604 HFILE_TYPE_PRIMARY, 2605 NULL, 2606 NULL, 2607 NULL, 2608 NULL, 2609 NULL, 2610 0); 2611 if (!NT_SUCCESS(Status)) goto Cleanup; 2612 2613 /* Copy the key recursively into the new hive */ 2614 Status = CmpDeepCopyKey(Kcb->KeyHive, 2615 Kcb->KeyCell, 2616 &KeyHive->Hive, 2617 Stable, 2618 &KeyHive->Hive.BaseBlock->RootCell); 2619 if (!NT_SUCCESS(Status)) goto Cleanup; 2620 2621 /* Set the primary handle of the hive */ 2622 KeyHive->FileHandles[HFILE_TYPE_PRIMARY] = FileHandle; 2623 2624 /* Dump the hive into the file */ 2625 HvWriteHive(&KeyHive->Hive); 2626 2627 Cleanup: 2628 2629 /* Free the hive */ 2630 if (KeyHive) CmpDestroyHive(KeyHive); 2631 2632 /* Release the locks */ 2633 CmpReleaseKcbLock(Kcb); 2634 CmpUnlockRegistry(); 2635 2636 return Status; 2637 } 2638 2639 NTSTATUS 2640 NTAPI 2641 CmSaveMergedKeys(IN PCM_KEY_CONTROL_BLOCK HighKcb, 2642 IN PCM_KEY_CONTROL_BLOCK LowKcb, 2643 IN HANDLE FileHandle) 2644 { 2645 PCMHIVE KeyHive = NULL; 2646 NTSTATUS Status = STATUS_SUCCESS; 2647 2648 PAGED_CODE(); 2649 2650 DPRINT("CmSaveKey(%p, %p, %p)\n", HighKcb, LowKcb, FileHandle); 2651 2652 /* Lock the registry and the KCBs */ 2653 CmpLockRegistry(); 2654 CmpAcquireKcbLockShared(HighKcb); 2655 CmpAcquireKcbLockShared(LowKcb); 2656 2657 if (LowKcb->Delete || HighKcb->Delete) 2658 { 2659 /* The source key has been deleted, do nothing */ 2660 Status = STATUS_KEY_DELETED; 2661 goto done; 2662 } 2663 2664 /* Create a new hive that will hold the key */ 2665 Status = CmpInitializeHive(&KeyHive, 2666 HINIT_CREATE, 2667 HIVE_VOLATILE, 2668 HFILE_TYPE_PRIMARY, 2669 NULL, 2670 NULL, 2671 NULL, 2672 NULL, 2673 NULL, 2674 0); 2675 if (!NT_SUCCESS(Status)) 2676 goto done; 2677 2678 /* Copy the low precedence key recursively into the new hive */ 2679 Status = CmpDeepCopyKey(LowKcb->KeyHive, 2680 LowKcb->KeyCell, 2681 &KeyHive->Hive, 2682 Stable, 2683 &KeyHive->Hive.BaseBlock->RootCell); 2684 if (!NT_SUCCESS(Status)) 2685 goto done; 2686 2687 /* Copy the high precedence key recursively into the new hive */ 2688 Status = CmpDeepCopyKey(HighKcb->KeyHive, 2689 HighKcb->KeyCell, 2690 &KeyHive->Hive, 2691 Stable, 2692 &KeyHive->Hive.BaseBlock->RootCell); 2693 if (!NT_SUCCESS(Status)) 2694 goto done; 2695 2696 /* Set the primary handle of the hive */ 2697 KeyHive->FileHandles[HFILE_TYPE_PRIMARY] = FileHandle; 2698 2699 /* Dump the hive into the file */ 2700 HvWriteHive(&KeyHive->Hive); 2701 2702 done: 2703 /* Free the hive */ 2704 if (KeyHive) 2705 CmpDestroyHive(KeyHive); 2706 2707 /* Release the locks */ 2708 CmpReleaseKcbLock(LowKcb); 2709 CmpReleaseKcbLock(HighKcb); 2710 CmpUnlockRegistry(); 2711 2712 return Status; 2713 } 2714