/* * Copyright (C) 2021 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. */ #define LOG_TAG "android.hardware.power.stats-service.pixel" #include #include #include #include #include #include #include #include #include #include #include using aidl::android::hardware::power::stats::DisplayStateResidencyDataProvider; using aidl::android::hardware::power::stats::EnergyConsumerType; using aidl::android::hardware::power::stats::GenericStateResidencyDataProvider; using aidl::android::hardware::power::stats::PowerStatsEnergyConsumer; const char kBootRevision[] = "ro.boot.revision"; std::map displayChannelNames = { {"PROTO1.0", "PPVAR_VSYS_PWR_DISP"}, {"EVT1.0", "PPVAR_VSYS_PWR_DISP"}, {"EVT1.1", "VSYS_PWR_DISPLAY"}, }; void addDisplay(std::shared_ptr p) { // Add display residency stats struct stat buffer; if (!stat("/sys/class/drm/card0/device/primary-panel/time_in_state", &buffer)) { // time_in_state exists addDisplayMrr(p); } else { // time_in_state doesn't exist std::vector states = { "Off", "LP: 1440x3120@10", "LP: 1440x3120@30", "On: 1440x3120@10", "On: 1440x3120@30", "On: 1440x3120@60", "On: 1440x3120@90", "On: 1440x3120@120", "HBM: 1440x3120@60", }; p->addStateResidencyDataProvider(std::make_unique("Display", "/sys/class/backlight/panel0-backlight/state", states)); } std::string rev = android::base::GetProperty(kBootRevision, ""); std::string channelName; if (displayChannelNames.find(rev) == displayChannelNames.end()) { channelName = displayChannelNames["EVT1.1"]; } else { channelName = displayChannelNames[rev]; } // Add display energy consumer /* * TODO(b/167216667): Add correct display power model here. Must read from display rail * and include proper coefficients for display states. */ p->addEnergyConsumer(PowerStatsEnergyConsumer::createMeterAndEntityConsumer(p, EnergyConsumerType::DISPLAY, "display", {channelName}, "Display", {{"LP: 1440x3120@10", 1}, {"LP: 1440x3120@30", 2}, {"On: 1440x3120@10", 3}, {"On: 1440x3120@30", 4}, {"On: 1440x3120@60", 5}, {"On: 1440x3120@90", 6}, {"On: 1440x3120@120", 7}, {"HBM: 1440x3120@60", 8}})); } void addUwb(std::shared_ptr p) { // A constant to represent the number of nanoseconds in one millisecond. const int NS_TO_MS = 1000000; // ACPM stats are reported in nanoseconds. The transform function // converts nanoseconds to milliseconds. std::function uwbNsToMs = [](uint64_t a) { return a / NS_TO_MS; }; const GenericStateResidencyDataProvider::StateResidencyConfig stateConfig = { .entryCountSupported = true, .entryCountPrefix = "count:", .totalTimeSupported = true, .totalTimePrefix = "dur ns:", .totalTimeTransform = uwbNsToMs, .lastEntrySupported = false, }; const std::vector> stateHeaders = { std::make_pair("Off", "Off state:"), std::make_pair("Run", "Run state:"), std::make_pair("Idle", "Idle state:"), std::make_pair("Tx", "Tx state:"), std::make_pair("Rx", "Rx state:"), }; std::vector cfgs; cfgs.emplace_back(generateGenericStateResidencyConfigs(stateConfig, stateHeaders), "UWB", ""); p->addStateResidencyDataProvider(std::make_unique( "/sys/devices/platform/10d30000.spi/spi_master/spi10/spi10.0/uwb/power_stats", cfgs)); } int main() { struct stat buffer; LOG(INFO) << "Pixel PowerStats HAL AIDL Service is starting."; // single thread ABinderProcess_setThreadPoolMaxThreadCount(0); std::shared_ptr p = ndk::SharedRefBase::make(); addGs101CommonDataProviders(p); addDisplay(p); if (!stat("/sys/devices/platform/10960000.hsi2c/i2c-7/7-0008/power_stats", &buffer)) { addNFC(p, "/sys/devices/platform/10960000.hsi2c/i2c-7/7-0008/power_stats"); } const std::string instance = std::string() + PowerStats::descriptor + "/default"; binder_status_t status = AServiceManager_addService(p->asBinder().get(), instance.c_str()); LOG_ALWAYS_FATAL_IF(status != STATUS_OK); ABinderProcess_joinThreadPool(); return EXIT_FAILURE; // should not reach }