/* * Copyright (C) 2019 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 "src/android_internal/power_stats_hal.h" #include <string.h> #include <algorithm> #include <android/hardware/power/stats/1.0/IPowerStats.h> namespace perfetto { namespace android_internal { using android::hardware::hidl_vec; using android::hardware::Return; using android::hardware::power::stats::V1_0::EnergyData; using android::hardware::power::stats::V1_0::IPowerStats; using android::hardware::power::stats::V1_0::RailInfo; using android::hardware::power::stats::V1_0::Status; namespace { android::sp<IPowerStats> g_svc; bool GetService() { if (!g_svc) g_svc = IPowerStats::getService(); return g_svc != nullptr; } } // namespace bool GetAvailableRails(RailDescriptor* rail_descriptors, size_t* size_of_arr) { const size_t in_array_size = *size_of_arr; *size_of_arr = 0; if (!GetService()) return false; Status status; auto rails_cb = [rail_descriptors, size_of_arr, &in_array_size, &status]( hidl_vec<RailInfo> r, Status s) { status = s; if (status == Status::SUCCESS) { *size_of_arr = std::min(in_array_size, r.size()); for (int i = 0; i < *size_of_arr; ++i) { const RailInfo& rail_info = r[i]; RailDescriptor& descriptor = rail_descriptors[i]; descriptor.index = rail_info.index; descriptor.sampling_rate = rail_info.samplingRate; strncpy(descriptor.rail_name, rail_info.railName.c_str(), sizeof(descriptor.rail_name)); strncpy(descriptor.subsys_name, rail_info.subsysName.c_str(), sizeof(descriptor.subsys_name)); descriptor.rail_name[sizeof(descriptor.rail_name) - 1] = '\0'; descriptor.subsys_name[sizeof(descriptor.subsys_name) - 1] = '\0'; } } }; Return<void> ret = g_svc->getRailInfo(rails_cb); return status == Status::SUCCESS; } bool GetRailEnergyData(RailEnergyData* rail_energy_array, size_t* size_of_arr) { const size_t in_array_size = *size_of_arr; *size_of_arr = 0; if (!GetService()) return false; Status status; auto energy_cb = [rail_energy_array, size_of_arr, &in_array_size, &status]( hidl_vec<EnergyData> m, Status s) { status = s; if (status == Status::SUCCESS) { *size_of_arr = std::min(in_array_size, m.size()); for (int i = 0; i < *size_of_arr; ++i) { const EnergyData& measurement = m[i]; RailEnergyData& element = rail_energy_array[i]; element.index = measurement.index; element.timestamp = measurement.timestamp; element.energy = measurement.energy; } } }; Return<void> ret = g_svc->getEnergyData(hidl_vec<uint32_t>(), energy_cb); return status == Status::SUCCESS; } } // namespace android_internal } // namespace perfetto