summaryrefslogtreecommitdiffstats
path: root/gfx/layers/composite/CompositableHost.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'gfx/layers/composite/CompositableHost.cpp')
-rw-r--r--gfx/layers/composite/CompositableHost.cpp292
1 files changed, 292 insertions, 0 deletions
diff --git a/gfx/layers/composite/CompositableHost.cpp b/gfx/layers/composite/CompositableHost.cpp
new file mode 100644
index 000000000..5ed3d3fe9
--- /dev/null
+++ b/gfx/layers/composite/CompositableHost.cpp
@@ -0,0 +1,292 @@
+/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * 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 "CompositableHost.h"
+#include <map> // for _Rb_tree_iterator, map, etc
+#include <utility> // for pair
+#include "ContentHost.h" // for ContentHostDoubleBuffered, etc
+#include "Effects.h" // for EffectMask, Effect, etc
+#include "gfxUtils.h"
+#include "ImageHost.h" // for ImageHostBuffered, etc
+#include "TiledContentHost.h" // for TiledContentHost
+#include "mozilla/layers/ImageContainerParent.h"
+#include "mozilla/layers/LayersSurfaces.h" // for SurfaceDescriptor
+#include "mozilla/layers/TextureHost.h" // for TextureHost, etc
+#include "mozilla/RefPtr.h" // for nsRefPtr
+#include "nsDebug.h" // for NS_WARNING
+#include "nsISupportsImpl.h" // for MOZ_COUNT_CTOR, etc
+#include "gfxPlatform.h" // for gfxPlatform
+#include "mozilla/layers/PCompositableParent.h"
+#include "IPDLActor.h"
+
+namespace mozilla {
+
+using namespace gfx;
+
+namespace layers {
+
+class Compositor;
+
+/**
+ * IPDL actor used by CompositableHost to match with its corresponding
+ * CompositableClient on the content side.
+ *
+ * CompositableParent is owned by the IPDL system. It's deletion is triggered
+ * by either the CompositableChild's deletion, or by the IPDL communication
+ * going down.
+ */
+class CompositableParent : public ParentActor<PCompositableParent>
+{
+public:
+ CompositableParent(CompositableParentManager* aMgr,
+ const TextureInfo& aTextureInfo,
+ uint64_t aID = 0,
+ PImageContainerParent* aImageContainer = nullptr)
+ {
+ MOZ_COUNT_CTOR(CompositableParent);
+ mHost = CompositableHost::Create(aTextureInfo);
+ mHost->SetAsyncID(aID);
+ if (aID) {
+ CompositableMap::Set(aID, this);
+ }
+ if (aImageContainer) {
+ mHost->SetImageContainer(
+ static_cast<ImageContainerParent*>(aImageContainer));
+ }
+ }
+
+ ~CompositableParent()
+ {
+ MOZ_COUNT_DTOR(CompositableParent);
+ CompositableMap::Erase(mHost->GetAsyncID());
+ }
+
+ virtual void Destroy() override
+ {
+ if (mHost) {
+ mHost->Detach(nullptr, CompositableHost::FORCE_DETACH);
+ }
+ }
+
+ RefPtr<CompositableHost> mHost;
+};
+
+CompositableHost::CompositableHost(const TextureInfo& aTextureInfo)
+ : mTextureInfo(aTextureInfo)
+ , mAsyncID(0)
+ , mCompositorID(0)
+ , mCompositor(nullptr)
+ , mLayer(nullptr)
+ , mFlashCounter(0)
+ , mAttached(false)
+ , mKeepAttached(false)
+{
+ MOZ_COUNT_CTOR(CompositableHost);
+}
+
+CompositableHost::~CompositableHost()
+{
+ MOZ_COUNT_DTOR(CompositableHost);
+}
+
+PCompositableParent*
+CompositableHost::CreateIPDLActor(CompositableParentManager* aMgr,
+ const TextureInfo& aTextureInfo,
+ uint64_t aID,
+ PImageContainerParent* aImageContainer)
+{
+ return new CompositableParent(aMgr, aTextureInfo, aID, aImageContainer);
+}
+
+bool
+CompositableHost::DestroyIPDLActor(PCompositableParent* aActor)
+{
+ delete aActor;
+ return true;
+}
+
+CompositableHost*
+CompositableHost::FromIPDLActor(PCompositableParent* aActor)
+{
+ MOZ_ASSERT(aActor);
+ return static_cast<CompositableParent*>(aActor)->mHost;
+}
+
+void
+CompositableHost::UseTextureHost(const nsTArray<TimedTexture>& aTextures)
+{
+ if (GetCompositor()) {
+ for (auto& texture : aTextures) {
+ texture.mTexture->SetCompositor(GetCompositor());
+ }
+ }
+}
+
+void
+CompositableHost::UseComponentAlphaTextures(TextureHost* aTextureOnBlack,
+ TextureHost* aTextureOnWhite)
+{
+ MOZ_ASSERT(aTextureOnBlack && aTextureOnWhite);
+ if (GetCompositor()) {
+ aTextureOnBlack->SetCompositor(GetCompositor());
+ aTextureOnWhite->SetCompositor(GetCompositor());
+ }
+}
+
+void
+CompositableHost::RemoveTextureHost(TextureHost* aTexture)
+{}
+
+void
+CompositableHost::SetCompositor(Compositor* aCompositor)
+{
+ MOZ_ASSERT(aCompositor);
+ mCompositor = aCompositor;
+}
+
+bool
+CompositableHost::AddMaskEffect(EffectChain& aEffects,
+ const gfx::Matrix4x4& aTransform)
+{
+ CompositableTextureSourceRef source;
+ RefPtr<TextureHost> host = GetAsTextureHost();
+
+ if (!host) {
+ NS_WARNING("Using compositable with no valid TextureHost as mask");
+ return false;
+ }
+
+ if (!host->Lock()) {
+ NS_WARNING("Failed to lock the mask texture");
+ return false;
+ }
+
+ if (!host->BindTextureSource(source)) {
+ NS_WARNING("The TextureHost was successfully locked but can't provide a TextureSource");
+ host->Unlock();
+ return false;
+ }
+ MOZ_ASSERT(source);
+
+ RefPtr<EffectMask> effect = new EffectMask(source,
+ source->GetSize(),
+ aTransform);
+ aEffects.mSecondaryEffects[EffectTypes::MASK] = effect;
+ return true;
+}
+
+void
+CompositableHost::RemoveMaskEffect()
+{
+ RefPtr<TextureHost> host = GetAsTextureHost();
+ if (host) {
+ host->Unlock();
+ }
+}
+
+/* static */ already_AddRefed<CompositableHost>
+CompositableHost::Create(const TextureInfo& aTextureInfo)
+{
+ RefPtr<CompositableHost> result;
+ switch (aTextureInfo.mCompositableType) {
+ case CompositableType::IMAGE_BRIDGE:
+ NS_ERROR("Cannot create an image bridge compositable this way");
+ break;
+ case CompositableType::CONTENT_TILED:
+ result = new TiledContentHost(aTextureInfo);
+ break;
+ case CompositableType::IMAGE:
+ result = new ImageHost(aTextureInfo);
+ break;
+ case CompositableType::CONTENT_SINGLE:
+ result = new ContentHostSingleBuffered(aTextureInfo);
+ break;
+ case CompositableType::CONTENT_DOUBLE:
+ result = new ContentHostDoubleBuffered(aTextureInfo);
+ break;
+ default:
+ NS_ERROR("Unknown CompositableType");
+ }
+ return result.forget();
+}
+
+void
+CompositableHost::DumpTextureHost(std::stringstream& aStream, TextureHost* aTexture)
+{
+ if (!aTexture) {
+ return;
+ }
+ RefPtr<gfx::DataSourceSurface> dSurf = aTexture->GetAsSurface();
+ if (!dSurf) {
+ return;
+ }
+ aStream << gfxUtils::GetAsDataURI(dSurf).get();
+}
+
+void
+CompositableHost::ReceivedDestroy(PCompositableParent* aActor)
+{
+ static_cast<CompositableParent*>(aActor)->RecvDestroy();
+}
+
+namespace CompositableMap {
+
+typedef std::map<uint64_t, PCompositableParent*> CompositableMap_t;
+static CompositableMap_t* sCompositableMap = nullptr;
+bool IsCreated() {
+ return sCompositableMap != nullptr;
+}
+PCompositableParent* Get(uint64_t aID)
+{
+ if (!IsCreated() || aID == 0) {
+ return nullptr;
+ }
+ CompositableMap_t::iterator it = sCompositableMap->find(aID);
+ if (it == sCompositableMap->end()) {
+ return nullptr;
+ }
+ return it->second;
+}
+void Set(uint64_t aID, PCompositableParent* aParent)
+{
+ if (!IsCreated() || aID == 0) {
+ return;
+ }
+ (*sCompositableMap)[aID] = aParent;
+}
+void Erase(uint64_t aID)
+{
+ if (!IsCreated() || aID == 0) {
+ return;
+ }
+ CompositableMap_t::iterator it = sCompositableMap->find(aID);
+ if (it != sCompositableMap->end()) {
+ sCompositableMap->erase(it);
+ }
+}
+void Clear()
+{
+ if (!IsCreated()) {
+ return;
+ }
+ sCompositableMap->clear();
+}
+void Create()
+{
+ if (sCompositableMap == nullptr) {
+ sCompositableMap = new CompositableMap_t;
+ }
+}
+void Destroy()
+{
+ Clear();
+ delete sCompositableMap;
+ sCompositableMap = nullptr;
+}
+
+} // namespace CompositableMap
+
+} // namespace layers
+} // namespace mozilla