| /* |
| * Copyright 2018 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. |
| */ |
| |
| #pragma once |
| |
| #include <atomic> |
| #include <chrono> |
| #include <unordered_set> |
| |
| #include <utils/Mutex.h> |
| |
| #include "../Scheduler/OneShotTimer.h" |
| #include "DisplayIdentification.h" |
| |
| using namespace std::chrono_literals; |
| |
| namespace android { |
| |
| class SurfaceFlinger; |
| |
| namespace Hwc2 { |
| |
| class PowerAdvisor { |
| public: |
| virtual ~PowerAdvisor(); |
| |
| // Initializes resources that cannot be initialized on construction |
| virtual void init() = 0; |
| virtual void onBootFinished() = 0; |
| virtual void setExpensiveRenderingExpected(DisplayId displayId, bool expected) = 0; |
| virtual bool isUsingExpensiveRendering() = 0; |
| virtual void notifyDisplayUpdateImminent() = 0; |
| virtual bool usePowerHintSession() = 0; |
| virtual bool supportsPowerHintSession() = 0; |
| virtual bool isPowerHintSessionRunning() = 0; |
| virtual void setTargetWorkDuration(int64_t targetDurationNanos) = 0; |
| virtual void sendActualWorkDuration(int64_t actualDurationNanos, nsecs_t timestamp) = 0; |
| virtual void enablePowerHint(bool enabled) = 0; |
| virtual bool startPowerHintSession(const std::vector<int32_t>& threadIds) = 0; |
| }; |
| |
| namespace impl { |
| |
| // PowerAdvisor is a wrapper around IPower HAL which takes into account the |
| // full state of the system when sending out power hints to things like the GPU. |
| class PowerAdvisor final : public Hwc2::PowerAdvisor { |
| public: |
| class HalWrapper { |
| public: |
| virtual ~HalWrapper() = default; |
| |
| virtual bool setExpensiveRendering(bool enabled) = 0; |
| virtual bool notifyDisplayUpdateImminent() = 0; |
| virtual bool supportsPowerHintSession() = 0; |
| virtual bool isPowerHintSessionRunning() = 0; |
| virtual void restartPowerHintSession() = 0; |
| virtual void setPowerHintSessionThreadIds(const std::vector<int32_t>& threadIds) = 0; |
| virtual bool startPowerHintSession() = 0; |
| virtual void setTargetWorkDuration(int64_t targetDurationNanos) = 0; |
| virtual void sendActualWorkDuration(int64_t actualDurationNanos, |
| nsecs_t timeStampNanos) = 0; |
| virtual bool shouldReconnectHAL() = 0; |
| virtual std::vector<int32_t> getPowerHintSessionThreadIds() = 0; |
| virtual std::optional<int64_t> getTargetWorkDuration() = 0; |
| }; |
| |
| PowerAdvisor(SurfaceFlinger& flinger); |
| ~PowerAdvisor() override; |
| |
| void init() override; |
| void onBootFinished() override; |
| void setExpensiveRenderingExpected(DisplayId displayId, bool expected) override; |
| bool isUsingExpensiveRendering() override { return mNotifiedExpensiveRendering; }; |
| void notifyDisplayUpdateImminent() override; |
| bool usePowerHintSession() override; |
| bool supportsPowerHintSession() override; |
| bool isPowerHintSessionRunning() override; |
| void setTargetWorkDuration(int64_t targetDurationNanos) override; |
| void sendActualWorkDuration(int64_t actualDurationNanos, nsecs_t timestamp) override; |
| void enablePowerHint(bool enabled) override; |
| bool startPowerHintSession(const std::vector<int32_t>& threadIds) override; |
| |
| private: |
| HalWrapper* getPowerHal() REQUIRES(mPowerHalMutex); |
| bool mReconnectPowerHal GUARDED_BY(mPowerHalMutex) = false; |
| std::mutex mPowerHalMutex; |
| |
| std::atomic_bool mBootFinished = false; |
| std::optional<bool> mPowerHintEnabled; |
| std::optional<bool> mSupportsPowerHint; |
| bool mPowerHintSessionRunning = false; |
| |
| // An adjustable safety margin which moves the "target" earlier to allow flinger to |
| // go a bit over without dropping a frame, especially since we can't measure |
| // the exact time HWC finishes composition so "actual" durations are measured |
| // from the end of present() instead, which is a bit later. |
| static constexpr const std::chrono::nanoseconds kTargetSafetyMargin = 2ms; |
| |
| std::unordered_set<DisplayId> mExpensiveDisplays; |
| bool mNotifiedExpensiveRendering = false; |
| |
| SurfaceFlinger& mFlinger; |
| const bool mUseScreenUpdateTimer; |
| std::atomic_bool mSendUpdateImminent = true; |
| scheduler::OneShotTimer mScreenUpdateTimer; |
| }; |
| |
| } // namespace impl |
| } // namespace Hwc2 |
| } // namespace android |