/* * Copyright 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. */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace android::compositionengine::impl { // The implementation class contains the common implementation, but does not // actually contain the final output state. class Output : public virtual compositionengine::Output { public: Output() = default; ~Output() override; // compositionengine::Output overrides bool isValid() const override; std::optional getDisplayId() const override; void setCompositionEnabled(bool) override; void setLayerCachingEnabled(bool) override; void setLayerCachingTexturePoolEnabled(bool) override; void setProjection(ui::Rotation orientation, const Rect& layerStackSpaceRect, const Rect& orientedDisplaySpaceRect) override; void setNextBrightness(float brightness) override; void setDisplaySize(const ui::Size&) override; void setLayerFilter(ui::LayerFilter) override; ui::Transform::RotationFlags getTransformHint() const override; void setColorTransform(const compositionengine::CompositionRefreshArgs&) override; void setColorProfile(const ColorProfile&) override; void setDisplayBrightness(float sdrWhitePointNits, float displayBrightnessNits) override; void dump(std::string&) const override; void dumpPlannerInfo(const Vector& args, std::string&) const override; const std::string& getName() const override; void setName(const std::string&) override; compositionengine::DisplayColorProfile* getDisplayColorProfile() const override; void setDisplayColorProfile(std::unique_ptr) override; compositionengine::RenderSurface* getRenderSurface() const override; void setRenderSurface(std::unique_ptr) override; Region getDirtyRegion() const override; bool includesLayer(ui::LayerFilter) const override; bool includesLayer(const sp&) const override; compositionengine::OutputLayer* getOutputLayerForLayer(const sp&) const override; void setReleasedLayers(ReleasedLayers&&) override; void prepare(const CompositionRefreshArgs&, LayerFESet&) override; ftl::Future present(const CompositionRefreshArgs&) override; bool supportsOffloadPresent() const override { return false; } void offloadPresentNextFrame() override; void uncacheBuffers(const std::vector& bufferIdsToUncache) override; void rebuildLayerStacks(const CompositionRefreshArgs&, LayerFESet&) override; void collectVisibleLayers(const CompositionRefreshArgs&, compositionengine::Output::CoverageState&) override; void ensureOutputLayerIfVisible(sp&, compositionengine::Output::CoverageState&) override; void setReleasedLayers(const compositionengine::CompositionRefreshArgs&) override; void updateCompositionState(const compositionengine::CompositionRefreshArgs&) override; void planComposition() override; void writeCompositionState(const compositionengine::CompositionRefreshArgs&) override; void updateColorProfile(const compositionengine::CompositionRefreshArgs&) override; void beginFrame() override; void prepareFrame() override; GpuCompositionResult prepareFrameAsync() override; void devOptRepaintFlash(const CompositionRefreshArgs&) override; void finishFrame(GpuCompositionResult&&) override; std::optional composeSurfaces(const Region&, std::shared_ptr, base::unique_fd&) override; void presentFrameAndReleaseLayers(bool flushEvenWhenDisabled) override; void renderCachedSets(const CompositionRefreshArgs&) override; void cacheClientCompositionRequests(uint32_t) override; bool canPredictCompositionStrategy(const CompositionRefreshArgs&) override; void setPredictCompositionStrategy(bool) override; void setTreat170mAsSrgb(bool) override; // Testing const ReleasedLayers& getReleasedLayersForTest() const; void setDisplayColorProfileForTest(std::unique_ptr); void setRenderSurfaceForTest(std::unique_ptr); bool plannerEnabled() const { return mPlanner != nullptr; } virtual bool anyLayersRequireClientComposition() const; virtual void updateProtectedContentState(); virtual bool dequeueRenderBuffer(base::unique_fd*, std::shared_ptr*); virtual std::future chooseCompositionStrategyAsync( std::optional*); virtual void resetCompositionStrategy(); virtual ftl::Future presentFrameAndReleaseLayersAsync( bool flushEvenWhenDisabled); protected: std::unique_ptr createOutputLayer(const sp&) const; std::optional findCurrentOutputLayerForLayer( const sp&) const; using DeviceRequestedChanges = android::HWComposer::DeviceRequestedChanges; bool chooseCompositionStrategy( std::optional*) override { return true; }; void applyCompositionStrategy(const std::optional&) override{}; bool getSkipColorTransform() const override; compositionengine::Output::FrameFences presentFrame() override; void executeCommands() override {} virtual renderengine::DisplaySettings generateClientCompositionDisplaySettings( const std::shared_ptr& buffer) const; std::vector generateClientCompositionRequests( bool supportsProtectedContent, ui::Dataspace outputDataspace, std::vector& outLayerFEs) override; void appendRegionFlashRequests(const Region&, std::vector&) override; void setExpensiveRenderingExpected(bool enabled) override; void setHintSessionGpuStart(TimePoint startTime) override; void setHintSessionGpuFence(std::unique_ptr&& gpuFence) override; void setHintSessionRequiresRenderEngine(bool requiresRenderEngine) override; bool isPowerHintSessionEnabled() override; bool isPowerHintSessionGpuReportingEnabled() override; void dumpBase(std::string&) const; // Implemented by the final implementation for the final state it uses. virtual compositionengine::OutputLayer* ensureOutputLayer(std::optional, const sp&) = 0; virtual compositionengine::OutputLayer* injectOutputLayerForTest(const sp&) = 0; virtual void finalizePendingOutputLayers() = 0; virtual const compositionengine::CompositionEngine& getCompositionEngine() const = 0; virtual void dumpState(std::string& out) const = 0; bool mustRecompose() const; const std::string& getNamePlusId() const { return mNamePlusId; } private: void dirtyEntireOutput(); compositionengine::OutputLayer* findLayerRequestingBackgroundComposition() const; void finishPrepareFrame(); ui::Dataspace getBestDataspace(ui::Dataspace*, bool*) const; compositionengine::Output::ColorProfile pickColorProfile( const compositionengine::CompositionRefreshArgs&) const; void updateHwcAsyncWorker(); float getHdrSdrRatio(const std::shared_ptr& buffer) const; std::string mName; std::string mNamePlusId; std::unique_ptr mDisplayColorProfile; std::unique_ptr mRenderSurface; ReleasedLayers mReleasedLayers; OutputLayer* mLayerRequestingBackgroundBlur = nullptr; std::unique_ptr mClientCompositionRequestCache; std::unique_ptr mPlanner; std::unique_ptr mHwComposerAsyncWorker; bool mPredictCompositionStrategy = false; bool mOffloadPresent = false; // Whether the content must be recomposed this frame. bool mMustRecompose = false; }; // This template factory function standardizes the implementation details of the // final class using the types actually required by the implementation. This is // not possible to do in the base class as those types may not even be visible // to the base code. template std::shared_ptr createOutputTemplated(const CompositionEngine& compositionEngine, Args... args) { class Output final : public BaseOutput { public: // Clang incorrectly complains that these are unused. #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wunused-local-typedef" using OutputCompositionState = std::remove_const_t< std::remove_reference_t().getState())>>; using OutputLayer = std::remove_pointer_t().getOutputLayerOrderedByZByIndex(0))>; #pragma clang diagnostic pop explicit Output(const CompositionEngine& compositionEngine, Args... args) : BaseOutput(std::forward(args)...), mCompositionEngine(compositionEngine) {} ~Output() override = default; private: // compositionengine::Output overrides const OutputCompositionState& getState() const override { return mState; } OutputCompositionState& editState() override { return mState; } size_t getOutputLayerCount() const override { return mCurrentOutputLayersOrderedByZ.size(); } OutputLayer* getOutputLayerOrderedByZByIndex(size_t index) const override { if (index >= mCurrentOutputLayersOrderedByZ.size()) { return nullptr; } return mCurrentOutputLayersOrderedByZ[index].get(); } // compositionengine::impl::Output overrides const CompositionEngine& getCompositionEngine() const override { return mCompositionEngine; }; OutputLayer* ensureOutputLayer(std::optional prevIndex, const sp& layerFE) { auto outputLayer = (prevIndex && *prevIndex <= mCurrentOutputLayersOrderedByZ.size()) ? std::move(mCurrentOutputLayersOrderedByZ[*prevIndex]) : BaseOutput::createOutputLayer(layerFE); auto result = outputLayer.get(); mPendingOutputLayersOrderedByZ.emplace_back(std::move(outputLayer)); return result; } void finalizePendingOutputLayers() override { // The pending layers are added in reverse order. Reverse them to // get the back-to-front ordered list of layers. std::reverse(mPendingOutputLayersOrderedByZ.begin(), mPendingOutputLayersOrderedByZ.end()); mCurrentOutputLayersOrderedByZ = std::move(mPendingOutputLayersOrderedByZ); } void dumpState(std::string& out) const override { mState.dump(out); } OutputLayer* injectOutputLayerForTest(const sp& layerFE) override { auto outputLayer = BaseOutput::createOutputLayer(layerFE); auto result = outputLayer.get(); mCurrentOutputLayersOrderedByZ.emplace_back(std::move(outputLayer)); return result; } // Note: This is declared as a private virtual non-override so it can be // an override implementation in the unit tests, but otherwise is not an // accessible override for the normal implementation. virtual void injectOutputLayerForTest(std::unique_ptr outputLayer) { mCurrentOutputLayersOrderedByZ.emplace_back(std::move(outputLayer)); } void clearOutputLayers() override { mCurrentOutputLayersOrderedByZ.clear(); mPendingOutputLayersOrderedByZ.clear(); } const CompositionEngine& mCompositionEngine; OutputCompositionState mState; std::vector> mCurrentOutputLayersOrderedByZ; std::vector> mPendingOutputLayersOrderedByZ; }; return std::make_shared(compositionEngine, std::forward(args)...); } std::shared_ptr createOutput(const compositionengine::CompositionEngine&); } // namespace android::compositionengine::impl