2013-04-17 23:29:41 -04:00
// Copyright 2013 Dolphin Emulator Project
// Licensed under GPLv2
// Refer to the license.txt file included.
2011-12-09 17:30:05 -06:00
2011-11-30 21:00:21 -06:00
# include "ProgramShaderCache.h"
2013-06-11 08:41:30 -05:00
# include "DriverDetails.h"
2011-12-10 15:58:44 -06:00
# include "MathUtil.h"
2013-01-31 23:11:53 +01:00
# include "StreamBuffer.h"
2013-02-13 13:12:19 +01:00
# include "Debugger.h"
# include "Statistics.h"
2013-03-04 10:20:55 +01:00
# include "ImageWrite.h"
2013-03-25 15:14:24 +01:00
# include "Render.h"
2013-10-07 17:19:47 +02:00
# include "PixelShaderManager.h"
# include "VertexShaderManager.h"
2011-12-10 15:58:44 -06:00
2011-11-30 21:00:21 -06:00
namespace OGL
{
2011-12-21 00:15:48 -06:00
2013-02-22 10:25:38 +01:00
static const u32 UBO_LENGTH = 32 * 1024 * 1024 ;
2013-01-25 13:20:42 +01:00
2013-01-14 13:58:11 +01:00
u32 ProgramShaderCache : : s_ubo_buffer_size ;
2013-10-07 17:19:47 +02:00
s32 ProgramShaderCache : : s_ubo_align ;
2011-12-26 00:15:54 -05:00
2013-01-31 23:11:53 +01:00
static StreamBuffer * s_buffer ;
2013-05-04 23:30:13 +02:00
static int num_failures = 0 ;
2013-01-31 23:11:53 +01:00
2013-02-13 21:34:48 +01:00
LinearDiskCache < SHADERUID , u8 > g_program_disk_cache ;
static GLuint CurrentProgram = 0 ;
ProgramShaderCache : : PCache ProgramShaderCache : : pshaders ;
2013-02-13 13:12:19 +01:00
ProgramShaderCache : : PCacheEntry * ProgramShaderCache : : last_entry ;
SHADERUID ProgramShaderCache : : last_uid ;
2013-04-29 21:00:39 +02:00
UidChecker < PixelShaderUid , PixelShaderCode > ProgramShaderCache : : pixel_uid_checker ;
UidChecker < VertexShaderUid , VertexShaderCode > ProgramShaderCache : : vertex_uid_checker ;
2013-01-28 15:32:38 -06:00
2013-02-13 21:34:48 +01:00
static char s_glsl_header [ 1024 ] = " " ;
2013-10-07 17:19:47 +02:00
// Annoying sure, can be removed once we drop our UBO workaround
2011-12-26 00:15:54 -05:00
const char * UniformNames [ NUM_UNIFORMS ] =
{
// PIXEL SHADER UNIFORMS
I_COLORS ,
I_KCOLORS ,
I_ALPHA ,
I_TEXDIMS ,
I_ZBIAS ,
I_INDTEXSCALE ,
I_INDTEXMTX ,
I_FOG ,
I_PLIGHTS ,
I_PMATERIALS ,
// VERTEX SHADER UNIFORMS
I_POSNORMALMATRIX ,
I_PROJECTION ,
I_MATERIALS ,
I_LIGHTS ,
I_TEXMATRICES ,
I_TRANSFORMMATRICES ,
I_NORMALMATRICES ,
I_POSTTRANSFORMMATRICES ,
I_DEPTHPARAMS ,
} ;
2013-10-08 14:34:42 +02:00
const static int PSVar_Loc [ ] = {
offsetof ( PixelShaderConstants , colors ) / 16 ,
offsetof ( PixelShaderConstants , kcolors ) / 16 ,
offsetof ( PixelShaderConstants , alpha ) / 16 ,
offsetof ( PixelShaderConstants , texdims ) / 16 ,
offsetof ( PixelShaderConstants , zbias ) / 16 ,
offsetof ( PixelShaderConstants , indtexscale ) / 16 ,
offsetof ( PixelShaderConstants , indtexmtx ) / 16 ,
offsetof ( PixelShaderConstants , fog ) / 16 ,
offsetof ( PixelShaderConstants , plights ) / 16 ,
offsetof ( PixelShaderConstants , pmaterials ) / 16 ,
2013-10-07 17:19:47 +02:00
} ;
2013-10-08 14:34:42 +02:00
const static int VSVar_Loc [ ] = {
offsetof ( VertexShaderConstants , posnormalmatrix ) / 16 ,
offsetof ( VertexShaderConstants , projection ) / 16 ,
offsetof ( VertexShaderConstants , materials ) / 16 ,
offsetof ( VertexShaderConstants , lights ) / 16 ,
offsetof ( VertexShaderConstants , texmatrices ) / 16 ,
offsetof ( VertexShaderConstants , transformmatrices ) / 16 ,
offsetof ( VertexShaderConstants , normalmatrices ) / 16 ,
offsetof ( VertexShaderConstants , posttransformmatrices ) / 16 ,
offsetof ( VertexShaderConstants , depthparams ) / 16 ,
} ;
2013-10-07 17:19:47 +02:00
// End of UBO workaround
2013-02-13 13:12:19 +01:00
void SHADER : : SetProgramVariables ( )
2011-12-26 00:15:54 -05:00
{
2013-02-13 13:12:19 +01:00
// glsl shader must be bind to set samplers
Bind ( ) ;
2013-10-29 01:23:17 -04:00
// Bind UBO
2014-01-05 10:38:45 +01:00
if ( g_ActiveConfig . backend_info . bSupportsGLSLUBO & & ! g_ActiveConfig . backend_info . bSupportShadingLanguage420pack )
2011-12-11 11:08:18 +01:00
{
2013-02-13 13:12:19 +01:00
GLint PSBlock_id = glGetUniformBlockIndex ( glprogid , " PSBlock " ) ;
GLint VSBlock_id = glGetUniformBlockIndex ( glprogid , " VSBlock " ) ;
2013-10-29 01:23:17 -04:00
2013-01-02 16:56:08 +01:00
if ( PSBlock_id ! = - 1 )
2013-02-13 13:12:19 +01:00
glUniformBlockBinding ( glprogid , PSBlock_id , 1 ) ;
2013-01-02 16:56:08 +01:00
if ( VSBlock_id ! = - 1 )
2013-02-13 13:12:19 +01:00
glUniformBlockBinding ( glprogid , VSBlock_id , 2 ) ;
2011-12-26 00:15:54 -05:00
}
2013-10-29 01:23:17 -04:00
2013-08-23 17:52:47 +02:00
// UBO workaround
if ( ! g_ActiveConfig . backend_info . bSupportsGLSLUBO )
{
2014-01-05 09:52:26 +01:00
for ( int a = 0 ; a < NUM_UNIFORMS ; + + a )
{
UniformLocations [ a ] = glGetUniformLocation ( glprogid , UniformNames [ a ] ) ;
UniformSize [ a ] = 0 ;
}
2013-08-23 17:52:47 +02:00
int max_uniforms = 0 ;
char name [ 50 ] ;
int size ;
2013-10-29 01:23:17 -04:00
2013-08-23 17:52:47 +02:00
glGetProgramiv ( glprogid , GL_ACTIVE_UNIFORMS , & max_uniforms ) ;
for ( int i = 0 ; i < max_uniforms ; i + + )
{
glGetActiveUniform ( glprogid , i , sizeof ( name ) , NULL , & size , NULL , name ) ;
for ( int a = 0 ; a < NUM_UNIFORMS ; + + a )
{
if ( strstr ( name , UniformNames [ a ] ) )
{
UniformSize [ a ] = size ;
break ;
}
}
}
}
2011-12-11 11:14:02 +01:00
2013-01-02 16:56:08 +01:00
// Bind Texture Sampler
2013-01-19 00:39:31 +01:00
for ( int a = 0 ; a < = 9 ; + + a )
2011-12-26 00:15:54 -05:00
{
2013-01-19 00:12:02 +01:00
char name [ 8 ] ;
snprintf ( name , 8 , " samp%d " , a ) ;
2013-10-29 01:23:17 -04:00
2013-01-02 16:56:08 +01:00
// Still need to get sampler locations since we aren't binding them statically in the shaders
2013-02-13 13:12:19 +01:00
int loc = glGetUniformLocation ( glprogid , name ) ;
2013-01-19 00:12:02 +01:00
if ( loc ! = - 1 )
glUniform1i ( loc , a ) ;
2012-12-31 02:34:27 +01:00
}
2013-01-02 16:56:08 +01:00
2012-12-31 02:34:27 +01:00
}
2013-02-13 13:12:19 +01:00
void SHADER : : SetProgramBindings ( )
2012-12-31 02:34:27 +01:00
{
2013-01-14 20:00:33 +01:00
if ( g_ActiveConfig . backend_info . bSupportsDualSourceBlend )
2012-12-31 02:34:27 +01:00
{
2013-01-02 16:56:08 +01:00
// So we do support extended blending
// So we need to set a few more things here.
// Bind our out locations
2013-08-20 14:00:24 +02:00
# ifndef USE_GLES3
2013-02-13 13:12:19 +01:00
glBindFragDataLocationIndexed ( glprogid , 0 , 0 , " ocol0 " ) ;
glBindFragDataLocationIndexed ( glprogid , 0 , 1 , " ocol1 " ) ;
2013-05-06 06:43:04 -05:00
# endif
2013-08-20 14:00:24 +02:00
}
2012-12-31 02:34:27 +01:00
// Need to set some attribute locations
2013-02-13 13:12:19 +01:00
glBindAttribLocation ( glprogid , SHADER_POSITION_ATTRIB , " rawpos " ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
glBindAttribLocation ( glprogid , SHADER_POSMTX_ATTRIB , " fposmtx " ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
glBindAttribLocation ( glprogid , SHADER_COLOR0_ATTRIB , " color0 " ) ;
glBindAttribLocation ( glprogid , SHADER_COLOR1_ATTRIB , " color1 " ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
glBindAttribLocation ( glprogid , SHADER_NORM0_ATTRIB , " rawnorm0 " ) ;
glBindAttribLocation ( glprogid , SHADER_NORM1_ATTRIB , " rawnorm1 " ) ;
glBindAttribLocation ( glprogid , SHADER_NORM2_ATTRIB , " rawnorm2 " ) ;
2013-10-29 01:23:17 -04:00
2013-01-14 22:59:08 +01:00
for ( int i = 0 ; i < 8 ; i + + ) {
char attrib_name [ 8 ] ;
snprintf ( attrib_name , 8 , " tex%d " , i ) ;
2013-02-13 13:12:19 +01:00
glBindAttribLocation ( glprogid , SHADER_TEXTURE0_ATTRIB + i , attrib_name ) ;
2013-01-14 22:59:08 +01:00
}
2011-12-26 00:15:54 -05:00
}
2011-12-11 11:14:02 +01:00
2013-02-13 13:12:19 +01:00
void SHADER : : Bind ( )
{
if ( CurrentProgram ! = glprogid )
{
glUseProgram ( glprogid ) ;
CurrentProgram = glprogid ;
}
}
void ProgramShaderCache : : UploadConstants ( )
2011-12-26 00:15:54 -05:00
{
2013-10-07 17:19:47 +02:00
if ( g_ActiveConfig . backend_info . bSupportsGLSLUBO )
{
if ( PixelShaderManager : : dirty | | VertexShaderManager : : dirty )
{
s_buffer - > Alloc ( s_ubo_buffer_size ) ;
2013-10-16 11:27:58 +00:00
if ( DriverDetails : : HasBug ( DriverDetails : : BUG_BROKENBUFFERSTREAM ) )
{
// This is just a hack to support our BUFFERDATA upload method
// as it's broken to uploaded in a splited way
static u8 * tmpbuffer = new u8 [ s_ubo_buffer_size ] ;
memcpy ( tmpbuffer , & PixelShaderManager : : constants , sizeof ( PixelShaderConstants ) ) ;
memcpy ( tmpbuffer + ROUND_UP ( sizeof ( PixelShaderConstants ) , s_ubo_align ) , & VertexShaderManager : : constants , sizeof ( VertexShaderConstants ) ) ;
size_t offset = s_buffer - > Upload ( tmpbuffer , s_ubo_buffer_size ) ;
2013-10-29 01:23:17 -04:00
glBindBufferRange ( GL_UNIFORM_BUFFER , 1 ,
2013-10-16 11:27:58 +00:00
s_buffer - > getBuffer ( ) , offset , sizeof ( PixelShaderConstants ) ) ;
2013-10-29 01:23:17 -04:00
glBindBufferRange ( GL_UNIFORM_BUFFER , 2 ,
2013-10-16 11:27:58 +00:00
s_buffer - > getBuffer ( ) , offset + ROUND_UP ( sizeof ( PixelShaderConstants ) , s_ubo_align ) , sizeof ( VertexShaderConstants ) ) ;
}
else
{
size_t offset = s_buffer - > Upload ( ( u8 * ) & PixelShaderManager : : constants , ROUND_UP ( sizeof ( PixelShaderConstants ) , s_ubo_align ) ) ;
2013-10-29 01:23:17 -04:00
glBindBufferRange ( GL_UNIFORM_BUFFER , 1 ,
2013-10-16 11:27:58 +00:00
s_buffer - > getBuffer ( ) , offset , sizeof ( PixelShaderConstants ) ) ;
offset = s_buffer - > Upload ( ( u8 * ) & VertexShaderManager : : constants , ROUND_UP ( sizeof ( VertexShaderConstants ) , s_ubo_align ) ) ;
2013-10-29 01:23:17 -04:00
glBindBufferRange ( GL_UNIFORM_BUFFER , 2 ,
2013-10-16 11:27:58 +00:00
s_buffer - > getBuffer ( ) , offset , sizeof ( VertexShaderConstants ) ) ;
}
2013-10-07 17:19:47 +02:00
PixelShaderManager : : dirty = false ;
VertexShaderManager : : dirty = false ;
2013-10-29 01:23:17 -04:00
2013-10-07 17:19:47 +02:00
ADDSTAT ( stats . thisFrame . bytesUniformStreamed , s_ubo_buffer_size ) ;
}
}
else
{
// UBO workaround
// this must be updated per shader switch, so also update it when it's not dirty
for ( unsigned int a = 0 ; a < 10 ; + + a )
{
if ( last_entry - > shader . UniformSize [ a ] > 0 )
2013-10-08 14:34:42 +02:00
glUniform4fv ( last_entry - > shader . UniformLocations [ a ] , last_entry - > shader . UniformSize [ a ] , ( float * ) & PixelShaderManager : : constants + 4 * PSVar_Loc [ a ] ) ;
2013-10-07 17:19:47 +02:00
}
for ( unsigned int a = 0 ; a < 9 ; + + a )
{
if ( last_entry - > shader . UniformSize [ a + 10 ] > 0 )
2013-10-08 14:34:42 +02:00
glUniform4fv ( last_entry - > shader . UniformLocations [ a + 10 ] , last_entry - > shader . UniformSize [ a + 10 ] , ( float * ) & VertexShaderManager : : constants + 4 * VSVar_Loc [ a ] ) ;
2013-10-07 17:19:47 +02:00
}
2013-10-29 01:23:17 -04:00
2013-05-23 21:07:01 +02:00
ADDSTAT ( stats . thisFrame . bytesUniformStreamed , s_ubo_buffer_size ) ;
2011-12-11 04:32:57 -06:00
}
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
}
GLuint ProgramShaderCache : : GetCurrentProgram ( void )
{
return CurrentProgram ;
}
SHADER * ProgramShaderCache : : SetShader ( DSTALPHA_MODE dstAlphaMode , u32 components )
{
SHADERUID uid ;
GetShaderId ( & uid , dstAlphaMode , components ) ;
2013-03-26 22:16:29 +01:00
2013-02-13 13:12:19 +01:00
// Check if the shader is already set
if ( last_entry )
{
if ( uid = = last_uid )
{
GFX_DEBUGGER_PAUSE_AT ( NEXT_PIXEL_SHADER_CHANGE , true ) ;
last_entry - > shader . Bind ( ) ;
return & last_entry - > shader ;
}
}
2013-03-29 20:33:28 +01:00
2013-02-13 13:12:19 +01:00
last_uid = uid ;
2013-03-29 20:33:28 +01:00
2013-02-13 13:12:19 +01:00
// Check if shader is already in cache
PCache : : iterator iter = pshaders . find ( uid ) ;
2011-12-26 00:15:54 -05:00
if ( iter ! = pshaders . end ( ) )
2011-12-11 04:32:57 -06:00
{
2013-02-13 13:12:19 +01:00
PCacheEntry * entry = & iter - > second ;
last_entry = entry ;
GFX_DEBUGGER_PAUSE_AT ( NEXT_PIXEL_SHADER_CHANGE , true ) ;
last_entry - > shader . Bind ( ) ;
return & last_entry - > shader ;
}
2013-03-29 20:33:28 +01:00
2013-02-13 13:12:19 +01:00
// Make an entry in the table
PCacheEntry & newentry = pshaders [ uid ] ;
last_entry = & newentry ;
2013-02-13 16:30:15 +01:00
newentry . in_cache = 0 ;
2013-03-29 20:33:28 +01:00
2013-03-26 22:16:29 +01:00
VertexShaderCode vcode ;
PixelShaderCode pcode ;
GenerateVertexShaderCode ( vcode , components , API_OPENGL ) ;
GeneratePixelShaderCode ( pcode , dstAlphaMode , API_OPENGL , components ) ;
2013-03-29 20:33:28 +01:00
2013-02-13 13:12:19 +01:00
if ( g_ActiveConfig . bEnableShaderDebugging )
{
2013-03-26 22:16:29 +01:00
newentry . shader . strvprog = vcode . GetBuffer ( ) ;
newentry . shader . strpprog = pcode . GetBuffer ( ) ;
2011-12-11 05:13:05 -06:00
}
2011-12-11 11:08:18 +01:00
2013-02-13 13:12:19 +01:00
# if defined(_DEBUG) || defined(DEBUGFAST)
if ( g_ActiveConfig . iLog & CONF_SAVESHADERS ) {
static int counter = 0 ;
char szTemp [ MAX_PATH ] ;
sprintf ( szTemp , " %svs_%04i.txt " , File : : GetUserPath ( D_DUMP_IDX ) . c_str ( ) , counter + + ) ;
2013-03-26 22:16:29 +01:00
SaveData ( szTemp , vcode . GetBuffer ( ) ) ;
2013-02-13 13:12:19 +01:00
sprintf ( szTemp , " %sps_%04i.txt " , File : : GetUserPath ( D_DUMP_IDX ) . c_str ( ) , counter + + ) ;
2013-03-26 22:16:29 +01:00
SaveData ( szTemp , pcode . GetBuffer ( ) ) ;
2013-02-13 13:12:19 +01:00
}
# endif
2011-12-26 00:15:54 -05:00
2013-03-26 22:16:29 +01:00
if ( ! CompileShader ( newentry . shader , vcode . GetBuffer ( ) , pcode . GetBuffer ( ) ) ) {
2013-02-13 13:12:19 +01:00
GFX_DEBUGGER_PAUSE_AT ( NEXT_ERROR , true ) ;
return NULL ;
}
INCSTAT ( stats . numPixelShadersCreated ) ;
SETSTAT ( stats . numPixelShadersAlive , pshaders . size ( ) ) ;
GFX_DEBUGGER_PAUSE_AT ( NEXT_PIXEL_SHADER_CHANGE , true ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
last_entry - > shader . Bind ( ) ;
return & last_entry - > shader ;
}
bool ProgramShaderCache : : CompileShader ( SHADER & shader , const char * vcode , const char * pcode )
{
GLuint vsid = CompileSingleShader ( GL_VERTEX_SHADER , vcode ) ;
GLuint psid = CompileSingleShader ( GL_FRAGMENT_SHADER , pcode ) ;
2013-03-29 20:33:28 +01:00
2013-02-13 13:12:19 +01:00
if ( ! vsid | | ! psid )
{
glDeleteShader ( vsid ) ;
glDeleteShader ( psid ) ;
return false ;
}
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
GLuint pid = shader . glprogid = glCreateProgram ( ) ; ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
glAttachShader ( pid , vsid ) ;
glAttachShader ( pid , psid ) ;
2011-12-26 00:15:54 -05:00
2013-03-25 15:14:24 +01:00
if ( g_ogl_config . bSupportsGLSLCache )
2013-02-13 13:12:19 +01:00
glProgramParameteri ( pid , GL_PROGRAM_BINARY_RETRIEVABLE_HINT , GL_TRUE ) ;
2011-12-26 00:15:54 -05:00
2013-02-13 13:12:19 +01:00
shader . SetProgramBindings ( ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
glLinkProgram ( pid ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
// original shaders aren't needed any more
glDeleteShader ( vsid ) ;
glDeleteShader ( psid ) ;
2013-10-29 01:23:17 -04:00
2013-03-07 20:26:56 +01:00
GLint linkStatus ;
glGetProgramiv ( pid , GL_LINK_STATUS , & linkStatus ) ;
2013-02-07 12:47:41 +01:00
GLsizei length = 0 ;
2013-02-13 13:12:19 +01:00
glGetProgramiv ( pid , GL_INFO_LOG_LENGTH , & length ) ;
2013-03-07 20:26:56 +01:00
if ( linkStatus ! = GL_TRUE | | ( length > 1 & & DEBUG_GLSL ) )
2013-02-07 12:47:41 +01:00
{
GLsizei charsWritten ;
GLchar * infoLog = new GLchar [ length ] ;
2013-02-13 13:12:19 +01:00
glGetProgramInfoLog ( pid , length , & charsWritten , infoLog ) ;
2013-02-07 12:47:41 +01:00
ERROR_LOG ( VIDEO , " Program info log: \n %s " , infoLog ) ;
char szTemp [ MAX_PATH ] ;
2013-05-04 23:30:13 +02:00
sprintf ( szTemp , " %sbad_p_%d.txt " , File : : GetUserPath ( D_DUMP_IDX ) . c_str ( ) , num_failures + + ) ;
2013-03-07 20:37:28 +01:00
std : : ofstream file ;
OpenFStream ( file , szTemp , std : : ios_base : : out ) ;
2013-08-16 21:05:35 +00:00
file < < s_glsl_header < < vcode < < s_glsl_header < < pcode < < infoLog ;
2013-03-07 20:37:28 +01:00
file . close ( ) ;
2013-10-29 01:23:17 -04:00
2013-09-18 11:47:44 +02:00
if ( linkStatus ! = GL_TRUE )
PanicAlert ( " Failed to link shaders! \n This usually happens when trying to use Dolphin with an outdated GPU or integrated GPU like the Intel GMA series. \n \n If you're sure this is Dolphin's error anyway, post the contents of %s along with this error message at the forums. \n \n Debug info (%s, %s, %s): \n %s " ,
szTemp ,
g_ogl_config . gl_vendor ,
g_ogl_config . gl_renderer ,
g_ogl_config . gl_version ,
infoLog ) ;
2013-10-29 01:23:17 -04:00
2013-03-07 20:37:28 +01:00
delete [ ] infoLog ;
2013-02-07 12:47:41 +01:00
}
if ( linkStatus ! = GL_TRUE )
{
// Compile failed
ERROR_LOG ( VIDEO , " Program linking failed; see info log " ) ;
// Don't try to use this shader
2013-02-13 13:12:19 +01:00
glDeleteProgram ( pid ) ;
return false ;
2013-02-07 12:47:41 +01:00
}
2011-12-26 00:15:54 -05:00
2013-02-13 13:12:19 +01:00
shader . SetProgramVariables ( ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
return true ;
2011-12-26 00:15:54 -05:00
}
2013-02-13 13:12:19 +01:00
GLuint ProgramShaderCache : : CompileSingleShader ( GLuint type , const char * code )
2011-12-26 00:15:54 -05:00
{
2013-02-13 13:12:19 +01:00
GLuint result = glCreateShader ( type ) ;
2013-02-13 21:34:48 +01:00
const char * src [ ] = { s_glsl_header , code } ;
2013-10-29 01:23:17 -04:00
2013-02-13 21:34:48 +01:00
glShaderSource ( result , 2 , src , NULL ) ;
2013-02-13 13:12:19 +01:00
glCompileShader ( result ) ;
2013-03-07 20:26:56 +01:00
GLint compileStatus ;
glGetShaderiv ( result , GL_COMPILE_STATUS , & compileStatus ) ;
2013-02-13 13:12:19 +01:00
GLsizei length = 0 ;
glGetShaderiv ( result , GL_INFO_LOG_LENGTH , & length ) ;
2013-06-18 10:24:36 -05:00
if ( DriverDetails : : HasBug ( DriverDetails : : BUG_BROKENINFOLOG ) )
length = 1024 ;
2013-03-07 20:26:56 +01:00
if ( compileStatus ! = GL_TRUE | | ( length > 1 & & DEBUG_GLSL ) )
2013-02-13 13:12:19 +01:00
{
GLsizei charsWritten ;
GLchar * infoLog = new GLchar [ length ] ;
glGetShaderInfoLog ( result , length , & charsWritten , infoLog ) ;
2013-12-18 22:21:14 -06:00
ERROR_LOG ( VIDEO , " %s Shader info log: \n %s " , type = = GL_VERTEX_SHADER ? " VS " : " PS " , infoLog ) ;
2013-02-13 13:12:19 +01:00
char szTemp [ MAX_PATH ] ;
2013-10-29 01:23:17 -04:00
sprintf ( szTemp ,
" %sbad_%s_%04i.txt " ,
File : : GetUserPath ( D_DUMP_IDX ) . c_str ( ) ,
type = = GL_VERTEX_SHADER ? " vs " : " ps " ,
2013-05-04 23:30:13 +02:00
num_failures + + ) ;
2013-03-07 20:37:28 +01:00
std : : ofstream file ;
OpenFStream ( file , szTemp , std : : ios_base : : out ) ;
2013-08-16 21:05:35 +00:00
file < < s_glsl_header < < code < < infoLog ;
2013-03-07 20:37:28 +01:00
file . close ( ) ;
2013-10-29 01:23:17 -04:00
2013-09-18 11:47:44 +02:00
if ( compileStatus ! = GL_TRUE )
PanicAlert ( " Failed to compile %s shader! \n This usually happens when trying to use Dolphin with an outdated GPU or integrated GPU like the Intel GMA series. \n \n If you're sure this is Dolphin's error anyway, post the contents of %s along with this error message at the forums. \n \n Debug info (%s, %s, %s): \n %s " ,
type = = GL_VERTEX_SHADER ? " vertex " : " pixel " ,
szTemp ,
g_ogl_config . gl_vendor ,
g_ogl_config . gl_renderer ,
g_ogl_config . gl_version ,
infoLog ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
delete [ ] infoLog ;
}
if ( compileStatus ! = GL_TRUE )
{
// Compile failed
ERROR_LOG ( VIDEO , " Shader compilation failed; see info log " ) ;
// Don't try to use this shader
glDeleteShader ( result ) ;
return 0 ;
}
( void ) GL_REPORT_ERROR ( ) ;
return result ;
2011-12-26 00:15:54 -05:00
}
2013-03-29 14:54:44 +01:00
void ProgramShaderCache : : GetShaderId ( SHADERUID * uid , DSTALPHA_MODE dstAlphaMode , u32 components )
2011-12-26 00:15:54 -05:00
{
2013-03-26 22:16:29 +01:00
GetPixelShaderUid ( uid - > puid , dstAlphaMode , API_OPENGL , components ) ;
GetVertexShaderUid ( uid - > vuid , components , API_OPENGL ) ;
2013-04-29 21:00:39 +02:00
if ( g_ActiveConfig . bEnableShaderDebugging )
{
PixelShaderCode pcode ;
GeneratePixelShaderCode ( pcode , dstAlphaMode , API_OPENGL , components ) ;
pixel_uid_checker . AddToIndexAndCheck ( pcode , uid - > puid , " Pixel " , " p " ) ;
VertexShaderCode vcode ;
GenerateVertexShaderCode ( vcode , components , API_OPENGL ) ;
vertex_uid_checker . AddToIndexAndCheck ( vcode , uid - > vuid , " Vertex " , " v " ) ;
}
2013-03-29 14:54:44 +01:00
}
2013-02-13 13:12:19 +01:00
2011-12-26 02:58:52 -05:00
ProgramShaderCache : : PCacheEntry ProgramShaderCache : : GetShaderProgram ( void )
2011-12-26 00:15:54 -05:00
{
2013-02-13 13:12:19 +01:00
return * last_entry ;
2011-12-26 00:15:54 -05:00
}
void ProgramShaderCache : : Init ( void )
{
// We have to get the UBO alignment here because
// if we generate a buffer that isn't aligned
// then the UBO will fail.
if ( g_ActiveConfig . backend_info . bSupportsGLSLUBO )
2011-12-11 11:08:18 +01:00
{
2013-10-07 17:19:47 +02:00
glGetIntegerv ( GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT , & s_ubo_align ) ;
2011-12-26 00:15:54 -05:00
2013-10-07 17:19:47 +02:00
s_ubo_buffer_size = ROUND_UP ( sizeof ( PixelShaderConstants ) , s_ubo_align ) + ROUND_UP ( sizeof ( VertexShaderConstants ) , s_ubo_align ) ;
2011-12-26 00:15:54 -05:00
// We multiply by *4*4 because we need to get down to basic machine units.
// So multiply by four to get how many floats we have from vec4s
// Then once more to get bytes
2013-01-31 23:11:53 +01:00
s_buffer = new StreamBuffer ( GL_UNIFORM_BUFFER , UBO_LENGTH ) ;
2011-12-11 11:08:18 +01:00
}
2011-12-26 00:15:54 -05:00
// Read our shader cache, only if supported
2013-12-09 16:45:20 +01:00
if ( g_ogl_config . bSupportsGLSLCache & & ! g_Config . bEnableShaderDebugging )
2011-12-11 04:19:11 -06:00
{
2013-02-13 16:30:15 +01:00
GLint Supported ;
glGetIntegerv ( GL_NUM_PROGRAM_BINARY_FORMATS , & Supported ) ;
if ( ! Supported )
{
ERROR_LOG ( VIDEO , " GL_ARB_get_program_binary is supported, but no binary format is known. So disable shader cache. " ) ;
2013-03-25 15:14:24 +01:00
g_ogl_config . bSupportsGLSLCache = false ;
2013-02-13 16:30:15 +01:00
}
else
{
2013-02-13 16:50:56 +01:00
if ( ! File : : Exists ( File : : GetUserPath ( D_SHADERCACHE_IDX ) ) )
File : : CreateDir ( File : : GetUserPath ( D_SHADERCACHE_IDX ) . c_str ( ) ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 16:30:15 +01:00
char cache_filename [ MAX_PATH ] ;
sprintf ( cache_filename , " %sogl-%s-shaders.cache " , File : : GetUserPath ( D_SHADERCACHE_IDX ) . c_str ( ) ,
SConfig : : GetInstance ( ) . m_LocalCoreStartupParameter . m_strUniqueID . c_str ( ) ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 16:30:15 +01:00
ProgramShaderCacheInserter inserter ;
g_program_disk_cache . OpenAndRead ( cache_filename , inserter ) ;
}
SETSTAT ( stats . numPixelShadersAlive , pshaders . size ( ) ) ;
2011-12-11 04:19:11 -06:00
}
2013-10-29 01:23:17 -04:00
2013-02-13 21:34:48 +01:00
CreateHeader ( ) ;
2013-10-29 01:23:17 -04:00
2013-01-15 14:22:40 +01:00
CurrentProgram = 0 ;
2013-02-13 13:12:19 +01:00
last_entry = NULL ;
2011-12-26 00:15:54 -05:00
}
void ProgramShaderCache : : Shutdown ( void )
{
2013-02-13 16:30:15 +01:00
// store all shaders in cache on disk
2013-12-09 16:45:20 +01:00
if ( g_ogl_config . bSupportsGLSLCache & & ! g_Config . bEnableShaderDebugging )
2011-12-11 11:08:18 +01:00
{
2011-12-11 11:14:02 +01:00
PCache : : iterator iter = pshaders . begin ( ) ;
2011-12-11 04:11:57 -06:00
for ( ; iter ! = pshaders . end ( ) ; + + iter )
2011-12-26 10:27:18 -05:00
{
2013-02-13 16:30:15 +01:00
if ( iter - > second . in_cache ) continue ;
2013-10-29 01:23:17 -04:00
2013-02-13 16:30:15 +01:00
GLint binary_size ;
glGetProgramiv ( iter - > second . shader . glprogid , GL_PROGRAM_BINARY_LENGTH , & binary_size ) ;
if ( ! binary_size ) continue ;
2013-10-29 01:23:17 -04:00
2013-02-13 16:30:15 +01:00
u8 * data = new u8 [ binary_size + sizeof ( GLenum ) ] ;
u8 * binary = data + sizeof ( GLenum ) ;
GLenum * prog_format = ( GLenum * ) data ;
glGetProgramBinary ( iter - > second . shader . glprogid , binary_size , NULL , prog_format , binary ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 16:30:15 +01:00
g_program_disk_cache . Append ( iter - > first , data , binary_size + sizeof ( GLenum ) ) ;
delete [ ] data ;
2011-12-26 10:27:18 -05:00
}
2011-12-11 04:11:57 -06:00
2011-12-26 00:15:54 -05:00
g_program_disk_cache . Sync ( ) ;
g_program_disk_cache . Close ( ) ;
}
2013-01-15 17:10:43 +01:00
glUseProgram ( 0 ) ;
2013-10-29 01:23:17 -04:00
2011-12-26 00:15:54 -05:00
PCache : : iterator iter = pshaders . begin ( ) ;
for ( ; iter ! = pshaders . end ( ) ; + + iter )
iter - > second . Destroy ( ) ;
pshaders . clear ( ) ;
2013-04-29 21:00:39 +02:00
pixel_uid_checker . Invalidate ( ) ;
vertex_uid_checker . Invalidate ( ) ;
2011-12-26 00:15:54 -05:00
if ( g_ActiveConfig . backend_info . bSupportsGLSLUBO )
{
2013-01-31 23:11:53 +01:00
delete s_buffer ;
s_buffer = 0 ;
2011-12-11 11:08:18 +01:00
}
2011-11-30 21:00:21 -06:00
}
2013-02-13 21:34:48 +01:00
void ProgramShaderCache : : CreateHeader ( void )
{
2013-04-08 14:50:58 +02:00
GLSL_VERSION v = g_ogl_config . eSupportedGLSLVersion ;
2013-10-29 01:23:17 -04:00
snprintf ( s_glsl_header , sizeof ( s_glsl_header ) ,
2013-10-06 03:12:13 -05:00
" %s \n "
2013-02-13 21:34:48 +01:00
" %s \n " // ubo
2013-08-21 11:48:39 +02:00
" %s \n " // early-z
2014-01-05 10:38:45 +01:00
" %s \n " // 420pack
2013-10-29 01:23:17 -04:00
2013-11-24 15:49:23 -06:00
// Precision defines for GLSLES3
2013-10-06 03:12:13 -05:00
" %s \n "
2013-02-13 21:34:48 +01:00
" \n " // A few required defines and ones that will make our lives a lot easier
2013-11-24 15:49:23 -06:00
" #define ATTRIN in \n "
" #define ATTROUT out \n "
2013-10-06 03:12:13 -05:00
" #define VARYIN %s \n "
" #define VARYOUT %s \n "
2013-02-13 21:34:48 +01:00
// Silly differences
" #define float2 vec2 \n "
" #define float3 vec3 \n "
" #define float4 vec4 \n "
2013-11-25 15:49:13 +01:00
" #define int2 ivec2 \n "
" #define int3 ivec3 \n "
" #define int4 ivec4 \n "
2013-02-13 21:34:48 +01:00
2013-06-18 07:52:36 -05:00
// hlsl to glsl function translation
2013-06-18 12:42:14 -05:00
" #define frac fract \n "
" #define lerp mix \n "
2013-06-18 07:52:36 -05:00
2013-12-19 17:30:39 -06:00
// Terrible hack, look at DriverDetails.h
" %s \n "
2013-11-24 15:49:23 -06:00
, v = = GLSLES3 ? " #version 300 es " : v = = GLSL_130 ? " #version 130 " : v = = GLSL_140 ? " #version 140 " : " #version 150 "
2013-05-05 23:22:21 -05:00
, g_ActiveConfig . backend_info . bSupportsGLSLUBO & & v < GLSL_140 ? " #extension GL_ARB_uniform_buffer_object : enable " : " "
2013-08-21 11:48:39 +02:00
, g_ActiveConfig . backend_info . bSupportsEarlyZ ? " #extension GL_ARB_shader_image_load_store : enable " : " "
2014-01-05 10:38:45 +01:00
, g_ActiveConfig . backend_info . bSupportShadingLanguage420pack ? " #extension GL_ARB_shading_language_420pack : enable " : " "
2013-10-29 01:23:17 -04:00
2013-11-24 15:49:23 -06:00
, v = = GLSLES3 ? " precision highp float; " : " "
2013-10-29 01:23:17 -04:00
2013-11-24 15:49:23 -06:00
, DriverDetails : : HasBug ( DriverDetails : : BUG_BROKENCENTROID ) ? " in " : " centroid in "
, DriverDetails : : HasBug ( DriverDetails : : BUG_BROKENCENTROID ) ? " out " : " centroid out "
2013-12-19 17:30:39 -06:00
, DriverDetails : : HasBug ( DriverDetails : : BUG_BROKENTEXTURESIZE ) ? " #define textureSize(x, y) ivec2(1, 1) " : " "
2013-02-13 21:34:48 +01:00
) ;
}
2013-02-13 13:12:19 +01:00
void ProgramShaderCache : : ProgramShaderCacheInserter : : Read ( const SHADERUID & key , const u8 * value , u32 value_size )
{
2013-02-13 16:30:15 +01:00
const u8 * binary = value + sizeof ( GLenum ) ;
GLenum * prog_format = ( GLenum * ) value ;
GLint binary_size = value_size - sizeof ( GLenum ) ;
2013-10-29 01:23:17 -04:00
2013-02-13 13:12:19 +01:00
PCacheEntry entry ;
2013-02-13 16:30:15 +01:00
entry . in_cache = 1 ;
2013-02-13 13:12:19 +01:00
entry . shader . glprogid = glCreateProgram ( ) ;
2013-02-13 16:30:15 +01:00
glProgramBinary ( entry . shader . glprogid , * prog_format , binary , binary_size ) ;
2013-02-13 13:12:19 +01:00
GLint success ;
glGetProgramiv ( entry . shader . glprogid , GL_LINK_STATUS , & success ) ;
if ( success )
{
pshaders [ key ] = entry ;
entry . shader . SetProgramVariables ( ) ;
}
2013-02-13 16:30:15 +01:00
else
glDeleteProgram ( entry . shader . glprogid ) ;
2013-02-13 13:12:19 +01:00
}
2011-12-26 00:15:54 -05:00
} // namespace OGL