gpu: Make pattern creation always succeed

If creation fails, create an offscreen image instead and draw that as a
texture.

Because offscreens basically always succeed, we can pretty much assume
success everywhere - apart from pattern creation functions that also
create images, because they can run out of shader space.
This commit is contained in:
Benjamin Otte 2023-09-02 01:18:04 +02:00
parent 4cbd384e39
commit 9224efd95e
4 changed files with 106 additions and 5 deletions

View File

@ -18,6 +18,21 @@ gsk_gpu_buffer_writer_abort (GskGpuBufferWriter *self)
self->finish (self, FALSE); self->finish (self, FALSE);
} }
gsize
gsk_gpu_buffer_writer_get_size (GskGpuBufferWriter *self)
{
return self->size;
}
void
gsk_gpu_buffer_writer_rewind (GskGpuBufferWriter *self,
gsize size)
{
g_assert (size <= self->size);
self->size = size;
}
void void
gsk_gpu_buffer_writer_ensure_size (GskGpuBufferWriter *self, gsk_gpu_buffer_writer_ensure_size (GskGpuBufferWriter *self,
gsize size) gsize size)

View File

@ -24,6 +24,9 @@ struct _GskGpuBufferWriter
gsize gsk_gpu_buffer_writer_commit (GskGpuBufferWriter *self); gsize gsk_gpu_buffer_writer_commit (GskGpuBufferWriter *self);
void gsk_gpu_buffer_writer_abort (GskGpuBufferWriter *self); void gsk_gpu_buffer_writer_abort (GskGpuBufferWriter *self);
gsize gsk_gpu_buffer_writer_get_size (GskGpuBufferWriter *self);
void gsk_gpu_buffer_writer_rewind (GskGpuBufferWriter *self,
gsize size);
void gsk_gpu_buffer_writer_ensure_size (GskGpuBufferWriter *self, void gsk_gpu_buffer_writer_ensure_size (GskGpuBufferWriter *self,
gsize size); gsize size);
void gsk_gpu_buffer_writer_append (GskGpuBufferWriter *self, void gsk_gpu_buffer_writer_append (GskGpuBufferWriter *self,

View File

@ -319,6 +319,74 @@ gsk_gpu_node_processor_clip_node_bounds (GskGpuNodeProcessor *self,
return TRUE; return TRUE;
} }
static GskGpuImage *
gsk_gpu_node_procesor_get_node_as_image (GskGpuNodeProcessor *self,
GskRenderNode *node,
graphene_rect_t *out_bounds)
{
GskGpuImage *result;
switch ((guint) gsk_render_node_get_node_type (node))
{
case GSK_TEXTURE_NODE:
{
GdkTexture *texture = gsk_texture_node_get_texture (node);
GskGpuDevice *device = gsk_gpu_frame_get_device (self->frame);
gint64 timestamp = gsk_gpu_frame_get_timestamp (self->frame);
result = gsk_gpu_device_lookup_texture_image (device, texture, timestamp);
if (result == NULL)
{
result = gsk_gpu_upload_texture_op (self->frame, texture);
g_object_ref (result);
gsk_gpu_device_cache_texture_image (device, texture, timestamp, result);
}
*out_bounds = node->bounds;
return result;
}
case GSK_CAIRO_NODE:
{
graphene_rect_t clipped;
graphene_rect_offset_r (&self->clip.rect.bounds, - self->offset.x, - self->offset.y, &clipped);
graphene_rect_intersection (&clipped, &node->bounds, &clipped);
if (clipped.size.width == 0 || clipped.size.height == 0)
return NULL;
result = gsk_gpu_upload_cairo_op (self->frame,
node,
&self->scale,
&clipped);
g_object_ref (result);
*out_bounds = clipped;
return result;
}
default:
{
graphene_rect_t clipped;
graphene_rect_offset_r (&self->clip.rect.bounds, - self->offset.x, - self->offset.y, &clipped);
graphene_rect_intersection (&clipped, &node->bounds, &clipped);
if (clipped.size.width == 0 || clipped.size.height == 0)
return NULL;
GSK_DEBUG (FALLBACK, "Offscreening node '%s'", g_type_name_from_instance ((GTypeInstance *) node));
result = gsk_gpu_render_pass_op_offscreen (self->frame,
&self->scale,
&clipped,
node);
*out_bounds = clipped;
return result;
}
}
}
static void static void
gsk_gpu_node_processor_add_fallback_node (GskGpuNodeProcessor *self, gsk_gpu_node_processor_add_fallback_node (GskGpuNodeProcessor *self,
GskRenderNode *node) GskRenderNode *node)
@ -949,6 +1017,7 @@ gsk_gpu_node_processor_create_node_pattern (GskGpuNodeProcessor *self,
gsize *out_n_images) gsize *out_n_images)
{ {
GskRenderNodeType node_type; GskRenderNodeType node_type;
graphene_rect_t bounds;
node_type = gsk_render_node_get_node_type (node); node_type = gsk_render_node_get_node_type (node);
if (node_type >= G_N_ELEMENTS (nodes_vtable)) if (node_type >= G_N_ELEMENTS (nodes_vtable))
@ -959,11 +1028,27 @@ gsk_gpu_node_processor_create_node_pattern (GskGpuNodeProcessor *self,
*out_n_images = 0; *out_n_images = 0;
if (nodes_vtable[node_type].create_pattern == NULL) if (nodes_vtable[node_type].create_pattern != NULL)
{
gsize size_before = gsk_gpu_buffer_writer_get_size (writer);
if (nodes_vtable[node_type].create_pattern (self, writer, node, images, n_images, out_n_images))
return TRUE;
gsk_gpu_buffer_writer_rewind (writer, size_before);
}
if (n_images == 0)
return FALSE; return FALSE;
if (!nodes_vtable[node_type].create_pattern (self, writer, node, images, n_images, out_n_images)) images[0].image = gsk_gpu_node_procesor_get_node_as_image (self, node, &bounds);
return FALSE; images[0].sampler = GSK_GPU_SAMPLER_DEFAULT;
images[0].descriptor = gsk_gpu_frame_get_image_descriptor (self->frame,
images[0].image,
images[0].sampler);
*out_n_images = 1;
gsk_gpu_buffer_writer_append_uint (writer, GSK_GPU_PATTERN_TEXTURE);
gsk_gpu_buffer_writer_append_uint (writer, images[0].descriptor);
gsk_gpu_buffer_writer_append_rect (writer, &bounds, &self->offset);
return TRUE; return TRUE;
} }

View File

@ -333,7 +333,5 @@ gsk_gpu_render_pass_op_offscreen (GskGpuFrame *frame,
image, image,
GSK_RENDER_PASS_OFFSCREEN); GSK_RENDER_PASS_OFFSCREEN);
g_object_unref (image);
return image; return image;
} }