/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: sw=2 ts=8 et : */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "LayerTransactionParent.h" #include // for vector #include "apz/src/AsyncPanZoomController.h" #include "CompositableHost.h" // for CompositableParent, Get, etc #include "ImageLayers.h" // for ImageLayer #include "Layers.h" // for Layer, ContainerLayer, etc #include "CompositableTransactionParent.h" // for EditReplyVector #include "CompositorBridgeParent.h" #include "gfxPrefs.h" #include "mozilla/gfx/BasePoint3D.h" // for BasePoint3D #include "mozilla/layers/CanvasLayerComposite.h" #include "mozilla/layers/ColorLayerComposite.h" #include "mozilla/layers/Compositor.h" // for Compositor #include "mozilla/layers/ContainerLayerComposite.h" #include "mozilla/layers/ImageBridgeParent.h" // for ImageBridgeParent #include "mozilla/layers/ImageLayerComposite.h" #include "mozilla/layers/LayerManagerComposite.h" #include "mozilla/layers/LayersMessages.h" // for EditReply, etc #include "mozilla/layers/LayersTypes.h" // for MOZ_LAYERS_LOG #include "mozilla/layers/PCompositableParent.h" #include "mozilla/layers/TextureHostOGL.h" // for TextureHostOGL #include "mozilla/layers/PaintedLayerComposite.h" #include "mozilla/mozalloc.h" // for operator delete, etc #include "mozilla/Unused.h" #include "nsCoord.h" // for NSAppUnitsToFloatPixels #include "nsDebug.h" // for NS_RUNTIMEABORT #include "nsDeviceContext.h" // for AppUnitsPerCSSPixel #include "nsISupportsImpl.h" // for Layer::Release, etc #include "nsLayoutUtils.h" // for nsLayoutUtils #include "nsMathUtils.h" // for NS_round #include "nsPoint.h" // for nsPoint #include "nsTArray.h" // for nsTArray, nsTArray_Impl, etc #include "TreeTraversal.h" // for ForEachNode #include "GeckoProfiler.h" #include "mozilla/layers/TextureHost.h" #include "mozilla/layers/AsyncCompositionManager.h" typedef std::vector EditReplyVector; using mozilla::layout::RenderFrameParent; namespace mozilla { namespace layers { //-------------------------------------------------- // LayerTransactionParent LayerTransactionParent::LayerTransactionParent(HostLayerManager* aManager, CompositorBridgeParentBase* aBridge, uint64_t aId) : mLayerManager(aManager) , mCompositorBridge(aBridge) , mId(aId) , mChildEpoch(0) , mParentEpoch(0) , mPendingTransaction(0) , mPendingCompositorUpdates(0) , mDestroyed(false) , mIPCOpen(false) { } LayerTransactionParent::~LayerTransactionParent() { } mozilla::ipc::IPCResult LayerTransactionParent::RecvShutdown() { Destroy(); IProtocol* mgr = Manager(); if (!Send__delete__(this)) { return IPC_FAIL_NO_REASON(mgr); } return IPC_OK(); } void LayerTransactionParent::Destroy() { mDestroyed = true; } mozilla::ipc::IPCResult LayerTransactionParent::RecvUpdateNoSwap(const TransactionInfo& txn) { return RecvUpdate(txn, nullptr); } class MOZ_STACK_CLASS AutoLayerTransactionParentAsyncMessageSender { public: explicit AutoLayerTransactionParentAsyncMessageSender(LayerTransactionParent* aLayerTransaction, const InfallibleTArray* aDestroyActors = nullptr) : mLayerTransaction(aLayerTransaction) , mActorsToDestroy(aDestroyActors) { mLayerTransaction->SetAboutToSendAsyncMessages(); ImageBridgeParent::SetAboutToSendAsyncMessages(mLayerTransaction->GetChildProcessId()); } ~AutoLayerTransactionParentAsyncMessageSender() { mLayerTransaction->SendPendingAsyncMessages(); ImageBridgeParent::SendPendingAsyncMessages(mLayerTransaction->GetChildProcessId()); if (mActorsToDestroy) { // Destroy the actors after sending the async messages because the latter may contain // references to some actors. for (const auto& op : *mActorsToDestroy) { mLayerTransaction->DestroyActor(op); } } } private: LayerTransactionParent* mLayerTransaction; const InfallibleTArray* mActorsToDestroy; }; mozilla::ipc::IPCResult LayerTransactionParent::RecvPaintTime(const uint64_t& aTransactionId, const TimeDuration& aPaintTime) { mCompositorBridge->UpdatePaintTime(this, aPaintTime); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvUpdate(const TransactionInfo& aInfo, InfallibleTArray* reply) { GeckoProfilerTracingRAII tracer("Paint", "LayerTransaction"); PROFILER_LABEL("LayerTransactionParent", "RecvUpdate", js::ProfileEntry::Category::GRAPHICS); #ifdef COMPOSITOR_PERFORMANCE_WARNING TimeStamp updateStart = TimeStamp::Now(); #endif MOZ_LAYERS_LOG(("[ParentSide] received txn with %d edits", aInfo.cset().Length())); UpdateFwdTransactionId(aInfo.fwdTransactionId()); if (mDestroyed || !layer_manager() || layer_manager()->IsDestroyed()) { for (const auto& op : aInfo.toDestroy()) { DestroyActor(op); } return IPC_OK(); } // This ensures that destroy operations are always processed. It is not safe // to early-return from RecvUpdate without doing so. AutoLayerTransactionParentAsyncMessageSender autoAsyncMessageSender(this, &aInfo.toDestroy()); EditReplyVector replyv; { AutoResolveRefLayers resolve(mCompositorBridge->GetCompositionManager(this)); layer_manager()->BeginTransaction(); } // not all edits require an update to the hit testing tree bool updateHitTestingTree = false; for (EditArray::index_type i = 0; i < aInfo.cset().Length(); ++i) { const Edit& edit = const_cast(aInfo.cset()[i]); switch (edit.type()) { // Create* ops case Edit::TOpCreatePaintedLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreatePaintedLayer")); RefPtr layer = layer_manager()->CreatePaintedLayer(); if (!BindLayer(layer, edit.get_OpCreatePaintedLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateContainerLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateContainerLayer")); RefPtr layer = layer_manager()->CreateContainerLayer(); if (!BindLayer(layer, edit.get_OpCreateContainerLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateImageLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateImageLayer")); RefPtr layer = layer_manager()->CreateImageLayer(); if (!BindLayer(layer, edit.get_OpCreateImageLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateColorLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateColorLayer")); RefPtr layer = layer_manager()->CreateColorLayer(); if (!BindLayer(layer, edit.get_OpCreateColorLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateTextLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateTextLayer")); RefPtr layer = layer_manager()->CreateTextLayer(); if (!BindLayer(layer, edit.get_OpCreateTextLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateBorderLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateTextLayer")); RefPtr layer = layer_manager()->CreateBorderLayer(); if (!BindLayer(layer, edit.get_OpCreateBorderLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateCanvasLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateCanvasLayer")); RefPtr layer = layer_manager()->CreateCanvasLayer(); if (!BindLayer(layer, edit.get_OpCreateCanvasLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpCreateRefLayer: { MOZ_LAYERS_LOG(("[ParentSide] CreateRefLayer")); RefPtr layer = layer_manager()->CreateRefLayer(); if (!BindLayer(layer, edit.get_OpCreateRefLayer())) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } // Attributes case Edit::TOpSetLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] SetLayerAttributes")); const OpSetLayerAttributes& osla = edit.get_OpSetLayerAttributes(); Layer* layer = AsLayer(osla.layer()); if (!layer) { return IPC_FAIL_NO_REASON(this); } const LayerAttributes& attrs = osla.attrs(); const CommonLayerAttributes& common = attrs.common(); layer->SetLayerBounds(common.layerBounds()); layer->SetVisibleRegion(common.visibleRegion()); layer->SetEventRegions(common.eventRegions()); layer->SetContentFlags(common.contentFlags()); layer->SetOpacity(common.opacity()); layer->SetClipRect(common.useClipRect() ? Some(common.clipRect()) : Nothing()); layer->SetScrolledClip(common.scrolledClip()); layer->SetBaseTransform(common.transform().value()); layer->SetTransformIsPerspective(common.transformIsPerspective()); layer->SetPostScale(common.postXScale(), common.postYScale()); layer->SetIsFixedPosition(common.isFixedPosition()); if (common.isFixedPosition()) { layer->SetFixedPositionData(common.fixedPositionScrollContainerId(), common.fixedPositionAnchor(), common.fixedPositionSides()); } if (common.isStickyPosition()) { layer->SetStickyPositionData(common.stickyScrollContainerId(), common.stickyScrollRangeOuter(), common.stickyScrollRangeInner()); } layer->SetScrollbarData(common.scrollbarTargetContainerId(), static_cast(common.scrollbarDirection()), common.scrollbarThumbRatio()); if (common.isScrollbarContainer()) { layer->SetIsScrollbarContainer(); } layer->SetMixBlendMode((gfx::CompositionOp)common.mixBlendMode()); layer->SetForceIsolatedGroup(common.forceIsolatedGroup()); if (LayerHandle maskLayer = common.maskLayer()) { layer->SetMaskLayer(AsLayer(maskLayer)); } else { layer->SetMaskLayer(nullptr); } layer->SetAnimations(common.animations()); layer->SetScrollMetadata(common.scrollMetadata()); layer->SetDisplayListLog(common.displayListLog().get()); // The updated invalid region is added to the existing one, since we can // update multiple times before the next composite. layer->AddInvalidRegion(common.invalidRegion()); nsTArray> maskLayers; for (size_t i = 0; i < common.ancestorMaskLayers().Length(); i++) { Layer* maskLayer = AsLayer(common.ancestorMaskLayers().ElementAt(i)); if (!maskLayer) { return IPC_FAIL_NO_REASON(this); } maskLayers.AppendElement(maskLayer); } layer->SetAncestorMaskLayers(maskLayers); typedef SpecificLayerAttributes Specific; const SpecificLayerAttributes& specific = attrs.specific(); switch (specific.type()) { case Specific::Tnull_t: break; case Specific::TPaintedLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] painted layer")); PaintedLayer* paintedLayer = layer->AsPaintedLayer(); if (!paintedLayer) { return IPC_FAIL_NO_REASON(this); } const PaintedLayerAttributes& attrs = specific.get_PaintedLayerAttributes(); paintedLayer->SetValidRegion(attrs.validRegion()); break; } case Specific::TContainerLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] container layer")); ContainerLayer* containerLayer = layer->AsContainerLayer(); if (!containerLayer) { return IPC_FAIL_NO_REASON(this); } const ContainerLayerAttributes& attrs = specific.get_ContainerLayerAttributes(); containerLayer->SetPreScale(attrs.preXScale(), attrs.preYScale()); containerLayer->SetInheritedScale(attrs.inheritedXScale(), attrs.inheritedYScale()); containerLayer->SetScaleToResolution(attrs.scaleToResolution(), attrs.presShellResolution()); containerLayer->SetEventRegionsOverride(attrs.eventRegionsOverride()); break; } case Specific::TColorLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] color layer")); ColorLayer* colorLayer = layer->AsColorLayer(); if (!colorLayer) { return IPC_FAIL_NO_REASON(this); } colorLayer->SetColor(specific.get_ColorLayerAttributes().color().value()); colorLayer->SetBounds(specific.get_ColorLayerAttributes().bounds()); break; } case Specific::TTextLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] text layer")); TextLayer* textLayer = layer->AsTextLayer(); if (!textLayer) { return IPC_FAIL_NO_REASON(this); } const auto& tla = specific.get_TextLayerAttributes(); textLayer->SetBounds(tla.bounds()); textLayer->SetGlyphs(Move(const_cast&>(tla.glyphs()))); textLayer->SetScaledFont(reinterpret_cast(tla.scaledFont())); break; } case Specific::TBorderLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] border layer")); BorderLayer* borderLayer = layer->AsBorderLayer(); if (!borderLayer) { return IPC_FAIL_NO_REASON(this); } borderLayer->SetRect(specific.get_BorderLayerAttributes().rect()); borderLayer->SetColors(specific.get_BorderLayerAttributes().colors()); borderLayer->SetCornerRadii(specific.get_BorderLayerAttributes().corners()); borderLayer->SetWidths(specific.get_BorderLayerAttributes().widths()); break; } case Specific::TCanvasLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] canvas layer")); CanvasLayer* canvasLayer = layer->AsCanvasLayer(); if (!canvasLayer) { return IPC_FAIL_NO_REASON(this); } canvasLayer->SetSamplingFilter(specific.get_CanvasLayerAttributes().samplingFilter()); canvasLayer->SetBounds(specific.get_CanvasLayerAttributes().bounds()); break; } case Specific::TRefLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] ref layer")); RefLayer* refLayer = layer->AsRefLayer(); if (!refLayer) { return IPC_FAIL_NO_REASON(this); } refLayer->SetReferentId(specific.get_RefLayerAttributes().id()); refLayer->SetEventRegionsOverride(specific.get_RefLayerAttributes().eventRegionsOverride()); break; } case Specific::TImageLayerAttributes: { MOZ_LAYERS_LOG(("[ParentSide] image layer")); ImageLayer* imageLayer = layer->AsImageLayer(); if (!imageLayer) { return IPC_FAIL_NO_REASON(this); } const ImageLayerAttributes& attrs = specific.get_ImageLayerAttributes(); imageLayer->SetSamplingFilter(attrs.samplingFilter()); imageLayer->SetScaleToSize(attrs.scaleToSize(), attrs.scaleMode()); break; } default: MOZ_CRASH("not reached"); } updateHitTestingTree = true; break; } case Edit::TOpSetDiagnosticTypes: { mLayerManager->GetCompositor()->SetDiagnosticTypes( edit.get_OpSetDiagnosticTypes().diagnostics()); break; } case Edit::TOpWindowOverlayChanged: { mLayerManager->SetWindowOverlayChanged(); break; } // Tree ops case Edit::TOpSetRoot: { MOZ_LAYERS_LOG(("[ParentSide] SetRoot")); Layer* newRoot = AsLayer(edit.get_OpSetRoot().root()); if (!newRoot) { return IPC_FAIL_NO_REASON(this); } if (newRoot->GetParent()) { // newRoot is not a root! return IPC_FAIL_NO_REASON(this); } mRoot = newRoot; updateHitTestingTree = true; break; } case Edit::TOpInsertAfter: { MOZ_LAYERS_LOG(("[ParentSide] InsertAfter")); const OpInsertAfter& oia = edit.get_OpInsertAfter(); Layer* child = AsLayer(oia.childLayer()); Layer* layer = AsLayer(oia.container()); Layer* after = AsLayer(oia.after()); if (!child || !layer || !after) { return IPC_FAIL_NO_REASON(this); } ContainerLayer* container = layer->AsContainerLayer(); if (!container || !container->InsertAfter(child, after)) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpPrependChild: { MOZ_LAYERS_LOG(("[ParentSide] PrependChild")); const OpPrependChild& oac = edit.get_OpPrependChild(); Layer* child = AsLayer(oac.childLayer()); Layer* layer = AsLayer(oac.container()); if (!child || !layer) { return IPC_FAIL_NO_REASON(this); } ContainerLayer* container = layer->AsContainerLayer(); if (!container || !container->InsertAfter(child, nullptr)) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpRemoveChild: { MOZ_LAYERS_LOG(("[ParentSide] RemoveChild")); const OpRemoveChild& orc = edit.get_OpRemoveChild(); Layer* childLayer = AsLayer(orc.childLayer()); Layer* layer = AsLayer(orc.container()); if (!childLayer || !layer) { return IPC_FAIL_NO_REASON(this); } ContainerLayer* container = layer->AsContainerLayer(); if (!container || !container->RemoveChild(childLayer)) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpRepositionChild: { MOZ_LAYERS_LOG(("[ParentSide] RepositionChild")); const OpRepositionChild& orc = edit.get_OpRepositionChild(); Layer* child = AsLayer(orc.childLayer()); Layer* after = AsLayer(orc.after()); Layer* layer = AsLayer(orc.container()); if (!child || !layer || !after) { return IPC_FAIL_NO_REASON(this); } ContainerLayer* container = layer->AsContainerLayer(); if (!container || !container->RepositionChild(child, after)) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TOpRaiseToTopChild: { MOZ_LAYERS_LOG(("[ParentSide] RaiseToTopChild")); const OpRaiseToTopChild& rtc = edit.get_OpRaiseToTopChild(); Layer* child = AsLayer(rtc.childLayer()); if (!child) { return IPC_FAIL_NO_REASON(this); } Layer* layer = AsLayer(rtc.container()); if (!layer) { return IPC_FAIL_NO_REASON(this); } ContainerLayer* container = layer->AsContainerLayer(); if (!container || !container->RepositionChild(child, nullptr)) { return IPC_FAIL_NO_REASON(this); } updateHitTestingTree = true; break; } case Edit::TCompositableOperation: { if (!ReceiveCompositableUpdate(edit.get_CompositableOperation(), replyv)) { return IPC_FAIL_NO_REASON(this); } break; } case Edit::TOpAttachCompositable: { const OpAttachCompositable& op = edit.get_OpAttachCompositable(); CompositableHost* host = CompositableHost::FromIPDLActor(op.compositableParent()); if (mPendingCompositorUpdates) { // Do not attach compositables from old layer trees. Return true since // content cannot handle errors. return IPC_OK(); } if (!Attach(AsLayer(op.layer()), host, false)) { return IPC_FAIL_NO_REASON(this); } if (mLayerManager->GetCompositor()) { host->SetCompositorID(mLayerManager->GetCompositor()->GetCompositorID()); } break; } case Edit::TOpAttachAsyncCompositable: { const OpAttachAsyncCompositable& op = edit.get_OpAttachAsyncCompositable(); if (mPendingCompositorUpdates) { // Do not attach compositables from old layer trees. Return true since // content cannot handle errors. return IPC_OK(); } ImageBridgeParent* imageBridge = ImageBridgeParent::GetInstance(OtherPid()); if (!imageBridge) { return IPC_FAIL_NO_REASON(this); } CompositableHost* host = imageBridge->FindCompositable(op.compositable()); if (!host) { NS_ERROR("CompositableHost not found in the map"); return IPC_FAIL_NO_REASON(this); } if (!Attach(AsLayer(op.layer()), host, true)) { return IPC_FAIL_NO_REASON(this); } if (mLayerManager->GetCompositor()) { host->SetCompositorID(mLayerManager->GetCompositor()->GetCompositorID()); } break; } default: MOZ_CRASH("not reached"); } } mCompositorBridge->ShadowLayersUpdated(this, aInfo, updateHitTestingTree); { AutoResolveRefLayers resolve(mCompositorBridge->GetCompositionManager(this)); layer_manager()->EndTransaction(TimeStamp(), LayerManager::END_NO_IMMEDIATE_REDRAW); } if (reply) { reply->SetCapacity(replyv.size()); if (replyv.size() > 0) { reply->AppendElements(&replyv.front(), replyv.size()); } } if (!IsSameProcess()) { // Ensure that any pending operations involving back and front // buffers have completed, so that neither process stomps on the // other's buffer contents. LayerManagerComposite::PlatformSyncBeforeReplyUpdate(); } #ifdef COMPOSITOR_PERFORMANCE_WARNING int compositeTime = (int)(mozilla::TimeStamp::Now() - updateStart).ToMilliseconds(); if (compositeTime > 15) { printf_stderr("Compositor: Layers update took %i ms (blocking gecko).\n", compositeTime); } #endif // Enable visual warning for long transaction when draw FPS option is enabled bool drawFps = gfxPrefs::LayersDrawFPS(); if (drawFps) { uint32_t visualWarningTrigger = gfxPrefs::LayerTransactionWarning(); // The default theshold is 200ms to trigger, hit red when it take 4 times longer TimeDuration latency = TimeStamp::Now() - aInfo.transactionStart(); if (latency > TimeDuration::FromMilliseconds(visualWarningTrigger)) { float severity = (latency - TimeDuration::FromMilliseconds(visualWarningTrigger)).ToMilliseconds() / (4 * visualWarningTrigger); if (severity > 1.f) { severity = 1.f; } mLayerManager->VisualFrameWarning(severity); PR_LogPrint("LayerTransactionParent::RecvUpdate transaction from process %d took %f ms", OtherPid(), latency.ToMilliseconds()); } } return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvSetLayerObserverEpoch(const uint64_t& aLayerObserverEpoch) { mChildEpoch = aLayerObserverEpoch; return IPC_OK(); } bool LayerTransactionParent::ShouldParentObserveEpoch() { if (mParentEpoch == mChildEpoch) { return false; } mParentEpoch = mChildEpoch; return true; } mozilla::ipc::IPCResult LayerTransactionParent::RecvSetTestSampleTime(const TimeStamp& aTime) { if (!mCompositorBridge->SetTestSampleTime(this, aTime)) { return IPC_FAIL_NO_REASON(this); } return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvLeaveTestMode() { mCompositorBridge->LeaveTestMode(this); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvGetAnimationOpacity(const LayerHandle& aParent, float* aOpacity, bool* aHasAnimationOpacity) { *aHasAnimationOpacity = false; if (mDestroyed || !layer_manager() || layer_manager()->IsDestroyed()) { return IPC_FAIL_NO_REASON(this); } RefPtr layer = AsLayer(aParent); if (!layer) { return IPC_FAIL_NO_REASON(this); } mCompositorBridge->ApplyAsyncProperties(this); if (!layer->AsHostLayer()->GetShadowOpacitySetByAnimation()) { return IPC_OK(); } *aOpacity = layer->GetLocalOpacity(); *aHasAnimationOpacity = true; return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvGetAnimationTransform(const LayerHandle& aLayerHandle, MaybeTransform* aTransform) { if (mDestroyed || !layer_manager() || layer_manager()->IsDestroyed()) { return IPC_FAIL_NO_REASON(this); } Layer* layer = AsLayer(aLayerHandle); if (!layer) { return IPC_FAIL_NO_REASON(this); } // Make sure we apply the latest animation style or else we can end up with // a race between when we temporarily clear the animation transform (in // CompositorBridgeParent::SetShadowProperties) and when animation recalculates // the value. mCompositorBridge->ApplyAsyncProperties(this); // This method is specific to transforms applied by animation. // This is because this method uses the information stored with an animation // such as the origin of the reference frame corresponding to the layer, to // recover the untranslated transform from the shadow transform. For // transforms that are not set by animation we don't have this information // available. if (!layer->AsHostLayer()->GetShadowTransformSetByAnimation()) { *aTransform = mozilla::void_t(); return IPC_OK(); } // The following code recovers the untranslated transform // from the shadow transform by undoing the translations in // AsyncCompositionManager::SampleValue. Matrix4x4 transform = layer->AsHostLayer()->GetShadowBaseTransform(); if (ContainerLayer* c = layer->AsContainerLayer()) { // Undo the scale transform applied by AsyncCompositionManager::SampleValue transform.PostScale(1.0f/c->GetInheritedXScale(), 1.0f/c->GetInheritedYScale(), 1.0f); } float scale = 1; Point3D scaledOrigin; Point3D transformOrigin; for (uint32_t i=0; i < layer->GetAnimations().Length(); i++) { if (layer->GetAnimations()[i].data().type() == AnimationData::TTransformData) { const TransformData& data = layer->GetAnimations()[i].data().get_TransformData(); scale = data.appUnitsPerDevPixel(); scaledOrigin = Point3D(NS_round(NSAppUnitsToFloatPixels(data.origin().x, scale)), NS_round(NSAppUnitsToFloatPixels(data.origin().y, scale)), 0.0f); transformOrigin = data.transformOrigin(); break; } } // If our parent isn't a perspective layer, then the offset into reference // frame coordinates will have been applied to us. Add an inverse translation // to cancel it out. if (!layer->GetParent() || !layer->GetParent()->GetTransformIsPerspective()) { transform.PostTranslate(-scaledOrigin.x, -scaledOrigin.y, -scaledOrigin.z); } // Undo the rebasing applied by // nsDisplayTransform::GetResultingTransformMatrixInternal transform.ChangeBasis(-transformOrigin); // Convert to CSS pixels (this undoes the operations performed by // nsStyleTransformMatrix::ProcessTranslatePart which is called from // nsDisplayTransform::GetResultingTransformMatrix) double devPerCss = double(scale) / double(nsDeviceContext::AppUnitsPerCSSPixel()); transform._41 *= devPerCss; transform._42 *= devPerCss; transform._43 *= devPerCss; *aTransform = transform; return IPC_OK(); } static AsyncPanZoomController* GetAPZCForViewID(Layer* aLayer, FrameMetrics::ViewID aScrollID) { AsyncPanZoomController* resultApzc = nullptr; ForEachNode( aLayer, [aScrollID, &resultApzc] (Layer* layer) { for (uint32_t i = 0; i < layer->GetScrollMetadataCount(); i++) { if (layer->GetFrameMetrics(i).GetScrollId() == aScrollID) { resultApzc = layer->GetAsyncPanZoomController(i); return TraversalFlag::Abort; } } return TraversalFlag::Continue; }); return resultApzc; } mozilla::ipc::IPCResult LayerTransactionParent::RecvSetAsyncScrollOffset(const FrameMetrics::ViewID& aScrollID, const float& aX, const float& aY) { if (mDestroyed || !layer_manager() || layer_manager()->IsDestroyed()) { return IPC_FAIL_NO_REASON(this); } AsyncPanZoomController* controller = GetAPZCForViewID(mRoot, aScrollID); if (!controller) { return IPC_FAIL_NO_REASON(this); } controller->SetTestAsyncScrollOffset(CSSPoint(aX, aY)); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvSetAsyncZoom(const FrameMetrics::ViewID& aScrollID, const float& aValue) { if (mDestroyed || !layer_manager() || layer_manager()->IsDestroyed()) { return IPC_FAIL_NO_REASON(this); } AsyncPanZoomController* controller = GetAPZCForViewID(mRoot, aScrollID); if (!controller) { return IPC_FAIL_NO_REASON(this); } controller->SetTestAsyncZoom(LayerToParentLayerScale(aValue)); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvFlushApzRepaints() { mCompositorBridge->FlushApzRepaints(this); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvGetAPZTestData(APZTestData* aOutData) { mCompositorBridge->GetAPZTestData(this, aOutData); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvRequestProperty(const nsString& aProperty, float* aValue) { if (aProperty.Equals(NS_LITERAL_STRING("overdraw"))) { *aValue = layer_manager()->GetCompositor()->GetFillRatio(); } else { *aValue = -1; } return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvSetConfirmedTargetAPZC(const uint64_t& aBlockId, nsTArray&& aTargets) { mCompositorBridge->SetConfirmedTargetAPZC(this, aBlockId, aTargets); return IPC_OK(); } bool LayerTransactionParent::Attach(Layer* aLayer, CompositableHost* aCompositable, bool aIsAsync) { if (!aCompositable || !aLayer) { return false; } HostLayer* layer = aLayer->AsHostLayer(); if (!layer) { return false; } Compositor* compositor = static_cast(aLayer->Manager())->GetCompositor(); if (!layer->SetCompositableHost(aCompositable)) { // not all layer types accept a compositable, see bug 967824 return false; } aCompositable->Attach(aLayer, compositor, aIsAsync ? CompositableHost::ALLOW_REATTACH | CompositableHost::KEEP_ATTACHED : CompositableHost::NO_FLAGS); return true; } mozilla::ipc::IPCResult LayerTransactionParent::RecvClearCachedResources() { if (mRoot) { // NB: |mRoot| here is the *child* context's root. In this parent // context, it's just a subtree root. We need to scope the clear // of resources to exactly that subtree, so we specify it here. mLayerManager->ClearCachedResources(mRoot); } mCompositorBridge->NotifyClearCachedResources(this); return IPC_OK(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvForceComposite() { mCompositorBridge->ForceComposite(this); return IPC_OK(); } PCompositableParent* LayerTransactionParent::AllocPCompositableParent(const TextureInfo& aInfo) { return CompositableHost::CreateIPDLActor(this, aInfo); } bool LayerTransactionParent::DeallocPCompositableParent(PCompositableParent* aActor) { return CompositableHost::DestroyIPDLActor(aActor); } void LayerTransactionParent::ActorDestroy(ActorDestroyReason why) { } bool LayerTransactionParent::AllocShmem(size_t aSize, ipc::SharedMemory::SharedMemoryType aType, ipc::Shmem* aShmem) { if (!mIPCOpen || mDestroyed) { return false; } return PLayerTransactionParent::AllocShmem(aSize, aType, aShmem); } bool LayerTransactionParent::AllocUnsafeShmem(size_t aSize, ipc::SharedMemory::SharedMemoryType aType, ipc::Shmem* aShmem) { if (!mIPCOpen || mDestroyed) { return false; } return PLayerTransactionParent::AllocUnsafeShmem(aSize, aType, aShmem); } void LayerTransactionParent::DeallocShmem(ipc::Shmem& aShmem) { if (!mIPCOpen || mDestroyed) { return; } PLayerTransactionParent::DeallocShmem(aShmem); } bool LayerTransactionParent::IsSameProcess() const { return OtherPid() == base::GetCurrentProcId(); } void LayerTransactionParent::SendAsyncMessage(const InfallibleTArray& aMessage) { MOZ_ASSERT_UNREACHABLE("unexpected to be called"); } void LayerTransactionParent::SendPendingAsyncMessages() { mCompositorBridge->SendPendingAsyncMessages(); } void LayerTransactionParent::SetAboutToSendAsyncMessages() { mCompositorBridge->SetAboutToSendAsyncMessages(); } void LayerTransactionParent::NotifyNotUsed(PTextureParent* aTexture, uint64_t aTransactionId) { MOZ_ASSERT_UNREACHABLE("unexpected to be called"); } bool LayerTransactionParent::BindLayerToHandle(RefPtr aLayer, const LayerHandle& aHandle) { if (!aHandle || !aLayer || mLayerMap.Contains(aHandle.Value())) { return false; } mLayerMap.Put(aHandle.Value(), aLayer); return true; } Layer* LayerTransactionParent::AsLayer(const LayerHandle& aHandle) { if (!aHandle) { return nullptr; } return mLayerMap.Get(aHandle.Value()).get(); } mozilla::ipc::IPCResult LayerTransactionParent::RecvReleaseLayer(const LayerHandle& aHandle) { if (!aHandle || !mLayerMap.Contains(aHandle.Value())) { return IPC_FAIL_NO_REASON(this); } Maybe> maybeLayer = mLayerMap.GetAndRemove(aHandle.Value()); if (maybeLayer) { (*maybeLayer)->Disconnect(); } return IPC_OK(); } } // namespace layers } // namespace mozilla