Display team introduces new refresh rate residency in kernel. Current display data provider is polling data from state sysfs. To reduce CPU loading and improve efficiency, we should get rid of polling data provider and create new data provider to read data from time_in_state sysfs in PowerStatsHAL. Bug: 316260832 Test: vts-tradefed run vts -m VtsHalPowerStatsTargetTest Change-Id: I4d9886f13207e41f13defd89ea2c19614918a570 Signed-off-by: Darren Hsu <darrenhsu@google.com>
170 lines
5.6 KiB
C++
170 lines
5.6 KiB
C++
/*
|
|
* Copyright (C) 2023 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.
|
|
*/
|
|
#include "DisplayMrrStateResidencyDataProvider.h"
|
|
|
|
#include <android-base/logging.h>
|
|
#include <android-base/parseint.h>
|
|
#include <android-base/strings.h>
|
|
|
|
using android::base::ParseInt;
|
|
using android::base::ParseUint;
|
|
using android::base::Split;
|
|
using android::base::Trim;
|
|
|
|
static const std::string TIME_IN_STATE = "time_in_state";
|
|
static const std::string AVAILABLE_STATE = "available_disp_stats";
|
|
static const std::vector<std::string> DISP_STATE = { "On", "HBM", "LP", "Off" };
|
|
|
|
namespace aidl {
|
|
namespace android {
|
|
namespace hardware {
|
|
namespace power {
|
|
namespace stats {
|
|
|
|
DisplayMrrStateResidencyDataProvider::DisplayMrrStateResidencyDataProvider(
|
|
const std::string& name, const std::string& path) : mName(name), mPath(path) {
|
|
mConfigs = std::vector<Config>();
|
|
std::string statePath = mPath + AVAILABLE_STATE;
|
|
std::unique_ptr<FILE, decltype(&fclose)> fp(fopen(statePath.c_str(), "r"), fclose);
|
|
if (fp) {
|
|
char *line = nullptr;
|
|
size_t len = 0;
|
|
Config config = { .state = 0, .resX = 0, .resY = 0, .rr = 0 };
|
|
while (getline(&line, &len, fp.get()) != -1) {
|
|
if (parseAvailableState(line, &config)) {
|
|
mConfigs.push_back(config);
|
|
} else {
|
|
PLOG(ERROR) << "Failed to parse display config for [" << std::string(line)
|
|
<< "] from " << statePath;
|
|
mConfigs.clear();
|
|
break;
|
|
}
|
|
}
|
|
free(line);
|
|
} else {
|
|
PLOG(ERROR) << "Failed to open file " << statePath;
|
|
}
|
|
}
|
|
|
|
bool DisplayMrrStateResidencyDataProvider::parseConfig(
|
|
char const *line, Config *config, uint64_t *duration) {
|
|
std::vector<std::string> parts = Split(line, " ");
|
|
|
|
if (duration == nullptr) {
|
|
if (parts.size() != 4) return false;
|
|
} else {
|
|
if (parts.size() != 5) return false;
|
|
|
|
if (!ParseUint(Trim(parts[4]), duration)) return false;
|
|
}
|
|
|
|
if (!ParseInt(Trim(parts[0]), &config->state)) return false;
|
|
if (!ParseInt(Trim(parts[1]), &config->resX)) return false;
|
|
if (!ParseInt(Trim(parts[2]), &config->resY)) return false;
|
|
if (!ParseInt(Trim(parts[3]), &config->rr)) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DisplayMrrStateResidencyDataProvider::parseAvailableState(
|
|
char const *line, Config *config) {
|
|
return parseConfig(line, config, nullptr);
|
|
}
|
|
|
|
bool DisplayMrrStateResidencyDataProvider::parseTimeInState(
|
|
char const *line, Config *config, uint64_t *duration) {
|
|
return parseConfig(line, config, duration);
|
|
}
|
|
|
|
bool DisplayMrrStateResidencyDataProvider::getStateResidencies(
|
|
std::unordered_map<std::string, std::vector<StateResidency>> *residencies) {
|
|
if (mConfigs.empty()) {
|
|
LOG(ERROR) << "Display MRR state list is empty!";
|
|
return false;
|
|
}
|
|
|
|
std::string path = mPath + TIME_IN_STATE;
|
|
std::unique_ptr<FILE, decltype(&fclose)> fp(fopen(path.c_str(), "r"), fclose);
|
|
if (!fp) {
|
|
PLOG(ERROR) << "Failed to open file " << path;
|
|
return false;
|
|
}
|
|
|
|
std::vector<StateResidency> stateResidencies;
|
|
for (int i = 0; i < mConfigs.size(); i++) {
|
|
StateResidency s = {.id = i, .totalTimeInStateMs = 0};
|
|
stateResidencies.push_back(s);
|
|
}
|
|
|
|
char *line = nullptr;
|
|
size_t len = 0;
|
|
Config config = { .state = 0, .resX = 0, .resY = 0, .rr = 0 };
|
|
uint64_t duration;
|
|
std::vector<Config>::const_iterator found;
|
|
while (getline(&line, &len, fp.get()) != -1) {
|
|
if (parseTimeInState(line, &config, &duration)) {
|
|
found = std::find(mConfigs.begin(), mConfigs.end(), config);
|
|
if (found != mConfigs.end()) {
|
|
stateResidencies[found - mConfigs.begin()].totalTimeInStateMs = duration;
|
|
} else {
|
|
LOG(ERROR) << "Failed to find config for [" << std::string(line)
|
|
<< "] in display MRR state list";
|
|
}
|
|
} else {
|
|
LOG(ERROR) << "Failed to parse state and duration from [" << std::string(line) << "]";
|
|
free(line);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
residencies->emplace(mName, stateResidencies);
|
|
|
|
free(line);
|
|
|
|
return true;
|
|
}
|
|
|
|
std::unordered_map<std::string, std::vector<State>> DisplayMrrStateResidencyDataProvider::getInfo()
|
|
{
|
|
int32_t dispId;
|
|
std::string name;
|
|
std::vector<State> states;
|
|
for (int32_t id = 0; id < mConfigs.size(); id++) {
|
|
dispId = mConfigs[id].state;
|
|
if (dispId >= DISP_STATE.size()) {
|
|
LOG(ERROR) << "Display state id " << dispId << " is out of bound";
|
|
return {};
|
|
}
|
|
|
|
name = DISP_STATE[dispId];
|
|
if (dispId != DISP_STATE.size() - 1) {
|
|
name += ": " + std::to_string(mConfigs[id].resX) +
|
|
"x" + std::to_string(mConfigs[id].resY) +
|
|
"@" + std::to_string(mConfigs[id].rr);
|
|
}
|
|
State s = { .id = id, .name = name };
|
|
states.push_back(s);
|
|
}
|
|
|
|
return {{ mName, states }};
|
|
}
|
|
|
|
} // namespace stats
|
|
} // namespace power
|
|
} // namespace hardware
|
|
} // namespace android
|
|
} // namespace aidl
|