blob: 5723904bbb204adbd4c6489a1d972f91331d18c5 [file] [log] [blame]
Benjamin Schwartz6d141702018-11-14 11:56:43 -08001/*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
Benjamin Schwartz7f27af72020-09-08 16:44:02 -070016// TODO(b/167628903): Delete this file
Benjamin Schwartz6d141702018-11-14 11:56:43 -080017#define LOG_TAG "libpixelpowerstats"
18
19#include <android-base/logging.h>
Benjamin Schwartzf1616352019-03-26 15:07:02 -070020#include <android-base/strings.h>
Benjamin Schwartz6d141702018-11-14 11:56:43 -080021#include <pixelpowerstats/GenericStateResidencyDataProvider.h>
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -080022#include <pixelpowerstats/PowerStatsUtils.h>
Sujee Rajayogame78733b2020-04-15 13:58:51 -070023
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -080024#include <cstdio>
25#include <cstring>
Benjamin Schwartzf1616352019-03-26 15:07:02 -070026#include <memory>
27#include <string>
28#include <unordered_map>
29#include <utility>
30#include <vector>
Benjamin Schwartz6d141702018-11-14 11:56:43 -080031
32namespace android {
33namespace hardware {
34namespace google {
35namespace pixel {
36namespace powerstats {
37
Benjamin Schwartza991ef52018-12-04 13:48:52 -080038std::vector<StateResidencyConfig> generateGenericStateResidencyConfigs(
Sujee Rajayogame78733b2020-04-15 13:58:51 -070039 const StateResidencyConfig &stateConfig,
40 const std::vector<std::pair<std::string, std::string>> &stateHeaders) {
Benjamin Schwartza991ef52018-12-04 13:48:52 -080041 std::vector<StateResidencyConfig> stateResidencyConfigs;
42 stateResidencyConfigs.reserve(stateHeaders.size());
43 for (auto h : stateHeaders) {
44 StateResidencyConfig cfg = {stateConfig};
45 cfg.name = h.first;
46 cfg.header = h.second;
47 stateResidencyConfigs.emplace_back(cfg);
48 }
49 return stateResidencyConfigs;
50}
51
52PowerEntityConfig::PowerEntityConfig(const std::vector<StateResidencyConfig> &stateResidencyConfigs)
Benjamin Schwartz6d141702018-11-14 11:56:43 -080053 : PowerEntityConfig("", stateResidencyConfigs) {}
54
Benjamin Schwartza991ef52018-12-04 13:48:52 -080055PowerEntityConfig::PowerEntityConfig(const std::string &header,
56 const std::vector<StateResidencyConfig> &stateResidencyConfigs)
Sujee Rajayogame78733b2020-04-15 13:58:51 -070057 : PowerEntityConfig(0, header, stateResidencyConfigs) {}
58
59PowerEntityConfig::PowerEntityConfig(const uint32_t start_id, const std::string &header,
60 const std::vector<StateResidencyConfig> &stateResidencyConfigs)
Benjamin Schwartz6d141702018-11-14 11:56:43 -080061 : mHeader(header) {
Benjamin Schwartza991ef52018-12-04 13:48:52 -080062 mStateResidencyConfigs.reserve(stateResidencyConfigs.size());
Sujee Rajayogame78733b2020-04-15 13:58:51 -070063 for (uint32_t i = start_id; i < start_id + stateResidencyConfigs.size(); ++i) {
64 mStateResidencyConfigs.emplace_back(i, stateResidencyConfigs[i - start_id]);
Benjamin Schwartz6d141702018-11-14 11:56:43 -080065 }
66}
67
Sujee Rajayogame78733b2020-04-15 13:58:51 -070068static bool parseState(PowerEntityStateResidencyData *data, const StateResidencyConfig &config,
69 FILE *fp, char **line, size_t *len) {
Benjamin Schwartz6d141702018-11-14 11:56:43 -080070 size_t numFieldsRead = 0;
71 const size_t numFields =
Sujee Rajayogame78733b2020-04-15 13:58:51 -070072 config.entryCountSupported + config.totalTimeSupported + config.lastEntrySupported;
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -080073
Sujee Rajayogame78733b2020-04-15 13:58:51 -070074 while ((numFieldsRead < numFields) && (getline(line, len, fp) != -1)) {
Benjamin Schwartz6d141702018-11-14 11:56:43 -080075 uint64_t stat = 0;
76 // Attempt to extract data from the current line
Sujee Rajayogame78733b2020-04-15 13:58:51 -070077 if (config.entryCountSupported &&
78 utils::extractStat(*line, config.entryCountPrefix, stat)) {
79 data->totalStateEntryCount =
80 config.entryCountTransform ? config.entryCountTransform(stat) : stat;
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -080081 ++numFieldsRead;
82 } else if (config.totalTimeSupported &&
Sujee Rajayogame78733b2020-04-15 13:58:51 -070083 utils::extractStat(*line, config.totalTimePrefix, stat)) {
84 data->totalTimeInStateMs =
85 config.totalTimeTransform ? config.totalTimeTransform(stat) : stat;
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -080086 ++numFieldsRead;
87 } else if (config.lastEntrySupported &&
Sujee Rajayogame78733b2020-04-15 13:58:51 -070088 utils::extractStat(*line, config.lastEntryPrefix, stat)) {
89 data->lastEntryTimestampMs =
90 config.lastEntryTransform ? config.lastEntryTransform(stat) : stat;
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -080091 ++numFieldsRead;
Benjamin Schwartz6d141702018-11-14 11:56:43 -080092 }
93 }
94
95 // End of file was reached and not all state data was parsed. Something
96 // went wrong
97 if (numFieldsRead != numFields) {
98 LOG(ERROR) << __func__ << ": failed to parse stats for:" << config.name;
99 return false;
100 }
101
102 return true;
103}
104
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800105template <class T, class Func>
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700106static auto findNext(const std::vector<T> &collection, FILE *fp, char **line, size_t *len,
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800107 Func pred) {
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -0800108 // handling the case when there is no header to look for
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800109 if (pred(collection.front(), "")) {
110 return collection.cbegin();
111 }
112
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700113 while (getline(line, len, fp) != -1) {
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800114 for (auto it = collection.cbegin(); it != collection.cend(); ++it) {
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700115 if (pred(*it, *line)) {
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800116 return it;
117 }
118 }
119 }
120
121 return collection.cend();
122}
123
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -0800124static bool getStateData(
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700125 PowerEntityStateResidencyResult *result,
126 const std::vector<std::pair<uint32_t, StateResidencyConfig>> &stateResidencyConfigs,
127 FILE *fp, char **line, size_t *len) {
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800128 size_t numStatesRead = 0;
129 size_t numStates = stateResidencyConfigs.size();
130 auto nextState = stateResidencyConfigs.cbegin();
131 auto endState = stateResidencyConfigs.cend();
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800132 auto pred = [](auto a, char const *b) {
133 // return true if b matches the header contained in a, ignoring whitespace
Benjamin Schwartzf1616352019-03-26 15:07:02 -0700134 return (a.second.header == android::base::Trim(std::string(b)));
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800135 };
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800136
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700137 result->stateResidencyData.resize(numStates);
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800138
139 // Search for state headers until we have found them all or can't find anymore
140 while ((numStatesRead < numStates) &&
Benjamin Schwartza991ef52018-12-04 13:48:52 -0800141 (nextState = findNext<std::pair<uint32_t, StateResidencyConfig>>(
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700142 stateResidencyConfigs, fp, line, len, pred)) != endState) {
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800143 // Found a matching state header. Parse the contents
144 PowerEntityStateResidencyData data = {.powerEntityStateId = nextState->first};
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700145 if (parseState(&data, nextState->second, fp, line, len)) {
146 result->stateResidencyData[numStatesRead] = data;
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800147 ++numStatesRead;
148 } else {
149 break;
150 }
151 }
152
153 // There was a problem parsing and we failed to get data for all of the states
154 if (numStatesRead != numStates) {
155 return false;
156 }
157
158 return true;
159}
160
161bool GenericStateResidencyDataProvider::getResults(
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700162 std::unordered_map<uint32_t, PowerEntityStateResidencyResult> &results) {
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800163 // Using FILE* instead of std::ifstream for performance reasons (b/122253123)
164 std::unique_ptr<FILE, decltype(&fclose)> fp(fopen(mPath.c_str(), "r"), fclose);
165 if (!fp) {
166 PLOG(ERROR) << __func__ << ":Failed to open file " << mPath
167 << " Error = " << strerror(errno);
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800168 return false;
169 }
170
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800171 size_t len = 0;
172 char *line = nullptr;
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800173 size_t numEntitiesRead = 0;
174 size_t numEntities = mPowerEntityConfigs.size();
175 auto nextConfig = mPowerEntityConfigs.cbegin();
176 auto endConfig = mPowerEntityConfigs.cend();
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800177 auto pred = [](auto a, char const *b) {
178 // return true if b matches the header contained in a, ignoring whitespace
Benjamin Schwartzf1616352019-03-26 15:07:02 -0700179 return (a.second.mHeader == android::base::Trim(std::string(b)));
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800180 };
Sujee Rajayogam70888812020-05-13 12:53:51 -0700181 bool skipFindNext = false;
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800182
183 // Search for entity headers until we have found them all or can't find anymore
184 while ((numEntitiesRead < numEntities) &&
Sujee Rajayogam70888812020-05-13 12:53:51 -0700185 (skipFindNext ||
186 (nextConfig = findNext<decltype(mPowerEntityConfigs)::value_type>(
187 mPowerEntityConfigs, fp.get(), &line, &len, pred)) != endConfig)) {
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -0800188 // Found a matching header. Retrieve its state data
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800189 PowerEntityStateResidencyResult result = {.powerEntityId = nextConfig->first};
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700190 if (getStateData(&result, nextConfig->second.mStateResidencyConfigs, fp.get(), &line,
191 &len)) {
192 // If a power entity already exists, then merge in the
193 // StateResidencyData.
194 if (results.find(nextConfig->first) != results.end()) {
195 uint32_t size = results[nextConfig->first].stateResidencyData.size();
196 results[nextConfig->first].stateResidencyData.resize(
197 size + result.stateResidencyData.size());
198 for (uint32_t i = 0; i < result.stateResidencyData.size(); i++) {
199 results[nextConfig->first].stateResidencyData[size + i] =
200 result.stateResidencyData[i];
201 }
202 } else {
203 results.emplace(nextConfig->first, result);
204 }
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800205 ++numEntitiesRead;
206 } else {
207 break;
208 }
Sujee Rajayogam70888812020-05-13 12:53:51 -0700209
210 // If the header of the next PowerEntityConfig is equal to the
211 // current, don't search for it within the file since we'll be search
212 // for more states.
213 auto currConfig = nextConfig++;
214 if (nextConfig != endConfig && nextConfig->second.mHeader == currConfig->second.mHeader) {
215 skipFindNext = true;
216 } else {
217 skipFindNext = false;
218 }
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800219 }
220
Benjamin Schwartz6c6e7872019-01-02 15:38:12 -0800221 free(line);
222
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800223 // There was a problem gathering state residency data for one or more entities
224 if (numEntitiesRead != numEntities) {
225 LOG(ERROR) << __func__ << ":Failed to get results for " << mPath;
226 return false;
227 }
228
229 return true;
230}
231
Benjamin Schwartza991ef52018-12-04 13:48:52 -0800232void GenericStateResidencyDataProvider::addEntity(uint32_t id, const PowerEntityConfig &config) {
233 mPowerEntityConfigs.emplace_back(id, config);
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800234}
235
236std::vector<PowerEntityStateSpace> GenericStateResidencyDataProvider::getStateSpaces() {
237 std::vector<PowerEntityStateSpace> stateSpaces;
Benjamin Schwartz3b2518d2018-11-28 09:34:02 -0800238 stateSpaces.reserve(mPowerEntityConfigs.size());
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800239 for (auto config : mPowerEntityConfigs) {
240 PowerEntityStateSpace s = {.powerEntityId = config.first};
241 s.states.resize(config.second.mStateResidencyConfigs.size());
242
243 for (uint32_t i = 0; i < config.second.mStateResidencyConfigs.size(); ++i) {
244 s.states[i] = {
Sujee Rajayogame78733b2020-04-15 13:58:51 -0700245 .powerEntityStateId = config.second.mStateResidencyConfigs[i].first,
246 .powerEntityStateName = config.second.mStateResidencyConfigs[i].second.name};
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800247 }
Benjamin Schwartza991ef52018-12-04 13:48:52 -0800248 stateSpaces.emplace_back(s);
Benjamin Schwartz6d141702018-11-14 11:56:43 -0800249 }
250 return stateSpaces;
251}
252
253} // namespace powerstats
254} // namespace pixel
255} // namespace google
256} // namespace hardware
Benjamin Schwartzf1616352019-03-26 15:07:02 -0700257} // namespace android