2010-11-14 17:22:05 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
/*
|
|
|
|
* Utilities for use with Cogl
|
|
|
|
*
|
|
|
|
* Copyright 2010 Red Hat, Inc.
|
|
|
|
* Copyright 2010 Intel Corporation
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2014-01-12 01:42:06 +00:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2010-11-14 17:22:05 +00:00
|
|
|
*/
|
|
|
|
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "clutter/clutter.h"
|
|
|
|
#include "compositor/cogl-utils.h"
|
2010-11-14 17:22:05 +00:00
|
|
|
|
|
|
|
/* Based on gnome-shell/src/st/st-private.c:_st_create_texture_material.c */
|
|
|
|
|
|
|
|
/**
|
2013-02-20 00:34:47 +00:00
|
|
|
* meta_create_texture_pipeline:
|
2014-05-28 19:44:23 +00:00
|
|
|
* @src_texture: (nullable): texture to use initially for the layer
|
2010-11-14 17:22:05 +00:00
|
|
|
*
|
2013-02-20 00:34:47 +00:00
|
|
|
* Creates a pipeline with a single layer. Using a common template
|
2013-09-04 12:38:54 +00:00
|
|
|
* makes it easier for Cogl to share a shader for different uses in
|
|
|
|
* Mutter.
|
2010-11-14 17:22:05 +00:00
|
|
|
*
|
2013-02-20 00:34:47 +00:00
|
|
|
* Return value: (transfer full): a newly created #CoglPipeline
|
2010-11-14 17:22:05 +00:00
|
|
|
*/
|
2013-02-20 00:34:47 +00:00
|
|
|
CoglPipeline *
|
|
|
|
meta_create_texture_pipeline (CoglTexture *src_texture)
|
2010-11-14 17:22:05 +00:00
|
|
|
{
|
2013-02-20 00:34:47 +00:00
|
|
|
static CoglPipeline *texture_pipeline_template = NULL;
|
|
|
|
CoglPipeline *pipeline;
|
2010-11-14 17:22:05 +00:00
|
|
|
|
2013-09-04 12:38:54 +00:00
|
|
|
/* The only state used in the pipeline that would affect the shader
|
|
|
|
generation is the texture type on the layer. Therefore we create
|
|
|
|
a template pipeline which sets this state and all texture
|
|
|
|
pipelines are created as a copy of this. That way Cogl can find
|
|
|
|
the shader state for the pipeline more quickly by looking at the
|
|
|
|
pipeline ancestry instead of resorting to the shader cache. */
|
2013-02-20 00:34:47 +00:00
|
|
|
if (G_UNLIKELY (texture_pipeline_template == NULL))
|
2010-11-14 17:22:05 +00:00
|
|
|
{
|
2013-09-04 12:38:54 +00:00
|
|
|
CoglContext *ctx =
|
|
|
|
clutter_backend_get_cogl_context (clutter_get_default_backend ());
|
2013-02-20 00:34:47 +00:00
|
|
|
|
|
|
|
texture_pipeline_template = cogl_pipeline_new (ctx);
|
2013-09-04 12:38:54 +00:00
|
|
|
cogl_pipeline_set_layer_null_texture (texture_pipeline_template,
|
|
|
|
0, /* layer */
|
|
|
|
COGL_TEXTURE_TYPE_2D);
|
2010-11-14 17:22:05 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 00:34:47 +00:00
|
|
|
pipeline = cogl_pipeline_copy (texture_pipeline_template);
|
2010-11-14 17:22:05 +00:00
|
|
|
|
2013-02-20 00:34:47 +00:00
|
|
|
if (src_texture != NULL)
|
|
|
|
cogl_pipeline_set_layer_texture (pipeline, 0, src_texture);
|
2010-11-14 17:22:05 +00:00
|
|
|
|
2013-02-20 00:34:47 +00:00
|
|
|
return pipeline;
|
2010-11-14 17:22:05 +00:00
|
|
|
}
|
2014-08-10 15:26:14 +00:00
|
|
|
|
|
|
|
static gboolean is_pot(int x)
|
|
|
|
{
|
|
|
|
return x > 0 && (x & (x - 1)) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_create_texture:
|
|
|
|
* @width: width of the texture to create
|
|
|
|
* @height: height of the texture to create
|
|
|
|
* @components; components to store in the texture (color or alpha)
|
|
|
|
* @flags: flags that affect the allocation behavior
|
|
|
|
*
|
|
|
|
* Creates a texture of the given size with the specified components
|
|
|
|
* for use as a frame buffer object.
|
|
|
|
*
|
|
|
|
* If non-power-of-two textures are not supported on the system, then
|
|
|
|
* the texture will be created as a texture rectangle; in this case,
|
|
|
|
* hardware repeating isn't possible, and texture coordinates are also
|
|
|
|
* different, but Cogl hides these issues from the application, except from
|
|
|
|
* GLSL shaders. Since GLSL is never (or at least almost never)
|
|
|
|
* present on such a system, this is not typically an issue.
|
|
|
|
*
|
|
|
|
* If %META_TEXTURE_ALLOW_SLICING is present in @flags, and the texture
|
|
|
|
* is larger than the texture size limits of the system, then the texture
|
|
|
|
* will be created as a sliced texture. This also will cause problems
|
|
|
|
* with using the texture with GLSL, and is more likely to be an issue
|
|
|
|
* since all GL implementations have texture size limits, and they can
|
|
|
|
* be as small as 2048x2048 on reasonably current systems.
|
|
|
|
*/
|
|
|
|
CoglTexture *
|
|
|
|
meta_create_texture (int width,
|
|
|
|
int height,
|
|
|
|
CoglTextureComponents components,
|
|
|
|
MetaTextureFlags flags)
|
|
|
|
{
|
|
|
|
ClutterBackend *backend = clutter_get_default_backend ();
|
|
|
|
CoglContext *ctx = clutter_backend_get_cogl_context (backend);
|
|
|
|
CoglTexture *texture;
|
|
|
|
|
|
|
|
gboolean should_use_rectangle = FALSE;
|
|
|
|
|
|
|
|
if (!(is_pot (width) && is_pot (height)) &&
|
|
|
|
!cogl_has_feature (ctx, COGL_FEATURE_ID_TEXTURE_NPOT))
|
|
|
|
{
|
|
|
|
if (cogl_has_feature (ctx, COGL_FEATURE_ID_TEXTURE_RECTANGLE))
|
|
|
|
should_use_rectangle = TRUE;
|
|
|
|
else
|
|
|
|
g_error ("Cannot create texture. Support for GL_ARB_texture_non_power_of_two or "
|
|
|
|
"ARB_texture_rectangle is required");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (should_use_rectangle)
|
|
|
|
texture = COGL_TEXTURE (cogl_texture_rectangle_new_with_size (ctx, width, height));
|
|
|
|
else
|
|
|
|
texture = COGL_TEXTURE (cogl_texture_2d_new_with_size (ctx, width, height));
|
|
|
|
cogl_texture_set_components (texture, components);
|
|
|
|
|
|
|
|
if ((flags & META_TEXTURE_ALLOW_SLICING) != 0)
|
|
|
|
{
|
|
|
|
/* To find out if we need to slice the texture, we have to go ahead and force storage
|
|
|
|
* to be allocated
|
|
|
|
*/
|
|
|
|
CoglError *catch_error = NULL;
|
|
|
|
if (!cogl_texture_allocate (texture, &catch_error))
|
|
|
|
{
|
|
|
|
cogl_error_free (catch_error);
|
|
|
|
cogl_object_unref (texture);
|
|
|
|
texture = COGL_TEXTURE (cogl_texture_2d_sliced_new_with_size (ctx, width, height, COGL_TEXTURE_MAX_WASTE));
|
|
|
|
cogl_texture_set_components (texture, components);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return texture;
|
|
|
|
}
|