summaryrefslogtreecommitdiffstats
path: root/gfx/webrender_bindings/RenderBufferTextureHost.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'gfx/webrender_bindings/RenderBufferTextureHost.cpp')
-rw-r--r--gfx/webrender_bindings/RenderBufferTextureHost.cpp160
1 files changed, 160 insertions, 0 deletions
diff --git a/gfx/webrender_bindings/RenderBufferTextureHost.cpp b/gfx/webrender_bindings/RenderBufferTextureHost.cpp
new file mode 100644
index 0000000000..7f8e454633
--- /dev/null
+++ b/gfx/webrender_bindings/RenderBufferTextureHost.cpp
@@ -0,0 +1,160 @@
+/* -*- 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 "RenderBufferTextureHost.h"
+
+#include "mozilla/gfx/Logging.h"
+#include "mozilla/layers/ImageDataSerializer.h"
+
+namespace mozilla {
+namespace wr {
+
+RenderBufferTextureHost::RenderBufferTextureHost(
+ uint8_t* aBuffer, const layers::BufferDescriptor& aDescriptor)
+ : mBuffer(aBuffer),
+ mDescriptor(aDescriptor),
+ mMap(),
+ mYMap(),
+ mCbMap(),
+ mCrMap(),
+ mLocked(false) {
+ MOZ_COUNT_CTOR_INHERITED(RenderBufferTextureHost, RenderTextureHost);
+
+ switch (mDescriptor.type()) {
+ case layers::BufferDescriptor::TYCbCrDescriptor: {
+ const layers::YCbCrDescriptor& ycbcr = mDescriptor.get_YCbCrDescriptor();
+ mSize = ycbcr.display().Size();
+ 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");
+ }
+}
+
+RenderBufferTextureHost::~RenderBufferTextureHost() {
+ MOZ_COUNT_DTOR_INHERITED(RenderBufferTextureHost, RenderTextureHost);
+}
+
+wr::WrExternalImage RenderBufferTextureHost::Lock(
+ uint8_t aChannelIndex, gl::GLContext* aGL, wr::ImageRendering aRendering) {
+ if (!mLocked) {
+ if (!GetBuffer()) {
+ // We hit some problems to get the shmem.
+ gfxCriticalNote << "GetBuffer Failed";
+ return InvalidToWrExternalImage();
+ }
+ if (mFormat != gfx::SurfaceFormat::YUV) {
+ mSurface = gfx::Factory::CreateWrappingDataSourceSurface(
+ GetBuffer(),
+ layers::ImageDataSerializer::GetRGBStride(
+ mDescriptor.get_RGBDescriptor()),
+ mSize, mFormat);
+ if (NS_WARN_IF(!mSurface)) {
+ gfxCriticalNote << "DataSourceSurface is null";
+ return InvalidToWrExternalImage();
+ }
+ if (NS_WARN_IF(!mSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
+ &mMap))) {
+ mSurface = nullptr;
+ gfxCriticalNote << "Failed to map Surface";
+ return InvalidToWrExternalImage();
+ }
+ } else {
+ const layers::YCbCrDescriptor& desc = mDescriptor.get_YCbCrDescriptor();
+
+ mYSurface = gfx::Factory::CreateWrappingDataSourceSurface(
+ layers::ImageDataSerializer::GetYChannel(GetBuffer(), desc),
+ desc.yStride(), desc.ySize(), gfx::SurfaceFormat::A8);
+ mCbSurface = gfx::Factory::CreateWrappingDataSourceSurface(
+ layers::ImageDataSerializer::GetCbChannel(GetBuffer(), desc),
+ desc.cbCrStride(), desc.cbCrSize(), gfx::SurfaceFormat::A8);
+ mCrSurface = gfx::Factory::CreateWrappingDataSourceSurface(
+ layers::ImageDataSerializer::GetCrChannel(GetBuffer(), desc),
+ desc.cbCrStride(), desc.cbCrSize(), gfx::SurfaceFormat::A8);
+ if (NS_WARN_IF(!mYSurface || !mCbSurface || !mCrSurface)) {
+ mYSurface = mCbSurface = mCrSurface = nullptr;
+ gfxCriticalNote << "YCbCr Surface is null";
+ return InvalidToWrExternalImage();
+ }
+ if (NS_WARN_IF(
+ !mYSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
+ &mYMap) ||
+ !mCbSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
+ &mCbMap) ||
+ !mCrSurface->Map(gfx::DataSourceSurface::MapType::READ_WRITE,
+ &mCrMap))) {
+ mYSurface = mCbSurface = mCrSurface = nullptr;
+ gfxCriticalNote << "Failed to map YCbCr Surface";
+ return InvalidToWrExternalImage();
+ }
+ }
+ mLocked = true;
+ }
+
+ RenderBufferData data = GetBufferDataForRender(aChannelIndex);
+ return RawDataToWrExternalImage(data.mData, data.mBufferSize);
+}
+
+void RenderBufferTextureHost::Unlock() {
+ if (mLocked) {
+ if (mSurface) {
+ mSurface->Unmap();
+ mSurface = nullptr;
+ } else if (mYSurface) {
+ mYSurface->Unmap();
+ mCbSurface->Unmap();
+ mCrSurface->Unmap();
+ mYSurface = mCbSurface = mCrSurface = nullptr;
+ }
+ mLocked = false;
+ }
+}
+
+RenderBufferTextureHost::RenderBufferData
+RenderBufferTextureHost::GetBufferDataForRender(uint8_t aChannelIndex) {
+ MOZ_ASSERT(mFormat != gfx::SurfaceFormat::YUV || aChannelIndex < 3);
+ MOZ_ASSERT(mFormat == gfx::SurfaceFormat::YUV || aChannelIndex < 1);
+ MOZ_ASSERT(mLocked);
+
+ if (mFormat != gfx::SurfaceFormat::YUV) {
+ MOZ_ASSERT(mSurface);
+
+ return RenderBufferData(mMap.mData,
+ mMap.mStride * mSurface->GetSize().height);
+ } else {
+ MOZ_ASSERT(mYSurface && mCbSurface && mCrSurface);
+
+ switch (aChannelIndex) {
+ case 0:
+ return RenderBufferData(mYMap.mData,
+ mYMap.mStride * mYSurface->GetSize().height);
+ break;
+ case 1:
+ return RenderBufferData(mCbMap.mData,
+ mCbMap.mStride * mCbSurface->GetSize().height);
+ break;
+ case 2:
+ return RenderBufferData(mCrMap.mData,
+ mCrMap.mStride * mCrSurface->GetSize().height);
+ break;
+ default:
+ MOZ_ASSERT_UNREACHABLE("unexpected to be called");
+ return RenderBufferData(nullptr, 0);
+ }
+ }
+}
+
+} // namespace wr
+} // namespace mozilla