/* * 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. */ #define LOG_TAG "powerhal-libperfmgr" #include "Power.h" #include #include #include #include #include #include #include #include #include #include #include #include "AdpfTypes.h" #include "PowerHintSession.h" #include "PowerSessionManager.h" #include "disp-power/DisplayLowPower.h" namespace aidl { namespace google { namespace hardware { namespace power { namespace impl { namespace pixel { using ::android::perfmgr::HintManager; constexpr char kPowerHalStateProp[] = "vendor.powerhal.state"; constexpr char kPowerHalAudioProp[] = "vendor.powerhal.audio"; constexpr char kPowerHalRenderingProp[] = "vendor.powerhal.rendering"; Power::Power(std::shared_ptr dlpw) : mDisplayLowPower(dlpw), mInteractionHandler(nullptr), mVRModeOn(false), mSustainedPerfModeOn(false) { mInteractionHandler = std::make_unique(); 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 if (state == "VR") { LOG(INFO) << "Initialize with VR on"; HintManager::GetInstance()->DoHint(state); mVRModeOn = true; } else if (state == "VR_SUSTAINED_PERFORMANCE") { LOG(INFO) << "Initialize with SUSTAINED_PERFORMANCE and VR on"; HintManager::GetInstance()->DoHint("VR_SUSTAINED_PERFORMANCE"); mSustainedPerfModeOn = true; mVRModeOn = 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"); } auto status = this->getInterfaceVersion(&mServiceVersion); LOG(INFO) << "PowerHAL InterfaceVersion:" << mServiceVersion << " isOK: " << status.isOk(); } 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::LOW_POWER: mDisplayLowPower->SetDisplayLowPower(enabled); if (enabled) { HintManager::GetInstance()->DoHint(toString(type)); } else { HintManager::GetInstance()->EndHint(toString(type)); } break; case Mode::SUSTAINED_PERFORMANCE: if (enabled && !mSustainedPerfModeOn) { if (!mVRModeOn) { // Sustained mode only. HintManager::GetInstance()->DoHint("SUSTAINED_PERFORMANCE"); } else { // Sustained + VR mode. HintManager::GetInstance()->EndHint("VR"); HintManager::GetInstance()->DoHint("VR_SUSTAINED_PERFORMANCE"); } mSustainedPerfModeOn = true; } else if (!enabled && mSustainedPerfModeOn) { HintManager::GetInstance()->EndHint("VR_SUSTAINED_PERFORMANCE"); HintManager::GetInstance()->EndHint("SUSTAINED_PERFORMANCE"); if (mVRModeOn) { // Switch back to VR Mode. HintManager::GetInstance()->DoHint("VR"); } mSustainedPerfModeOn = false; } break; case Mode::VR: if (enabled && !mVRModeOn) { if (!mSustainedPerfModeOn) { // VR mode only. HintManager::GetInstance()->DoHint("VR"); } else { // Sustained + VR mode. HintManager::GetInstance()->EndHint("SUSTAINED_PERFORMANCE"); HintManager::GetInstance()->DoHint("VR_SUSTAINED_PERFORMANCE"); } mVRModeOn = true; } else if (!enabled && mVRModeOn) { HintManager::GetInstance()->EndHint("VR_SUSTAINED_PERFORMANCE"); HintManager::GetInstance()->EndHint("VR"); if (mSustainedPerfModeOn) { // Switch back to sustained Mode. HintManager::GetInstance()->DoHint("SUSTAINED_PERFORMANCE"); } mVRModeOn = false; } break; case Mode::LAUNCH: if (mVRModeOn || mSustainedPerfModeOn) { break; } [[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]]; case Mode::CAMERA_STREAMING_SECURE: [[fallthrough]]; case Mode::CAMERA_STREAMING_LOW: [[fallthrough]]; case Mode::CAMERA_STREAMING_MID: [[fallthrough]]; case Mode::CAMERA_STREAMING_HIGH: [[fallthrough]]; case Mode::GAME_LOADING: [[fallthrough]]; default: 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) { switch (mServiceVersion) { case 5: if (static_cast(type) <= static_cast(Mode::AUTOMOTIVE_PROJECTION)) break; [[fallthrough]]; case 4: [[fallthrough]]; case 3: if (static_cast(type) <= static_cast(Mode::GAME_LOADING)) break; [[fallthrough]]; case 2: [[fallthrough]]; case 1: if (static_cast(type) <= static_cast(Mode::CAMERA_STREAMING_HIGH)) break; [[fallthrough]]; default: *_aidl_return = false; return ndk::ScopedAStatus::ok(); } bool supported = HintManager::GetInstance()->IsHintSupported(toString(type)); // LOW_POWER handled insides PowerHAL specifically if (type == Mode::LOW_POWER) { supported = true; } if (!supported && HintManager::GetInstance()->IsAdpfProfileSupported(toString(type))) { supported = true; } 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 (mVRModeOn || mSustainedPerfModeOn) { break; } mInteractionHandler->Acquire(durationMs); break; case Boost::DISPLAY_UPDATE_IMMINENT: [[fallthrough]]; case Boost::ML_ACC: [[fallthrough]]; case Boost::AUDIO_LAUNCH: [[fallthrough]]; case Boost::CAMERA_LAUNCH: [[fallthrough]]; case Boost::CAMERA_SHOT: [[fallthrough]]; default: if (mVRModeOn || mSustainedPerfModeOn) { 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) { switch (mServiceVersion) { case 5: [[fallthrough]]; case 4: [[fallthrough]]; case 3: [[fallthrough]]; case 2: [[fallthrough]]; case 1: if (static_cast(type) <= static_cast(Boost::CAMERA_SHOT)) break; [[fallthrough]]; default: *_aidl_return = false; return ndk::ScopedAStatus::ok(); } bool supported = HintManager::GetInstance()->IsHintSupported(toString(type)); if (!supported && HintManager::GetInstance()->IsAdpfProfileSupported(toString(type))) { supported = true; } 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" "VRMode: %s\n" "SustainedPerformanceMode: %s\n", boolToString(HintManager::GetInstance()->IsRunning()), boolToString(mVRModeOn), boolToString(mSustainedPerfModeOn))); // 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 &threadIds, int64_t durationNanos, std::shared_ptr *_aidl_return) { SessionConfig config; return createHintSessionWithConfig(tgid, uid, threadIds, durationNanos, SessionTag::OTHER, &config, _aidl_return); } 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(); } ndk::ScopedAStatus Power::createHintSessionWithConfig( int32_t tgid, int32_t uid, const std::vector &threadIds, int64_t durationNanos, SessionTag tag, SessionConfig *config, std::shared_ptr *_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> session = ndk::SharedRefBase::make>(tgid, uid, threadIds, durationNanos, tag); *_aidl_return = session; session->getSessionConfig(config); PowerSessionManager<>::getInstance()->registerSession(session, config->id); return ndk::ScopedAStatus::ok(); } ndk::ScopedAStatus Power::getSessionChannel(int32_t, int32_t, ChannelConfig *_aidl_return) { static AidlMessageQueue stubQueue{20, true}; static std::thread stubThread([&] { ChannelMessage data; // This loop will only run while there is data waiting // to be processed, and blocks on a futex all other times while (stubQueue.readBlocking(&data, 1, 0)) { } }); _aidl_return->channelDescriptor = stubQueue.dupeDesc(); _aidl_return->readFlagBitmask = 0x01; _aidl_return->writeFlagBitmask = 0x02; _aidl_return->eventFlagDescriptor = std::nullopt; return ndk::ScopedAStatus::ok(); } ndk::ScopedAStatus Power::closeSessionChannel(int32_t, int32_t) { return ndk::ScopedAStatus::ok(); } } // namespace pixel } // namespace impl } // namespace power } // namespace hardware } // namespace google } // namespace aidl