/*
* 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