2010-02-26 12:38:22 +00:00
|
|
|
/*
|
|
|
|
* Cogl
|
|
|
|
*
|
|
|
|
* An object oriented GL/GLES Abstraction/Utility Layer
|
|
|
|
*
|
|
|
|
* Copyright (C) 2011 Intel Corporation.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Neil Roberts <neil@linux.intel.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
#include <cogl/cogl.h>
|
|
|
|
#include "cogl-pango-pipeline-cache.h"
|
|
|
|
|
2012-02-18 16:03:10 +00:00
|
|
|
#include "cogl/cogl-context-private.h"
|
|
|
|
|
2010-02-26 12:38:22 +00:00
|
|
|
typedef struct _CoglPangoPipelineCacheEntry CoglPangoPipelineCacheEntry;
|
|
|
|
|
|
|
|
struct _CoglPangoPipelineCache
|
|
|
|
{
|
|
|
|
GHashTable *hash_table;
|
|
|
|
|
2010-02-26 13:01:54 +00:00
|
|
|
CoglPipeline *base_texture_alpha_pipeline;
|
|
|
|
CoglPipeline *base_texture_rgba_pipeline;
|
2010-02-26 12:38:22 +00:00
|
|
|
|
|
|
|
gboolean use_mipmapping;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _CoglPangoPipelineCacheEntry
|
|
|
|
{
|
|
|
|
/* This will take a reference or it can be NULL to represent the
|
|
|
|
pipeline used to render colors */
|
2011-08-24 21:30:34 +01:00
|
|
|
CoglTexture *texture;
|
2010-02-26 12:38:22 +00:00
|
|
|
|
|
|
|
/* This will only take a weak reference */
|
|
|
|
CoglHandle pipeline;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
_cogl_pango_pipeline_cache_key_destroy (gpointer data)
|
|
|
|
{
|
|
|
|
if (data)
|
|
|
|
cogl_object_unref (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_cogl_pango_pipeline_cache_value_destroy (gpointer data)
|
|
|
|
{
|
|
|
|
CoglPangoPipelineCacheEntry *cache_entry = data;
|
|
|
|
|
|
|
|
if (cache_entry->texture)
|
|
|
|
cogl_object_unref (cache_entry->texture);
|
|
|
|
|
|
|
|
/* We don't need to unref the pipeline because it only takes a weak
|
|
|
|
reference */
|
|
|
|
|
|
|
|
g_slice_free (CoglPangoPipelineCacheEntry, cache_entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
CoglPangoPipelineCache *
|
|
|
|
_cogl_pango_pipeline_cache_new (gboolean use_mipmapping)
|
|
|
|
{
|
|
|
|
CoglPangoPipelineCache *cache = g_new (CoglPangoPipelineCache, 1);
|
|
|
|
|
|
|
|
/* The key is the pipeline pointer. A reference is taken when the
|
|
|
|
pipeline is used as a key so we should unref it again in the
|
|
|
|
destroy function */
|
|
|
|
cache->hash_table =
|
|
|
|
g_hash_table_new_full (g_direct_hash,
|
|
|
|
g_direct_equal,
|
|
|
|
_cogl_pango_pipeline_cache_key_destroy,
|
|
|
|
_cogl_pango_pipeline_cache_value_destroy);
|
|
|
|
|
2010-02-26 13:01:54 +00:00
|
|
|
cache->base_texture_rgba_pipeline = NULL;
|
|
|
|
cache->base_texture_alpha_pipeline = NULL;
|
2010-02-26 12:38:22 +00:00
|
|
|
|
|
|
|
cache->use_mipmapping = use_mipmapping;
|
|
|
|
|
|
|
|
return cache;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglPipeline *
|
2010-02-26 13:01:54 +00:00
|
|
|
get_base_texture_rgba_pipeline (CoglPangoPipelineCache *cache)
|
2010-02-26 12:38:22 +00:00
|
|
|
{
|
2010-02-26 13:01:54 +00:00
|
|
|
if (cache->base_texture_rgba_pipeline == NULL)
|
2010-02-26 12:38:22 +00:00
|
|
|
{
|
|
|
|
CoglPipeline *pipeline;
|
|
|
|
|
2012-02-18 16:03:10 +00:00
|
|
|
_COGL_GET_CONTEXT (ctx, NULL);
|
|
|
|
|
|
|
|
pipeline = cache->base_texture_rgba_pipeline = cogl_pipeline_new (ctx);
|
2010-02-26 13:01:54 +00:00
|
|
|
|
|
|
|
cogl_pipeline_set_layer_wrap_mode (pipeline, 0,
|
2012-02-18 14:43:01 +00:00
|
|
|
COGL_PIPELINE_WRAP_MODE_CLAMP_TO_EDGE);
|
2010-02-26 13:01:54 +00:00
|
|
|
|
|
|
|
if (cache->use_mipmapping)
|
|
|
|
cogl_pipeline_set_layer_filters
|
|
|
|
(pipeline, 0,
|
|
|
|
COGL_PIPELINE_FILTER_LINEAR_MIPMAP_LINEAR,
|
|
|
|
COGL_PIPELINE_FILTER_LINEAR);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cache->base_texture_rgba_pipeline;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglPipeline *
|
|
|
|
get_base_texture_alpha_pipeline (CoglPangoPipelineCache *cache)
|
|
|
|
{
|
|
|
|
if (cache->base_texture_alpha_pipeline == NULL)
|
|
|
|
{
|
|
|
|
CoglPipeline *pipeline;
|
|
|
|
|
|
|
|
pipeline = cogl_pipeline_copy (get_base_texture_rgba_pipeline (cache));
|
|
|
|
cache->base_texture_alpha_pipeline = pipeline;
|
2010-02-26 12:38:22 +00:00
|
|
|
|
|
|
|
/* The default combine mode of materials is to modulate (A x B)
|
|
|
|
* the texture RGBA channels with the RGBA channels of the
|
|
|
|
* previous layer (which in our case is just the font color)
|
|
|
|
*
|
|
|
|
* Since the RGB for an alpha texture is defined as 0, this gives us:
|
|
|
|
*
|
|
|
|
* result.rgb = color.rgb * 0
|
|
|
|
* result.a = color.a * texture.a
|
|
|
|
*
|
|
|
|
* What we want is premultiplied rgba values:
|
|
|
|
*
|
|
|
|
* result.rgba = color.rgb * texture.a
|
|
|
|
* result.a = color.a * texture.a
|
|
|
|
*/
|
|
|
|
cogl_pipeline_set_layer_combine (pipeline, 0, /* layer */
|
|
|
|
"RGBA = MODULATE (PREVIOUS, TEXTURE[A])",
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2010-02-26 13:01:54 +00:00
|
|
|
return cache->base_texture_alpha_pipeline;
|
2010-02-26 12:38:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
CoglPangoPipelineCache *cache;
|
2011-08-24 21:30:34 +01:00
|
|
|
CoglTexture *texture;
|
2010-02-26 12:38:22 +00:00
|
|
|
} PipelineDestroyNotifyData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
pipeline_destroy_notify_cb (void *user_data)
|
|
|
|
{
|
|
|
|
PipelineDestroyNotifyData *data = user_data;
|
|
|
|
|
|
|
|
g_hash_table_remove (data->cache->hash_table, data->texture);
|
|
|
|
g_slice_free (PipelineDestroyNotifyData, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
CoglPipeline *
|
|
|
|
_cogl_pango_pipeline_cache_get (CoglPangoPipelineCache *cache,
|
|
|
|
CoglHandle texture)
|
|
|
|
{
|
|
|
|
CoglPangoPipelineCacheEntry *entry;
|
|
|
|
PipelineDestroyNotifyData *destroy_data;
|
|
|
|
static CoglUserDataKey pipeline_destroy_notify_key;
|
|
|
|
|
|
|
|
/* Look for an existing entry */
|
|
|
|
entry = g_hash_table_lookup (cache->hash_table, texture);
|
|
|
|
|
|
|
|
if (entry)
|
|
|
|
return cogl_object_ref (entry->pipeline);
|
|
|
|
|
|
|
|
/* No existing pipeline was found so let's create another */
|
|
|
|
entry = g_slice_new (CoglPangoPipelineCacheEntry);
|
|
|
|
|
|
|
|
if (texture)
|
|
|
|
{
|
2010-02-26 13:01:54 +00:00
|
|
|
CoglPipeline *base;
|
|
|
|
|
2011-08-24 21:30:34 +01:00
|
|
|
entry->texture = cogl_object_ref (texture);
|
2010-02-26 13:01:54 +00:00
|
|
|
|
|
|
|
if (cogl_texture_get_format (entry->texture) == COGL_PIXEL_FORMAT_A_8)
|
|
|
|
base = get_base_texture_alpha_pipeline (cache);
|
|
|
|
else
|
|
|
|
base = get_base_texture_rgba_pipeline (cache);
|
|
|
|
|
|
|
|
entry->pipeline = cogl_pipeline_copy (base);
|
2010-02-26 12:38:22 +00:00
|
|
|
|
|
|
|
cogl_pipeline_set_layer_texture (entry->pipeline, 0 /* layer */, texture);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-02-18 16:03:10 +00:00
|
|
|
_COGL_GET_CONTEXT (ctx, NULL);
|
|
|
|
|
2010-02-26 12:38:22 +00:00
|
|
|
entry->texture = NULL;
|
2012-02-18 16:03:10 +00:00
|
|
|
entry->pipeline = cogl_pipeline_new (ctx);
|
2010-02-26 12:38:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Add a weak reference to the pipeline so we can remove it from the
|
|
|
|
hash table when it is destroyed */
|
|
|
|
destroy_data = g_slice_new (PipelineDestroyNotifyData);
|
|
|
|
destroy_data->cache = cache;
|
|
|
|
destroy_data->texture = texture;
|
|
|
|
cogl_object_set_user_data (entry->pipeline,
|
|
|
|
&pipeline_destroy_notify_key,
|
|
|
|
destroy_data,
|
|
|
|
pipeline_destroy_notify_cb);
|
|
|
|
|
|
|
|
g_hash_table_insert (cache->hash_table,
|
2011-08-24 21:30:34 +01:00
|
|
|
texture ? cogl_object_ref (texture) : NULL,
|
2010-02-26 12:38:22 +00:00
|
|
|
entry);
|
|
|
|
|
|
|
|
/* This doesn't take a reference on the pipeline so that it will use
|
|
|
|
the newly created reference */
|
|
|
|
return entry->pipeline;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_cogl_pango_pipeline_cache_free (CoglPangoPipelineCache *cache)
|
|
|
|
{
|
2010-02-26 13:01:54 +00:00
|
|
|
if (cache->base_texture_rgba_pipeline)
|
|
|
|
cogl_object_unref (cache->base_texture_rgba_pipeline);
|
|
|
|
if (cache->base_texture_alpha_pipeline)
|
|
|
|
cogl_object_unref (cache->base_texture_alpha_pipeline);
|
2010-02-26 12:38:22 +00:00
|
|
|
|
|
|
|
g_hash_table_destroy (cache->hash_table);
|
|
|
|
|
|
|
|
g_free (cache);
|
|
|
|
}
|