gtk/gsk/vulkan/gskvulkanmaskop.c
Benjamin Otte c0b185bee9 vulkan: Make Op->command() return the next op
This way, ops can batch themselves.

They don't dothat yet, but you know where this is going...
2023-07-16 12:13:00 +02:00

173 lines
5.3 KiB
C

#include "config.h"
#include "gskvulkanmaskopprivate.h"
#include "gskvulkanprivate.h"
#include "vulkan/resources/mask.vert.h"
typedef struct _GskVulkanMaskOp GskVulkanMaskOp;
struct _GskVulkanMaskOp
{
GskVulkanOp op;
struct {
GskVulkanImage *image;
graphene_rect_t rect;
graphene_rect_t tex_rect;
guint32 image_descriptor;
} source, mask;
GskMaskMode mask_mode;
gsize vertex_offset;
};
static void
gsk_vulkan_mask_op_finish (GskVulkanOp *op)
{
GskVulkanMaskOp *self = (GskVulkanMaskOp *) op;
g_object_unref (self->source.image);
g_object_unref (self->mask.image);
}
static void
gsk_vulkan_mask_op_print (GskVulkanOp *op,
GString *string,
guint indent)
{
GskVulkanMaskOp *self = (GskVulkanMaskOp *) op;
print_indent (string, indent);
print_rect (string, &self->source.rect);
g_string_append (string, "mask ");
print_rect (string, &self->mask.rect);
switch (self->mask_mode)
{
case GSK_MASK_MODE_ALPHA:
g_string_append (string, "alpha ");
break;
case GSK_MASK_MODE_INVERTED_ALPHA:
g_string_append (string, "inverted-alpha ");
break;
case GSK_MASK_MODE_LUMINANCE:
g_string_append (string, "luminance ");
break;
case GSK_MASK_MODE_INVERTED_LUMINANCE:
g_string_append (string, "inverted-luminance ");
break;
default:
g_assert_not_reached ();
break;
}
print_newline (string);
}
static void
gsk_vulkan_mask_op_upload (GskVulkanOp *op,
GskVulkanUploader *uploader)
{
}
static inline gsize
round_up (gsize number, gsize divisor)
{
return (number + divisor - 1) / divisor * divisor;
}
static gsize
gsk_vulkan_mask_op_count_vertex_data (GskVulkanOp *op,
gsize n_bytes)
{
GskVulkanMaskOp *self = (GskVulkanMaskOp *) op;
gsize vertex_stride;
vertex_stride = gsk_vulkan_mask_info.pVertexBindingDescriptions[0].stride;
n_bytes = round_up (n_bytes, vertex_stride);
self->vertex_offset = n_bytes;
n_bytes += vertex_stride;
return n_bytes;
}
static void
gsk_vulkan_mask_op_collect_vertex_data (GskVulkanOp *op,
guchar *data)
{
GskVulkanMaskOp *self = (GskVulkanMaskOp *) op;
GskVulkanMaskInstance *instance = (GskVulkanMaskInstance *) (data + self->vertex_offset);
gsk_vulkan_rect_to_float (&self->source.rect, instance->source_rect);
gsk_vulkan_rect_to_float (&self->source.tex_rect, instance->source_tex_rect);
instance->source_id = self->source.image_descriptor;
gsk_vulkan_rect_to_float (&self->mask.rect, instance->mask_rect);
gsk_vulkan_rect_to_float (&self->mask.tex_rect, instance->mask_tex_rect);
instance->mask_id = self->mask.image_descriptor;
instance->mask_mode = self->mask_mode;
}
static void
gsk_vulkan_mask_op_reserve_descriptor_sets (GskVulkanOp *op,
GskVulkanRender *render)
{
GskVulkanMaskOp *self = (GskVulkanMaskOp *) op;
self->source.image_descriptor = gsk_vulkan_render_get_image_descriptor (render, self->source.image, GSK_VULKAN_SAMPLER_DEFAULT);
self->mask.image_descriptor = gsk_vulkan_render_get_image_descriptor (render, self->mask.image, GSK_VULKAN_SAMPLER_DEFAULT);
}
static GskVulkanOp *
gsk_vulkan_mask_op_command (GskVulkanOp *op,
GskVulkanRender *render,
VkPipelineLayout pipeline_layout,
VkCommandBuffer command_buffer)
{
GskVulkanMaskOp *self = (GskVulkanMaskOp *) op;
vkCmdDraw (command_buffer,
6, 1,
0, self->vertex_offset / gsk_vulkan_mask_info.pVertexBindingDescriptions[0].stride);
return op->next;
}
static const GskVulkanOpClass GSK_VULKAN_COLOR_MASK_OP_CLASS = {
GSK_VULKAN_OP_SIZE (GskVulkanMaskOp),
GSK_VULKAN_STAGE_COMMAND,
"mask",
&gsk_vulkan_mask_info,
gsk_vulkan_mask_op_finish,
gsk_vulkan_mask_op_print,
gsk_vulkan_mask_op_upload,
gsk_vulkan_mask_op_count_vertex_data,
gsk_vulkan_mask_op_collect_vertex_data,
gsk_vulkan_mask_op_reserve_descriptor_sets,
gsk_vulkan_mask_op_command
};
void
gsk_vulkan_mask_op (GskVulkanRenderPass *render_pass,
const char *clip_type,
const graphene_point_t *offset,
GskVulkanImage *source,
const graphene_rect_t *source_rect,
const graphene_rect_t *source_tex_rect,
GskVulkanImage *mask,
const graphene_rect_t *mask_rect,
const graphene_rect_t *mask_tex_rect,
GskMaskMode mask_mode)
{
GskVulkanMaskOp *self;
self = (GskVulkanMaskOp *) gsk_vulkan_op_alloc (render_pass, &GSK_VULKAN_COLOR_MASK_OP_CLASS);
((GskVulkanOp *) self)->clip_type = g_intern_string (clip_type);
self->source.image = g_object_ref (source);
graphene_rect_offset_r (source_rect, offset->x, offset->y, &self->source.rect);
gsk_vulkan_normalize_tex_coords (&self->source.tex_rect, source_rect, source_tex_rect);
self->mask.image = g_object_ref (mask);
graphene_rect_offset_r (mask_rect, offset->x, offset->y, &self->mask.rect);
gsk_vulkan_normalize_tex_coords (&self->mask.tex_rect, mask_rect, mask_tex_rect);
self->mask_mode = mask_mode;
}