mirror of
https://github.com/Evolution-X/hardware_interfaces
synced 2026-02-01 11:36:00 +00:00
Data providers that provided data for multiple entities were being added to the data structure as nullptr due to move semantics. Now they will only be added once (ensuring no more nullptr entries) and an index will map each power entity id to its corresponding data provider. Bug: 184290936 Test: Presubmit Change-Id: I858269beb36ba5f87bb14a228079f3abd6c2332f
180 lines
6.1 KiB
C++
180 lines
6.1 KiB
C++
/*
|
|
* Copyright (C) 2020 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 "PowerStats.h"
|
|
|
|
#include <android-base/logging.h>
|
|
|
|
#include <numeric>
|
|
|
|
namespace aidl {
|
|
namespace android {
|
|
namespace hardware {
|
|
namespace power {
|
|
namespace stats {
|
|
|
|
void PowerStats::addStateResidencyDataProvider(std::unique_ptr<IStateResidencyDataProvider> p) {
|
|
if (!p) {
|
|
return;
|
|
}
|
|
|
|
int32_t id = mPowerEntityInfos.size();
|
|
auto info = p->getInfo();
|
|
|
|
size_t index = mStateResidencyDataProviders.size();
|
|
mStateResidencyDataProviders.emplace_back(std::move(p));
|
|
|
|
for (const auto& [entityName, states] : info) {
|
|
PowerEntity i = {
|
|
.id = id++,
|
|
.name = entityName,
|
|
.states = states,
|
|
};
|
|
mPowerEntityInfos.emplace_back(i);
|
|
mStateResidencyDataProviderIndex.emplace_back(index);
|
|
}
|
|
}
|
|
|
|
void PowerStats::addEnergyConsumer(std::unique_ptr<IEnergyConsumer> p) {
|
|
if (!p) {
|
|
return;
|
|
}
|
|
|
|
EnergyConsumerType type = p->getType();
|
|
std::string name = p->getName();
|
|
int32_t count = count_if(mEnergyConsumerInfos.begin(), mEnergyConsumerInfos.end(),
|
|
[&type](const EnergyConsumer& c) { return type == c.type; });
|
|
int32_t id = mEnergyConsumers.size();
|
|
mEnergyConsumerInfos.emplace_back(
|
|
EnergyConsumer{.id = id, .ordinal = count, .type = type, .name = name});
|
|
mEnergyConsumers.emplace_back(std::move(p));
|
|
}
|
|
|
|
void PowerStats::setEnergyMeter(std::unique_ptr<IEnergyMeter> p) {
|
|
mEnergyMeter = std::move(p);
|
|
}
|
|
|
|
ndk::ScopedAStatus PowerStats::getPowerEntityInfo(std::vector<PowerEntity>* _aidl_return) {
|
|
*_aidl_return = mPowerEntityInfos;
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus PowerStats::getStateResidency(const std::vector<int32_t>& in_powerEntityIds,
|
|
std::vector<StateResidencyResult>* _aidl_return) {
|
|
if (mPowerEntityInfos.empty()) {
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
// If in_powerEntityIds is empty then return data for all supported entities
|
|
if (in_powerEntityIds.empty()) {
|
|
std::vector<int32_t> v(mPowerEntityInfos.size());
|
|
std::iota(std::begin(v), std::end(v), 0);
|
|
return getStateResidency(v, _aidl_return);
|
|
}
|
|
|
|
std::unordered_map<std::string, std::vector<StateResidency>> stateResidencies;
|
|
|
|
for (const int32_t id : in_powerEntityIds) {
|
|
// check for invalid ids
|
|
if (id < 0 || id >= mPowerEntityInfos.size()) {
|
|
return ndk::ScopedAStatus(AStatus_fromExceptionCode(EX_ILLEGAL_ARGUMENT));
|
|
}
|
|
|
|
// Check to see if we already have data for the given id
|
|
std::string powerEntityName = mPowerEntityInfos[id].name;
|
|
if (stateResidencies.find(powerEntityName) == stateResidencies.end()) {
|
|
mStateResidencyDataProviders.at(mStateResidencyDataProviderIndex.at(id))
|
|
->getStateResidencies(&stateResidencies);
|
|
}
|
|
|
|
// Append results if we have them
|
|
auto stateResidency = stateResidencies.find(powerEntityName);
|
|
if (stateResidency != stateResidencies.end()) {
|
|
StateResidencyResult res = {
|
|
.id = id,
|
|
.stateResidencyData = stateResidency->second,
|
|
};
|
|
_aidl_return->emplace_back(res);
|
|
} else {
|
|
// Failed to get results for the given id.
|
|
LOG(ERROR) << "Failed to get results for " << powerEntityName;
|
|
}
|
|
}
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus PowerStats::getEnergyConsumerInfo(std::vector<EnergyConsumer>* _aidl_return) {
|
|
*_aidl_return = mEnergyConsumerInfos;
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus PowerStats::getEnergyConsumed(const std::vector<int32_t>& in_energyConsumerIds,
|
|
std::vector<EnergyConsumerResult>* _aidl_return) {
|
|
if (mEnergyConsumers.empty()) {
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
// If in_powerEntityIds is empty then return data for all supported energy consumers
|
|
if (in_energyConsumerIds.empty()) {
|
|
std::vector<int32_t> v(mEnergyConsumerInfos.size());
|
|
std::iota(std::begin(v), std::end(v), 0);
|
|
return getEnergyConsumed(v, _aidl_return);
|
|
}
|
|
|
|
for (const auto id : in_energyConsumerIds) {
|
|
// check for invalid ids
|
|
if (id < 0 || id >= mEnergyConsumers.size()) {
|
|
return ndk::ScopedAStatus(AStatus_fromExceptionCode(EX_ILLEGAL_ARGUMENT));
|
|
}
|
|
|
|
auto optionalResult = mEnergyConsumers[id]->getEnergyConsumed();
|
|
if (optionalResult) {
|
|
EnergyConsumerResult result = optionalResult.value();
|
|
result.id = id;
|
|
_aidl_return->emplace_back(result);
|
|
} else {
|
|
// Failed to get results for the given id.
|
|
LOG(ERROR) << "Failed to get results for " << mEnergyConsumerInfos[id].name;
|
|
}
|
|
}
|
|
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
ndk::ScopedAStatus PowerStats::getEnergyMeterInfo(std::vector<Channel>* _aidl_return) {
|
|
if (!mEnergyMeter) {
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
return mEnergyMeter->getEnergyMeterInfo(_aidl_return);
|
|
}
|
|
|
|
ndk::ScopedAStatus PowerStats::readEnergyMeter(const std::vector<int32_t>& in_channelIds,
|
|
std::vector<EnergyMeasurement>* _aidl_return) {
|
|
if (!mEnergyMeter) {
|
|
return ndk::ScopedAStatus::ok();
|
|
}
|
|
|
|
return mEnergyMeter->readEnergyMeter(in_channelIds, _aidl_return);
|
|
}
|
|
|
|
} // namespace stats
|
|
} // namespace power
|
|
} // namespace hardware
|
|
} // namespace android
|
|
} // namespace aidl
|