2008-06-02 10:58:57 +00:00
|
|
|
/*
|
2009-04-27 14:48:12 +00:00
|
|
|
* Cogl
|
2008-06-02 10:58:57 +00:00
|
|
|
*
|
2009-04-27 14:48:12 +00:00
|
|
|
* An object oriented GL/GLES Abstraction/Utility Layer
|
2008-06-02 10:58:57 +00:00
|
|
|
*
|
2010-08-02 20:40:55 +00:00
|
|
|
* Copyright (C) 2007,2008,2009,2010 Intel Corporation.
|
2008-06-02 10:58:57 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2010-03-01 12:56:10 +00:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*
|
2008-06-02 10:58:57 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "cogl.h"
|
2008-10-30 17:25:00 +00:00
|
|
|
#include "cogl-shader-private.h"
|
2008-06-02 10:58:57 +00:00
|
|
|
#include "cogl-internal.h"
|
|
|
|
#include "cogl-context.h"
|
|
|
|
#include "cogl-handle.h"
|
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
#include <glib.h>
|
|
|
|
|
2010-08-04 16:53:51 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
#ifdef HAVE_COGL_GL
|
2010-10-15 17:00:29 +00:00
|
|
|
#define glCreateShader ctx->drv.pf_glCreateShader
|
|
|
|
#define glGetShaderiv ctx->drv.pf_glGetShaderiv
|
|
|
|
#define glGetShaderInfoLog ctx->drv.pf_glGetShaderInfoLog
|
|
|
|
#define glCompileShader ctx->drv.pf_glCompileShader
|
|
|
|
#define glShaderSource ctx->drv.pf_glShaderSource
|
|
|
|
#define glDeleteShader ctx->drv.pf_glDeleteShader
|
|
|
|
#define glProgramString ctx->drv.pf_glProgramString
|
|
|
|
#define glBindProgram ctx->drv.pf_glBindProgram
|
|
|
|
#define glDeletePrograms ctx->drv.pf_glDeletePrograms
|
|
|
|
#define glGenPrograms ctx->drv.pf_glGenPrograms
|
2010-08-02 20:40:55 +00:00
|
|
|
#define GET_CONTEXT _COGL_GET_CONTEXT
|
|
|
|
#else
|
|
|
|
#define GET_CONTEXT(CTXVAR,RETVAL) G_STMT_START { } G_STMT_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef HAVE_COGL_GLES
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
static void _cogl_shader_free (CoglShader *shader);
|
|
|
|
|
2009-04-01 16:16:44 +00:00
|
|
|
COGL_HANDLE_DEFINE (Shader, shader);
|
2010-07-09 16:59:16 +00:00
|
|
|
COGL_OBJECT_DEFINE_DEPRECATED_REF_COUNTING (shader);
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
_cogl_shader_free (CoglShader *shader)
|
|
|
|
{
|
|
|
|
/* Frees shader resources but its handle is not
|
|
|
|
released! Do that separately before this! */
|
|
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
2010-08-04 16:53:51 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_COGL_GL
|
|
|
|
if (shader->language == COGL_SHADER_LANGUAGE_ARBFP)
|
2010-10-15 17:00:29 +00:00
|
|
|
{
|
|
|
|
if (shader->gl_handle)
|
|
|
|
GE (glDeletePrograms (1, &shader->gl_handle));
|
|
|
|
}
|
2010-08-04 16:53:51 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
if (shader->gl_handle)
|
|
|
|
GE (glDeleteShader (shader->gl_handle));
|
2010-07-22 15:58:23 +00:00
|
|
|
|
|
|
|
g_slice_free (CoglShader, shader);
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CoglHandle
|
2009-05-12 13:15:18 +00:00
|
|
|
cogl_create_shader (CoglShaderType type)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
|
|
|
CoglShader *shader;
|
2009-05-12 13:15:18 +00:00
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
GET_CONTEXT (ctx, COGL_INVALID_HANDLE);
|
|
|
|
|
2010-08-04 16:53:51 +00:00
|
|
|
switch (type)
|
2009-05-12 13:15:18 +00:00
|
|
|
{
|
2010-08-04 16:53:51 +00:00
|
|
|
case COGL_SHADER_TYPE_VERTEX:
|
|
|
|
case COGL_SHADER_TYPE_FRAGMENT:
|
|
|
|
break;
|
|
|
|
default:
|
2009-05-12 13:15:18 +00:00
|
|
|
g_warning ("Unexpected shader type (0x%08lX) given to "
|
|
|
|
"cogl_create_shader", (unsigned long) type);
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
}
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
shader = g_slice_new (CoglShader);
|
2010-08-04 16:53:51 +00:00
|
|
|
shader->language = COGL_SHADER_LANGUAGE_GLSL;
|
|
|
|
shader->gl_handle = 0;
|
2010-07-14 18:39:24 +00:00
|
|
|
shader->type = type;
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
return _cogl_shader_handle_new (shader);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cogl_shader_source (CoglHandle handle,
|
2009-05-12 13:15:18 +00:00
|
|
|
const char *source)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
|
|
|
CoglShader *shader;
|
2010-08-04 16:53:51 +00:00
|
|
|
CoglShaderLanguage language;
|
|
|
|
|
2008-06-02 10:58:57 +00:00
|
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
2009-04-27 14:48:12 +00:00
|
|
|
|
2008-06-02 10:58:57 +00:00
|
|
|
if (!cogl_is_shader (handle))
|
|
|
|
return;
|
|
|
|
|
|
|
|
shader = _cogl_shader_pointer_from_handle (handle);
|
|
|
|
|
2010-08-04 16:53:51 +00:00
|
|
|
#ifdef HAVE_COGL_GL
|
|
|
|
if (strncmp (source, "!!ARBfp1.0", 10) == 0)
|
|
|
|
language = COGL_SHADER_LANGUAGE_ARBFP;
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
language = COGL_SHADER_LANGUAGE_GLSL;
|
|
|
|
|
|
|
|
/* Delete the old object if the language is changing... */
|
|
|
|
if (G_UNLIKELY (language != shader->language))
|
|
|
|
{
|
|
|
|
#ifdef HAVE_COGL_GL
|
|
|
|
if (shader->language == COGL_SHADER_LANGUAGE_ARBFP)
|
|
|
|
{
|
2010-10-15 17:00:29 +00:00
|
|
|
if (shader->gl_handle)
|
|
|
|
GE (glDeletePrograms (1, &shader->gl_handle));
|
2010-08-04 16:53:51 +00:00
|
|
|
}
|
|
|
|
else
|
2010-08-12 09:40:39 +00:00
|
|
|
#endif
|
2010-08-04 16:53:51 +00:00
|
|
|
{
|
|
|
|
if (shader->gl_handle)
|
|
|
|
GE (glDeleteShader (shader->gl_handle));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef HAVE_COGL_GL
|
|
|
|
if (language == COGL_SHADER_LANGUAGE_ARBFP)
|
2010-10-15 17:00:29 +00:00
|
|
|
{
|
|
|
|
#ifdef COGL_GL_DEBUG
|
|
|
|
GLenum gl_error;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
GE (glGenPrograms (1, &shader->gl_handle));
|
|
|
|
|
|
|
|
GE (glBindProgram (GL_FRAGMENT_PROGRAM_ARB, shader->gl_handle));
|
|
|
|
|
|
|
|
#ifdef COGL_GL_DEBUG
|
|
|
|
while ((gl_error = glGetError ()) != GL_NO_ERROR)
|
|
|
|
;
|
|
|
|
#endif
|
|
|
|
glProgramString (GL_FRAGMENT_PROGRAM_ARB,
|
|
|
|
GL_PROGRAM_FORMAT_ASCII_ARB,
|
|
|
|
strlen (source),
|
|
|
|
source);
|
|
|
|
#ifdef COGL_GL_DEBUG
|
|
|
|
gl_error = glGetError ();
|
|
|
|
if (gl_error != GL_NO_ERROR)
|
|
|
|
{
|
|
|
|
g_warning ("%s: GL error (%d): Failed to compile ARBfp:\n%s\n%s",
|
|
|
|
G_STRLOC,
|
|
|
|
gl_error,
|
|
|
|
source,
|
|
|
|
glGetString (GL_PROGRAM_ERROR_STRING_ARB));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
2010-08-04 16:53:51 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
if (!shader->gl_handle)
|
|
|
|
{
|
|
|
|
GLenum gl_type;
|
|
|
|
|
|
|
|
switch (shader->type)
|
|
|
|
{
|
|
|
|
case COGL_SHADER_TYPE_VERTEX:
|
|
|
|
gl_type = GL_VERTEX_SHADER;
|
|
|
|
break;
|
|
|
|
case COGL_SHADER_TYPE_FRAGMENT:
|
|
|
|
gl_type = GL_FRAGMENT_SHADER;
|
|
|
|
break;
|
2010-08-09 18:38:23 +00:00
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
break;
|
2010-08-04 16:53:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
shader->gl_handle = glCreateShader (gl_type);
|
|
|
|
}
|
|
|
|
glShaderSource (shader->gl_handle, 1, &source, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
shader->language = language;
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cogl_shader_compile (CoglHandle handle)
|
|
|
|
{
|
|
|
|
CoglShader *shader;
|
|
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
|
|
|
|
|
|
|
if (!cogl_is_shader (handle))
|
|
|
|
return;
|
|
|
|
|
|
|
|
shader = _cogl_shader_pointer_from_handle (handle);
|
|
|
|
|
2010-08-04 16:53:51 +00:00
|
|
|
if (shader->language == COGL_SHADER_LANGUAGE_GLSL)
|
|
|
|
GE (glCompileShader (shader->gl_handle));
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
cogl: improves header and coding style consistency
We've had complaints that our Cogl code/headers are a bit "special" so
this is a first pass at tidying things up by giving them some
consistency. These changes are all consistent with how new code in Cogl
is being written, but the style isn't consistently applied across all
code yet.
There are two parts to this patch; but since each one required a large
amount of effort to maintain tidy indenting it made sense to combine the
changes to reduce the time spent re indenting the same lines.
The first change is to use a consistent style for declaring function
prototypes in headers. Cogl headers now consistently use this style for
prototypes:
return_type
cogl_function_name (CoglType arg0,
CoglType arg1);
Not everyone likes this style, but it seems that most of the currently
active Cogl developers agree on it.
The second change is to constrain the use of redundant glib data types
in Cogl. Uses of gint, guint, gfloat, glong, gulong and gchar have all
been replaced with int, unsigned int, float, long, unsigned long and char
respectively. When talking about pixel data; use of guchar has been
replaced with guint8, otherwise unsigned char can be used.
The glib types that we continue to use for portability are gboolean,
gint{8,16,32,64}, guint{8,16,32,64} and gsize.
The general intention is that Cogl should look palatable to the widest
range of C programmers including those outside the Gnome community so
- especially for the public API - we want to minimize the number of
foreign looking typedefs.
2010-02-10 01:57:32 +00:00
|
|
|
char *
|
2009-06-01 15:31:32 +00:00
|
|
|
cogl_shader_get_info_log (CoglHandle handle)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
|
|
|
CoglShader *shader;
|
2010-08-02 20:40:55 +00:00
|
|
|
|
|
|
|
GET_CONTEXT (ctx, NULL);
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
if (!cogl_is_shader (handle))
|
2009-06-01 15:31:32 +00:00
|
|
|
return NULL;
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
shader = _cogl_shader_pointer_from_handle (handle);
|
|
|
|
|
2010-08-12 09:43:22 +00:00
|
|
|
#ifdef HAVE_COGL_GL
|
2010-08-04 16:53:51 +00:00
|
|
|
if (shader->language == COGL_SHADER_LANGUAGE_ARBFP)
|
|
|
|
{
|
|
|
|
/* ARBfp exposes a program error string, but since cogl_program
|
|
|
|
* doesn't have any API to query an error log it is not currently
|
|
|
|
* exposed. */
|
2010-08-12 09:43:22 +00:00
|
|
|
return g_strdup ("");
|
2010-08-04 16:53:51 +00:00
|
|
|
}
|
|
|
|
else
|
2010-08-12 09:43:22 +00:00
|
|
|
#endif
|
2010-08-04 16:53:51 +00:00
|
|
|
{
|
|
|
|
char buffer[512];
|
|
|
|
int len = 0;
|
|
|
|
glGetShaderInfoLog (shader->gl_handle, 511, &len, buffer);
|
|
|
|
buffer[len] = '\0';
|
|
|
|
return g_strdup (buffer);
|
|
|
|
}
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
2009-05-12 13:15:18 +00:00
|
|
|
CoglShaderType
|
|
|
|
cogl_shader_get_type (CoglHandle handle)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
|
|
|
CoglShader *shader;
|
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
GET_CONTEXT (ctx, COGL_SHADER_TYPE_VERTEX);
|
|
|
|
|
2008-06-02 10:58:57 +00:00
|
|
|
if (!cogl_is_shader (handle))
|
2009-05-12 13:15:18 +00:00
|
|
|
{
|
|
|
|
g_warning ("Non shader handle type passed to cogl_shader_get_type");
|
|
|
|
return COGL_SHADER_TYPE_VERTEX;
|
|
|
|
}
|
|
|
|
|
|
|
|
shader = _cogl_shader_pointer_from_handle (handle);
|
2010-07-23 16:44:53 +00:00
|
|
|
return shader->type;
|
2009-05-12 13:15:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
cogl_shader_is_compiled (CoglHandle handle)
|
|
|
|
{
|
|
|
|
GLint status;
|
|
|
|
CoglShader *shader;
|
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
GET_CONTEXT (ctx, FALSE);
|
|
|
|
|
2009-05-12 13:15:18 +00:00
|
|
|
if (!cogl_is_shader (handle))
|
|
|
|
return FALSE;
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
shader = _cogl_shader_pointer_from_handle (handle);
|
|
|
|
|
2010-08-04 16:53:51 +00:00
|
|
|
#ifdef HAVE_COGL_GL
|
|
|
|
if (shader->language == COGL_SHADER_LANGUAGE_ARBFP)
|
2009-05-12 13:15:18 +00:00
|
|
|
return TRUE;
|
|
|
|
else
|
2010-08-04 16:53:51 +00:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
GE (glGetShaderiv (shader->gl_handle, GL_COMPILE_STATUS, &status));
|
|
|
|
if (status == GL_TRUE)
|
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
#else /* HAVE_COGL_GLES */
|
2008-06-02 10:58:57 +00:00
|
|
|
|
|
|
|
/* No support on regular OpenGL 1.1 */
|
|
|
|
|
|
|
|
CoglHandle
|
2009-05-12 13:15:18 +00:00
|
|
|
cogl_create_shader (CoglShaderType type)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
cogl_is_shader (CoglHandle handle)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
CoglHandle
|
|
|
|
cogl_shader_ref (CoglHandle handle)
|
|
|
|
{
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cogl_shader_unref (CoglHandle handle)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cogl_shader_source (CoglHandle shader,
|
2009-05-12 13:15:18 +00:00
|
|
|
const char *source)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cogl_shader_compile (CoglHandle shader_handle)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
cogl: improves header and coding style consistency
We've had complaints that our Cogl code/headers are a bit "special" so
this is a first pass at tidying things up by giving them some
consistency. These changes are all consistent with how new code in Cogl
is being written, but the style isn't consistently applied across all
code yet.
There are two parts to this patch; but since each one required a large
amount of effort to maintain tidy indenting it made sense to combine the
changes to reduce the time spent re indenting the same lines.
The first change is to use a consistent style for declaring function
prototypes in headers. Cogl headers now consistently use this style for
prototypes:
return_type
cogl_function_name (CoglType arg0,
CoglType arg1);
Not everyone likes this style, but it seems that most of the currently
active Cogl developers agree on it.
The second change is to constrain the use of redundant glib data types
in Cogl. Uses of gint, guint, gfloat, glong, gulong and gchar have all
been replaced with int, unsigned int, float, long, unsigned long and char
respectively. When talking about pixel data; use of guchar has been
replaced with guint8, otherwise unsigned char can be used.
The glib types that we continue to use for portability are gboolean,
gint{8,16,32,64}, guint{8,16,32,64} and gsize.
The general intention is that Cogl should look palatable to the widest
range of C programmers including those outside the Gnome community so
- especially for the public API - we want to minimize the number of
foreign looking typedefs.
2010-02-10 01:57:32 +00:00
|
|
|
char *
|
2009-06-04 11:04:47 +00:00
|
|
|
cogl_shader_get_info_log (CoglHandle handle)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
2009-06-04 11:04:47 +00:00
|
|
|
return NULL;
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
2009-05-12 13:15:18 +00:00
|
|
|
CoglShaderType
|
|
|
|
cogl_shader_get_type (CoglHandle handle)
|
2008-06-02 10:58:57 +00:00
|
|
|
{
|
2009-05-12 13:15:18 +00:00
|
|
|
return COGL_SHADER_TYPE_VERTEX;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
cogl_shader_is_compiled (CoglHandle handle)
|
|
|
|
{
|
|
|
|
return FALSE;
|
2008-06-02 10:58:57 +00:00
|
|
|
}
|
|
|
|
|
2010-08-02 20:40:55 +00:00
|
|
|
#endif /* HAVE_COGL_GLES */
|
|
|
|
|