android_device_google_gs-co.../powerstats/DisplayMrrStateResidencyDataProvider.cpp
Darren Hsu 40b75d505c powerstats: introduce display MRR state residency data provider
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>
2023-12-22 10:08:20 +08:00

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