mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-04 16:15:25 +00:00
8631de9c2b
Backed out changeset 5be7514914b6 (bug 709490) Backed out changeset 04b6f94fbe8a (bug 709490) Backed out changeset 00c0e85dd8cd (bug 709490) Backed out changeset 221385b7b81a (bug 709490) Backed out changeset ecc38c18734f (bug 709490) Backed out changeset 22878c936384 (bug 709490) Backed out changeset 0edcbb60eee3 (bug 709490) Backed out changeset 5feceec2014b (bug 709490) Backed out changeset 835b655cb873 (bug 709490) Backed out changeset 6fbb4a3f8cf7 (bug 709490) Backed out changeset a5f8646fa156 (bug 709490) Backed out changeset 2ae1386916b3 (bug 709490) Backed out changeset 6b29a2a0a8fb (bug 709490)
196 lines
6.2 KiB
C++
196 lines
6.2 KiB
C++
/* -*- Mode: C++; tab-width: 2; 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 "CopyableCanvasLayer.h"
|
|
|
|
#include "BasicLayersImpl.h" // for FillWithMask, etc
|
|
#include "GLContext.h" // for GLContext
|
|
#include "GLScreenBuffer.h" // for GLScreenBuffer
|
|
#include "SharedSurface.h" // for SharedSurface
|
|
#include "SharedSurfaceGL.h" // for SharedSurface
|
|
#include "gfxPattern.h" // for gfxPattern, etc
|
|
#include "gfxPlatform.h" // for gfxPlatform, gfxImageFormat
|
|
#include "gfxRect.h" // for gfxRect
|
|
#include "gfxUtils.h" // for gfxUtils
|
|
#include "gfx2DGlue.h" // for thebes --> moz2d transition
|
|
#include "mozilla/gfx/BaseSize.h" // for BaseSize
|
|
#include "mozilla/gfx/Tools.h"
|
|
#include "mozilla/gfx/Point.h" // for IntSize
|
|
#include "mozilla/layers/PersistentBufferProvider.h"
|
|
#include "nsDebug.h" // for NS_ASSERTION, NS_WARNING, etc
|
|
#include "nsISupportsImpl.h" // for gfxContext::AddRef, etc
|
|
#include "nsRect.h" // for mozilla::gfx::IntRect
|
|
#include "gfxUtils.h"
|
|
#include "client/TextureClientSharedSurface.h"
|
|
|
|
namespace mozilla {
|
|
namespace layers {
|
|
|
|
using namespace mozilla::gfx;
|
|
using namespace mozilla::gl;
|
|
|
|
CopyableCanvasLayer::CopyableCanvasLayer(LayerManager* aLayerManager, void *aImplData) :
|
|
CanvasLayer(aLayerManager, aImplData)
|
|
, mGLFrontbuffer(nullptr)
|
|
, mIsAlphaPremultiplied(true)
|
|
, mOriginPos(gl::OriginPos::TopLeft)
|
|
{
|
|
MOZ_COUNT_CTOR(CopyableCanvasLayer);
|
|
}
|
|
|
|
CopyableCanvasLayer::~CopyableCanvasLayer()
|
|
{
|
|
MOZ_COUNT_DTOR(CopyableCanvasLayer);
|
|
}
|
|
|
|
void
|
|
CopyableCanvasLayer::Initialize(const Data& aData)
|
|
{
|
|
NS_ASSERTION(mSurface == nullptr, "BasicCanvasLayer::Initialize called twice!");
|
|
|
|
if (aData.mGLContext) {
|
|
mGLContext = aData.mGLContext;
|
|
mIsAlphaPremultiplied = aData.mIsGLAlphaPremult;
|
|
mOriginPos = gl::OriginPos::BottomLeft;
|
|
|
|
MOZ_ASSERT(mGLContext->IsOffscreen(), "canvas gl context isn't offscreen");
|
|
|
|
if (aData.mFrontbufferGLTex) {
|
|
gfx::IntSize size(aData.mSize.width, aData.mSize.height);
|
|
mGLFrontbuffer = SharedSurface_Basic::Wrap(aData.mGLContext, size, aData.mHasAlpha,
|
|
aData.mFrontbufferGLTex);
|
|
}
|
|
} else if (aData.mBufferProvider) {
|
|
mBufferProvider = aData.mBufferProvider;
|
|
} else {
|
|
MOZ_CRASH("CanvasLayer created without mSurface, mDrawTarget or mGLContext?");
|
|
}
|
|
|
|
mBounds.SetRect(0, 0, aData.mSize.width, aData.mSize.height);
|
|
}
|
|
|
|
bool
|
|
CopyableCanvasLayer::IsDataValid(const Data& aData)
|
|
{
|
|
return mGLContext == aData.mGLContext;
|
|
}
|
|
|
|
void
|
|
CopyableCanvasLayer::UpdateTarget(DrawTarget* aDestTarget)
|
|
{
|
|
if (mBufferProvider) {
|
|
mSurface = mBufferProvider->GetSnapshot();
|
|
}
|
|
|
|
if (!mGLContext && aDestTarget) {
|
|
NS_ASSERTION(mSurface, "Must have surface to draw!");
|
|
if (mSurface) {
|
|
aDestTarget->CopySurface(mSurface,
|
|
IntRect(0, 0, mBounds.width, mBounds.height),
|
|
IntPoint(0, 0));
|
|
mSurface = nullptr;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (mBufferProvider) {
|
|
return;
|
|
}
|
|
|
|
MOZ_ASSERT(mGLContext);
|
|
|
|
SharedSurface* frontbuffer = nullptr;
|
|
if (mGLFrontbuffer) {
|
|
frontbuffer = mGLFrontbuffer.get();
|
|
} else {
|
|
GLScreenBuffer* screen = mGLContext->Screen();
|
|
const auto& front = screen->Front();
|
|
if (front) {
|
|
frontbuffer = front->Surf();
|
|
}
|
|
}
|
|
|
|
if (!frontbuffer) {
|
|
NS_WARNING("Null frame received.");
|
|
return;
|
|
}
|
|
|
|
IntSize readSize(frontbuffer->mSize);
|
|
SurfaceFormat format = (GetContentFlags() & CONTENT_OPAQUE)
|
|
? SurfaceFormat::B8G8R8X8
|
|
: SurfaceFormat::B8G8R8A8;
|
|
bool needsPremult = frontbuffer->mHasAlpha && !mIsAlphaPremultiplied;
|
|
|
|
// Try to read back directly into aDestTarget's output buffer
|
|
if (aDestTarget) {
|
|
uint8_t* destData;
|
|
IntSize destSize;
|
|
int32_t destStride;
|
|
SurfaceFormat destFormat;
|
|
if (aDestTarget->LockBits(&destData, &destSize, &destStride, &destFormat)) {
|
|
if (destSize == readSize && destFormat == format) {
|
|
RefPtr<DataSourceSurface> data =
|
|
Factory::CreateWrappingDataSourceSurface(destData, destStride, destSize, destFormat);
|
|
mGLContext->Readback(frontbuffer, data);
|
|
if (needsPremult) {
|
|
gfxUtils::PremultiplyDataSurface(data, data);
|
|
}
|
|
aDestTarget->ReleaseBits(destData);
|
|
return;
|
|
}
|
|
aDestTarget->ReleaseBits(destData);
|
|
}
|
|
}
|
|
|
|
RefPtr<DataSourceSurface> resultSurf = GetTempSurface(readSize, format);
|
|
// There will already be a warning from inside of GetTempSurface, but
|
|
// it doesn't hurt to complain:
|
|
if (NS_WARN_IF(!resultSurf)) {
|
|
return;
|
|
}
|
|
|
|
// Readback handles Flush/MarkDirty.
|
|
mGLContext->Readback(frontbuffer, resultSurf);
|
|
if (needsPremult) {
|
|
gfxUtils::PremultiplyDataSurface(resultSurf, resultSurf);
|
|
}
|
|
MOZ_ASSERT(resultSurf);
|
|
|
|
if (aDestTarget) {
|
|
aDestTarget->CopySurface(resultSurf,
|
|
IntRect(0, 0, readSize.width, readSize.height),
|
|
IntPoint(0, 0));
|
|
} else {
|
|
// If !aDestSurface then we will end up painting using mSurface, so
|
|
// stick our surface into mSurface, so that the Paint() path is the same.
|
|
mSurface = resultSurf;
|
|
}
|
|
}
|
|
|
|
DataSourceSurface*
|
|
CopyableCanvasLayer::GetTempSurface(const IntSize& aSize,
|
|
const SurfaceFormat aFormat)
|
|
{
|
|
if (!mCachedTempSurface ||
|
|
aSize != mCachedTempSurface->GetSize() ||
|
|
aFormat != mCachedTempSurface->GetFormat())
|
|
{
|
|
// Create a surface aligned to 8 bytes since that's the highest alignment WebGL can handle.
|
|
uint32_t stride = GetAlignedStride<8>(aSize.width * BytesPerPixel(aFormat));
|
|
mCachedTempSurface = Factory::CreateDataSourceSurfaceWithStride(aSize, aFormat, stride);
|
|
}
|
|
|
|
return mCachedTempSurface;
|
|
}
|
|
|
|
void
|
|
CopyableCanvasLayer::DiscardTempSurface()
|
|
{
|
|
mCachedTempSurface = nullptr;
|
|
}
|
|
|
|
} // namespace layers
|
|
} // namespace mozilla
|