2013-06-23 07:49:34 +00:00
/*
* * r_opengl . cpp
* *
* * OpenGL system interface
* *
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* * Copyright 2005 Tim Stump
2013-08-18 12:16:33 +00:00
* * Copyright 2005 - 2013 Christoph Oelckers
2013-06-23 07:49:34 +00:00
* * All rights reserved .
* *
* * Redistribution and use in source and binary forms , with or without
* * modification , are permitted provided that the following conditions
* * are met :
* *
* * 1. Redistributions of source code must retain the above copyright
* * notice , this list of conditions and the following disclaimer .
* * 2. Redistributions in binary form must reproduce the above copyright
* * notice , this list of conditions and the following disclaimer in the
* * documentation and / or other materials provided with the distribution .
* * 3. The name of the author may not be used to endorse or promote products
* * derived from this software without specific prior written permission .
* * 4. Full disclosure of the entire project ' s source code , except for third
* * party libraries is mandatory . ( NOTE : This clause is non - negotiable ! )
* *
* * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* * IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* * INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* * NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* * DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* * THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* * ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* * THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* *
*/
# include "gl/system/gl_system.h"
# include "tarray.h"
# include "doomtype.h"
# include "m_argv.h"
# include "zstring.h"
# include "version.h"
# include "i_system.h"
2013-09-02 06:43:56 +00:00
# include "v_text.h"
2016-04-26 16:24:02 +00:00
# include "r_data/r_translate.h"
2013-09-03 16:29:39 +00:00
# include "gl/system/gl_interface.h"
2013-06-23 07:49:34 +00:00
# include "gl/system/gl_cvars.h"
2016-05-01 11:09:13 +00:00
void gl_PatchMenu ( ) ;
2013-06-23 07:49:34 +00:00
static TArray < FString > m_Extensions ;
2013-08-18 13:41:52 +00:00
RenderContext gl ;
2013-06-23 07:49:34 +00:00
//==========================================================================
//
//
//
//==========================================================================
static void CollectExtensions ( )
{
2014-07-15 00:26:23 +00:00
const char * extension ;
2013-06-23 07:49:34 +00:00
2014-07-15 00:26:23 +00:00
int max = 0 ;
glGetIntegerv ( GL_NUM_EXTENSIONS , & max ) ;
2013-06-23 07:49:34 +00:00
2016-04-30 14:23:32 +00:00
if ( 0 = = max )
2013-06-23 07:49:34 +00:00
{
2016-04-30 14:23:32 +00:00
// Try old method to collect extensions
const char * supported = ( char * ) glGetString ( GL_EXTENSIONS ) ;
if ( nullptr ! = supported )
{
char * extensions = new char [ strlen ( supported ) + 1 ] ;
strcpy ( extensions , supported ) ;
char * extension = strtok ( extensions , " " ) ;
while ( extension )
{
m_Extensions . Push ( FString ( extension ) ) ;
extension = strtok ( nullptr , " " ) ;
}
delete [ ] extensions ;
}
}
else
{
// Use modern method to collect extensions
for ( int i = 0 ; i < max ; i + + )
{
extension = ( const char * ) glGetStringi ( GL_EXTENSIONS , i ) ;
m_Extensions . Push ( FString ( extension ) ) ;
}
2013-06-23 07:49:34 +00:00
}
}
//==========================================================================
//
//
//
//==========================================================================
static bool CheckExtension ( const char * ext )
{
for ( unsigned int i = 0 ; i < m_Extensions . Size ( ) ; + + i )
{
if ( m_Extensions [ i ] . CompareNoCase ( ext ) = = 0 ) return true ;
}
return false ;
}
2013-08-18 13:41:52 +00:00
2013-06-23 07:49:34 +00:00
//==========================================================================
//
//
//
//==========================================================================
2013-08-18 13:41:52 +00:00
static void InitContext ( )
2013-06-23 07:49:34 +00:00
{
2013-08-18 13:41:52 +00:00
gl . flags = 0 ;
}
//==========================================================================
//
//
//
//==========================================================================
2016-04-26 11:50:05 +00:00
# define FUDGE_FUNC(name, ext) if (_ptrc_##name == NULL) _ptrc_##name = _ptrc_##name##ext;
2013-08-18 13:41:52 +00:00
void gl_LoadExtensions ( )
{
InitContext ( ) ;
2013-06-23 07:49:34 +00:00
CollectExtensions ( ) ;
2014-06-13 23:24:28 +00:00
const char * version = Args - > CheckValue ( " -glversion " ) ;
2016-04-26 11:50:05 +00:00
const char * glversion = ( const char * ) glGetString ( GL_VERSION ) ;
if ( version = = NULL )
{
version = glversion ;
}
else
{
double v1 = strtod ( version , NULL ) ;
double v2 = strtod ( glversion , NULL ) ;
if ( v2 < v1 ) version = glversion ;
else Printf ( " Emulating OpenGL v %s \n " , version ) ;
}
gl . version = strtod ( version , NULL ) + 0.01f ;
2014-06-13 23:24:28 +00:00
2016-08-22 13:31:23 +00:00
// Don't even start if it's lower than 2.0 or no framebuffers are available
2016-04-26 11:50:05 +00:00
if ( ( gl . version < 2.0 | | ! CheckExtension ( " GL_EXT_framebuffer_object " ) ) & & gl . version < 3.0 )
2013-06-23 07:49:34 +00:00
{
2016-04-26 11:50:05 +00:00
I_FatalError ( " Unsupported OpenGL version. \n At least OpenGL 2.0 with framebuffer support is required to run " GAMENAME " . \n " ) ;
2013-06-23 07:49:34 +00:00
}
2014-07-15 00:26:23 +00:00
// add 0.01 to account for roundoff errors making the number a tad smaller than the actual version
gl . glslversion = strtod ( ( char * ) glGetString ( GL_SHADING_LANGUAGE_VERSION ) , NULL ) + 0.01f ;
2013-06-23 07:49:34 +00:00
2014-07-27 11:46:35 +00:00
gl . vendorstring = ( char * ) glGetString ( GL_VENDOR ) ;
2016-04-26 11:50:05 +00:00
gl . lightmethod = LM_SOFTWARE ;
2016-08-06 10:03:16 +00:00
gl . buffermethod = BM_CLIENTARRAY ;
2013-06-23 07:49:34 +00:00
2016-05-03 11:39:41 +00:00
if ( ( gl . version > = 3.3f | | CheckExtension ( " GL_ARB_sampler_objects " ) ) & & ! Args - > CheckParm ( " -nosampler " ) )
2016-04-23 13:47:51 +00:00
{
gl . flags | = RFL_SAMPLER_OBJECTS ;
}
2016-04-26 11:50:05 +00:00
// Buffer lighting is only feasible with GLSL 1.3 and higher, even if 1.2 supports the extension.
if ( gl . version > 3.0f & & ( gl . version > = 3.3f | | CheckExtension ( " GL_ARB_uniform_buffer_object " ) ) )
{
gl . lightmethod = LM_DEFERRED ;
2016-08-29 09:33:20 +00:00
gl . buffermethod = BM_DEFERRED ;
2016-04-26 11:50:05 +00:00
}
if ( CheckExtension ( " GL_ARB_texture_compression " ) ) gl . flags | = RFL_TEXTURE_COMPRESSION ;
if ( CheckExtension ( " GL_EXT_texture_compression_s3tc " ) ) gl . flags | = RFL_TEXTURE_COMPRESSION_S3TC ;
2016-04-23 13:47:51 +00:00
2016-08-06 11:06:55 +00:00
if ( Args - > CheckParm ( " -noshader " ) /* || gl.glslversion < 1.2f*/ )
2016-04-26 11:50:05 +00:00
{
2016-04-26 14:26:34 +00:00
gl . version = 2.11f ;
gl . glslversion = 0 ;
2016-04-26 14:44:03 +00:00
gl . lightmethod = LM_SOFTWARE ;
2016-08-08 10:55:09 +00:00
gl . flags | = RFL_NO_CLIP_PLANES ;
2016-04-26 11:50:05 +00:00
}
2016-04-26 13:01:23 +00:00
else if ( gl . version < 3.0f )
2016-04-26 11:50:05 +00:00
{
2016-08-29 08:43:03 +00:00
if ( CheckExtension ( " GL_NV_GPU_shader4 " ) | | CheckExtension ( " GL_EXT_GPU_shader4 " ) ) gl . glslversion = 1.21f ; // for pre-3.0 drivers that support capable hardware. Needed for Apple.
else
{
gl . buffermethod = BM_CLIENTARRAY ;
gl . glslversion = 0 ;
}
2016-07-29 19:31:20 +00:00
if ( ! CheckExtension ( " GL_EXT_packed_float " ) ) gl . flags | = RFL_NO_RGBA16F ;
if ( ! CheckExtension ( " GL_EXT_packed_depth_stencil " ) ) gl . flags | = RFL_NO_DEPTHSTENCIL ;
2016-08-08 10:55:09 +00:00
gl . flags | = RFL_NO_CLIP_PLANES ;
2016-04-26 11:50:05 +00:00
}
else if ( gl . version < 4.f )
{
2016-08-08 10:55:09 +00:00
# ifdef _WIN32
2016-04-26 11:50:05 +00:00
if ( strstr ( gl . vendorstring , " ATI Tech " ) )
{
2016-08-08 10:55:09 +00:00
gl . flags | = RFL_NO_CLIP_PLANES ; // gl_ClipDistance is horribly broken on ATI GL3 drivers for Windows.
2016-04-26 11:50:05 +00:00
}
2016-08-08 10:55:09 +00:00
# endif
2016-04-26 11:50:05 +00:00
}
2016-08-08 10:55:09 +00:00
else if ( gl . version < 4.5f )
2014-08-19 13:56:33 +00:00
{
// don't use GL 4.x features when running in GL 3 emulation mode.
2014-09-17 07:01:16 +00:00
if ( CheckExtension ( " GL_ARB_buffer_storage " ) )
{
// work around a problem with older AMD drivers: Their implementation of shader storage buffer objects is piss-poor and does not match uniform buffers even closely.
// Recent drivers, GL 4.4 don't have this problem, these can easily be recognized by also supporting the GL_ARB_buffer_storage extension.
if ( CheckExtension ( " GL_ARB_shader_storage_buffer_object " ) )
{
2014-12-01 08:58:23 +00:00
// Shader storage buffer objects are broken on current Intel drivers.
if ( strstr ( gl . vendorstring , " Intel " ) = = NULL )
{
gl . flags | = RFL_SHADER_STORAGE_BUFFER ;
}
2014-09-17 07:01:16 +00:00
}
gl . flags | = RFL_BUFFER_STORAGE ;
2016-04-26 11:50:05 +00:00
gl . lightmethod = LM_DIRECT ;
2016-08-06 10:03:16 +00:00
gl . buffermethod = BM_PERSISTENT ;
2016-04-26 11:50:05 +00:00
}
else
{
2016-04-26 13:01:23 +00:00
gl . version = 3.3f ;
2014-09-17 07:01:16 +00:00
}
2014-08-19 13:56:33 +00:00
}
2016-08-08 10:55:09 +00:00
else
{
// Assume that everything works without problems on GL 4.5 drivers where these things are core features.
gl . flags | = RFL_SHADER_STORAGE_BUFFER | RFL_BUFFER_STORAGE ;
gl . lightmethod = LM_DIRECT ;
gl . buffermethod = BM_PERSISTENT ;
}
2016-04-26 11:50:05 +00:00
2016-08-17 15:37:13 +00:00
if ( gl . version > = 4.3f | | CheckExtension ( " GL_ARB_invalidate_subdata " ) ) gl . flags | = RFL_INVALIDATE_BUFFER ;
2016-08-22 17:25:13 +00:00
if ( gl . version > = 4.3f | | CheckExtension ( " GL_KHR_debug " ) ) gl . flags | = RFL_DEBUG ;
2016-08-17 15:37:13 +00:00
2016-04-26 11:50:05 +00:00
const char * lm = Args - > CheckValue ( " -lightmethod " ) ;
if ( lm ! = NULL )
{
if ( ! stricmp ( lm , " deferred " ) & & gl . lightmethod = = LM_DIRECT ) gl . lightmethod = LM_DEFERRED ;
if ( ! stricmp ( lm , " textured " ) ) gl . lightmethod = LM_SOFTWARE ;
}
2016-08-06 10:03:16 +00:00
lm = Args - > CheckValue ( " -buffermethod " ) ;
if ( lm ! = NULL )
{
2016-08-22 13:31:23 +00:00
if ( ! stricmp ( lm , " deferred " ) & & gl . buffermethod = = BM_PERSISTENT ) gl . buffermethod = BM_DEFERRED ;
2016-08-06 10:03:16 +00:00
if ( ! stricmp ( lm , " clientarray " ) ) gl . buffermethod = BM_CLIENTARRAY ;
}
2014-09-14 21:01:57 +00:00
int v ;
2016-04-30 14:31:09 +00:00
if ( gl . lightmethod ! = LM_SOFTWARE & & ! ( gl . flags & RFL_SHADER_STORAGE_BUFFER ) )
{
glGetIntegerv ( GL_MAX_FRAGMENT_UNIFORM_COMPONENTS , & v ) ;
gl . maxuniforms = v ;
glGetIntegerv ( GL_MAX_UNIFORM_BLOCK_SIZE , & v ) ;
gl . maxuniformblock = v ;
glGetIntegerv ( GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT , & v ) ;
gl . uniformblockalignment = v ;
}
else
{
gl . maxuniforms = 0 ;
gl . maxuniformblock = 0 ;
gl . uniformblockalignment = 0 ;
}
2016-04-26 11:50:05 +00:00
glGetIntegerv ( GL_MAX_TEXTURE_SIZE , & gl . max_texturesize ) ;
2013-06-23 07:49:34 +00:00
glPixelStorei ( GL_UNPACK_ALIGNMENT , 1 ) ;
2016-04-26 11:50:05 +00:00
// fudge a bit with the framebuffer stuff to avoid redundancies in the main code. Some of the older cards do not have the ARB stuff but the calls are nearly identical.
FUDGE_FUNC ( glGenerateMipmap , EXT ) ;
FUDGE_FUNC ( glGenFramebuffers , EXT ) ;
FUDGE_FUNC ( glBindFramebuffer , EXT ) ;
FUDGE_FUNC ( glDeleteFramebuffers , EXT ) ;
FUDGE_FUNC ( glFramebufferTexture2D , EXT ) ;
FUDGE_FUNC ( glGenerateMipmap , EXT ) ;
FUDGE_FUNC ( glGenFramebuffers , EXT ) ;
FUDGE_FUNC ( glBindFramebuffer , EXT ) ;
FUDGE_FUNC ( glDeleteFramebuffers , EXT ) ;
FUDGE_FUNC ( glFramebufferTexture2D , EXT ) ;
FUDGE_FUNC ( glFramebufferRenderbuffer , EXT ) ;
FUDGE_FUNC ( glGenRenderbuffers , EXT ) ;
FUDGE_FUNC ( glDeleteRenderbuffers , EXT ) ;
FUDGE_FUNC ( glRenderbufferStorage , EXT ) ;
FUDGE_FUNC ( glBindRenderbuffer , EXT ) ;
2016-08-26 06:30:47 +00:00
FUDGE_FUNC ( glCheckFramebufferStatus , EXT ) ;
2016-05-01 11:09:13 +00:00
gl_PatchMenu ( ) ;
2013-06-23 07:49:34 +00:00
}
//==========================================================================
//
//
//
//==========================================================================
2013-08-18 13:41:52 +00:00
void gl_PrintStartupLog ( )
2013-06-23 07:49:34 +00:00
{
2015-04-27 19:25:16 +00:00
int v = 0 ;
2016-01-30 22:01:11 +00:00
if ( gl . version > = 3.2 ) glGetIntegerv ( GL_CONTEXT_PROFILE_MASK , & v ) ;
2014-08-23 23:09:44 +00:00
2013-06-23 07:49:34 +00:00
Printf ( " GL_VENDOR: %s \n " , glGetString ( GL_VENDOR ) ) ;
Printf ( " GL_RENDERER: %s \n " , glGetString ( GL_RENDERER ) ) ;
2014-08-23 23:09:44 +00:00
Printf ( " GL_VERSION: %s (%s profile) \n " , glGetString ( GL_VERSION ) , ( v & GL_CONTEXT_CORE_PROFILE_BIT ) ? " Core " : " Compatibility " ) ;
2013-06-23 07:49:34 +00:00
Printf ( " GL_SHADING_LANGUAGE_VERSION: %s \n " , glGetString ( GL_SHADING_LANGUAGE_VERSION ) ) ;
2014-07-15 00:26:23 +00:00
Printf ( " GL_EXTENSIONS: " ) ;
for ( unsigned i = 0 ; i < m_Extensions . Size ( ) ; i + + )
{
Printf ( " %s " , m_Extensions [ i ] . GetChars ( ) ) ;
}
2013-06-23 07:49:34 +00:00
2014-04-06 12:35:44 +00:00
glGetIntegerv ( GL_MAX_TEXTURE_SIZE , & v ) ;
2014-07-15 00:26:23 +00:00
Printf ( " \n Max. texture size: %d \n " , v ) ;
2013-06-23 07:49:34 +00:00
glGetIntegerv ( GL_MAX_TEXTURE_IMAGE_UNITS , & v ) ;
Printf ( " Max. texture units: %d \n " , v ) ;
glGetIntegerv ( GL_MAX_VARYING_FLOATS , & v ) ;
Printf ( " Max. varying: %d \n " , v ) ;
2016-04-30 14:31:09 +00:00
if ( gl . lightmethod ! = LM_SOFTWARE & & ! ( gl . flags & RFL_SHADER_STORAGE_BUFFER ) )
{
glGetIntegerv ( GL_MAX_UNIFORM_BLOCK_SIZE , & v ) ;
Printf ( " Max. uniform block size: %d \n " , v ) ;
glGetIntegerv ( GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT , & v ) ;
Printf ( " Uniform block alignment: %d \n " , v ) ;
}
if ( gl . flags & RFL_SHADER_STORAGE_BUFFER )
{
glGetIntegerv ( GL_MAX_COMBINED_SHADER_STORAGE_BLOCKS , & v ) ;
Printf ( " Max. combined shader storage blocks: %d \n " , v ) ;
glGetIntegerv ( GL_MAX_VERTEX_SHADER_STORAGE_BLOCKS , & v ) ;
Printf ( " Max. vertex shader storage blocks: %d \n " , v ) ;
}
2013-06-23 07:49:34 +00:00
2016-04-26 16:24:02 +00:00
// For shader-less, the special alphatexture translation must be changed to actually set the alpha, because it won't get translated by a shader.
if ( gl . glslversion = = 0 )
{
FRemapTable * remap = translationtables [ TRANSLATION_Standard ] [ 8 ] ;
for ( int i = 0 ; i < 256 ; i + + )
{
remap - > Remap [ i ] = i ;
2016-04-26 17:11:32 +00:00
remap - > Palette [ i ] = PalEntry ( i , 255 , 255 , 255 ) ;
2016-04-26 16:24:02 +00:00
}
}
2014-07-30 21:13:16 +00:00
2013-06-23 07:49:34 +00:00
}