2013-07-24 13:06:59 +00:00
|
|
|
/*
|
2013-07-23 16:39:07 +00:00
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2016-05-02 13:56:16 +00:00
|
|
|
* Copyright (C) 2013-2016 CERN
|
2013-07-23 16:39:07 +00:00
|
|
|
* @author Maciej Suminski <maciej.suminski@cern.ch>
|
|
|
|
*
|
|
|
|
* 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, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
2014-02-24 10:17:49 +00:00
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
2013-07-23 16:39:07 +00:00
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file opengl_compositor.cpp
|
2015-02-15 01:18:35 +00:00
|
|
|
* @brief Class that handles multitarget rendering (i.e. to different textures/surfaces) and
|
2013-07-23 16:39:07 +00:00
|
|
|
* later compositing into a single image (OpenGL flavour).
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <gal/opengl/opengl_compositor.h>
|
2016-05-02 13:56:16 +00:00
|
|
|
#include <gal/opengl/utils.h>
|
2015-02-15 01:18:35 +00:00
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <cassert>
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-10-14 14:13:35 +00:00
|
|
|
using namespace KIGFX;
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
OPENGL_COMPOSITOR::OPENGL_COMPOSITOR() :
|
2016-05-02 13:56:16 +00:00
|
|
|
m_initialized( false ), m_curBuffer( 0 ),
|
|
|
|
m_mainFbo( 0 ), m_depthBuffer( 0 ), m_curFbo( DIRECT_RENDERING )
|
2013-07-23 16:39:07 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
OPENGL_COMPOSITOR::~OPENGL_COMPOSITOR()
|
|
|
|
{
|
|
|
|
if( m_initialized )
|
|
|
|
clean();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void OPENGL_COMPOSITOR::Initialize()
|
|
|
|
{
|
|
|
|
if( m_initialized )
|
|
|
|
return;
|
|
|
|
|
|
|
|
// We need framebuffer objects for drawing the screen contents
|
|
|
|
// Generate framebuffer and a depth buffer
|
2016-05-02 13:56:16 +00:00
|
|
|
glGenFramebuffersEXT( 1, &m_mainFbo );
|
|
|
|
checkGlError( "generating framebuffer" );
|
|
|
|
bindFb( m_mainFbo );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
// Allocate memory for the depth buffer
|
|
|
|
// Attach the depth buffer to the framebuffer
|
2013-10-29 16:28:29 +00:00
|
|
|
glGenRenderbuffersEXT( 1, &m_depthBuffer );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "generating renderbuffer" );
|
2013-10-29 16:28:29 +00:00
|
|
|
glBindRenderbufferEXT( GL_RENDERBUFFER_EXT, m_depthBuffer );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "binding renderbuffer" );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2016-05-02 14:15:24 +00:00
|
|
|
glRenderbufferStorageEXT( GL_RENDERBUFFER_EXT, GL_DEPTH24_STENCIL8, m_width, m_height );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "creating renderbuffer storage" );
|
2016-05-02 14:15:24 +00:00
|
|
|
glFramebufferRenderbufferEXT( GL_FRAMEBUFFER_EXT, GL_DEPTH_STENCIL_ATTACHMENT,
|
2014-07-09 09:22:42 +00:00
|
|
|
GL_RENDERBUFFER_EXT, m_depthBuffer );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "attaching renderbuffer" );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
// Unbind the framebuffer, so by default all the rendering goes directly to the display
|
2016-05-02 13:56:16 +00:00
|
|
|
bindFb( DIRECT_RENDERING );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
m_initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void OPENGL_COMPOSITOR::Resize( unsigned int aWidth, unsigned int aHeight )
|
|
|
|
{
|
|
|
|
if( m_initialized )
|
|
|
|
clean();
|
|
|
|
|
|
|
|
m_width = aWidth;
|
|
|
|
m_height = aHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-19 09:02:38 +00:00
|
|
|
unsigned int OPENGL_COMPOSITOR::CreateBuffer()
|
2013-07-23 16:39:07 +00:00
|
|
|
{
|
2015-02-15 01:18:35 +00:00
|
|
|
assert( m_initialized );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
int maxBuffers, maxTextureSize;
|
2014-07-09 09:22:42 +00:00
|
|
|
|
2014-11-13 15:37:15 +00:00
|
|
|
// Get the maximum number of buffers
|
2014-07-09 09:22:42 +00:00
|
|
|
glGetIntegerv( GL_MAX_COLOR_ATTACHMENTS, (GLint*) &maxBuffers );
|
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
if( (int) usedBuffers() >= maxBuffers )
|
2013-10-14 09:39:21 +00:00
|
|
|
{
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "Cannot create more framebuffers. OpenGL rendering "
|
2013-11-11 09:48:49 +00:00
|
|
|
"backend requires at least 3 framebuffers. You may try to update/change "
|
2016-05-02 13:56:16 +00:00
|
|
|
"your graphic drivers." );
|
|
|
|
}
|
|
|
|
|
|
|
|
glGetIntegerv( GL_MAX_TEXTURE_SIZE, (GLint*) &maxTextureSize );
|
|
|
|
|
|
|
|
if( maxTextureSize < (int) m_width || maxTextureSize < (int) m_height )
|
|
|
|
{
|
|
|
|
throw std::runtime_error( "Requested texture size is not supported. "
|
|
|
|
"Could not create a buffer." );
|
2013-10-14 09:39:21 +00:00
|
|
|
}
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
// GL_COLOR_ATTACHMENTn are consecutive integers
|
|
|
|
GLuint attachmentPoint = GL_COLOR_ATTACHMENT0 + usedBuffers();
|
|
|
|
GLuint textureTarget;
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
// Generate the texture for the pixel storage
|
|
|
|
glGenTextures( 1, &textureTarget );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "generating framebuffer texture target" );
|
2013-07-24 13:06:59 +00:00
|
|
|
glBindTexture( GL_TEXTURE_2D, textureTarget );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "binding framebuffer texture target" );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
// Set texture parameters
|
|
|
|
glTexEnvf( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE );
|
2014-07-09 09:22:42 +00:00
|
|
|
glTexImage2D( GL_TEXTURE_2D, 0, GL_RGBA8, m_width, m_height, 0, GL_RGBA,
|
2013-07-24 13:06:59 +00:00
|
|
|
GL_UNSIGNED_BYTE, NULL );
|
2016-05-02 13:56:16 +00:00
|
|
|
checkGlError( "creating framebuffer texture" );
|
2013-07-24 13:06:59 +00:00
|
|
|
glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST );
|
|
|
|
glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
// Bind the texture to the specific attachment point, clear and rebind the screen
|
2016-05-02 13:56:16 +00:00
|
|
|
bindFb( m_mainFbo );
|
2014-07-09 09:22:42 +00:00
|
|
|
glFramebufferTexture2DEXT( GL_FRAMEBUFFER_EXT, attachmentPoint,
|
|
|
|
GL_TEXTURE_2D, textureTarget, 0 );
|
2013-08-02 08:34:23 +00:00
|
|
|
|
|
|
|
// Check the status, exit if the framebuffer can't be created
|
2013-10-29 16:28:29 +00:00
|
|
|
GLenum status = glCheckFramebufferStatusEXT( GL_FRAMEBUFFER_EXT );
|
2013-08-02 08:34:23 +00:00
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
if( status != GL_FRAMEBUFFER_COMPLETE_EXT )
|
2013-08-02 08:34:23 +00:00
|
|
|
{
|
|
|
|
switch( status )
|
|
|
|
{
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT_EXT:
|
2015-02-15 01:18:35 +00:00
|
|
|
throw std::runtime_error( "Cannot create the framebuffer." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT_EXT:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "The framebuffer attachment points are incomplete." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER_EXT:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "The framebuffer does not have at least one "
|
2015-02-15 01:18:35 +00:00
|
|
|
"image attached to it." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER_EXT:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "The framebuffer read buffer is incomplete." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_UNSUPPORTED_EXT:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "The combination of internal formats of the attached "
|
2015-02-15 01:18:35 +00:00
|
|
|
"images violates an implementation-dependent set of restrictions." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE_EXT:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "GL_RENDERBUFFER_SAMPLES is not the same for "
|
2015-02-15 01:18:35 +00:00
|
|
|
"all attached renderbuffers" );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
2013-10-29 16:28:29 +00:00
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS_EXT:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "Framebuffer incomplete layer targets errors." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2016-05-02 13:56:16 +00:00
|
|
|
throw std::runtime_error( "Cannot create the framebuffer." );
|
2013-08-02 08:34:23 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-10-14 09:39:21 +00:00
|
|
|
|
|
|
|
return 0;
|
2013-08-02 08:34:23 +00:00
|
|
|
}
|
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
ClearBuffer();
|
2013-09-04 14:25:57 +00:00
|
|
|
|
|
|
|
// Return to direct rendering (we were asked only to create a buffer, not switch to one)
|
2016-05-02 13:56:16 +00:00
|
|
|
bindFb( DIRECT_RENDERING );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
// Store the new buffer
|
|
|
|
OPENGL_BUFFER buffer = { textureTarget, attachmentPoint };
|
|
|
|
m_buffers.push_back( buffer );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
return usedBuffers();
|
2013-07-23 16:39:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void OPENGL_COMPOSITOR::SetBuffer( unsigned int aBufferHandle )
|
|
|
|
{
|
2016-05-02 13:56:16 +00:00
|
|
|
assert( m_initialized );
|
|
|
|
assert( aBufferHandle <= usedBuffers() );
|
2013-07-24 13:06:59 +00:00
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
// Either unbind the FBO for direct rendering, or bind the one with target textures
|
|
|
|
bindFb( aBufferHandle == DIRECT_RENDERING ? DIRECT_RENDERING : m_mainFbo );
|
2013-07-24 13:06:59 +00:00
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
// Switch the target texture
|
|
|
|
if( m_curFbo != DIRECT_RENDERING )
|
2013-07-24 13:06:59 +00:00
|
|
|
{
|
2016-05-02 13:56:16 +00:00
|
|
|
m_curBuffer = aBufferHandle - 1;
|
|
|
|
glDrawBuffer( m_buffers[m_curBuffer].attachmentPoint );
|
|
|
|
checkGlError( "setting draw buffer" );
|
2013-07-23 16:39:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void OPENGL_COMPOSITOR::ClearBuffer()
|
|
|
|
{
|
2015-02-15 01:18:35 +00:00
|
|
|
assert( m_initialized );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
glClearColor( 0.0f, 0.0f, 0.0f, 0.0f );
|
2016-05-02 14:15:24 +00:00
|
|
|
glClear( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT );
|
2013-07-23 16:39:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-24 13:06:59 +00:00
|
|
|
void OPENGL_COMPOSITOR::DrawBuffer( unsigned int aBufferHandle )
|
2013-07-23 16:39:07 +00:00
|
|
|
{
|
2015-02-15 01:18:35 +00:00
|
|
|
assert( m_initialized );
|
|
|
|
assert( aBufferHandle != 0 && aBufferHandle <= usedBuffers() );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
// Switch to the main framebuffer and blit the scene
|
2016-05-02 13:56:16 +00:00
|
|
|
bindFb( DIRECT_RENDERING );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
// Depth test has to be disabled to make transparency working
|
|
|
|
glDisable( GL_DEPTH_TEST );
|
|
|
|
glBlendFunc( GL_ONE, GL_ONE_MINUS_SRC_ALPHA );
|
|
|
|
|
|
|
|
// Enable texturing and bind the main texture
|
|
|
|
glEnable( GL_TEXTURE_2D );
|
|
|
|
glBindTexture( GL_TEXTURE_2D, m_buffers[aBufferHandle - 1].textureTarget );
|
|
|
|
|
|
|
|
// Draw a full screen quad with the texture
|
|
|
|
glMatrixMode( GL_MODELVIEW );
|
|
|
|
glPushMatrix();
|
|
|
|
glLoadIdentity();
|
|
|
|
glMatrixMode( GL_PROJECTION );
|
|
|
|
glPushMatrix();
|
|
|
|
glLoadIdentity();
|
|
|
|
|
|
|
|
glBegin( GL_TRIANGLES );
|
|
|
|
glTexCoord2f( 0.0f, 1.0f );
|
2013-07-24 13:06:59 +00:00
|
|
|
glVertex2f( -1.0f, -1.0f );
|
2013-07-23 16:39:07 +00:00
|
|
|
glTexCoord2f( 1.0f, 1.0f );
|
2013-07-24 13:06:59 +00:00
|
|
|
glVertex2f( 1.0f, -1.0f );
|
2013-07-23 16:39:07 +00:00
|
|
|
glTexCoord2f( 1.0f, 0.0f );
|
2013-07-24 13:06:59 +00:00
|
|
|
glVertex2f( 1.0f, 1.0f );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
|
|
|
glTexCoord2f( 0.0f, 1.0f );
|
2013-07-24 13:06:59 +00:00
|
|
|
glVertex2f( -1.0f, -1.0f );
|
2013-07-23 16:39:07 +00:00
|
|
|
glTexCoord2f( 1.0f, 0.0f );
|
2013-07-24 13:06:59 +00:00
|
|
|
glVertex2f( 1.0f, 1.0f );
|
2013-07-23 16:39:07 +00:00
|
|
|
glTexCoord2f( 0.0f, 0.0f );
|
2013-07-24 13:06:59 +00:00
|
|
|
glVertex2f( -1.0f, 1.0f );
|
2013-07-23 16:39:07 +00:00
|
|
|
glEnd();
|
|
|
|
|
|
|
|
glPopMatrix();
|
|
|
|
glMatrixMode( GL_MODELVIEW );
|
|
|
|
glPopMatrix();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
void OPENGL_COMPOSITOR::bindFb( unsigned int aFb ) {
|
|
|
|
// Currently there are only 2 valid FBOs
|
|
|
|
assert( aFb == DIRECT_RENDERING || aFb == m_mainFbo );
|
|
|
|
|
|
|
|
if( m_curFbo != aFb )
|
|
|
|
{
|
|
|
|
glBindFramebufferEXT( GL_FRAMEBUFFER, aFb );
|
|
|
|
checkGlError( "switching framebuffer" );
|
|
|
|
m_curFbo = aFb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-23 16:39:07 +00:00
|
|
|
void OPENGL_COMPOSITOR::clean()
|
|
|
|
{
|
2015-02-15 01:18:35 +00:00
|
|
|
assert( m_initialized );
|
2013-07-23 16:39:07 +00:00
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
bindFb( DIRECT_RENDERING );
|
2013-10-14 14:13:35 +00:00
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
for( OPENGL_BUFFERS::const_iterator it = m_buffers.begin(); it != m_buffers.end(); ++it )
|
2013-07-23 16:39:07 +00:00
|
|
|
{
|
|
|
|
glDeleteTextures( 1, &it->textureTarget );
|
|
|
|
}
|
2013-07-30 16:29:54 +00:00
|
|
|
|
2013-07-23 16:39:07 +00:00
|
|
|
m_buffers.clear();
|
|
|
|
|
2016-05-02 13:56:16 +00:00
|
|
|
glDeleteFramebuffersEXT( 1, &m_mainFbo );
|
2014-07-09 09:22:42 +00:00
|
|
|
glDeleteRenderbuffersEXT( 1, &m_depthBuffer );
|
|
|
|
|
2013-07-23 16:39:07 +00:00
|
|
|
m_initialized = false;
|
|
|
|
}
|