1 /* 2 * PROJECT: ReactOS Kernel 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: ntoskrnl/config/ntapi.c 5 * PURPOSE: Configuration Manager - Internal Registry APIs 6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org) 7 * Eric Kohl 8 */ 9 10 /* INCLUDES ******************************************************************/ 11 12 #include "ntoskrnl.h" 13 #define NDEBUG 14 #include "debug.h" 15 16 BOOLEAN CmBootAcceptFirstTime = TRUE; 17 BOOLEAN CmFirstTime = TRUE; 18 extern ULONG InitSafeBootMode; 19 20 21 /* PRIVATE FUNCTIONS *********************************************************/ 22 23 /* 24 * Adapted from ntoskrnl/include/internal/ob_x.h:ObpReleaseObjectCreateInformation() 25 */ 26 VOID 27 ReleaseCapturedObjectAttributes( 28 _In_ POBJECT_ATTRIBUTES CapturedObjectAttributes, 29 _In_ KPROCESSOR_MODE AccessMode) 30 { 31 /* Check if we have a security descriptor */ 32 if (CapturedObjectAttributes->SecurityDescriptor) 33 { 34 /* Release it */ 35 SeReleaseSecurityDescriptor(CapturedObjectAttributes->SecurityDescriptor, 36 AccessMode, 37 TRUE); 38 CapturedObjectAttributes->SecurityDescriptor = NULL; 39 } 40 41 /* Check if we have an object name */ 42 if (CapturedObjectAttributes->ObjectName) 43 { 44 /* Release it */ 45 ReleaseCapturedUnicodeString(CapturedObjectAttributes->ObjectName, AccessMode); 46 } 47 } 48 49 /* 50 * Adapted from ntoskrnl/ob/oblife.c:ObpCaptureObjectCreateInformation() 51 */ 52 NTSTATUS 53 ProbeAndCaptureObjectAttributes( 54 _Out_ POBJECT_ATTRIBUTES CapturedObjectAttributes, 55 _Out_ PUNICODE_STRING ObjectName, 56 _In_ KPROCESSOR_MODE AccessMode, 57 _In_ POBJECT_ATTRIBUTES ObjectAttributes, 58 _In_ BOOLEAN CaptureSecurity) 59 { 60 NTSTATUS Status = STATUS_SUCCESS; 61 PSECURITY_DESCRIPTOR SecurityDescriptor; 62 // PSECURITY_QUALITY_OF_SERVICE SecurityQos; 63 PUNICODE_STRING LocalObjectName = NULL; 64 65 /* Zero out the Capture Data */ 66 RtlZeroMemory(CapturedObjectAttributes, sizeof(*CapturedObjectAttributes)); 67 68 /* SEH everything here for protection */ 69 _SEH2_TRY 70 { 71 /* Check if we got attributes */ 72 if (ObjectAttributes) 73 { 74 /* Check if we're in user mode */ 75 if (AccessMode != KernelMode) 76 { 77 /* Probe the attributes */ 78 ProbeForRead(ObjectAttributes, 79 sizeof(OBJECT_ATTRIBUTES), 80 sizeof(ULONG)); 81 } 82 83 /* Validate the Size and Attributes */ 84 if ((ObjectAttributes->Length != sizeof(OBJECT_ATTRIBUTES)) || 85 (ObjectAttributes->Attributes & ~OBJ_VALID_KERNEL_ATTRIBUTES)) // Understood as all the possible valid attributes 86 { 87 /* Invalid combination, fail */ 88 _SEH2_YIELD(return STATUS_INVALID_PARAMETER); 89 } 90 91 /* Set some Create Info and do not allow user-mode kernel handles */ 92 CapturedObjectAttributes->Length = sizeof(OBJECT_ATTRIBUTES); 93 CapturedObjectAttributes->RootDirectory = ObjectAttributes->RootDirectory; 94 CapturedObjectAttributes->Attributes = ObpValidateAttributes(ObjectAttributes->Attributes, AccessMode); 95 LocalObjectName = ObjectAttributes->ObjectName; 96 SecurityDescriptor = ObjectAttributes->SecurityDescriptor; 97 // SecurityQos = ObjectAttributes->SecurityQualityOfService; 98 99 /* Check if we have a security descriptor */ 100 if (CaptureSecurity && SecurityDescriptor) 101 { 102 /* 103 * Capture it. 104 * Note: This has an implicit memory barrier due 105 * to the function call, so cleanup is safe here. 106 */ 107 Status = SeCaptureSecurityDescriptor(SecurityDescriptor, 108 AccessMode, 109 NonPagedPool, 110 TRUE, 111 &CapturedObjectAttributes-> 112 SecurityDescriptor); 113 if (!NT_SUCCESS(Status)) 114 { 115 /* Capture failed, quit */ 116 CapturedObjectAttributes->SecurityDescriptor = NULL; 117 _SEH2_YIELD(return Status); 118 } 119 } 120 else 121 { 122 CapturedObjectAttributes->SecurityDescriptor = NULL; 123 } 124 125 #if 0 126 // We don't use the QoS! 127 128 /* Check if we have QoS */ 129 if (SecurityQos) 130 { 131 /* Check if we came from user mode */ 132 if (AccessMode != KernelMode) 133 { 134 /* Validate the QoS */ 135 ProbeForRead(SecurityQos, 136 sizeof(SECURITY_QUALITY_OF_SERVICE), 137 sizeof(ULONG)); 138 } 139 140 /* Save Info */ 141 CapturedObjectAttributes->SecurityQualityOfService = *SecurityQos; 142 CapturedObjectAttributes->SecurityQos = 143 &CapturedObjectAttributes->SecurityQualityOfService; 144 } 145 #else 146 CapturedObjectAttributes->SecurityQualityOfService = NULL; 147 #endif 148 } 149 else 150 { 151 /* We don't have a name */ 152 LocalObjectName = NULL; 153 } 154 } 155 _SEH2_EXCEPT(ExSystemExceptionFilter()) 156 { 157 /* Cleanup and return the exception code */ 158 ReleaseCapturedObjectAttributes(CapturedObjectAttributes, AccessMode); 159 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 160 } 161 _SEH2_END; 162 163 /* Now check if the Object Attributes had an Object Name */ 164 if (LocalObjectName) 165 { 166 Status = ProbeAndCaptureUnicodeString(ObjectName, AccessMode, LocalObjectName); 167 } 168 else 169 { 170 /* Clear the string */ 171 RtlInitEmptyUnicodeString(ObjectName, NULL, 0); 172 173 /* It cannot have specified a Root Directory */ 174 if (CapturedObjectAttributes->RootDirectory) 175 { 176 Status = STATUS_OBJECT_NAME_INVALID; 177 } 178 } 179 180 /* Set the caputured object attributes name pointer to the one the user gave to us */ 181 CapturedObjectAttributes->ObjectName = ObjectName; 182 183 /* Cleanup if we failed */ 184 if (!NT_SUCCESS(Status)) 185 { 186 ReleaseCapturedObjectAttributes(CapturedObjectAttributes, AccessMode); 187 } 188 189 /* Return status to caller */ 190 return Status; 191 } 192 193 static 194 NTSTATUS 195 CmpConvertHandleToKernelHandle( 196 _In_ HANDLE SourceHandle, 197 _In_opt_ POBJECT_TYPE ObjectType, 198 _In_ ACCESS_MASK DesiredAccess, 199 _In_ KPROCESSOR_MODE AccessMode, 200 _Out_ PHANDLE KernelHandle) 201 { 202 NTSTATUS Status; 203 PVOID Object; 204 205 *KernelHandle = NULL; 206 207 /* NULL handle is valid */ 208 if (SourceHandle == NULL) 209 return STATUS_SUCCESS; 210 211 /* Get the object pointer */ 212 Status = ObReferenceObjectByHandle(SourceHandle, 213 DesiredAccess, 214 ObjectType, 215 AccessMode, 216 &Object, 217 NULL); 218 if (!NT_SUCCESS(Status)) 219 return Status; 220 221 /* Create a kernel handle from the pointer */ 222 Status = ObOpenObjectByPointer(Object, 223 OBJ_KERNEL_HANDLE, 224 NULL, 225 DesiredAccess, 226 ObjectType, 227 KernelMode, 228 KernelHandle); 229 230 /* Dereference the object */ 231 ObDereferenceObject(Object); 232 return Status; 233 } 234 235 236 /* FUNCTIONS *****************************************************************/ 237 238 NTSTATUS 239 NTAPI 240 NtCreateKey(OUT PHANDLE KeyHandle, 241 IN ACCESS_MASK DesiredAccess, 242 IN POBJECT_ATTRIBUTES ObjectAttributes, 243 IN ULONG TitleIndex, 244 IN PUNICODE_STRING Class OPTIONAL, 245 IN ULONG CreateOptions, 246 OUT PULONG Disposition OPTIONAL) 247 { 248 NTSTATUS Status; 249 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 250 CM_PARSE_CONTEXT ParseContext = {0}; 251 HANDLE Handle; 252 PAGED_CODE(); 253 254 DPRINT("NtCreateKey(Path: %wZ, Root %x, Access: %x, CreateOptions %x)\n", 255 ObjectAttributes->ObjectName, ObjectAttributes->RootDirectory, 256 DesiredAccess, CreateOptions); 257 258 /* Ignore the WOW64 flag, it's not valid in the kernel */ 259 DesiredAccess &= ~KEY_WOW64_RES; 260 261 /* Check for user-mode caller */ 262 if (PreviousMode != KernelMode) 263 { 264 /* Prepare to probe parameters */ 265 _SEH2_TRY 266 { 267 /* Check if we have a class */ 268 if (Class) 269 { 270 /* Probe it */ 271 ParseContext.Class = ProbeForReadUnicodeString(Class); 272 ProbeForRead(ParseContext.Class.Buffer, 273 ParseContext.Class.Length, 274 sizeof(WCHAR)); 275 } 276 277 /* Probe the key handle */ 278 ProbeForWriteHandle(KeyHandle); 279 *KeyHandle = NULL; 280 281 /* Probe object attributes */ 282 ProbeForRead(ObjectAttributes, 283 sizeof(OBJECT_ATTRIBUTES), 284 sizeof(ULONG)); 285 286 if (Disposition) 287 ProbeForWriteUlong(Disposition); 288 } 289 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 290 { 291 /* Return the exception code */ 292 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 293 } 294 _SEH2_END; 295 } 296 else 297 { 298 /* Save the class directly */ 299 if (Class) ParseContext.Class = *Class; 300 } 301 302 /* Setup the parse context */ 303 ParseContext.CreateOperation = TRUE; 304 ParseContext.CreateOptions = CreateOptions; 305 306 /* Do the create */ 307 Status = ObOpenObjectByName(ObjectAttributes, 308 CmpKeyObjectType, 309 PreviousMode, 310 NULL, 311 DesiredAccess, 312 &ParseContext, 313 &Handle); 314 315 _SEH2_TRY 316 { 317 /* Return data to user */ 318 if (NT_SUCCESS(Status)) *KeyHandle = Handle; 319 if (Disposition) *Disposition = ParseContext.Disposition; 320 } 321 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 322 { 323 /* Get the status */ 324 Status = _SEH2_GetExceptionCode(); 325 } 326 _SEH2_END; 327 328 DPRINT("Returning handle %x, Status %x.\n", Handle, Status); 329 330 /* Return status */ 331 return Status; 332 } 333 334 NTSTATUS 335 NTAPI 336 NtOpenKey(OUT PHANDLE KeyHandle, 337 IN ACCESS_MASK DesiredAccess, 338 IN POBJECT_ATTRIBUTES ObjectAttributes) 339 { 340 CM_PARSE_CONTEXT ParseContext = {0}; 341 HANDLE Handle; 342 NTSTATUS Status; 343 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 344 PAGED_CODE(); 345 DPRINT("NtOpenKey(Path: %wZ, Root %x, Access: %x)\n", 346 ObjectAttributes->ObjectName, ObjectAttributes->RootDirectory, DesiredAccess); 347 348 /* Ignore the WOW64 flag, it's not valid in the kernel */ 349 DesiredAccess &= ~KEY_WOW64_RES; 350 351 /* Check for user-mode caller */ 352 if (PreviousMode != KernelMode) 353 { 354 /* Prepare to probe parameters */ 355 _SEH2_TRY 356 { 357 /* Probe the key handle */ 358 ProbeForWriteHandle(KeyHandle); 359 *KeyHandle = NULL; 360 361 /* Probe object attributes */ 362 ProbeForRead(ObjectAttributes, 363 sizeof(OBJECT_ATTRIBUTES), 364 sizeof(ULONG)); 365 } 366 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 367 { 368 /* Return the exception code */ 369 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 370 } 371 _SEH2_END; 372 } 373 374 /* Just let the object manager handle this */ 375 Status = ObOpenObjectByName(ObjectAttributes, 376 CmpKeyObjectType, 377 PreviousMode, 378 NULL, 379 DesiredAccess, 380 &ParseContext, 381 &Handle); 382 383 /* Only do this if we succeeded */ 384 if (NT_SUCCESS(Status)) 385 { 386 _SEH2_TRY 387 { 388 /* Return the handle to caller */ 389 *KeyHandle = Handle; 390 } 391 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 392 { 393 /* Get the status */ 394 Status = _SEH2_GetExceptionCode(); 395 } 396 _SEH2_END; 397 } 398 399 DPRINT("Returning handle %x, Status %x.\n", Handle, Status); 400 401 /* Return status */ 402 return Status; 403 } 404 405 406 NTSTATUS 407 NTAPI 408 NtDeleteKey(IN HANDLE KeyHandle) 409 { 410 PCM_KEY_BODY KeyObject; 411 NTSTATUS Status; 412 REG_DELETE_KEY_INFORMATION DeleteKeyInfo; 413 REG_POST_OPERATION_INFORMATION PostOperationInfo; 414 PAGED_CODE(); 415 DPRINT("NtDeleteKey(KH 0x%p)\n", KeyHandle); 416 417 /* Verify that the handle is valid and is a registry key */ 418 Status = ObReferenceObjectByHandle(KeyHandle, 419 DELETE, 420 CmpKeyObjectType, 421 ExGetPreviousMode(), 422 (PVOID*)&KeyObject, 423 NULL); 424 if (!NT_SUCCESS(Status)) return Status; 425 426 /* Setup the callback */ 427 PostOperationInfo.Object = (PVOID)KeyObject; 428 DeleteKeyInfo.Object = (PVOID)KeyObject; 429 Status = CmiCallRegisteredCallbacks(RegNtPreDeleteKey, &DeleteKeyInfo); 430 if (NT_SUCCESS(Status)) 431 { 432 /* Check if we are read-only */ 433 if ((KeyObject->KeyControlBlock->ExtFlags & CM_KCB_READ_ONLY_KEY) || 434 (KeyObject->KeyControlBlock->ParentKcb->ExtFlags & CM_KCB_READ_ONLY_KEY)) 435 { 436 /* Fail */ 437 Status = STATUS_ACCESS_DENIED; 438 } 439 else 440 { 441 /* Call the internal API */ 442 Status = CmDeleteKey(KeyObject); 443 } 444 445 /* Do post callback */ 446 PostOperationInfo.Status = Status; 447 CmiCallRegisteredCallbacks(RegNtPostDeleteKey, &PostOperationInfo); 448 } 449 450 /* Dereference and return status */ 451 ObDereferenceObject(KeyObject); 452 return Status; 453 } 454 455 NTSTATUS 456 NTAPI 457 NtEnumerateKey(IN HANDLE KeyHandle, 458 IN ULONG Index, 459 IN KEY_INFORMATION_CLASS KeyInformationClass, 460 OUT PVOID KeyInformation, 461 IN ULONG Length, 462 OUT PULONG ResultLength) 463 { 464 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 465 NTSTATUS Status; 466 PCM_KEY_BODY KeyObject; 467 REG_ENUMERATE_KEY_INFORMATION EnumerateKeyInfo; 468 REG_POST_OPERATION_INFORMATION PostOperationInfo; 469 PAGED_CODE(); 470 DPRINT("NtEnumerateKey() KH 0x%p, Index 0x%x, KIC %d, Length %lu\n", 471 KeyHandle, Index, KeyInformationClass, Length); 472 473 /* Reject classes we don't know about */ 474 if ((KeyInformationClass != KeyBasicInformation) && 475 (KeyInformationClass != KeyNodeInformation) && 476 (KeyInformationClass != KeyFullInformation)) 477 { 478 /* Fail */ 479 return STATUS_INVALID_PARAMETER; 480 } 481 482 /* Verify that the handle is valid and is a registry key */ 483 Status = ObReferenceObjectByHandle(KeyHandle, 484 KEY_ENUMERATE_SUB_KEYS, 485 CmpKeyObjectType, 486 PreviousMode, 487 (PVOID*)&KeyObject, 488 NULL); 489 if (!NT_SUCCESS(Status)) return Status; 490 491 if (PreviousMode != KernelMode) 492 { 493 _SEH2_TRY 494 { 495 ProbeForWriteUlong(ResultLength); 496 ProbeForWrite(KeyInformation, 497 Length, 498 sizeof(ULONG)); 499 } 500 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 501 { 502 /* Dereference and return status */ 503 ObDereferenceObject(KeyObject); 504 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 505 } 506 _SEH2_END; 507 } 508 509 /* Setup the callback */ 510 PostOperationInfo.Object = (PVOID)KeyObject; 511 EnumerateKeyInfo.Object = (PVOID)KeyObject; 512 EnumerateKeyInfo.Index = Index; 513 EnumerateKeyInfo.KeyInformationClass = KeyInformationClass; 514 EnumerateKeyInfo.Length = Length; 515 EnumerateKeyInfo.ResultLength = ResultLength; 516 517 /* Do the callback */ 518 Status = CmiCallRegisteredCallbacks(RegNtPreEnumerateKey, &EnumerateKeyInfo); 519 if (NT_SUCCESS(Status)) 520 { 521 /* Call the internal API */ 522 Status = CmEnumerateKey(KeyObject->KeyControlBlock, 523 Index, 524 KeyInformationClass, 525 KeyInformation, 526 Length, 527 ResultLength); 528 529 /* Do the post callback */ 530 PostOperationInfo.Status = Status; 531 CmiCallRegisteredCallbacks(RegNtPostEnumerateKey, &PostOperationInfo); 532 } 533 534 /* Dereference and return status */ 535 ObDereferenceObject(KeyObject); 536 DPRINT("Returning status %x.\n", Status); 537 return Status; 538 } 539 540 NTSTATUS 541 NTAPI 542 NtEnumerateValueKey(IN HANDLE KeyHandle, 543 IN ULONG Index, 544 IN KEY_VALUE_INFORMATION_CLASS KeyValueInformationClass, 545 OUT PVOID KeyValueInformation, 546 IN ULONG Length, 547 OUT PULONG ResultLength) 548 { 549 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 550 NTSTATUS Status; 551 PCM_KEY_BODY KeyObject; 552 REG_ENUMERATE_VALUE_KEY_INFORMATION EnumerateValueKeyInfo; 553 REG_POST_OPERATION_INFORMATION PostOperationInfo; 554 555 PAGED_CODE(); 556 557 DPRINT("NtEnumerateValueKey() KH 0x%p, Index 0x%x, KVIC %d, Length %lu\n", 558 KeyHandle, Index, KeyValueInformationClass, Length); 559 560 /* Reject classes we don't know about */ 561 if ((KeyValueInformationClass != KeyValueBasicInformation) && 562 (KeyValueInformationClass != KeyValueFullInformation) && 563 (KeyValueInformationClass != KeyValuePartialInformation) && 564 (KeyValueInformationClass != KeyValueFullInformationAlign64) && 565 (KeyValueInformationClass != KeyValuePartialInformationAlign64)) 566 { 567 /* Fail */ 568 return STATUS_INVALID_PARAMETER; 569 } 570 571 /* Verify that the handle is valid and is a registry key */ 572 Status = ObReferenceObjectByHandle(KeyHandle, 573 KEY_QUERY_VALUE, 574 CmpKeyObjectType, 575 PreviousMode, 576 (PVOID*)&KeyObject, 577 NULL); 578 if (!NT_SUCCESS(Status)) return Status; 579 580 if (PreviousMode != KernelMode) 581 { 582 _SEH2_TRY 583 { 584 ProbeForWriteUlong(ResultLength); 585 ProbeForWrite(KeyValueInformation, 586 Length, 587 sizeof(ULONG)); 588 } 589 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 590 { 591 /* Dereference and return status */ 592 ObDereferenceObject(KeyObject); 593 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 594 } 595 _SEH2_END; 596 } 597 598 /* Setup the callback */ 599 PostOperationInfo.Object = (PVOID)KeyObject; 600 EnumerateValueKeyInfo.Object = (PVOID)KeyObject; 601 EnumerateValueKeyInfo.Index = Index; 602 EnumerateValueKeyInfo.KeyValueInformationClass = KeyValueInformationClass; 603 EnumerateValueKeyInfo.KeyValueInformation = KeyValueInformation; 604 EnumerateValueKeyInfo.Length = Length; 605 EnumerateValueKeyInfo.ResultLength = ResultLength; 606 607 /* Do the callback */ 608 Status = CmiCallRegisteredCallbacks(RegNtPreEnumerateValueKey, 609 &EnumerateValueKeyInfo); 610 if (NT_SUCCESS(Status)) 611 { 612 /* Call the internal API */ 613 Status = CmEnumerateValueKey(KeyObject->KeyControlBlock, 614 Index, 615 KeyValueInformationClass, 616 KeyValueInformation, 617 Length, 618 ResultLength); 619 620 /* Do the post callback */ 621 PostOperationInfo.Status = Status; 622 CmiCallRegisteredCallbacks(RegNtPostEnumerateValueKey, &PostOperationInfo); 623 } 624 625 /* Dereference and return status */ 626 ObDereferenceObject(KeyObject); 627 return Status; 628 } 629 630 NTSTATUS 631 NTAPI 632 NtQueryKey(IN HANDLE KeyHandle, 633 IN KEY_INFORMATION_CLASS KeyInformationClass, 634 OUT PVOID KeyInformation, 635 IN ULONG Length, 636 OUT PULONG ResultLength) 637 { 638 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 639 NTSTATUS Status; 640 PCM_KEY_BODY KeyObject; 641 REG_QUERY_KEY_INFORMATION QueryKeyInfo; 642 REG_POST_OPERATION_INFORMATION PostOperationInfo; 643 OBJECT_HANDLE_INFORMATION HandleInfo; 644 PAGED_CODE(); 645 DPRINT("NtQueryKey() KH 0x%p, KIC %d, Length %lu\n", 646 KeyHandle, KeyInformationClass, Length); 647 648 /* Reject invalid classes */ 649 if ((KeyInformationClass != KeyBasicInformation) && 650 (KeyInformationClass != KeyNodeInformation) && 651 (KeyInformationClass != KeyFullInformation) && 652 (KeyInformationClass != KeyNameInformation) && 653 (KeyInformationClass != KeyCachedInformation) && 654 (KeyInformationClass != KeyFlagsInformation)) 655 { 656 /* Fail */ 657 return STATUS_INVALID_PARAMETER; 658 } 659 660 /* Check if just the name is required */ 661 if (KeyInformationClass == KeyNameInformation) 662 { 663 /* Ignore access level */ 664 Status = ObReferenceObjectByHandle(KeyHandle, 665 0, 666 CmpKeyObjectType, 667 PreviousMode, 668 (PVOID*)&KeyObject, 669 &HandleInfo); 670 if (NT_SUCCESS(Status)) 671 { 672 /* At least a single bit of access is required */ 673 if (!HandleInfo.GrantedAccess) 674 { 675 /* No such luck */ 676 ObDereferenceObject(KeyObject); 677 Status = STATUS_ACCESS_DENIED; 678 } 679 } 680 } 681 else 682 { 683 /* Get a reference */ 684 Status = ObReferenceObjectByHandle(KeyHandle, 685 KEY_QUERY_VALUE, 686 CmpKeyObjectType, 687 PreviousMode, 688 (PVOID*)&KeyObject, 689 NULL); 690 } 691 692 /* Quit on failure */ 693 if (!NT_SUCCESS(Status)) return Status; 694 695 if (PreviousMode != KernelMode) 696 { 697 _SEH2_TRY 698 { 699 ProbeForWriteUlong(ResultLength); 700 ProbeForWrite(KeyInformation, 701 Length, 702 sizeof(ULONG)); 703 } 704 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 705 { 706 /* Dereference and return status */ 707 ObDereferenceObject(KeyObject); 708 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 709 } 710 _SEH2_END; 711 } 712 713 /* Setup the callback */ 714 PostOperationInfo.Object = (PVOID)KeyObject; 715 QueryKeyInfo.Object = (PVOID)KeyObject; 716 QueryKeyInfo.KeyInformationClass = KeyInformationClass; 717 QueryKeyInfo.KeyInformation = KeyInformation; 718 QueryKeyInfo.Length = Length; 719 QueryKeyInfo.ResultLength = ResultLength; 720 721 /* Do the callback */ 722 Status = CmiCallRegisteredCallbacks(RegNtPreQueryKey, &QueryKeyInfo); 723 if (NT_SUCCESS(Status)) 724 { 725 /* Call the internal API */ 726 Status = CmQueryKey(KeyObject->KeyControlBlock, 727 KeyInformationClass, 728 KeyInformation, 729 Length, 730 ResultLength); 731 732 /* Do the post callback */ 733 PostOperationInfo.Status = Status; 734 CmiCallRegisteredCallbacks(RegNtPostQueryKey, &PostOperationInfo); 735 } 736 737 /* Dereference and return status */ 738 ObDereferenceObject(KeyObject); 739 return Status; 740 } 741 742 NTSTATUS 743 NTAPI 744 NtQueryValueKey(IN HANDLE KeyHandle, 745 IN PUNICODE_STRING ValueName, 746 IN KEY_VALUE_INFORMATION_CLASS KeyValueInformationClass, 747 OUT PVOID KeyValueInformation, 748 IN ULONG Length, 749 OUT PULONG ResultLength) 750 { 751 NTSTATUS Status; 752 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 753 PCM_KEY_BODY KeyObject; 754 REG_QUERY_VALUE_KEY_INFORMATION QueryValueKeyInfo; 755 REG_POST_OPERATION_INFORMATION PostOperationInfo; 756 UNICODE_STRING ValueNameCopy; 757 758 PAGED_CODE(); 759 760 DPRINT("NtQueryValueKey() KH 0x%p, VN '%wZ', KVIC %d, Length %lu\n", 761 KeyHandle, ValueName, KeyValueInformationClass, Length); 762 763 /* Reject classes we don't know about */ 764 if ((KeyValueInformationClass != KeyValueBasicInformation) && 765 (KeyValueInformationClass != KeyValueFullInformation) && 766 (KeyValueInformationClass != KeyValuePartialInformation) && 767 (KeyValueInformationClass != KeyValueFullInformationAlign64) && 768 (KeyValueInformationClass != KeyValuePartialInformationAlign64)) 769 { 770 /* Fail */ 771 return STATUS_INVALID_PARAMETER; 772 } 773 774 /* Verify that the handle is valid and is a registry key */ 775 Status = ObReferenceObjectByHandle(KeyHandle, 776 KEY_QUERY_VALUE, 777 CmpKeyObjectType, 778 PreviousMode, 779 (PVOID*)&KeyObject, 780 NULL); 781 if (!NT_SUCCESS(Status)) 782 return Status; 783 784 if (PreviousMode != KernelMode) 785 { 786 _SEH2_TRY 787 { 788 ProbeForWriteUlong(ResultLength); 789 ProbeForWrite(KeyValueInformation, 790 Length, 791 sizeof(ULONG)); 792 } 793 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 794 { 795 /* Dereference and return status */ 796 ObDereferenceObject(KeyObject); 797 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 798 } 799 _SEH2_END; 800 } 801 802 /* Capture the string */ 803 Status = ProbeAndCaptureUnicodeString(&ValueNameCopy, PreviousMode, ValueName); 804 if (!NT_SUCCESS(Status)) 805 goto Quit; 806 807 /* Make sure the name is aligned properly */ 808 if (ValueNameCopy.Length & (sizeof(WCHAR) - 1)) 809 { 810 /* It isn't, so we'll fail */ 811 Status = STATUS_INVALID_PARAMETER; 812 goto Quit; 813 } 814 815 /* Ignore any null characters at the end */ 816 while (ValueNameCopy.Length && 817 !(ValueNameCopy.Buffer[ValueNameCopy.Length / sizeof(WCHAR) - 1])) 818 { 819 /* Skip it */ 820 ValueNameCopy.Length -= sizeof(WCHAR); 821 } 822 823 /* Setup the callback */ 824 PostOperationInfo.Object = (PVOID)KeyObject; 825 QueryValueKeyInfo.Object = (PVOID)KeyObject; 826 QueryValueKeyInfo.ValueName = &ValueNameCopy; 827 QueryValueKeyInfo.KeyValueInformationClass = KeyValueInformationClass; 828 QueryValueKeyInfo.Length = Length; 829 QueryValueKeyInfo.ResultLength = ResultLength; 830 831 /* Do the callback */ 832 Status = CmiCallRegisteredCallbacks(RegNtPreQueryValueKey, &QueryValueKeyInfo); 833 if (NT_SUCCESS(Status)) 834 { 835 /* Call the internal API */ 836 Status = CmQueryValueKey(KeyObject->KeyControlBlock, 837 ValueNameCopy, 838 KeyValueInformationClass, 839 KeyValueInformation, 840 Length, 841 ResultLength); 842 843 /* Do the post callback */ 844 PostOperationInfo.Status = Status; 845 CmiCallRegisteredCallbacks(RegNtPostQueryValueKey, &PostOperationInfo); 846 } 847 848 Quit: 849 if (ValueNameCopy.Buffer) 850 ReleaseCapturedUnicodeString(&ValueNameCopy, PreviousMode); 851 852 /* Dereference and return status */ 853 ObDereferenceObject(KeyObject); 854 return Status; 855 } 856 857 NTSTATUS 858 NTAPI 859 NtSetValueKey(IN HANDLE KeyHandle, 860 IN PUNICODE_STRING ValueName, 861 IN ULONG TitleIndex, 862 IN ULONG Type, 863 IN PVOID Data, 864 IN ULONG DataSize) 865 { 866 NTSTATUS Status = STATUS_SUCCESS; 867 KPROCESSOR_MODE PreviousMode; 868 PCM_KEY_BODY KeyObject; 869 REG_SET_VALUE_KEY_INFORMATION SetValueKeyInfo; 870 REG_POST_OPERATION_INFORMATION PostOperationInfo; 871 UNICODE_STRING ValueNameCopy; 872 873 PAGED_CODE(); 874 875 PreviousMode = ExGetPreviousMode(); 876 877 /* Verify that the handle is valid and is a registry key */ 878 Status = ObReferenceObjectByHandle(KeyHandle, 879 KEY_SET_VALUE, 880 CmpKeyObjectType, 881 PreviousMode, 882 (PVOID*)&KeyObject, 883 NULL); 884 if (!NT_SUCCESS(Status)) 885 return Status; 886 887 if (!DataSize) 888 Data = NULL; 889 890 /* Probe and copy the data */ 891 if ((PreviousMode != KernelMode) && (DataSize != 0)) 892 { 893 PVOID DataCopy = NULL; 894 895 _SEH2_TRY 896 { 897 ProbeForRead(Data, DataSize, 1); 898 } 899 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 900 { 901 Status = _SEH2_GetExceptionCode(); 902 } 903 _SEH2_END; 904 905 if (!NT_SUCCESS(Status)) 906 { 907 /* Dereference and return status */ 908 ObDereferenceObject(KeyObject); 909 return Status; 910 } 911 912 DataCopy = ExAllocatePoolWithTag(PagedPool, DataSize, TAG_CM); 913 if (!DataCopy) 914 { 915 /* Dereference and return status */ 916 ObDereferenceObject(KeyObject); 917 return STATUS_INSUFFICIENT_RESOURCES; 918 } 919 920 _SEH2_TRY 921 { 922 RtlCopyMemory(DataCopy, Data, DataSize); 923 } 924 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 925 { 926 Status = _SEH2_GetExceptionCode(); 927 } 928 _SEH2_END; 929 930 if (!NT_SUCCESS(Status)) 931 { 932 /* Dereference and return status */ 933 ExFreePoolWithTag(DataCopy, TAG_CM); 934 ObDereferenceObject(KeyObject); 935 return Status; 936 } 937 938 Data = DataCopy; 939 } 940 941 /* Capture the string */ 942 Status = ProbeAndCaptureUnicodeString(&ValueNameCopy, PreviousMode, ValueName); 943 if (!NT_SUCCESS(Status)) 944 goto Quit; 945 946 DPRINT("NtSetValueKey() KH 0x%p, VN '%wZ', TI %x, T %lu, DS %lu\n", 947 KeyHandle, &ValueNameCopy, TitleIndex, Type, DataSize); 948 949 /* Make sure the name is aligned, not too long, and the data under 4GB */ 950 if ((ValueNameCopy.Length > 32767) || 951 (ValueNameCopy.Length & (sizeof(WCHAR) - 1)) || 952 (DataSize > 0x80000000)) 953 { 954 /* Fail */ 955 Status = STATUS_INVALID_PARAMETER; 956 goto Quit; 957 } 958 959 /* Ignore any null characters at the end */ 960 while (ValueNameCopy.Length && 961 !(ValueNameCopy.Buffer[ValueNameCopy.Length / sizeof(WCHAR) - 1])) 962 { 963 /* Skip it */ 964 ValueNameCopy.Length -= sizeof(WCHAR); 965 } 966 967 /* Don't touch read-only keys */ 968 if (KeyObject->KeyControlBlock->ExtFlags & CM_KCB_READ_ONLY_KEY) 969 { 970 /* Fail */ 971 Status = STATUS_ACCESS_DENIED; 972 goto Quit; 973 } 974 975 /* Setup callback */ 976 PostOperationInfo.Object = (PVOID)KeyObject; 977 SetValueKeyInfo.Object = (PVOID)KeyObject; 978 SetValueKeyInfo.ValueName = &ValueNameCopy; 979 SetValueKeyInfo.TitleIndex = TitleIndex; 980 SetValueKeyInfo.Type = Type; 981 SetValueKeyInfo.Data = Data; 982 SetValueKeyInfo.DataSize = DataSize; 983 984 /* Do the callback */ 985 Status = CmiCallRegisteredCallbacks(RegNtPreSetValueKey, &SetValueKeyInfo); 986 if (NT_SUCCESS(Status)) 987 { 988 /* Call the internal API */ 989 Status = CmSetValueKey(KeyObject->KeyControlBlock, 990 &ValueNameCopy, 991 Type, 992 Data, 993 DataSize); 994 995 /* Do the post-callback */ 996 PostOperationInfo.Status = Status; 997 CmiCallRegisteredCallbacks(RegNtPostSetValueKey, &PostOperationInfo); 998 } 999 1000 Quit: 1001 if (ValueNameCopy.Buffer) 1002 ReleaseCapturedUnicodeString(&ValueNameCopy, PreviousMode); 1003 1004 if ((PreviousMode != KernelMode) && Data) 1005 ExFreePoolWithTag(Data, TAG_CM); 1006 1007 /* Dereference and return status */ 1008 ObDereferenceObject(KeyObject); 1009 return Status; 1010 } 1011 1012 NTSTATUS 1013 NTAPI 1014 NtDeleteValueKey(IN HANDLE KeyHandle, 1015 IN PUNICODE_STRING ValueName) 1016 { 1017 NTSTATUS Status; 1018 PCM_KEY_BODY KeyObject; 1019 REG_DELETE_VALUE_KEY_INFORMATION DeleteValueKeyInfo; 1020 REG_POST_OPERATION_INFORMATION PostOperationInfo; 1021 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 1022 UNICODE_STRING ValueNameCopy; 1023 1024 PAGED_CODE(); 1025 1026 /* Verify that the handle is valid and is a registry key */ 1027 Status = ObReferenceObjectByHandle(KeyHandle, 1028 KEY_SET_VALUE, 1029 CmpKeyObjectType, 1030 PreviousMode, 1031 (PVOID*)&KeyObject, 1032 NULL); 1033 if (!NT_SUCCESS(Status)) 1034 return Status; 1035 1036 /* Capture the string */ 1037 Status = ProbeAndCaptureUnicodeString(&ValueNameCopy, PreviousMode, ValueName); 1038 if (!NT_SUCCESS(Status)) 1039 goto Quit; 1040 1041 /* Make sure the name is aligned properly */ 1042 if (ValueNameCopy.Length & (sizeof(WCHAR) - 1)) 1043 { 1044 /* It isn't, so we'll fail */ 1045 Status = STATUS_INVALID_PARAMETER; 1046 goto Quit; 1047 } 1048 1049 /* Don't touch read-only keys */ 1050 if (KeyObject->KeyControlBlock->ExtFlags & CM_KCB_READ_ONLY_KEY) 1051 { 1052 /* Fail */ 1053 Status = STATUS_ACCESS_DENIED; 1054 goto Quit; 1055 } 1056 1057 /* Do the callback */ 1058 DeleteValueKeyInfo.Object = (PVOID)KeyObject; 1059 DeleteValueKeyInfo.ValueName = ValueName; 1060 Status = CmiCallRegisteredCallbacks(RegNtPreDeleteValueKey, 1061 &DeleteValueKeyInfo); 1062 if (NT_SUCCESS(Status)) 1063 { 1064 /* Call the internal API */ 1065 Status = CmDeleteValueKey(KeyObject->KeyControlBlock, ValueNameCopy); 1066 1067 /* Do the post callback */ 1068 PostOperationInfo.Object = (PVOID)KeyObject; 1069 PostOperationInfo.Status = Status; 1070 CmiCallRegisteredCallbacks(RegNtPostDeleteValueKey, 1071 &PostOperationInfo); 1072 } 1073 1074 Quit: 1075 if (ValueNameCopy.Buffer) 1076 ReleaseCapturedUnicodeString(&ValueNameCopy, PreviousMode); 1077 1078 /* Dereference and return status */ 1079 ObDereferenceObject(KeyObject); 1080 return Status; 1081 } 1082 1083 NTSTATUS 1084 NTAPI 1085 NtFlushKey(IN HANDLE KeyHandle) 1086 { 1087 NTSTATUS Status; 1088 PCM_KEY_BODY KeyObject; 1089 PAGED_CODE(); 1090 1091 /* Get the key object */ 1092 Status = ObReferenceObjectByHandle(KeyHandle, 1093 0, 1094 CmpKeyObjectType, 1095 ExGetPreviousMode(), 1096 (PVOID*)&KeyObject, 1097 NULL); 1098 if (!NT_SUCCESS(Status)) return Status; 1099 1100 /* Lock the registry */ 1101 CmpLockRegistry(); 1102 1103 /* Lock the KCB */ 1104 CmpAcquireKcbLockShared(KeyObject->KeyControlBlock); 1105 1106 /* Make sure KCB isn't deleted */ 1107 if (KeyObject->KeyControlBlock->Delete) 1108 { 1109 /* Fail */ 1110 Status = STATUS_KEY_DELETED; 1111 } 1112 else 1113 { 1114 /* Call the internal API */ 1115 Status = CmFlushKey(KeyObject->KeyControlBlock, FALSE); 1116 } 1117 1118 /* Release the locks */ 1119 CmpReleaseKcbLock(KeyObject->KeyControlBlock); 1120 CmpUnlockRegistry(); 1121 1122 /* Dereference the object and return status */ 1123 ObDereferenceObject(KeyObject); 1124 return Status; 1125 } 1126 1127 NTSTATUS 1128 NTAPI 1129 NtLoadKey(IN POBJECT_ATTRIBUTES KeyObjectAttributes, 1130 IN POBJECT_ATTRIBUTES FileObjectAttributes) 1131 { 1132 /* Call the newer API */ 1133 return NtLoadKeyEx(KeyObjectAttributes, FileObjectAttributes, 0, NULL); 1134 } 1135 1136 NTSTATUS 1137 NTAPI 1138 NtLoadKey2(IN POBJECT_ATTRIBUTES KeyObjectAttributes, 1139 IN POBJECT_ATTRIBUTES FileObjectAttributes, 1140 IN ULONG Flags) 1141 { 1142 /* Call the newer API */ 1143 return NtLoadKeyEx(KeyObjectAttributes, FileObjectAttributes, Flags, NULL); 1144 } 1145 1146 NTSTATUS 1147 NTAPI 1148 NtLoadKeyEx(IN POBJECT_ATTRIBUTES TargetKey, 1149 IN POBJECT_ATTRIBUTES SourceFile, 1150 IN ULONG Flags, 1151 IN HANDLE TrustClassKey) 1152 { 1153 NTSTATUS Status; 1154 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 1155 OBJECT_ATTRIBUTES CapturedTargetKey; 1156 OBJECT_ATTRIBUTES CapturedSourceFile; 1157 UNICODE_STRING TargetKeyName, SourceFileName; 1158 HANDLE KmTargetKeyRootDir = NULL, KmSourceFileRootDir = NULL; 1159 PCM_KEY_BODY KeyBody = NULL; 1160 1161 PAGED_CODE(); 1162 1163 /* Validate flags */ 1164 if (Flags & ~REG_NO_LAZY_FLUSH) 1165 return STATUS_INVALID_PARAMETER; 1166 1167 /* Validate privilege */ 1168 if (!SeSinglePrivilegeCheck(SeRestorePrivilege, PreviousMode)) 1169 { 1170 DPRINT1("Restore Privilege missing!\n"); 1171 return STATUS_PRIVILEGE_NOT_HELD; 1172 } 1173 1174 /* Block APCs */ 1175 KeEnterCriticalRegion(); 1176 1177 /* Check for user-mode caller */ 1178 if (PreviousMode != KernelMode) 1179 { 1180 /* Prepare to probe parameters */ 1181 _SEH2_TRY 1182 { 1183 /* Probe target key */ 1184 ProbeForRead(TargetKey, 1185 sizeof(OBJECT_ATTRIBUTES), 1186 sizeof(ULONG)); 1187 1188 /* Probe source file */ 1189 ProbeForRead(SourceFile, 1190 sizeof(OBJECT_ATTRIBUTES), 1191 sizeof(ULONG)); 1192 } 1193 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1194 { 1195 /* Return the exception code */ 1196 Status = _SEH2_GetExceptionCode(); 1197 _SEH2_YIELD(goto Quit); 1198 } 1199 _SEH2_END; 1200 } 1201 1202 /* Probe and capture the target key attributes, including the security */ 1203 Status = ProbeAndCaptureObjectAttributes(&CapturedTargetKey, 1204 &TargetKeyName, 1205 PreviousMode, 1206 TargetKey, 1207 TRUE); 1208 if (!NT_SUCCESS(Status)) 1209 goto Quit; 1210 1211 /* 1212 * Probe and capture the source file attributes, but not the security. 1213 * A proper security context is built by CmLoadKey(). 1214 */ 1215 Status = ProbeAndCaptureObjectAttributes(&CapturedSourceFile, 1216 &SourceFileName, 1217 PreviousMode, 1218 SourceFile, 1219 FALSE); 1220 if (!NT_SUCCESS(Status)) 1221 { 1222 ReleaseCapturedObjectAttributes(&CapturedTargetKey, PreviousMode); 1223 goto Quit; 1224 } 1225 1226 /* Make sure the target key root directory handle is a kernel handle */ 1227 Status = CmpConvertHandleToKernelHandle(CapturedTargetKey.RootDirectory, 1228 CmpKeyObjectType, 1229 KEY_READ, 1230 PreviousMode, 1231 &KmTargetKeyRootDir); 1232 if (!NT_SUCCESS(Status)) 1233 goto Cleanup; 1234 CapturedTargetKey.RootDirectory = KmTargetKeyRootDir; 1235 CapturedTargetKey.Attributes |= OBJ_KERNEL_HANDLE; 1236 1237 /* Make sure the source file root directory handle is a kernel handle */ 1238 Status = CmpConvertHandleToKernelHandle(CapturedSourceFile.RootDirectory, 1239 IoFileObjectType, 1240 FILE_TRAVERSE, 1241 PreviousMode, 1242 &KmSourceFileRootDir); 1243 if (!NT_SUCCESS(Status)) 1244 goto Cleanup; 1245 CapturedSourceFile.RootDirectory = KmSourceFileRootDir; 1246 CapturedSourceFile.Attributes |= OBJ_KERNEL_HANDLE; 1247 1248 /* Check if we have a trust class */ 1249 if (TrustClassKey) 1250 { 1251 /* Reference it */ 1252 Status = ObReferenceObjectByHandle(TrustClassKey, 1253 0, 1254 CmpKeyObjectType, 1255 PreviousMode, 1256 (PVOID*)&KeyBody, 1257 NULL); 1258 } 1259 1260 /* Call the internal API */ 1261 Status = CmLoadKey(&CapturedTargetKey, 1262 &CapturedSourceFile, 1263 Flags, 1264 KeyBody); 1265 1266 /* Dereference the trust key, if any */ 1267 if (KeyBody) ObDereferenceObject(KeyBody); 1268 1269 Cleanup: 1270 /* Close the local kernel handles */ 1271 if (KmSourceFileRootDir) 1272 ObCloseHandle(KmSourceFileRootDir, KernelMode); 1273 if (KmTargetKeyRootDir) 1274 ObCloseHandle(KmTargetKeyRootDir, KernelMode); 1275 1276 /* Release the captured object attributes */ 1277 ReleaseCapturedObjectAttributes(&CapturedSourceFile, PreviousMode); 1278 ReleaseCapturedObjectAttributes(&CapturedTargetKey, PreviousMode); 1279 1280 Quit: 1281 /* Bring back APCs */ 1282 KeLeaveCriticalRegion(); 1283 1284 /* Return status */ 1285 return Status; 1286 } 1287 1288 NTSTATUS 1289 NTAPI 1290 NtNotifyChangeKey(IN HANDLE KeyHandle, 1291 IN HANDLE Event, 1292 IN PIO_APC_ROUTINE ApcRoutine OPTIONAL, 1293 IN PVOID ApcContext OPTIONAL, 1294 OUT PIO_STATUS_BLOCK IoStatusBlock, 1295 IN ULONG CompletionFilter, 1296 IN BOOLEAN WatchTree, 1297 OUT PVOID Buffer, 1298 IN ULONG Length, 1299 IN BOOLEAN Asynchronous) 1300 { 1301 /* Call the newer API */ 1302 return NtNotifyChangeMultipleKeys(KeyHandle, 1303 0, 1304 NULL, 1305 Event, 1306 ApcRoutine, 1307 ApcContext, 1308 IoStatusBlock, 1309 CompletionFilter, 1310 WatchTree, 1311 Buffer, 1312 Length, 1313 Asynchronous); 1314 } 1315 1316 NTSTATUS 1317 NTAPI 1318 NtInitializeRegistry(IN USHORT Flag) 1319 { 1320 BOOLEAN SetupBoot; 1321 NTSTATUS Status = STATUS_SUCCESS; 1322 PAGED_CODE(); 1323 1324 /* Always do this as kernel mode */ 1325 if (KeGetPreviousMode() == UserMode) 1326 return ZwInitializeRegistry(Flag); 1327 1328 /* Enough of the system has booted by now */ 1329 Ki386PerfEnd(); 1330 1331 /* Validate flag */ 1332 if (Flag > CM_BOOT_FLAG_MAX) return STATUS_INVALID_PARAMETER; 1333 1334 /* Check if boot was accepted */ 1335 if ((Flag >= CM_BOOT_FLAG_ACCEPTED) && (Flag <= CM_BOOT_FLAG_MAX)) 1336 { 1337 /* Only allow once */ 1338 if (!CmBootAcceptFirstTime) return STATUS_ACCESS_DENIED; 1339 CmBootAcceptFirstTime = FALSE; 1340 1341 /* Get the control set accepted */ 1342 Flag -= CM_BOOT_FLAG_ACCEPTED; 1343 if (Flag) 1344 { 1345 /* Save the last known good boot */ 1346 Status = CmpSaveBootControlSet(Flag); 1347 1348 /* Notify HAL */ 1349 HalEndOfBoot(); 1350 1351 /* Enable lazy flush */ 1352 CmpHoldLazyFlush = FALSE; 1353 CmpLazyFlush(); 1354 return Status; 1355 } 1356 1357 /* Otherwise, invalid boot */ 1358 return STATUS_INVALID_PARAMETER; 1359 } 1360 1361 /* Check if this was a setup boot */ 1362 SetupBoot = (Flag == CM_BOOT_FLAG_SETUP ? TRUE : FALSE); 1363 1364 /* Make sure we're only called once */ 1365 if (!CmFirstTime) return STATUS_ACCESS_DENIED; 1366 CmFirstTime = FALSE; 1367 1368 /* Lock the registry exclusively */ 1369 CmpLockRegistryExclusive(); 1370 1371 /* Initialize the hives and lazy flusher */ 1372 CmpCmdInit(SetupBoot); 1373 1374 /* Save version data */ 1375 CmpSetVersionData(); 1376 1377 /* Release the registry lock */ 1378 CmpUnlockRegistry(); 1379 return STATUS_SUCCESS; 1380 } 1381 1382 NTSTATUS 1383 NTAPI 1384 NtCompactKeys(IN ULONG Count, 1385 IN PHANDLE KeyArray) 1386 { 1387 UNIMPLEMENTED; 1388 return STATUS_NOT_IMPLEMENTED; 1389 } 1390 1391 NTSTATUS 1392 NTAPI 1393 NtCompressKey(IN HANDLE Key) 1394 { 1395 UNIMPLEMENTED; 1396 return STATUS_NOT_IMPLEMENTED; 1397 } 1398 1399 // FIXME: different for different windows versions! 1400 #define PRODUCT_ACTIVATION_VERSION 7749 1401 1402 NTSTATUS 1403 NTAPI 1404 NtLockProductActivationKeys(IN PULONG pPrivateVer, 1405 IN PULONG pSafeMode) 1406 { 1407 KPROCESSOR_MODE PreviousMode; 1408 1409 PreviousMode = ExGetPreviousMode(); 1410 _SEH2_TRY 1411 { 1412 /* Check if the caller asked for the version */ 1413 if (pPrivateVer != NULL) 1414 { 1415 /* For user mode, probe it */ 1416 if (PreviousMode != KernelMode) 1417 { 1418 ProbeForWriteUlong(pPrivateVer); 1419 } 1420 1421 /* Return the expected version */ 1422 *pPrivateVer = PRODUCT_ACTIVATION_VERSION; 1423 } 1424 1425 /* Check if the caller asked for safe mode mode state */ 1426 if (pSafeMode != NULL) 1427 { 1428 /* For user mode, probe it */ 1429 if (PreviousMode != KernelMode) 1430 { 1431 ProbeForWriteUlong(pSafeMode); 1432 } 1433 1434 /* Return the safe boot mode state */ 1435 *pSafeMode = InitSafeBootMode; 1436 } 1437 } 1438 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1439 { 1440 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 1441 } 1442 _SEH2_END; 1443 1444 return STATUS_SUCCESS; 1445 } 1446 1447 NTSTATUS 1448 NTAPI 1449 NtLockRegistryKey(IN HANDLE KeyHandle) 1450 { 1451 UNIMPLEMENTED; 1452 return STATUS_NOT_IMPLEMENTED; 1453 } 1454 1455 NTSTATUS 1456 NTAPI 1457 NtNotifyChangeMultipleKeys(IN HANDLE MasterKeyHandle, 1458 IN ULONG Count, 1459 IN POBJECT_ATTRIBUTES SlaveObjects, 1460 IN HANDLE Event, 1461 IN PIO_APC_ROUTINE ApcRoutine OPTIONAL, 1462 IN PVOID ApcContext OPTIONAL, 1463 OUT PIO_STATUS_BLOCK IoStatusBlock, 1464 IN ULONG CompletionFilter, 1465 IN BOOLEAN WatchTree, 1466 OUT PVOID Buffer, 1467 IN ULONG Length, 1468 IN BOOLEAN Asynchronous) 1469 { 1470 UNIMPLEMENTED_ONCE; 1471 return STATUS_NOT_IMPLEMENTED; 1472 } 1473 1474 NTSTATUS 1475 NTAPI 1476 NtQueryMultipleValueKey(IN HANDLE KeyHandle, 1477 IN OUT PKEY_VALUE_ENTRY ValueList, 1478 IN ULONG NumberOfValues, 1479 OUT PVOID Buffer, 1480 IN OUT PULONG Length, 1481 OUT PULONG ReturnLength) 1482 { 1483 UNIMPLEMENTED; 1484 return STATUS_NOT_IMPLEMENTED; 1485 } 1486 1487 NTSTATUS 1488 NTAPI 1489 NtQueryOpenSubKeys(IN POBJECT_ATTRIBUTES TargetKey, 1490 OUT PULONG HandleCount) 1491 { 1492 KPROCESSOR_MODE PreviousMode; 1493 PCM_KEY_BODY KeyBody = NULL; 1494 HANDLE KeyHandle; 1495 NTSTATUS Status; 1496 ULONG SubKeys; 1497 1498 DPRINT("NtQueryOpenSubKeys()\n"); 1499 1500 PAGED_CODE(); 1501 1502 /* Get the processor mode */ 1503 PreviousMode = KeGetPreviousMode(); 1504 1505 /* Check for user-mode caller */ 1506 if (PreviousMode != KernelMode) 1507 { 1508 /* Prepare to probe parameters */ 1509 _SEH2_TRY 1510 { 1511 /* Probe target key */ 1512 ProbeForRead(TargetKey, 1513 sizeof(OBJECT_ATTRIBUTES), 1514 sizeof(ULONG)); 1515 1516 /* Probe handle count */ 1517 ProbeForWriteUlong(HandleCount); 1518 } 1519 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1520 { 1521 /* Return the exception code */ 1522 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 1523 } 1524 _SEH2_END; 1525 } 1526 1527 /* Open a handle to the key */ 1528 Status = ObOpenObjectByName(TargetKey, 1529 CmpKeyObjectType, 1530 PreviousMode, 1531 NULL, 1532 KEY_READ, 1533 NULL, 1534 &KeyHandle); 1535 if (NT_SUCCESS(Status)) 1536 { 1537 /* Reference the key object */ 1538 Status = ObReferenceObjectByHandle(KeyHandle, 1539 KEY_READ, 1540 CmpKeyObjectType, 1541 PreviousMode, 1542 (PVOID*)&KeyBody, 1543 NULL); 1544 1545 /* Close the handle */ 1546 NtClose(KeyHandle); 1547 } 1548 1549 /* Fail, if the key object could not be referenced */ 1550 if (!NT_SUCCESS(Status)) 1551 return Status; 1552 1553 /* Lock the registry exclusively */ 1554 CmpLockRegistryExclusive(); 1555 1556 /* Fail, if we did not open a hive root key */ 1557 if (KeyBody->KeyControlBlock->KeyCell != 1558 KeyBody->KeyControlBlock->KeyHive->BaseBlock->RootCell) 1559 { 1560 DPRINT("Error: Key is not a hive root key!\n"); 1561 CmpUnlockRegistry(); 1562 ObDereferenceObject(KeyBody); 1563 return STATUS_INVALID_PARAMETER; 1564 } 1565 1566 /* Call the internal API */ 1567 SubKeys = CmpEnumerateOpenSubKeys(KeyBody->KeyControlBlock, 1568 FALSE, FALSE); 1569 1570 /* Unlock the registry */ 1571 CmpUnlockRegistry(); 1572 1573 /* Dereference the key object */ 1574 ObDereferenceObject(KeyBody); 1575 1576 /* Write back the result */ 1577 _SEH2_TRY 1578 { 1579 *HandleCount = SubKeys; 1580 } 1581 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1582 { 1583 Status = _SEH2_GetExceptionCode(); 1584 } 1585 _SEH2_END; 1586 1587 DPRINT("Done.\n"); 1588 1589 return Status; 1590 } 1591 1592 NTSTATUS 1593 NTAPI 1594 NtQueryOpenSubKeysEx(IN POBJECT_ATTRIBUTES TargetKey, 1595 IN ULONG BufferLength, 1596 IN PVOID Buffer, 1597 IN PULONG RequiredSize) 1598 { 1599 UNIMPLEMENTED; 1600 return STATUS_NOT_IMPLEMENTED; 1601 } 1602 1603 NTSTATUS 1604 NTAPI 1605 NtRenameKey(IN HANDLE KeyHandle, 1606 IN PUNICODE_STRING ReplacementName) 1607 { 1608 UNIMPLEMENTED; 1609 return STATUS_NOT_IMPLEMENTED; 1610 } 1611 1612 NTSTATUS 1613 NTAPI 1614 NtReplaceKey(IN POBJECT_ATTRIBUTES ObjectAttributes, 1615 IN HANDLE Key, 1616 IN POBJECT_ATTRIBUTES ReplacedObjectAttributes) 1617 { 1618 UNIMPLEMENTED; 1619 return STATUS_NOT_IMPLEMENTED; 1620 } 1621 1622 NTSTATUS 1623 NTAPI 1624 NtRestoreKey(IN HANDLE KeyHandle, 1625 IN HANDLE FileHandle, 1626 IN ULONG RestoreFlags) 1627 { 1628 UNIMPLEMENTED; 1629 return STATUS_NOT_IMPLEMENTED; 1630 } 1631 1632 NTSTATUS 1633 NTAPI 1634 NtSaveKey(IN HANDLE KeyHandle, 1635 IN HANDLE FileHandle) 1636 { 1637 /* Call the extended API */ 1638 return NtSaveKeyEx(KeyHandle, FileHandle, REG_STANDARD_FORMAT); 1639 } 1640 1641 NTSTATUS 1642 NTAPI 1643 NtSaveKeyEx(IN HANDLE KeyHandle, 1644 IN HANDLE FileHandle, 1645 IN ULONG Flags) 1646 { 1647 NTSTATUS Status; 1648 HANDLE KmFileHandle = NULL; 1649 PCM_KEY_BODY KeyObject; 1650 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 1651 1652 PAGED_CODE(); 1653 1654 DPRINT("NtSaveKeyEx(0x%p, 0x%p, %lu)\n", KeyHandle, FileHandle, Flags); 1655 1656 /* Verify the flags */ 1657 if ((Flags != REG_STANDARD_FORMAT) 1658 && (Flags != REG_LATEST_FORMAT) 1659 && (Flags != REG_NO_COMPRESSION)) 1660 { 1661 /* Only one of these values can be specified */ 1662 return STATUS_INVALID_PARAMETER; 1663 } 1664 1665 /* Validate privilege */ 1666 if (!SeSinglePrivilegeCheck(SeBackupPrivilege, PreviousMode)) 1667 { 1668 return STATUS_PRIVILEGE_NOT_HELD; 1669 } 1670 1671 /* Make sure the target file handle is a kernel handle */ 1672 Status = CmpConvertHandleToKernelHandle(FileHandle, 1673 IoFileObjectType, 1674 FILE_WRITE_DATA, 1675 PreviousMode, 1676 &KmFileHandle); 1677 if (!NT_SUCCESS(Status)) 1678 goto Quit; 1679 1680 /* Verify that the handle is valid and is a registry key */ 1681 Status = ObReferenceObjectByHandle(KeyHandle, 1682 KEY_READ, 1683 CmpKeyObjectType, 1684 PreviousMode, 1685 (PVOID*)&KeyObject, 1686 NULL); 1687 if (!NT_SUCCESS(Status)) 1688 goto Quit; 1689 1690 /* Call the internal API */ 1691 Status = CmSaveKey(KeyObject->KeyControlBlock, KmFileHandle, Flags); 1692 1693 /* Dereference the registry key */ 1694 ObDereferenceObject(KeyObject); 1695 1696 Quit: 1697 /* Close the local kernel handle */ 1698 if (KmFileHandle) 1699 ObCloseHandle(KmFileHandle, KernelMode); 1700 1701 return Status; 1702 } 1703 1704 NTSTATUS 1705 NTAPI 1706 NtSaveMergedKeys(IN HANDLE HighPrecedenceKeyHandle, 1707 IN HANDLE LowPrecedenceKeyHandle, 1708 IN HANDLE FileHandle) 1709 { 1710 NTSTATUS Status; 1711 KPROCESSOR_MODE PreviousMode; 1712 HANDLE KmFileHandle = NULL; 1713 PCM_KEY_BODY HighPrecedenceKeyObject = NULL; 1714 PCM_KEY_BODY LowPrecedenceKeyObject = NULL; 1715 1716 PAGED_CODE(); 1717 1718 DPRINT("NtSaveMergedKeys(0x%p, 0x%p, 0x%p)\n", 1719 HighPrecedenceKeyHandle, LowPrecedenceKeyHandle, FileHandle); 1720 1721 PreviousMode = ExGetPreviousMode(); 1722 1723 /* Validate privilege */ 1724 if (!SeSinglePrivilegeCheck(SeBackupPrivilege, PreviousMode)) 1725 { 1726 return STATUS_PRIVILEGE_NOT_HELD; 1727 } 1728 1729 /* Make sure the target file handle is a kernel handle */ 1730 Status = CmpConvertHandleToKernelHandle(FileHandle, 1731 IoFileObjectType, 1732 FILE_WRITE_DATA, 1733 PreviousMode, 1734 &KmFileHandle); 1735 if (!NT_SUCCESS(Status)) 1736 goto Quit; 1737 1738 /* Verify that the handles are valid and are registry keys */ 1739 Status = ObReferenceObjectByHandle(HighPrecedenceKeyHandle, 1740 KEY_READ, 1741 CmpKeyObjectType, 1742 PreviousMode, 1743 (PVOID*)&HighPrecedenceKeyObject, 1744 NULL); 1745 if (!NT_SUCCESS(Status)) 1746 goto Quit; 1747 1748 Status = ObReferenceObjectByHandle(LowPrecedenceKeyHandle, 1749 KEY_READ, 1750 CmpKeyObjectType, 1751 PreviousMode, 1752 (PVOID*)&LowPrecedenceKeyObject, 1753 NULL); 1754 if (!NT_SUCCESS(Status)) 1755 goto Quit; 1756 1757 /* Call the internal API */ 1758 Status = CmSaveMergedKeys(HighPrecedenceKeyObject->KeyControlBlock, 1759 LowPrecedenceKeyObject->KeyControlBlock, 1760 KmFileHandle); 1761 1762 Quit: 1763 /* Dereference the opened key objects */ 1764 if (LowPrecedenceKeyObject) 1765 ObDereferenceObject(LowPrecedenceKeyObject); 1766 if (HighPrecedenceKeyObject) 1767 ObDereferenceObject(HighPrecedenceKeyObject); 1768 1769 /* Close the local kernel handle */ 1770 if (KmFileHandle) 1771 ObCloseHandle(KmFileHandle, KernelMode); 1772 1773 return Status; 1774 } 1775 1776 NTSTATUS 1777 NTAPI 1778 NtSetInformationKey(IN HANDLE KeyHandle, 1779 IN KEY_SET_INFORMATION_CLASS KeyInformationClass, 1780 IN PVOID KeyInformation, 1781 IN ULONG KeyInformationLength) 1782 { 1783 UNIMPLEMENTED; 1784 return STATUS_NOT_IMPLEMENTED; 1785 } 1786 1787 NTSTATUS 1788 NTAPI 1789 NtUnloadKey(IN POBJECT_ATTRIBUTES KeyObjectAttributes) 1790 { 1791 return NtUnloadKey2(KeyObjectAttributes, 0); 1792 } 1793 1794 NTSTATUS 1795 NTAPI 1796 NtUnloadKey2(IN POBJECT_ATTRIBUTES TargetKey, 1797 IN ULONG Flags) 1798 { 1799 NTSTATUS Status; 1800 OBJECT_ATTRIBUTES CapturedTargetKey; 1801 UNICODE_STRING ObjectName; 1802 HANDLE KmTargetKeyRootDir = NULL; 1803 CM_PARSE_CONTEXT ParseContext = {0}; 1804 KPROCESSOR_MODE PreviousMode = ExGetPreviousMode(); 1805 PCM_KEY_BODY KeyBody = NULL; 1806 HANDLE Handle; 1807 1808 PAGED_CODE(); 1809 1810 /* Validate privilege */ 1811 if (!SeSinglePrivilegeCheck(SeRestorePrivilege, PreviousMode)) 1812 { 1813 DPRINT1("Restore Privilege missing!\n"); 1814 return STATUS_PRIVILEGE_NOT_HELD; 1815 } 1816 1817 /* Check for user-mode caller */ 1818 if (PreviousMode != KernelMode) 1819 { 1820 /* Prepare to probe parameters */ 1821 _SEH2_TRY 1822 { 1823 /* Probe object attributes */ 1824 ProbeForRead(TargetKey, 1825 sizeof(OBJECT_ATTRIBUTES), 1826 sizeof(ULONG)); 1827 1828 CapturedTargetKey = *TargetKey; 1829 1830 /* Probe the string */ 1831 ObjectName = ProbeForReadUnicodeString(CapturedTargetKey.ObjectName); 1832 ProbeForRead(ObjectName.Buffer, 1833 ObjectName.Length, 1834 sizeof(WCHAR)); 1835 1836 CapturedTargetKey.ObjectName = &ObjectName; 1837 } 1838 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1839 { 1840 /* Return the exception code */ 1841 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 1842 } 1843 _SEH2_END; 1844 } 1845 else 1846 { 1847 /* Save the target attributes directly */ 1848 CapturedTargetKey = *TargetKey; 1849 } 1850 1851 /* Make sure the target key root directory handle is a kernel handle */ 1852 Status = CmpConvertHandleToKernelHandle(CapturedTargetKey.RootDirectory, 1853 CmpKeyObjectType, 1854 KEY_WRITE, 1855 PreviousMode, 1856 &KmTargetKeyRootDir); 1857 if (!NT_SUCCESS(Status)) 1858 return Status; 1859 CapturedTargetKey.RootDirectory = KmTargetKeyRootDir; 1860 CapturedTargetKey.Attributes |= OBJ_KERNEL_HANDLE; 1861 1862 /* Setup the parse context */ 1863 ParseContext.CreateOperation = TRUE; 1864 ParseContext.CreateOptions = REG_OPTION_BACKUP_RESTORE; 1865 1866 /* Do the create */ 1867 /* Open a local handle to the key */ 1868 Status = ObOpenObjectByName(&CapturedTargetKey, 1869 CmpKeyObjectType, 1870 KernelMode, 1871 NULL, 1872 KEY_WRITE, 1873 &ParseContext, 1874 &Handle); 1875 if (NT_SUCCESS(Status)) 1876 { 1877 /* Reference the key object */ 1878 Status = ObReferenceObjectByHandle(Handle, 1879 KEY_WRITE, 1880 CmpKeyObjectType, 1881 KernelMode, 1882 (PVOID*)&KeyBody, 1883 NULL); 1884 1885 /* Close the handle */ 1886 ObCloseHandle(Handle, KernelMode); 1887 } 1888 1889 /* Close the local kernel handle */ 1890 if (KmTargetKeyRootDir) 1891 ObCloseHandle(KmTargetKeyRootDir, KernelMode); 1892 1893 /* Return if a failure was encountered */ 1894 if (!NT_SUCCESS(Status)) 1895 return Status; 1896 1897 /* 1898 * Lock down the entire registry when we unload a hive. 1899 * 1900 * NOTE: We might block other threads of other processes that do 1901 * operations with unrelated keys of other hives when we lock 1902 * the registry for exclusive use by the calling thread that does 1903 * the unloading. If this turns out to cause a major overhead we 1904 * have to rethink the locking mechanism here (prior commit - f1d2a44). 1905 */ 1906 CmpLockRegistryExclusive(); 1907 ExAcquirePushLockExclusive(&CmpLoadHiveLock); 1908 1909 /* Check if it's being deleted already */ 1910 if (KeyBody->KeyControlBlock->Delete) 1911 { 1912 /* Return appropriate status */ 1913 Status = STATUS_KEY_DELETED; 1914 goto Quit; 1915 } 1916 1917 /* Check if it's a read-only key */ 1918 if (KeyBody->KeyControlBlock->ExtFlags & CM_KCB_READ_ONLY_KEY) 1919 { 1920 /* Return appropriate status */ 1921 Status = STATUS_ACCESS_DENIED; 1922 goto Quit; 1923 } 1924 1925 /* Call the internal API. Note that CmUnloadKey() unlocks the registry only on success. */ 1926 Status = CmUnloadKey(KeyBody->KeyControlBlock, Flags); 1927 1928 Quit: 1929 /* If CmUnloadKey() failed we need to unlock registry ourselves */ 1930 if (!NT_SUCCESS(Status)) 1931 { 1932 /* Unlock the hive loading and registry locks */ 1933 ExReleasePushLockExclusive(&CmpLoadHiveLock); 1934 CmpUnlockRegistry(); 1935 } 1936 1937 /* Dereference the key */ 1938 ObDereferenceObject(KeyBody); 1939 1940 /* Return status */ 1941 return Status; 1942 } 1943 1944 NTSTATUS 1945 NTAPI 1946 NtUnloadKeyEx(IN POBJECT_ATTRIBUTES TargetKey, 1947 IN HANDLE Event) 1948 { 1949 UNIMPLEMENTED; 1950 return STATUS_NOT_IMPLEMENTED; 1951 } 1952 1953 /* EOF */ 1954