mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-10 12:40:10 +00:00
e00e6f3d5d
The memory isn't guaranteed to be all zeroes and we don't want glitches.
479 lines
17 KiB
C
479 lines
17 KiB
C
#include "config.h"
|
|
|
|
#include "gskvulkanuploadopprivate.h"
|
|
|
|
#include "gskvulkanprivate.h"
|
|
|
|
#include "gdk/gdkmemoryformatprivate.h"
|
|
|
|
static gsize
|
|
gsk_vulkan_upload_op_count_vertex_data (GskVulkanOp *op,
|
|
gsize n_bytes)
|
|
{
|
|
return n_bytes;
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_op_collect_vertex_data (GskVulkanOp *op,
|
|
guchar *data)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_op_reserve_descriptor_sets (GskVulkanOp *op,
|
|
GskVulkanRender *render)
|
|
{
|
|
}
|
|
|
|
static GskVulkanOp *
|
|
gsk_vulkan_upload_op_command_with_area (GskVulkanOp *op,
|
|
GskVulkanRender *render,
|
|
VkCommandBuffer command_buffer,
|
|
GskVulkanImage *image,
|
|
const cairo_rectangle_int_t *area,
|
|
void (* draw_func) (GskVulkanOp *, guchar *, gsize),
|
|
GskVulkanBuffer **buffer)
|
|
{
|
|
gsize stride;
|
|
guchar *data;
|
|
|
|
stride = area->width * gdk_memory_format_bytes_per_pixel (gsk_vulkan_image_get_format (image));
|
|
*buffer = gsk_vulkan_buffer_new_map (gsk_vulkan_render_get_context (render),
|
|
area->height * stride,
|
|
GSK_VULKAN_WRITE);
|
|
data = gsk_vulkan_buffer_map (*buffer);
|
|
|
|
draw_func (op, data, stride);
|
|
|
|
gsk_vulkan_buffer_unmap (*buffer);
|
|
|
|
vkCmdPipelineBarrier (command_buffer,
|
|
VK_PIPELINE_STAGE_HOST_BIT,
|
|
VK_PIPELINE_STAGE_TRANSFER_BIT,
|
|
0,
|
|
0, NULL,
|
|
1, &(VkBufferMemoryBarrier) {
|
|
.sType = VK_STRUCTURE_TYPE_BUFFER_MEMORY_BARRIER,
|
|
.srcAccessMask = VK_ACCESS_HOST_WRITE_BIT,
|
|
.dstAccessMask = VK_ACCESS_TRANSFER_READ_BIT,
|
|
.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
|
|
.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
|
|
.buffer = gsk_vulkan_buffer_get_buffer (*buffer),
|
|
.offset = 0,
|
|
.size = VK_WHOLE_SIZE,
|
|
},
|
|
0, NULL);
|
|
gsk_vulkan_image_transition (image,
|
|
command_buffer,
|
|
VK_PIPELINE_STAGE_TRANSFER_BIT,
|
|
VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
|
|
VK_ACCESS_TRANSFER_WRITE_BIT);
|
|
|
|
vkCmdCopyBufferToImage (command_buffer,
|
|
gsk_vulkan_buffer_get_buffer (*buffer),
|
|
gsk_vulkan_image_get_vk_image (image),
|
|
VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
|
|
1,
|
|
(VkBufferImageCopy[1]) {
|
|
{
|
|
.bufferOffset = 0,
|
|
.bufferRowLength = area->width,
|
|
.bufferImageHeight = area->height,
|
|
.imageSubresource = {
|
|
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
|
|
.mipLevel = 0,
|
|
.baseArrayLayer = 0,
|
|
.layerCount = 1
|
|
},
|
|
.imageOffset = {
|
|
.x = area->x,
|
|
.y = area->y,
|
|
.z = 0
|
|
},
|
|
.imageExtent = {
|
|
.width = area->width,
|
|
.height = area->height,
|
|
.depth = 1
|
|
}
|
|
}
|
|
});
|
|
|
|
return op->next;
|
|
}
|
|
|
|
static GskVulkanOp *
|
|
gsk_vulkan_upload_op_command (GskVulkanOp *op,
|
|
GskVulkanRender *render,
|
|
VkCommandBuffer command_buffer,
|
|
GskVulkanImage *image,
|
|
void (* draw_func) (GskVulkanOp *, guchar *, gsize),
|
|
GskVulkanBuffer **buffer)
|
|
{
|
|
gsize stride;
|
|
guchar *data;
|
|
|
|
data = gsk_vulkan_image_try_map (image, &stride);
|
|
if (data)
|
|
{
|
|
draw_func (op, data, stride);
|
|
|
|
gsk_vulkan_image_unmap (image);
|
|
*buffer = NULL;
|
|
|
|
return op->next;
|
|
}
|
|
|
|
return gsk_vulkan_upload_op_command_with_area (op,
|
|
render,
|
|
command_buffer,
|
|
image,
|
|
&(cairo_rectangle_int_t) {
|
|
0, 0,
|
|
gsk_vulkan_image_get_width (image),
|
|
gsk_vulkan_image_get_height (image),
|
|
},
|
|
draw_func,
|
|
buffer);
|
|
|
|
}
|
|
|
|
typedef struct _GskVulkanUploadTextureOp GskVulkanUploadTextureOp;
|
|
|
|
struct _GskVulkanUploadTextureOp
|
|
{
|
|
GskVulkanOp op;
|
|
|
|
GskVulkanImage *image;
|
|
GskVulkanBuffer *buffer;
|
|
GdkTexture *texture;
|
|
};
|
|
|
|
static void
|
|
gsk_vulkan_upload_texture_op_finish (GskVulkanOp *op)
|
|
{
|
|
GskVulkanUploadTextureOp *self = (GskVulkanUploadTextureOp *) op;
|
|
|
|
g_object_unref (self->image);
|
|
g_clear_pointer (&self->buffer, gsk_vulkan_buffer_free);
|
|
g_object_unref (self->texture);
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_texture_op_print (GskVulkanOp *op,
|
|
GString *string,
|
|
guint indent)
|
|
{
|
|
GskVulkanUploadTextureOp *self = (GskVulkanUploadTextureOp *) op;
|
|
|
|
print_indent (string, indent);
|
|
g_string_append (string, "upload-texture ");
|
|
print_image (string, self->image);
|
|
print_newline (string);
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_texture_op_draw (GskVulkanOp *op,
|
|
guchar *data,
|
|
gsize stride)
|
|
{
|
|
GskVulkanUploadTextureOp *self = (GskVulkanUploadTextureOp *) op;
|
|
GdkTextureDownloader *downloader;
|
|
|
|
downloader = gdk_texture_downloader_new (self->texture);
|
|
gdk_texture_downloader_set_format (downloader, gsk_vulkan_image_get_format (self->image));
|
|
gdk_texture_downloader_download_into (downloader, data, stride);
|
|
gdk_texture_downloader_free (downloader);
|
|
}
|
|
|
|
static GskVulkanOp *
|
|
gsk_vulkan_upload_texture_op_command (GskVulkanOp *op,
|
|
GskVulkanRender *render,
|
|
VkRenderPass render_pass,
|
|
VkCommandBuffer command_buffer)
|
|
{
|
|
GskVulkanUploadTextureOp *self = (GskVulkanUploadTextureOp *) op;
|
|
|
|
return gsk_vulkan_upload_op_command (op,
|
|
render,
|
|
command_buffer,
|
|
self->image,
|
|
gsk_vulkan_upload_texture_op_draw,
|
|
&self->buffer);
|
|
}
|
|
|
|
static const GskVulkanOpClass GSK_VULKAN_UPLOAD_TEXTURE_OP_CLASS = {
|
|
GSK_VULKAN_OP_SIZE (GskVulkanUploadTextureOp),
|
|
GSK_VULKAN_STAGE_UPLOAD,
|
|
gsk_vulkan_upload_texture_op_finish,
|
|
gsk_vulkan_upload_texture_op_print,
|
|
gsk_vulkan_upload_op_count_vertex_data,
|
|
gsk_vulkan_upload_op_collect_vertex_data,
|
|
gsk_vulkan_upload_op_reserve_descriptor_sets,
|
|
gsk_vulkan_upload_texture_op_command
|
|
};
|
|
|
|
GskVulkanImage *
|
|
gsk_vulkan_upload_texture_op (GskVulkanRender *render,
|
|
GdkTexture *texture)
|
|
{
|
|
GskVulkanUploadTextureOp *self;
|
|
|
|
self = (GskVulkanUploadTextureOp *) gsk_vulkan_op_alloc (render, &GSK_VULKAN_UPLOAD_TEXTURE_OP_CLASS);
|
|
|
|
self->texture = g_object_ref (texture);
|
|
self->image = gsk_vulkan_image_new_for_upload (gsk_vulkan_render_get_context (render),
|
|
gdk_texture_get_format (texture),
|
|
gdk_texture_get_width (texture),
|
|
gdk_texture_get_height (texture));
|
|
|
|
return self->image;
|
|
}
|
|
|
|
typedef struct _GskVulkanUploadCairoOp GskVulkanUploadCairoOp;
|
|
|
|
struct _GskVulkanUploadCairoOp
|
|
{
|
|
GskVulkanOp op;
|
|
|
|
GskVulkanImage *image;
|
|
GskRenderNode *node;
|
|
graphene_rect_t viewport;
|
|
|
|
GskVulkanBuffer *buffer;
|
|
};
|
|
|
|
static void
|
|
gsk_vulkan_upload_cairo_op_finish (GskVulkanOp *op)
|
|
{
|
|
GskVulkanUploadCairoOp *self = (GskVulkanUploadCairoOp *) op;
|
|
|
|
g_object_unref (self->image);
|
|
gsk_render_node_unref (self->node);
|
|
|
|
g_clear_pointer (&self->buffer, gsk_vulkan_buffer_free);
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_cairo_op_print (GskVulkanOp *op,
|
|
GString *string,
|
|
guint indent)
|
|
{
|
|
GskVulkanUploadCairoOp *self = (GskVulkanUploadCairoOp *) op;
|
|
|
|
print_indent (string, indent);
|
|
g_string_append (string, "upload-cairo ");
|
|
print_image (string, self->image);
|
|
print_newline (string);
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_cairo_op_draw (GskVulkanOp *op,
|
|
guchar *data,
|
|
gsize stride)
|
|
{
|
|
GskVulkanUploadCairoOp *self = (GskVulkanUploadCairoOp *) op;
|
|
cairo_surface_t *surface;
|
|
cairo_t *cr;
|
|
int width, height;
|
|
|
|
width = gsk_vulkan_image_get_width (self->image);
|
|
height = gsk_vulkan_image_get_height (self->image);
|
|
|
|
surface = cairo_image_surface_create_for_data (data,
|
|
CAIRO_FORMAT_ARGB32,
|
|
width, height,
|
|
stride);
|
|
cairo_surface_set_device_scale (surface,
|
|
width / self->viewport.size.width,
|
|
height / self->viewport.size.height);
|
|
cr = cairo_create (surface);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
|
|
cairo_paint (cr);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
|
|
cairo_translate (cr, -self->viewport.origin.x, -self->viewport.origin.y);
|
|
|
|
gsk_render_node_draw (self->node, cr);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
cairo_surface_finish (surface);
|
|
cairo_surface_destroy (surface);
|
|
}
|
|
|
|
static GskVulkanOp *
|
|
gsk_vulkan_upload_cairo_op_command (GskVulkanOp *op,
|
|
GskVulkanRender *render,
|
|
VkRenderPass render_pass,
|
|
VkCommandBuffer command_buffer)
|
|
{
|
|
GskVulkanUploadCairoOp *self = (GskVulkanUploadCairoOp *) op;
|
|
|
|
return gsk_vulkan_upload_op_command (op,
|
|
render,
|
|
command_buffer,
|
|
self->image,
|
|
gsk_vulkan_upload_cairo_op_draw,
|
|
&self->buffer);
|
|
}
|
|
|
|
static const GskVulkanOpClass GSK_VULKAN_UPLOAD_CAIRO_OP_CLASS = {
|
|
GSK_VULKAN_OP_SIZE (GskVulkanUploadCairoOp),
|
|
GSK_VULKAN_STAGE_UPLOAD,
|
|
gsk_vulkan_upload_cairo_op_finish,
|
|
gsk_vulkan_upload_cairo_op_print,
|
|
gsk_vulkan_upload_op_count_vertex_data,
|
|
gsk_vulkan_upload_op_collect_vertex_data,
|
|
gsk_vulkan_upload_op_reserve_descriptor_sets,
|
|
gsk_vulkan_upload_cairo_op_command
|
|
};
|
|
|
|
GskVulkanImage *
|
|
gsk_vulkan_upload_cairo_op (GskVulkanRender *render,
|
|
GskRenderNode *node,
|
|
const graphene_vec2_t *scale,
|
|
const graphene_rect_t *viewport)
|
|
{
|
|
GskVulkanUploadCairoOp *self;
|
|
|
|
self = (GskVulkanUploadCairoOp *) gsk_vulkan_op_alloc (render, &GSK_VULKAN_UPLOAD_CAIRO_OP_CLASS);
|
|
|
|
self->node = gsk_render_node_ref (node);
|
|
self->image = gsk_vulkan_image_new_for_upload (gsk_vulkan_render_get_context (render),
|
|
GDK_MEMORY_DEFAULT,
|
|
ceil (graphene_vec2_get_x (scale) * viewport->size.width),
|
|
ceil (graphene_vec2_get_y (scale) * viewport->size.height));
|
|
g_assert (gsk_vulkan_image_get_postprocess (self->image) == 0);
|
|
self->viewport = *viewport;
|
|
|
|
return self->image;
|
|
}
|
|
|
|
typedef struct _GskVulkanUploadGlyphOp GskVulkanUploadGlyphOp;
|
|
|
|
struct _GskVulkanUploadGlyphOp
|
|
{
|
|
GskVulkanOp op;
|
|
|
|
GskVulkanImage *image;
|
|
cairo_rectangle_int_t area;
|
|
PangoFont *font;
|
|
PangoGlyphInfo glyph_info;
|
|
float scale;
|
|
|
|
GskVulkanBuffer *buffer;
|
|
};
|
|
|
|
static void
|
|
gsk_vulkan_upload_glyph_op_finish (GskVulkanOp *op)
|
|
{
|
|
GskVulkanUploadGlyphOp *self = (GskVulkanUploadGlyphOp *) op;
|
|
|
|
g_object_unref (self->image);
|
|
g_object_unref (self->font);
|
|
|
|
g_clear_pointer (&self->buffer, gsk_vulkan_buffer_free);
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_glyph_op_print (GskVulkanOp *op,
|
|
GString *string,
|
|
guint indent)
|
|
{
|
|
GskVulkanUploadGlyphOp *self = (GskVulkanUploadGlyphOp *) op;
|
|
|
|
print_indent (string, indent);
|
|
g_string_append (string, "upload-glyph ");
|
|
print_int_rect (string, &self->area);
|
|
g_string_append_printf (string, "glyph %u @ %g ", self->glyph_info.glyph, self->scale);
|
|
print_newline (string);
|
|
}
|
|
|
|
static void
|
|
gsk_vulkan_upload_glyph_op_draw (GskVulkanOp *op,
|
|
guchar *data,
|
|
gsize stride)
|
|
{
|
|
GskVulkanUploadGlyphOp *self = (GskVulkanUploadGlyphOp *) op;
|
|
cairo_surface_t *surface;
|
|
cairo_t *cr;
|
|
|
|
surface = cairo_image_surface_create_for_data (data,
|
|
CAIRO_FORMAT_ARGB32,
|
|
self->area.width,
|
|
self->area.height,
|
|
stride);
|
|
cairo_surface_set_device_scale (surface, self->scale, self->scale);
|
|
|
|
cr = cairo_create (surface);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
|
|
cairo_paint (cr);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
|
|
|
|
/* Make sure the entire surface is initialized to black */
|
|
cairo_set_source_rgba (cr, 0, 0, 0, 0);
|
|
cairo_rectangle (cr, 0.0, 0.0, self->area.width, self->area.height);
|
|
cairo_fill (cr);
|
|
|
|
/* Draw glyph */
|
|
cairo_set_source_rgba (cr, 1, 1, 1, 1);
|
|
|
|
pango_cairo_show_glyph_string (cr,
|
|
self->font,
|
|
&(PangoGlyphString) {
|
|
.num_glyphs = 1,
|
|
.glyphs = &self->glyph_info
|
|
});
|
|
|
|
cairo_destroy (cr);
|
|
|
|
cairo_surface_finish (surface);
|
|
cairo_surface_destroy (surface);
|
|
}
|
|
|
|
static GskVulkanOp *
|
|
gsk_vulkan_upload_glyph_op_command (GskVulkanOp *op,
|
|
GskVulkanRender *render,
|
|
VkRenderPass render_pass,
|
|
VkCommandBuffer command_buffer)
|
|
{
|
|
GskVulkanUploadGlyphOp *self = (GskVulkanUploadGlyphOp *) op;
|
|
|
|
return gsk_vulkan_upload_op_command_with_area (op,
|
|
render,
|
|
command_buffer,
|
|
self->image,
|
|
&self->area,
|
|
gsk_vulkan_upload_glyph_op_draw,
|
|
&self->buffer);
|
|
}
|
|
|
|
static const GskVulkanOpClass GSK_VULKAN_UPLOAD_GLYPH_OP_CLASS = {
|
|
GSK_VULKAN_OP_SIZE (GskVulkanUploadGlyphOp),
|
|
GSK_VULKAN_STAGE_UPLOAD,
|
|
gsk_vulkan_upload_glyph_op_finish,
|
|
gsk_vulkan_upload_glyph_op_print,
|
|
gsk_vulkan_upload_op_count_vertex_data,
|
|
gsk_vulkan_upload_op_collect_vertex_data,
|
|
gsk_vulkan_upload_op_reserve_descriptor_sets,
|
|
gsk_vulkan_upload_glyph_op_command
|
|
};
|
|
|
|
void
|
|
gsk_vulkan_upload_glyph_op (GskVulkanRender *render,
|
|
GskVulkanImage *image,
|
|
cairo_rectangle_int_t *area,
|
|
PangoFont *font,
|
|
PangoGlyphInfo *glyph_info,
|
|
float scale)
|
|
{
|
|
GskVulkanUploadGlyphOp *self;
|
|
|
|
self = (GskVulkanUploadGlyphOp *) gsk_vulkan_op_alloc (render, &GSK_VULKAN_UPLOAD_GLYPH_OP_CLASS);
|
|
|
|
self->image = g_object_ref (image);
|
|
self->area = *area;
|
|
self->font = g_object_ref (font);
|
|
self->glyph_info = *glyph_info;
|
|
self->scale = scale;
|
|
}
|