summaryrefslogtreecommitdiffstats
path: root/gfx/webrender_bindings/RenderMacIOSurfaceTextureHost.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'gfx/webrender_bindings/RenderMacIOSurfaceTextureHost.cpp')
-rw-r--r--gfx/webrender_bindings/RenderMacIOSurfaceTextureHost.cpp165
1 files changed, 165 insertions, 0 deletions
diff --git a/gfx/webrender_bindings/RenderMacIOSurfaceTextureHost.cpp b/gfx/webrender_bindings/RenderMacIOSurfaceTextureHost.cpp
new file mode 100644
index 0000000000..6d2f5505c8
--- /dev/null
+++ b/gfx/webrender_bindings/RenderMacIOSurfaceTextureHost.cpp
@@ -0,0 +1,165 @@
+/* -*- 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 "RenderMacIOSurfaceTextureHost.h"
+
+#include "GLContextCGL.h"
+#include "mozilla/gfx/Logging.h"
+#include "ScopedGLHelpers.h"
+
+namespace mozilla {
+namespace wr {
+
+static CGLError CreateTextureForPlane(uint8_t aPlaneID, gl::GLContext* aGL,
+ MacIOSurface* aSurface, GLuint* aTexture,
+ wr::ImageRendering aRendering) {
+ MOZ_ASSERT(aGL && aSurface && aTexture);
+
+ aGL->fGenTextures(1, aTexture);
+ ActivateBindAndTexParameteri(aGL, LOCAL_GL_TEXTURE0,
+ LOCAL_GL_TEXTURE_RECTANGLE_ARB, *aTexture,
+ aRendering);
+ aGL->fTexParameteri(LOCAL_GL_TEXTURE_RECTANGLE_ARB, LOCAL_GL_TEXTURE_WRAP_T,
+ LOCAL_GL_CLAMP_TO_EDGE);
+ aGL->fTexParameteri(LOCAL_GL_TEXTURE_RECTANGLE_ARB, LOCAL_GL_TEXTURE_WRAP_S,
+ LOCAL_GL_CLAMP_TO_EDGE);
+
+ CGLError result = kCGLNoError;
+ gfx::SurfaceFormat readFormat = gfx::SurfaceFormat::UNKNOWN;
+ result = aSurface->CGLTexImageIOSurface2D(
+ aGL, gl::GLContextCGL::Cast(aGL)->GetCGLContext(), aPlaneID, &readFormat);
+ // If this is a yuv format, the Webrender only supports YUV422 interleaving
+ // format.
+ MOZ_ASSERT(aSurface->GetFormat() != gfx::SurfaceFormat::YUV422 ||
+ readFormat == gfx::SurfaceFormat::YUV422);
+
+ return result;
+}
+
+RenderMacIOSurfaceTextureHost::RenderMacIOSurfaceTextureHost(
+ MacIOSurface* aSurface)
+ : mSurface(aSurface), mTextureHandles{0, 0, 0} {
+ MOZ_COUNT_CTOR_INHERITED(RenderMacIOSurfaceTextureHost, RenderTextureHost);
+}
+
+RenderMacIOSurfaceTextureHost::~RenderMacIOSurfaceTextureHost() {
+ MOZ_COUNT_DTOR_INHERITED(RenderMacIOSurfaceTextureHost, RenderTextureHost);
+ DeleteTextureHandle();
+}
+
+GLuint RenderMacIOSurfaceTextureHost::GetGLHandle(uint8_t aChannelIndex) const {
+ MOZ_ASSERT(mSurface);
+ MOZ_ASSERT((mSurface->GetPlaneCount() == 0)
+ ? (aChannelIndex == mSurface->GetPlaneCount())
+ : (aChannelIndex < mSurface->GetPlaneCount()));
+ return mTextureHandles[aChannelIndex];
+}
+
+gfx::IntSize RenderMacIOSurfaceTextureHost::GetSize(
+ uint8_t aChannelIndex) const {
+ MOZ_ASSERT(mSurface);
+ MOZ_ASSERT((mSurface->GetPlaneCount() == 0)
+ ? (aChannelIndex == mSurface->GetPlaneCount())
+ : (aChannelIndex < mSurface->GetPlaneCount()));
+
+ if (!mSurface) {
+ return gfx::IntSize();
+ }
+ return gfx::IntSize(mSurface->GetDevicePixelWidth(aChannelIndex),
+ mSurface->GetDevicePixelHeight(aChannelIndex));
+}
+
+wr::WrExternalImage RenderMacIOSurfaceTextureHost::Lock(
+ uint8_t aChannelIndex, gl::GLContext* aGL, wr::ImageRendering aRendering) {
+ if (mGL.get() != aGL) {
+ // release the texture handle in the previous gl context
+ DeleteTextureHandle();
+ mGL = aGL;
+ mGL->MakeCurrent();
+ }
+
+ if (!mSurface || !mGL || !mGL->MakeCurrent()) {
+ return InvalidToWrExternalImage();
+ }
+
+ if (!mTextureHandles[0]) {
+ MOZ_ASSERT(gl::GLContextCGL::Cast(mGL.get())->GetCGLContext());
+
+ mCachedRendering = aRendering;
+ // The result of GetPlaneCount() is 0 for single plane format, but it will
+ // be 2 if the format has 2 planar data.
+ CreateTextureForPlane(0, mGL, mSurface, &(mTextureHandles[0]), aRendering);
+ for (size_t i = 1; i < mSurface->GetPlaneCount(); ++i) {
+ CreateTextureForPlane(i, mGL, mSurface, &(mTextureHandles[i]),
+ aRendering);
+ }
+ // update filter if filter was changed
+ } else if (IsFilterUpdateNecessary(aRendering)) {
+ ActivateBindAndTexParameteri(aGL, LOCAL_GL_TEXTURE0,
+ LOCAL_GL_TEXTURE_RECTANGLE_ARB,
+ mTextureHandles[0], aRendering);
+ // Cache new rendering filter.
+ mCachedRendering = aRendering;
+ for (size_t i = 1; i < mSurface->GetPlaneCount(); ++i) {
+ ActivateBindAndTexParameteri(aGL, LOCAL_GL_TEXTURE0,
+ LOCAL_GL_TEXTURE_RECTANGLE_ARB,
+ mTextureHandles[i], aRendering);
+ }
+ }
+
+ gfx::IntSize size = GetSize(aChannelIndex);
+ return NativeTextureToWrExternalImage(GetGLHandle(aChannelIndex), 0, 0,
+ size.width, size.height);
+}
+
+void RenderMacIOSurfaceTextureHost::Unlock() {}
+
+void RenderMacIOSurfaceTextureHost::DeleteTextureHandle() {
+ if (mTextureHandles[0] != 0 && mGL && mGL->MakeCurrent()) {
+ // Calling glDeleteTextures on 0 isn't an error. So, just make them a single
+ // call.
+ mGL->fDeleteTextures(3, mTextureHandles);
+ for (size_t i = 0; i < 3; ++i) {
+ mTextureHandles[i] = 0;
+ }
+ }
+}
+
+size_t RenderMacIOSurfaceTextureHost::GetPlaneCount() const {
+ size_t planeCount = mSurface->GetPlaneCount();
+ return planeCount > 0 ? planeCount : 1;
+}
+
+gfx::SurfaceFormat RenderMacIOSurfaceTextureHost::GetFormat() const {
+ return mSurface->GetFormat();
+}
+
+gfx::ColorDepth RenderMacIOSurfaceTextureHost::GetColorDepth() const {
+ return gfx::ColorDepth::COLOR_8;
+}
+
+gfx::YUVColorSpace RenderMacIOSurfaceTextureHost::GetYUVColorSpace() const {
+ return mSurface->GetYUVColorSpace();
+}
+
+bool RenderMacIOSurfaceTextureHost::MapPlane(RenderCompositor* aCompositor,
+ uint8_t aChannelIndex,
+ PlaneInfo& aPlaneInfo) {
+ if (!aChannelIndex) {
+ mSurface->Lock();
+ }
+ aPlaneInfo.mData = mSurface->GetBaseAddressOfPlane(aChannelIndex);
+ aPlaneInfo.mStride = mSurface->GetBytesPerRow(aChannelIndex);
+ aPlaneInfo.mSize =
+ gfx::IntSize(mSurface->GetDevicePixelWidth(aChannelIndex),
+ mSurface->GetDevicePixelHeight(aChannelIndex));
+ return true;
+}
+
+void RenderMacIOSurfaceTextureHost::UnmapPlanes() { mSurface->Unlock(); }
+
+} // namespace wr
+} // namespace mozilla