| /* | 
 |  * 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. | 
 |  */ | 
 |  | 
 | #include <android-base/stringprintf.h> | 
 | #include <compositionengine/CompositionEngine.h> | 
 | #include <compositionengine/DisplayColorProfile.h> | 
 | #include <compositionengine/LayerFE.h> | 
 | #include <compositionengine/RenderSurface.h> | 
 | #include <compositionengine/impl/Output.h> | 
 | #include <compositionengine/impl/OutputLayer.h> | 
 | #include <ui/DebugUtils.h> | 
 |  | 
 | namespace android::compositionengine { | 
 |  | 
 | Output::~Output() = default; | 
 |  | 
 | namespace impl { | 
 |  | 
 | Output::Output(const CompositionEngine& compositionEngine) | 
 |       : mCompositionEngine(compositionEngine) {} | 
 |  | 
 | Output::~Output() = default; | 
 |  | 
 | const CompositionEngine& Output::getCompositionEngine() const { | 
 |     return mCompositionEngine; | 
 | } | 
 |  | 
 | bool Output::isValid() const { | 
 |     return mDisplayColorProfile && mDisplayColorProfile->isValid() && mRenderSurface && | 
 |             mRenderSurface->isValid(); | 
 | } | 
 |  | 
 | const std::string& Output::getName() const { | 
 |     return mName; | 
 | } | 
 |  | 
 | void Output::setName(const std::string& name) { | 
 |     mName = name; | 
 | } | 
 |  | 
 | void Output::setCompositionEnabled(bool enabled) { | 
 |     if (mState.isEnabled == enabled) { | 
 |         return; | 
 |     } | 
 |  | 
 |     mState.isEnabled = enabled; | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | void Output::setProjection(const ui::Transform& transform, int32_t orientation, const Rect& frame, | 
 |                            const Rect& viewport, const Rect& scissor, bool needsFiltering) { | 
 |     mState.transform = transform; | 
 |     mState.orientation = orientation; | 
 |     mState.scissor = scissor; | 
 |     mState.frame = frame; | 
 |     mState.viewport = viewport; | 
 |     mState.needsFiltering = needsFiltering; | 
 |  | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | // TODO(lpique): Rename setSize() once more is moved. | 
 | void Output::setBounds(const ui::Size& size) { | 
 |     mRenderSurface->setDisplaySize(size); | 
 |     // TODO(lpique): Rename mState.size once more is moved. | 
 |     mState.bounds = Rect(mRenderSurface->getSize()); | 
 |  | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | void Output::setLayerStackFilter(uint32_t layerStackId, bool isInternal) { | 
 |     mState.layerStackId = layerStackId; | 
 |     mState.layerStackInternal = isInternal; | 
 |  | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | void Output::setColorTransform(const mat4& transform) { | 
 |     if (mState.colorTransformMat == transform) { | 
 |         return; | 
 |     } | 
 |  | 
 |     const bool isIdentity = (transform == mat4()); | 
 |     const auto newColorTransform = | 
 |             isIdentity ? HAL_COLOR_TRANSFORM_IDENTITY : HAL_COLOR_TRANSFORM_ARBITRARY_MATRIX; | 
 |  | 
 |     mState.colorTransform = newColorTransform; | 
 |     mState.colorTransformMat = transform; | 
 |  | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | void Output::setColorMode(ui::ColorMode mode, ui::Dataspace dataspace, | 
 |                           ui::RenderIntent renderIntent) { | 
 |     if (mState.colorMode == mode && mState.dataspace == dataspace && | 
 |         mState.renderIntent == renderIntent) { | 
 |         return; | 
 |     } | 
 |  | 
 |     mState.colorMode = mode; | 
 |     mState.dataspace = dataspace; | 
 |     mState.renderIntent = renderIntent; | 
 |  | 
 |     mRenderSurface->setBufferDataspace(dataspace); | 
 |  | 
 |     ALOGV("Set active color mode: %s (%d), active render intent: %s (%d)", | 
 |           decodeColorMode(mode).c_str(), mode, decodeRenderIntent(renderIntent).c_str(), | 
 |           renderIntent); | 
 |  | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | void Output::dump(std::string& out) const { | 
 |     using android::base::StringAppendF; | 
 |  | 
 |     StringAppendF(&out, "   Composition Output State: [\"%s\"]", mName.c_str()); | 
 |  | 
 |     out.append("\n   "); | 
 |  | 
 |     dumpBase(out); | 
 | } | 
 |  | 
 | void Output::dumpBase(std::string& out) const { | 
 |     mState.dump(out); | 
 |  | 
 |     if (mDisplayColorProfile) { | 
 |         mDisplayColorProfile->dump(out); | 
 |     } else { | 
 |         out.append("    No display color profile!\n"); | 
 |     } | 
 |  | 
 |     if (mRenderSurface) { | 
 |         mRenderSurface->dump(out); | 
 |     } else { | 
 |         out.append("    No render surface!\n"); | 
 |     } | 
 |  | 
 |     android::base::StringAppendF(&out, "\n   %zu Layers\b", mOutputLayersOrderedByZ.size()); | 
 |     for (const auto& outputLayer : mOutputLayersOrderedByZ) { | 
 |         if (!outputLayer) { | 
 |             continue; | 
 |         } | 
 |         outputLayer->dump(out); | 
 |     } | 
 | } | 
 |  | 
 | compositionengine::DisplayColorProfile* Output::getDisplayColorProfile() const { | 
 |     return mDisplayColorProfile.get(); | 
 | } | 
 |  | 
 | void Output::setDisplayColorProfile(std::unique_ptr<compositionengine::DisplayColorProfile> mode) { | 
 |     mDisplayColorProfile = std::move(mode); | 
 | } | 
 |  | 
 | void Output::setDisplayColorProfileForTest( | 
 |         std::unique_ptr<compositionengine::DisplayColorProfile> mode) { | 
 |     mDisplayColorProfile = std::move(mode); | 
 | } | 
 |  | 
 | compositionengine::RenderSurface* Output::getRenderSurface() const { | 
 |     return mRenderSurface.get(); | 
 | } | 
 |  | 
 | void Output::setRenderSurface(std::unique_ptr<compositionengine::RenderSurface> surface) { | 
 |     mRenderSurface = std::move(surface); | 
 |     mState.bounds = Rect(mRenderSurface->getSize()); | 
 |  | 
 |     dirtyEntireOutput(); | 
 | } | 
 |  | 
 | void Output::setRenderSurfaceForTest(std::unique_ptr<compositionengine::RenderSurface> surface) { | 
 |     mRenderSurface = std::move(surface); | 
 | } | 
 |  | 
 | const OutputCompositionState& Output::getState() const { | 
 |     return mState; | 
 | } | 
 |  | 
 | OutputCompositionState& Output::editState() { | 
 |     return mState; | 
 | } | 
 |  | 
 | Region Output::getDirtyRegion(bool repaintEverything) const { | 
 |     Region dirty(mState.viewport); | 
 |     if (!repaintEverything) { | 
 |         dirty.andSelf(mState.dirtyRegion); | 
 |     } | 
 |     return dirty; | 
 | } | 
 |  | 
 | bool Output::belongsInOutput(uint32_t layerStackId, bool internalOnly) const { | 
 |     // The layerStackId's must match, and also the layer must not be internal | 
 |     // only when not on an internal output. | 
 |     return (layerStackId == mState.layerStackId) && (!internalOnly || mState.layerStackInternal); | 
 | } | 
 |  | 
 | compositionengine::OutputLayer* Output::getOutputLayerForLayer( | 
 |         compositionengine::Layer* layer) const { | 
 |     for (const auto& outputLayer : mOutputLayersOrderedByZ) { | 
 |         if (outputLayer && &outputLayer->getLayer() == layer) { | 
 |             return outputLayer.get(); | 
 |         } | 
 |     } | 
 |     return nullptr; | 
 | } | 
 |  | 
 | std::unique_ptr<compositionengine::OutputLayer> Output::getOrCreateOutputLayer( | 
 |         std::optional<DisplayId> displayId, std::shared_ptr<compositionengine::Layer> layer, | 
 |         sp<compositionengine::LayerFE> layerFE) { | 
 |     for (auto& outputLayer : mOutputLayersOrderedByZ) { | 
 |         if (outputLayer && &outputLayer->getLayer() == layer.get()) { | 
 |             return std::move(outputLayer); | 
 |         } | 
 |     } | 
 |     return createOutputLayer(mCompositionEngine, displayId, *this, layer, layerFE); | 
 | } | 
 |  | 
 | void Output::setOutputLayersOrderedByZ(OutputLayers&& layers) { | 
 |     mOutputLayersOrderedByZ = std::move(layers); | 
 | } | 
 |  | 
 | const Output::OutputLayers& Output::getOutputLayersOrderedByZ() const { | 
 |     return mOutputLayersOrderedByZ; | 
 | } | 
 |  | 
 | void Output::dirtyEntireOutput() { | 
 |     mState.dirtyRegion.set(mState.bounds); | 
 | } | 
 |  | 
 | } // namespace impl | 
 | } // namespace android::compositionengine |