2013-07-10 00:45:36 +00:00
|
|
|
/*
|
|
|
|
* libjingle
|
|
|
|
* Copyright 2008 Google Inc.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
|
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
|
|
|
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "talk/base/systeminfo.h"
|
|
|
|
|
|
|
|
#if defined(WIN32)
|
|
|
|
#include <winsock2.h>
|
|
|
|
#ifndef EXCLUDE_D3D9
|
|
|
|
#include <d3d9.h>
|
|
|
|
#endif
|
|
|
|
#include <intrin.h> // for __cpuid()
|
|
|
|
#elif defined(OSX)
|
|
|
|
#include <ApplicationServices/ApplicationServices.h>
|
|
|
|
#include <CoreServices/CoreServices.h>
|
|
|
|
#elif defined(LINUX) || defined(ANDROID)
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#if defined(OSX) || defined(IOS)
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(WIN32)
|
|
|
|
#include "talk/base/scoped_ptr.h"
|
|
|
|
#include "talk/base/win32.h"
|
|
|
|
#elif defined(OSX)
|
|
|
|
#include "talk/base/macconversion.h"
|
|
|
|
#elif defined(LINUX) || defined(ANDROID)
|
|
|
|
#include "talk/base/linux.h"
|
|
|
|
#endif
|
|
|
|
#include "talk/base/common.h"
|
|
|
|
#include "talk/base/logging.h"
|
|
|
|
#include "talk/base/stringutils.h"
|
|
|
|
|
|
|
|
namespace talk_base {
|
|
|
|
|
|
|
|
// See Also: http://msdn.microsoft.com/en-us/library/ms683194(v=vs.85).aspx
|
|
|
|
#if defined(WIN32)
|
|
|
|
typedef BOOL (WINAPI *LPFN_GLPI)(
|
|
|
|
PSYSTEM_LOGICAL_PROCESSOR_INFORMATION,
|
|
|
|
PDWORD);
|
|
|
|
|
|
|
|
static void GetProcessorInformation(int* physical_cpus, int* cache_size) {
|
|
|
|
// GetLogicalProcessorInformation() is available on Windows XP SP3 and beyond.
|
|
|
|
LPFN_GLPI glpi = reinterpret_cast<LPFN_GLPI>(GetProcAddress(
|
|
|
|
GetModuleHandle(L"kernel32"),
|
|
|
|
"GetLogicalProcessorInformation"));
|
|
|
|
if (NULL == glpi) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Determine buffer size, allocate and get processor information.
|
|
|
|
// Size can change between calls (unlikely), so a loop is done.
|
|
|
|
DWORD return_length = 0;
|
2013-10-25 21:18:33 +00:00
|
|
|
scoped_ptr<SYSTEM_LOGICAL_PROCESSOR_INFORMATION[]> infos;
|
2013-07-10 00:45:36 +00:00
|
|
|
while (!glpi(infos.get(), &return_length)) {
|
|
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
|
|
|
|
infos.reset(new SYSTEM_LOGICAL_PROCESSOR_INFORMATION[
|
|
|
|
return_length / sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION)]);
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*physical_cpus = 0;
|
|
|
|
*cache_size = 0;
|
|
|
|
for (size_t i = 0;
|
|
|
|
i < return_length / sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION); ++i) {
|
|
|
|
if (infos[i].Relationship == RelationProcessorCore) {
|
|
|
|
++*physical_cpus;
|
|
|
|
} else if (infos[i].Relationship == RelationCache) {
|
|
|
|
int next_cache_size = static_cast<int>(infos[i].Cache.Size);
|
|
|
|
if (next_cache_size >= *cache_size) {
|
|
|
|
*cache_size = next_cache_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// TODO(fbarchard): Use gcc 4.4 provided cpuid intrinsic
|
|
|
|
// 32 bit fpic requires ebx be preserved
|
|
|
|
#if (defined(__pic__) || defined(__APPLE__)) && defined(__i386__)
|
|
|
|
static inline void __cpuid(int cpu_info[4], int info_type) {
|
|
|
|
__asm__ volatile ( // NOLINT
|
|
|
|
"mov %%ebx, %%edi\n"
|
|
|
|
"cpuid\n"
|
|
|
|
"xchg %%edi, %%ebx\n"
|
|
|
|
: "=a"(cpu_info[0]), "=D"(cpu_info[1]), "=c"(cpu_info[2]), "=d"(cpu_info[3])
|
|
|
|
: "a"(info_type)
|
|
|
|
); // NOLINT
|
|
|
|
}
|
|
|
|
#elif defined(__i386__) || defined(__x86_64__)
|
|
|
|
static inline void __cpuid(int cpu_info[4], int info_type) {
|
|
|
|
__asm__ volatile ( // NOLINT
|
|
|
|
"cpuid\n"
|
|
|
|
: "=a"(cpu_info[0]), "=b"(cpu_info[1]), "=c"(cpu_info[2]), "=d"(cpu_info[3])
|
|
|
|
: "a"(info_type)
|
|
|
|
); // NOLINT
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif // WIN32
|
|
|
|
|
|
|
|
// Note(fbarchard):
|
|
|
|
// Family and model are extended family and extended model. 8 bits each.
|
|
|
|
SystemInfo::SystemInfo()
|
|
|
|
: physical_cpus_(1), logical_cpus_(1), cache_size_(0),
|
|
|
|
cpu_family_(0), cpu_model_(0), cpu_stepping_(0),
|
|
|
|
cpu_speed_(0), memory_(0) {
|
|
|
|
// Initialize the basic information.
|
|
|
|
#if defined(__arm__) || defined(_M_ARM)
|
|
|
|
cpu_arch_ = SI_ARCH_ARM;
|
|
|
|
#elif defined(__x86_64__) || defined(_M_X64)
|
|
|
|
cpu_arch_ = SI_ARCH_X64;
|
|
|
|
#elif defined(__i386__) || defined(_M_IX86)
|
|
|
|
cpu_arch_ = SI_ARCH_X86;
|
|
|
|
#else
|
|
|
|
cpu_arch_ = SI_ARCH_UNKNOWN;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(WIN32)
|
|
|
|
SYSTEM_INFO si;
|
|
|
|
GetSystemInfo(&si);
|
|
|
|
logical_cpus_ = si.dwNumberOfProcessors;
|
|
|
|
GetProcessorInformation(&physical_cpus_, &cache_size_);
|
|
|
|
if (physical_cpus_ <= 0) {
|
|
|
|
physical_cpus_ = logical_cpus_;
|
|
|
|
}
|
|
|
|
cpu_family_ = si.wProcessorLevel;
|
|
|
|
cpu_model_ = si.wProcessorRevision >> 8;
|
|
|
|
cpu_stepping_ = si.wProcessorRevision & 0xFF;
|
|
|
|
#elif defined(OSX) || defined(IOS)
|
|
|
|
uint32_t sysctl_value;
|
|
|
|
size_t length = sizeof(sysctl_value);
|
|
|
|
if (!sysctlbyname("hw.physicalcpu_max", &sysctl_value, &length, NULL, 0)) {
|
|
|
|
physical_cpus_ = static_cast<int>(sysctl_value);
|
|
|
|
}
|
|
|
|
length = sizeof(sysctl_value);
|
|
|
|
if (!sysctlbyname("hw.logicalcpu_max", &sysctl_value, &length, NULL, 0)) {
|
|
|
|
logical_cpus_ = static_cast<int>(sysctl_value);
|
|
|
|
}
|
|
|
|
uint64_t sysctl_value64;
|
|
|
|
length = sizeof(sysctl_value64);
|
|
|
|
if (!sysctlbyname("hw.l3cachesize", &sysctl_value64, &length, NULL, 0)) {
|
|
|
|
cache_size_ = static_cast<int>(sysctl_value64);
|
|
|
|
}
|
|
|
|
if (!cache_size_) {
|
|
|
|
length = sizeof(sysctl_value64);
|
|
|
|
if (!sysctlbyname("hw.l2cachesize", &sysctl_value64, &length, NULL, 0)) {
|
|
|
|
cache_size_ = static_cast<int>(sysctl_value64);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
length = sizeof(sysctl_value);
|
|
|
|
if (!sysctlbyname("machdep.cpu.family", &sysctl_value, &length, NULL, 0)) {
|
|
|
|
cpu_family_ = static_cast<int>(sysctl_value);
|
|
|
|
}
|
|
|
|
length = sizeof(sysctl_value);
|
|
|
|
if (!sysctlbyname("machdep.cpu.model", &sysctl_value, &length, NULL, 0)) {
|
|
|
|
cpu_model_ = static_cast<int>(sysctl_value);
|
|
|
|
}
|
|
|
|
length = sizeof(sysctl_value);
|
|
|
|
if (!sysctlbyname("machdep.cpu.stepping", &sysctl_value, &length, NULL, 0)) {
|
|
|
|
cpu_stepping_ = static_cast<int>(sysctl_value);
|
|
|
|
}
|
2013-10-25 21:18:33 +00:00
|
|
|
#elif defined(__native_client__)
|
|
|
|
// TODO(ryanpetrie): Implement this via PPAPI when it's available.
|
2013-07-10 00:45:36 +00:00
|
|
|
#else // LINUX || ANDROID
|
|
|
|
ProcCpuInfo proc_info;
|
|
|
|
if (proc_info.LoadFromSystem()) {
|
|
|
|
proc_info.GetNumCpus(&logical_cpus_);
|
|
|
|
proc_info.GetNumPhysicalCpus(&physical_cpus_);
|
|
|
|
proc_info.GetCpuFamily(&cpu_family_);
|
|
|
|
#if defined(CPU_X86)
|
|
|
|
// These values only apply to x86 systems.
|
|
|
|
proc_info.GetSectionIntValue(0, "model", &cpu_model_);
|
|
|
|
proc_info.GetSectionIntValue(0, "stepping", &cpu_stepping_);
|
|
|
|
proc_info.GetSectionIntValue(0, "cpu MHz", &cpu_speed_);
|
|
|
|
proc_info.GetSectionIntValue(0, "cache size", &cache_size_);
|
|
|
|
cache_size_ *= 1024;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
// ProcCpuInfo reads cpu speed from "cpu MHz" under /proc/cpuinfo.
|
|
|
|
// But that number is a moving target which can change on-the-fly according to
|
|
|
|
// many factors including system workload.
|
|
|
|
// See /sys/devices/system/cpu/cpu0/cpufreq/scaling_available_governors.
|
|
|
|
// The one in /sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq is more
|
|
|
|
// accurate. We use it as our cpu speed when it is available.
|
|
|
|
// cpuinfo_max_freq is measured in KHz and requires conversion to MHz.
|
|
|
|
int max_freq = talk_base::ReadCpuMaxFreq();
|
|
|
|
if (max_freq > 0) {
|
|
|
|
cpu_speed_ = max_freq / 1000;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
// For L2 CacheSize see also
|
|
|
|
// http://www.flounder.com/cpuid_explorer2.htm#CPUID(0x800000006)
|
|
|
|
#ifdef CPU_X86
|
|
|
|
if (cache_size_ == 0) {
|
|
|
|
int cpu_info[4];
|
|
|
|
__cpuid(cpu_info, 0x80000000); // query maximum extended cpuid function.
|
|
|
|
if (static_cast<uint32>(cpu_info[0]) >= 0x80000006) {
|
|
|
|
__cpuid(cpu_info, 0x80000006);
|
|
|
|
cache_size_ = (cpu_info[2] >> 16) * 1024;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the number of cpu threads available to the system.
|
|
|
|
int SystemInfo::GetMaxCpus() {
|
|
|
|
return logical_cpus_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the number of cpu cores available to the system.
|
|
|
|
int SystemInfo::GetMaxPhysicalCpus() {
|
|
|
|
return physical_cpus_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the number of cpus available to the process. Since affinity can be
|
|
|
|
// changed on the fly, do not cache this value.
|
|
|
|
// Can be affected by heat.
|
|
|
|
int SystemInfo::GetCurCpus() {
|
|
|
|
int cur_cpus;
|
|
|
|
#if defined(WIN32)
|
|
|
|
DWORD_PTR process_mask, system_mask;
|
|
|
|
::GetProcessAffinityMask(::GetCurrentProcess(), &process_mask, &system_mask);
|
|
|
|
for (cur_cpus = 0; process_mask; ++cur_cpus) {
|
|
|
|
// Sparse-ones algorithm. There are slightly faster methods out there but
|
|
|
|
// they are unintuitive and won't make a difference on a single dword.
|
|
|
|
process_mask &= (process_mask - 1);
|
|
|
|
}
|
|
|
|
#elif defined(OSX) || defined(IOS)
|
|
|
|
uint32_t sysctl_value;
|
|
|
|
size_t length = sizeof(sysctl_value);
|
|
|
|
int error = sysctlbyname("hw.ncpu", &sysctl_value, &length, NULL, 0);
|
|
|
|
cur_cpus = !error ? static_cast<int>(sysctl_value) : 1;
|
|
|
|
#else
|
|
|
|
// Linux, Solaris, ANDROID
|
|
|
|
cur_cpus = static_cast<int>(sysconf(_SC_NPROCESSORS_ONLN));
|
|
|
|
#endif
|
|
|
|
return cur_cpus;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the type of this CPU.
|
|
|
|
SystemInfo::Architecture SystemInfo::GetCpuArchitecture() {
|
|
|
|
return cpu_arch_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the vendor string from the cpu, e.g. "GenuineIntel", "AuthenticAMD".
|
|
|
|
// See "Intel Processor Identification and the CPUID Instruction"
|
|
|
|
// (Intel document number: 241618)
|
|
|
|
std::string SystemInfo::GetCpuVendor() {
|
|
|
|
if (cpu_vendor_.empty()) {
|
|
|
|
#if defined(CPU_X86)
|
|
|
|
int cpu_info[4];
|
|
|
|
__cpuid(cpu_info, 0);
|
|
|
|
cpu_info[0] = cpu_info[1]; // Reorder output
|
|
|
|
cpu_info[1] = cpu_info[3];
|
|
|
|
cpu_info[2] = cpu_info[2];
|
|
|
|
cpu_info[3] = 0;
|
|
|
|
cpu_vendor_ = std::string(reinterpret_cast<char*>(&cpu_info[0]));
|
|
|
|
#elif defined(CPU_ARM)
|
|
|
|
cpu_vendor_ = std::string("ARM");
|
|
|
|
#else
|
|
|
|
cpu_vendor_ = std::string("Undefined");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return cpu_vendor_;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SystemInfo::GetCpuCacheSize() {
|
|
|
|
return cache_size_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the "family" of this CPU.
|
|
|
|
int SystemInfo::GetCpuFamily() {
|
|
|
|
return cpu_family_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the "model" of this CPU.
|
|
|
|
int SystemInfo::GetCpuModel() {
|
|
|
|
return cpu_model_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the "stepping" of this CPU.
|
|
|
|
int SystemInfo::GetCpuStepping() {
|
|
|
|
return cpu_stepping_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the clockrate of the primary processor in Mhz. This value can be
|
|
|
|
// cached. Returns -1 on error.
|
|
|
|
int SystemInfo::GetMaxCpuSpeed() {
|
|
|
|
if (cpu_speed_) {
|
|
|
|
return cpu_speed_;
|
|
|
|
}
|
|
|
|
#if defined(WIN32)
|
|
|
|
HKEY key;
|
|
|
|
static const WCHAR keyName[] =
|
|
|
|
L"HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0";
|
|
|
|
|
|
|
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, keyName , 0, KEY_QUERY_VALUE, &key)
|
|
|
|
== ERROR_SUCCESS) {
|
|
|
|
DWORD data, len;
|
|
|
|
len = sizeof(data);
|
|
|
|
|
|
|
|
if (RegQueryValueEx(key, L"~Mhz", 0, 0, reinterpret_cast<LPBYTE>(&data),
|
|
|
|
&len) == ERROR_SUCCESS) {
|
|
|
|
cpu_speed_ = data;
|
|
|
|
} else {
|
|
|
|
LOG(LS_WARNING) << "Failed to query registry value HKLM\\" << keyName
|
|
|
|
<< "\\~Mhz";
|
|
|
|
cpu_speed_ = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
RegCloseKey(key);
|
|
|
|
} else {
|
|
|
|
LOG(LS_WARNING) << "Failed to open registry key HKLM\\" << keyName;
|
|
|
|
cpu_speed_ = -1;
|
|
|
|
}
|
|
|
|
#elif defined(IOS) || defined(OSX)
|
|
|
|
uint64_t sysctl_value;
|
|
|
|
size_t length = sizeof(sysctl_value);
|
|
|
|
int error = sysctlbyname("hw.cpufrequency_max", &sysctl_value, &length,
|
|
|
|
NULL, 0);
|
|
|
|
cpu_speed_ = !error ? static_cast<int>(sysctl_value/1000000) : -1;
|
|
|
|
#else
|
|
|
|
// TODO(fbarchard): Implement using proc/cpuinfo
|
|
|
|
cpu_speed_ = 0;
|
|
|
|
#endif
|
|
|
|
return cpu_speed_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Dynamically check the current clockrate, which could be reduced because of
|
|
|
|
// powersaving profiles. Eventually for windows we want to query WMI for
|
|
|
|
// root\WMI::ProcessorPerformance.InstanceName="Processor_Number_0".frequency
|
|
|
|
int SystemInfo::GetCurCpuSpeed() {
|
|
|
|
#if defined(WIN32)
|
|
|
|
// TODO(fbarchard): Add WMI check, requires COM initialization
|
|
|
|
// NOTE(fbarchard): Testable on Sandy Bridge.
|
|
|
|
return GetMaxCpuSpeed();
|
|
|
|
#elif defined(IOS) || defined(OSX)
|
|
|
|
uint64_t sysctl_value;
|
|
|
|
size_t length = sizeof(sysctl_value);
|
|
|
|
int error = sysctlbyname("hw.cpufrequency", &sysctl_value, &length, NULL, 0);
|
|
|
|
return !error ? static_cast<int>(sysctl_value/1000000) : GetMaxCpuSpeed();
|
|
|
|
#else // LINUX || ANDROID
|
|
|
|
// TODO(fbarchard): Use proc/cpuinfo for Cur speed on Linux.
|
|
|
|
return GetMaxCpuSpeed();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the amount of installed physical memory in Bytes. Cacheable.
|
|
|
|
// Returns -1 on error.
|
|
|
|
int64 SystemInfo::GetMemorySize() {
|
|
|
|
if (memory_) {
|
|
|
|
return memory_;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(WIN32)
|
|
|
|
MEMORYSTATUSEX status = {0};
|
|
|
|
status.dwLength = sizeof(status);
|
|
|
|
|
|
|
|
if (GlobalMemoryStatusEx(&status)) {
|
|
|
|
memory_ = status.ullTotalPhys;
|
|
|
|
} else {
|
|
|
|
LOG_GLE(LS_WARNING) << "GlobalMemoryStatusEx failed.";
|
|
|
|
memory_ = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif defined(OSX) || defined(IOS)
|
|
|
|
size_t len = sizeof(memory_);
|
|
|
|
int error = sysctlbyname("hw.memsize", &memory_, &len, NULL, 0);
|
|
|
|
if (error || memory_ == 0) {
|
|
|
|
memory_ = -1;
|
|
|
|
}
|
|
|
|
#else // LINUX || ANDROID
|
|
|
|
memory_ = static_cast<int64>(sysconf(_SC_PHYS_PAGES)) *
|
|
|
|
static_cast<int64>(sysconf(_SC_PAGESIZE));
|
|
|
|
if (memory_ < 0) {
|
|
|
|
LOG(LS_WARNING) << "sysconf(_SC_PHYS_PAGES) failed."
|
|
|
|
<< "sysconf(_SC_PHYS_PAGES) " << sysconf(_SC_PHYS_PAGES)
|
|
|
|
<< "sysconf(_SC_PAGESIZE) " << sysconf(_SC_PAGESIZE);
|
|
|
|
memory_ = -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return memory_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Return the name of the machine model we are currently running on.
|
|
|
|
// This is a human readable string that consists of the name and version
|
|
|
|
// number of the hardware, i.e 'MacBookAir1,1'. Returns an empty string if
|
|
|
|
// model can not be determined. The string is cached for subsequent calls.
|
|
|
|
std::string SystemInfo::GetMachineModel() {
|
|
|
|
if (!machine_model_.empty()) {
|
|
|
|
return machine_model_;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(OSX) || defined(IOS)
|
|
|
|
char buffer[128];
|
|
|
|
size_t length = sizeof(buffer);
|
|
|
|
int error = sysctlbyname("hw.model", buffer, &length, NULL, 0);
|
|
|
|
if (!error) {
|
|
|
|
machine_model_.assign(buffer, length - 1);
|
|
|
|
} else {
|
|
|
|
machine_model_.clear();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
machine_model_ = "Not available";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return machine_model_;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef OSX
|
|
|
|
// Helper functions to query IOKit for video hardware properties.
|
|
|
|
static CFTypeRef SearchForProperty(io_service_t port, CFStringRef name) {
|
|
|
|
return IORegistryEntrySearchCFProperty(port, kIOServicePlane,
|
|
|
|
name, kCFAllocatorDefault,
|
|
|
|
kIORegistryIterateRecursively | kIORegistryIterateParents);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GetProperty(io_service_t port, CFStringRef name, int* value) {
|
|
|
|
if (!value) return;
|
|
|
|
CFTypeRef ref = SearchForProperty(port, name);
|
|
|
|
if (ref) {
|
|
|
|
CFTypeID refType = CFGetTypeID(ref);
|
|
|
|
if (CFNumberGetTypeID() == refType) {
|
|
|
|
CFNumberRef number = reinterpret_cast<CFNumberRef>(ref);
|
|
|
|
p_convertCFNumberToInt(number, value);
|
|
|
|
} else if (CFDataGetTypeID() == refType) {
|
|
|
|
CFDataRef data = reinterpret_cast<CFDataRef>(ref);
|
|
|
|
if (CFDataGetLength(data) == sizeof(UInt32)) {
|
|
|
|
*value = *reinterpret_cast<const UInt32*>(CFDataGetBytePtr(data));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CFRelease(ref);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GetProperty(io_service_t port, CFStringRef name,
|
|
|
|
std::string* value) {
|
|
|
|
if (!value) return;
|
|
|
|
CFTypeRef ref = SearchForProperty(port, name);
|
|
|
|
if (ref) {
|
|
|
|
CFTypeID refType = CFGetTypeID(ref);
|
|
|
|
if (CFStringGetTypeID() == refType) {
|
|
|
|
CFStringRef stringRef = reinterpret_cast<CFStringRef>(ref);
|
|
|
|
p_convertHostCFStringRefToCPPString(stringRef, *value);
|
|
|
|
} else if (CFDataGetTypeID() == refType) {
|
|
|
|
CFDataRef dataRef = reinterpret_cast<CFDataRef>(ref);
|
|
|
|
*value = std::string(reinterpret_cast<const char*>(
|
|
|
|
CFDataGetBytePtr(dataRef)), CFDataGetLength(dataRef));
|
|
|
|
}
|
|
|
|
CFRelease(ref);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Fills a struct with information on the graphics adapater and returns true
|
|
|
|
// iff successful.
|
|
|
|
bool SystemInfo::GetGpuInfo(GpuInfo *info) {
|
|
|
|
if (!info) return false;
|
|
|
|
#if defined(WIN32) && !defined(EXCLUDE_D3D9)
|
|
|
|
D3DADAPTER_IDENTIFIER9 identifier;
|
|
|
|
HRESULT hr = E_FAIL;
|
|
|
|
HINSTANCE d3d_lib = LoadLibrary(L"d3d9.dll");
|
|
|
|
|
|
|
|
if (d3d_lib) {
|
|
|
|
typedef IDirect3D9* (WINAPI *D3DCreate9Proc)(UINT);
|
|
|
|
D3DCreate9Proc d3d_create_proc = reinterpret_cast<D3DCreate9Proc>(
|
|
|
|
GetProcAddress(d3d_lib, "Direct3DCreate9"));
|
|
|
|
if (d3d_create_proc) {
|
|
|
|
IDirect3D9* d3d = d3d_create_proc(D3D_SDK_VERSION);
|
|
|
|
if (d3d) {
|
|
|
|
hr = d3d->GetAdapterIdentifier(D3DADAPTER_DEFAULT, 0, &identifier);
|
|
|
|
d3d->Release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FreeLibrary(d3d_lib);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hr != D3D_OK) {
|
|
|
|
LOG(LS_ERROR) << "Failed to access Direct3D9 information.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->device_name = identifier.DeviceName;
|
|
|
|
info->description = identifier.Description;
|
|
|
|
info->vendor_id = identifier.VendorId;
|
|
|
|
info->device_id = identifier.DeviceId;
|
|
|
|
info->driver = identifier.Driver;
|
|
|
|
// driver_version format: product.version.subversion.build
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << HIWORD(identifier.DriverVersion.HighPart) << "."
|
|
|
|
<< LOWORD(identifier.DriverVersion.HighPart) << "."
|
|
|
|
<< HIWORD(identifier.DriverVersion.LowPart) << "."
|
|
|
|
<< LOWORD(identifier.DriverVersion.LowPart);
|
|
|
|
info->driver_version = ss.str();
|
|
|
|
return true;
|
|
|
|
#elif defined(OSX)
|
|
|
|
// We'll query the IOKit for the gpu of the main display.
|
|
|
|
io_service_t display_service_port = CGDisplayIOServicePort(
|
|
|
|
kCGDirectMainDisplay);
|
|
|
|
GetProperty(display_service_port, CFSTR("vendor-id"), &info->vendor_id);
|
|
|
|
GetProperty(display_service_port, CFSTR("device-id"), &info->device_id);
|
|
|
|
GetProperty(display_service_port, CFSTR("model"), &info->description);
|
|
|
|
return true;
|
|
|
|
#else // LINUX || ANDROID
|
|
|
|
// TODO(fbarchard): Implement this on Linux
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
} // namespace talk_base
|