1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS kernel 4 * FILE: ntoskrnl/ex/sysinfo.c 5 * PURPOSE: System information functions 6 * 7 * PROGRAMMERS: David Welch (welch@mcmail.com) 8 * Aleksey Bragin (aleksey@reactos.org) 9 */ 10 11 /* INCLUDES *****************************************************************/ 12 13 #include <ntoskrnl.h> 14 #include <wmidata.h> 15 #include <wmistr.h> 16 #define NDEBUG 17 #include <debug.h> 18 19 /* The maximum size of an environment value (in bytes) */ 20 #define MAX_ENVVAL_SIZE 1024 21 22 #define SIG_ACPI 0x41435049 23 #define SIG_FIRM 0x4649524D 24 #define SIG_RSMB 0x52534D42 25 26 extern LIST_ENTRY HandleTableListHead; 27 extern EX_PUSH_LOCK HandleTableListLock; 28 29 FAST_MUTEX ExpEnvironmentLock; 30 ERESOURCE ExpFirmwareTableResource; 31 LIST_ENTRY ExpFirmwareTableProviderListHead; 32 33 FORCEINLINE 34 NTSTATUS 35 ExpConvertLdrModuleToRtlModule(IN ULONG ModuleCount, 36 IN PLDR_DATA_TABLE_ENTRY LdrEntry, 37 OUT PRTL_PROCESS_MODULE_INFORMATION ModuleInfo) 38 { 39 PCHAR p; 40 NTSTATUS Status; 41 ANSI_STRING ModuleName; 42 43 /* Fill it out */ 44 ModuleInfo->MappedBase = NULL; 45 ModuleInfo->ImageBase = LdrEntry->DllBase; 46 ModuleInfo->ImageSize = LdrEntry->SizeOfImage; 47 ModuleInfo->Flags = LdrEntry->Flags; 48 ModuleInfo->LoadCount = LdrEntry->LoadCount; 49 ModuleInfo->LoadOrderIndex = (USHORT)ModuleCount; 50 ModuleInfo->InitOrderIndex = 0; 51 52 /* Setup name */ 53 RtlInitEmptyAnsiString(&ModuleName, 54 ModuleInfo->FullPathName, 55 sizeof(ModuleInfo->FullPathName)); 56 57 /* Convert it */ 58 Status = RtlUnicodeStringToAnsiString(&ModuleName, 59 &LdrEntry->FullDllName, 60 FALSE); 61 if ((NT_SUCCESS(Status)) || (Status == STATUS_BUFFER_OVERFLOW)) 62 { 63 /* Calculate offset to name */ 64 p = ModuleName.Buffer + ModuleName.Length; 65 while ((p > ModuleName.Buffer) && (*--p)) 66 { 67 /* Check if we found the separator */ 68 if (*p == OBJ_NAME_PATH_SEPARATOR) 69 { 70 /* We did, break out */ 71 p++; 72 break; 73 } 74 } 75 76 /* Set the offset */ 77 ModuleInfo->OffsetToFileName = (USHORT)(p - ModuleName.Buffer); 78 } 79 else 80 { 81 /* Return empty name */ 82 ModuleInfo->FullPathName[0] = ANSI_NULL; 83 ModuleInfo->OffsetToFileName = 0; 84 } 85 86 return Status; 87 } 88 89 NTSTATUS 90 NTAPI 91 ExpQueryModuleInformation(IN PLIST_ENTRY KernelModeList, 92 IN PLIST_ENTRY UserModeList, 93 OUT PRTL_PROCESS_MODULES Modules, 94 IN ULONG Length, 95 OUT PULONG ReturnLength) 96 { 97 NTSTATUS Status = STATUS_SUCCESS; 98 ULONG RequiredLength; 99 PRTL_PROCESS_MODULE_INFORMATION ModuleInfo; 100 PLDR_DATA_TABLE_ENTRY LdrEntry; 101 ULONG ModuleCount = 0; 102 PLIST_ENTRY NextEntry; 103 104 /* Setup defaults */ 105 RequiredLength = FIELD_OFFSET(RTL_PROCESS_MODULES, Modules); 106 ModuleInfo = &Modules->Modules[0]; 107 108 /* Loop the kernel list */ 109 NextEntry = KernelModeList->Flink; 110 while (NextEntry != KernelModeList) 111 { 112 /* Get the entry */ 113 LdrEntry = CONTAINING_RECORD(NextEntry, 114 LDR_DATA_TABLE_ENTRY, 115 InLoadOrderLinks); 116 117 /* Update size and check if we can manage one more entry */ 118 RequiredLength += sizeof(RTL_PROCESS_MODULE_INFORMATION); 119 if (Length >= RequiredLength) 120 { 121 Status = ExpConvertLdrModuleToRtlModule(ModuleCount, 122 LdrEntry, 123 ModuleInfo); 124 125 /* Go to the next module */ 126 ModuleInfo++; 127 } 128 else 129 { 130 /* Set error code */ 131 Status = STATUS_INFO_LENGTH_MISMATCH; 132 } 133 134 /* Update count and move to next entry */ 135 ModuleCount++; 136 NextEntry = NextEntry->Flink; 137 } 138 139 /* Check if caller also wanted user modules */ 140 if (UserModeList) 141 { 142 NextEntry = UserModeList->Flink; 143 while (NextEntry != UserModeList) 144 { 145 /* Get the entry */ 146 LdrEntry = CONTAINING_RECORD(NextEntry, 147 LDR_DATA_TABLE_ENTRY, 148 InLoadOrderLinks); 149 150 /* Update size and check if we can manage one more entry */ 151 RequiredLength += sizeof(RTL_PROCESS_MODULE_INFORMATION); 152 if (Length >= RequiredLength) 153 { 154 Status = ExpConvertLdrModuleToRtlModule(ModuleCount, 155 LdrEntry, 156 ModuleInfo); 157 158 /* Go to the next module */ 159 ModuleInfo++; 160 } 161 else 162 { 163 /* Set error code */ 164 Status = STATUS_INFO_LENGTH_MISMATCH; 165 } 166 167 /* Update count and move to next entry */ 168 ModuleCount++; 169 NextEntry = NextEntry->Flink; 170 } 171 } 172 173 /* Update return length */ 174 if (ReturnLength) *ReturnLength = RequiredLength; 175 176 /* Validate the length again */ 177 if (Length >= FIELD_OFFSET(RTL_PROCESS_MODULES, Modules)) 178 { 179 /* Set the final count */ 180 Modules->NumberOfModules = ModuleCount; 181 } 182 else 183 { 184 /* Otherwise, we failed */ 185 Status = STATUS_INFO_LENGTH_MISMATCH; 186 } 187 188 /* Done */ 189 return Status; 190 } 191 192 VOID 193 NTAPI 194 ExUnlockUserBuffer(PMDL Mdl) 195 { 196 MmUnlockPages(Mdl); 197 ExFreePoolWithTag(Mdl, TAG_MDL); 198 } 199 200 NTSTATUS 201 NTAPI 202 ExLockUserBuffer( 203 PVOID BaseAddress, 204 ULONG Length, 205 KPROCESSOR_MODE AccessMode, 206 LOCK_OPERATION Operation, 207 PVOID *MappedSystemVa, 208 PMDL *OutMdl) 209 { 210 PMDL Mdl; 211 PAGED_CODE(); 212 213 *MappedSystemVa = NULL; 214 *OutMdl = NULL; 215 216 /* Allocate an MDL for the buffer */ 217 Mdl = IoAllocateMdl(BaseAddress, Length, FALSE, TRUE, NULL); 218 if (Mdl == NULL) 219 { 220 return STATUS_INSUFFICIENT_RESOURCES; 221 } 222 223 /* Enter SEH for probing */ 224 _SEH2_TRY 225 { 226 MmProbeAndLockPages(Mdl, AccessMode, Operation); 227 } 228 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 229 { 230 ExFreePoolWithTag(Mdl, TAG_MDL); 231 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 232 } 233 _SEH2_END; 234 235 /* Return the safe kernel mode buffer */ 236 *MappedSystemVa = MmGetSystemAddressForMdlSafe(Mdl, NormalPagePriority); 237 if (*MappedSystemVa == NULL) 238 { 239 ExUnlockUserBuffer(Mdl); 240 return STATUS_INSUFFICIENT_RESOURCES; 241 } 242 243 /* Return the MDL */ 244 *OutMdl = Mdl; 245 return STATUS_SUCCESS; 246 } 247 248 NTSTATUS 249 NTAPI 250 ExpGetRawSMBiosTable( 251 _Out_opt_ PVOID Buffer, 252 _Out_ ULONG * OutSize, 253 _In_ ULONG BufferSize) 254 { 255 NTSTATUS Status; 256 PVOID DataBlockObject; 257 PWNODE_ALL_DATA AllData; 258 ULONG WMIBufSize; 259 260 ASSERT(OutSize != NULL); 261 *OutSize = 0; 262 263 /* Open the data block object for the SMBIOS table */ 264 Status = IoWMIOpenBlock(&MSSmBios_RawSMBiosTables_GUID, 265 WMIGUID_QUERY, 266 &DataBlockObject); 267 if (!NT_SUCCESS(Status)) 268 { 269 DPRINT1("IoWMIOpenBlock failed: 0x%08lx\n", Status); 270 return Status; 271 } 272 273 /* Query the required buffer size */ 274 WMIBufSize = 0; 275 Status = IoWMIQueryAllData(DataBlockObject, &WMIBufSize, NULL); 276 if (!NT_SUCCESS(Status)) 277 { 278 DPRINT1("IoWMIOpenBlock failed: 0x%08lx\n", Status); 279 return Status; 280 } 281 282 AllData = ExAllocatePoolWithTag(PagedPool, WMIBufSize, 'itfS'); 283 if (AllData == NULL) 284 { 285 DPRINT1("Failed to allocate %lu bytes for SMBIOS tables\n", WMIBufSize); 286 return STATUS_INSUFFICIENT_RESOURCES; 287 } 288 289 /* Query the buffer data */ 290 Status = IoWMIQueryAllData(DataBlockObject, &WMIBufSize, AllData); 291 if (!NT_SUCCESS(Status)) 292 { 293 DPRINT1("IoWMIOpenBlock failed: 0x%08lx\n", Status); 294 ExFreePoolWithTag(AllData, 'itfS'); 295 return Status; 296 } 297 298 Status = STATUS_SUCCESS; 299 *OutSize = AllData->FixedInstanceSize; 300 if (Buffer != NULL) 301 { 302 if (BufferSize >= *OutSize) 303 { 304 RtlMoveMemory(Buffer, AllData + 1, *OutSize); 305 } 306 else 307 { 308 Status = STATUS_BUFFER_TOO_SMALL; 309 } 310 } 311 312 /* Free the buffer */ 313 ExFreePoolWithTag(AllData, 'itfS'); 314 return Status; 315 } 316 317 /* FUNCTIONS *****************************************************************/ 318 319 /* 320 * @implemented 321 */ 322 VOID 323 NTAPI 324 ExGetCurrentProcessorCpuUsage(PULONG CpuUsage) 325 { 326 PKPRCB Prcb; 327 ULONG TotalTime; 328 ULONGLONG ScaledIdle; 329 330 Prcb = KeGetCurrentPrcb(); 331 332 ScaledIdle = (ULONGLONG)Prcb->IdleThread->KernelTime * 100; 333 TotalTime = Prcb->KernelTime + Prcb->UserTime; 334 if (TotalTime != 0) 335 *CpuUsage = (ULONG)(100 - (ScaledIdle / TotalTime)); 336 else 337 *CpuUsage = 0; 338 } 339 340 /* 341 * @implemented 342 */ 343 VOID 344 NTAPI 345 ExGetCurrentProcessorCounts(PULONG IdleTime, 346 PULONG KernelAndUserTime, 347 PULONG ProcessorNumber) 348 { 349 PKPRCB Prcb; 350 351 Prcb = KeGetCurrentPrcb(); 352 353 *IdleTime = Prcb->IdleThread->KernelTime; 354 *KernelAndUserTime = Prcb->KernelTime + Prcb->UserTime; 355 *ProcessorNumber = (ULONG)Prcb->Number; 356 } 357 358 /* 359 * @implemented 360 */ 361 BOOLEAN 362 NTAPI 363 ExIsProcessorFeaturePresent(IN ULONG ProcessorFeature) 364 { 365 /* Quick check to see if it exists at all */ 366 if (ProcessorFeature >= PROCESSOR_FEATURE_MAX) return(FALSE); 367 368 /* Return our support for it */ 369 return(SharedUserData->ProcessorFeatures[ProcessorFeature]); 370 } 371 372 /* 373 * @implemented 374 */ 375 BOOLEAN 376 NTAPI 377 ExVerifySuite(SUITE_TYPE SuiteType) 378 { 379 if (SuiteType == Personal) return TRUE; 380 return FALSE; 381 } 382 383 NTSTATUS 384 NTAPI 385 NtQuerySystemEnvironmentValue(IN PUNICODE_STRING VariableName, 386 OUT PWSTR ValueBuffer, 387 IN ULONG ValueBufferLength, 388 IN OUT PULONG ReturnLength OPTIONAL) 389 { 390 ANSI_STRING AName; 391 UNICODE_STRING WName; 392 ARC_STATUS Result; 393 PCH AnsiValueBuffer; 394 ANSI_STRING AValue; 395 UNICODE_STRING WValue; 396 KPROCESSOR_MODE PreviousMode; 397 NTSTATUS Status; 398 PAGED_CODE(); 399 400 /* Check if the call came from user mode */ 401 PreviousMode = ExGetPreviousMode(); 402 if (PreviousMode != KernelMode) 403 { 404 _SEH2_TRY 405 { 406 /* Probe the input and output buffers */ 407 ProbeForRead(VariableName, sizeof(UNICODE_STRING), sizeof(ULONG)); 408 ProbeForWrite(ValueBuffer, ValueBufferLength, sizeof(WCHAR)); 409 if (ReturnLength != NULL) ProbeForWriteUlong(ReturnLength); 410 } 411 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 412 { 413 /* Return the exception code */ 414 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 415 } 416 _SEH2_END; 417 } 418 419 /* According to NTInternals the SeSystemEnvironmentName privilege is required! */ 420 if (!SeSinglePrivilegeCheck(SeSystemEnvironmentPrivilege, PreviousMode)) 421 { 422 DPRINT1("NtQuerySystemEnvironmentValue: Caller requires the SeSystemEnvironmentPrivilege privilege!\n"); 423 return STATUS_PRIVILEGE_NOT_HELD; 424 } 425 426 /* Copy the name to kernel space if necessary */ 427 Status = ProbeAndCaptureUnicodeString(&WName, PreviousMode, VariableName); 428 if (!NT_SUCCESS(Status)) return Status; 429 430 /* Convert the name to ANSI and release the captured UNICODE string */ 431 Status = RtlUnicodeStringToAnsiString(&AName, &WName, TRUE); 432 ReleaseCapturedUnicodeString(&WName, PreviousMode); 433 if (!NT_SUCCESS(Status)) return Status; 434 435 /* Allocate a buffer for the ANSI environment variable */ 436 AnsiValueBuffer = ExAllocatePoolWithTag(NonPagedPool, MAX_ENVVAL_SIZE, 'rvnE'); 437 if (AnsiValueBuffer == NULL) 438 { 439 RtlFreeAnsiString(&AName); 440 return STATUS_INSUFFICIENT_RESOURCES; 441 } 442 443 /* Get the environment variable and free the ANSI name */ 444 Result = HalGetEnvironmentVariable(AName.Buffer, 445 MAX_ENVVAL_SIZE, 446 AnsiValueBuffer); 447 RtlFreeAnsiString(&AName); 448 449 /* Check if we had success */ 450 if (Result == ESUCCESS) 451 { 452 /* Copy the result back to the caller. */ 453 _SEH2_TRY 454 { 455 /* Initialize ANSI string from the result */ 456 RtlInitAnsiString(&AValue, AnsiValueBuffer); 457 458 /* Initialize a UNICODE string from the callers buffer */ 459 RtlInitEmptyUnicodeString(&WValue, ValueBuffer, (USHORT)ValueBufferLength); 460 461 /* Convert the result to UNICODE */ 462 Status = RtlAnsiStringToUnicodeString(&WValue, &AValue, FALSE); 463 464 if (ReturnLength != NULL) 465 *ReturnLength = WValue.Length; 466 } 467 _SEH2_EXCEPT(ExSystemExceptionFilter()) 468 { 469 Status = _SEH2_GetExceptionCode(); 470 } 471 _SEH2_END; 472 } 473 else 474 { 475 Status = STATUS_UNSUCCESSFUL; 476 } 477 478 /* Free the allocated ANSI value buffer */ 479 ExFreePoolWithTag(AnsiValueBuffer, 'rvnE'); 480 481 return Status; 482 } 483 484 485 NTSTATUS 486 NTAPI 487 NtSetSystemEnvironmentValue(IN PUNICODE_STRING VariableName, 488 IN PUNICODE_STRING Value) 489 { 490 UNICODE_STRING CapturedName, CapturedValue; 491 ANSI_STRING AName, AValue; 492 KPROCESSOR_MODE PreviousMode; 493 NTSTATUS Status; 494 495 PAGED_CODE(); 496 497 PreviousMode = ExGetPreviousMode(); 498 499 /* 500 * Copy the strings to kernel space if necessary 501 */ 502 Status = ProbeAndCaptureUnicodeString(&CapturedName, 503 PreviousMode, 504 VariableName); 505 if (NT_SUCCESS(Status)) 506 { 507 Status = ProbeAndCaptureUnicodeString(&CapturedValue, 508 PreviousMode, 509 Value); 510 if (NT_SUCCESS(Status)) 511 { 512 /* 513 * according to ntinternals the SeSystemEnvironmentName privilege is required! 514 */ 515 if (SeSinglePrivilegeCheck(SeSystemEnvironmentPrivilege, 516 PreviousMode)) 517 { 518 /* 519 * convert the strings to ANSI 520 */ 521 Status = RtlUnicodeStringToAnsiString(&AName, 522 &CapturedName, 523 TRUE); 524 if (NT_SUCCESS(Status)) 525 { 526 Status = RtlUnicodeStringToAnsiString(&AValue, 527 &CapturedValue, 528 TRUE); 529 if (NT_SUCCESS(Status)) 530 { 531 ARC_STATUS Result = HalSetEnvironmentVariable(AName.Buffer, 532 AValue.Buffer); 533 534 Status = (Result ? STATUS_SUCCESS : STATUS_UNSUCCESSFUL); 535 } 536 } 537 } 538 else 539 { 540 DPRINT1("NtSetSystemEnvironmentValue: Caller requires the SeSystemEnvironmentPrivilege privilege!\n"); 541 Status = STATUS_PRIVILEGE_NOT_HELD; 542 } 543 544 ReleaseCapturedUnicodeString(&CapturedValue, 545 PreviousMode); 546 } 547 548 ReleaseCapturedUnicodeString(&CapturedName, 549 PreviousMode); 550 } 551 552 return Status; 553 } 554 555 NTSTATUS 556 NTAPI 557 NtEnumerateSystemEnvironmentValuesEx(IN ULONG InformationClass, 558 IN PVOID Buffer, 559 IN ULONG BufferLength) 560 { 561 UNIMPLEMENTED; 562 return STATUS_NOT_IMPLEMENTED; 563 } 564 565 NTSTATUS 566 NTAPI 567 NtQuerySystemEnvironmentValueEx( 568 _In_ PUNICODE_STRING VariableName, 569 _In_ LPGUID VendorGuid, 570 _Out_opt_ PVOID Value, 571 _Inout_ PULONG ReturnLength, 572 _Out_opt_ PULONG Attributes) 573 { 574 UNIMPLEMENTED; 575 return STATUS_NOT_IMPLEMENTED; 576 } 577 578 NTSTATUS 579 NTAPI 580 NtSetSystemEnvironmentValueEx( 581 _In_ PUNICODE_STRING VariableName, 582 _In_ LPGUID VendorGuid, 583 _In_reads_bytes_opt_(ValueLength) PVOID Value, 584 _In_ ULONG ValueLength, 585 _In_ ULONG Attributes) 586 { 587 UNIMPLEMENTED; 588 return STATUS_NOT_IMPLEMENTED; 589 } 590 591 /* --- Query/Set System Information --- */ 592 593 /* 594 * NOTE: QSI_DEF(n) and SSI_DEF(n) define _cdecl function symbols 595 * so the stack is popped only in one place on x86 platform. 596 */ 597 #define QSI_USE(n) QSI##n 598 #define QSI_DEF(n) \ 599 static NTSTATUS QSI_USE(n) (PVOID Buffer, ULONG Size, PULONG ReqSize) 600 601 #define SSI_USE(n) SSI##n 602 #define SSI_DEF(n) \ 603 static NTSTATUS SSI_USE(n) (PVOID Buffer, ULONG Size) 604 605 VOID 606 NTAPI 607 ExQueryPoolUsage(OUT PULONG PagedPoolPages, 608 OUT PULONG NonPagedPoolPages, 609 OUT PULONG PagedPoolAllocs, 610 OUT PULONG PagedPoolFrees, 611 OUT PULONG PagedPoolLookasideHits, 612 OUT PULONG NonPagedPoolAllocs, 613 OUT PULONG NonPagedPoolFrees, 614 OUT PULONG NonPagedPoolLookasideHits); 615 616 /* Class 0 - Basic Information */ 617 QSI_DEF(SystemBasicInformation) 618 { 619 PSYSTEM_BASIC_INFORMATION Sbi 620 = (PSYSTEM_BASIC_INFORMATION) Buffer; 621 622 *ReqSize = sizeof(SYSTEM_BASIC_INFORMATION); 623 624 /* Check user buffer's size */ 625 if (Size != sizeof(SYSTEM_BASIC_INFORMATION)) 626 { 627 return STATUS_INFO_LENGTH_MISMATCH; 628 } 629 630 RtlZeroMemory(Sbi, Size); 631 Sbi->Reserved = 0; 632 Sbi->TimerResolution = KeMaximumIncrement; 633 Sbi->PageSize = PAGE_SIZE; 634 Sbi->NumberOfPhysicalPages = MmNumberOfPhysicalPages; 635 Sbi->LowestPhysicalPageNumber = (ULONG)MmLowestPhysicalPage; 636 Sbi->HighestPhysicalPageNumber = (ULONG)MmHighestPhysicalPage; 637 Sbi->AllocationGranularity = MM_VIRTMEM_GRANULARITY; /* hard coded on Intel? */ 638 Sbi->MinimumUserModeAddress = 0x10000; /* Top of 64k */ 639 Sbi->MaximumUserModeAddress = (ULONG_PTR)MmHighestUserAddress; 640 Sbi->ActiveProcessorsAffinityMask = KeActiveProcessors; 641 Sbi->NumberOfProcessors = KeNumberProcessors; 642 643 return STATUS_SUCCESS; 644 } 645 646 /* Class 1 - Processor Information */ 647 QSI_DEF(SystemProcessorInformation) 648 { 649 PSYSTEM_PROCESSOR_INFORMATION Spi 650 = (PSYSTEM_PROCESSOR_INFORMATION) Buffer; 651 652 *ReqSize = sizeof(SYSTEM_PROCESSOR_INFORMATION); 653 654 /* Check user buffer's size */ 655 if (Size < sizeof(SYSTEM_PROCESSOR_INFORMATION)) 656 { 657 return STATUS_INFO_LENGTH_MISMATCH; 658 } 659 Spi->ProcessorArchitecture = KeProcessorArchitecture; 660 Spi->ProcessorLevel = KeProcessorLevel; 661 Spi->ProcessorRevision = KeProcessorRevision; 662 #if (NTDDI_VERSION < NTDDI_WIN8) 663 Spi->Reserved = 0; 664 #else 665 Spi->MaximumProcessors = 0; 666 #endif 667 Spi->ProcessorFeatureBits = KeFeatureBits; 668 669 DPRINT("Arch %u Level %u Rev 0x%x\n", Spi->ProcessorArchitecture, 670 Spi->ProcessorLevel, Spi->ProcessorRevision); 671 672 return STATUS_SUCCESS; 673 } 674 675 /* Class 2 - Performance Information */ 676 QSI_DEF(SystemPerformanceInformation) 677 { 678 LONG i; 679 ULONG IdleUser, IdleKernel; 680 PKPRCB Prcb; 681 PSYSTEM_PERFORMANCE_INFORMATION Spi 682 = (PSYSTEM_PERFORMANCE_INFORMATION) Buffer; 683 684 PEPROCESS TheIdleProcess; 685 686 *ReqSize = sizeof(SYSTEM_PERFORMANCE_INFORMATION); 687 688 /* Check user buffer's size */ 689 if (Size < sizeof(SYSTEM_PERFORMANCE_INFORMATION)) 690 { 691 return STATUS_INFO_LENGTH_MISMATCH; 692 } 693 694 TheIdleProcess = PsIdleProcess; 695 696 IdleKernel = KeQueryRuntimeProcess(&TheIdleProcess->Pcb, &IdleUser); 697 Spi->IdleProcessTime.QuadPart = UInt32x32To64(IdleKernel, KeMaximumIncrement); 698 Spi->IoReadTransferCount = IoReadTransferCount; 699 Spi->IoWriteTransferCount = IoWriteTransferCount; 700 Spi->IoOtherTransferCount = IoOtherTransferCount; 701 Spi->IoReadOperationCount = IoReadOperationCount; 702 Spi->IoWriteOperationCount = IoWriteOperationCount; 703 Spi->IoOtherOperationCount = IoOtherOperationCount; 704 for (i = 0; i < KeNumberProcessors; i ++) 705 { 706 Prcb = KiProcessorBlock[i]; 707 if (Prcb) 708 { 709 Spi->IoReadTransferCount.QuadPart += Prcb->IoReadTransferCount.QuadPart; 710 Spi->IoWriteTransferCount.QuadPart += Prcb->IoWriteTransferCount.QuadPart; 711 Spi->IoOtherTransferCount.QuadPart += Prcb->IoOtherTransferCount.QuadPart; 712 Spi->IoReadOperationCount += Prcb->IoReadOperationCount; 713 Spi->IoWriteOperationCount += Prcb->IoWriteOperationCount; 714 Spi->IoOtherOperationCount += Prcb->IoOtherOperationCount; 715 } 716 } 717 718 Spi->AvailablePages = (ULONG)MmAvailablePages; 719 720 Spi->CommittedPages = MmTotalCommittedPages; 721 /* 722 * Add up the full system total + pagefile. 723 * All this make Taskmgr happy but not sure it is the right numbers. 724 * This too, fixes some of GlobalMemoryStatusEx numbers. 725 */ 726 Spi->CommitLimit = MmNumberOfPhysicalPages + MiFreeSwapPages + MiUsedSwapPages; 727 728 Spi->PeakCommitment = MmPeakCommitment; 729 Spi->PageFaultCount = 0; /* FIXME */ 730 Spi->CopyOnWriteCount = 0; /* FIXME */ 731 Spi->TransitionCount = 0; /* FIXME */ 732 Spi->CacheTransitionCount = 0; /* FIXME */ 733 Spi->DemandZeroCount = 0; /* FIXME */ 734 Spi->PageReadCount = 0; /* FIXME */ 735 Spi->PageReadIoCount = 0; /* FIXME */ 736 Spi->CacheReadCount = 0; /* FIXME */ 737 Spi->CacheIoCount = 0; /* FIXME */ 738 Spi->DirtyPagesWriteCount = 0; /* FIXME */ 739 Spi->DirtyWriteIoCount = 0; /* FIXME */ 740 Spi->MappedPagesWriteCount = 0; /* FIXME */ 741 Spi->MappedWriteIoCount = 0; /* FIXME */ 742 743 Spi->PagedPoolPages = 0; 744 Spi->NonPagedPoolPages = 0; 745 Spi->PagedPoolAllocs = 0; 746 Spi->PagedPoolFrees = 0; 747 Spi->PagedPoolLookasideHits = 0; 748 Spi->NonPagedPoolAllocs = 0; 749 Spi->NonPagedPoolFrees = 0; 750 Spi->NonPagedPoolLookasideHits = 0; 751 ExQueryPoolUsage(&Spi->PagedPoolPages, 752 &Spi->NonPagedPoolPages, 753 &Spi->PagedPoolAllocs, 754 &Spi->PagedPoolFrees, 755 &Spi->PagedPoolLookasideHits, 756 &Spi->NonPagedPoolAllocs, 757 &Spi->NonPagedPoolFrees, 758 &Spi->NonPagedPoolLookasideHits); 759 Spi->FreeSystemPtes = 0; /* FIXME */ 760 761 Spi->ResidentSystemCodePage = 0; /* FIXME */ 762 763 Spi->TotalSystemDriverPages = 0; /* FIXME */ 764 Spi->Spare3Count = 0; /* FIXME */ 765 766 Spi->ResidentSystemCachePage = MiMemoryConsumers[MC_USER].PagesUsed; /* FIXME */ 767 Spi->ResidentPagedPoolPage = 0; /* FIXME */ 768 769 Spi->ResidentSystemDriverPage = 0; /* FIXME */ 770 Spi->CcFastReadNoWait = 0; /* FIXME */ 771 Spi->CcFastReadWait = 0; /* FIXME */ 772 Spi->CcFastReadResourceMiss = 0; /* FIXME */ 773 Spi->CcFastReadNotPossible = 0; /* FIXME */ 774 775 Spi->CcFastMdlReadNoWait = 0; /* FIXME */ 776 Spi->CcFastMdlReadWait = 0; /* FIXME */ 777 Spi->CcFastMdlReadResourceMiss = 0; /* FIXME */ 778 Spi->CcFastMdlReadNotPossible = 0; /* FIXME */ 779 780 Spi->CcMapDataNoWait = CcMapDataNoWait; 781 Spi->CcMapDataWait = CcMapDataWait; 782 Spi->CcMapDataNoWaitMiss = 0; /* FIXME */ 783 Spi->CcMapDataWaitMiss = 0; /* FIXME */ 784 785 Spi->CcPinMappedDataCount = CcPinMappedDataCount; 786 Spi->CcPinReadNoWait = CcPinReadNoWait; 787 Spi->CcPinReadWait = CcPinReadWait; 788 Spi->CcPinReadNoWaitMiss = 0; /* FIXME */ 789 Spi->CcPinReadWaitMiss = 0; /* FIXME */ 790 Spi->CcCopyReadNoWait = 0; /* FIXME */ 791 Spi->CcCopyReadWait = 0; /* FIXME */ 792 Spi->CcCopyReadNoWaitMiss = 0; /* FIXME */ 793 Spi->CcCopyReadWaitMiss = 0; /* FIXME */ 794 795 Spi->CcMdlReadNoWait = 0; /* FIXME */ 796 Spi->CcMdlReadWait = 0; /* FIXME */ 797 Spi->CcMdlReadNoWaitMiss = 0; /* FIXME */ 798 Spi->CcMdlReadWaitMiss = 0; /* FIXME */ 799 Spi->CcReadAheadIos = 0; /* FIXME */ 800 Spi->CcLazyWriteIos = CcLazyWriteIos; 801 Spi->CcLazyWritePages = CcLazyWritePages; 802 Spi->CcDataFlushes = CcDataFlushes; 803 Spi->CcDataPages = CcDataPages; 804 805 Spi->ContextSwitches = 0; 806 Spi->FirstLevelTbFills = 0; 807 Spi->SecondLevelTbFills = 0; 808 Spi->SystemCalls = 0; 809 for (i = 0; i < KeNumberProcessors; i ++) 810 { 811 Prcb = KiProcessorBlock[i]; 812 if (Prcb) 813 { 814 Spi->ContextSwitches += KeGetContextSwitches(Prcb); 815 Spi->FirstLevelTbFills += Prcb->KeFirstLevelTbFills; 816 Spi->SecondLevelTbFills += Prcb->KeSecondLevelTbFills; 817 Spi->SystemCalls += Prcb->KeSystemCalls; 818 } 819 } 820 821 return STATUS_SUCCESS; 822 } 823 824 /* Class 3 - Time Of Day Information */ 825 QSI_DEF(SystemTimeOfDayInformation) 826 { 827 SYSTEM_TIMEOFDAY_INFORMATION Sti; 828 LARGE_INTEGER CurrentTime; 829 830 /* Set amount of written information to 0 */ 831 *ReqSize = 0; 832 833 /* Check user buffer's size */ 834 if (Size > sizeof(SYSTEM_TIMEOFDAY_INFORMATION)) 835 { 836 return STATUS_INFO_LENGTH_MISMATCH; 837 } 838 839 /* Get current time */ 840 KeQuerySystemTime(&CurrentTime); 841 842 /* Zero local buffer */ 843 RtlZeroMemory(&Sti, sizeof(SYSTEM_TIMEOFDAY_INFORMATION)); 844 845 /* Fill local time structure */ 846 Sti.BootTime= KeBootTime; 847 Sti.CurrentTime = CurrentTime; 848 Sti.TimeZoneBias.QuadPart = ExpTimeZoneBias.QuadPart; 849 Sti.TimeZoneId = ExpTimeZoneId; 850 Sti.Reserved = 0; 851 852 /* Copy as much as requested by caller */ 853 RtlCopyMemory(Buffer, &Sti, Size); 854 855 /* Set amount of information we copied */ 856 *ReqSize = Size; 857 858 return STATUS_SUCCESS; 859 } 860 861 /* Class 4 - Path Information (DEPRECATED) */ 862 QSI_DEF(SystemPathInformation) 863 { 864 /* 865 * Since NT 3.51, this information class is trivially implemented. 866 * The path to the NT directory is now stored in KUSER_SHARED_DATA 867 * as the NtSystemRoot member. 868 * Windows Checked builds show the following message and break to 869 * the debugger before failing the function as not implemented. 870 */ 871 #if DBG 872 DPRINT1("EX: SystemPathInformation now available via SharedUserData\n"); 873 // DbgBreakPoint(); // Not needed in ReactOS. 874 #endif 875 return STATUS_NOT_IMPLEMENTED; 876 } 877 878 /* Class 5 - Process Information */ 879 QSI_DEF(SystemProcessInformation) 880 { 881 PSYSTEM_PROCESS_INFORMATION SpiCurrent; 882 PSYSTEM_THREAD_INFORMATION ThreadInfo; 883 PEPROCESS Process = NULL, SystemProcess; 884 PETHREAD CurrentThread; 885 ANSI_STRING ImageName; 886 ULONG CurrentSize; 887 USHORT ImageNameMaximumLength; // image name length in bytes 888 USHORT ImageNameLength; 889 PLIST_ENTRY CurrentEntry; 890 ULONG TotalSize = 0, ThreadsCount; 891 ULONG TotalUser, TotalKernel; 892 PUCHAR Current; 893 NTSTATUS Status = STATUS_SUCCESS; 894 PUNICODE_STRING TempProcessImageName; 895 _SEH2_VOLATILE PUNICODE_STRING ProcessImageName = NULL; 896 PWCHAR szSrc; 897 BOOLEAN Overflow = FALSE; 898 899 _SEH2_TRY 900 { 901 /* scan the process list */ 902 903 PSYSTEM_PROCESS_INFORMATION Spi 904 = (PSYSTEM_PROCESS_INFORMATION) Buffer; 905 906 *ReqSize = sizeof(SYSTEM_PROCESS_INFORMATION); 907 908 /* Check for overflow */ 909 if (Size < sizeof(SYSTEM_PROCESS_INFORMATION)) 910 { 911 Overflow = TRUE; 912 } 913 914 /* Zero user's buffer */ 915 if (!Overflow) RtlZeroMemory(Spi, Size); 916 917 SystemProcess = PsIdleProcess; 918 Process = SystemProcess; 919 Current = (PUCHAR) Spi; 920 921 do 922 { 923 SpiCurrent = (PSYSTEM_PROCESS_INFORMATION) Current; 924 925 /* Lock the Process */ 926 KeEnterCriticalRegion(); 927 ExAcquirePushLockShared(&Process->ProcessLock); 928 929 if ((Process->ProcessExiting) && 930 (Process->Pcb.Header.SignalState) && 931 !(Process->ActiveThreads) && 932 (IsListEmpty(&Process->Pcb.ThreadListHead))) 933 { 934 DPRINT1("Process %p (%s:%p) is a zombie\n", 935 Process, Process->ImageFileName, Process->UniqueProcessId); 936 CurrentSize = 0; 937 ImageNameMaximumLength = 0; 938 939 /* Unlock the Process */ 940 ExReleasePushLockShared(&Process->ProcessLock); 941 KeLeaveCriticalRegion(); 942 goto Skip; 943 } 944 945 ThreadsCount = 0; 946 CurrentEntry = Process->Pcb.ThreadListHead.Flink; 947 while (CurrentEntry != &Process->Pcb.ThreadListHead) 948 { 949 ThreadsCount++; 950 CurrentEntry = CurrentEntry->Flink; 951 } 952 953 // size of the structure for every process 954 CurrentSize = sizeof(SYSTEM_PROCESS_INFORMATION) + sizeof(SYSTEM_THREAD_INFORMATION) * ThreadsCount; 955 ImageNameLength = 0; 956 Status = SeLocateProcessImageName(Process, &TempProcessImageName); 957 ProcessImageName = TempProcessImageName; 958 szSrc = NULL; 959 if (NT_SUCCESS(Status) && (ProcessImageName->Length > 0)) 960 { 961 szSrc = (PWCHAR)((PCHAR)ProcessImageName->Buffer + ProcessImageName->Length); 962 /* Loop the file name*/ 963 while (szSrc > ProcessImageName->Buffer) 964 { 965 /* Make sure this isn't a backslash */ 966 if (*--szSrc == OBJ_NAME_PATH_SEPARATOR) 967 { 968 szSrc++; 969 break; 970 } 971 else 972 { 973 ImageNameLength += sizeof(WCHAR); 974 } 975 } 976 } 977 if (!ImageNameLength && Process != PsIdleProcess) 978 { 979 ImageNameLength = (USHORT)strlen(Process->ImageFileName) * sizeof(WCHAR); 980 } 981 982 /* Round up the image name length as NT does */ 983 if (ImageNameLength > 0) 984 ImageNameMaximumLength = ROUND_UP(ImageNameLength + sizeof(WCHAR), 8); 985 else 986 ImageNameMaximumLength = 0; 987 988 TotalSize += CurrentSize + ImageNameMaximumLength; 989 990 /* Check for overflow */ 991 if (TotalSize > Size) 992 { 993 Overflow = TRUE; 994 } 995 996 /* Fill system information */ 997 if (!Overflow) 998 { 999 SpiCurrent->NextEntryOffset = CurrentSize + ImageNameMaximumLength; // relative offset to the beginning of the next structure 1000 SpiCurrent->NumberOfThreads = ThreadsCount; 1001 SpiCurrent->CreateTime = Process->CreateTime; 1002 SpiCurrent->ImageName.Length = ImageNameLength; 1003 SpiCurrent->ImageName.MaximumLength = ImageNameMaximumLength; 1004 SpiCurrent->ImageName.Buffer = (void*)(Current + CurrentSize); 1005 1006 /* Copy name to the end of the struct */ 1007 if(Process != PsIdleProcess) 1008 { 1009 if (szSrc) 1010 { 1011 RtlCopyMemory(SpiCurrent->ImageName.Buffer, szSrc, SpiCurrent->ImageName.Length); 1012 } 1013 else 1014 { 1015 RtlInitAnsiString(&ImageName, Process->ImageFileName); 1016 Status = RtlAnsiStringToUnicodeString(&SpiCurrent->ImageName, &ImageName, FALSE); 1017 if (!NT_SUCCESS(Status)) 1018 { 1019 SpiCurrent->ImageName.Length = 0; 1020 } 1021 } 1022 } 1023 else 1024 { 1025 RtlInitUnicodeString(&SpiCurrent->ImageName, NULL); 1026 } 1027 1028 SpiCurrent->BasePriority = Process->Pcb.BasePriority; 1029 SpiCurrent->UniqueProcessId = Process->UniqueProcessId; 1030 SpiCurrent->InheritedFromUniqueProcessId = Process->InheritedFromUniqueProcessId; 1031 1032 /* PsIdleProcess shares its handle table with PsInitialSystemProcess, 1033 * so return the handle count for System only, not Idle one. */ 1034 SpiCurrent->HandleCount = (Process == PsIdleProcess) ? 0 : ObGetProcessHandleCount(Process); 1035 1036 SpiCurrent->PeakVirtualSize = Process->PeakVirtualSize; 1037 SpiCurrent->VirtualSize = Process->VirtualSize; 1038 SpiCurrent->PageFaultCount = Process->Vm.PageFaultCount; 1039 SpiCurrent->PeakWorkingSetSize = Process->Vm.PeakWorkingSetSize; 1040 SpiCurrent->WorkingSetSize = Process->Vm.WorkingSetSize; 1041 SpiCurrent->QuotaPeakPagedPoolUsage = Process->QuotaPeak[PsPagedPool]; 1042 SpiCurrent->QuotaPagedPoolUsage = Process->QuotaUsage[PsPagedPool]; 1043 SpiCurrent->QuotaPeakNonPagedPoolUsage = Process->QuotaPeak[PsNonPagedPool]; 1044 SpiCurrent->QuotaNonPagedPoolUsage = Process->QuotaUsage[PsNonPagedPool]; 1045 SpiCurrent->PagefileUsage = Process->QuotaUsage[PsPageFile]; 1046 SpiCurrent->PeakPagefileUsage = Process->QuotaPeak[PsPageFile]; 1047 SpiCurrent->PrivatePageCount = Process->CommitCharge; 1048 ThreadInfo = (PSYSTEM_THREAD_INFORMATION)(SpiCurrent + 1); 1049 1050 CurrentEntry = Process->Pcb.ThreadListHead.Flink; 1051 while (CurrentEntry != &Process->Pcb.ThreadListHead) 1052 { 1053 CurrentThread = CONTAINING_RECORD(CurrentEntry, ETHREAD, Tcb.ThreadListEntry); 1054 1055 ThreadInfo->KernelTime.QuadPart = UInt32x32To64(CurrentThread->Tcb.KernelTime, KeMaximumIncrement); 1056 ThreadInfo->UserTime.QuadPart = UInt32x32To64(CurrentThread->Tcb.UserTime, KeMaximumIncrement); 1057 ThreadInfo->CreateTime.QuadPart = CurrentThread->CreateTime.QuadPart; 1058 ThreadInfo->WaitTime = CurrentThread->Tcb.WaitTime; 1059 ThreadInfo->StartAddress = (PVOID) CurrentThread->StartAddress; 1060 ThreadInfo->ClientId = CurrentThread->Cid; 1061 ThreadInfo->Priority = CurrentThread->Tcb.Priority; 1062 ThreadInfo->BasePriority = CurrentThread->Tcb.BasePriority; 1063 ThreadInfo->ContextSwitches = CurrentThread->Tcb.ContextSwitches; 1064 ThreadInfo->ThreadState = CurrentThread->Tcb.State; 1065 ThreadInfo->WaitReason = CurrentThread->Tcb.WaitReason; 1066 1067 ThreadInfo++; 1068 CurrentEntry = CurrentEntry->Flink; 1069 } 1070 1071 /* Query total user/kernel times of a process */ 1072 TotalKernel = KeQueryRuntimeProcess(&Process->Pcb, &TotalUser); 1073 SpiCurrent->UserTime.QuadPart = UInt32x32To64(TotalUser, KeMaximumIncrement); 1074 SpiCurrent->KernelTime.QuadPart = UInt32x32To64(TotalKernel, KeMaximumIncrement); 1075 } 1076 1077 if (ProcessImageName) 1078 { 1079 /* Release the memory allocated by SeLocateProcessImageName */ 1080 ExFreePoolWithTag(ProcessImageName, TAG_SEPA); 1081 ProcessImageName = NULL; 1082 } 1083 1084 /* Unlock the Process */ 1085 ExReleasePushLockShared(&Process->ProcessLock); 1086 KeLeaveCriticalRegion(); 1087 1088 /* Handle idle process entry */ 1089 Skip: 1090 if (Process == PsIdleProcess) Process = NULL; 1091 1092 Process = PsGetNextProcess(Process); 1093 ThreadsCount = 0; 1094 if ((Process == SystemProcess) || (Process == NULL)) 1095 { 1096 if (!Overflow) 1097 SpiCurrent->NextEntryOffset = 0; 1098 break; 1099 } 1100 else 1101 Current += CurrentSize + ImageNameMaximumLength; 1102 } while ((Process != SystemProcess) && (Process != NULL)); 1103 1104 if(Process != NULL) 1105 ObDereferenceObject(Process); 1106 Status = STATUS_SUCCESS; 1107 } 1108 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1109 { 1110 if(Process != NULL) 1111 ObDereferenceObject(Process); 1112 if (ProcessImageName) 1113 { 1114 /* Release the memory allocated by SeLocateProcessImageName */ 1115 ExFreePoolWithTag(ProcessImageName, TAG_SEPA); 1116 } 1117 1118 Status = _SEH2_GetExceptionCode(); 1119 } 1120 _SEH2_END 1121 1122 if (Overflow) 1123 Status = STATUS_INFO_LENGTH_MISMATCH; 1124 1125 *ReqSize = TotalSize; 1126 return Status; 1127 } 1128 1129 /* Class 6 - Call Count Information */ 1130 QSI_DEF(SystemCallCountInformation) 1131 { 1132 /* FIXME */ 1133 DPRINT1("NtQuerySystemInformation - SystemCallCountInformation not implemented\n"); 1134 return STATUS_NOT_IMPLEMENTED; 1135 } 1136 1137 /* Class 7 - Device Information */ 1138 QSI_DEF(SystemDeviceInformation) 1139 { 1140 PSYSTEM_DEVICE_INFORMATION Sdi 1141 = (PSYSTEM_DEVICE_INFORMATION) Buffer; 1142 PCONFIGURATION_INFORMATION ConfigInfo; 1143 1144 *ReqSize = sizeof(SYSTEM_DEVICE_INFORMATION); 1145 1146 /* Check user buffer's size */ 1147 if (Size < sizeof(SYSTEM_DEVICE_INFORMATION)) 1148 { 1149 return STATUS_INFO_LENGTH_MISMATCH; 1150 } 1151 1152 ConfigInfo = IoGetConfigurationInformation(); 1153 1154 Sdi->NumberOfDisks = ConfigInfo->DiskCount; 1155 Sdi->NumberOfFloppies = ConfigInfo->FloppyCount; 1156 Sdi->NumberOfCdRoms = ConfigInfo->CdRomCount; 1157 Sdi->NumberOfTapes = ConfigInfo->TapeCount; 1158 Sdi->NumberOfSerialPorts = ConfigInfo->SerialCount; 1159 Sdi->NumberOfParallelPorts = ConfigInfo->ParallelCount; 1160 1161 return STATUS_SUCCESS; 1162 } 1163 1164 /* Class 8 - Processor Performance Information */ 1165 QSI_DEF(SystemProcessorPerformanceInformation) 1166 { 1167 PSYSTEM_PROCESSOR_PERFORMANCE_INFORMATION Spi 1168 = (PSYSTEM_PROCESSOR_PERFORMANCE_INFORMATION) Buffer; 1169 1170 LONG i; 1171 ULONG TotalTime; 1172 PKPRCB Prcb; 1173 1174 *ReqSize = KeNumberProcessors * sizeof(SYSTEM_PROCESSOR_PERFORMANCE_INFORMATION); 1175 1176 /* Check user buffer's size */ 1177 if (Size < *ReqSize) 1178 { 1179 return STATUS_INFO_LENGTH_MISMATCH; 1180 } 1181 1182 for (i = 0; i < KeNumberProcessors; i++) 1183 { 1184 /* Get the PRCB on this processor */ 1185 Prcb = KiProcessorBlock[i]; 1186 1187 /* Calculate total user and kernel times */ 1188 TotalTime = Prcb->IdleThread->KernelTime + Prcb->IdleThread->UserTime; 1189 Spi->IdleTime.QuadPart = UInt32x32To64(TotalTime, KeMaximumIncrement); 1190 Spi->KernelTime.QuadPart = UInt32x32To64(Prcb->KernelTime, KeMaximumIncrement); 1191 Spi->UserTime.QuadPart = UInt32x32To64(Prcb->UserTime, KeMaximumIncrement); 1192 Spi->DpcTime.QuadPart = UInt32x32To64(Prcb->DpcTime, KeMaximumIncrement); 1193 Spi->InterruptTime.QuadPart = UInt32x32To64(Prcb->InterruptTime, KeMaximumIncrement); 1194 Spi->InterruptCount = Prcb->InterruptCount; 1195 Spi++; 1196 } 1197 1198 return STATUS_SUCCESS; 1199 } 1200 1201 /* Class 9 - Flags Information */ 1202 QSI_DEF(SystemFlagsInformation) 1203 { 1204 #if (NTDDI_VERSION >= NTDDI_VISTA) 1205 *ReqSize = sizeof(SYSTEM_FLAGS_INFORMATION); 1206 #endif 1207 1208 if (sizeof(SYSTEM_FLAGS_INFORMATION) != Size) 1209 { 1210 return STATUS_INFO_LENGTH_MISMATCH; 1211 } 1212 1213 ((PSYSTEM_FLAGS_INFORMATION) Buffer)->Flags = NtGlobalFlag; 1214 #if (NTDDI_VERSION < NTDDI_VISTA) 1215 *ReqSize = sizeof(SYSTEM_FLAGS_INFORMATION); 1216 #endif 1217 1218 return STATUS_SUCCESS; 1219 } 1220 1221 SSI_DEF(SystemFlagsInformation) 1222 { 1223 if (sizeof(SYSTEM_FLAGS_INFORMATION) != Size) 1224 { 1225 return STATUS_INFO_LENGTH_MISMATCH; 1226 } 1227 1228 if (!SeSinglePrivilegeCheck(SeDebugPrivilege, ExGetPreviousMode())) 1229 { 1230 #if (NTDDI_VERSION < NTDDI_WIN7) 1231 return STATUS_ACCESS_VIOLATION; 1232 #else 1233 return STATUS_ACCESS_DENIED; 1234 #endif 1235 } 1236 1237 NtGlobalFlag = ((PSYSTEM_FLAGS_INFORMATION) Buffer)->Flags; 1238 return STATUS_SUCCESS; 1239 } 1240 1241 /* Class 10 - Call Time Information */ 1242 QSI_DEF(SystemCallTimeInformation) 1243 { 1244 /* FIXME */ 1245 DPRINT1("NtQuerySystemInformation - SystemCallTimeInformation not implemented\n"); 1246 return STATUS_NOT_IMPLEMENTED; 1247 } 1248 1249 /* Class 11 - Module Information */ 1250 QSI_DEF(SystemModuleInformation) 1251 { 1252 NTSTATUS Status; 1253 1254 /* Acquire system module list lock */ 1255 KeEnterCriticalRegion(); 1256 ExAcquireResourceExclusiveLite(&PsLoadedModuleResource, TRUE); 1257 1258 /* Call the generic handler with the system module list */ 1259 Status = ExpQueryModuleInformation(&PsLoadedModuleList, 1260 &MmLoadedUserImageList, 1261 (PRTL_PROCESS_MODULES)Buffer, 1262 Size, 1263 ReqSize); 1264 1265 /* Release list lock and return status */ 1266 ExReleaseResourceLite(&PsLoadedModuleResource); 1267 KeLeaveCriticalRegion(); 1268 return Status; 1269 } 1270 1271 /* Class 12 - Locks Information */ 1272 QSI_DEF(SystemLocksInformation) 1273 { 1274 /* FIXME */ 1275 DPRINT1("NtQuerySystemInformation - SystemLocksInformation not implemented\n"); 1276 return STATUS_NOT_IMPLEMENTED; 1277 } 1278 1279 /* Class 13 - Stack Trace Information */ 1280 QSI_DEF(SystemStackTraceInformation) 1281 { 1282 /* FIXME */ 1283 DPRINT1("NtQuerySystemInformation - SystemStackTraceInformation not implemented\n"); 1284 return STATUS_NOT_IMPLEMENTED; 1285 } 1286 1287 /* Class 14 - Paged Pool Information */ 1288 QSI_DEF(SystemPagedPoolInformation) 1289 { 1290 /* FIXME */ 1291 DPRINT1("NtQuerySystemInformation - SystemPagedPoolInformation not implemented\n"); 1292 return STATUS_NOT_IMPLEMENTED; 1293 } 1294 1295 /* Class 15 - Non Paged Pool Information */ 1296 QSI_DEF(SystemNonPagedPoolInformation) 1297 { 1298 /* FIXME */ 1299 DPRINT1("NtQuerySystemInformation - SystemNonPagedPoolInformation not implemented\n"); 1300 return STATUS_NOT_IMPLEMENTED; 1301 } 1302 1303 /* Class 16 - Handle Information */ 1304 QSI_DEF(SystemHandleInformation) 1305 { 1306 PSYSTEM_HANDLE_INFORMATION HandleInformation; 1307 PLIST_ENTRY NextTableEntry; 1308 PHANDLE_TABLE HandleTable; 1309 PHANDLE_TABLE_ENTRY HandleTableEntry; 1310 EXHANDLE Handle; 1311 ULONG Index = 0; 1312 NTSTATUS Status; 1313 PMDL Mdl; 1314 PAGED_CODE(); 1315 1316 DPRINT("NtQuerySystemInformation - SystemHandleInformation\n"); 1317 1318 /* Set initial required buffer size */ 1319 *ReqSize = FIELD_OFFSET(SYSTEM_HANDLE_INFORMATION, Handles); 1320 1321 /* Check user's buffer size */ 1322 if (Size < *ReqSize) 1323 { 1324 return STATUS_INFO_LENGTH_MISMATCH; 1325 } 1326 1327 /* We need to lock down the memory */ 1328 Status = ExLockUserBuffer(Buffer, 1329 Size, 1330 ExGetPreviousMode(), 1331 IoWriteAccess, 1332 (PVOID*)&HandleInformation, 1333 &Mdl); 1334 if (!NT_SUCCESS(Status)) 1335 { 1336 DPRINT1("Failed to lock the user buffer: 0x%lx\n", Status); 1337 return Status; 1338 } 1339 1340 /* Reset of count of handles */ 1341 HandleInformation->NumberOfHandles = 0; 1342 1343 /* Enter a critical region */ 1344 KeEnterCriticalRegion(); 1345 1346 /* Acquire the handle table lock */ 1347 ExAcquirePushLockShared(&HandleTableListLock); 1348 1349 /* Enumerate all system handles */ 1350 for (NextTableEntry = HandleTableListHead.Flink; 1351 NextTableEntry != &HandleTableListHead; 1352 NextTableEntry = NextTableEntry->Flink) 1353 { 1354 /* Get current handle table */ 1355 HandleTable = CONTAINING_RECORD(NextTableEntry, HANDLE_TABLE, HandleTableList); 1356 1357 /* Set the initial value and loop the entries */ 1358 Handle.Value = 0; 1359 while ((HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, Handle))) 1360 { 1361 /* Validate the entry */ 1362 if ((HandleTableEntry->Object) && 1363 (HandleTableEntry->NextFreeTableEntry != -2)) 1364 { 1365 /* Increase of count of handles */ 1366 ++HandleInformation->NumberOfHandles; 1367 1368 /* Lock the entry */ 1369 if (ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 1370 { 1371 /* Increase required buffer size */ 1372 *ReqSize += sizeof(SYSTEM_HANDLE_TABLE_ENTRY_INFO); 1373 1374 /* Check user's buffer size */ 1375 if (*ReqSize > Size) 1376 { 1377 Status = STATUS_INFO_LENGTH_MISMATCH; 1378 } 1379 else 1380 { 1381 POBJECT_HEADER ObjectHeader = ObpGetHandleObject(HandleTableEntry); 1382 1383 /* Filling handle information */ 1384 HandleInformation->Handles[Index].UniqueProcessId = 1385 (USHORT)(ULONG_PTR) HandleTable->UniqueProcessId; 1386 1387 HandleInformation->Handles[Index].CreatorBackTraceIndex = 0; 1388 1389 #if 0 /* FIXME!!! Type field currupted */ 1390 HandleInformation->Handles[Index].ObjectTypeIndex = 1391 (UCHAR) ObjectHeader->Type->Index; 1392 #else 1393 HandleInformation->Handles[Index].ObjectTypeIndex = 0; 1394 #endif 1395 1396 HandleInformation->Handles[Index].HandleAttributes = 1397 HandleTableEntry->ObAttributes & OBJ_HANDLE_ATTRIBUTES; 1398 1399 HandleInformation->Handles[Index].HandleValue = 1400 (USHORT)(ULONG_PTR) Handle.GenericHandleOverlay; 1401 1402 HandleInformation->Handles[Index].Object = &ObjectHeader->Body; 1403 1404 HandleInformation->Handles[Index].GrantedAccess = 1405 HandleTableEntry->GrantedAccess; 1406 1407 ++Index; 1408 } 1409 1410 /* Unlock it */ 1411 ExUnlockHandleTableEntry(HandleTable, HandleTableEntry); 1412 } 1413 } 1414 1415 /* Go to the next entry */ 1416 Handle.Value += sizeof(HANDLE); 1417 } 1418 } 1419 1420 /* Release the lock */ 1421 ExReleasePushLockShared(&HandleTableListLock); 1422 1423 /* Leave the critical region */ 1424 KeLeaveCriticalRegion(); 1425 1426 /* Release the locked user buffer */ 1427 ExUnlockUserBuffer(Mdl); 1428 1429 return Status; 1430 } 1431 1432 /* Class 17 - Information */ 1433 QSI_DEF(SystemObjectInformation) 1434 { 1435 /* FIXME */ 1436 DPRINT1("NtQuerySystemInformation - SystemObjectInformation not implemented\n"); 1437 return STATUS_NOT_IMPLEMENTED; 1438 } 1439 1440 /* Class 18 - Information */ 1441 QSI_DEF(SystemPageFileInformation) 1442 { 1443 UNICODE_STRING FileName; /* FIXME */ 1444 SYSTEM_PAGEFILE_INFORMATION *Spfi = (SYSTEM_PAGEFILE_INFORMATION *) Buffer; 1445 1446 if (Size < sizeof(SYSTEM_PAGEFILE_INFORMATION)) 1447 { 1448 * ReqSize = sizeof(SYSTEM_PAGEFILE_INFORMATION); 1449 return STATUS_INFO_LENGTH_MISMATCH; 1450 } 1451 1452 RtlInitUnicodeString(&FileName, NULL); /* FIXME */ 1453 1454 /* FIXME */ 1455 Spfi->NextEntryOffset = 0; 1456 1457 Spfi->TotalSize = MiFreeSwapPages + MiUsedSwapPages; 1458 Spfi->TotalInUse = MiUsedSwapPages; 1459 Spfi->PeakUsage = MiUsedSwapPages; /* FIXME */ 1460 Spfi->PageFileName = FileName; 1461 return STATUS_SUCCESS; 1462 } 1463 1464 /* Class 19 - Vdm Instemul Information */ 1465 QSI_DEF(SystemVdmInstemulInformation) 1466 { 1467 /* FIXME */ 1468 DPRINT1("NtQuerySystemInformation - SystemVdmInstemulInformation not implemented\n"); 1469 return STATUS_NOT_IMPLEMENTED; 1470 } 1471 1472 /* Class 20 - Vdm Bop Information */ 1473 QSI_DEF(SystemVdmBopInformation) 1474 { 1475 /* FIXME */ 1476 DPRINT1("NtQuerySystemInformation - SystemVdmBopInformation not implemented\n"); 1477 return STATUS_NOT_IMPLEMENTED; 1478 } 1479 1480 /* Class 21 - File Cache Information */ 1481 QSI_DEF(SystemFileCacheInformation) 1482 { 1483 SYSTEM_FILECACHE_INFORMATION *Sci = (SYSTEM_FILECACHE_INFORMATION *) Buffer; 1484 1485 *ReqSize = sizeof(SYSTEM_FILECACHE_INFORMATION); 1486 1487 if (Size < *ReqSize) 1488 { 1489 return STATUS_INFO_LENGTH_MISMATCH; 1490 } 1491 1492 RtlZeroMemory(Sci, sizeof(SYSTEM_FILECACHE_INFORMATION)); 1493 1494 /* Return the Byte size not the page size. */ 1495 Sci->CurrentSize = MiMemoryConsumers[MC_USER].PagesUsed; /* FIXME */ 1496 Sci->PeakSize = MiMemoryConsumers[MC_USER].PagesUsed; /* FIXME */ 1497 /* Taskmgr multiplies this one by page size right away */ 1498 Sci->CurrentSizeIncludingTransitionInPages = MiMemoryConsumers[MC_USER].PagesUsed; /* FIXME: Should be */ 1499 /* system working set and standby pages. */ 1500 Sci->PageFaultCount = 0; /* FIXME */ 1501 Sci->MinimumWorkingSet = 0; /* FIXME */ 1502 Sci->MaximumWorkingSet = 0; /* FIXME */ 1503 1504 return STATUS_SUCCESS; 1505 } 1506 1507 SSI_DEF(SystemFileCacheInformation) 1508 { 1509 if (Size < sizeof(SYSTEM_FILECACHE_INFORMATION)) 1510 { 1511 return STATUS_INFO_LENGTH_MISMATCH; 1512 } 1513 /* FIXME */ 1514 DPRINT1("NtSetSystemInformation - SystemFileCacheInformation not implemented\n"); 1515 return STATUS_NOT_IMPLEMENTED; 1516 } 1517 1518 /* Class 22 - Pool Tag Information */ 1519 QSI_DEF(SystemPoolTagInformation) 1520 { 1521 if (Size < sizeof(SYSTEM_POOLTAG_INFORMATION)) return STATUS_INFO_LENGTH_MISMATCH; 1522 return ExGetPoolTagInfo(Buffer, Size, ReqSize); 1523 } 1524 1525 /* Class 23 - Interrupt Information for all processors */ 1526 QSI_DEF(SystemInterruptInformation) 1527 { 1528 PKPRCB Prcb; 1529 LONG i; 1530 ULONG ti; 1531 PSYSTEM_INTERRUPT_INFORMATION sii = (PSYSTEM_INTERRUPT_INFORMATION)Buffer; 1532 1533 if(Size < KeNumberProcessors * sizeof(SYSTEM_INTERRUPT_INFORMATION)) 1534 { 1535 return STATUS_INFO_LENGTH_MISMATCH; 1536 } 1537 1538 ti = KeQueryTimeIncrement(); 1539 1540 for (i = 0; i < KeNumberProcessors; i++) 1541 { 1542 Prcb = KiProcessorBlock[i]; 1543 sii->ContextSwitches = KeGetContextSwitches(Prcb); 1544 sii->DpcCount = Prcb->DpcData[0].DpcCount; 1545 sii->DpcRate = Prcb->DpcRequestRate; 1546 sii->TimeIncrement = ti; 1547 sii->DpcBypassCount = 0; 1548 sii->ApcBypassCount = 0; 1549 sii++; 1550 } 1551 1552 return STATUS_SUCCESS; 1553 } 1554 1555 /* Class 24 - DPC Behaviour Information */ 1556 QSI_DEF(SystemDpcBehaviourInformation) 1557 { 1558 PSYSTEM_DPC_BEHAVIOR_INFORMATION sdbi = (PSYSTEM_DPC_BEHAVIOR_INFORMATION)Buffer; 1559 1560 if (Size < sizeof(SYSTEM_DPC_BEHAVIOR_INFORMATION)) 1561 { 1562 return STATUS_INFO_LENGTH_MISMATCH; 1563 } 1564 1565 sdbi->DpcQueueDepth = KiMaximumDpcQueueDepth; 1566 sdbi->MinimumDpcRate = KiMinimumDpcRate; 1567 sdbi->AdjustDpcThreshold = KiAdjustDpcThreshold; 1568 sdbi->IdealDpcRate = KiIdealDpcRate; 1569 1570 return STATUS_SUCCESS; 1571 } 1572 1573 SSI_DEF(SystemDpcBehaviourInformation) 1574 { 1575 /* FIXME */ 1576 DPRINT1("NtSetSystemInformation - SystemDpcBehaviourInformation not implemented\n"); 1577 return STATUS_NOT_IMPLEMENTED; 1578 } 1579 1580 /* Class 25 - Full Memory Information */ 1581 QSI_DEF(SystemFullMemoryInformation) 1582 { 1583 PULONG Spi = (PULONG) Buffer; 1584 1585 PEPROCESS TheIdleProcess; 1586 1587 *ReqSize = sizeof(ULONG); 1588 1589 if (sizeof(ULONG) != Size) 1590 { 1591 return STATUS_INFO_LENGTH_MISMATCH; 1592 } 1593 1594 DPRINT("SystemFullMemoryInformation\n"); 1595 1596 TheIdleProcess = PsIdleProcess; 1597 1598 DPRINT("PID: %p, KernelTime: %u PFFree: %lu PFUsed: %lu\n", 1599 TheIdleProcess->UniqueProcessId, 1600 TheIdleProcess->Pcb.KernelTime, 1601 MiFreeSwapPages, 1602 MiUsedSwapPages); 1603 1604 *Spi = MiMemoryConsumers[MC_USER].PagesUsed; 1605 1606 return STATUS_SUCCESS; 1607 } 1608 1609 /* Class 26 - Load Image */ 1610 SSI_DEF(SystemLoadGdiDriverInformation) 1611 { 1612 PSYSTEM_GDI_DRIVER_INFORMATION DriverInfo = (PVOID)Buffer; 1613 UNICODE_STRING ImageName; 1614 PVOID ImageBase; 1615 PVOID SectionPointer; 1616 ULONG_PTR EntryPoint; 1617 NTSTATUS Status; 1618 ULONG DirSize; 1619 PIMAGE_NT_HEADERS NtHeader; 1620 1621 /* Validate size */ 1622 if (Size != sizeof(SYSTEM_GDI_DRIVER_INFORMATION)) 1623 { 1624 /* Incorrect buffer length, fail */ 1625 return STATUS_INFO_LENGTH_MISMATCH; 1626 } 1627 1628 /* Only kernel mode can call this function */ 1629 if (ExGetPreviousMode() != KernelMode) return STATUS_PRIVILEGE_NOT_HELD; 1630 1631 /* Load the driver */ 1632 ImageName = DriverInfo->DriverName; 1633 Status = MmLoadSystemImage(&ImageName, 1634 NULL, 1635 NULL, 1636 0, 1637 &SectionPointer, 1638 &ImageBase); 1639 if (!NT_SUCCESS(Status)) return Status; 1640 1641 /* Return the export pointer */ 1642 DriverInfo->ExportSectionPointer = 1643 RtlImageDirectoryEntryToData(ImageBase, 1644 TRUE, 1645 IMAGE_DIRECTORY_ENTRY_EXPORT, 1646 &DirSize); 1647 1648 /* Get the entrypoint */ 1649 NtHeader = RtlImageNtHeader(ImageBase); 1650 EntryPoint = NtHeader->OptionalHeader.AddressOfEntryPoint; 1651 EntryPoint += (ULONG_PTR)ImageBase; 1652 1653 /* Save other data */ 1654 DriverInfo->ImageAddress = ImageBase; 1655 DriverInfo->SectionPointer = SectionPointer; 1656 DriverInfo->EntryPoint = (PVOID)EntryPoint; 1657 DriverInfo->ImageLength = NtHeader->OptionalHeader.SizeOfImage; 1658 1659 /* All is good */ 1660 return STATUS_SUCCESS; 1661 } 1662 1663 /* Class 27 - Unload Image */ 1664 SSI_DEF(SystemUnloadGdiDriverInformation) 1665 { 1666 PVOID *SectionPointer = Buffer; 1667 1668 /* Validate size */ 1669 if (Size != sizeof(PVOID)) 1670 { 1671 /* Incorrect length, fail */ 1672 return STATUS_INFO_LENGTH_MISMATCH; 1673 } 1674 1675 /* Only kernel mode can call this function */ 1676 if (ExGetPreviousMode() != KernelMode) return STATUS_PRIVILEGE_NOT_HELD; 1677 1678 /* Unload the image */ 1679 MmUnloadSystemImage(*SectionPointer); 1680 return STATUS_SUCCESS; 1681 } 1682 1683 /* Class 28 - Time Adjustment Information */ 1684 QSI_DEF(SystemTimeAdjustmentInformation) 1685 { 1686 PSYSTEM_QUERY_TIME_ADJUST_INFORMATION TimeInfo = 1687 (PSYSTEM_QUERY_TIME_ADJUST_INFORMATION)Buffer; 1688 1689 /* Check if enough storage was provided */ 1690 if (sizeof(SYSTEM_QUERY_TIME_ADJUST_INFORMATION) > Size) 1691 { 1692 * ReqSize = sizeof(SYSTEM_QUERY_TIME_ADJUST_INFORMATION); 1693 return STATUS_INFO_LENGTH_MISMATCH; 1694 } 1695 1696 /* Give time values to our caller */ 1697 TimeInfo->TimeIncrement = KeMaximumIncrement; 1698 TimeInfo->TimeAdjustment = KeTimeAdjustment; 1699 TimeInfo->Enable = !KiTimeAdjustmentEnabled; 1700 1701 return STATUS_SUCCESS; 1702 } 1703 1704 SSI_DEF(SystemTimeAdjustmentInformation) 1705 { 1706 KPROCESSOR_MODE PreviousMode = KeGetPreviousMode(); 1707 PSYSTEM_SET_TIME_ADJUST_INFORMATION TimeInfo = 1708 (PSYSTEM_SET_TIME_ADJUST_INFORMATION)Buffer; 1709 1710 /* Check size of a buffer, it must match our expectations */ 1711 if (sizeof(SYSTEM_SET_TIME_ADJUST_INFORMATION) != Size) 1712 return STATUS_INFO_LENGTH_MISMATCH; 1713 1714 /* Check who is calling */ 1715 if (PreviousMode != KernelMode) 1716 { 1717 /* Check access rights */ 1718 if (!SeSinglePrivilegeCheck(SeSystemtimePrivilege, PreviousMode)) 1719 { 1720 return STATUS_PRIVILEGE_NOT_HELD; 1721 } 1722 } 1723 1724 /* FIXME: behaviour suggests the member be named 'Disable' */ 1725 if (TimeInfo->Enable) 1726 { 1727 /* Disable time adjustment and set default value */ 1728 KiTimeAdjustmentEnabled = FALSE; 1729 KeTimeAdjustment = KeMaximumIncrement; 1730 } 1731 else 1732 { 1733 /* Check if a valid time adjustment value is given */ 1734 if (TimeInfo->TimeAdjustment == 0) return STATUS_INVALID_PARAMETER_2; 1735 1736 /* Enable time adjustment and set the adjustment value */ 1737 KiTimeAdjustmentEnabled = TRUE; 1738 KeTimeAdjustment = TimeInfo->TimeAdjustment; 1739 } 1740 1741 return STATUS_SUCCESS; 1742 } 1743 1744 /* Class 29 - Summary Memory Information */ 1745 QSI_DEF(SystemSummaryMemoryInformation) 1746 { 1747 /* FIXME */ 1748 DPRINT1("NtQuerySystemInformation - SystemSummaryMemoryInformation not implemented\n"); 1749 return STATUS_NOT_IMPLEMENTED; 1750 } 1751 1752 /* Class 30 - Next Event Id Information */ 1753 QSI_DEF(SystemNextEventIdInformation) 1754 { 1755 /* FIXME */ 1756 DPRINT1("NtQuerySystemInformation - SystemNextEventIdInformation not implemented\n"); 1757 return STATUS_NOT_IMPLEMENTED; 1758 } 1759 1760 /* Class 31 */ 1761 QSI_DEF(SystemPerformanceTraceInformation) 1762 { 1763 /* FIXME */ 1764 DPRINT1("NtQuerySystemInformation - SystemPerformanceTraceInformation not implemented\n"); 1765 return STATUS_NOT_IMPLEMENTED; 1766 } 1767 1768 /* Class 32 - Crash Dump Information */ 1769 QSI_DEF(SystemCrashDumpInformation) 1770 { 1771 /* FIXME */ 1772 DPRINT1("NtQuerySystemInformation - SystemCrashDumpInformation not implemented\n"); 1773 return STATUS_NOT_IMPLEMENTED; 1774 } 1775 1776 /* Class 33 - Exception Information */ 1777 QSI_DEF(SystemExceptionInformation) 1778 { 1779 PSYSTEM_EXCEPTION_INFORMATION ExceptionInformation = 1780 (PSYSTEM_EXCEPTION_INFORMATION)Buffer; 1781 PKPRCB Prcb; 1782 ULONG AlignmentFixupCount = 0, ExceptionDispatchCount = 0; 1783 ULONG FloatingEmulationCount = 0, ByteWordEmulationCount = 0; 1784 CHAR i; 1785 1786 /* Check size of a buffer, it must match our expectations */ 1787 if (sizeof(SYSTEM_EXCEPTION_INFORMATION) != Size) 1788 return STATUS_INFO_LENGTH_MISMATCH; 1789 1790 /* Sum up exception count information from all processors */ 1791 for (i = 0; i < KeNumberProcessors; i++) 1792 { 1793 Prcb = KiProcessorBlock[i]; 1794 if (Prcb) 1795 { 1796 AlignmentFixupCount += Prcb->KeAlignmentFixupCount; 1797 ExceptionDispatchCount += Prcb->KeExceptionDispatchCount; 1798 #ifndef _M_ARM 1799 FloatingEmulationCount += Prcb->KeFloatingEmulationCount; 1800 #endif // _M_ARM 1801 } 1802 } 1803 1804 /* Save information in user's buffer */ 1805 ExceptionInformation->AlignmentFixupCount = AlignmentFixupCount; 1806 ExceptionInformation->ExceptionDispatchCount = ExceptionDispatchCount; 1807 ExceptionInformation->FloatingEmulationCount = FloatingEmulationCount; 1808 ExceptionInformation->ByteWordEmulationCount = ByteWordEmulationCount; 1809 1810 return STATUS_SUCCESS; 1811 } 1812 1813 /* Class 34 - Crash Dump State Information */ 1814 QSI_DEF(SystemCrashDumpStateInformation) 1815 { 1816 /* FIXME */ 1817 DPRINT1("NtQuerySystemInformation - SystemCrashDumpStateInformation not implemented\n"); 1818 return STATUS_NOT_IMPLEMENTED; 1819 } 1820 1821 /* Class 35 - Kernel Debugger Information */ 1822 QSI_DEF(SystemKernelDebuggerInformation) 1823 { 1824 PSYSTEM_KERNEL_DEBUGGER_INFORMATION skdi = (PSYSTEM_KERNEL_DEBUGGER_INFORMATION) Buffer; 1825 1826 #if (NTDDI_VERSION >= NTDDI_VISTA) 1827 *ReqSize = sizeof(SYSTEM_KERNEL_DEBUGGER_INFORMATION); 1828 #endif 1829 1830 if (Size < sizeof(SYSTEM_KERNEL_DEBUGGER_INFORMATION)) 1831 { 1832 return STATUS_INFO_LENGTH_MISMATCH; 1833 } 1834 1835 skdi->KernelDebuggerEnabled = KD_DEBUGGER_ENABLED; 1836 skdi->KernelDebuggerNotPresent = KD_DEBUGGER_NOT_PRESENT; 1837 1838 #if (NTDDI_VERSION < NTDDI_VISTA) 1839 *ReqSize = sizeof(SYSTEM_KERNEL_DEBUGGER_INFORMATION); 1840 #endif 1841 1842 return STATUS_SUCCESS; 1843 } 1844 1845 /* Class 36 - Context Switch Information */ 1846 QSI_DEF(SystemContextSwitchInformation) 1847 { 1848 PSYSTEM_CONTEXT_SWITCH_INFORMATION ContextSwitchInformation = 1849 (PSYSTEM_CONTEXT_SWITCH_INFORMATION)Buffer; 1850 ULONG ContextSwitches; 1851 PKPRCB Prcb; 1852 CHAR i; 1853 1854 /* Check size of a buffer, it must match our expectations */ 1855 if (sizeof(SYSTEM_CONTEXT_SWITCH_INFORMATION) != Size) 1856 return STATUS_INFO_LENGTH_MISMATCH; 1857 1858 /* Calculate total value of context switches across all processors */ 1859 ContextSwitches = 0; 1860 for (i = 0; i < KeNumberProcessors; i ++) 1861 { 1862 Prcb = KiProcessorBlock[i]; 1863 if (Prcb) 1864 { 1865 ContextSwitches += KeGetContextSwitches(Prcb); 1866 } 1867 } 1868 1869 ContextSwitchInformation->ContextSwitches = ContextSwitches; 1870 1871 /* FIXME */ 1872 ContextSwitchInformation->FindAny = 0; 1873 ContextSwitchInformation->FindLast = 0; 1874 ContextSwitchInformation->FindIdeal = 0; 1875 ContextSwitchInformation->IdleAny = 0; 1876 ContextSwitchInformation->IdleCurrent = 0; 1877 ContextSwitchInformation->IdleLast = 0; 1878 ContextSwitchInformation->IdleIdeal = 0; 1879 ContextSwitchInformation->PreemptAny = 0; 1880 ContextSwitchInformation->PreemptCurrent = 0; 1881 ContextSwitchInformation->PreemptLast = 0; 1882 ContextSwitchInformation->SwitchToIdle = 0; 1883 1884 return STATUS_SUCCESS; 1885 } 1886 1887 /* Class 37 - Registry Quota Information */ 1888 QSI_DEF(SystemRegistryQuotaInformation) 1889 { 1890 PSYSTEM_REGISTRY_QUOTA_INFORMATION srqi = (PSYSTEM_REGISTRY_QUOTA_INFORMATION) Buffer; 1891 1892 *ReqSize = sizeof(SYSTEM_REGISTRY_QUOTA_INFORMATION); 1893 if (Size < sizeof(SYSTEM_REGISTRY_QUOTA_INFORMATION)) 1894 { 1895 return STATUS_INFO_LENGTH_MISMATCH; 1896 } 1897 1898 DPRINT1("Faking max registry size of 32 MB\n"); 1899 srqi->RegistryQuotaAllowed = 0x2000000; 1900 srqi->RegistryQuotaUsed = 0x200000; 1901 srqi->PagedPoolSize = 0x200000; 1902 1903 return STATUS_SUCCESS; 1904 } 1905 1906 SSI_DEF(SystemRegistryQuotaInformation) 1907 { 1908 /* FIXME */ 1909 DPRINT1("NtSetSystemInformation - SystemRegistryQuotaInformation not implemented\n"); 1910 return STATUS_NOT_IMPLEMENTED; 1911 } 1912 1913 /* Class 38 - Load And Call Image */ 1914 SSI_DEF(SystemExtendServiceTableInformation) 1915 { 1916 UNICODE_STRING ImageName; 1917 KPROCESSOR_MODE PreviousMode = KeGetPreviousMode(); 1918 PLDR_DATA_TABLE_ENTRY ModuleObject; 1919 NTSTATUS Status; 1920 PIMAGE_NT_HEADERS NtHeader; 1921 DRIVER_OBJECT Win32k; 1922 PDRIVER_INITIALIZE DriverInit; 1923 PVOID ImageBase; 1924 ULONG_PTR EntryPoint; 1925 1926 /* Validate the size */ 1927 if (Size != sizeof(UNICODE_STRING)) return STATUS_INFO_LENGTH_MISMATCH; 1928 1929 /* Check who is calling */ 1930 if (PreviousMode != KernelMode) 1931 { 1932 static const UNICODE_STRING Win32kName = 1933 RTL_CONSTANT_STRING(L"\\SystemRoot\\System32\\win32k.sys"); 1934 1935 /* Make sure we can load drivers */ 1936 if (!SeSinglePrivilegeCheck(SeLoadDriverPrivilege, UserMode)) 1937 { 1938 /* FIXME: We can't, fail */ 1939 return STATUS_PRIVILEGE_NOT_HELD; 1940 } 1941 1942 _SEH2_TRY 1943 { 1944 /* Probe and copy the unicode string */ 1945 ProbeForRead(Buffer, sizeof(ImageName), 1); 1946 ImageName = *(PUNICODE_STRING)Buffer; 1947 1948 /* Probe the string buffer */ 1949 ProbeForRead(ImageName.Buffer, ImageName.Length, sizeof(WCHAR)); 1950 1951 /* Check if we have the correct name (nothing else is allowed!) */ 1952 if (!RtlEqualUnicodeString(&ImageName, &Win32kName, FALSE)) 1953 { 1954 _SEH2_YIELD(return STATUS_PRIVILEGE_NOT_HELD); 1955 } 1956 } 1957 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 1958 { 1959 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 1960 } 1961 _SEH2_END; 1962 1963 /* Recursively call the function, so that we are from kernel mode */ 1964 return ZwSetSystemInformation(SystemExtendServiceTableInformation, 1965 (PVOID)&Win32kName, 1966 sizeof(Win32kName)); 1967 } 1968 1969 /* Load the image */ 1970 Status = MmLoadSystemImage((PUNICODE_STRING)Buffer, 1971 NULL, 1972 NULL, 1973 0, 1974 (PVOID)&ModuleObject, 1975 &ImageBase); 1976 1977 if (!NT_SUCCESS(Status)) return Status; 1978 1979 /* Get the headers */ 1980 NtHeader = RtlImageNtHeader(ImageBase); 1981 if (!NtHeader) 1982 { 1983 /* Fail */ 1984 MmUnloadSystemImage(ModuleObject); 1985 return STATUS_INVALID_IMAGE_FORMAT; 1986 } 1987 1988 /* Get the entrypoint */ 1989 EntryPoint = NtHeader->OptionalHeader.AddressOfEntryPoint; 1990 EntryPoint += (ULONG_PTR)ImageBase; 1991 DriverInit = (PDRIVER_INITIALIZE)EntryPoint; 1992 1993 /* Create a dummy device */ 1994 RtlZeroMemory(&Win32k, sizeof(Win32k)); 1995 ASSERT(KeGetCurrentIrql() == PASSIVE_LEVEL); 1996 Win32k.DriverStart = ImageBase; 1997 1998 /* Call it */ 1999 Status = (DriverInit)(&Win32k, NULL); 2000 ASSERT(KeGetCurrentIrql() == PASSIVE_LEVEL); 2001 2002 /* Unload if we failed */ 2003 if (!NT_SUCCESS(Status)) MmUnloadSystemImage(ModuleObject); 2004 return Status; 2005 } 2006 2007 /* Class 39 - Priority Separation */ 2008 SSI_DEF(SystemPrioritySeperation) 2009 { 2010 /* Check if the size is correct */ 2011 if (Size != sizeof(ULONG)) 2012 { 2013 return STATUS_INFO_LENGTH_MISMATCH; 2014 } 2015 2016 /* We need the TCB privilege */ 2017 if (!SeSinglePrivilegeCheck(SeTcbPrivilege, ExGetPreviousMode())) 2018 { 2019 return STATUS_PRIVILEGE_NOT_HELD; 2020 } 2021 2022 /* Modify the quantum table */ 2023 PsChangeQuantumTable(TRUE, *(PULONG)Buffer); 2024 2025 return STATUS_SUCCESS; 2026 } 2027 2028 /* Class 40 */ 2029 QSI_DEF(SystemVerifierAddDriverInformation) 2030 { 2031 /* FIXME */ 2032 DPRINT1("NtQuerySystemInformation - SystemVerifierAddDriverInformation not implemented\n"); 2033 return STATUS_NOT_IMPLEMENTED; 2034 } 2035 2036 /* Class 41 */ 2037 QSI_DEF(SystemVerifierRemoveDriverInformation) 2038 { 2039 /* FIXME */ 2040 DPRINT1("NtQuerySystemInformation - SystemVerifierRemoveDriverInformation not implemented\n"); 2041 return STATUS_NOT_IMPLEMENTED; 2042 } 2043 2044 /* Class 42 - Power Information */ 2045 QSI_DEF(SystemProcessorIdleInformation) 2046 { 2047 *ReqSize = sizeof(PROCESSOR_POWER_INFORMATION) * KeNumberProcessors; 2048 2049 if (sizeof(PROCESSOR_POWER_INFORMATION) * KeNumberProcessors > Size) 2050 { 2051 return STATUS_INFO_LENGTH_MISMATCH; 2052 } 2053 2054 /* FIXME */ 2055 DPRINT1("NtQuerySystemInformation - SystemPowerInformation not implemented\n"); 2056 return STATUS_NOT_IMPLEMENTED; 2057 } 2058 2059 /* Class 43 */ 2060 QSI_DEF(SystemLegacyDriverInformation) 2061 { 2062 /* FIXME */ 2063 DPRINT1("NtQuerySystemInformation - SystemLegacyDriverInformation not implemented\n"); 2064 return STATUS_NOT_IMPLEMENTED; 2065 } 2066 2067 /* Class 44 - Current Time Zone Information */ 2068 QSI_DEF(SystemCurrentTimeZoneInformation) 2069 { 2070 *ReqSize = sizeof(RTL_TIME_ZONE_INFORMATION); 2071 2072 if (sizeof(RTL_TIME_ZONE_INFORMATION) != Size) 2073 { 2074 return STATUS_INFO_LENGTH_MISMATCH; 2075 } 2076 2077 /* Copy the time zone information struct */ 2078 memcpy(Buffer, 2079 &ExpTimeZoneInfo, 2080 sizeof(RTL_TIME_ZONE_INFORMATION)); 2081 2082 return STATUS_SUCCESS; 2083 } 2084 2085 SSI_DEF(SystemCurrentTimeZoneInformation) 2086 { 2087 /* Check user buffer's size */ 2088 if (Size < sizeof(RTL_TIME_ZONE_INFORMATION)) 2089 { 2090 return STATUS_INFO_LENGTH_MISMATCH; 2091 } 2092 2093 return ExpSetTimeZoneInformation((PRTL_TIME_ZONE_INFORMATION)Buffer); 2094 } 2095 2096 static 2097 VOID 2098 ExpCopyLookasideInformation( 2099 PSYSTEM_LOOKASIDE_INFORMATION *InfoPointer, 2100 PULONG RemainingPointer, 2101 PLIST_ENTRY ListHead, 2102 BOOLEAN ListUsesMisses) 2103 2104 { 2105 PSYSTEM_LOOKASIDE_INFORMATION Info; 2106 PGENERAL_LOOKASIDE LookasideList; 2107 PLIST_ENTRY ListEntry; 2108 ULONG Remaining; 2109 2110 /* Get info pointer and remaining count of free array element */ 2111 Info = *InfoPointer; 2112 Remaining = *RemainingPointer; 2113 2114 /* Loop as long as we have lookaside lists and free array elements */ 2115 for (ListEntry = ListHead->Flink; 2116 (ListEntry != ListHead) && (Remaining > 0); 2117 ListEntry = ListEntry->Flink, Remaining--) 2118 { 2119 LookasideList = CONTAINING_RECORD(ListEntry, GENERAL_LOOKASIDE, ListEntry); 2120 2121 /* Fill the next array element */ 2122 Info->CurrentDepth = LookasideList->Depth; 2123 Info->MaximumDepth = LookasideList->MaximumDepth; 2124 Info->TotalAllocates = LookasideList->TotalAllocates; 2125 Info->TotalFrees = LookasideList->TotalFrees; 2126 Info->Type = LookasideList->Type; 2127 Info->Tag = LookasideList->Tag; 2128 Info->Size = LookasideList->Size; 2129 2130 /* Check how the lists track misses/hits */ 2131 if (ListUsesMisses) 2132 { 2133 /* Copy misses */ 2134 Info->AllocateMisses = LookasideList->AllocateMisses; 2135 Info->FreeMisses = LookasideList->FreeMisses; 2136 } 2137 else 2138 { 2139 /* Calculate misses */ 2140 Info->AllocateMisses = LookasideList->TotalAllocates 2141 - LookasideList->AllocateHits; 2142 Info->FreeMisses = LookasideList->TotalFrees 2143 - LookasideList->FreeHits; 2144 } 2145 } 2146 2147 /* Return the updated pointer and remaining count */ 2148 *InfoPointer = Info; 2149 *RemainingPointer = Remaining; 2150 } 2151 2152 /* Class 45 - Lookaside Information */ 2153 QSI_DEF(SystemLookasideInformation) 2154 { 2155 KPROCESSOR_MODE PreviousMode; 2156 PSYSTEM_LOOKASIDE_INFORMATION Info; 2157 PMDL Mdl; 2158 ULONG MaxCount, Remaining; 2159 KIRQL OldIrql; 2160 NTSTATUS Status; 2161 2162 /* First we need to lock down the memory, since we are going to access it 2163 at high IRQL */ 2164 PreviousMode = ExGetPreviousMode(); 2165 Status = ExLockUserBuffer(Buffer, 2166 Size, 2167 PreviousMode, 2168 IoWriteAccess, 2169 (PVOID*)&Info, 2170 &Mdl); 2171 if (!NT_SUCCESS(Status)) 2172 { 2173 DPRINT1("Failed to lock the user buffer: 0x%lx\n", Status); 2174 return Status; 2175 } 2176 2177 /* Calculate how many items we can store */ 2178 Remaining = MaxCount = Size / sizeof(SYSTEM_LOOKASIDE_INFORMATION); 2179 if (Remaining == 0) 2180 { 2181 goto Leave; 2182 } 2183 2184 /* Copy info from pool lookaside lists */ 2185 ExpCopyLookasideInformation(&Info, 2186 &Remaining, 2187 &ExPoolLookasideListHead, 2188 FALSE); 2189 if (Remaining == 0) 2190 { 2191 goto Leave; 2192 } 2193 2194 /* Copy info from system lookaside lists */ 2195 ExpCopyLookasideInformation(&Info, 2196 &Remaining, 2197 &ExSystemLookasideListHead, 2198 TRUE); 2199 if (Remaining == 0) 2200 { 2201 goto Leave; 2202 } 2203 2204 /* Acquire spinlock for ExpNonPagedLookasideListHead */ 2205 KeAcquireSpinLock(&ExpNonPagedLookasideListLock, &OldIrql); 2206 2207 /* Copy info from non-paged lookaside lists */ 2208 ExpCopyLookasideInformation(&Info, 2209 &Remaining, 2210 &ExpNonPagedLookasideListHead, 2211 TRUE); 2212 2213 /* Release spinlock for ExpNonPagedLookasideListHead */ 2214 KeReleaseSpinLock(&ExpNonPagedLookasideListLock, OldIrql); 2215 2216 if (Remaining == 0) 2217 { 2218 goto Leave; 2219 } 2220 2221 /* Acquire spinlock for ExpPagedLookasideListHead */ 2222 KeAcquireSpinLock(&ExpPagedLookasideListLock, &OldIrql); 2223 2224 /* Copy info from paged lookaside lists */ 2225 ExpCopyLookasideInformation(&Info, 2226 &Remaining, 2227 &ExpPagedLookasideListHead, 2228 TRUE); 2229 2230 /* Release spinlock for ExpPagedLookasideListHead */ 2231 KeReleaseSpinLock(&ExpPagedLookasideListLock, OldIrql); 2232 2233 Leave: 2234 2235 /* Release the locked user buffer */ 2236 ExUnlockUserBuffer(Mdl); 2237 2238 /* Return the size of the actually written data */ 2239 *ReqSize = (MaxCount - Remaining) * sizeof(SYSTEM_LOOKASIDE_INFORMATION); 2240 return STATUS_SUCCESS; 2241 } 2242 2243 /* Class 46 - Set time slip event */ 2244 SSI_DEF(SystemTimeSlipNotification) 2245 { 2246 /* FIXME */ 2247 DPRINT1("NtSetSystemInformation - SystemTimeSlipNotification not implemented\n"); 2248 return STATUS_NOT_IMPLEMENTED; 2249 } 2250 2251 NTSTATUS 2252 NTAPI 2253 MmSessionCreate(OUT PULONG SessionId); 2254 2255 NTSTATUS 2256 NTAPI 2257 MmSessionDelete(IN ULONG SessionId); 2258 2259 /* Class 47 - Create a new session (TSE) */ 2260 SSI_DEF(SystemSessionCreate) 2261 { 2262 ULONG SessionId; 2263 KPROCESSOR_MODE PreviousMode = KeGetPreviousMode(); 2264 NTSTATUS Status; 2265 2266 if (Size != sizeof(ULONG)) return STATUS_INFO_LENGTH_MISMATCH; 2267 2268 if (PreviousMode != KernelMode) 2269 { 2270 if (!SeSinglePrivilegeCheck(SeLoadDriverPrivilege, PreviousMode)) 2271 { 2272 return STATUS_PRIVILEGE_NOT_HELD; 2273 } 2274 2275 ProbeForWriteUlong(Buffer); 2276 } 2277 2278 Status = MmSessionCreate(&SessionId); 2279 if (NT_SUCCESS(Status)) *(PULONG)Buffer = SessionId; 2280 2281 return Status; 2282 } 2283 2284 /* Class 48 - Delete an existing session (TSE) */ 2285 SSI_DEF(SystemSessionDetach) 2286 { 2287 ULONG SessionId; 2288 KPROCESSOR_MODE PreviousMode = KeGetPreviousMode(); 2289 2290 if (Size != sizeof(ULONG)) return STATUS_INFO_LENGTH_MISMATCH; 2291 2292 if (PreviousMode != KernelMode) 2293 { 2294 if (!SeSinglePrivilegeCheck(SeLoadDriverPrivilege, PreviousMode)) 2295 { 2296 return STATUS_PRIVILEGE_NOT_HELD; 2297 } 2298 } 2299 2300 SessionId = *(PULONG)Buffer; 2301 2302 return MmSessionDelete(SessionId); 2303 } 2304 2305 /* Class 49 - UNKNOWN */ 2306 QSI_DEF(SystemSessionInformation) 2307 { 2308 /* FIXME */ 2309 DPRINT1("NtQuerySystemInformation - SystemSessionInformation not implemented\n"); 2310 return STATUS_NOT_IMPLEMENTED; 2311 } 2312 2313 /* Class 50 - System range start address */ 2314 QSI_DEF(SystemRangeStartInformation) 2315 { 2316 /* Check user buffer's size */ 2317 if (Size != sizeof(ULONG_PTR)) return STATUS_INFO_LENGTH_MISMATCH; 2318 2319 *(PULONG_PTR)Buffer = (ULONG_PTR)MmSystemRangeStart; 2320 2321 if (ReqSize) *ReqSize = sizeof(ULONG_PTR); 2322 2323 return STATUS_SUCCESS; 2324 } 2325 2326 /* Class 51 - Driver verifier information */ 2327 QSI_DEF(SystemVerifierInformation) 2328 { 2329 /* FIXME */ 2330 DPRINT1("NtQuerySystemInformation - SystemVerifierInformation not implemented\n"); 2331 return STATUS_NOT_IMPLEMENTED; 2332 } 2333 2334 SSI_DEF(SystemVerifierInformation) 2335 { 2336 /* FIXME */ 2337 DPRINT1("NtSetSystemInformation - SystemVerifierInformation not implemented\n"); 2338 return STATUS_NOT_IMPLEMENTED; 2339 } 2340 2341 /* Class 52 */ 2342 SSI_DEF(SystemVerifierThunkExtend) 2343 { 2344 /* FIXME */ 2345 DPRINT1("NtSetSystemInformation - SystemVerifierThunkExtend not implemented\n"); 2346 return STATUS_NOT_IMPLEMENTED; 2347 } 2348 2349 /* Class 53 - A session's processes */ 2350 QSI_DEF(SystemSessionProcessesInformation) 2351 { 2352 /* FIXME */ 2353 DPRINT1("NtQuerySystemInformation - SystemSessionProcessInformation not implemented\n"); 2354 return STATUS_NOT_IMPLEMENTED; 2355 } 2356 2357 /* Class 54 - Load & map in system space */ 2358 SSI_DEF(SystemLoadGdiDriverInSystemSpaceInformation) 2359 { 2360 /* FIXME */ 2361 DPRINT1("NtSetSystemInformation - SystemLoadGdiDriverInSystemSpaceInformation not implemented\n"); 2362 return STATUS_NOT_IMPLEMENTED; 2363 } 2364 2365 /* Class 55 - NUMA processor information */ 2366 QSI_DEF(SystemNumaProcessorMap) 2367 { 2368 ULONG MaxEntries, Node; 2369 PSYSTEM_NUMA_INFORMATION NumaInformation = (PSYSTEM_NUMA_INFORMATION)Buffer; 2370 2371 /* Validate input size */ 2372 if (Size < sizeof(ULONG)) 2373 { 2374 return STATUS_INFO_LENGTH_MISMATCH; 2375 } 2376 2377 /* Return highest node */ 2378 NumaInformation->HighestNodeNumber = KeNumberNodes - 1; 2379 2380 /* Compute how much entries we will be able to put in output structure */ 2381 MaxEntries = (Size - FIELD_OFFSET(SYSTEM_NUMA_INFORMATION, ActiveProcessorsAffinityMask)) / sizeof(ULONGLONG); 2382 /* Make sure we don't overflow KeNodeBlock */ 2383 if (MaxEntries > KeNumberNodes) 2384 { 2385 MaxEntries = KeNumberNodes; 2386 } 2387 2388 /* If we have entries to write, and room for it */ 2389 if (Size >= FIELD_OFFSET(SYSTEM_NUMA_INFORMATION, ActiveProcessorsAffinityMask) && 2390 MaxEntries != 0) 2391 { 2392 /* Already set size we return */ 2393 *ReqSize = FIELD_OFFSET(SYSTEM_NUMA_INFORMATION, ActiveProcessorsAffinityMask) + 2394 MaxEntries * sizeof(ULONGLONG); 2395 2396 /* For each node, return processor mask */ 2397 for (Node = 0; Node < MaxEntries; ++Node) 2398 { 2399 NumaInformation->ActiveProcessorsAffinityMask[Node] = KeNodeBlock[Node]->ProcessorMask; 2400 } 2401 } 2402 else 2403 { 2404 /* We only returned highest node number */ 2405 *ReqSize = sizeof(ULONG); 2406 } 2407 2408 return STATUS_SUCCESS; 2409 } 2410 2411 /* Class 56 - Prefetcher information */ 2412 QSI_DEF(SystemPrefetcherInformation) 2413 { 2414 /* FIXME */ 2415 DPRINT1("NtQuerySystemInformation - SystemPrefetcherInformation not implemented\n"); 2416 return STATUS_NOT_IMPLEMENTED; 2417 } 2418 2419 /* Class 57 - Extended process information */ 2420 QSI_DEF(SystemExtendedProcessInformation) 2421 { 2422 /* FIXME */ 2423 DPRINT1("NtQuerySystemInformation - SystemExtendedProcessInformation not implemented\n"); 2424 return STATUS_NOT_IMPLEMENTED; 2425 } 2426 2427 /* Class 58 - Recommended shared data alignment */ 2428 QSI_DEF(SystemRecommendedSharedDataAlignment) 2429 { 2430 /* FIXME */ 2431 DPRINT1("NtQuerySystemInformation - SystemRecommendedSharedDataAlignment not implemented\n"); 2432 return STATUS_NOT_IMPLEMENTED; 2433 } 2434 2435 /* Class 60 - NUMA memory information */ 2436 QSI_DEF(SystemNumaAvailableMemory) 2437 { 2438 ULONG MaxEntries, Node; 2439 PSYSTEM_NUMA_INFORMATION NumaInformation = (PSYSTEM_NUMA_INFORMATION)Buffer; 2440 2441 /* Validate input size */ 2442 if (Size < sizeof(ULONG)) 2443 { 2444 return STATUS_INFO_LENGTH_MISMATCH; 2445 } 2446 2447 /* Return highest node */ 2448 NumaInformation->HighestNodeNumber = KeNumberNodes - 1; 2449 2450 /* Compute how much entries we will be able to put in output structure */ 2451 MaxEntries = (Size - FIELD_OFFSET(SYSTEM_NUMA_INFORMATION, AvailableMemory)) / sizeof(ULONGLONG); 2452 /* Make sure we don't overflow KeNodeBlock */ 2453 if (MaxEntries > KeNumberNodes) 2454 { 2455 MaxEntries = KeNumberNodes; 2456 } 2457 2458 /* If we have entries to write, and room for it */ 2459 if (Size >= FIELD_OFFSET(SYSTEM_NUMA_INFORMATION, AvailableMemory) && 2460 MaxEntries != 0) 2461 { 2462 /* Already set size we return */ 2463 *ReqSize = FIELD_OFFSET(SYSTEM_NUMA_INFORMATION, AvailableMemory) + 2464 MaxEntries * sizeof(ULONGLONG); 2465 2466 /* If we have a single entry (us), directly return MM information */ 2467 if (MaxEntries == 1) 2468 { 2469 NumaInformation->AvailableMemory[0] = MmAvailablePages << PAGE_SHIFT; 2470 } 2471 else 2472 { 2473 /* Otherwise, for each node, return available bytes */ 2474 for (Node = 0; Node < MaxEntries; ++Node) 2475 { 2476 NumaInformation->AvailableMemory[Node] = (KeNodeBlock[Node]->FreeCount[0] + KeNodeBlock[Node]->FreeCount[1]) << PAGE_SHIFT; 2477 } 2478 } 2479 } 2480 else 2481 { 2482 /* We only returned highest node number */ 2483 *ReqSize = sizeof(ULONG); 2484 } 2485 2486 return STATUS_SUCCESS; 2487 } 2488 2489 /* Class 64 - Extended handle information */ 2490 QSI_DEF(SystemExtendedHandleInformation) 2491 { 2492 PSYSTEM_HANDLE_INFORMATION_EX HandleInformation; 2493 PLIST_ENTRY NextTableEntry; 2494 PHANDLE_TABLE HandleTable; 2495 PHANDLE_TABLE_ENTRY HandleTableEntry; 2496 EXHANDLE Handle; 2497 ULONG Index = 0; 2498 NTSTATUS Status; 2499 PMDL Mdl; 2500 PAGED_CODE(); 2501 2502 DPRINT("NtQuerySystemInformation - SystemExtendedHandleInformation\n"); 2503 2504 /* Set initial required buffer size */ 2505 *ReqSize = FIELD_OFFSET(SYSTEM_HANDLE_INFORMATION_EX, Handle); 2506 2507 /* Check user's buffer size */ 2508 if (Size < *ReqSize) 2509 { 2510 return STATUS_INFO_LENGTH_MISMATCH; 2511 } 2512 2513 /* We need to lock down the memory */ 2514 Status = ExLockUserBuffer(Buffer, 2515 Size, 2516 ExGetPreviousMode(), 2517 IoWriteAccess, 2518 (PVOID*)&HandleInformation, 2519 &Mdl); 2520 if (!NT_SUCCESS(Status)) 2521 { 2522 DPRINT1("Failed to lock the user buffer: 0x%lx\n", Status); 2523 return Status; 2524 } 2525 2526 /* Reset of count of handles */ 2527 HandleInformation->Count = 0; 2528 2529 /* Enter a critical region */ 2530 KeEnterCriticalRegion(); 2531 2532 /* Acquire the handle table lock */ 2533 ExAcquirePushLockShared(&HandleTableListLock); 2534 2535 /* Enumerate all system handles */ 2536 for (NextTableEntry = HandleTableListHead.Flink; 2537 NextTableEntry != &HandleTableListHead; 2538 NextTableEntry = NextTableEntry->Flink) 2539 { 2540 /* Get current handle table */ 2541 HandleTable = CONTAINING_RECORD(NextTableEntry, HANDLE_TABLE, HandleTableList); 2542 2543 /* Set the initial value and loop the entries */ 2544 Handle.Value = 0; 2545 while ((HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, Handle))) 2546 { 2547 /* Validate the entry */ 2548 if ((HandleTableEntry->Object) && 2549 (HandleTableEntry->NextFreeTableEntry != -2)) 2550 { 2551 /* Increase of count of handles */ 2552 ++HandleInformation->Count; 2553 2554 /* Lock the entry */ 2555 if (ExpLockHandleTableEntry(HandleTable, HandleTableEntry)) 2556 { 2557 /* Increase required buffer size */ 2558 *ReqSize += sizeof(SYSTEM_HANDLE_TABLE_ENTRY_INFO_EX); 2559 2560 /* Check user's buffer size */ 2561 if (*ReqSize > Size) 2562 { 2563 Status = STATUS_INFO_LENGTH_MISMATCH; 2564 } 2565 else 2566 { 2567 POBJECT_HEADER ObjectHeader = ObpGetHandleObject(HandleTableEntry); 2568 2569 /* Filling handle information */ 2570 HandleInformation->Handle[Index].UniqueProcessId = 2571 (USHORT)(ULONG_PTR) HandleTable->UniqueProcessId; 2572 2573 HandleInformation->Handle[Index].CreatorBackTraceIndex = 0; 2574 2575 #if 0 /* FIXME!!! Type field currupted */ 2576 HandleInformation->Handles[Index].ObjectTypeIndex = 2577 (UCHAR) ObjectHeader->Type->Index; 2578 #else 2579 HandleInformation->Handle[Index].ObjectTypeIndex = 0; 2580 #endif 2581 2582 HandleInformation->Handle[Index].HandleAttributes = 2583 HandleTableEntry->ObAttributes & OBJ_HANDLE_ATTRIBUTES; 2584 2585 HandleInformation->Handle[Index].HandleValue = 2586 (USHORT)(ULONG_PTR) Handle.GenericHandleOverlay; 2587 2588 HandleInformation->Handle[Index].Object = &ObjectHeader->Body; 2589 2590 HandleInformation->Handle[Index].GrantedAccess = 2591 HandleTableEntry->GrantedAccess; 2592 2593 HandleInformation->Handle[Index].Reserved = 0; 2594 2595 ++Index; 2596 } 2597 2598 /* Unlock it */ 2599 ExUnlockHandleTableEntry(HandleTable, HandleTableEntry); 2600 } 2601 } 2602 2603 /* Go to the next entry */ 2604 Handle.Value += sizeof(HANDLE); 2605 } 2606 } 2607 2608 /* Release the lock */ 2609 ExReleasePushLockShared(&HandleTableListLock); 2610 2611 /* Leave the critical region */ 2612 KeLeaveCriticalRegion(); 2613 2614 /* Release the locked user buffer */ 2615 ExUnlockUserBuffer(Mdl); 2616 2617 return Status; 2618 } 2619 2620 /* Class 70 - System object security mode information */ 2621 QSI_DEF(SystemObjectSecurityMode) 2622 { 2623 PULONG ObjectSecurityInfo = (PULONG)Buffer; 2624 2625 /* Validate input size */ 2626 if (Size != sizeof(ULONG)) 2627 { 2628 return STATUS_INFO_LENGTH_MISMATCH; 2629 } 2630 2631 *ObjectSecurityInfo = ObpObjectSecurityMode; 2632 2633 return STATUS_SUCCESS; 2634 } 2635 2636 /* Class 73 - Logical processor information */ 2637 QSI_DEF(SystemLogicalProcessorInformation) 2638 { 2639 LONG i; 2640 PKPRCB Prcb; 2641 KAFFINITY CurrentProc; 2642 NTSTATUS Status = STATUS_SUCCESS; 2643 ULONG DataSize = 0, ProcessorFlags; 2644 PSYSTEM_LOGICAL_PROCESSOR_INFORMATION CurrentInfo; 2645 2646 /* First, browse active processors, thanks to the map */ 2647 i = 0; 2648 CurrentInfo = Buffer; 2649 CurrentProc = KeActiveProcessors; 2650 do 2651 { 2652 /* If current processor is active and is main in case of HT/MC, return it */ 2653 Prcb = KiProcessorBlock[i]; 2654 if ((CurrentProc & 1) && 2655 Prcb == Prcb->MultiThreadSetMaster) 2656 { 2657 /* Assume processor can do HT or multicore */ 2658 ProcessorFlags = 1; 2659 2660 /* If set is the same for PRCB and multithread, then 2661 * actually, the processor is single core 2662 */ 2663 if (Prcb->SetMember == Prcb->MultiThreadProcessorSet) 2664 { 2665 ProcessorFlags = 0; 2666 } 2667 2668 /* Check we have enough room to return */ 2669 DataSize += sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION); 2670 if (DataSize > Size) 2671 { 2672 Status = STATUS_INFO_LENGTH_MISMATCH; 2673 } 2674 else 2675 { 2676 /* Zero output and return */ 2677 RtlZeroMemory(CurrentInfo, sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION)); 2678 CurrentInfo->ProcessorMask = Prcb->MultiThreadProcessorSet; 2679 2680 /* Processor core needs 1 if HT/MC is supported */ 2681 CurrentInfo->Relationship = RelationProcessorCore; 2682 CurrentInfo->ProcessorCore.Flags = ProcessorFlags; 2683 ++CurrentInfo; 2684 } 2685 } 2686 2687 /* Move to the next proc */ 2688 CurrentProc >>= 1; 2689 ++i; 2690 /* Loop while there's someone in the bitmask */ 2691 } while (CurrentProc != 0); 2692 2693 /* Now, return the NUMA nodes */ 2694 for (i = 0; i < KeNumberNodes; ++i) 2695 { 2696 /* Check we have enough room to return */ 2697 DataSize += sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION); 2698 if (DataSize > Size) 2699 { 2700 Status = STATUS_INFO_LENGTH_MISMATCH; 2701 } 2702 else 2703 { 2704 /* Zero output and return */ 2705 RtlZeroMemory(CurrentInfo, sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION)); 2706 CurrentInfo->ProcessorMask = KeActiveProcessors; 2707 2708 /* NUMA node needs its ID */ 2709 CurrentInfo->Relationship = RelationNumaNode; 2710 CurrentInfo->NumaNode.NodeNumber = i; 2711 ++CurrentInfo; 2712 } 2713 } 2714 2715 *ReqSize = DataSize; 2716 2717 return Status; 2718 } 2719 2720 /* Class 76 - System firmware table information */ 2721 QSI_DEF(SystemFirmwareTableInformation) 2722 { 2723 PSYSTEM_FIRMWARE_TABLE_INFORMATION SysFirmwareInfo = (PSYSTEM_FIRMWARE_TABLE_INFORMATION)Buffer; 2724 NTSTATUS Status = STATUS_SUCCESS; 2725 ULONG InputBufSize; 2726 ULONG DataSize = 0; 2727 ULONG TableCount = 0; 2728 2729 DPRINT("NtQuerySystemInformation - SystemFirmwareTableInformation\n"); 2730 2731 /* Set initial required buffer size */ 2732 *ReqSize = FIELD_OFFSET(SYSTEM_FIRMWARE_TABLE_INFORMATION, TableBuffer); 2733 2734 /* Check user's buffer size */ 2735 if (Size < *ReqSize) 2736 { 2737 return STATUS_INFO_LENGTH_MISMATCH; 2738 } 2739 2740 InputBufSize = SysFirmwareInfo->TableBufferLength; 2741 switch (SysFirmwareInfo->ProviderSignature) 2742 { 2743 /* 2744 * ExpFirmwareTableResource and ExpFirmwareTableProviderListHead 2745 * variables should be used there somehow... 2746 */ 2747 case SIG_ACPI: 2748 { 2749 /* FIXME: Not implemented yet */ 2750 DPRINT1("ACPI provider not implemented\n"); 2751 Status = STATUS_NOT_IMPLEMENTED; 2752 break; 2753 } 2754 case SIG_FIRM: 2755 { 2756 /* FIXME: Not implemented yet */ 2757 DPRINT1("FIRM provider not implemented\n"); 2758 Status = STATUS_NOT_IMPLEMENTED; 2759 break; 2760 } 2761 case SIG_RSMB: 2762 { 2763 Status = ExpGetRawSMBiosTable(NULL, &DataSize, 0); 2764 if (DataSize > 0) 2765 { 2766 TableCount = 1; 2767 if (SysFirmwareInfo->Action == SystemFirmwareTable_Enumerate) 2768 { 2769 DataSize = TableCount * sizeof(ULONG); 2770 if (DataSize <= InputBufSize) 2771 { 2772 *(ULONG *)SysFirmwareInfo->TableBuffer = 0; 2773 } 2774 } 2775 else if (SysFirmwareInfo->Action == SystemFirmwareTable_Get 2776 && DataSize <= InputBufSize) 2777 { 2778 Status = ExpGetRawSMBiosTable(SysFirmwareInfo->TableBuffer, &DataSize, InputBufSize); 2779 } 2780 SysFirmwareInfo->TableBufferLength = DataSize; 2781 } 2782 break; 2783 } 2784 default: 2785 { 2786 DPRINT1("SystemFirmwareTableInformation: Unsupported provider (0x%x)\n", 2787 SysFirmwareInfo->ProviderSignature); 2788 Status = STATUS_ILLEGAL_FUNCTION; 2789 } 2790 } 2791 2792 if (NT_SUCCESS(Status)) 2793 { 2794 switch (SysFirmwareInfo->Action) 2795 { 2796 case SystemFirmwareTable_Enumerate: 2797 case SystemFirmwareTable_Get: 2798 { 2799 if (SysFirmwareInfo->TableBufferLength > InputBufSize) 2800 { 2801 Status = STATUS_BUFFER_TOO_SMALL; 2802 } 2803 break; 2804 } 2805 default: 2806 { 2807 DPRINT1("SystemFirmwareTableInformation: Unsupported action (0x%x)\n", 2808 SysFirmwareInfo->Action); 2809 Status = STATUS_ILLEGAL_FUNCTION; 2810 } 2811 } 2812 } 2813 else 2814 { 2815 SysFirmwareInfo->TableBufferLength = 0; 2816 } 2817 return Status; 2818 } 2819 2820 /* Query/Set Calls Table */ 2821 typedef 2822 struct _QSSI_CALLS 2823 { 2824 NTSTATUS (* Query) (PVOID,ULONG,PULONG); 2825 NTSTATUS (* Set) (PVOID,ULONG); 2826 } QSSI_CALLS; 2827 2828 // QS Query & Set 2829 // QX Query 2830 // XS Set 2831 // XX unknown behaviour 2832 // 2833 #define SI_QS(n) {QSI_USE(n),SSI_USE(n)} 2834 #define SI_QX(n) {QSI_USE(n),NULL} 2835 #define SI_XS(n) {NULL,SSI_USE(n)} 2836 #define SI_XX(n) {NULL,NULL} 2837 2838 static 2839 QSSI_CALLS 2840 CallQS[] = 2841 { 2842 SI_QX(SystemBasicInformation), 2843 SI_QX(SystemProcessorInformation), 2844 SI_QX(SystemPerformanceInformation), 2845 SI_QX(SystemTimeOfDayInformation), 2846 SI_QX(SystemPathInformation), 2847 SI_QX(SystemProcessInformation), 2848 SI_QX(SystemCallCountInformation), 2849 SI_QX(SystemDeviceInformation), 2850 SI_QX(SystemProcessorPerformanceInformation), 2851 SI_QS(SystemFlagsInformation), 2852 SI_QX(SystemCallTimeInformation), /* should be SI_XX */ 2853 SI_QX(SystemModuleInformation), 2854 SI_QX(SystemLocksInformation), 2855 SI_QX(SystemStackTraceInformation), /* should be SI_XX */ 2856 SI_QX(SystemPagedPoolInformation), /* should be SI_XX */ 2857 SI_QX(SystemNonPagedPoolInformation), /* should be SI_XX */ 2858 SI_QX(SystemHandleInformation), 2859 SI_QX(SystemObjectInformation), 2860 SI_QX(SystemPageFileInformation), 2861 SI_QX(SystemVdmInstemulInformation), 2862 SI_QX(SystemVdmBopInformation), /* it should be SI_XX */ 2863 SI_QS(SystemFileCacheInformation), 2864 SI_QX(SystemPoolTagInformation), 2865 SI_QX(SystemInterruptInformation), 2866 SI_QS(SystemDpcBehaviourInformation), 2867 SI_QX(SystemFullMemoryInformation), /* it should be SI_XX */ 2868 SI_XS(SystemLoadGdiDriverInformation), 2869 SI_XS(SystemUnloadGdiDriverInformation), 2870 SI_QS(SystemTimeAdjustmentInformation), 2871 SI_QX(SystemSummaryMemoryInformation), /* it should be SI_XX */ 2872 SI_QX(SystemNextEventIdInformation), /* it should be SI_XX */ 2873 SI_QX(SystemPerformanceTraceInformation), /* it should be SI_XX */ 2874 SI_QX(SystemCrashDumpInformation), 2875 SI_QX(SystemExceptionInformation), 2876 SI_QX(SystemCrashDumpStateInformation), 2877 SI_QX(SystemKernelDebuggerInformation), 2878 SI_QX(SystemContextSwitchInformation), 2879 SI_QS(SystemRegistryQuotaInformation), 2880 SI_XS(SystemExtendServiceTableInformation), 2881 SI_XS(SystemPrioritySeperation), 2882 SI_QX(SystemVerifierAddDriverInformation), /* it should be SI_XX */ 2883 SI_QX(SystemVerifierRemoveDriverInformation), /* it should be SI_XX */ 2884 SI_QX(SystemProcessorIdleInformation), /* it should be SI_XX */ 2885 SI_QX(SystemLegacyDriverInformation), /* it should be SI_XX */ 2886 SI_QS(SystemCurrentTimeZoneInformation), /* it should be SI_QX */ 2887 SI_QX(SystemLookasideInformation), 2888 SI_XS(SystemTimeSlipNotification), 2889 SI_XS(SystemSessionCreate), 2890 SI_XS(SystemSessionDetach), 2891 SI_QX(SystemSessionInformation), /* it should be SI_XX */ 2892 SI_QX(SystemRangeStartInformation), 2893 SI_QS(SystemVerifierInformation), 2894 SI_XS(SystemVerifierThunkExtend), 2895 SI_QX(SystemSessionProcessesInformation), 2896 SI_XS(SystemLoadGdiDriverInSystemSpaceInformation), 2897 SI_QX(SystemNumaProcessorMap), 2898 SI_QX(SystemPrefetcherInformation), 2899 SI_QX(SystemExtendedProcessInformation), 2900 SI_QX(SystemRecommendedSharedDataAlignment), 2901 SI_XX(SystemComPlusPackage), 2902 SI_QX(SystemNumaAvailableMemory), 2903 SI_XX(SystemProcessorPowerInformation), /* FIXME: not implemented */ 2904 SI_XX(SystemEmulationBasicInformation), /* FIXME: not implemented */ 2905 SI_XX(SystemEmulationProcessorInformation), /* FIXME: not implemented */ 2906 SI_QX(SystemExtendedHandleInformation), 2907 SI_XX(SystemLostDelayedWriteInformation), /* FIXME: not implemented */ 2908 SI_XX(SystemBigPoolInformation), /* FIXME: not implemented */ 2909 SI_XX(SystemSessionPoolTagInformation), /* FIXME: not implemented */ 2910 SI_XX(SystemSessionMappedViewInformation), /* FIXME: not implemented */ 2911 SI_XX(SystemHotpatchInformation), /* FIXME: not implemented */ 2912 SI_QX(SystemObjectSecurityMode), 2913 SI_XX(SystemWatchdogTimerHandler), /* FIXME: not implemented */ 2914 SI_XX(SystemWatchdogTimerInformation), /* FIXME: not implemented */ 2915 SI_QX(SystemLogicalProcessorInformation), 2916 SI_XX(SystemWow64SharedInformation), /* FIXME: not implemented */ 2917 SI_XX(SystemRegisterFirmwareTableInformationHandler), /* FIXME: not implemented */ 2918 SI_QX(SystemFirmwareTableInformation), 2919 }; 2920 2921 C_ASSERT(SystemBasicInformation == 0); 2922 #define MIN_SYSTEM_INFO_CLASS (SystemBasicInformation) 2923 #define MAX_SYSTEM_INFO_CLASS RTL_NUMBER_OF(CallQS) 2924 2925 /* 2926 * @implemented 2927 */ 2928 __kernel_entry 2929 NTSTATUS 2930 NTAPI 2931 NtQuerySystemInformation( 2932 _In_ SYSTEM_INFORMATION_CLASS SystemInformationClass, 2933 _Out_writes_bytes_to_opt_(SystemInformationLength, *ReturnLength) PVOID SystemInformation, 2934 _In_ ULONG SystemInformationLength, 2935 _Out_opt_ PULONG ReturnLength) 2936 { 2937 NTSTATUS Status = STATUS_NOT_IMPLEMENTED; 2938 ULONG CapturedResultLength = 0; 2939 ULONG Alignment = TYPE_ALIGNMENT(ULONG); 2940 KPROCESSOR_MODE PreviousMode; 2941 2942 PAGED_CODE(); 2943 2944 PreviousMode = ExGetPreviousMode(); 2945 2946 _SEH2_TRY 2947 { 2948 #if (NTDDI_VERSION >= NTDDI_VISTA) 2949 /* 2950 * Check whether the request is valid. 2951 */ 2952 if (SystemInformationClass < MIN_SYSTEM_INFO_CLASS || 2953 SystemInformationClass >= MAX_SYSTEM_INFO_CLASS) 2954 { 2955 _SEH2_YIELD(return STATUS_INVALID_INFO_CLASS); 2956 } 2957 #endif 2958 2959 if (PreviousMode != KernelMode) 2960 { 2961 /* SystemKernelDebuggerInformation needs only BOOLEAN alignment */ 2962 if (SystemInformationClass == SystemKernelDebuggerInformation) 2963 Alignment = TYPE_ALIGNMENT(BOOLEAN); 2964 2965 ProbeForWrite(SystemInformation, SystemInformationLength, Alignment); 2966 if (ReturnLength != NULL) 2967 ProbeForWriteUlong(ReturnLength); 2968 } 2969 2970 if (ReturnLength) 2971 *ReturnLength = 0; 2972 2973 #if (NTDDI_VERSION < NTDDI_VISTA) 2974 /* 2975 * Check whether the request is valid. 2976 */ 2977 if (SystemInformationClass < MIN_SYSTEM_INFO_CLASS || 2978 SystemInformationClass >= MAX_SYSTEM_INFO_CLASS) 2979 { 2980 _SEH2_YIELD(return STATUS_INVALID_INFO_CLASS); 2981 } 2982 #endif 2983 2984 if (CallQS[SystemInformationClass].Query != NULL) 2985 { 2986 /* Hand the request to a subhandler */ 2987 Status = CallQS[SystemInformationClass].Query(SystemInformation, 2988 SystemInformationLength, 2989 &CapturedResultLength); 2990 2991 /* Save the result length to the caller */ 2992 if (ReturnLength) 2993 *ReturnLength = CapturedResultLength; 2994 } 2995 } 2996 _SEH2_EXCEPT(ExSystemExceptionFilter()) 2997 { 2998 Status = _SEH2_GetExceptionCode(); 2999 } 3000 _SEH2_END; 3001 3002 return Status; 3003 } 3004 3005 __kernel_entry 3006 NTSTATUS 3007 NTAPI 3008 NtSetSystemInformation( 3009 _In_ SYSTEM_INFORMATION_CLASS SystemInformationClass, 3010 _In_reads_bytes_(SystemInformationLength) PVOID SystemInformation, 3011 _In_ ULONG SystemInformationLength) 3012 { 3013 NTSTATUS Status = STATUS_INVALID_INFO_CLASS; 3014 KPROCESSOR_MODE PreviousMode; 3015 3016 PAGED_CODE(); 3017 3018 PreviousMode = ExGetPreviousMode(); 3019 3020 _SEH2_TRY 3021 { 3022 /* 3023 * If called from user mode, check possible unsafe arguments. 3024 */ 3025 if (PreviousMode != KernelMode) 3026 { 3027 ProbeForRead(SystemInformation, SystemInformationLength, sizeof(ULONG)); 3028 } 3029 3030 /* 3031 * Check whether the request is valid. 3032 */ 3033 if ((SystemInformationClass >= MIN_SYSTEM_INFO_CLASS) && 3034 (SystemInformationClass < MAX_SYSTEM_INFO_CLASS)) 3035 { 3036 if (CallQS[SystemInformationClass].Set != NULL) 3037 { 3038 /* Hand the request to a subhandler */ 3039 Status = CallQS[SystemInformationClass].Set(SystemInformation, 3040 SystemInformationLength); 3041 } 3042 } 3043 } 3044 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 3045 { 3046 Status = _SEH2_GetExceptionCode(); 3047 } 3048 _SEH2_END; 3049 3050 return Status; 3051 } 3052 3053 ULONG 3054 NTAPI 3055 NtGetCurrentProcessorNumber(VOID) 3056 { 3057 /* Just use Ke */ 3058 return KeGetCurrentProcessorNumber(); 3059 } 3060 3061 #undef ExGetPreviousMode 3062 KPROCESSOR_MODE 3063 NTAPI 3064 ExGetPreviousMode(VOID) 3065 { 3066 /* Just use Ke */ 3067 return KeGetPreviousMode(); 3068 } 3069