|  | /* | 
|  | * 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. | 
|  | */ | 
|  |  | 
|  | #include <compositionengine/CompositionRefreshArgs.h> | 
|  | #include <compositionengine/LayerFE.h> | 
|  | #include <compositionengine/OutputLayer.h> | 
|  | #include <compositionengine/impl/CompositionEngine.h> | 
|  | #include <compositionengine/impl/Display.h> | 
|  | #include <compositionengine/impl/Layer.h> | 
|  | #include <renderengine/RenderEngine.h> | 
|  | #include <utils/Trace.h> | 
|  |  | 
|  | #include "DisplayHardware/HWComposer.h" | 
|  |  | 
|  | namespace android::compositionengine { | 
|  |  | 
|  | CompositionEngine::~CompositionEngine() = default; | 
|  |  | 
|  | namespace impl { | 
|  |  | 
|  | std::unique_ptr<compositionengine::CompositionEngine> createCompositionEngine() { | 
|  | return std::make_unique<CompositionEngine>(); | 
|  | } | 
|  |  | 
|  | CompositionEngine::CompositionEngine() = default; | 
|  | CompositionEngine::~CompositionEngine() = default; | 
|  |  | 
|  | std::shared_ptr<compositionengine::Display> CompositionEngine::createDisplay( | 
|  | const DisplayCreationArgs& args) { | 
|  | return compositionengine::impl::createDisplay(*this, args); | 
|  | } | 
|  |  | 
|  | std::shared_ptr<compositionengine::Layer> CompositionEngine::createLayer( | 
|  | const LayerCreationArgs& args) { | 
|  | return compositionengine::impl::createLayer(args); | 
|  | } | 
|  |  | 
|  | HWComposer& CompositionEngine::getHwComposer() const { | 
|  | return *mHwComposer.get(); | 
|  | } | 
|  |  | 
|  | void CompositionEngine::setHwComposer(std::unique_ptr<HWComposer> hwComposer) { | 
|  | mHwComposer = std::move(hwComposer); | 
|  | } | 
|  |  | 
|  | renderengine::RenderEngine& CompositionEngine::getRenderEngine() const { | 
|  | return *mRenderEngine.get(); | 
|  | } | 
|  |  | 
|  | void CompositionEngine::setRenderEngine(std::unique_ptr<renderengine::RenderEngine> renderEngine) { | 
|  | mRenderEngine = std::move(renderEngine); | 
|  | } | 
|  |  | 
|  | TimeStats& CompositionEngine::getTimeStats() const { | 
|  | return *mTimeStats.get(); | 
|  | } | 
|  |  | 
|  | void CompositionEngine::setTimeStats(const std::shared_ptr<TimeStats>& timeStats) { | 
|  | mTimeStats = timeStats; | 
|  | } | 
|  |  | 
|  | bool CompositionEngine::needsAnotherUpdate() const { | 
|  | return mNeedsAnotherUpdate; | 
|  | } | 
|  |  | 
|  | nsecs_t CompositionEngine::getLastFrameRefreshTimestamp() const { | 
|  | return mRefreshStartTime; | 
|  | } | 
|  |  | 
|  | void CompositionEngine::present(CompositionRefreshArgs& args) { | 
|  | ATRACE_CALL(); | 
|  | ALOGV(__FUNCTION__); | 
|  |  | 
|  | preComposition(args); | 
|  |  | 
|  | { | 
|  | // latchedLayers is used to track the set of front-end layer state that | 
|  | // has been latched across all outputs for the prepare step, and is not | 
|  | // needed for anything else. | 
|  | LayerFESet latchedLayers; | 
|  |  | 
|  | for (const auto& output : args.outputs) { | 
|  | output->prepare(args, latchedLayers); | 
|  | } | 
|  | } | 
|  |  | 
|  | updateLayerStateFromFE(args); | 
|  |  | 
|  | for (const auto& output : args.outputs) { | 
|  | output->present(args); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CompositionEngine::updateCursorAsync(CompositionRefreshArgs& args) { | 
|  | std::unordered_map<compositionengine::LayerFE*, compositionengine::LayerFECompositionState*> | 
|  | uniqueVisibleLayers; | 
|  |  | 
|  | for (const auto& output : args.outputs) { | 
|  | for (auto* layer : output->getOutputLayersOrderedByZ()) { | 
|  | if (layer->isHardwareCursor()) { | 
|  | // Latch the cursor composition state from each front-end layer. | 
|  | layer->getLayerFE().latchCursorCompositionState(layer->getLayer().editFEState()); | 
|  |  | 
|  | layer->writeCursorPositionToHWC(); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void CompositionEngine::preComposition(CompositionRefreshArgs& args) { | 
|  | ATRACE_CALL(); | 
|  | ALOGV(__FUNCTION__); | 
|  |  | 
|  | bool needsAnotherUpdate = false; | 
|  |  | 
|  | mRefreshStartTime = systemTime(SYSTEM_TIME_MONOTONIC); | 
|  |  | 
|  | for (auto& layer : args.layers) { | 
|  | sp<compositionengine::LayerFE> layerFE = layer->getLayerFE(); | 
|  | if (layerFE && layerFE->onPreComposition(mRefreshStartTime)) { | 
|  | needsAnotherUpdate = true; | 
|  | } | 
|  | } | 
|  |  | 
|  | mNeedsAnotherUpdate = needsAnotherUpdate; | 
|  | } | 
|  |  | 
|  | void CompositionEngine::dump(std::string&) const { | 
|  | // The base class has no state to dump, but derived classes might. | 
|  | } | 
|  |  | 
|  | void CompositionEngine::setNeedsAnotherUpdateForTest(bool value) { | 
|  | mNeedsAnotherUpdate = value; | 
|  | } | 
|  |  | 
|  | void CompositionEngine::updateLayerStateFromFE(CompositionRefreshArgs& args) { | 
|  | // Update the composition state from each front-end layer | 
|  | for (const auto& output : args.outputs) { | 
|  | output->updateLayerStateFromFE(args); | 
|  | } | 
|  | } | 
|  |  | 
|  | } // namespace impl | 
|  | } // namespace android::compositionengine |