| /* |
| * Copyright (C) 2008 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 "LayerState" |
| |
| #include <inttypes.h> |
| |
| #include <utils/Errors.h> |
| #include <binder/Parcel.h> |
| #include <gui/ISurfaceComposerClient.h> |
| #include <gui/IGraphicBufferProducer.h> |
| #include <gui/LayerState.h> |
| |
| #include <cmath> |
| |
| namespace android { |
| |
| status_t layer_state_t::write(Parcel& output) const |
| { |
| output.writeStrongBinder(surface); |
| output.writeUint64(what); |
| output.writeFloat(x); |
| output.writeFloat(y); |
| output.writeInt32(z); |
| output.writeUint32(w); |
| output.writeUint32(h); |
| output.writeUint32(layerStack); |
| output.writeFloat(alpha); |
| output.writeUint32(flags); |
| output.writeUint32(mask); |
| *reinterpret_cast<layer_state_t::matrix22_t *>( |
| output.writeInplace(sizeof(layer_state_t::matrix22_t))) = matrix; |
| output.write(crop_legacy); |
| output.writeStrongBinder(barrierHandle_legacy); |
| output.writeStrongBinder(reparentHandle); |
| output.writeUint64(frameNumber_legacy); |
| output.writeInt32(overrideScalingMode); |
| output.writeStrongBinder(IInterface::asBinder(barrierGbp_legacy)); |
| output.writeStrongBinder(relativeLayerHandle); |
| output.writeStrongBinder(parentHandleForChild); |
| output.writeFloat(color.r); |
| output.writeFloat(color.g); |
| output.writeFloat(color.b); |
| #ifndef NO_INPUT |
| inputInfo.write(output); |
| #endif |
| output.write(transparentRegion); |
| output.writeUint32(transform); |
| output.writeBool(transformToDisplayInverse); |
| output.write(crop); |
| output.write(frame); |
| if (buffer) { |
| output.writeBool(true); |
| output.write(*buffer); |
| } else { |
| output.writeBool(false); |
| } |
| if (acquireFence) { |
| output.writeBool(true); |
| output.write(*acquireFence); |
| } else { |
| output.writeBool(false); |
| } |
| output.writeUint32(static_cast<uint32_t>(dataspace)); |
| output.write(hdrMetadata); |
| output.write(surfaceDamageRegion); |
| output.writeInt32(api); |
| if (sidebandStream) { |
| output.writeBool(true); |
| output.writeNativeHandle(sidebandStream->handle()); |
| } else { |
| output.writeBool(false); |
| } |
| |
| memcpy(output.writeInplace(16 * sizeof(float)), |
| colorTransform.asArray(), 16 * sizeof(float)); |
| output.writeFloat(cornerRadius); |
| output.writeUint32(backgroundBlurRadius); |
| output.writeStrongBinder(cachedBuffer.token.promote()); |
| output.writeUint64(cachedBuffer.id); |
| output.writeParcelable(metadata); |
| |
| output.writeFloat(bgColorAlpha); |
| output.writeUint32(static_cast<uint32_t>(bgColorDataspace)); |
| output.writeBool(colorSpaceAgnostic); |
| |
| auto err = output.writeVectorSize(listeners); |
| if (err) { |
| return err; |
| } |
| |
| for (auto listener : listeners) { |
| err = output.writeStrongBinder(listener.transactionCompletedListener); |
| if (err) { |
| return err; |
| } |
| err = output.writeInt64Vector(listener.callbackIds); |
| if (err) { |
| return err; |
| } |
| } |
| output.writeFloat(shadowRadius); |
| output.writeInt32(frameRateSelectionPriority); |
| output.writeFloat(frameRate); |
| output.writeByte(frameRateCompatibility); |
| output.writeUint32(fixedTransformHint); |
| return NO_ERROR; |
| } |
| |
| status_t layer_state_t::read(const Parcel& input) |
| { |
| surface = input.readStrongBinder(); |
| what = input.readUint64(); |
| x = input.readFloat(); |
| y = input.readFloat(); |
| z = input.readInt32(); |
| w = input.readUint32(); |
| h = input.readUint32(); |
| layerStack = input.readUint32(); |
| alpha = input.readFloat(); |
| flags = static_cast<uint8_t>(input.readUint32()); |
| mask = static_cast<uint8_t>(input.readUint32()); |
| const void* matrix_data = input.readInplace(sizeof(layer_state_t::matrix22_t)); |
| if (matrix_data) { |
| matrix = *reinterpret_cast<layer_state_t::matrix22_t const *>(matrix_data); |
| } else { |
| return BAD_VALUE; |
| } |
| input.read(crop_legacy); |
| barrierHandle_legacy = input.readStrongBinder(); |
| reparentHandle = input.readStrongBinder(); |
| frameNumber_legacy = input.readUint64(); |
| overrideScalingMode = input.readInt32(); |
| barrierGbp_legacy = interface_cast<IGraphicBufferProducer>(input.readStrongBinder()); |
| relativeLayerHandle = input.readStrongBinder(); |
| parentHandleForChild = input.readStrongBinder(); |
| color.r = input.readFloat(); |
| color.g = input.readFloat(); |
| color.b = input.readFloat(); |
| |
| #ifndef NO_INPUT |
| inputInfo = InputWindowInfo::read(input); |
| #endif |
| |
| input.read(transparentRegion); |
| transform = input.readUint32(); |
| transformToDisplayInverse = input.readBool(); |
| input.read(crop); |
| input.read(frame); |
| buffer = new GraphicBuffer(); |
| if (input.readBool()) { |
| input.read(*buffer); |
| } |
| acquireFence = new Fence(); |
| if (input.readBool()) { |
| input.read(*acquireFence); |
| } |
| dataspace = static_cast<ui::Dataspace>(input.readUint32()); |
| input.read(hdrMetadata); |
| input.read(surfaceDamageRegion); |
| api = input.readInt32(); |
| if (input.readBool()) { |
| sidebandStream = NativeHandle::create(input.readNativeHandle(), true); |
| } |
| |
| const void* color_transform_data = input.readInplace(16 * sizeof(float)); |
| if (color_transform_data) { |
| colorTransform = mat4(static_cast<const float*>(color_transform_data)); |
| } else { |
| return BAD_VALUE; |
| } |
| cornerRadius = input.readFloat(); |
| backgroundBlurRadius = input.readUint32(); |
| cachedBuffer.token = input.readStrongBinder(); |
| cachedBuffer.id = input.readUint64(); |
| input.readParcelable(&metadata); |
| |
| bgColorAlpha = input.readFloat(); |
| bgColorDataspace = static_cast<ui::Dataspace>(input.readUint32()); |
| colorSpaceAgnostic = input.readBool(); |
| |
| int32_t numListeners = input.readInt32(); |
| listeners.clear(); |
| for (int i = 0; i < numListeners; i++) { |
| auto listener = input.readStrongBinder(); |
| std::vector<CallbackId> callbackIds; |
| input.readInt64Vector(&callbackIds); |
| listeners.emplace_back(listener, callbackIds); |
| } |
| shadowRadius = input.readFloat(); |
| frameRateSelectionPriority = input.readInt32(); |
| frameRate = input.readFloat(); |
| frameRateCompatibility = input.readByte(); |
| fixedTransformHint = static_cast<ui::Transform::RotationFlags>(input.readUint32()); |
| return NO_ERROR; |
| } |
| |
| status_t ComposerState::write(Parcel& output) const { |
| return state.write(output); |
| } |
| |
| status_t ComposerState::read(const Parcel& input) { |
| return state.read(input); |
| } |
| |
| |
| DisplayState::DisplayState() : |
| what(0), |
| layerStack(0), |
| viewport(Rect::EMPTY_RECT), |
| frame(Rect::EMPTY_RECT), |
| width(0), |
| height(0) { |
| } |
| |
| status_t DisplayState::write(Parcel& output) const { |
| output.writeStrongBinder(token); |
| output.writeStrongBinder(IInterface::asBinder(surface)); |
| output.writeUint32(what); |
| output.writeUint32(layerStack); |
| output.writeUint32(toRotationInt(orientation)); |
| output.write(viewport); |
| output.write(frame); |
| output.writeUint32(width); |
| output.writeUint32(height); |
| return NO_ERROR; |
| } |
| |
| status_t DisplayState::read(const Parcel& input) { |
| token = input.readStrongBinder(); |
| surface = interface_cast<IGraphicBufferProducer>(input.readStrongBinder()); |
| what = input.readUint32(); |
| layerStack = input.readUint32(); |
| orientation = ui::toRotation(input.readUint32()); |
| input.read(viewport); |
| input.read(frame); |
| width = input.readUint32(); |
| height = input.readUint32(); |
| return NO_ERROR; |
| } |
| |
| void DisplayState::merge(const DisplayState& other) { |
| if (other.what & eSurfaceChanged) { |
| what |= eSurfaceChanged; |
| surface = other.surface; |
| } |
| if (other.what & eLayerStackChanged) { |
| what |= eLayerStackChanged; |
| layerStack = other.layerStack; |
| } |
| if (other.what & eDisplayProjectionChanged) { |
| what |= eDisplayProjectionChanged; |
| orientation = other.orientation; |
| viewport = other.viewport; |
| frame = other.frame; |
| } |
| if (other.what & eDisplaySizeChanged) { |
| what |= eDisplaySizeChanged; |
| width = other.width; |
| height = other.height; |
| } |
| } |
| |
| void layer_state_t::merge(const layer_state_t& other) { |
| if (other.what & ePositionChanged) { |
| what |= ePositionChanged; |
| x = other.x; |
| y = other.y; |
| } |
| if (other.what & eLayerChanged) { |
| what |= eLayerChanged; |
| what &= ~eRelativeLayerChanged; |
| z = other.z; |
| } |
| if (other.what & eSizeChanged) { |
| what |= eSizeChanged; |
| w = other.w; |
| h = other.h; |
| } |
| if (other.what & eAlphaChanged) { |
| what |= eAlphaChanged; |
| alpha = other.alpha; |
| } |
| if (other.what & eMatrixChanged) { |
| what |= eMatrixChanged; |
| matrix = other.matrix; |
| } |
| if (other.what & eTransparentRegionChanged) { |
| what |= eTransparentRegionChanged; |
| transparentRegion = other.transparentRegion; |
| } |
| if (other.what & eFlagsChanged) { |
| what |= eFlagsChanged; |
| flags &= ~other.mask; |
| flags |= (other.flags & other.mask); |
| mask |= other.mask; |
| } |
| if (other.what & eLayerStackChanged) { |
| what |= eLayerStackChanged; |
| layerStack = other.layerStack; |
| } |
| if (other.what & eCropChanged_legacy) { |
| what |= eCropChanged_legacy; |
| crop_legacy = other.crop_legacy; |
| } |
| if (other.what & eCornerRadiusChanged) { |
| what |= eCornerRadiusChanged; |
| cornerRadius = other.cornerRadius; |
| } |
| if (other.what & eBackgroundBlurRadiusChanged) { |
| what |= eBackgroundBlurRadiusChanged; |
| backgroundBlurRadius = other.backgroundBlurRadius; |
| } |
| if (other.what & eDeferTransaction_legacy) { |
| what |= eDeferTransaction_legacy; |
| barrierHandle_legacy = other.barrierHandle_legacy; |
| barrierGbp_legacy = other.barrierGbp_legacy; |
| frameNumber_legacy = other.frameNumber_legacy; |
| } |
| if (other.what & eOverrideScalingModeChanged) { |
| what |= eOverrideScalingModeChanged; |
| overrideScalingMode = other.overrideScalingMode; |
| } |
| if (other.what & eReparentChildren) { |
| what |= eReparentChildren; |
| reparentHandle = other.reparentHandle; |
| } |
| if (other.what & eDetachChildren) { |
| what |= eDetachChildren; |
| } |
| if (other.what & eRelativeLayerChanged) { |
| what |= eRelativeLayerChanged; |
| what &= ~eLayerChanged; |
| z = other.z; |
| relativeLayerHandle = other.relativeLayerHandle; |
| } |
| if (other.what & eReparent) { |
| what |= eReparent; |
| parentHandleForChild = other.parentHandleForChild; |
| } |
| if (other.what & eDestroySurface) { |
| what |= eDestroySurface; |
| } |
| if (other.what & eTransformChanged) { |
| what |= eTransformChanged; |
| transform = other.transform; |
| } |
| if (other.what & eTransformToDisplayInverseChanged) { |
| what |= eTransformToDisplayInverseChanged; |
| transformToDisplayInverse = other.transformToDisplayInverse; |
| } |
| if (other.what & eCropChanged) { |
| what |= eCropChanged; |
| crop = other.crop; |
| } |
| if (other.what & eFrameChanged) { |
| what |= eFrameChanged; |
| frame = other.frame; |
| } |
| if (other.what & eBufferChanged) { |
| what |= eBufferChanged; |
| buffer = other.buffer; |
| } |
| if (other.what & eAcquireFenceChanged) { |
| what |= eAcquireFenceChanged; |
| acquireFence = other.acquireFence; |
| } |
| if (other.what & eDataspaceChanged) { |
| what |= eDataspaceChanged; |
| dataspace = other.dataspace; |
| } |
| if (other.what & eHdrMetadataChanged) { |
| what |= eHdrMetadataChanged; |
| hdrMetadata = other.hdrMetadata; |
| } |
| if (other.what & eSurfaceDamageRegionChanged) { |
| what |= eSurfaceDamageRegionChanged; |
| surfaceDamageRegion = other.surfaceDamageRegion; |
| } |
| if (other.what & eApiChanged) { |
| what |= eApiChanged; |
| api = other.api; |
| } |
| if (other.what & eSidebandStreamChanged) { |
| what |= eSidebandStreamChanged; |
| sidebandStream = other.sidebandStream; |
| } |
| if (other.what & eColorTransformChanged) { |
| what |= eColorTransformChanged; |
| colorTransform = other.colorTransform; |
| } |
| if (other.what & eHasListenerCallbacksChanged) { |
| what |= eHasListenerCallbacksChanged; |
| } |
| |
| #ifndef NO_INPUT |
| if (other.what & eInputInfoChanged) { |
| what |= eInputInfoChanged; |
| inputInfo = other.inputInfo; |
| } |
| #endif |
| |
| if (other.what & eCachedBufferChanged) { |
| what |= eCachedBufferChanged; |
| cachedBuffer = other.cachedBuffer; |
| } |
| if (other.what & eBackgroundColorChanged) { |
| what |= eBackgroundColorChanged; |
| color = other.color; |
| bgColorAlpha = other.bgColorAlpha; |
| bgColorDataspace = other.bgColorDataspace; |
| } |
| if (other.what & eMetadataChanged) { |
| what |= eMetadataChanged; |
| metadata.merge(other.metadata); |
| } |
| if (other.what & eShadowRadiusChanged) { |
| what |= eShadowRadiusChanged; |
| shadowRadius = other.shadowRadius; |
| } |
| if (other.what & eFrameRateSelectionPriority) { |
| what |= eFrameRateSelectionPriority; |
| frameRateSelectionPriority = other.frameRateSelectionPriority; |
| } |
| if (other.what & eFrameRateChanged) { |
| what |= eFrameRateChanged; |
| frameRate = other.frameRate; |
| frameRateCompatibility = other.frameRateCompatibility; |
| } |
| if (other.what & eFixedTransformHintChanged) { |
| what |= eFixedTransformHintChanged; |
| fixedTransformHint = other.fixedTransformHint; |
| } |
| if ((other.what & what) != other.what) { |
| ALOGE("Unmerged SurfaceComposer Transaction properties. LayerState::merge needs updating? " |
| "other.what=0x%" PRIu64 " what=0x%" PRIu64, |
| other.what, what); |
| } |
| } |
| |
| // ------------------------------- InputWindowCommands ---------------------------------------- |
| |
| void InputWindowCommands::merge(const InputWindowCommands& other) { |
| syncInputWindows |= other.syncInputWindows; |
| } |
| |
| void InputWindowCommands::clear() { |
| syncInputWindows = false; |
| } |
| |
| void InputWindowCommands::write(Parcel& output) const { |
| output.writeBool(syncInputWindows); |
| } |
| |
| void InputWindowCommands::read(const Parcel& input) { |
| syncInputWindows = input.readBool(); |
| } |
| |
| bool ValidateFrameRate(float frameRate, int8_t compatibility, const char* inFunctionName) { |
| const char* functionName = inFunctionName != nullptr ? inFunctionName : "call"; |
| int floatClassification = std::fpclassify(frameRate); |
| if (frameRate < 0 || floatClassification == FP_INFINITE || floatClassification == FP_NAN) { |
| ALOGE("%s failed - invalid frame rate %f", functionName, frameRate); |
| return false; |
| } |
| |
| if (compatibility != ANATIVEWINDOW_FRAME_RATE_COMPATIBILITY_DEFAULT && |
| compatibility != ANATIVEWINDOW_FRAME_RATE_COMPATIBILITY_FIXED_SOURCE) { |
| ALOGE("%s failed - invalid compatibility value %d", functionName, compatibility); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| }; // namespace android |