mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-15 06:40:08 +00:00
3bc1e0534f
After the device/cache split, this was forgotten.
293 lines
8.0 KiB
C
293 lines
8.0 KiB
C
#include "config.h"
|
|
|
|
#include "gskgpudeviceprivate.h"
|
|
|
|
#include "gskgpucacheprivate.h"
|
|
|
|
#include "gdk/gdkprofilerprivate.h"
|
|
|
|
#include "gsk/gskdebugprivate.h"
|
|
|
|
#define CACHE_TIMEOUT 15 /* seconds */
|
|
|
|
typedef struct _GskGpuDevicePrivate GskGpuDevicePrivate;
|
|
|
|
struct _GskGpuDevicePrivate
|
|
{
|
|
GdkDisplay *display;
|
|
gsize max_image_size;
|
|
gsize tile_size;
|
|
|
|
GskGpuCache *cache; /* we don't own a ref, but manage the cache */
|
|
guint cache_gc_source;
|
|
int cache_timeout; /* in seconds, or -1 to disable gc */
|
|
};
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GskGpuDevice, gsk_gpu_device, G_TYPE_OBJECT)
|
|
|
|
/* Returns TRUE if everything was GC'ed */
|
|
static gboolean
|
|
gsk_gpu_device_gc (GskGpuDevice *self,
|
|
gint64 timestamp)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
gint64 before G_GNUC_UNUSED = GDK_PROFILER_CURRENT_TIME;
|
|
gboolean result;
|
|
|
|
if (priv->cache == NULL)
|
|
return TRUE;
|
|
|
|
gsk_gpu_device_make_current (self);
|
|
|
|
result = gsk_gpu_cache_gc (priv->cache,
|
|
priv->cache_timeout >= 0 ? priv->cache_timeout * G_TIME_SPAN_SECOND : -1,
|
|
timestamp);
|
|
if (result)
|
|
g_clear_object (&priv->cache);
|
|
|
|
gdk_profiler_end_mark (before, "Glyph cache GC", NULL);
|
|
|
|
return result;
|
|
}
|
|
|
|
static gboolean
|
|
cache_gc_cb (gpointer data)
|
|
{
|
|
GskGpuDevice *self = data;
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
gint64 timestamp;
|
|
gboolean result = G_SOURCE_CONTINUE;
|
|
|
|
timestamp = g_get_monotonic_time ();
|
|
GSK_DEBUG (CACHE, "Periodic GC (timestamp %lld)", (long long) timestamp);
|
|
|
|
/* gc can collect the device if all windows are closed and only
|
|
* the cache is keeping it alive */
|
|
g_object_ref (self);
|
|
|
|
if (gsk_gpu_device_gc (self, timestamp))
|
|
{
|
|
priv->cache_gc_source = 0;
|
|
result = G_SOURCE_REMOVE;
|
|
}
|
|
|
|
g_object_unref (self);
|
|
|
|
return result;
|
|
}
|
|
|
|
void
|
|
gsk_gpu_device_maybe_gc (GskGpuDevice *self)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
gsize dead_texture_pixels;
|
|
|
|
if (priv->cache_timeout < 0)
|
|
return;
|
|
|
|
if (priv->cache == NULL)
|
|
return;
|
|
|
|
dead_texture_pixels = gsk_gpu_cache_get_dead_texture_pixels (priv->cache);
|
|
|
|
if (priv->cache_timeout == 0 || dead_texture_pixels > 1000000)
|
|
{
|
|
GSK_DEBUG (CACHE, "Pre-frame GC (%" G_GSIZE_FORMAT " dead pixels)", dead_texture_pixels);
|
|
gsk_gpu_device_gc (self, g_get_monotonic_time ());
|
|
}
|
|
}
|
|
|
|
void
|
|
gsk_gpu_device_queue_gc (GskGpuDevice *self)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
if (priv->cache_timeout > 0 && !priv->cache_gc_source)
|
|
priv->cache_gc_source = g_timeout_add_seconds (priv->cache_timeout, cache_gc_cb, self);
|
|
}
|
|
|
|
static void
|
|
gsk_gpu_device_dispose (GObject *object)
|
|
{
|
|
GskGpuDevice *self = GSK_GPU_DEVICE (object);
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
g_clear_handle_id (&priv->cache_gc_source, g_source_remove);
|
|
|
|
G_OBJECT_CLASS (gsk_gpu_device_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gsk_gpu_device_finalize (GObject *object)
|
|
{
|
|
GskGpuDevice *self = GSK_GPU_DEVICE (object);
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
g_object_unref (priv->display);
|
|
|
|
G_OBJECT_CLASS (gsk_gpu_device_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gsk_gpu_device_class_init (GskGpuDeviceClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->dispose = gsk_gpu_device_dispose;
|
|
object_class->finalize = gsk_gpu_device_finalize;
|
|
}
|
|
|
|
static void
|
|
gsk_gpu_device_init (GskGpuDevice *self)
|
|
{
|
|
}
|
|
void
|
|
gsk_gpu_device_setup (GskGpuDevice *self,
|
|
GdkDisplay *display,
|
|
gsize max_image_size,
|
|
gsize tile_size)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
const char *str;
|
|
|
|
priv->display = g_object_ref (display);
|
|
priv->max_image_size = max_image_size;
|
|
priv->tile_size = tile_size;
|
|
priv->cache_timeout = CACHE_TIMEOUT;
|
|
|
|
str = g_getenv ("GSK_CACHE_TIMEOUT");
|
|
if (str != NULL)
|
|
{
|
|
gint64 value;
|
|
GError *error = NULL;
|
|
|
|
if (!g_ascii_string_to_signed (str, 10, -1, G_MAXINT, &value, &error))
|
|
{
|
|
g_warning ("Failed to parse GSK_CACHE_TIMEOUT: %s", error->message);
|
|
g_error_free (error);
|
|
}
|
|
else
|
|
{
|
|
priv->cache_timeout = (int) value;
|
|
}
|
|
}
|
|
|
|
if (GSK_DEBUG_CHECK (CACHE))
|
|
{
|
|
if (priv->cache_timeout < 0)
|
|
gdk_debug_message ("Cache GC disabled");
|
|
else if (priv->cache_timeout == 0)
|
|
gdk_debug_message ("Cache GC before every frame");
|
|
else
|
|
gdk_debug_message ("Cache GC timeout: %d seconds", priv->cache_timeout);
|
|
}
|
|
}
|
|
|
|
GdkDisplay *
|
|
gsk_gpu_device_get_display (GskGpuDevice *self)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
return priv->display;
|
|
}
|
|
|
|
GskGpuCache *
|
|
gsk_gpu_device_get_cache (GskGpuDevice *self)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
if (G_LIKELY (priv->cache))
|
|
return priv->cache;
|
|
|
|
priv->cache = gsk_gpu_cache_new (self);
|
|
|
|
return priv->cache;
|
|
}
|
|
|
|
/*<private>
|
|
* gsk_gpu_device_get_max_image_size:
|
|
* @self: a device
|
|
*
|
|
* Returns the max image size supported by this device.
|
|
*
|
|
* This maps to GL_MAX_TEXTURE_SIZE on GL, but Vulkan is more flexible with
|
|
* per-format size limits, so this is an estimate and code should still handle
|
|
* failures of image creation at smaller sizes. (Besides handling them anyway
|
|
* in case of OOM.)
|
|
*
|
|
* Returns: The maximum size in pixels for creating a GskGpuImage
|
|
**/
|
|
gsize
|
|
gsk_gpu_device_get_max_image_size (GskGpuDevice *self)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
return priv->max_image_size;
|
|
}
|
|
|
|
/*<private>
|
|
* gsk_gpu_device_get_tile_size:
|
|
* @self: a device
|
|
*
|
|
* The suggested size for tiling images. This value will be small enough so that
|
|
* image creation never fails due to size constraints. It should also not be too
|
|
* large to allow efficient caching of tiles and evictions of unused tiles
|
|
* (think of an image editor showing only a section of a large image).
|
|
*
|
|
* Returns: The suggested size of tiles when tiling images.
|
|
**/
|
|
gsize
|
|
gsk_gpu_device_get_tile_size (GskGpuDevice *self)
|
|
{
|
|
GskGpuDevicePrivate *priv = gsk_gpu_device_get_instance_private (self);
|
|
|
|
return priv->tile_size;
|
|
}
|
|
|
|
GskGpuImage *
|
|
gsk_gpu_device_create_offscreen_image (GskGpuDevice *self,
|
|
gboolean with_mipmap,
|
|
GdkMemoryDepth depth,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
return GSK_GPU_DEVICE_GET_CLASS (self)->create_offscreen_image (self, with_mipmap, depth, width, height);
|
|
}
|
|
|
|
GskGpuImage *
|
|
gsk_gpu_device_create_atlas_image (GskGpuDevice *self,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
return GSK_GPU_DEVICE_GET_CLASS (self)->create_atlas_image (self, width, height);
|
|
}
|
|
|
|
GskGpuImage *
|
|
gsk_gpu_device_create_upload_image (GskGpuDevice *self,
|
|
gboolean with_mipmap,
|
|
GdkMemoryFormat format,
|
|
gboolean try_srgb,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
return GSK_GPU_DEVICE_GET_CLASS (self)->create_upload_image (self, with_mipmap, format, try_srgb, width, height);
|
|
}
|
|
|
|
GskGpuImage *
|
|
gsk_gpu_device_create_download_image (GskGpuDevice *self,
|
|
GdkMemoryDepth depth,
|
|
gsize width,
|
|
gsize height)
|
|
{
|
|
return GSK_GPU_DEVICE_GET_CLASS (self)->create_download_image (self, depth, width, height);
|
|
}
|
|
|
|
void
|
|
gsk_gpu_device_make_current (GskGpuDevice *self)
|
|
{
|
|
GSK_GPU_DEVICE_GET_CLASS (self)->make_current (self);
|
|
}
|
|
|
|
/* }}} */
|
|
/* vim:set foldmethod=marker expandtab: */
|