// Copyright 2011 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/system/sys_info.h" #include #include #include #include #include #include "base/check.h" #include "base/files/file_util.h" #include "base/lazy_instance.h" #include "base/notreached.h" #include "base/numerics/safe_conversions.h" #include "base/process/process_metrics.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "base/strings/stringprintf.h" #include "base/system/sys_info_internal.h" #include "build/build_config.h" namespace { uint64_t AmountOfMemory(int pages_name) { long pages = sysconf(pages_name); long page_size = sysconf(_SC_PAGESIZE); if (pages < 0 || page_size < 0) return 0; return static_cast(pages) * static_cast(page_size); } uint64_t AmountOfPhysicalMemory() { return AmountOfMemory(_SC_PHYS_PAGES); } base::LazyInstance< base::internal::LazySysInfoValue>::Leaky g_lazy_physical_memory = LAZY_INSTANCE_INITIALIZER; } // namespace namespace base { // static uint64_t SysInfo::AmountOfPhysicalMemoryImpl() { return g_lazy_physical_memory.Get().value(); } // static uint64_t SysInfo::AmountOfAvailablePhysicalMemoryImpl() { SystemMemoryInfoKB info; if (!GetSystemMemoryInfo(&info)) return 0; return AmountOfAvailablePhysicalMemory(info); } // static uint64_t SysInfo::AmountOfAvailablePhysicalMemory( const SystemMemoryInfoKB& info) { // See details here: // https://git.kernel.org/cgit/linux/kernel/git/torvalds/linux.git/commit/?id=34e431b0ae398fc54ea69ff85ec700722c9da773 // The fallback logic (when there is no MemAvailable) would be more precise // if we had info about zones watermarks (/proc/zoneinfo). int res_kb = info.available != 0 ? std::max(info.available - info.active_file, 0) : info.free + info.reclaimable + info.inactive_file; return checked_cast(res_kb) * 1024; } // static std::string SysInfo::CPUModelName() { #if BUILDFLAG(IS_CHROMEOS) && defined(ARCH_CPU_ARMEL) const char kCpuModelPrefix[] = "Hardware"; #else const char kCpuModelPrefix[] = "model name"; #endif std::string contents; ReadFileToString(FilePath("/proc/cpuinfo"), &contents); DCHECK(!contents.empty()); if (!contents.empty()) { std::istringstream iss(contents); std::string line; while (std::getline(iss, line)) { if (line.compare(0, strlen(kCpuModelPrefix), kCpuModelPrefix) == 0) { size_t pos = line.find(": "); return line.substr(pos + 2); } } } #if defined(ARCH_CPU_ARMEL) // /proc/cpuinfo does not have a defined ABI and so devices may fall // through without a model name. // For ARM devices use /sys/devices/socX/soc_id // // https://www.kernel.org/doc/Documentation/ABI/testing/sysfs-devices-soc: // On many of ARM based silicon with SMCCC v1.2+ compliant firmware // this will contain the SOC ID appended to the family attribute // to ensure there is no conflict in this namespace across various // vendors. The format is "jep106:XXYY:ZZZZ" where XX is identity // code, YY is continuation code and ZZZZ is the SOC ID. const char kSocIdDirectory[] = "/sys/devices/soc%u"; const char kSocIdFile[] = "/sys/devices/soc%u/soc_id"; const char kJEP106[] = "jep106"; // There can be multiple /sys/bus/soc/devices/socX on a system. // Iterate through until one with jep106:XXYY:ZZZZ is found. for (int soc_instance = 0;; ++soc_instance) { if (!PathExists( FilePath(base::StringPrintf(kSocIdDirectory, soc_instance)))) break; std::string soc_id; ReadFileToString(FilePath(base::StringPrintf(kSocIdFile, soc_instance)), &soc_id); if (soc_id.find(kJEP106) == 0) return soc_id; } #endif return std::string(); } #if !BUILDFLAG(IS_CHROMEOS) && !BUILDFLAG(IS_ANDROID) // static SysInfo::HardwareInfo SysInfo::GetHardwareInfoSync() { static const size_t kMaxStringSize = 100u; HardwareInfo info; std::string data; if (ReadFileToStringWithMaxSize( FilePath("/sys/devices/virtual/dmi/id/sys_vendor"), &data, kMaxStringSize)) { TrimWhitespaceASCII(data, TrimPositions::TRIM_ALL, &info.manufacturer); } if (ReadFileToStringWithMaxSize( FilePath("/sys/devices/virtual/dmi/id/product_name"), &data, kMaxStringSize)) { TrimWhitespaceASCII(data, TrimPositions::TRIM_ALL, &info.model); } DCHECK(IsStringUTF8(info.manufacturer)); DCHECK(IsStringUTF8(info.model)); return info; } #endif } // namespace base