* Brings us closer to core profile, all matrices are working, and apart from a problem with text drawing, Blender is working fine. * Reduce the coding overhead of having to setup/teardown when alternating between 2D and 3D drawing sessions. * Gives us fewer modes and states we need to keep track of. Unfortunatelly this also "rejects a fundamental change" the original design was trying to make - that 2D is different from 3D and deserves its own best implementation. That said, it is still aligned with the function API design as originally implemented (i.e., it still uses gpuTranslate2D, ...). Finally, if you build with core profile and this patch you get: https://developer.blender.org/F545352 [The text glitch is an unrelated issue]. Reviewers: merwin, sergey, brecht Differential Revision: https://developer.blender.org/D2626
672 lines
18 KiB
C
672 lines
18 KiB
C
/*
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* The Original Code is Copyright (C) 2005 Blender Foundation.
|
|
* All rights reserved.
|
|
*
|
|
* The Original Code is: all of this file.
|
|
*
|
|
* Contributor(s): Brecht Van Lommel.
|
|
*
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
*/
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "BLI_blenlib.h"
|
|
#include "BLI_utildefines.h"
|
|
|
|
#include "BKE_global.h"
|
|
|
|
#include "GPU_batch.h"
|
|
#include "GPU_draw.h"
|
|
#include "GPU_framebuffer.h"
|
|
#include "GPU_matrix.h"
|
|
#include "GPU_shader.h"
|
|
#include "GPU_texture.h"
|
|
|
|
static struct GPUFrameBufferGlobal {
|
|
GLuint currentfb;
|
|
} GG = {0};
|
|
|
|
/* Number of maximum output slots.
|
|
* We support 4 outputs for now (usually we wouldn't need more to preserve fill rate) */
|
|
#define GPU_FB_MAX_SLOTS 4
|
|
|
|
struct GPUFrameBuffer {
|
|
GLuint object;
|
|
GPUTexture *colortex[GPU_FB_MAX_SLOTS];
|
|
GPUTexture *depthtex;
|
|
struct GPUStateValues attribs;
|
|
};
|
|
|
|
static void gpu_print_framebuffer_error(GLenum status, char err_out[256])
|
|
{
|
|
const char *format = "GPUFrameBuffer: framebuffer status %s\n";
|
|
const char *err = "unknown";
|
|
|
|
#define format_status(X) \
|
|
case GL_FRAMEBUFFER_##X: err = "GL_FRAMEBUFFER_"#X; \
|
|
break;
|
|
|
|
switch (status) {
|
|
/* success */
|
|
format_status(COMPLETE)
|
|
/* errors shared by OpenGL desktop & ES */
|
|
format_status(INCOMPLETE_ATTACHMENT)
|
|
format_status(INCOMPLETE_MISSING_ATTACHMENT)
|
|
format_status(UNSUPPORTED)
|
|
#if 0 /* for OpenGL ES only */
|
|
format_status(INCOMPLETE_DIMENSIONS)
|
|
#else /* for desktop GL only */
|
|
format_status(INCOMPLETE_DRAW_BUFFER)
|
|
format_status(INCOMPLETE_READ_BUFFER)
|
|
format_status(INCOMPLETE_MULTISAMPLE)
|
|
format_status(UNDEFINED)
|
|
#endif
|
|
}
|
|
|
|
#undef format_status
|
|
|
|
if (err_out) {
|
|
BLI_snprintf(err_out, 256, format, err);
|
|
}
|
|
else {
|
|
fprintf(stderr, format, err);
|
|
}
|
|
}
|
|
|
|
/* GPUFrameBuffer */
|
|
|
|
GPUFrameBuffer *GPU_framebuffer_create(void)
|
|
{
|
|
GPUFrameBuffer *fb;
|
|
|
|
fb = MEM_callocN(sizeof(GPUFrameBuffer), "GPUFrameBuffer");
|
|
glGenFramebuffers(1, &fb->object);
|
|
|
|
if (!fb->object) {
|
|
fprintf(stderr, "GPUFFrameBuffer: framebuffer gen failed.\n");
|
|
GPU_framebuffer_free(fb);
|
|
return NULL;
|
|
}
|
|
|
|
/* make sure no read buffer is enabled, so completeness check will not fail. We set those at binding time */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
glReadBuffer(GL_NONE);
|
|
glDrawBuffer(GL_NONE);
|
|
glBindFramebuffer(GL_FRAMEBUFFER, 0);
|
|
|
|
return fb;
|
|
}
|
|
|
|
bool GPU_framebuffer_texture_attach(GPUFrameBuffer *fb, GPUTexture *tex, int slot)
|
|
{
|
|
GLenum attachment;
|
|
|
|
if (slot >= GPU_FB_MAX_SLOTS) {
|
|
fprintf(stderr,
|
|
"Attaching to index %d framebuffer slot unsupported. "
|
|
"Use at most %d\n", slot, GPU_FB_MAX_SLOTS);
|
|
return false;
|
|
}
|
|
|
|
if ((G.debug & G_DEBUG)) {
|
|
if (GPU_texture_bound_number(tex) != -1) {
|
|
fprintf(stderr,
|
|
"Feedback loop warning!: "
|
|
"Attempting to attach texture to framebuffer while still bound to texture unit for drawing!\n");
|
|
}
|
|
}
|
|
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
GG.currentfb = fb->object;
|
|
|
|
if (GPU_texture_stencil(tex) && GPU_texture_depth(tex))
|
|
attachment = GL_DEPTH_STENCIL_ATTACHMENT;
|
|
else if (GPU_texture_depth(tex))
|
|
attachment = GL_DEPTH_ATTACHMENT;
|
|
else
|
|
attachment = GL_COLOR_ATTACHMENT0 + slot;
|
|
|
|
#if defined(__APPLE__) && defined(WITH_GL_PROFILE_COMPAT)
|
|
/* Mac workaround, remove after we switch to core profile */
|
|
glFramebufferTextureEXT(GL_FRAMEBUFFER, attachment, GPU_texture_opengl_bindcode(tex), 0);
|
|
#else
|
|
glFramebufferTexture(GL_FRAMEBUFFER, attachment, GPU_texture_opengl_bindcode(tex), 0);
|
|
#endif
|
|
|
|
if (GPU_texture_depth(tex))
|
|
fb->depthtex = tex;
|
|
else
|
|
fb->colortex[slot] = tex;
|
|
|
|
GPU_texture_framebuffer_set(tex, fb, slot);
|
|
|
|
return true;
|
|
}
|
|
|
|
void GPU_framebuffer_texture_detach(GPUTexture *tex)
|
|
{
|
|
GLenum attachment;
|
|
GPUFrameBuffer *fb = GPU_texture_framebuffer(tex);
|
|
int fb_attachment = GPU_texture_framebuffer_attachment(tex);
|
|
|
|
if (!fb)
|
|
return;
|
|
|
|
if (GG.currentfb != fb->object) {
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
GG.currentfb = fb->object;
|
|
}
|
|
|
|
if (GPU_texture_stencil(tex) && GPU_texture_depth(tex)) {
|
|
fb->depthtex = NULL;
|
|
attachment = GL_DEPTH_STENCIL_ATTACHMENT;
|
|
}
|
|
else if (GPU_texture_depth(tex)) {
|
|
fb->depthtex = NULL;
|
|
attachment = GL_DEPTH_ATTACHMENT;
|
|
}
|
|
else {
|
|
BLI_assert(fb->colortex[fb_attachment] == tex);
|
|
fb->colortex[fb_attachment] = NULL;
|
|
attachment = GL_COLOR_ATTACHMENT0 + fb_attachment;
|
|
}
|
|
|
|
glFramebufferTexture2D(GL_FRAMEBUFFER, attachment, GPU_texture_target(tex), 0, 0);
|
|
|
|
GPU_texture_framebuffer_set(tex, NULL, -1);
|
|
}
|
|
|
|
void GPU_texture_bind_as_framebuffer(GPUTexture *tex)
|
|
{
|
|
GPUFrameBuffer *fb = GPU_texture_framebuffer(tex);
|
|
int fb_attachment = GPU_texture_framebuffer_attachment(tex);
|
|
|
|
if (!fb) {
|
|
fprintf(stderr, "Error, texture not bound to framebuffer!\n");
|
|
return;
|
|
}
|
|
|
|
/* push attributes */
|
|
gpuSaveState(&fb->attribs, GPU_ENABLE_BIT | GPU_VIEWPORT_BIT);
|
|
glDisable(GL_SCISSOR_TEST);
|
|
|
|
/* bind framebuffer */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
|
|
if (GPU_texture_depth(tex)) {
|
|
glDrawBuffer(GL_NONE);
|
|
glReadBuffer(GL_NONE);
|
|
}
|
|
else {
|
|
/* last bound prevails here, better allow explicit control here too */
|
|
glDrawBuffer(GL_COLOR_ATTACHMENT0 + fb_attachment);
|
|
glReadBuffer(GL_COLOR_ATTACHMENT0 + fb_attachment);
|
|
}
|
|
|
|
if (GPU_texture_target(tex) == GL_TEXTURE_2D_MULTISAMPLE) {
|
|
glEnable(GL_MULTISAMPLE);
|
|
}
|
|
|
|
/* set default viewport */
|
|
glViewport(0, 0, GPU_texture_width(tex), GPU_texture_height(tex));
|
|
GG.currentfb = fb->object;
|
|
}
|
|
|
|
void GPU_framebuffer_slots_bind(GPUFrameBuffer *fb, int slot)
|
|
{
|
|
int numslots = 0, i;
|
|
GLenum attachments[4];
|
|
|
|
if (!fb->colortex[slot]) {
|
|
fprintf(stderr, "Error, framebuffer slot empty!\n");
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
if (fb->colortex[i]) {
|
|
attachments[numslots] = GL_COLOR_ATTACHMENT0 + i;
|
|
numslots++;
|
|
}
|
|
}
|
|
|
|
/* push attributes */
|
|
gpuSaveState(&fb->attribs, GPU_ENABLE_BIT | GPU_VIEWPORT_BIT);
|
|
glDisable(GL_SCISSOR_TEST);
|
|
|
|
/* bind framebuffer */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
|
|
/* last bound prevails here, better allow explicit control here too */
|
|
glDrawBuffers(numslots, attachments);
|
|
glReadBuffer(GL_COLOR_ATTACHMENT0 + slot);
|
|
|
|
/* set default viewport */
|
|
glViewport(0, 0, GPU_texture_width(fb->colortex[slot]), GPU_texture_height(fb->colortex[slot]));
|
|
GG.currentfb = fb->object;
|
|
}
|
|
|
|
void GPU_framebuffer_bind(GPUFrameBuffer *fb)
|
|
{
|
|
int numslots = 0, i;
|
|
GLenum attachments[4];
|
|
GLenum readattachement = 0;
|
|
GPUTexture *tex;
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
if (fb->colortex[i]) {
|
|
attachments[numslots] = GL_COLOR_ATTACHMENT0 + i;
|
|
tex = fb->colortex[i];
|
|
|
|
if (!readattachement)
|
|
readattachement = GL_COLOR_ATTACHMENT0 + i;
|
|
|
|
numslots++;
|
|
}
|
|
}
|
|
|
|
/* bind framebuffer */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
|
|
if (numslots == 0) {
|
|
glDrawBuffer(GL_NONE);
|
|
glReadBuffer(GL_NONE);
|
|
tex = fb->depthtex;
|
|
}
|
|
else {
|
|
/* last bound prevails here, better allow explicit control here too */
|
|
glDrawBuffers(numslots, attachments);
|
|
glReadBuffer(readattachement);
|
|
}
|
|
|
|
glViewport(0, 0, GPU_texture_width(tex), GPU_texture_height(tex));
|
|
GG.currentfb = fb->object;
|
|
}
|
|
|
|
|
|
void GPU_framebuffer_texture_unbind(GPUFrameBuffer *fb, GPUTexture *UNUSED(tex))
|
|
{
|
|
/* restore attributes */
|
|
gpuRestoreState(&fb->attribs);
|
|
}
|
|
|
|
void GPU_framebuffer_bind_no_save(GPUFrameBuffer *fb, int slot)
|
|
{
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
/* last bound prevails here, better allow explicit control here too */
|
|
glDrawBuffer(GL_COLOR_ATTACHMENT0 + slot);
|
|
glReadBuffer(GL_COLOR_ATTACHMENT0 + slot);
|
|
|
|
/* push matrices and set default viewport and matrix */
|
|
glViewport(0, 0, GPU_texture_width(fb->colortex[slot]), GPU_texture_height(fb->colortex[slot]));
|
|
GG.currentfb = fb->object;
|
|
}
|
|
|
|
bool GPU_framebuffer_bound(GPUFrameBuffer *fb)
|
|
{
|
|
return fb->object == GG.currentfb;
|
|
}
|
|
|
|
bool GPU_framebuffer_check_valid(GPUFrameBuffer *fb, char err_out[256])
|
|
{
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
GG.currentfb = fb->object;
|
|
|
|
GLenum status = glCheckFramebufferStatus(GL_FRAMEBUFFER);
|
|
|
|
if (status != GL_FRAMEBUFFER_COMPLETE) {
|
|
GPU_framebuffer_restore();
|
|
gpu_print_framebuffer_error(status, err_out);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void GPU_framebuffer_free(GPUFrameBuffer *fb)
|
|
{
|
|
int i;
|
|
if (fb->depthtex)
|
|
GPU_framebuffer_texture_detach(fb->depthtex);
|
|
|
|
for (i = 0; i < GPU_FB_MAX_SLOTS; i++) {
|
|
if (fb->colortex[i]) {
|
|
GPU_framebuffer_texture_detach(fb->colortex[i]);
|
|
}
|
|
}
|
|
|
|
if (fb->object) {
|
|
glDeleteFramebuffers(1, &fb->object);
|
|
|
|
if (GG.currentfb == fb->object) {
|
|
glBindFramebuffer(GL_FRAMEBUFFER, 0);
|
|
GG.currentfb = 0;
|
|
}
|
|
}
|
|
|
|
MEM_freeN(fb);
|
|
}
|
|
|
|
void GPU_framebuffer_restore(void)
|
|
{
|
|
if (GG.currentfb != 0) {
|
|
glBindFramebuffer(GL_FRAMEBUFFER, 0);
|
|
GG.currentfb = 0;
|
|
}
|
|
}
|
|
|
|
void GPU_framebuffer_blur(
|
|
GPUFrameBuffer *fb, GPUTexture *tex,
|
|
GPUFrameBuffer *blurfb, GPUTexture *blurtex)
|
|
{
|
|
const float fullscreencos[4][2] = {{-1.0f, -1.0f}, {1.0f, -1.0f}, {-1.0f, 1.0f}, {1.0f, 1.0f}};
|
|
const float fullscreenuvs[4][2] = {{0.0f, 0.0f}, {1.0f, 0.0f}, {0.0f, 1.0f}, {1.0f, 1.0f}};
|
|
|
|
static VertexFormat format = {0};
|
|
static VertexBuffer vbo = {{0}};
|
|
static Batch batch = {{0}};
|
|
|
|
const float scaleh[2] = {1.0f / GPU_texture_width(blurtex), 0.0f};
|
|
const float scalev[2] = {0.0f, 1.0f / GPU_texture_height(tex)};
|
|
|
|
GPUShader *blur_shader = GPU_shader_get_builtin_shader(GPU_SHADER_SEP_GAUSSIAN_BLUR);
|
|
|
|
if (!blur_shader)
|
|
return;
|
|
|
|
/* Preparing to draw quad */
|
|
if (format.attrib_ct == 0) {
|
|
unsigned int i = 0;
|
|
/* Vertex format */
|
|
unsigned int pos = VertexFormat_add_attrib(&format, "pos", COMP_F32, 2, KEEP_FLOAT);
|
|
unsigned int uvs = VertexFormat_add_attrib(&format, "uvs", COMP_F32, 2, KEEP_FLOAT);
|
|
|
|
/* Vertices */
|
|
VertexBuffer_init_with_format(&vbo, &format);
|
|
VertexBuffer_allocate_data(&vbo, 36);
|
|
|
|
for (int j = 0; j < 3; ++j) {
|
|
VertexBuffer_set_attrib(&vbo, uvs, i, fullscreenuvs[j]);
|
|
VertexBuffer_set_attrib(&vbo, pos, i++, fullscreencos[j]);
|
|
}
|
|
for (int j = 1; j < 4; ++j) {
|
|
VertexBuffer_set_attrib(&vbo, uvs, i, fullscreenuvs[j]);
|
|
VertexBuffer_set_attrib(&vbo, pos, i++, fullscreencos[j]);
|
|
}
|
|
|
|
Batch_init(&batch, GL_TRIANGLES, &vbo, NULL);
|
|
}
|
|
|
|
glDisable(GL_DEPTH_TEST);
|
|
|
|
/* Blurring horizontally */
|
|
/* We do the bind ourselves rather than using GPU_framebuffer_texture_bind() to avoid
|
|
* pushing unnecessary matrices onto the OpenGL stack. */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, blurfb->object);
|
|
glDrawBuffer(GL_COLOR_ATTACHMENT0);
|
|
|
|
/* avoid warnings from texture binding */
|
|
GG.currentfb = blurfb->object;
|
|
|
|
glViewport(0, 0, GPU_texture_width(blurtex), GPU_texture_height(blurtex));
|
|
|
|
GPU_texture_bind(tex, 0);
|
|
|
|
Batch_set_builtin_program(&batch, GPU_SHADER_SEP_GAUSSIAN_BLUR);
|
|
Batch_Uniform2f(&batch, "ScaleU", scaleh[0], scaleh[1]);
|
|
Batch_Uniform1i(&batch, "textureSource", GL_TEXTURE0);
|
|
Batch_draw(&batch);
|
|
|
|
/* Blurring vertically */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fb->object);
|
|
glDrawBuffer(GL_COLOR_ATTACHMENT0);
|
|
|
|
GG.currentfb = fb->object;
|
|
|
|
glViewport(0, 0, GPU_texture_width(tex), GPU_texture_height(tex));
|
|
|
|
GPU_texture_bind(blurtex, 0);
|
|
|
|
/* Hack to make the following uniform stick */
|
|
Batch_set_builtin_program(&batch, GPU_SHADER_SEP_GAUSSIAN_BLUR);
|
|
Batch_Uniform2f(&batch, "ScaleU", scalev[0], scalev[1]);
|
|
Batch_Uniform1i(&batch, "textureSource", GL_TEXTURE0);
|
|
Batch_draw(&batch);
|
|
}
|
|
|
|
void GPU_framebuffer_blit(GPUFrameBuffer *fb_read, int read_slot, GPUFrameBuffer *fb_write, int write_slot, bool use_depth)
|
|
{
|
|
GPUTexture *read_tex = (use_depth) ? fb_read->depthtex : fb_read->colortex[read_slot];
|
|
GPUTexture *write_tex = (use_depth) ? fb_write->depthtex : fb_write->colortex[write_slot];
|
|
int read_attach = (use_depth) ? GL_DEPTH_ATTACHMENT : GL_COLOR_ATTACHMENT0 + GPU_texture_framebuffer_attachment(read_tex);
|
|
int write_attach = (use_depth) ? GL_DEPTH_ATTACHMENT : GL_COLOR_ATTACHMENT0 + GPU_texture_framebuffer_attachment(write_tex);
|
|
int read_bind = GPU_texture_opengl_bindcode(read_tex);
|
|
int write_bind = GPU_texture_opengl_bindcode(write_tex);
|
|
const int read_w = GPU_texture_width(read_tex);
|
|
const int read_h = GPU_texture_height(read_tex);
|
|
const int write_w = GPU_texture_width(write_tex);
|
|
const int write_h = GPU_texture_height(write_tex);
|
|
|
|
/* read from multi-sample buffer */
|
|
glBindFramebuffer(GL_READ_FRAMEBUFFER, fb_read->object);
|
|
glFramebufferTexture2D(
|
|
GL_READ_FRAMEBUFFER, read_attach,
|
|
GL_TEXTURE_2D, read_bind, 0);
|
|
BLI_assert(glCheckFramebufferStatus(GL_READ_FRAMEBUFFER) == GL_FRAMEBUFFER_COMPLETE);
|
|
|
|
/* write into new single-sample buffer */
|
|
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fb_write->object);
|
|
glFramebufferTexture2D(
|
|
GL_DRAW_FRAMEBUFFER, write_attach,
|
|
GL_TEXTURE_2D, write_bind, 0);
|
|
BLI_assert(glCheckFramebufferStatus(GL_DRAW_FRAMEBUFFER) == GL_FRAMEBUFFER_COMPLETE);
|
|
|
|
glBlitFramebuffer(0, 0, read_w, read_h, 0, 0, write_w, write_h, (use_depth) ? GL_DEPTH_BUFFER_BIT : GL_COLOR_BUFFER_BIT, GL_NEAREST);
|
|
|
|
/* Restore previous framebuffer */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, GG.currentfb);
|
|
glDrawBuffer(GL_COLOR_ATTACHMENT0);
|
|
}
|
|
|
|
/* GPUOffScreen */
|
|
|
|
struct GPUOffScreen {
|
|
GPUFrameBuffer *fb;
|
|
GPUTexture *color;
|
|
GPUTexture *depth;
|
|
};
|
|
|
|
GPUOffScreen *GPU_offscreen_create(int width, int height, int samples, char err_out[256])
|
|
{
|
|
GPUOffScreen *ofs;
|
|
|
|
ofs = MEM_callocN(sizeof(GPUOffScreen), "GPUOffScreen");
|
|
|
|
ofs->fb = GPU_framebuffer_create();
|
|
if (!ofs->fb) {
|
|
GPU_offscreen_free(ofs);
|
|
return NULL;
|
|
}
|
|
|
|
if (samples) {
|
|
if (!GLEW_ARB_texture_multisample ||
|
|
/* This is required when blitting from a multi-sampled buffers,
|
|
* even though we're not scaling. */
|
|
!GLEW_EXT_framebuffer_multisample_blit_scaled)
|
|
{
|
|
samples = 0;
|
|
}
|
|
}
|
|
|
|
ofs->depth = GPU_texture_create_depth_multisample(width, height, samples, err_out);
|
|
if (!ofs->depth) {
|
|
GPU_offscreen_free(ofs);
|
|
return NULL;
|
|
}
|
|
|
|
if (!GPU_framebuffer_texture_attach(ofs->fb, ofs->depth, 0)) {
|
|
GPU_offscreen_free(ofs);
|
|
return NULL;
|
|
}
|
|
|
|
ofs->color = GPU_texture_create_2D_multisample(width, height, NULL, samples, err_out);
|
|
if (!ofs->color) {
|
|
GPU_offscreen_free(ofs);
|
|
return NULL;
|
|
}
|
|
|
|
if (!GPU_framebuffer_texture_attach(ofs->fb, ofs->color, 0)) {
|
|
GPU_offscreen_free(ofs);
|
|
return NULL;
|
|
}
|
|
|
|
/* check validity at the very end! */
|
|
if (!GPU_framebuffer_check_valid(ofs->fb, err_out)) {
|
|
GPU_offscreen_free(ofs);
|
|
return NULL;
|
|
}
|
|
|
|
GPU_framebuffer_restore();
|
|
|
|
return ofs;
|
|
}
|
|
|
|
void GPU_offscreen_free(GPUOffScreen *ofs)
|
|
{
|
|
if (ofs->fb)
|
|
GPU_framebuffer_free(ofs->fb);
|
|
if (ofs->color)
|
|
GPU_texture_free(ofs->color);
|
|
if (ofs->depth)
|
|
GPU_texture_free(ofs->depth);
|
|
|
|
MEM_freeN(ofs);
|
|
}
|
|
|
|
void GPU_offscreen_bind(GPUOffScreen *ofs, bool save)
|
|
{
|
|
glDisable(GL_SCISSOR_TEST);
|
|
if (save)
|
|
GPU_texture_bind_as_framebuffer(ofs->color);
|
|
else {
|
|
GPU_framebuffer_bind_no_save(ofs->fb, 0);
|
|
}
|
|
}
|
|
|
|
void GPU_offscreen_unbind(GPUOffScreen *ofs, bool restore)
|
|
{
|
|
if (restore)
|
|
GPU_framebuffer_texture_unbind(ofs->fb, ofs->color);
|
|
GPU_framebuffer_restore();
|
|
glEnable(GL_SCISSOR_TEST);
|
|
}
|
|
|
|
void GPU_offscreen_read_pixels(GPUOffScreen *ofs, int type, void *pixels)
|
|
{
|
|
const int w = GPU_texture_width(ofs->color);
|
|
const int h = GPU_texture_height(ofs->color);
|
|
|
|
if (GPU_texture_target(ofs->color) == GL_TEXTURE_2D_MULTISAMPLE) {
|
|
/* For a multi-sample texture,
|
|
* we need to create an intermediate buffer to blit to,
|
|
* before its copied using 'glReadPixels' */
|
|
|
|
/* not needed since 'ofs' needs to be bound to the framebuffer already */
|
|
// #define USE_FBO_CTX_SWITCH
|
|
|
|
GLuint fbo_blit = 0;
|
|
GLuint tex_blit = 0;
|
|
GLenum status;
|
|
|
|
/* create texture for new 'fbo_blit' */
|
|
glGenTextures(1, &tex_blit);
|
|
if (!tex_blit) {
|
|
goto finally;
|
|
}
|
|
|
|
glBindTexture(GL_TEXTURE_2D, tex_blit);
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, w, h, 0, GL_RGBA, type, 0);
|
|
|
|
#ifdef USE_FBO_CTX_SWITCH
|
|
/* read from multi-sample buffer */
|
|
glBindFramebuffer(GL_READ_FRAMEBUFFER, ofs->color->fb->object);
|
|
glFramebufferTexture2D(
|
|
GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0 + ofs->color->fb_attachment,
|
|
GL_TEXTURE_2D_MULTISAMPLE, ofs->color->bindcode, 0);
|
|
status = glCheckFramebufferStatus(GL_READ_FRAMEBUFFER);
|
|
if (status != GL_FRAMEBUFFER_COMPLETE) {
|
|
goto finally;
|
|
}
|
|
#endif
|
|
|
|
/* write into new single-sample buffer */
|
|
glGenFramebuffers(1, &fbo_blit);
|
|
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fbo_blit);
|
|
glFramebufferTexture2D(
|
|
GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
|
|
GL_TEXTURE_2D, tex_blit, 0);
|
|
status = glCheckFramebufferStatus(GL_DRAW_FRAMEBUFFER);
|
|
if (status != GL_FRAMEBUFFER_COMPLETE) {
|
|
goto finally;
|
|
}
|
|
|
|
/* perform the copy */
|
|
glBlitFramebuffer(0, 0, w, h, 0, 0, w, h, GL_COLOR_BUFFER_BIT, GL_NEAREST);
|
|
|
|
/* read the results */
|
|
glBindFramebuffer(GL_READ_FRAMEBUFFER, fbo_blit);
|
|
glReadPixels(0, 0, w, h, GL_RGBA, type, pixels);
|
|
|
|
#ifdef USE_FBO_CTX_SWITCH
|
|
/* restore the original frame-bufer */
|
|
glBindFramebuffer(GL_FRAMEBUFFER, ofs->color->fb->object);
|
|
#undef USE_FBO_CTX_SWITCH
|
|
#endif
|
|
|
|
|
|
finally:
|
|
/* cleanup */
|
|
if (tex_blit) {
|
|
glDeleteTextures(1, &tex_blit);
|
|
}
|
|
if (fbo_blit) {
|
|
glDeleteFramebuffers(1, &fbo_blit);
|
|
}
|
|
}
|
|
else {
|
|
glReadPixels(0, 0, w, h, GL_RGBA, type, pixels);
|
|
}
|
|
}
|
|
|
|
int GPU_offscreen_width(const GPUOffScreen *ofs)
|
|
{
|
|
return GPU_texture_width(ofs->color);
|
|
}
|
|
|
|
int GPU_offscreen_height(const GPUOffScreen *ofs)
|
|
{
|
|
return GPU_texture_height(ofs->color);
|
|
}
|
|
|
|
int GPU_offscreen_color_texture(const GPUOffScreen *ofs)
|
|
{
|
|
return GPU_texture_opengl_bindcode(ofs->color);
|
|
}
|
|
|