/* * Copyright 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "Swappy" #include "CpuInfo.h" #include <limits> #include <bitset> #include <cstdlib> #include <cstring> #include "Log.h" namespace { bool startsWith(std::string &mainStr, const char *toMatch) { // std::string::find returns 0 if toMatch is found at beginning return mainStr.find(toMatch) == 0; } std::vector<std::string> split(const std::string& s, char c) { std::vector<std::string> v; std::string::size_type i = 0; std::string::size_type j = s.find(c); while (j != std::string::npos) { v.push_back(s.substr(i, j-i)); i = ++j; j = s.find(c, j); if (j == std::string::npos) { v.push_back(s.substr(i, s.length())); } } return v; } std::string ReadFile(const std::string& path) { char buf[10240]; FILE *fp = fopen(path.c_str(), "r"); if (fp == nullptr) return std::string(); fgets(buf, 10240, fp); fclose(fp); return std::string(buf); } } // anonymous namespace namespace swappy { std::string to_string(int n) { constexpr int kBufSize = 12; // strlen("−2147483648")+1 static char buf[kBufSize]; snprintf(buf, kBufSize, "%d", n); return buf; } CpuInfo::CpuInfo() { const auto BUFFER_LENGTH = 10240; char buf[BUFFER_LENGTH]; FILE *fp = fopen("/proc/cpuinfo", "r"); if (!fp) { return; } long mMaxFrequency = 0; long mMinFrequency = std::numeric_limits<long>::max(); while (fgets(buf, BUFFER_LENGTH, fp) != NULL) { buf[strlen(buf) - 1] = '\0'; // eat the newline fgets() stores std::string line = buf; if (startsWith(line, "processor")) { Cpu core; core.id = mCpus.size(); auto core_path = std::string("/sys/devices/system/cpu/cpu") + to_string(core.id); auto package_id = ReadFile(core_path + "/topology/physical_package_id"); auto frequency = ReadFile(core_path + "/cpufreq/cpuinfo_max_freq"); core.package_id = atol(package_id.c_str()); core.frequency = atol(frequency.c_str()); mMinFrequency = std::min(mMinFrequency, core.frequency); mMaxFrequency = std::max(mMaxFrequency, core.frequency); mCpus.push_back(core); } else if (startsWith(line, "Hardware")) { mHardware = split(line, ':')[1]; } } fclose(fp); CPU_ZERO(&mLittleCoresMask); CPU_ZERO(&mBigCoresMask); for (auto cpu : mCpus) { if (cpu.frequency == mMinFrequency) { ++mNumberOfLittleCores; cpu.type = Cpu::Type::Little; CPU_SET(cpu.id, &mLittleCoresMask); } else { ++mNumberOfBigCores; cpu.type = Cpu::Type::Big; CPU_SET(cpu.id, &mBigCoresMask); } } } unsigned int CpuInfo::getNumberOfCpus() const { return mCpus.size(); } const std::vector<CpuInfo::Cpu>& CpuInfo::getCpus() const { return mCpus; } const std::string CpuInfo::getHardware() const { return mHardware; } unsigned int CpuInfo::getNumberOfLittleCores() const { return mNumberOfLittleCores; } unsigned int CpuInfo::getNumberOfBigCores() const { return mNumberOfBigCores; } cpu_set_t CpuInfo::getLittleCoresMask() const { return mLittleCoresMask; } cpu_set_t CpuInfo::getBigCoresMask() const { return mBigCoresMask; } unsigned int to_mask(cpu_set_t cpu_set) { std::bitset<32> mask; for (int i = 0; i < CPU_SETSIZE; ++i) { if (CPU_ISSET(i, &cpu_set)) mask[i] = 1; } return (int) mask.to_ulong(); } } // namespace swappy