mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-15 06:40:08 +00:00
d85ec2cbb4
If desired, try creating GL_SRGB images. Pass a try_srgb boolean down to the image creation functions and have them attempt to create images like that. When it is not possible to create srgb images in the given format, just fall back to regular images. The calling code is meant to check the GSK_GPU_IMAGE_SRGB flags to determine the actual format of the resulting image.
711 lines
21 KiB
C
711 lines
21 KiB
C
#include "config.h"
|
|
|
|
#include "gskgldeviceprivate.h"
|
|
|
|
#include "gskdebugprivate.h"
|
|
#include "gskgpushaderopprivate.h"
|
|
#include "gskglbufferprivate.h"
|
|
#include "gskglimageprivate.h"
|
|
|
|
#include "gdk/gdkdisplayprivate.h"
|
|
#include "gdk/gdkglcontextprivate.h"
|
|
#include "gdk/gdkprofilerprivate.h"
|
|
|
|
#include <glib/gi18n-lib.h>
|
|
|
|
struct _GskGLDevice
|
|
{
|
|
GskGpuDevice parent_instance;
|
|
|
|
GHashTable *gl_programs;
|
|
const char *version_string;
|
|
GdkGLAPI api;
|
|
|
|
guint sampler_ids[GSK_GPU_SAMPLER_N_SAMPLERS];
|
|
};
|
|
|
|
struct _GskGLDeviceClass
|
|
{
|
|
GskGpuDeviceClass parent_class;
|
|
};
|
|
|
|
typedef struct _GLProgramKey GLProgramKey;
|
|
|
|
struct _GLProgramKey
|
|
{
|
|
const GskGpuShaderOpClass *op_class;
|
|
guint32 variation;
|
|
GskGpuShaderClip clip;
|
|
guint n_external_textures;
|
|
};
|
|
|
|
G_DEFINE_TYPE (GskGLDevice, gsk_gl_device, GSK_TYPE_GPU_DEVICE)
|
|
|
|
static guint
|
|
gl_program_key_hash (gconstpointer data)
|
|
{
|
|
const GLProgramKey *key = data;
|
|
|
|
return GPOINTER_TO_UINT (key->op_class) ^
|
|
key->clip ^
|
|
(key->variation << 2) ^
|
|
(key->n_external_textures << 24);
|
|
}
|
|
|
|
static gboolean
|
|
gl_program_key_equal (gconstpointer a,
|
|
gconstpointer b)
|
|
{
|
|
const GLProgramKey *keya = a;
|
|
const GLProgramKey *keyb = b;
|
|
|
|
return keya->op_class == keyb->op_class &&
|
|
keya->variation == keyb->variation &&
|
|
keya->clip == keyb->clip &&
|
|
keya->n_external_textures == keyb->n_external_textures;
|
|
}
|
|
|
|
static GskGpuImage *
|
|
gsk_gl_device_create_offscreen_image (GskGpuDevice *device,
|
|
gboolean with_mipmap,
|
|
GdkMemoryDepth depth,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
GskGLDevice *self = GSK_GL_DEVICE (device);
|
|
|
|
return gsk_gl_image_new (self,
|
|
gdk_memory_depth_get_format (depth),
|
|
gdk_memory_depth_is_srgb (depth),
|
|
GSK_GPU_IMAGE_RENDERABLE | GSK_GPU_IMAGE_FILTERABLE,
|
|
width,
|
|
height);
|
|
}
|
|
|
|
static GskGpuImage *
|
|
gsk_gl_device_create_upload_image (GskGpuDevice *device,
|
|
gboolean with_mipmap,
|
|
GdkMemoryFormat format,
|
|
gboolean try_srgb,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
GskGLDevice *self = GSK_GL_DEVICE (device);
|
|
|
|
return gsk_gl_image_new (self,
|
|
format,
|
|
try_srgb,
|
|
0,
|
|
width,
|
|
height);
|
|
}
|
|
|
|
static GskGpuImage *
|
|
gsk_gl_device_create_download_image (GskGpuDevice *device,
|
|
GdkMemoryDepth depth,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
GskGLDevice *self = GSK_GL_DEVICE (device);
|
|
|
|
return gsk_gl_image_new (self,
|
|
gdk_memory_depth_get_format (depth),
|
|
gdk_memory_depth_is_srgb (depth),
|
|
GSK_GPU_IMAGE_RENDERABLE,
|
|
width,
|
|
height);
|
|
}
|
|
|
|
static GskGpuImage *
|
|
gsk_gl_device_create_atlas_image (GskGpuDevice *device,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
GskGLDevice *self = GSK_GL_DEVICE (device);
|
|
|
|
return gsk_gl_image_new (self,
|
|
GDK_MEMORY_DEFAULT,
|
|
FALSE,
|
|
GSK_GPU_IMAGE_RENDERABLE,
|
|
width,
|
|
height);
|
|
}
|
|
|
|
static void
|
|
gsk_gl_device_make_current (GskGpuDevice *device)
|
|
{
|
|
gdk_gl_context_make_current (gdk_display_get_gl_context (gsk_gpu_device_get_display (device)));
|
|
}
|
|
|
|
static void
|
|
gsk_gl_device_finalize (GObject *object)
|
|
{
|
|
GskGLDevice *self = GSK_GL_DEVICE (object);
|
|
GskGpuDevice *device = GSK_GPU_DEVICE (self);
|
|
|
|
g_object_steal_data (G_OBJECT (gsk_gpu_device_get_display (device)), "-gsk-gl-device");
|
|
|
|
gdk_gl_context_make_current (gdk_display_get_gl_context (gsk_gpu_device_get_display (device)));
|
|
|
|
g_hash_table_unref (self->gl_programs);
|
|
glDeleteSamplers (G_N_ELEMENTS (self->sampler_ids), self->sampler_ids);
|
|
|
|
G_OBJECT_CLASS (gsk_gl_device_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gsk_gl_device_class_init (GskGLDeviceClass *klass)
|
|
{
|
|
GskGpuDeviceClass *gpu_device_class = GSK_GPU_DEVICE_CLASS (klass);
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
gpu_device_class->create_offscreen_image = gsk_gl_device_create_offscreen_image;
|
|
gpu_device_class->create_atlas_image = gsk_gl_device_create_atlas_image;
|
|
gpu_device_class->create_upload_image = gsk_gl_device_create_upload_image;
|
|
gpu_device_class->create_download_image = gsk_gl_device_create_download_image;
|
|
gpu_device_class->make_current = gsk_gl_device_make_current;
|
|
|
|
object_class->finalize = gsk_gl_device_finalize;
|
|
}
|
|
|
|
static void
|
|
free_gl_program (gpointer program)
|
|
{
|
|
glDeleteProgram (GPOINTER_TO_UINT (program));
|
|
}
|
|
|
|
static void
|
|
gsk_gl_device_init (GskGLDevice *self)
|
|
{
|
|
self->gl_programs = g_hash_table_new_full (gl_program_key_hash, gl_program_key_equal, g_free, free_gl_program);
|
|
}
|
|
|
|
static void
|
|
gsk_gl_device_setup_samplers (GskGLDevice *self)
|
|
{
|
|
struct {
|
|
GLuint min_filter;
|
|
GLuint mag_filter;
|
|
GLuint wrap;
|
|
} sampler_flags[GSK_GPU_SAMPLER_N_SAMPLERS] = {
|
|
[GSK_GPU_SAMPLER_DEFAULT] = {
|
|
.min_filter = GL_LINEAR,
|
|
.mag_filter = GL_LINEAR,
|
|
.wrap = GL_CLAMP_TO_EDGE,
|
|
},
|
|
[GSK_GPU_SAMPLER_TRANSPARENT] = {
|
|
.min_filter = GL_LINEAR,
|
|
.mag_filter = GL_LINEAR,
|
|
.wrap = GL_CLAMP_TO_BORDER,
|
|
},
|
|
[GSK_GPU_SAMPLER_REPEAT] = {
|
|
.min_filter = GL_LINEAR,
|
|
.mag_filter = GL_LINEAR,
|
|
.wrap = GL_REPEAT,
|
|
},
|
|
[GSK_GPU_SAMPLER_NEAREST] = {
|
|
.min_filter = GL_NEAREST,
|
|
.mag_filter = GL_NEAREST,
|
|
.wrap = GL_CLAMP_TO_EDGE,
|
|
},
|
|
[GSK_GPU_SAMPLER_MIPMAP_DEFAULT] = {
|
|
.min_filter = GL_LINEAR_MIPMAP_LINEAR,
|
|
.mag_filter = GL_LINEAR,
|
|
.wrap = GL_CLAMP_TO_EDGE,
|
|
}
|
|
};
|
|
guint i;
|
|
|
|
glGenSamplers (G_N_ELEMENTS (self->sampler_ids), self->sampler_ids);
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (self->sampler_ids); i++)
|
|
{
|
|
glSamplerParameteri (self->sampler_ids[i], GL_TEXTURE_MIN_FILTER, sampler_flags[i].min_filter);
|
|
glSamplerParameteri (self->sampler_ids[i], GL_TEXTURE_MAG_FILTER, sampler_flags[i].mag_filter);
|
|
glSamplerParameteri (self->sampler_ids[i], GL_TEXTURE_WRAP_S, sampler_flags[i].wrap);
|
|
glSamplerParameteri (self->sampler_ids[i], GL_TEXTURE_WRAP_T, sampler_flags[i].wrap);
|
|
}
|
|
}
|
|
|
|
GskGpuDevice *
|
|
gsk_gl_device_get_for_display (GdkDisplay *display,
|
|
GError **error)
|
|
{
|
|
GskGLDevice *self;
|
|
GdkGLContext *context;
|
|
GLint max_texture_size;
|
|
|
|
self = g_object_get_data (G_OBJECT (display), "-gsk-gl-device");
|
|
if (self)
|
|
return GSK_GPU_DEVICE (g_object_ref (self));
|
|
|
|
if (!gdk_display_prepare_gl (display, error))
|
|
return NULL;
|
|
|
|
context = gdk_display_get_gl_context (display);
|
|
|
|
/* GLES 2 is not supported */
|
|
if (!gdk_gl_context_check_version (context, "3.0", "3.0"))
|
|
{
|
|
g_set_error (error, GDK_GL_ERROR, GDK_GL_ERROR_NOT_AVAILABLE,
|
|
_("OpenGL ES 3.0 is not supported by this renderer."));
|
|
return NULL;
|
|
}
|
|
|
|
self = g_object_new (GSK_TYPE_GL_DEVICE, NULL);
|
|
|
|
gdk_gl_context_make_current (context);
|
|
|
|
glGetIntegerv (GL_MAX_TEXTURE_SIZE, &max_texture_size);
|
|
gsk_gpu_device_setup (GSK_GPU_DEVICE (self), display, max_texture_size);
|
|
|
|
self->version_string = gdk_gl_context_get_glsl_version_string (context);
|
|
self->api = gdk_gl_context_get_api (context);
|
|
gsk_gl_device_setup_samplers (self);
|
|
|
|
g_object_set_data (G_OBJECT (display), "-gsk-gl-device", self);
|
|
|
|
return GSK_GPU_DEVICE (self);
|
|
}
|
|
|
|
static char *
|
|
prepend_line_numbers (char *code)
|
|
{
|
|
GString *s;
|
|
char *p;
|
|
int line;
|
|
|
|
s = g_string_new ("");
|
|
p = code;
|
|
line = 1;
|
|
while (*p)
|
|
{
|
|
char *end = strchr (p, '\n');
|
|
if (end)
|
|
end = end + 1; /* Include newline */
|
|
else
|
|
end = p + strlen (p);
|
|
|
|
g_string_append_printf (s, "%3d| ", line++);
|
|
g_string_append_len (s, p, end - p);
|
|
|
|
p = end;
|
|
}
|
|
|
|
g_free (code);
|
|
|
|
return g_string_free (s, FALSE);
|
|
}
|
|
|
|
static gboolean
|
|
gsk_gl_device_check_shader_error (int shader_id,
|
|
GError **error)
|
|
{
|
|
GLint status;
|
|
GLint log_len;
|
|
GLint code_len;
|
|
char *log;
|
|
char *code;
|
|
|
|
glGetShaderiv (shader_id, GL_COMPILE_STATUS, &status);
|
|
|
|
if G_LIKELY (status == GL_TRUE)
|
|
return TRUE;
|
|
|
|
glGetShaderiv (shader_id, GL_INFO_LOG_LENGTH, &log_len);
|
|
log = g_malloc0 (log_len + 1);
|
|
glGetShaderInfoLog (shader_id, log_len, NULL, log);
|
|
|
|
glGetShaderiv (shader_id, GL_SHADER_SOURCE_LENGTH, &code_len);
|
|
code = g_malloc0 (code_len + 1);
|
|
glGetShaderSource (shader_id, code_len, NULL, code);
|
|
|
|
code = prepend_line_numbers (code);
|
|
|
|
g_set_error (error,
|
|
GDK_GL_ERROR,
|
|
GDK_GL_ERROR_COMPILATION_FAILED,
|
|
"Compilation failure in shader.\n"
|
|
"Source Code:\n"
|
|
"%s\n"
|
|
"\n"
|
|
"Error Message:\n"
|
|
"%s\n"
|
|
"\n",
|
|
code,
|
|
log);
|
|
|
|
g_free (code);
|
|
g_free (log);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
print_shader_info (const char *prefix,
|
|
GLuint shader_id,
|
|
const char *name)
|
|
{
|
|
if (GSK_DEBUG_CHECK (SHADERS))
|
|
{
|
|
int code_len;
|
|
|
|
glGetShaderiv (shader_id, GL_SHADER_SOURCE_LENGTH, &code_len);
|
|
|
|
if (code_len > 0)
|
|
{
|
|
char *code;
|
|
|
|
code = g_malloc0 (code_len + 1);
|
|
glGetShaderSource (shader_id, code_len, NULL, code);
|
|
|
|
code = prepend_line_numbers (code);
|
|
|
|
g_message ("%s %d, %s:\n%s",
|
|
prefix, shader_id,
|
|
name ? name : "unnamed",
|
|
code);
|
|
g_free (code);
|
|
}
|
|
}
|
|
}
|
|
|
|
static GLuint
|
|
gsk_gl_device_load_shader (GskGLDevice *self,
|
|
const char *program_name,
|
|
GLenum shader_type,
|
|
guint32 variation,
|
|
GskGpuShaderClip clip,
|
|
guint n_external_textures,
|
|
GError **error)
|
|
{
|
|
GString *preamble;
|
|
char *resource_name;
|
|
GBytes *bytes;
|
|
GLuint shader_id;
|
|
|
|
preamble = g_string_new (NULL);
|
|
|
|
g_string_append (preamble, self->version_string);
|
|
g_string_append (preamble, "\n");
|
|
if (self->api == GDK_GL_API_GLES)
|
|
{
|
|
if (n_external_textures > 0)
|
|
{
|
|
g_string_append (preamble, "#extension GL_OES_EGL_image_external_essl3 : require\n");
|
|
g_string_append (preamble, "#extension GL_OES_EGL_image_external : require\n");
|
|
}
|
|
g_string_append (preamble, "#define GSK_GLES 1\n");
|
|
g_assert (3 * n_external_textures <= 16);
|
|
}
|
|
else
|
|
{
|
|
g_assert (n_external_textures == 0);
|
|
}
|
|
|
|
g_string_append_printf (preamble, "#define N_TEXTURES %u\n", 16 - 3 * n_external_textures);
|
|
g_string_append_printf (preamble, "#define N_EXTERNAL_TEXTURES %u\n", n_external_textures);
|
|
|
|
switch (shader_type)
|
|
{
|
|
case GL_VERTEX_SHADER:
|
|
g_string_append (preamble, "#define GSK_VERTEX_SHADER 1\n");
|
|
break;
|
|
|
|
case GL_FRAGMENT_SHADER:
|
|
g_string_append (preamble, "#define GSK_FRAGMENT_SHADER 1\n");
|
|
break;
|
|
|
|
default:
|
|
g_assert_not_reached ();
|
|
return 0;
|
|
}
|
|
|
|
g_string_append_printf (preamble, "#define GSK_VARIATION %uu\n", variation);
|
|
|
|
switch (clip)
|
|
{
|
|
case GSK_GPU_SHADER_CLIP_NONE:
|
|
g_string_append (preamble, "#define GSK_SHADER_CLIP GSK_GPU_SHADER_CLIP_NONE\n");
|
|
break;
|
|
case GSK_GPU_SHADER_CLIP_RECT:
|
|
g_string_append (preamble, "#define GSK_SHADER_CLIP GSK_GPU_SHADER_CLIP_RECT\n");
|
|
break;
|
|
case GSK_GPU_SHADER_CLIP_ROUNDED:
|
|
g_string_append (preamble, "#define GSK_SHADER_CLIP GSK_GPU_SHADER_CLIP_ROUNDED\n");
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
break;
|
|
}
|
|
|
|
resource_name = g_strconcat ("/org/gtk/libgsk/shaders/gl/", program_name, ".glsl", NULL);
|
|
bytes = g_resources_lookup_data (resource_name, 0, error);
|
|
g_free (resource_name);
|
|
if (bytes == NULL)
|
|
return 0;
|
|
|
|
shader_id = glCreateShader (shader_type);
|
|
|
|
glShaderSource (shader_id,
|
|
2,
|
|
(const char *[]) {
|
|
preamble->str,
|
|
g_bytes_get_data (bytes, NULL),
|
|
},
|
|
NULL);
|
|
|
|
g_bytes_unref (bytes);
|
|
g_string_free (preamble, TRUE);
|
|
|
|
glCompileShader (shader_id);
|
|
|
|
print_shader_info (shader_type == GL_FRAGMENT_SHADER ? "fragment" : "vertex", shader_id, program_name);
|
|
|
|
if (!gsk_gl_device_check_shader_error (shader_id, error))
|
|
{
|
|
glDeleteShader (shader_id);
|
|
return 0;
|
|
}
|
|
|
|
return shader_id;
|
|
}
|
|
|
|
static GLuint
|
|
gsk_gl_device_load_program (GskGLDevice *self,
|
|
const GskGpuShaderOpClass *op_class,
|
|
guint32 variation,
|
|
GskGpuShaderClip clip,
|
|
guint n_external_textures,
|
|
GError **error)
|
|
{
|
|
G_GNUC_UNUSED gint64 begin_time = GDK_PROFILER_CURRENT_TIME;
|
|
GLuint vertex_shader_id, fragment_shader_id, program_id;
|
|
GLint link_status;
|
|
|
|
vertex_shader_id = gsk_gl_device_load_shader (self, op_class->shader_name, GL_VERTEX_SHADER, variation, clip, n_external_textures, error);
|
|
if (vertex_shader_id == 0)
|
|
return 0;
|
|
|
|
fragment_shader_id = gsk_gl_device_load_shader (self, op_class->shader_name, GL_FRAGMENT_SHADER, variation, clip, n_external_textures, error);
|
|
if (fragment_shader_id == 0)
|
|
return 0;
|
|
|
|
program_id = glCreateProgram ();
|
|
|
|
glAttachShader (program_id, vertex_shader_id);
|
|
glAttachShader (program_id, fragment_shader_id);
|
|
|
|
op_class->setup_attrib_locations (program_id);
|
|
|
|
glLinkProgram (program_id);
|
|
|
|
glGetProgramiv (program_id, GL_LINK_STATUS, &link_status);
|
|
|
|
glDetachShader (program_id, vertex_shader_id);
|
|
glDeleteShader (vertex_shader_id);
|
|
glDetachShader (program_id, fragment_shader_id);
|
|
glDeleteShader (fragment_shader_id);
|
|
|
|
if (link_status == GL_FALSE)
|
|
{
|
|
char *buffer = NULL;
|
|
int log_len = 0;
|
|
|
|
glGetProgramiv (program_id, GL_INFO_LOG_LENGTH, &log_len);
|
|
|
|
if (log_len > 0)
|
|
{
|
|
/* log_len includes NULL */
|
|
buffer = g_malloc0 (log_len);
|
|
glGetProgramInfoLog (program_id, log_len, NULL, buffer);
|
|
}
|
|
|
|
g_set_error (error,
|
|
GDK_GL_ERROR,
|
|
GDK_GL_ERROR_LINK_FAILED,
|
|
"Linking failure in shader: %s",
|
|
buffer ? buffer : "");
|
|
|
|
g_free (buffer);
|
|
|
|
glDeleteProgram (program_id);
|
|
|
|
return 0;
|
|
}
|
|
|
|
gdk_profiler_end_markf (begin_time,
|
|
"Compile Program",
|
|
"name=%s id=%u frag=%u vert=%u",
|
|
op_class->shader_name, program_id, fragment_shader_id, vertex_shader_id);
|
|
|
|
return program_id;
|
|
}
|
|
|
|
void
|
|
gsk_gl_device_use_program (GskGLDevice *self,
|
|
const GskGpuShaderOpClass *op_class,
|
|
guint32 variation,
|
|
GskGpuShaderClip clip,
|
|
guint n_external_textures)
|
|
{
|
|
GError *error = NULL;
|
|
GLuint program_id;
|
|
GLProgramKey key = {
|
|
.op_class = op_class,
|
|
.variation = variation,
|
|
.clip = clip,
|
|
.n_external_textures = n_external_textures
|
|
};
|
|
guint i, n_textures;
|
|
|
|
program_id = GPOINTER_TO_UINT (g_hash_table_lookup (self->gl_programs, &key));
|
|
if (program_id)
|
|
{
|
|
glUseProgram (program_id);
|
|
return;
|
|
}
|
|
|
|
program_id = gsk_gl_device_load_program (self, op_class, variation, clip, n_external_textures, &error);
|
|
if (program_id == 0)
|
|
{
|
|
g_critical ("Failed to load shader program: %s", error->message);
|
|
g_clear_error (&error);
|
|
return;
|
|
}
|
|
|
|
g_hash_table_insert (self->gl_programs, g_memdup (&key, sizeof (GLProgramKey)), GUINT_TO_POINTER (program_id));
|
|
|
|
glUseProgram (program_id);
|
|
|
|
n_textures = 16 - 3 * n_external_textures;
|
|
|
|
for (i = 0; i < n_external_textures; i++)
|
|
{
|
|
char *name = g_strdup_printf ("external_textures[%u]", i);
|
|
glUniform1i (glGetUniformLocation (program_id, name), n_textures + 3 * i);
|
|
g_free (name);
|
|
}
|
|
|
|
for (i = 0; i < n_textures; i++)
|
|
{
|
|
char *name = g_strdup_printf ("textures[%u]", i);
|
|
glUniform1i (glGetUniformLocation (program_id, name), i);
|
|
g_free (name);
|
|
}
|
|
}
|
|
|
|
GLuint
|
|
gsk_gl_device_get_sampler_id (GskGLDevice *self,
|
|
GskGpuSampler sampler)
|
|
{
|
|
g_return_val_if_fail (sampler < G_N_ELEMENTS (self->sampler_ids), 0);
|
|
|
|
return self->sampler_ids[sampler];
|
|
}
|
|
|
|
static gboolean
|
|
gsk_gl_device_get_format_flags (GskGLDevice *self,
|
|
GdkGLContext *context,
|
|
GdkMemoryFormat format,
|
|
GskGpuImageFlags *out_flags)
|
|
{
|
|
GdkGLMemoryFlags gl_flags;
|
|
|
|
*out_flags = 0;
|
|
gl_flags = gdk_gl_context_get_format_flags (context, format);
|
|
|
|
if (!(gl_flags & GDK_GL_FORMAT_USABLE))
|
|
return FALSE;
|
|
|
|
if (gl_flags & GDK_GL_FORMAT_RENDERABLE)
|
|
*out_flags |= GSK_GPU_IMAGE_RENDERABLE;
|
|
else if (gdk_gl_context_get_use_es (context))
|
|
*out_flags |= GSK_GPU_IMAGE_NO_BLIT;
|
|
if (gl_flags & GDK_GL_FORMAT_FILTERABLE)
|
|
*out_flags |= GSK_GPU_IMAGE_FILTERABLE;
|
|
if ((gl_flags & (GDK_GL_FORMAT_RENDERABLE | GDK_GL_FORMAT_FILTERABLE)) == (GDK_GL_FORMAT_RENDERABLE | GDK_GL_FORMAT_FILTERABLE))
|
|
*out_flags |= GSK_GPU_IMAGE_CAN_MIPMAP;
|
|
|
|
if (gdk_memory_format_alpha (format) == GDK_MEMORY_ALPHA_STRAIGHT)
|
|
*out_flags |= GSK_GPU_IMAGE_STRAIGHT_ALPHA;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gsk_gl_device_find_gl_format (GskGLDevice *self,
|
|
GdkMemoryFormat format,
|
|
GskGpuImageFlags required_flags,
|
|
GdkMemoryFormat *out_format,
|
|
GskGpuImageFlags *out_flags,
|
|
GLint *out_gl_internal_format,
|
|
GLint *out_gl_internal_srgb_format,
|
|
GLenum *out_gl_format,
|
|
GLenum *out_gl_type,
|
|
GLint out_swizzle[4])
|
|
{
|
|
GdkGLContext *context = gdk_gl_context_get_current ();
|
|
GskGpuImageFlags flags;
|
|
GdkMemoryFormat alt_format;
|
|
const GdkMemoryFormat *fallbacks;
|
|
gsize i;
|
|
|
|
/* First, try the actual format */
|
|
if (gsk_gl_device_get_format_flags (self, context, format, &flags) &&
|
|
((flags & required_flags) == required_flags))
|
|
{
|
|
*out_format = format;
|
|
*out_flags = flags;
|
|
gdk_memory_format_gl_format (format,
|
|
gdk_gl_context_get_use_es (context),
|
|
out_gl_internal_format,
|
|
out_gl_internal_srgb_format,
|
|
out_gl_format,
|
|
out_gl_type,
|
|
out_swizzle);
|
|
return;
|
|
}
|
|
|
|
/* Second, try the potential RGBA format */
|
|
if (gdk_memory_format_gl_rgba_format (format,
|
|
gdk_gl_context_get_use_es (context),
|
|
&alt_format,
|
|
out_gl_internal_format,
|
|
out_gl_internal_srgb_format,
|
|
out_gl_format,
|
|
out_gl_type,
|
|
out_swizzle) &&
|
|
gsk_gl_device_get_format_flags (self, context, alt_format, &flags) &&
|
|
((flags & required_flags) == required_flags))
|
|
{
|
|
*out_format = format;
|
|
*out_flags = flags;
|
|
return;
|
|
}
|
|
|
|
/* Next, try the fallbacks */
|
|
fallbacks = gdk_memory_format_get_fallbacks (format);
|
|
for (i = 0; fallbacks[i] != -1; i++)
|
|
{
|
|
if (gsk_gl_device_get_format_flags (self, context, fallbacks[i], &flags) &&
|
|
((flags & required_flags) == required_flags))
|
|
{
|
|
*out_format = fallbacks[i];
|
|
*out_flags = flags;
|
|
gdk_memory_format_gl_format (fallbacks[i],
|
|
gdk_gl_context_get_use_es (context),
|
|
out_gl_internal_format,
|
|
out_gl_internal_srgb_format,
|
|
out_gl_format,
|
|
out_gl_type,
|
|
out_swizzle);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* fallbacks will always fallback to a supported format */
|
|
g_assert_not_reached ();
|
|
}
|
|
|