mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-13 07:24:47 +00:00
311 lines
8.5 KiB
C++
311 lines
8.5 KiB
C++
/* -*- 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/LayersSurfaces.h" // for SurfaceDescriptor
|
|
#include "mozilla/layers/TextureHost.h" // for TextureHost, etc
|
|
#include "nsAutoPtr.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"
|
|
|
|
namespace mozilla {
|
|
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
|
|
* goind down.
|
|
*/
|
|
class CompositableParent : public PCompositableParent
|
|
{
|
|
public:
|
|
CompositableParent(CompositableParentManager* aMgr,
|
|
const TextureInfo& aTextureInfo,
|
|
uint64_t aID = 0)
|
|
{
|
|
MOZ_COUNT_CTOR(CompositableParent);
|
|
mHost = CompositableHost::Create(aTextureInfo);
|
|
mHost->SetAsyncID(aID);
|
|
if (aID) {
|
|
CompositableMap::Set(aID, this);
|
|
}
|
|
}
|
|
|
|
~CompositableParent()
|
|
{
|
|
MOZ_COUNT_DTOR(CompositableParent);
|
|
CompositableMap::Erase(mHost->GetAsyncID());
|
|
}
|
|
|
|
virtual void ActorDestroy(ActorDestroyReason why) MOZ_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);
|
|
if (mBackendData) {
|
|
mBackendData->ClearData();
|
|
}
|
|
}
|
|
|
|
PCompositableParent*
|
|
CompositableHost::CreateIPDLActor(CompositableParentManager* aMgr,
|
|
const TextureInfo& aTextureInfo,
|
|
uint64_t aID)
|
|
{
|
|
return new CompositableParent(aMgr, aTextureInfo, aID);
|
|
}
|
|
|
|
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(TextureHost* aTexture)
|
|
{
|
|
if (!aTexture) {
|
|
return;
|
|
}
|
|
aTexture->SetCompositor(GetCompositor());
|
|
aTexture->SetCompositableBackendSpecificData(GetCompositableBackendSpecificData());
|
|
}
|
|
|
|
void
|
|
CompositableHost::UseComponentAlphaTextures(TextureHost* aTextureOnBlack,
|
|
TextureHost* aTextureOnWhite)
|
|
{
|
|
MOZ_ASSERT(aTextureOnBlack && aTextureOnWhite);
|
|
aTextureOnBlack->SetCompositor(GetCompositor());
|
|
aTextureOnBlack->SetCompositableBackendSpecificData(GetCompositableBackendSpecificData());
|
|
aTextureOnWhite->SetCompositor(GetCompositor());
|
|
aTextureOnWhite->SetCompositableBackendSpecificData(GetCompositableBackendSpecificData());
|
|
}
|
|
|
|
void
|
|
CompositableHost::RemoveTextureHost(TextureHost* aTexture)
|
|
{
|
|
// Clear strong refrence to CompositableBackendSpecificData
|
|
aTexture->SetCompositableBackendSpecificData(nullptr);
|
|
}
|
|
|
|
void
|
|
CompositableHost::SetCompositor(Compositor* aCompositor)
|
|
{
|
|
mCompositor = aCompositor;
|
|
}
|
|
|
|
bool
|
|
CompositableHost::AddMaskEffect(EffectChain& aEffects,
|
|
const gfx::Matrix4x4& aTransform,
|
|
bool aIs3D)
|
|
{
|
|
RefPtr<TextureSource> 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;
|
|
}
|
|
|
|
source = host->GetTextureSources();
|
|
MOZ_ASSERT(source);
|
|
|
|
if (!source) {
|
|
NS_WARNING("The TextureHost was successfully locked but can't provide a TextureSource");
|
|
host->Unlock();
|
|
return false;
|
|
}
|
|
|
|
RefPtr<EffectMask> effect = new EffectMask(source,
|
|
source->GetSize(),
|
|
aTransform);
|
|
effect->mIs3D = aIs3D;
|
|
aEffects.mSecondaryEffects[EffectTypes::MASK] = effect;
|
|
return true;
|
|
}
|
|
|
|
void
|
|
CompositableHost::RemoveMaskEffect()
|
|
{
|
|
RefPtr<TextureHost> host = GetAsTextureHost();
|
|
if (host) {
|
|
host->Unlock();
|
|
}
|
|
}
|
|
|
|
// implemented in TextureHostOGL.cpp
|
|
TemporaryRef<CompositableBackendSpecificData> CreateCompositableBackendSpecificDataOGL();
|
|
|
|
/* static */ TemporaryRef<CompositableHost>
|
|
CompositableHost::Create(const TextureInfo& aTextureInfo)
|
|
{
|
|
RefPtr<CompositableHost> result;
|
|
switch (aTextureInfo.mCompositableType) {
|
|
case CompositableType::BUFFER_BRIDGE:
|
|
NS_ERROR("Cannot create an image bridge compositable this way");
|
|
break;
|
|
case CompositableType::BUFFER_CONTENT_INC:
|
|
result = new ContentHostIncremental(aTextureInfo);
|
|
break;
|
|
case CompositableType::BUFFER_TILED:
|
|
case CompositableType::BUFFER_SIMPLE_TILED:
|
|
result = new TiledContentHost(aTextureInfo);
|
|
break;
|
|
case CompositableType::IMAGE:
|
|
result = new ImageHost(aTextureInfo);
|
|
break;
|
|
#ifdef MOZ_WIDGET_GONK
|
|
case CompositableType::IMAGE_OVERLAY:
|
|
result = new ImageHostOverlay(aTextureInfo);
|
|
break;
|
|
#endif
|
|
case CompositableType::CONTENT_SINGLE:
|
|
result = new ContentHostSingleBuffered(aTextureInfo);
|
|
break;
|
|
case CompositableType::CONTENT_DOUBLE:
|
|
result = new ContentHostDoubleBuffered(aTextureInfo);
|
|
break;
|
|
default:
|
|
NS_ERROR("Unknown CompositableType");
|
|
}
|
|
// We know that Tiled buffers don't use the compositable backend-specific
|
|
// data, so don't bother creating it.
|
|
if (result && aTextureInfo.mCompositableType != CompositableType::BUFFER_TILED) {
|
|
RefPtr<CompositableBackendSpecificData> data = CreateCompositableBackendSpecificDataOGL();
|
|
result->SetCompositableBackendSpecificData(data);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#ifdef MOZ_DUMP_PAINTING
|
|
void
|
|
CompositableHost::DumpTextureHost(std::stringstream& aStream, TextureHost* aTexture)
|
|
{
|
|
if (!aTexture) {
|
|
return;
|
|
}
|
|
RefPtr<gfx::DataSourceSurface> dSurf = aTexture->GetAsSurface();
|
|
if (!dSurf) {
|
|
return;
|
|
}
|
|
gfxPlatform *platform = gfxPlatform::GetPlatform();
|
|
RefPtr<gfx::DrawTarget> dt = platform->CreateDrawTargetForData(dSurf->GetData(),
|
|
dSurf->GetSize(),
|
|
dSurf->Stride(),
|
|
dSurf->GetFormat());
|
|
// TODO stream surface
|
|
gfxUtils::DumpAsDataURI(dt, stderr);
|
|
}
|
|
#endif
|
|
|
|
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
|