2020-07-07 20:57:22 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
|
|
/* 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 "RenderExternalTextureHost.h"
|
|
|
|
|
|
|
|
#include "mozilla/gfx/Logging.h"
|
|
|
|
#include "mozilla/layers/ImageDataSerializer.h"
|
|
|
|
|
|
|
|
#include "GLContext.h"
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace wr {
|
|
|
|
|
|
|
|
RenderExternalTextureHost::RenderExternalTextureHost(
|
|
|
|
uint8_t* aBuffer, const layers::BufferDescriptor& aDescriptor)
|
|
|
|
: mBuffer(aBuffer),
|
|
|
|
mDescriptor(aDescriptor),
|
|
|
|
mInitialized(false),
|
|
|
|
mTextureUpdateNeeded(true) {
|
|
|
|
MOZ_COUNT_CTOR_INHERITED(RenderExternalTextureHost, RenderTextureHost);
|
|
|
|
|
|
|
|
switch (mDescriptor.type()) {
|
|
|
|
case layers::BufferDescriptor::TYCbCrDescriptor: {
|
|
|
|
const layers::YCbCrDescriptor& ycbcr = mDescriptor.get_YCbCrDescriptor();
|
|
|
|
mSize = ycbcr.ySize();
|
|
|
|
mFormat = gfx::SurfaceFormat::YUV;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case layers::BufferDescriptor::TRGBDescriptor: {
|
|
|
|
const layers::RGBDescriptor& rgb = mDescriptor.get_RGBDescriptor();
|
|
|
|
mSize = rgb.size();
|
|
|
|
mFormat = rgb.format();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
gfxCriticalError() << "Bad buffer host descriptor "
|
|
|
|
<< (int)mDescriptor.type();
|
|
|
|
MOZ_CRASH("GFX: Bad descriptor");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RenderExternalTextureHost::~RenderExternalTextureHost() {
|
|
|
|
MOZ_COUNT_DTOR_INHERITED(RenderExternalTextureHost, RenderTextureHost);
|
|
|
|
|
|
|
|
if (NS_WARN_IF(!IsReadyForDeletion())) {
|
|
|
|
gfxCriticalNote << "RenderExternalTextureHost sync failed";
|
|
|
|
}
|
|
|
|
|
|
|
|
DeleteTextures();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RenderExternalTextureHost::CreateSurfaces() {
|
|
|
|
if (!IsYUV()) {
|
|
|
|
mSurfaces[0] = gfx::Factory::CreateWrappingDataSourceSurface(
|
|
|
|
GetBuffer(),
|
|
|
|
layers::ImageDataSerializer::GetRGBStride(
|
|
|
|
mDescriptor.get_RGBDescriptor()),
|
|
|
|
mSize, mFormat);
|
|
|
|
} else {
|
|
|
|
const layers::YCbCrDescriptor& desc = mDescriptor.get_YCbCrDescriptor();
|
2021-01-04 17:20:17 +03:00
|
|
|
const gfx::SurfaceFormat surfaceFormat =
|
2020-08-11 04:13:30 +03:00
|
|
|
SurfaceFormatForColorDepth(desc.colorDepth());
|
2020-07-07 20:57:22 +03:00
|
|
|
|
|
|
|
mSurfaces[0] = gfx::Factory::CreateWrappingDataSourceSurface(
|
|
|
|
layers::ImageDataSerializer::GetYChannel(GetBuffer(), desc),
|
2020-08-11 04:13:30 +03:00
|
|
|
desc.yStride(), desc.ySize(), surfaceFormat);
|
2020-07-07 20:57:22 +03:00
|
|
|
mSurfaces[1] = gfx::Factory::CreateWrappingDataSourceSurface(
|
|
|
|
layers::ImageDataSerializer::GetCbChannel(GetBuffer(), desc),
|
2020-08-11 04:13:30 +03:00
|
|
|
desc.cbCrStride(), desc.cbCrSize(), surfaceFormat);
|
2020-07-07 20:57:22 +03:00
|
|
|
mSurfaces[2] = gfx::Factory::CreateWrappingDataSourceSurface(
|
|
|
|
layers::ImageDataSerializer::GetCrChannel(GetBuffer(), desc),
|
2020-08-11 04:13:30 +03:00
|
|
|
desc.cbCrStride(), desc.cbCrSize(), surfaceFormat);
|
2020-07-07 20:57:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < PlaneCount(); ++i) {
|
|
|
|
if (NS_WARN_IF(!mSurfaces[i])) {
|
|
|
|
gfxCriticalNote << "Surface is null";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderExternalTextureHost::DeleteSurfaces() {
|
|
|
|
for (size_t i = 0; i < PlaneCount(); ++i) {
|
|
|
|
mSurfaces[i] = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderExternalTextureHost::DeleteTextures() {
|
|
|
|
for (size_t i = 0; i < PlaneCount(); ++i) {
|
|
|
|
mTextureSources[i] = nullptr;
|
|
|
|
mImages[i] = InvalidToWrExternalImage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RenderExternalTextureHost::InitializeIfNeeded() {
|
|
|
|
if (mInitialized) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!GetBuffer()) {
|
|
|
|
// We hit some problems to get the shmem.
|
|
|
|
gfxCriticalNote << "GetBuffer Failed";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CreateSurfaces()) {
|
|
|
|
DeleteSurfaces();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
mInitialized = true;
|
|
|
|
return mInitialized;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RenderExternalTextureHost::IsReadyForDeletion() {
|
|
|
|
if (!mInitialized) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& textureSource = mTextureSources[0];
|
|
|
|
if (textureSource) {
|
|
|
|
return textureSource->Sync(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
wr::WrExternalImage RenderExternalTextureHost::Lock(
|
|
|
|
uint8_t aChannelIndex, gl::GLContext* aGL, wr::ImageRendering aRendering) {
|
|
|
|
if (mGL.get() != aGL) {
|
|
|
|
mGL = aGL;
|
|
|
|
mGL->MakeCurrent();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mGL || !mGL->MakeCurrent()) {
|
|
|
|
return InvalidToWrExternalImage();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!InitializeIfNeeded()) {
|
|
|
|
return InvalidToWrExternalImage();
|
|
|
|
}
|
|
|
|
|
|
|
|
UpdateTextures(aRendering);
|
|
|
|
return mImages[aChannelIndex];
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderExternalTextureHost::PrepareForUse() { mTextureUpdateNeeded = true; }
|
|
|
|
|
|
|
|
void RenderExternalTextureHost::Unlock() {}
|
|
|
|
|
|
|
|
void RenderExternalTextureHost::UpdateTexture(size_t aIndex) {
|
|
|
|
MOZ_ASSERT(mSurfaces[aIndex]);
|
|
|
|
|
|
|
|
auto& texture = mTextureSources[aIndex];
|
|
|
|
|
|
|
|
if (texture) {
|
|
|
|
texture->Update(mSurfaces[aIndex]);
|
|
|
|
} else {
|
|
|
|
texture = new layers::DirectMapTextureSource(mGL, mSurfaces[aIndex]);
|
|
|
|
|
|
|
|
const GLuint handle = texture->GetTextureHandle();
|
|
|
|
const gfx::IntSize& size = texture->GetSize();
|
|
|
|
mImages[aIndex] =
|
|
|
|
NativeTextureToWrExternalImage(handle, 0, 0, size.width, size.height);
|
|
|
|
}
|
|
|
|
|
|
|
|
MOZ_ASSERT(mGL->GetError() == LOCAL_GL_NO_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderExternalTextureHost::UpdateTextures(wr::ImageRendering aRendering) {
|
|
|
|
const bool renderingChanged = IsFilterUpdateNecessary(aRendering);
|
|
|
|
|
|
|
|
if (!mTextureUpdateNeeded && !renderingChanged) {
|
|
|
|
// Nothing to do here.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < PlaneCount(); ++i) {
|
|
|
|
if (mTextureUpdateNeeded) {
|
|
|
|
UpdateTexture(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (renderingChanged) {
|
|
|
|
const GLuint handle = mTextureSources[i]->GetTextureHandle();
|
|
|
|
ActivateBindAndTexParameteri(mGL, LOCAL_GL_TEXTURE0,
|
|
|
|
LOCAL_GL_TEXTURE_RECTANGLE_ARB, handle,
|
|
|
|
aRendering);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mTextureSources[0]->MaybeFenceTexture();
|
|
|
|
mTextureUpdateNeeded = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace wr
|
|
|
|
} // namespace mozilla
|