|
Packit |
f0b94e |
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
Packit |
f0b94e |
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
Packit |
f0b94e |
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
Packit |
f0b94e |
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
#include "TextureImageEGL.h"
|
|
Packit |
f0b94e |
#include "GLLibraryEGL.h"
|
|
Packit |
f0b94e |
#include "GLContext.h"
|
|
Packit |
f0b94e |
#include "GLUploadHelpers.h"
|
|
Packit |
f0b94e |
#include "gfxPlatform.h"
|
|
Packit |
f0b94e |
#include "mozilla/gfx/Types.h"
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
namespace mozilla {
|
|
Packit |
f0b94e |
namespace gl {
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
static GLenum GLFormatForImage(gfx::SurfaceFormat aFormat) {
|
|
Packit |
f0b94e |
switch (aFormat) {
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::B8G8R8A8:
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::B8G8R8X8:
|
|
Packit |
f0b94e |
return LOCAL_GL_RGBA;
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::R5G6B5_UINT16:
|
|
Packit |
f0b94e |
return LOCAL_GL_RGB;
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::A8:
|
|
Packit |
f0b94e |
return LOCAL_GL_LUMINANCE;
|
|
Packit |
f0b94e |
default:
|
|
Packit |
f0b94e |
NS_WARNING("Unknown GL format for Surface format");
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
return 0;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
static GLenum GLTypeForImage(gfx::SurfaceFormat aFormat) {
|
|
Packit |
f0b94e |
switch (aFormat) {
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::B8G8R8A8:
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::B8G8R8X8:
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::A8:
|
|
Packit |
f0b94e |
return LOCAL_GL_UNSIGNED_BYTE;
|
|
Packit |
f0b94e |
case gfx::SurfaceFormat::R5G6B5_UINT16:
|
|
Packit |
f0b94e |
return LOCAL_GL_UNSIGNED_SHORT_5_6_5;
|
|
Packit |
f0b94e |
default:
|
|
Packit |
f0b94e |
NS_WARNING("Unknown GL format for Surface format");
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
return 0;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
TextureImageEGL::TextureImageEGL(GLuint aTexture, const gfx::IntSize& aSize,
|
|
Packit |
f0b94e |
GLenum aWrapMode, ContentType aContentType,
|
|
Packit |
f0b94e |
GLContext* aContext, Flags aFlags,
|
|
Packit |
f0b94e |
TextureState aTextureState,
|
|
Packit |
f0b94e |
TextureImage::ImageFormat aImageFormat)
|
|
Packit |
f0b94e |
: TextureImage(aSize, aWrapMode, aContentType, aFlags),
|
|
Packit |
f0b94e |
mGLContext(aContext),
|
|
Packit |
f0b94e |
mUpdateFormat(gfx::ImageFormatToSurfaceFormat(aImageFormat)),
|
|
Packit |
f0b94e |
mEGLImage(nullptr),
|
|
Packit |
f0b94e |
mTexture(aTexture),
|
|
Packit |
f0b94e |
mSurface(nullptr),
|
|
Packit |
f0b94e |
mConfig(nullptr),
|
|
Packit |
f0b94e |
mTextureState(aTextureState),
|
|
Packit |
f0b94e |
mBound(false) {
|
|
Packit |
f0b94e |
if (mUpdateFormat == gfx::SurfaceFormat::UNKNOWN) {
|
|
Packit |
f0b94e |
mUpdateFormat =
|
|
Packit |
f0b94e |
gfxPlatform::GetPlatform()->Optimal2DFormatForContent(GetContentType());
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (mUpdateFormat == gfx::SurfaceFormat::R5G6B5_UINT16) {
|
|
Packit |
f0b94e |
mTextureFormat = gfx::SurfaceFormat::R8G8B8X8;
|
|
Packit |
f0b94e |
} else if (mUpdateFormat == gfx::SurfaceFormat::B8G8R8X8) {
|
|
Packit |
f0b94e |
mTextureFormat = gfx::SurfaceFormat::B8G8R8X8;
|
|
Packit |
f0b94e |
} else {
|
|
Packit |
f0b94e |
mTextureFormat = gfx::SurfaceFormat::B8G8R8A8;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
TextureImageEGL::~TextureImageEGL() {
|
|
Packit |
f0b94e |
if (mGLContext->IsDestroyed() || !mGLContext->IsOwningThreadCurrent()) {
|
|
Packit |
f0b94e |
return;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// If we have a context, then we need to delete the texture;
|
|
Packit |
f0b94e |
// if we don't have a context (either real or shared),
|
|
Packit |
f0b94e |
// then they went away when the contex was deleted, because it
|
|
Packit |
f0b94e |
// was the only one that had access to it.
|
|
Packit |
f0b94e |
if (mGLContext->MakeCurrent()) {
|
|
Packit |
f0b94e |
mGLContext->fDeleteTextures(1, &mTexture);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
ReleaseTexImage();
|
|
Packit |
f0b94e |
DestroyEGLSurface();
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
bool TextureImageEGL::DirectUpdate(
|
|
Packit |
f0b94e |
gfx::DataSourceSurface* aSurf, const nsIntRegion& aRegion,
|
|
Packit |
f0b94e |
const gfx::IntPoint& aFrom /* = gfx::IntPoint(0,0) */) {
|
|
Packit |
f0b94e |
gfx::IntRect bounds = aRegion.GetBounds();
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsIntRegion region;
|
|
Packit |
f0b94e |
if (mTextureState != Valid) {
|
|
Packit |
f0b94e |
bounds = gfx::IntRect(0, 0, mSize.width, mSize.height);
|
|
Packit |
f0b94e |
region = nsIntRegion(bounds);
|
|
Packit |
f0b94e |
} else {
|
|
Packit |
f0b94e |
region = aRegion;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
bool needInit = mTextureState == Created;
|
|
Packit |
f0b94e |
size_t uploadSize = 0;
|
|
Packit |
f0b94e |
mTextureFormat = UploadSurfaceToTexture(mGLContext, aSurf, region, mTexture,
|
|
Packit |
f0b94e |
mSize, &uploadSize, needInit, aFrom);
|
|
Packit |
f0b94e |
if (mTextureFormat == SurfaceFormat::UNKNOWN) {
|
|
Packit |
f0b94e |
return false;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (uploadSize > 0) {
|
|
Packit |
f0b94e |
UpdateUploadSize(uploadSize);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mTextureState = Valid;
|
|
Packit |
f0b94e |
return true;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
void TextureImageEGL::BindTexture(GLenum aTextureUnit) {
|
|
Packit |
f0b94e |
// Ensure the texture is allocated before it is used.
|
|
Packit |
f0b94e |
if (mTextureState == Created) {
|
|
Packit |
f0b94e |
Resize(mSize);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mGLContext->fActiveTexture(aTextureUnit);
|
|
Packit |
f0b94e |
mGLContext->fBindTexture(LOCAL_GL_TEXTURE_2D, mTexture);
|
|
Packit |
f0b94e |
mGLContext->fActiveTexture(LOCAL_GL_TEXTURE0);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
void TextureImageEGL::Resize(const gfx::IntSize& aSize) {
|
|
Packit |
f0b94e |
if (mSize == aSize && mTextureState != Created) return;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mGLContext->fBindTexture(LOCAL_GL_TEXTURE_2D, mTexture);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mGLContext->fTexImage2D(LOCAL_GL_TEXTURE_2D, 0,
|
|
Packit |
f0b94e |
GLFormatForImage(mUpdateFormat), aSize.width,
|
|
Packit |
f0b94e |
aSize.height, 0, GLFormatForImage(mUpdateFormat),
|
|
Packit |
f0b94e |
GLTypeForImage(mUpdateFormat), nullptr);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mTextureState = Allocated;
|
|
Packit |
f0b94e |
mSize = aSize;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
bool TextureImageEGL::BindTexImage() {
|
|
Packit |
f0b94e |
if (mBound && !ReleaseTexImage()) return false;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
EGLBoolean success = sEGLLibrary.fBindTexImage(
|
|
Packit |
f0b94e |
EGL_DISPLAY(), (EGLSurface)mSurface, LOCAL_EGL_BACK_BUFFER);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (success == LOCAL_EGL_FALSE) return false;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mBound = true;
|
|
Packit |
f0b94e |
return true;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
bool TextureImageEGL::ReleaseTexImage() {
|
|
Packit |
f0b94e |
if (!mBound) return true;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
EGLBoolean success = sEGLLibrary.fReleaseTexImage(
|
|
Packit |
f0b94e |
EGL_DISPLAY(), (EGLSurface)mSurface, LOCAL_EGL_BACK_BUFFER);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (success == LOCAL_EGL_FALSE) return false;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mBound = false;
|
|
Packit |
f0b94e |
return true;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
void TextureImageEGL::DestroyEGLSurface(void) {
|
|
Packit |
f0b94e |
if (!mSurface) return;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
sEGLLibrary.fDestroySurface(EGL_DISPLAY(), mSurface);
|
|
Packit |
f0b94e |
mSurface = nullptr;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
already_AddRefed<TextureImage> CreateTextureImageEGL(
|
|
Packit |
f0b94e |
GLContext* gl, const gfx::IntSize& aSize,
|
|
Packit |
f0b94e |
TextureImage::ContentType aContentType, GLenum aWrapMode,
|
|
Packit |
f0b94e |
TextureImage::Flags aFlags, TextureImage::ImageFormat aImageFormat) {
|
|
Packit |
f0b94e |
RefPtr<TextureImage> t =
|
|
Packit |
f0b94e |
new gl::TiledTextureImage(gl, aSize, aContentType, aFlags, aImageFormat);
|
|
Packit |
f0b94e |
return t.forget();
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
already_AddRefed<TextureImage> TileGenFuncEGL(
|
|
Packit |
f0b94e |
GLContext* gl, const gfx::IntSize& aSize,
|
|
Packit |
f0b94e |
TextureImage::ContentType aContentType, TextureImage::Flags aFlags,
|
|
Packit |
f0b94e |
TextureImage::ImageFormat aImageFormat) {
|
|
Packit |
f0b94e |
gl->MakeCurrent();
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
GLuint texture;
|
|
Packit |
f0b94e |
gl->fGenTextures(1, &texture);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
RefPtr<TextureImageEGL> teximage =
|
|
Packit |
f0b94e |
new TextureImageEGL(texture, aSize, LOCAL_GL_CLAMP_TO_EDGE, aContentType,
|
|
Packit |
f0b94e |
gl, aFlags, TextureImage::Created, aImageFormat);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
teximage->BindTexture(LOCAL_GL_TEXTURE0);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
GLint texfilter = aFlags & TextureImage::UseNearestFilter ? LOCAL_GL_NEAREST
|
|
Packit |
f0b94e |
: LOCAL_GL_LINEAR;
|
|
Packit |
f0b94e |
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_MIN_FILTER,
|
|
Packit |
f0b94e |
texfilter);
|
|
Packit |
f0b94e |
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_MAG_FILTER,
|
|
Packit |
f0b94e |
texfilter);
|
|
Packit |
f0b94e |
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_WRAP_S,
|
|
Packit |
f0b94e |
LOCAL_GL_CLAMP_TO_EDGE);
|
|
Packit |
f0b94e |
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_WRAP_T,
|
|
Packit |
f0b94e |
LOCAL_GL_CLAMP_TO_EDGE);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
return teximage.forget();
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
} // namespace gl
|
|
Packit |
f0b94e |
} // namespace mozilla
|