2015-03-21 03:35:04 +08:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// Offscreen rendering in OpenGL using framebuffer objects.
|
|
|
|
//
|
|
|
|
// Copyright 2015 <whitequark@whitequark.org>
|
|
|
|
//-----------------------------------------------------------------------------
|
2015-03-24 14:45:53 +08:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <OpenGL/GL.h>
|
|
|
|
#else
|
2015-03-21 03:35:04 +08:00
|
|
|
#include <GL/glew.h>
|
2015-03-24 14:45:53 +08:00
|
|
|
#endif
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
#include "gloffscreen.h"
|
|
|
|
#include "solvespace.h"
|
|
|
|
|
2015-03-21 03:35:04 +08:00
|
|
|
GLOffscreen::GLOffscreen() : _pixels(NULL), _pixels_inv(NULL) {
|
2015-03-24 14:45:53 +08:00
|
|
|
#ifndef __APPLE__
|
2015-03-21 03:35:04 +08:00
|
|
|
if(glewInit() != GLEW_OK)
|
2015-03-21 03:53:44 +08:00
|
|
|
oops();
|
2015-03-24 14:45:53 +08:00
|
|
|
#endif
|
2015-03-21 03:53:44 +08:00
|
|
|
|
|
|
|
if(!GL_EXT_framebuffer_object)
|
|
|
|
oops();
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
glGenFramebuffersEXT(1, &_framebuffer);
|
|
|
|
glGenRenderbuffersEXT(1, &_color_renderbuffer);
|
|
|
|
glGenRenderbuffersEXT(1, &_depth_renderbuffer);
|
2015-03-21 03:35:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
GLOffscreen::~GLOffscreen() {
|
2015-03-21 03:53:44 +08:00
|
|
|
glDeleteRenderbuffersEXT(1, &_depth_renderbuffer);
|
|
|
|
glDeleteRenderbuffersEXT(1, &_color_renderbuffer);
|
|
|
|
glDeleteFramebuffersEXT(1, &_framebuffer);
|
2015-03-21 03:35:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GLOffscreen::begin(int width, int height) {
|
|
|
|
if(_width != width || _height != height) {
|
|
|
|
delete[] _pixels;
|
|
|
|
delete[] _pixels_inv;
|
|
|
|
|
|
|
|
_pixels = new uint32_t[width * height * 4];
|
|
|
|
_pixels_inv = new uint32_t[width * height * 4];
|
|
|
|
|
|
|
|
_width = width;
|
|
|
|
_height = height;
|
|
|
|
}
|
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, _framebuffer);
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, _color_renderbuffer);
|
|
|
|
glRenderbufferStorageEXT(GL_RENDERBUFFER_EXT, GL_RGBA8, _width, _height);
|
|
|
|
glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT,
|
|
|
|
GL_RENDERBUFFER_EXT, _color_renderbuffer);
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, _depth_renderbuffer);
|
|
|
|
glRenderbufferStorageEXT(GL_RENDERBUFFER_EXT, GL_DEPTH_COMPONENT24, _width, _height);
|
|
|
|
glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT,
|
|
|
|
GL_RENDERBUFFER_EXT, _depth_renderbuffer);
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
if(glCheckFramebufferStatusEXT(GL_FRAMEBUFFER_EXT) == GL_FRAMEBUFFER_COMPLETE_EXT)
|
2015-03-21 03:35:04 +08:00
|
|
|
return true;
|
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0);
|
2015-03-21 03:35:04 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-03-24 14:45:53 +08:00
|
|
|
uint8_t *GLOffscreen::end(bool flip) {
|
|
|
|
uint32_t *pixels_tgt = flip ? _pixels_inv : _pixels;
|
|
|
|
|
2015-03-21 03:35:04 +08:00
|
|
|
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
|
|
|
glReadPixels(0, 0, _width, _height,
|
|
|
|
GL_BGRA, GL_UNSIGNED_INT_8_8_8_8_REV, _pixels_inv);
|
|
|
|
#else
|
|
|
|
glReadPixels(0, 0, _width, _height,
|
|
|
|
GL_BGRA, GL_UNSIGNED_INT_8_8_8_8, _pixels_inv);
|
|
|
|
#endif
|
|
|
|
|
2015-03-24 14:45:53 +08:00
|
|
|
if(flip) {
|
|
|
|
/* in OpenGL coordinates, bottom is zero Y */
|
|
|
|
for(int i = 0; i < _height; i++)
|
|
|
|
memcpy(&_pixels[_width * i], &_pixels_inv[_width * (_height - i - 1)], _width * 4);
|
|
|
|
}
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-21 03:53:44 +08:00
|
|
|
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0);
|
2015-03-21 03:35:04 +08:00
|
|
|
|
2015-03-24 14:45:53 +08:00
|
|
|
return (uint8_t*) (flip ? _pixels : _pixels_inv);
|
2015-03-21 03:35:04 +08:00
|
|
|
}
|