| /* |
| * Copyright (C) 2020 The Android Open Source Project |
| * |
| * SPDX-License-Identifier: Apache-2.0 |
| */ |
| |
| #define LOG_TAG "powerhal-libperfmgr" |
| |
| #include "Power.h" |
| |
| #include <android-base/file.h> |
| #include <android-base/logging.h> |
| #include <android-base/properties.h> |
| #include <android-base/stringprintf.h> |
| #include <android-base/strings.h> |
| #include <perfmgr/HintManager.h> |
| #include <utils/Log.h> |
| |
| #include <mutex> |
| |
| #include "PowerHintSession.h" |
| #include "PowerSessionManager.h" |
| |
| namespace aidl { |
| namespace google { |
| namespace hardware { |
| namespace power { |
| namespace impl { |
| namespace pixel { |
| |
| using ::aidl::google::hardware::power::impl::pixel::PowerHintSession; |
| using ::android::perfmgr::HintManager; |
| |
| constexpr char kPowerHalStateProp[] = "vendor.powerhal.state"; |
| constexpr char kPowerHalAudioProp[] = "vendor.powerhal.audio"; |
| constexpr char kPowerHalRenderingProp[] = "vendor.powerhal.rendering"; |
| |
| static const std::vector<Mode> kAlwaysAllowedModes = { |
| Mode::DOUBLE_TAP_TO_WAKE, |
| Mode::INTERACTIVE, |
| Mode::DEVICE_IDLE, |
| Mode::DISPLAY_INACTIVE, |
| }; |
| |
| Power::Power() |
| : mInteractionHandler(nullptr), |
| mSustainedPerfModeOn(false), |
| mBatterySaverOn(false) { |
| mInteractionHandler = std::make_unique<InteractionHandler>(); |
| mInteractionHandler->Init(); |
| |
| std::string state = ::android::base::GetProperty(kPowerHalStateProp, ""); |
| if (state == "SUSTAINED_PERFORMANCE") { |
| LOG(INFO) << "Initialize with SUSTAINED_PERFORMANCE on"; |
| HintManager::GetInstance()->DoHint("SUSTAINED_PERFORMANCE"); |
| mSustainedPerfModeOn = true; |
| } else { |
| LOG(INFO) << "Initialize PowerHAL"; |
| } |
| |
| state = ::android::base::GetProperty(kPowerHalAudioProp, ""); |
| if (state == "AUDIO_STREAMING_LOW_LATENCY") { |
| LOG(INFO) << "Initialize with AUDIO_LOW_LATENCY on"; |
| HintManager::GetInstance()->DoHint(state); |
| } |
| |
| state = ::android::base::GetProperty(kPowerHalRenderingProp, ""); |
| if (state == "EXPENSIVE_RENDERING") { |
| LOG(INFO) << "Initialize with EXPENSIVE_RENDERING on"; |
| HintManager::GetInstance()->DoHint("EXPENSIVE_RENDERING"); |
| } |
| } |
| |
| static void endAllHints() { |
| std::shared_ptr<HintManager> hm = HintManager::GetInstance(); |
| for (auto hint : hm->GetHints()) { |
| if (std::any_of(kAlwaysAllowedModes.begin(), kAlwaysAllowedModes.end(), |
| [hint](auto mode) { return hint == toString(mode); })) { |
| continue; |
| } |
| hm->EndHint(hint); |
| } |
| } |
| |
| ndk::ScopedAStatus Power::setMode(Mode type, bool enabled) { |
| LOG(DEBUG) << "Power setMode: " << toString(type) << " to: " << enabled; |
| if (HintManager::GetInstance()->GetAdpfProfile() && |
| HintManager::GetInstance()->GetAdpfProfile()->mReportingRateLimitNs > 0) { |
| PowerSessionManager::getInstance()->updateHintMode(toString(type), enabled); |
| } |
| switch (type) { |
| case Mode::SUSTAINED_PERFORMANCE: |
| if (enabled) { |
| endAllHints(); |
| HintManager::GetInstance()->DoHint("SUSTAINED_PERFORMANCE"); |
| } else { |
| HintManager::GetInstance()->EndHint("SUSTAINED_PERFORMANCE"); |
| } |
| mSustainedPerfModeOn = enabled; |
| break; |
| case Mode::LOW_POWER: |
| if (enabled) { |
| endAllHints(); |
| HintManager::GetInstance()->DoHint("LOW_POWER"); |
| } else { |
| HintManager::GetInstance()->EndHint("LOW_POWER"); |
| } |
| mBatterySaverOn = enabled; |
| break; |
| case Mode::LAUNCH: |
| [[fallthrough]]; |
| case Mode::DOUBLE_TAP_TO_WAKE: |
| [[fallthrough]]; |
| case Mode::FIXED_PERFORMANCE: |
| [[fallthrough]]; |
| case Mode::EXPENSIVE_RENDERING: |
| [[fallthrough]]; |
| case Mode::INTERACTIVE: |
| [[fallthrough]]; |
| case Mode::DEVICE_IDLE: |
| [[fallthrough]]; |
| case Mode::DISPLAY_INACTIVE: |
| [[fallthrough]]; |
| case Mode::AUDIO_STREAMING_LOW_LATENCY: |
| [[fallthrough]]; |
| default: |
| if ((mBatterySaverOn || mSustainedPerfModeOn) && std::find(kAlwaysAllowedModes.begin(), |
| kAlwaysAllowedModes.end(), type) == kAlwaysAllowedModes.end()) { |
| break; |
| } |
| if (enabled) { |
| HintManager::GetInstance()->DoHint(toString(type)); |
| } else { |
| HintManager::GetInstance()->EndHint(toString(type)); |
| } |
| break; |
| } |
| |
| return ndk::ScopedAStatus::ok(); |
| } |
| |
| ndk::ScopedAStatus Power::isModeSupported(Mode type, bool *_aidl_return) { |
| bool supported = type == Mode::LOW_POWER |
| || HintManager::GetInstance()->IsHintSupported(toString(type)); |
| LOG(INFO) << "Power mode " << toString(type) << " isModeSupported: " << supported; |
| *_aidl_return = supported; |
| return ndk::ScopedAStatus::ok(); |
| } |
| |
| ndk::ScopedAStatus Power::setBoost(Boost type, int32_t durationMs) { |
| LOG(DEBUG) << "Power setBoost: " << toString(type) << " duration: " << durationMs; |
| if (HintManager::GetInstance()->GetAdpfProfile() && |
| HintManager::GetInstance()->GetAdpfProfile()->mReportingRateLimitNs > 0) { |
| PowerSessionManager::getInstance()->updateHintBoost(toString(type), durationMs); |
| } |
| switch (type) { |
| case Boost::INTERACTION: |
| if (mSustainedPerfModeOn || mBatterySaverOn) { |
| break; |
| } |
| mInteractionHandler->Acquire(durationMs); |
| break; |
| case Boost::DISPLAY_UPDATE_IMMINENT: |
| [[fallthrough]]; |
| case Boost::ML_ACC: |
| [[fallthrough]]; |
| case Boost::AUDIO_LAUNCH: |
| [[fallthrough]]; |
| default: |
| if (mSustainedPerfModeOn || mBatterySaverOn) { |
| break; |
| } |
| if (durationMs > 0) { |
| HintManager::GetInstance()->DoHint(toString(type), |
| std::chrono::milliseconds(durationMs)); |
| } else if (durationMs == 0) { |
| HintManager::GetInstance()->DoHint(toString(type)); |
| } else { |
| HintManager::GetInstance()->EndHint(toString(type)); |
| } |
| break; |
| } |
| |
| return ndk::ScopedAStatus::ok(); |
| } |
| |
| ndk::ScopedAStatus Power::isBoostSupported(Boost type, bool *_aidl_return) { |
| bool supported = HintManager::GetInstance()->IsHintSupported(toString(type)); |
| LOG(INFO) << "Power boost " << toString(type) << " isBoostSupported: " << supported; |
| *_aidl_return = supported; |
| return ndk::ScopedAStatus::ok(); |
| } |
| |
| constexpr const char *boolToString(bool b) { |
| return b ? "true" : "false"; |
| } |
| |
| binder_status_t Power::dump(int fd, const char **, uint32_t) { |
| std::string buf(::android::base::StringPrintf( |
| "HintManager Running: %s\n" |
| "SustainedPerformanceMode: %s\n" |
| "BatterySaverMode: %s\n", |
| boolToString(HintManager::GetInstance()->IsRunning()), |
| boolToString(mSustainedPerfModeOn), |
| boolToString(mBatterySaverOn))); |
| // Dump nodes through libperfmgr |
| HintManager::GetInstance()->DumpToFd(fd); |
| PowerSessionManager::getInstance()->dumpToFd(fd); |
| if (!::android::base::WriteStringToFd(buf, fd)) { |
| PLOG(ERROR) << "Failed to dump state to fd"; |
| } |
| fsync(fd); |
| return STATUS_OK; |
| } |
| |
| ndk::ScopedAStatus Power::createHintSession(int32_t tgid, int32_t uid, |
| const std::vector<int32_t> &threadIds, |
| int64_t durationNanos, |
| std::shared_ptr<IPowerHintSession> *_aidl_return) { |
| if (!HintManager::GetInstance()->GetAdpfProfile() || |
| HintManager::GetInstance()->GetAdpfProfile()->mReportingRateLimitNs <= 0) { |
| *_aidl_return = nullptr; |
| return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION); |
| } |
| if (threadIds.size() == 0) { |
| LOG(ERROR) << "Error: threadIds.size() shouldn't be " << threadIds.size(); |
| *_aidl_return = nullptr; |
| return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT); |
| } |
| std::shared_ptr<IPowerHintSession> session = ndk::SharedRefBase::make<PowerHintSession>( |
| tgid, uid, threadIds, durationNanos); |
| *_aidl_return = session; |
| return ndk::ScopedAStatus::ok(); |
| } |
| |
| ndk::ScopedAStatus Power::getHintSessionPreferredRate(int64_t *outNanoseconds) { |
| *outNanoseconds = HintManager::GetInstance()->GetAdpfProfile() |
| ? HintManager::GetInstance()->GetAdpfProfile()->mReportingRateLimitNs |
| : 0; |
| if (*outNanoseconds <= 0) { |
| return ndk::ScopedAStatus::fromExceptionCode(EX_UNSUPPORTED_OPERATION); |
| } |
| |
| return ndk::ScopedAStatus::ok(); |
| } |
| |
| } // namespace pixel |
| } // namespace impl |
| } // namespace power |
| } // namespace hardware |
| } // namespace google |
| } // namespace aidl |