This repository has been archived on 2024-07-18. You can view files and clone it, but cannot push or open issues or pull requests.
syntastic-glsl-drv/check_glsl.c
Emmanuel BENOîT cf1a1619fe Checker can use "normal" shaders now
Passing '/' as an argument before the shader type will use separable
shaders.
2015-03-11 08:56:20 +01:00

277 lines
6.1 KiB
C

#include <GL/glx.h>
#include <GL/glxext.h>
#include <GL/gl.h>
#include <GL/glext.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
static int vAttributes[] = {
None
};
static int cAttributes[] = {
GLX_CONTEXT_MAJOR_VERSION_ARB , 3 ,
GLX_CONTEXT_MINOR_VERSION_ARB , 0 ,
GLX_CONTEXT_FLAGS_ARB ,
GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB ,
GLX_CONTEXT_PROFILE_MASK_ARB ,
GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB ,
0
};
static int bAttributes[] = {
GLX_PBUFFER_WIDTH , 32 ,
GLX_PBUFFER_HEIGHT , 32 ,
None
};
#define gglp(t,n) \
(t)glXGetProcAddress( "gl" n )
static int errmsg( char const* msg )
{
fprintf( stderr , "%s\n" , msg );
return 0;
}
static Display * xDisplay;
static GLXContext glxContext;
static GLXPbuffer pBuffer;
static PFNGLCREATESHADERPROC glCreateShader;
static PFNGLCREATESHADERPROGRAMVPROC glCreateShaderProgramv;
static PFNGLSHADERSOURCEPROC glShaderSource;
static PFNGLCOMPILESHADERPROC glCompileShader;
static PFNGLGETPROGRAMINFOLOGPROC glGetProgramInfoLog;
static PFNGLGETSHADERINFOLOGPROC glGetShaderInfoLog;
static PFNGLGETSHADERIVPROC glGetShaderiv;
static PFNGLDELETEPROGRAMPROC glDeleteProgram;
static PFNGLDELETESHADERPROC glDeleteShader;
static int initContext( )
{
PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribs;
GLXFBConfig * fbc;
int fbcount;
xDisplay = XOpenDisplay( 0 );
if ( !xDisplay ) {
return errmsg( "Could not open display" );
}
fbc = glXChooseFBConfig( xDisplay , DefaultScreen( xDisplay ) ,
vAttributes , &fbcount );
if ( !fbc ) {
return errmsg( "No framebuffer config" );
}
glXCreateContextAttribs = gglp( PFNGLXCREATECONTEXTATTRIBSARBPROC ,
"XCreateContextAttribsARB" );
glxContext = glXCreateContextAttribs( xDisplay , fbc[ 0 ] , NULL ,
GL_TRUE , cAttributes );
if ( !glxContext ) {
return errmsg( "Could not create GL3.0+ context" );
}
pBuffer = glXCreatePbuffer( xDisplay , fbc[ 0 ] , bAttributes );
if ( !pBuffer ) {
return errmsg( "Could not create PBuffer" );
}
XFree( fbc );
XSync( xDisplay , False );
if ( ! glXMakeContextCurrent( xDisplay , pBuffer , pBuffer ,
glxContext ) ) {
return errmsg( "Could not activate context" );
}
glCreateShader = gglp( PFNGLCREATESHADERPROC , "CreateShader" );
glCreateShaderProgramv = gglp( PFNGLCREATESHADERPROGRAMVPROC ,
"CreateShaderProgramv" );
glShaderSource = gglp( PFNGLSHADERSOURCEPROC , "ShaderSource" );
glCompileShader = gglp( PFNGLCOMPILESHADERPROC , "CompileShader" );
glGetProgramInfoLog = gglp( PFNGLGETPROGRAMINFOLOGPROC ,
"GetProgramInfoLog" );
glGetShaderInfoLog = gglp( PFNGLGETSHADERINFOLOGPROC ,
"GetShaderInfoLog" );
glGetShaderiv = gglp( PFNGLGETSHADERIVPROC , "GetShaderiv" );
glDeleteProgram = gglp( PFNGLDELETEPROGRAMPROC , "DeleteProgram" );
glDeleteShader = gglp( PFNGLDELETESHADERPROC , "DeleteShader" );
return 1;
}
static void killContext( )
{
glXDestroyPbuffer( xDisplay , pBuffer );
glXMakeCurrent( xDisplay , 0 , 0 );
glXDestroyContext( xDisplay , glxContext );
XCloseDisplay( xDisplay );
}
static GLenum getShaderType( char const* argument )
{
if ( !strcmp( argument , "v" ) )
return GL_VERTEX_SHADER;
if ( !strcmp( argument , "f" ) )
return GL_FRAGMENT_SHADER;
if ( !strcmp( argument , "g" ) )
return GL_GEOMETRY_SHADER;
if ( !strcmp( argument , "tc" ) )
return GL_TESS_CONTROL_SHADER;
if ( !strcmp( argument , "te" ) )
return GL_TESS_EVALUATION_SHADER;
if ( !strcmp( argument , "c" ) )
return GL_COMPUTE_SHADER;
fprintf( stderr , "Invalid shader type '%s'\n" , argument );
return 0;
}
static char * loadFile( char const* name )
{
FILE * f = fopen( name , "r" );
if ( !f ) {
fprintf( stderr , "%s: file not found\n" , name );
return NULL;
}
int sz;
fseek( f , 0 , SEEK_END );
sz = ftell( f );
if ( sz <= 0 ) {
fprintf( stderr , "%s: fseek failed / empty file\n" , name );
return NULL;
}
fseek( f , 0 , SEEK_SET );
char * buffer = malloc( sz + 1 );
if ( !buffer ) {
fprintf( stderr , "%s: malloc() failed (%d bytes)\n" , name ,
sz );
return NULL;
}
if ( !fread( buffer , 1 , sz , f ) ) {
fprintf( stderr , "%s: could not read file contents\n" , name );
return NULL;
}
buffer[ sz ] = 0;
fclose(f);
return buffer;
}
static void loadShaderSeparable( GLenum type , char const* file )
{
GLuint program;
char * source = loadFile( file );
if ( source == NULL ) {
return;
}
program = glCreateShaderProgramv( type , 1 , (GLchar const**) &source );
free( source );
char infoLog[ 128 * 1024 ];
int sz , i , j;
glGetProgramInfoLog( program , sizeof( infoLog ) , &sz , infoLog );
glDeleteProgram( program );
i = 0;
while ( i < sz ) {
j = i;
while ( j < sz && infoLog[ j ] != '\n' ) {
j ++;
}
if ( j < sz ) {
infoLog[ j ] = 0;
}
fprintf( stderr , "%s: %s\n" , file , &infoLog[ i ] );
i = j + 1;
}
}
static void loadShader( GLenum type , char const* file )
{
GLenum shader;
shader = glCreateShader( type );
char * source = loadFile( file );
if ( source == NULL ) {
return;
}
glShaderSource( shader , 1 , (GLchar const**) &source , NULL );
free( source );
glCompileShader( shader );
GLint status;
glGetShaderiv( shader , GL_COMPILE_STATUS , &status );
char infoLog[ 128 * 1024 ];
int sz , i , j;
glGetShaderInfoLog( shader , sizeof( infoLog ) , &sz , infoLog );
glDeleteShader( shader );
i = 0;
while ( i < sz ) {
j = i;
while ( j < sz && infoLog[ j ] != '\n' ) {
j ++;
}
if ( j < sz ) {
infoLog[ j ] = 0;
}
fprintf( stderr , "%s: %s\n" , file , &infoLog[ i ] );
i = j + 1;
}
}
int main( int argc , char ** argv )
{
if ( !initContext( ) ) {
return 1;
}
if ( argc > 1 ) {
if ( argc < 3 ) {
fprintf( stderr ,
"Syntax: %s [/] v|f|g|tc|te|c file...\n" ,
argv[ 0 ] );
return 1;
}
int argPos = 1 , separable = 0;
if ( argc > 3 && !strcmp( "/" , argv[ 1 ] ) ) {
separable = 1;
argPos ++;
}
GLenum type = getShaderType( argv[ argPos ] );
if ( type == 0 ) {
return 1;
}
int i;
for ( i = argPos + 1 ; i < argc ; i ++ ) {
if ( separable ) {
loadShaderSeparable( type , argv[ i ] );
} else {
loadShader( type , argv[ i ] );
}
}
}
killContext( );
return 0;
}