#include "cpu.h" #include "util/windows/registry.h" #include "util/windows/nt.h" #include "util/mallocHelper.h" #include "util/smbiosHelper.h" #include static void ffPdhOpenCloseQuery(HQUERY* query) { assert(query); if (*query) { PdhCloseQuery(*query); *query = NULL; } } static const char* detectThermalTemp(double* result) { // typeperf.exe -sc 1 "\Thermal Zone Information(*)\Temperature" __attribute__((__cleanup__(ffPdhOpenCloseQuery))) HQUERY query = NULL; if (PdhOpenQueryW(NULL, 0, &query) != ERROR_SUCCESS) return "Failed to open PDH query"; HCOUNTER counter = NULL; if (PdhAddEnglishCounterW(query, L"\\Thermal Zone Information(*)\\Temperature", 0, &counter) != ERROR_SUCCESS) return "Failed to add TZI temperature counter"; if (PdhCollectQueryData(query) != ERROR_SUCCESS) return "Failed to collect query data"; PDH_FMT_COUNTERVALUE value; if (PdhGetFormattedCounterValue(counter, PDH_FMT_DOUBLE, NULL, &value) != ERROR_SUCCESS) return "Failed to format counter value"; *result = value.doubleValue - 273; return NULL; } // 7.5 typedef struct FFSmbiosProcessorInfo { FFSmbiosHeader Header; uint8_t SocketDesignation; // string uint8_t ProcessorType; // enum uint8_t ProcessorFamily; // enum uint8_t ProcessorManufacturer; // string uint64_t ProcessorID; // varies uint8_t ProcessorVersion; // string uint8_t Voltage; // varies uint16_t ExternalClock; // varies uint16_t MaxSpeed; // varies uint16_t CurrentSpeed; // varies uint8_t Status; // varies uint8_t ProcessorUpgrade; // enum // 2.1+ uint16_t L1CacheHandle; // varies uint16_t L2CacheHandle; // varies uint16_t L3CacheHandle; // varies // 2.3+ uint8_t SerialNumber; // string uint8_t AssertTag; // string uint8_t PartNumber; // string // 2.5+ uint8_t CoreCount; // varies uint8_t CoreEnabled; // varies uint8_t ThreadCount; // varies uint16_t ProcessorCharacteristics; // bit field // 2.6+ uint16_t ProcessorFamily2; // enum // 3.0+ uint16_t CoreCount2; // varies uint16_t CoreEnabled2; // varies uint16_t ThreadCount2; // varies // 3.6+ uint16_t ThreadEnabled; // varies } __attribute__((__packed__)) FFSmbiosProcessorInfo; static_assert(offsetof(FFSmbiosProcessorInfo, ThreadEnabled) == 0x30, "FFSmbiosProcessorInfo: Wrong struct alignment"); static const char* detectMaxSpeedBySmbios(FFCPUResult* cpu) { const FFSmbiosHeaderTable* smbiosTable = ffGetSmbiosHeaderTable(); if (!smbiosTable) return "Failed to get SMBIOS data"; const FFSmbiosProcessorInfo* data = (const FFSmbiosProcessorInfo*) (*smbiosTable)[FF_SMBIOS_TYPE_PROCESSOR_INFO]; if (!data) return "Processor information is not found in SMBIOS data"; while (data->ProcessorType != 0x03 /*Central Processor*/ || (data->Status & 0b00000111) != 1 /*Enabled*/) { data = (const FFSmbiosProcessorInfo*) ffSmbiosNextEntry(&data->Header); if (data->Header.Type != FF_SMBIOS_TYPE_PROCESSOR_INFO) return "No active CPU is found in SMBIOS data"; } uint32_t speed = data->MaxSpeed; // Sometimes SMBIOS reports invalid value. We assume that max speed is small than 2x of base if (speed < cpu->frequencyBase || speed > cpu->frequencyBase * 2) return "Possible invalid CPU max speed in SMBIOS data. See #800"; cpu->frequencyMax = speed; return NULL; } static const char* detectNCores(FFCPUResult* cpu) { DWORD length = 0; GetLogicalProcessorInformationEx(RelationAll, NULL, &length); if (length == 0) return "GetLogicalProcessorInformationEx(RelationAll, NULL, &length) failed"; SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX* FF_AUTO_FREE pProcessorInfo = (SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX*)malloc(length); if (!pProcessorInfo || !GetLogicalProcessorInformationEx(RelationAll, pProcessorInfo, &length)) return "GetLogicalProcessorInformationEx(RelationAll, pProcessorInfo, &length) failed"; for( SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX* ptr = pProcessorInfo; (uint8_t*)ptr < ((uint8_t*)pProcessorInfo) + length; ptr = (SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX*)(((uint8_t*)ptr) + ptr->Size) ) { if (ptr->Relationship == RelationGroup) { for (uint32_t index = 0; index < ptr->Group.ActiveGroupCount; ++index) { cpu->coresOnline += ptr->Group.GroupInfo[index].ActiveProcessorCount; cpu->coresLogical += ptr->Group.GroupInfo[index].MaximumProcessorCount; } } else if (ptr->Relationship == RelationProcessorCore) ++cpu->coresPhysical; else if (ptr->Relationship == RelationProcessorPackage) ++cpu->packages; } return NULL; } static const char* detectByRegistry(FFCPUResult* cpu) { FF_HKEY_AUTO_DESTROY hKey = NULL; if(!ffRegOpenKeyForRead(HKEY_LOCAL_MACHINE, L"HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0", &hKey, NULL)) return "ffRegOpenKeyForRead(HKEY_LOCAL_MACHINE, L\"HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0\", &hKey, NULL) failed"; ffRegReadStrbuf(hKey, L"ProcessorNameString", &cpu->name, NULL); ffRegReadStrbuf(hKey, L"VendorIdentifier", &cpu->vendor, NULL); if (cpu->coresLogical == 0) { FF_HKEY_AUTO_DESTROY hProcsKey = NULL; if (ffRegOpenKeyForRead(HKEY_LOCAL_MACHINE, L"HARDWARE\\DESCRIPTION\\System\\CentralProcessor", &hProcsKey, NULL)) { uint32_t cores; if (ffRegGetNSubKeys(hProcsKey, &cores, NULL)) cpu->coresOnline = cpu->coresPhysical = cpu->coresLogical = (uint16_t) cores; } } uint32_t mhz; if(ffRegReadUint(hKey, L"~MHz", &mhz, NULL)) cpu->frequencyBase = mhz; return NULL; } static const char* detectCoreTypes(FFCPUResult* cpu) { FF_AUTO_FREE PROCESSOR_POWER_INFORMATION* pinfo = calloc(cpu->coresLogical, sizeof(PROCESSOR_POWER_INFORMATION)); if (!NT_SUCCESS(NtPowerInformation(ProcessorInformation, NULL, 0, pinfo, (ULONG) sizeof(PROCESSOR_POWER_INFORMATION) * cpu->coresLogical))) return "NtPowerInformation(ProcessorInformation, NULL, 0, pinfo, size) failed"; for (uint32_t icore = 0; icore < cpu->coresLogical && pinfo[icore].MhzLimit; ++icore) { uint32_t ifreq = 0; while (cpu->coreTypes[ifreq].freq != pinfo[icore].MhzLimit && cpu->coreTypes[ifreq].freq > 0) ++ifreq; if (cpu->coreTypes[ifreq].freq == 0) cpu->coreTypes[ifreq].freq = pinfo[icore].MhzLimit; ++cpu->coreTypes[ifreq].count; } if (cpu->frequencyBase == 0) cpu->frequencyBase = pinfo->MaxMhz; return NULL; } const char* ffDetectCPUImpl(const FFCPUOptions* options, FFCPUResult* cpu) { detectNCores(cpu); const char* error = detectByRegistry(cpu); if (error) return error; ffCPUDetectSpeedByCpuid(cpu); if (options->showPeCoreCount) detectCoreTypes(cpu); if (cpu->frequencyMax == 0) detectMaxSpeedBySmbios(cpu); if(options->temp) detectThermalTemp(&cpu->temperature); return NULL; }