2023-08-12 20:10:16 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "gskgpuuploadopprivate.h"
|
|
|
|
|
|
|
|
#include "gskgpuframeprivate.h"
|
|
|
|
#include "gskgpuimageprivate.h"
|
|
|
|
#include "gskgpuprintprivate.h"
|
|
|
|
#include "gskgldeviceprivate.h"
|
|
|
|
#include "gskglimageprivate.h"
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
#include "gskvulkanbufferprivate.h"
|
|
|
|
#include "gskvulkanimageprivate.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "gdk/gdkglcontextprivate.h"
|
|
|
|
#include "gsk/gskrendernodeprivate.h"
|
|
|
|
|
|
|
|
static GskGpuOp *
|
|
|
|
gsk_gpu_upload_op_gl_command_with_area (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskGpuImage *image,
|
|
|
|
const cairo_rectangle_int_t *area,
|
|
|
|
void (* draw_func) (GskGpuOp *, guchar *, gsize))
|
|
|
|
{
|
|
|
|
GskGLImage *gl_image = GSK_GL_IMAGE (image);
|
|
|
|
GdkMemoryFormat format;
|
|
|
|
GdkGLContext *context;
|
|
|
|
gsize stride, bpp;
|
|
|
|
guchar *data;
|
|
|
|
guint gl_format, gl_type;
|
|
|
|
|
|
|
|
context = GDK_GL_CONTEXT (gsk_gpu_frame_get_context (frame));
|
|
|
|
format = gsk_gpu_image_get_format (image);
|
|
|
|
bpp = gdk_memory_format_bytes_per_pixel (format);
|
|
|
|
stride = area->width * bpp;
|
|
|
|
data = g_malloc (area->height * stride);
|
|
|
|
|
|
|
|
draw_func (op, data, stride);
|
|
|
|
|
|
|
|
gl_format = gsk_gl_image_get_gl_format (gl_image);
|
|
|
|
gl_type = gsk_gl_image_get_gl_type (gl_image);
|
|
|
|
|
|
|
|
glActiveTexture (GL_TEXTURE0);
|
|
|
|
gsk_gl_image_bind_texture (gl_image);
|
|
|
|
|
|
|
|
glPixelStorei (GL_UNPACK_ALIGNMENT, gdk_memory_format_alignment (format));
|
|
|
|
|
|
|
|
/* GL_UNPACK_ROW_LENGTH is available on desktop GL, OpenGL ES >= 3.0, or if
|
|
|
|
* the GL_EXT_unpack_subimage extension for OpenGL ES 2.0 is available
|
|
|
|
*/
|
|
|
|
if (stride == gsk_gpu_image_get_width (image) * bpp)
|
|
|
|
{
|
|
|
|
glTexSubImage2D (GL_TEXTURE_2D, 0, area->x, area->y, area->width, area->height, gl_format, gl_type, data);
|
|
|
|
}
|
|
|
|
else if (stride % bpp == 0 && gdk_gl_context_has_unpack_subimage (context))
|
|
|
|
{
|
|
|
|
glPixelStorei (GL_UNPACK_ROW_LENGTH, stride / bpp);
|
|
|
|
|
|
|
|
glTexSubImage2D (GL_TEXTURE_2D, 0, area->x, area->y, area->width, area->height, gl_format, gl_type, data);
|
|
|
|
|
|
|
|
glPixelStorei (GL_UNPACK_ROW_LENGTH, 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gsize i;
|
|
|
|
for (i = 0; i < area->height; i++)
|
|
|
|
glTexSubImage2D (GL_TEXTURE_2D, 0, area->x, area->y + i, area->width, 1, gl_format, gl_type, data + (i * stride));
|
|
|
|
}
|
|
|
|
|
|
|
|
glPixelStorei (GL_UNPACK_ALIGNMENT, 4);
|
|
|
|
|
|
|
|
g_free (data);
|
|
|
|
|
|
|
|
return op->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GskGpuOp *
|
|
|
|
gsk_gpu_upload_op_gl_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskGpuImage *image,
|
|
|
|
void (* draw_func) (GskGpuOp *, guchar *, gsize))
|
|
|
|
{
|
|
|
|
return gsk_gpu_upload_op_gl_command_with_area (op,
|
|
|
|
frame,
|
|
|
|
image,
|
|
|
|
&(cairo_rectangle_int_t) {
|
|
|
|
0, 0,
|
|
|
|
gsk_gpu_image_get_width (image),
|
|
|
|
gsk_gpu_image_get_height (image)
|
|
|
|
},
|
|
|
|
draw_func);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
static GskGpuOp *
|
|
|
|
gsk_gpu_upload_op_vk_command_with_area (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
VkCommandBuffer command_buffer,
|
|
|
|
GskVulkanImage *image,
|
|
|
|
const cairo_rectangle_int_t *area,
|
|
|
|
void (* draw_func) (GskGpuOp *, guchar *, gsize),
|
2023-08-21 00:18:37 +00:00
|
|
|
GskGpuBuffer **buffer)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
gsize stride;
|
|
|
|
guchar *data;
|
|
|
|
|
|
|
|
stride = area->width * gdk_memory_format_bytes_per_pixel (gsk_gpu_image_get_format (GSK_GPU_IMAGE (image)));
|
2023-08-21 00:18:37 +00:00
|
|
|
*buffer = gsk_vulkan_buffer_new_write (GSK_VULKAN_DEVICE (gsk_gpu_frame_get_device (frame)),
|
|
|
|
area->height * stride);
|
|
|
|
data = gsk_gpu_buffer_map (*buffer);
|
2023-08-12 20:10:16 +00:00
|
|
|
|
|
|
|
draw_func (op, data, stride);
|
2023-08-21 00:18:37 +00:00
|
|
|
|
|
|
|
gsk_gpu_buffer_unmap (*buffer);
|
2023-08-12 20:10:16 +00:00
|
|
|
|
|
|
|
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,
|
2023-08-21 00:18:37 +00:00
|
|
|
.buffer = gsk_vulkan_buffer_get_vk_buffer (GSK_VULKAN_BUFFER (*buffer)),
|
2023-08-12 20:10:16 +00:00
|
|
|
.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,
|
2023-08-21 00:18:37 +00:00
|
|
|
gsk_vulkan_buffer_get_vk_buffer (GSK_VULKAN_BUFFER (*buffer)),
|
2023-08-12 20:10:16 +00:00
|
|
|
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 GskGpuOp *
|
|
|
|
gsk_gpu_upload_op_vk_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
VkCommandBuffer command_buffer,
|
|
|
|
GskVulkanImage *image,
|
|
|
|
void (* draw_func) (GskGpuOp *, guchar *, gsize),
|
2023-08-21 00:18:37 +00:00
|
|
|
GskGpuBuffer **buffer)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
gsize stride;
|
|
|
|
guchar *data;
|
|
|
|
|
|
|
|
data = gsk_vulkan_image_get_data (image, &stride);
|
|
|
|
if (data)
|
|
|
|
{
|
|
|
|
draw_func (op, data, stride);
|
|
|
|
|
|
|
|
*buffer = NULL;
|
|
|
|
|
|
|
|
return op->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_vk_command_with_area (op,
|
|
|
|
frame,
|
|
|
|
command_buffer,
|
|
|
|
image,
|
|
|
|
&(cairo_rectangle_int_t) {
|
|
|
|
0, 0,
|
|
|
|
gsk_gpu_image_get_width (GSK_GPU_IMAGE (image)),
|
|
|
|
gsk_gpu_image_get_height (GSK_GPU_IMAGE (image)),
|
|
|
|
},
|
|
|
|
draw_func,
|
|
|
|
buffer);
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef struct _GskGpuUploadCairoOp GskGpuUploadCairoOp;
|
|
|
|
|
|
|
|
struct _GskGpuUploadCairoOp
|
|
|
|
{
|
|
|
|
GskGpuOp op;
|
|
|
|
|
|
|
|
GskGpuImage *image;
|
|
|
|
GskRenderNode *node;
|
|
|
|
graphene_rect_t viewport;
|
|
|
|
|
2023-08-21 00:18:37 +00:00
|
|
|
GskGpuBuffer *buffer;
|
2023-08-12 20:10:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_cairo_op_finish (GskGpuOp *op)
|
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self = (GskGpuUploadCairoOp *) op;
|
|
|
|
|
|
|
|
g_object_unref (self->image);
|
|
|
|
gsk_render_node_unref (self->node);
|
2023-08-21 00:18:37 +00:00
|
|
|
g_clear_object (&self->buffer);
|
2023-08-12 20:10:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2023-08-21 00:18:37 +00:00
|
|
|
gsk_gpu_upload_cairo_op_print (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GString *string,
|
|
|
|
guint indent)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self = (GskGpuUploadCairoOp *) op;
|
|
|
|
|
|
|
|
gsk_gpu_print_op (string, indent, "upload-cairo");
|
|
|
|
gsk_gpu_print_image (string, self->image);
|
|
|
|
gsk_gpu_print_newline (string);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_cairo_op_draw (GskGpuOp *op,
|
|
|
|
guchar *data,
|
|
|
|
gsize stride)
|
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self = (GskGpuUploadCairoOp *) op;
|
|
|
|
cairo_surface_t *surface;
|
|
|
|
cairo_t *cr;
|
|
|
|
int width, height;
|
|
|
|
|
|
|
|
width = gsk_gpu_image_get_width (self->image);
|
|
|
|
height = gsk_gpu_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_fallback (self->node, cr);
|
|
|
|
|
|
|
|
cairo_destroy (cr);
|
|
|
|
|
|
|
|
cairo_surface_finish (surface);
|
|
|
|
cairo_surface_destroy (surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
static GskGpuOp *
|
2023-08-21 00:18:37 +00:00
|
|
|
gsk_gpu_upload_cairo_op_vk_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
VkRenderPass render_pass,
|
|
|
|
VkFormat format,
|
|
|
|
VkCommandBuffer command_buffer)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self = (GskGpuUploadCairoOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_vk_command (op,
|
|
|
|
frame,
|
|
|
|
command_buffer,
|
|
|
|
GSK_VULKAN_IMAGE (self->image),
|
|
|
|
gsk_gpu_upload_cairo_op_draw,
|
|
|
|
&self->buffer);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static GskGpuOp *
|
|
|
|
gsk_gpu_upload_cairo_op_gl_command (GskGpuOp *op,
|
2023-08-29 06:34:55 +00:00
|
|
|
GskGpuFrame *frame,
|
|
|
|
gsize flip_y)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self = (GskGpuUploadCairoOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_gl_command (op,
|
|
|
|
frame,
|
|
|
|
self->image,
|
|
|
|
gsk_gpu_upload_cairo_op_draw);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const GskGpuOpClass GSK_GPU_UPLOAD_CAIRO_OP_CLASS = {
|
|
|
|
GSK_GPU_OP_SIZE (GskGpuUploadCairoOp),
|
|
|
|
GSK_GPU_STAGE_UPLOAD,
|
|
|
|
gsk_gpu_upload_cairo_op_finish,
|
|
|
|
gsk_gpu_upload_cairo_op_print,
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
gsk_gpu_upload_cairo_op_vk_command,
|
|
|
|
#endif
|
|
|
|
gsk_gpu_upload_cairo_op_gl_command
|
|
|
|
};
|
|
|
|
|
|
|
|
GskGpuImage *
|
|
|
|
gsk_gpu_upload_cairo_op (GskGpuFrame *frame,
|
|
|
|
GskRenderNode *node,
|
|
|
|
const graphene_vec2_t *scale,
|
|
|
|
const graphene_rect_t *viewport)
|
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self;
|
|
|
|
|
|
|
|
self = (GskGpuUploadCairoOp *) gsk_gpu_op_alloc (frame, &GSK_GPU_UPLOAD_CAIRO_OP_CLASS);
|
|
|
|
|
|
|
|
self->node = gsk_render_node_ref (node);
|
|
|
|
self->image = gsk_gpu_device_create_upload_image (gsk_gpu_frame_get_device (frame),
|
|
|
|
GDK_MEMORY_DEFAULT,
|
|
|
|
ceil (graphene_vec2_get_x (scale) * viewport->size.width),
|
|
|
|
ceil (graphene_vec2_get_y (scale) * viewport->size.height));
|
|
|
|
/* g_assert (gsk_gpu_image_get_postprocess (self->image) == 0); */
|
|
|
|
self->viewport = *viewport;
|
|
|
|
|
|
|
|
return self->image;
|
|
|
|
}
|
|
|
|
|