зеркало из https://github.com/mozilla/gecko-dev.git
226 строки
7.5 KiB
C++
226 строки
7.5 KiB
C++
/* -*- Mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 4; -*- */
|
|
/* 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 "MozFramebuffer.h"
|
|
|
|
#include "GLContext.h"
|
|
#include "mozilla/gfx/Logging.h"
|
|
#include "ScopedGLHelpers.h"
|
|
|
|
namespace mozilla {
|
|
namespace gl {
|
|
|
|
static void DeleteByTarget(GLContext* const gl, const GLenum target,
|
|
const GLuint name) {
|
|
if (target == LOCAL_GL_RENDERBUFFER) {
|
|
gl->DeleteRenderbuffer(name);
|
|
} else {
|
|
gl->DeleteTexture(name);
|
|
}
|
|
}
|
|
|
|
UniquePtr<MozFramebuffer> MozFramebuffer::Create(GLContext* const gl,
|
|
const gfx::IntSize& size,
|
|
const uint32_t samples,
|
|
const bool depthStencil) {
|
|
if (samples && !gl->IsSupported(GLFeature::framebuffer_multisample))
|
|
return nullptr;
|
|
|
|
if (uint32_t(size.width) > gl->MaxTexOrRbSize() ||
|
|
uint32_t(size.height) > gl->MaxTexOrRbSize() ||
|
|
samples > gl->MaxSamples()) {
|
|
return nullptr;
|
|
}
|
|
|
|
gl->MakeCurrent();
|
|
|
|
GLContext::LocalErrorScope errorScope(*gl);
|
|
|
|
GLenum colorTarget;
|
|
GLuint colorName;
|
|
if (samples) {
|
|
colorTarget = LOCAL_GL_RENDERBUFFER;
|
|
colorName = gl->CreateRenderbuffer();
|
|
const ScopedBindRenderbuffer bindRB(gl, colorName);
|
|
gl->fRenderbufferStorageMultisample(colorTarget, samples, LOCAL_GL_RGBA8,
|
|
size.width, size.height);
|
|
} else {
|
|
colorTarget = LOCAL_GL_TEXTURE_2D;
|
|
colorName = gl->CreateTexture();
|
|
const ScopedBindTexture bindTex(gl, colorName);
|
|
gl->TexParams_SetClampNoMips();
|
|
gl->fTexImage2D(colorTarget, 0, LOCAL_GL_RGBA, size.width, size.height, 0,
|
|
LOCAL_GL_RGBA, LOCAL_GL_UNSIGNED_BYTE, nullptr);
|
|
}
|
|
|
|
const auto err = errorScope.GetError();
|
|
if (err) {
|
|
if (err != LOCAL_GL_OUT_OF_MEMORY) {
|
|
gfxCriticalNote << "Unexpected error: " << gfx::hexa(err) << ": "
|
|
<< GLContext::GLErrorToString(err);
|
|
}
|
|
DeleteByTarget(gl, colorTarget, colorName);
|
|
return nullptr;
|
|
}
|
|
|
|
return CreateImpl(
|
|
gl, size, samples,
|
|
depthStencil ? DepthAndStencilBuffer::Create(gl, size, samples) : nullptr,
|
|
colorTarget, colorName);
|
|
}
|
|
|
|
UniquePtr<MozFramebuffer> MozFramebuffer::CreateForBacking(
|
|
GLContext* const gl, const gfx::IntSize& size, const uint32_t samples,
|
|
bool depthStencil, const GLenum colorTarget, const GLuint colorName) {
|
|
return CreateImpl(
|
|
gl, size, samples,
|
|
depthStencil ? DepthAndStencilBuffer::Create(gl, size, samples) : nullptr,
|
|
colorTarget, colorName);
|
|
}
|
|
|
|
/* static */ UniquePtr<MozFramebuffer>
|
|
MozFramebuffer::CreateForBackingWithSharedDepthAndStencil(
|
|
const gfx::IntSize& size, const uint32_t samples, GLenum colorTarget,
|
|
GLuint colorName,
|
|
const RefPtr<DepthAndStencilBuffer>& depthAndStencilBuffer) {
|
|
auto gl = depthAndStencilBuffer->gl();
|
|
if (!gl || !gl->MakeCurrent()) {
|
|
return nullptr;
|
|
}
|
|
return CreateImpl(gl, size, samples, depthAndStencilBuffer, colorTarget,
|
|
colorName);
|
|
}
|
|
|
|
/* static */ UniquePtr<MozFramebuffer> MozFramebuffer::CreateImpl(
|
|
GLContext* const gl, const gfx::IntSize& size, const uint32_t samples,
|
|
const RefPtr<DepthAndStencilBuffer>& depthAndStencilBuffer,
|
|
const GLenum colorTarget, const GLuint colorName) {
|
|
GLuint fb = gl->CreateFramebuffer();
|
|
const ScopedBindFramebuffer bindFB(gl, fb);
|
|
|
|
if (colorTarget == LOCAL_GL_RENDERBUFFER) {
|
|
gl->fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER,
|
|
LOCAL_GL_COLOR_ATTACHMENT0, colorTarget,
|
|
colorName);
|
|
} else {
|
|
gl->fFramebufferTexture2D(LOCAL_GL_FRAMEBUFFER, LOCAL_GL_COLOR_ATTACHMENT0,
|
|
colorTarget, colorName, 0);
|
|
}
|
|
|
|
if (depthAndStencilBuffer) {
|
|
gl->fFramebufferRenderbuffer(
|
|
LOCAL_GL_FRAMEBUFFER, LOCAL_GL_DEPTH_ATTACHMENT, LOCAL_GL_RENDERBUFFER,
|
|
depthAndStencilBuffer->mDepthRB);
|
|
gl->fFramebufferRenderbuffer(
|
|
LOCAL_GL_FRAMEBUFFER, LOCAL_GL_STENCIL_ATTACHMENT,
|
|
LOCAL_GL_RENDERBUFFER, depthAndStencilBuffer->mStencilRB);
|
|
}
|
|
|
|
const auto status = gl->fCheckFramebufferStatus(LOCAL_GL_FRAMEBUFFER);
|
|
if (status != LOCAL_GL_FRAMEBUFFER_COMPLETE) {
|
|
gfxCriticalNote << "MozFramebuffer::CreateImpl(size:" << size
|
|
<< ", samples:" << samples
|
|
<< ", depthAndStencil:" << bool(depthAndStencilBuffer)
|
|
<< ", colorTarget:" << gfx::hexa(colorTarget)
|
|
<< ", colorName:" << colorName << "): Incomplete: 0x"
|
|
<< gfx::hexa(status);
|
|
return nullptr;
|
|
}
|
|
|
|
return UniquePtr<MozFramebuffer>(new MozFramebuffer(
|
|
gl, size, fb, samples, depthAndStencilBuffer, colorTarget, colorName));
|
|
}
|
|
|
|
/* static */ RefPtr<DepthAndStencilBuffer> DepthAndStencilBuffer::Create(
|
|
GLContext* const gl, const gfx::IntSize& size, const uint32_t samples) {
|
|
const auto fnAllocRB = [&](GLenum format) {
|
|
GLuint rb = gl->CreateRenderbuffer();
|
|
const ScopedBindRenderbuffer bindRB(gl, rb);
|
|
if (samples) {
|
|
gl->fRenderbufferStorageMultisample(LOCAL_GL_RENDERBUFFER, samples,
|
|
format, size.width, size.height);
|
|
} else {
|
|
gl->fRenderbufferStorage(LOCAL_GL_RENDERBUFFER, format, size.width,
|
|
size.height);
|
|
}
|
|
return rb;
|
|
};
|
|
|
|
GLuint depthRB, stencilRB;
|
|
{
|
|
GLContext::LocalErrorScope errorScope(*gl);
|
|
|
|
if (gl->IsSupported(GLFeature::packed_depth_stencil)) {
|
|
depthRB = fnAllocRB(LOCAL_GL_DEPTH24_STENCIL8);
|
|
stencilRB = depthRB; // Ignore unused mStencilRB.
|
|
} else {
|
|
depthRB = fnAllocRB(LOCAL_GL_DEPTH_COMPONENT24);
|
|
stencilRB = fnAllocRB(LOCAL_GL_STENCIL_INDEX8);
|
|
}
|
|
|
|
const auto err = errorScope.GetError();
|
|
if (err) {
|
|
MOZ_ASSERT(err == LOCAL_GL_OUT_OF_MEMORY);
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
return new DepthAndStencilBuffer(gl, size, depthRB, stencilRB);
|
|
}
|
|
|
|
////////////////////
|
|
|
|
MozFramebuffer::MozFramebuffer(
|
|
GLContext* const gl, const gfx::IntSize& size, GLuint fb,
|
|
const uint32_t samples, RefPtr<DepthAndStencilBuffer> depthAndStencilBuffer,
|
|
const GLenum colorTarget, const GLuint colorName)
|
|
: mWeakGL(gl),
|
|
mSize(size),
|
|
mSamples(samples),
|
|
mFB(fb),
|
|
mColorTarget(colorTarget),
|
|
mDepthAndStencilBuffer(std::move(depthAndStencilBuffer)),
|
|
mColorName(colorName) {
|
|
MOZ_ASSERT(mColorTarget);
|
|
MOZ_ASSERT(mColorName);
|
|
}
|
|
|
|
MozFramebuffer::~MozFramebuffer() {
|
|
GLContext* const gl = mWeakGL;
|
|
if (!gl || !gl->MakeCurrent()) {
|
|
return;
|
|
}
|
|
|
|
gl->DeleteFramebuffer(mFB);
|
|
|
|
DeleteByTarget(gl, mColorTarget, mColorName);
|
|
}
|
|
|
|
bool MozFramebuffer::HasDepth() const {
|
|
return mDepthAndStencilBuffer && mDepthAndStencilBuffer->mDepthRB;
|
|
}
|
|
|
|
bool MozFramebuffer::HasStencil() const {
|
|
return mDepthAndStencilBuffer && mDepthAndStencilBuffer->mStencilRB;
|
|
}
|
|
|
|
DepthAndStencilBuffer::DepthAndStencilBuffer(GLContext* gl,
|
|
const gfx::IntSize& size,
|
|
GLuint depthRB, GLuint stencilRB)
|
|
: mWeakGL(gl), mSize(size), mDepthRB(depthRB), mStencilRB(stencilRB) {}
|
|
|
|
DepthAndStencilBuffer::~DepthAndStencilBuffer() {
|
|
GLContext* const gl = mWeakGL;
|
|
if (!gl || !gl->MakeCurrent()) {
|
|
return;
|
|
}
|
|
|
|
gl->DeleteRenderbuffer(mDepthRB);
|
|
gl->DeleteRenderbuffer(mStencilRB);
|
|
}
|
|
|
|
} // namespace gl
|
|
} // namespace mozilla
|