mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-07 20:17:37 +00:00
294 lines
8.8 KiB
C++
294 lines
8.8 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/. */
|
|
|
|
#ifndef MOZILLA_GFX_TEXTURECLIENT_H
|
|
#define MOZILLA_GFX_TEXTURECLIENT_H
|
|
|
|
#include "mozilla/layers/LayersSurfaces.h"
|
|
#include "gfxASurface.h"
|
|
#include "mozilla/layers/CompositorTypes.h" // for TextureInfo
|
|
#include "mozilla/RefPtr.h"
|
|
|
|
class gfxReusableSurfaceWrapper;
|
|
|
|
namespace mozilla {
|
|
|
|
namespace gl {
|
|
class GLContext;
|
|
}
|
|
|
|
namespace layers {
|
|
|
|
class ContentClient;
|
|
class PlanarYCbCrImage;
|
|
class Image;
|
|
class CompositableForwarder;
|
|
|
|
/**
|
|
* This class allows texture clients to draw into textures through Azure or
|
|
* thebes and applies locking semantics to allow GPU or CPU level
|
|
* synchronization.
|
|
* TextureClient's purpose is for the texture data to be
|
|
* forwarded to the right place on the compositor side and with correct locking
|
|
* semantics.
|
|
*
|
|
* When modifying a TextureClient's data, first call LockDescriptor, modify the
|
|
* data in the descriptor, and then call Unlock. This makes sure that if the
|
|
* data is shared with the compositor, the later will not try to read while the
|
|
* data is being modified (on the other side, TextureHost also has Lock/Unlock
|
|
* semantics).
|
|
* After unlocking, call Updated in order to add the modification to the current
|
|
* layer transaction.
|
|
* Depending on whether the data is shared or copied, Lock/Unlock and Updated
|
|
* can be no-ops. What's important is that the Client/Host pair implement the
|
|
* same semantics.
|
|
*
|
|
* Ownership of the surface descriptor depends on how the TextureClient/Host is
|
|
* used by the CompositableClient/Host.
|
|
*/
|
|
class TextureClient : public RefCounted<TextureClient>
|
|
{
|
|
public:
|
|
typedef gl::SharedTextureHandle SharedTextureHandle;
|
|
typedef gl::GLContext GLContext;
|
|
typedef gl::TextureImage TextureImage;
|
|
|
|
virtual ~TextureClient();
|
|
|
|
/* This will return an identifier that can be sent accross a process or
|
|
* thread boundary and used to construct a TextureHost object
|
|
* which can then be used as a texture for rendering by a compatible
|
|
* compositor. This texture should have been created with the
|
|
* TextureHostIdentifier specified by the compositor that this identifier
|
|
* is to be used with.
|
|
*/
|
|
virtual const TextureInfo& GetTextureInfo() const
|
|
{
|
|
return mTextureInfo;
|
|
}
|
|
|
|
virtual bool SupportsType(TextureClientType aType) { return false; }
|
|
|
|
/**
|
|
* The Lock* methods lock the texture client for drawing into, providing some
|
|
* object that can be used for drawing to. Once the user is finished
|
|
* with the object it should call Unlock.
|
|
*/
|
|
virtual gfxImageSurface* LockImageSurface() { return nullptr; }
|
|
virtual gfxASurface* LockSurface() { return nullptr; }
|
|
virtual gfx::DrawTarget* LockDrawTarget() { return nullptr; }
|
|
|
|
virtual SurfaceDescriptor* LockSurfaceDescriptor() { return GetDescriptor(); }
|
|
virtual void ReleaseResources() {}
|
|
/**
|
|
* This unlocks the current DrawableTexture and allows the host to composite
|
|
* it directly.
|
|
*/
|
|
virtual void Unlock() {}
|
|
|
|
/**
|
|
* Ensure that the texture client is suitable for the given size and content
|
|
* type and that any initialisation has taken place.
|
|
*/
|
|
virtual void EnsureAllocated(gfx::IntSize aSize,
|
|
gfxASurface::gfxContentType aType) = 0;
|
|
|
|
/**
|
|
* _Only_ used at the end of the layer transaction when receiving a reply from
|
|
* the compositor.
|
|
*/
|
|
virtual void SetDescriptorFromReply(const SurfaceDescriptor& aDescriptor)
|
|
{
|
|
// default implementation
|
|
SetDescriptor(aDescriptor);
|
|
}
|
|
virtual void SetDescriptor(const SurfaceDescriptor& aDescriptor)
|
|
{
|
|
mDescriptor = aDescriptor;
|
|
}
|
|
SurfaceDescriptor* GetDescriptor() { return &mDescriptor; }
|
|
|
|
CompositableForwarder* GetForwarder() const
|
|
{
|
|
return mForwarder;
|
|
}
|
|
|
|
void SetFlags(TextureFlags aFlags)
|
|
{
|
|
mTextureInfo.mTextureFlags = aFlags;
|
|
}
|
|
|
|
enum AccessMode
|
|
{
|
|
ACCESS_NONE = 0x0,
|
|
ACCESS_READ_ONLY = 0x1,
|
|
ACCESS_READ_WRITE = 0x2
|
|
};
|
|
|
|
void SetAccessMode(AccessMode aAccessMode)
|
|
{
|
|
mAccessMode = aAccessMode;
|
|
}
|
|
|
|
AccessMode GetAccessMode() const
|
|
{
|
|
return mAccessMode;
|
|
}
|
|
|
|
virtual gfxASurface::gfxContentType GetContentType() = 0;
|
|
|
|
protected:
|
|
TextureClient(CompositableForwarder* aForwarder,
|
|
const TextureInfo& aTextureInfo);
|
|
|
|
CompositableForwarder* mForwarder;
|
|
// So far all TextureClients use a SurfaceDescriptor, so it makes sense to
|
|
// keep the reference here.
|
|
SurfaceDescriptor mDescriptor;
|
|
TextureInfo mTextureInfo;
|
|
AccessMode mAccessMode;
|
|
};
|
|
|
|
class TextureClientShmem : public TextureClient
|
|
{
|
|
public:
|
|
TextureClientShmem(CompositableForwarder* aForwarder, const TextureInfo& aTextureInfo);
|
|
~TextureClientShmem() { ReleaseResources(); }
|
|
|
|
virtual bool SupportsType(TextureClientType aType) MOZ_OVERRIDE
|
|
{
|
|
return aType == TEXTURE_SHMEM || aType == TEXTURE_CONTENT;
|
|
}
|
|
virtual gfxImageSurface* LockImageSurface() MOZ_OVERRIDE;
|
|
virtual gfxASurface* LockSurface() MOZ_OVERRIDE { return GetSurface(); }
|
|
virtual void Unlock() MOZ_OVERRIDE;
|
|
virtual void EnsureAllocated(gfx::IntSize aSize, gfxASurface::gfxContentType aType) MOZ_OVERRIDE;
|
|
|
|
virtual void ReleaseResources() MOZ_OVERRIDE;
|
|
virtual void SetDescriptor(const SurfaceDescriptor& aDescriptor) MOZ_OVERRIDE;
|
|
virtual gfxASurface::gfxContentType GetContentType() MOZ_OVERRIDE { return mContentType; }
|
|
private:
|
|
gfxASurface* GetSurface();
|
|
|
|
nsRefPtr<gfxASurface> mSurface;
|
|
nsRefPtr<gfxImageSurface> mSurfaceAsImage;
|
|
|
|
gfxASurface::gfxContentType mContentType;
|
|
gfx::IntSize mSize;
|
|
|
|
friend class CompositingFactory;
|
|
};
|
|
|
|
class TextureClientShmemYCbCr : public TextureClient
|
|
{
|
|
public:
|
|
TextureClientShmemYCbCr(CompositableForwarder* aForwarder, const TextureInfo& aTextureInfo)
|
|
: TextureClient(aForwarder, aTextureInfo)
|
|
{ }
|
|
~TextureClientShmemYCbCr() { ReleaseResources(); }
|
|
|
|
virtual bool SupportsType(TextureClientType aType) MOZ_OVERRIDE { return aType == TEXTURE_YCBCR; }
|
|
void EnsureAllocated(gfx::IntSize aSize, gfxASurface::gfxContentType aType) MOZ_OVERRIDE;
|
|
virtual void SetDescriptorFromReply(const SurfaceDescriptor& aDescriptor) MOZ_OVERRIDE;
|
|
virtual void SetDescriptor(const SurfaceDescriptor& aDescriptor) MOZ_OVERRIDE;
|
|
virtual void ReleaseResources();
|
|
virtual gfxASurface::gfxContentType GetContentType() MOZ_OVERRIDE { return gfxASurface::CONTENT_COLOR_ALPHA; }
|
|
};
|
|
|
|
class TextureClientTile : public TextureClient
|
|
{
|
|
public:
|
|
TextureClientTile(const TextureClientTile& aOther);
|
|
TextureClientTile(CompositableForwarder* aForwarder,
|
|
const TextureInfo& aTextureInfo);
|
|
~TextureClientTile();
|
|
|
|
virtual void EnsureAllocated(gfx::IntSize aSize,
|
|
gfxASurface::gfxContentType aType) MOZ_OVERRIDE;
|
|
|
|
virtual gfxImageSurface* LockImageSurface() MOZ_OVERRIDE;
|
|
|
|
gfxReusableSurfaceWrapper* GetReusableSurfaceWrapper()
|
|
{
|
|
return mSurface;
|
|
}
|
|
|
|
virtual void SetDescriptor(const SurfaceDescriptor& aDescriptor) MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(false, "Tiled texture clients don't use SurfaceDescriptors.");
|
|
}
|
|
|
|
|
|
virtual gfxASurface::gfxContentType GetContentType() { return mContentType; }
|
|
|
|
private:
|
|
gfxASurface::gfxContentType mContentType;
|
|
nsRefPtr<gfxReusableSurfaceWrapper> mSurface;
|
|
|
|
friend class CompositingFactory;
|
|
};
|
|
|
|
/*
|
|
* The logic of converting input image data into a Surface descriptor should be
|
|
* outside of TextureClient. For Image layers we implement them in the AutoLock*
|
|
* idiom so that the states need for the purpose of convertion only exist within
|
|
* the conversion operation, and to avoid adding special interfaces in
|
|
* TextureClient tht are only used in one place and not implemented everywhere.
|
|
* We should do this for all the input data type.
|
|
*/
|
|
|
|
/**
|
|
* Base class for AutoLock*Client.
|
|
* handles lock/unlock
|
|
*/
|
|
class AutoLockTextureClient
|
|
{
|
|
public:
|
|
AutoLockTextureClient(TextureClient* aTexture)
|
|
{
|
|
mTextureClient = aTexture;
|
|
mDescriptor = aTexture->LockSurfaceDescriptor();
|
|
}
|
|
|
|
SurfaceDescriptor* GetSurfaceDescriptor()
|
|
{
|
|
return mDescriptor;
|
|
}
|
|
|
|
virtual ~AutoLockTextureClient()
|
|
{
|
|
mTextureClient->Unlock();
|
|
}
|
|
protected:
|
|
TextureClient* mTextureClient;
|
|
SurfaceDescriptor* mDescriptor;
|
|
};
|
|
|
|
/**
|
|
* Writes the content of a PlanarYCbCrImage into a SurfaceDescriptor.
|
|
*/
|
|
class AutoLockYCbCrClient : public AutoLockTextureClient
|
|
{
|
|
public:
|
|
AutoLockYCbCrClient(TextureClient* aTexture) : AutoLockTextureClient(aTexture) {}
|
|
bool Update(PlanarYCbCrImage* aImage);
|
|
protected:
|
|
bool EnsureTextureClient(PlanarYCbCrImage* aImage);
|
|
};
|
|
|
|
/**
|
|
* Writes the content of a gfxASurface into a SurfaceDescriptor.
|
|
*/
|
|
class AutoLockShmemClient : public AutoLockTextureClient
|
|
{
|
|
public:
|
|
AutoLockShmemClient(TextureClient* aTexture) : AutoLockTextureClient(aTexture) {}
|
|
bool Update(Image* aImage, uint32_t aContentFlags, gfxPattern* pat);
|
|
};
|
|
|
|
}
|
|
}
|
|
#endif
|