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
|
|
|
|
|
2024-06-26 16:17:56 +00:00
|
|
|
#include "gdk/gdkcolorstateprivate.h"
|
2023-08-12 20:10:16 +00:00
|
|
|
#include "gdk/gdkglcontextprivate.h"
|
2023-12-11 05:21:32 +00:00
|
|
|
#include "gsk/gskdebugprivate.h"
|
2023-08-12 20:10:16 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
gsize stride, bpp;
|
|
|
|
guchar *data;
|
|
|
|
guint gl_format, gl_type;
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
2024-09-26 15:39:31 +00:00
|
|
|
else if (stride % bpp == 0)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
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,
|
2023-11-04 02:57:38 +00:00
|
|
|
GskVulkanCommandState *state,
|
2023-08-12 20:10:16 +00:00
|
|
|
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);
|
2024-03-08 21:54:53 +00:00
|
|
|
|
|
|
|
gsk_gpu_buffer_unmap (*buffer, area->height * stride);
|
2023-08-12 20:10:16 +00:00
|
|
|
|
2023-11-04 02:57:38 +00:00
|
|
|
vkCmdPipelineBarrier (state->vk_command_buffer,
|
2023-08-12 20:10:16 +00:00
|
|
|
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,
|
2023-11-29 22:14:00 +00:00
|
|
|
state->semaphores,
|
2023-11-04 02:57:38 +00:00
|
|
|
state->vk_command_buffer,
|
2023-08-12 20:10:16 +00:00
|
|
|
VK_PIPELINE_STAGE_TRANSFER_BIT,
|
|
|
|
VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
|
|
|
|
VK_ACCESS_TRANSFER_WRITE_BIT);
|
|
|
|
|
2023-11-04 02:57:38 +00:00
|
|
|
vkCmdCopyBufferToImage (state->vk_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 *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_op_vk_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskVulkanCommandState *state,
|
|
|
|
GskVulkanImage *image,
|
|
|
|
void (* draw_func) (GskGpuOp *, guchar *, gsize),
|
|
|
|
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,
|
2023-11-04 02:57:38 +00:00
|
|
|
state,
|
2023-08-12 20:10:16 +00:00
|
|
|
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
|
|
|
|
|
2023-09-01 10:11:06 +00:00
|
|
|
typedef struct _GskGpuUploadTextureOp GskGpuUploadTextureOp;
|
|
|
|
|
|
|
|
struct _GskGpuUploadTextureOp
|
|
|
|
{
|
|
|
|
GskGpuOp op;
|
|
|
|
|
|
|
|
GskGpuImage *image;
|
|
|
|
GskGpuBuffer *buffer;
|
|
|
|
GdkTexture *texture;
|
2024-08-27 02:48:43 +00:00
|
|
|
guint lod_level;
|
memoryformat: Add linear/nearest choice for mipmaping
linear will average all the pixels for the lod, nearest will just pick
one (using the same method as OpenGL/Vulkan, picking bottom right
center).
This doesn't really make linear/nearest filtering work as it should
(because it's still a form of mipmaps), but it has 2 advantages:
1. it gets closer to the desired effect
2. it is a lot faster
Because only 1 pixel is chosen from the original image, instead of
averaging all pixels, a lot less memory needs to be accessed, and
because memory access is the bottleneck for large images, the speedup is
almost linear with the number of pixels not accessed.
And that means that even for lot level 3, aka 1/8th scale, only 1/64 of
the pixels need to be accessed, and everything is 50x faster.
Switching gtk4-demo --run=image_scaling to linear/nearest makes all the
lag go away for me, even with a 64k x 64k image.
2024-09-03 04:12:03 +00:00
|
|
|
GskScalingFilter lod_filter;
|
2023-09-01 10:11:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_texture_op_finish (GskGpuOp *op)
|
|
|
|
{
|
|
|
|
GskGpuUploadTextureOp *self = (GskGpuUploadTextureOp *) op;
|
|
|
|
|
|
|
|
g_object_unref (self->image);
|
|
|
|
g_clear_object (&self->buffer);
|
|
|
|
g_object_unref (self->texture);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_texture_op_print (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GString *string,
|
|
|
|
guint indent)
|
|
|
|
{
|
|
|
|
GskGpuUploadTextureOp *self = (GskGpuUploadTextureOp *) op;
|
|
|
|
|
|
|
|
gsk_gpu_print_op (string, indent, "upload-texture");
|
|
|
|
gsk_gpu_print_image (string, self->image);
|
2024-08-27 02:48:43 +00:00
|
|
|
if (self->lod_level > 0)
|
memoryformat: Add linear/nearest choice for mipmaping
linear will average all the pixels for the lod, nearest will just pick
one (using the same method as OpenGL/Vulkan, picking bottom right
center).
This doesn't really make linear/nearest filtering work as it should
(because it's still a form of mipmaps), but it has 2 advantages:
1. it gets closer to the desired effect
2. it is a lot faster
Because only 1 pixel is chosen from the original image, instead of
averaging all pixels, a lot less memory needs to be accessed, and
because memory access is the bottleneck for large images, the speedup is
almost linear with the number of pixels not accessed.
And that means that even for lot level 3, aka 1/8th scale, only 1/64 of
the pixels need to be accessed, and everything is 50x faster.
Switching gtk4-demo --run=image_scaling to linear/nearest makes all the
lag go away for me, even with a 64k x 64k image.
2024-09-03 04:12:03 +00:00
|
|
|
g_string_append_printf (string, " @%ux %s",
|
|
|
|
1 << self->lod_level,
|
|
|
|
self->lod_filter == GSK_SCALING_FILTER_TRILINEAR ? "linear" : "nearest");
|
2023-09-01 10:11:06 +00:00
|
|
|
gsk_gpu_print_newline (string);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_texture_op_draw (GskGpuOp *op,
|
|
|
|
guchar *data,
|
|
|
|
gsize stride)
|
|
|
|
{
|
|
|
|
GskGpuUploadTextureOp *self = (GskGpuUploadTextureOp *) op;
|
|
|
|
GdkTextureDownloader *downloader;
|
|
|
|
|
|
|
|
downloader = gdk_texture_downloader_new (self->texture);
|
2024-07-26 21:33:48 +00:00
|
|
|
gdk_texture_downloader_set_color_state (downloader, gdk_texture_get_color_state (self->texture));
|
2024-08-27 02:48:43 +00:00
|
|
|
if (self->lod_level == 0)
|
|
|
|
{
|
2024-08-27 03:59:54 +00:00
|
|
|
gdk_texture_downloader_set_format (downloader, gsk_gpu_image_get_format (self->image));
|
2024-08-27 02:48:43 +00:00
|
|
|
gdk_texture_downloader_download_into (downloader, data, stride);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GBytes *bytes;
|
|
|
|
gsize src_stride;
|
|
|
|
|
2024-08-27 03:59:54 +00:00
|
|
|
gdk_texture_downloader_set_format (downloader, gdk_texture_get_format (self->texture));
|
2024-08-27 02:48:43 +00:00
|
|
|
bytes = gdk_texture_downloader_download_bytes (downloader, &src_stride);
|
|
|
|
gdk_memory_mipmap (data,
|
|
|
|
stride,
|
|
|
|
gsk_gpu_image_get_format (self->image),
|
|
|
|
g_bytes_get_data (bytes, NULL),
|
|
|
|
src_stride,
|
2024-08-27 03:59:54 +00:00
|
|
|
gdk_texture_get_format (self->texture),
|
2024-08-27 02:48:43 +00:00
|
|
|
gdk_texture_get_width (self->texture),
|
|
|
|
gdk_texture_get_height (self->texture),
|
memoryformat: Add linear/nearest choice for mipmaping
linear will average all the pixels for the lod, nearest will just pick
one (using the same method as OpenGL/Vulkan, picking bottom right
center).
This doesn't really make linear/nearest filtering work as it should
(because it's still a form of mipmaps), but it has 2 advantages:
1. it gets closer to the desired effect
2. it is a lot faster
Because only 1 pixel is chosen from the original image, instead of
averaging all pixels, a lot less memory needs to be accessed, and
because memory access is the bottleneck for large images, the speedup is
almost linear with the number of pixels not accessed.
And that means that even for lot level 3, aka 1/8th scale, only 1/64 of
the pixels need to be accessed, and everything is 50x faster.
Switching gtk4-demo --run=image_scaling to linear/nearest makes all the
lag go away for me, even with a 64k x 64k image.
2024-09-03 04:12:03 +00:00
|
|
|
self->lod_level,
|
|
|
|
self->lod_filter == GSK_SCALING_FILTER_TRILINEAR ? TRUE : FALSE);
|
2024-08-27 02:48:43 +00:00
|
|
|
g_bytes_unref (bytes);
|
|
|
|
}
|
2023-09-01 10:11:06 +00:00
|
|
|
gdk_texture_downloader_free (downloader);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
static GskGpuOp *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_texture_op_vk_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskVulkanCommandState *state)
|
2023-09-01 10:11:06 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadTextureOp *self = (GskGpuUploadTextureOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_vk_command (op,
|
|
|
|
frame,
|
2023-11-04 02:57:38 +00:00
|
|
|
state,
|
2023-09-01 10:11:06 +00:00
|
|
|
GSK_VULKAN_IMAGE (self->image),
|
|
|
|
gsk_gpu_upload_texture_op_draw,
|
|
|
|
&self->buffer);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static GskGpuOp *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_texture_op_gl_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskGLCommandState *state)
|
2023-09-01 10:11:06 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadTextureOp *self = (GskGpuUploadTextureOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_gl_command (op,
|
|
|
|
frame,
|
|
|
|
self->image,
|
|
|
|
gsk_gpu_upload_texture_op_draw);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const GskGpuOpClass GSK_GPU_UPLOAD_TEXTURE_OP_CLASS = {
|
|
|
|
GSK_GPU_OP_SIZE (GskGpuUploadTextureOp),
|
|
|
|
GSK_GPU_STAGE_UPLOAD,
|
|
|
|
gsk_gpu_upload_texture_op_finish,
|
|
|
|
gsk_gpu_upload_texture_op_print,
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
gsk_gpu_upload_texture_op_vk_command,
|
|
|
|
#endif
|
|
|
|
gsk_gpu_upload_texture_op_gl_command
|
|
|
|
};
|
|
|
|
|
|
|
|
GskGpuImage *
|
memoryformat: Add linear/nearest choice for mipmaping
linear will average all the pixels for the lod, nearest will just pick
one (using the same method as OpenGL/Vulkan, picking bottom right
center).
This doesn't really make linear/nearest filtering work as it should
(because it's still a form of mipmaps), but it has 2 advantages:
1. it gets closer to the desired effect
2. it is a lot faster
Because only 1 pixel is chosen from the original image, instead of
averaging all pixels, a lot less memory needs to be accessed, and
because memory access is the bottleneck for large images, the speedup is
almost linear with the number of pixels not accessed.
And that means that even for lot level 3, aka 1/8th scale, only 1/64 of
the pixels need to be accessed, and everything is 50x faster.
Switching gtk4-demo --run=image_scaling to linear/nearest makes all the
lag go away for me, even with a 64k x 64k image.
2024-09-03 04:12:03 +00:00
|
|
|
gsk_gpu_upload_texture_op_try (GskGpuFrame *frame,
|
|
|
|
gboolean with_mipmap,
|
|
|
|
guint lod_level,
|
|
|
|
GskScalingFilter lod_filter,
|
|
|
|
GdkTexture *texture)
|
2023-09-01 10:11:06 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadTextureOp *self;
|
2023-10-12 20:59:09 +00:00
|
|
|
GskGpuImage *image;
|
2024-07-06 11:29:09 +00:00
|
|
|
GdkMemoryFormat format;
|
|
|
|
|
|
|
|
format = gdk_texture_get_format (texture);
|
2023-10-12 20:59:09 +00:00
|
|
|
|
|
|
|
image = gsk_gpu_device_create_upload_image (gsk_gpu_frame_get_device (frame),
|
2023-11-01 05:43:15 +00:00
|
|
|
with_mipmap,
|
2024-07-06 11:29:09 +00:00
|
|
|
format,
|
|
|
|
gdk_memory_format_alpha (format) != GDK_MEMORY_ALPHA_PREMULTIPLIED &&
|
2024-06-26 16:17:56 +00:00
|
|
|
gdk_color_state_get_no_srgb_tf (gdk_texture_get_color_state (texture)) != NULL,
|
2024-08-27 02:48:43 +00:00
|
|
|
(gdk_texture_get_width (texture) + (1 << lod_level) - 1) >> lod_level,
|
|
|
|
(gdk_texture_get_height (texture) + (1 << lod_level) - 1) >> lod_level);
|
2023-10-12 20:59:09 +00:00
|
|
|
if (image == NULL)
|
|
|
|
return NULL;
|
2023-09-01 10:11:06 +00:00
|
|
|
|
2023-12-11 05:21:32 +00:00
|
|
|
if (GSK_DEBUG_CHECK (FALLBACK))
|
|
|
|
{
|
|
|
|
GEnumClass *enum_class = g_type_class_ref (GDK_TYPE_MEMORY_FORMAT);
|
|
|
|
|
2024-04-25 23:08:03 +00:00
|
|
|
if (!GDK_IS_MEMORY_TEXTURE (texture))
|
|
|
|
{
|
|
|
|
gdk_debug_message ("Unoptimized upload for %s", G_OBJECT_TYPE_NAME (texture));
|
|
|
|
}
|
|
|
|
|
2023-12-11 05:21:32 +00:00
|
|
|
if (gdk_texture_get_format (texture) != gsk_gpu_image_get_format (image))
|
|
|
|
{
|
|
|
|
gdk_debug_message ("Unsupported format %s, converting on CPU to %s",
|
|
|
|
g_enum_get_value (enum_class, gdk_texture_get_format (texture))->value_nick,
|
|
|
|
g_enum_get_value (enum_class, gsk_gpu_image_get_format (image))->value_nick);
|
|
|
|
}
|
|
|
|
if (with_mipmap && !(gsk_gpu_image_get_flags (image) & GSK_GPU_IMAGE_CAN_MIPMAP))
|
|
|
|
{
|
|
|
|
gdk_debug_message ("Format %s does not support mipmaps",
|
|
|
|
g_enum_get_value (enum_class, gsk_gpu_image_get_format (image))->value_nick);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_type_class_unref (enum_class);
|
|
|
|
}
|
|
|
|
|
2023-09-01 10:11:06 +00:00
|
|
|
self = (GskGpuUploadTextureOp *) gsk_gpu_op_alloc (frame, &GSK_GPU_UPLOAD_TEXTURE_OP_CLASS);
|
|
|
|
|
|
|
|
self->texture = g_object_ref (texture);
|
2024-08-27 02:48:43 +00:00
|
|
|
self->lod_level = lod_level;
|
memoryformat: Add linear/nearest choice for mipmaping
linear will average all the pixels for the lod, nearest will just pick
one (using the same method as OpenGL/Vulkan, picking bottom right
center).
This doesn't really make linear/nearest filtering work as it should
(because it's still a form of mipmaps), but it has 2 advantages:
1. it gets closer to the desired effect
2. it is a lot faster
Because only 1 pixel is chosen from the original image, instead of
averaging all pixels, a lot less memory needs to be accessed, and
because memory access is the bottleneck for large images, the speedup is
almost linear with the number of pixels not accessed.
And that means that even for lot level 3, aka 1/8th scale, only 1/64 of
the pixels need to be accessed, and everything is 50x faster.
Switching gtk4-demo --run=image_scaling to linear/nearest makes all the
lag go away for me, even with a 64k x 64k image.
2024-09-03 04:12:03 +00:00
|
|
|
self->lod_filter = lod_filter;
|
2023-10-12 20:59:09 +00:00
|
|
|
self->image = image;
|
2023-09-01 10:11:06 +00:00
|
|
|
|
2024-07-12 06:53:37 +00:00
|
|
|
return g_object_ref (self->image);
|
2023-09-01 10:11:06 +00:00
|
|
|
}
|
|
|
|
|
2023-08-12 20:10:16 +00:00
|
|
|
typedef struct _GskGpuUploadCairoOp GskGpuUploadCairoOp;
|
|
|
|
|
|
|
|
struct _GskGpuUploadCairoOp
|
|
|
|
{
|
|
|
|
GskGpuOp op;
|
|
|
|
|
|
|
|
GskGpuImage *image;
|
|
|
|
graphene_rect_t viewport;
|
2023-12-26 02:32:50 +00:00
|
|
|
GskGpuCairoFunc func;
|
|
|
|
gpointer user_data;
|
|
|
|
GDestroyNotify user_destroy;
|
2023-08-12 20:10:16 +00:00
|
|
|
|
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);
|
2023-12-26 02:32:50 +00:00
|
|
|
if (self->user_destroy)
|
|
|
|
self->user_destroy (self->user_data);
|
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);
|
|
|
|
|
2023-12-26 02:32:50 +00:00
|
|
|
self->func (self->user_data, cr);
|
2023-08-12 20:10:16 +00:00
|
|
|
|
|
|
|
cairo_destroy (cr);
|
|
|
|
|
|
|
|
cairo_surface_finish (surface);
|
|
|
|
cairo_surface_destroy (surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
static GskGpuOp *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_cairo_op_vk_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskVulkanCommandState *state)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self = (GskGpuUploadCairoOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_vk_command (op,
|
|
|
|
frame,
|
2023-11-04 02:57:38 +00:00
|
|
|
state,
|
2023-08-12 20:10:16 +00:00
|
|
|
GSK_VULKAN_IMAGE (self->image),
|
|
|
|
gsk_gpu_upload_cairo_op_draw,
|
|
|
|
&self->buffer);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static GskGpuOp *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_cairo_op_gl_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskGLCommandState *state)
|
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,
|
|
|
|
const graphene_vec2_t *scale,
|
2023-12-26 02:32:50 +00:00
|
|
|
const graphene_rect_t *viewport,
|
|
|
|
GskGpuCairoFunc func,
|
|
|
|
gpointer user_data,
|
|
|
|
GDestroyNotify user_destroy)
|
2023-08-12 20:10:16 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadCairoOp *self;
|
|
|
|
|
|
|
|
self = (GskGpuUploadCairoOp *) gsk_gpu_op_alloc (frame, &GSK_GPU_UPLOAD_CAIRO_OP_CLASS);
|
|
|
|
|
|
|
|
self->image = gsk_gpu_device_create_upload_image (gsk_gpu_frame_get_device (frame),
|
2023-11-01 05:43:15 +00:00
|
|
|
FALSE,
|
2023-08-12 20:10:16 +00:00
|
|
|
GDK_MEMORY_DEFAULT,
|
2024-06-26 16:17:56 +00:00
|
|
|
gdk_color_state_get_no_srgb_tf (GDK_COLOR_STATE_SRGB) != NULL,
|
2023-08-12 20:10:16 +00:00
|
|
|
ceil (graphene_vec2_get_x (scale) * viewport->size.width),
|
|
|
|
ceil (graphene_vec2_get_y (scale) * viewport->size.height));
|
|
|
|
self->viewport = *viewport;
|
2023-12-26 02:32:50 +00:00
|
|
|
self->func = func;
|
|
|
|
self->user_data = user_data;
|
|
|
|
self->user_destroy = user_destroy;
|
2023-08-12 20:10:16 +00:00
|
|
|
|
|
|
|
return self->image;
|
|
|
|
}
|
|
|
|
|
2023-09-06 00:11:39 +00:00
|
|
|
typedef struct _GskGpuUploadGlyphOp GskGpuUploadGlyphOp;
|
|
|
|
|
|
|
|
struct _GskGpuUploadGlyphOp
|
|
|
|
{
|
|
|
|
GskGpuOp op;
|
|
|
|
|
|
|
|
GskGpuImage *image;
|
|
|
|
cairo_rectangle_int_t area;
|
|
|
|
PangoFont *font;
|
|
|
|
PangoGlyph glyph;
|
|
|
|
graphene_point_t origin;
|
|
|
|
|
|
|
|
GskGpuBuffer *buffer;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_glyph_op_finish (GskGpuOp *op)
|
|
|
|
{
|
|
|
|
GskGpuUploadGlyphOp *self = (GskGpuUploadGlyphOp *) op;
|
|
|
|
|
|
|
|
g_object_unref (self->image);
|
|
|
|
g_object_unref (self->font);
|
|
|
|
|
|
|
|
g_clear_object (&self->buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_glyph_op_print (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GString *string,
|
|
|
|
guint indent)
|
|
|
|
{
|
|
|
|
GskGpuUploadGlyphOp *self = (GskGpuUploadGlyphOp *) op;
|
2024-02-24 02:48:17 +00:00
|
|
|
PangoFontDescription *desc;
|
|
|
|
char *str;
|
|
|
|
|
|
|
|
desc = pango_font_describe_with_absolute_size (self->font);
|
|
|
|
str = pango_font_description_to_string (desc);
|
2023-09-06 00:11:39 +00:00
|
|
|
|
|
|
|
gsk_gpu_print_op (string, indent, "upload-glyph");
|
|
|
|
gsk_gpu_print_int_rect (string, &self->area);
|
2024-02-24 02:48:17 +00:00
|
|
|
g_string_append_printf (string, "glyph %u font %s ", self->glyph, str);
|
2023-09-06 00:11:39 +00:00
|
|
|
gsk_gpu_print_newline (string);
|
2024-02-24 02:48:17 +00:00
|
|
|
|
|
|
|
g_free (str);
|
|
|
|
pango_font_description_free (desc);
|
2023-09-06 00:11:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gsk_gpu_upload_glyph_op_draw (GskGpuOp *op,
|
|
|
|
guchar *data,
|
|
|
|
gsize stride)
|
|
|
|
{
|
|
|
|
GskGpuUploadGlyphOp *self = (GskGpuUploadGlyphOp *) op;
|
|
|
|
cairo_surface_t *surface;
|
|
|
|
cairo_t *cr;
|
2024-03-01 19:38:50 +00:00
|
|
|
PangoRectangle ink_rect = { 0, };
|
2023-09-06 00:11:39 +00:00
|
|
|
|
|
|
|
surface = cairo_image_surface_create_for_data (data,
|
|
|
|
CAIRO_FORMAT_ARGB32,
|
|
|
|
self->area.width,
|
|
|
|
self->area.height,
|
|
|
|
stride);
|
|
|
|
cairo_surface_set_device_offset (surface, self->origin.x, self->origin.y);
|
|
|
|
|
|
|
|
cr = cairo_create (surface);
|
|
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
|
|
|
|
cairo_paint (cr);
|
|
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
|
|
|
|
|
2023-09-25 19:12:20 +00:00
|
|
|
/* 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);
|
|
|
|
|
2023-09-06 00:11:39 +00:00
|
|
|
/* Draw glyph */
|
|
|
|
cairo_set_source_rgba (cr, 1, 1, 1, 1);
|
|
|
|
|
2024-03-01 19:38:50 +00:00
|
|
|
/* The pango code for drawing hex boxes uses the glyph width */
|
|
|
|
if (self->glyph & PANGO_GLYPH_UNKNOWN_FLAG)
|
|
|
|
pango_font_get_glyph_extents (self->font, self->glyph, &ink_rect, NULL);
|
|
|
|
|
2023-09-06 00:11:39 +00:00
|
|
|
pango_cairo_show_glyph_string (cr,
|
|
|
|
self->font,
|
|
|
|
&(PangoGlyphString) {
|
|
|
|
.num_glyphs = 1,
|
|
|
|
.glyphs = (PangoGlyphInfo[1]) { {
|
2024-03-01 19:38:50 +00:00
|
|
|
.glyph = self->glyph,
|
|
|
|
.geometry = {
|
|
|
|
.width = ink_rect.width,
|
|
|
|
}
|
2023-09-06 00:11:39 +00:00
|
|
|
} }
|
|
|
|
});
|
|
|
|
|
|
|
|
cairo_destroy (cr);
|
|
|
|
|
|
|
|
cairo_surface_finish (surface);
|
|
|
|
cairo_surface_destroy (surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
static GskGpuOp *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_glyph_op_vk_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskVulkanCommandState *state)
|
2023-09-06 00:11:39 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadGlyphOp *self = (GskGpuUploadGlyphOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_vk_command_with_area (op,
|
|
|
|
frame,
|
2023-11-04 02:57:38 +00:00
|
|
|
state,
|
2023-09-06 00:11:39 +00:00
|
|
|
GSK_VULKAN_IMAGE (self->image),
|
|
|
|
&self->area,
|
|
|
|
gsk_gpu_upload_glyph_op_draw,
|
|
|
|
&self->buffer);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static GskGpuOp *
|
2023-11-04 02:57:38 +00:00
|
|
|
gsk_gpu_upload_glyph_op_gl_command (GskGpuOp *op,
|
|
|
|
GskGpuFrame *frame,
|
|
|
|
GskGLCommandState *state)
|
2023-09-06 00:11:39 +00:00
|
|
|
{
|
|
|
|
GskGpuUploadGlyphOp *self = (GskGpuUploadGlyphOp *) op;
|
|
|
|
|
|
|
|
return gsk_gpu_upload_op_gl_command_with_area (op,
|
|
|
|
frame,
|
|
|
|
self->image,
|
|
|
|
&self->area,
|
|
|
|
gsk_gpu_upload_glyph_op_draw);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const GskGpuOpClass GSK_GPU_UPLOAD_GLYPH_OP_CLASS = {
|
|
|
|
GSK_GPU_OP_SIZE (GskGpuUploadGlyphOp),
|
|
|
|
GSK_GPU_STAGE_UPLOAD,
|
|
|
|
gsk_gpu_upload_glyph_op_finish,
|
|
|
|
gsk_gpu_upload_glyph_op_print,
|
|
|
|
#ifdef GDK_RENDERING_VULKAN
|
|
|
|
gsk_gpu_upload_glyph_op_vk_command,
|
|
|
|
#endif
|
|
|
|
gsk_gpu_upload_glyph_op_gl_command,
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
gsk_gpu_upload_glyph_op (GskGpuFrame *frame,
|
|
|
|
GskGpuImage *image,
|
|
|
|
PangoFont *font,
|
2024-10-21 07:09:37 +00:00
|
|
|
PangoGlyph glyph,
|
2023-09-06 00:11:39 +00:00
|
|
|
const cairo_rectangle_int_t *area,
|
|
|
|
const graphene_point_t *origin)
|
|
|
|
{
|
|
|
|
GskGpuUploadGlyphOp *self;
|
|
|
|
|
|
|
|
self = (GskGpuUploadGlyphOp *) gsk_gpu_op_alloc (frame, &GSK_GPU_UPLOAD_GLYPH_OP_CLASS);
|
|
|
|
|
|
|
|
self->image = g_object_ref (image);
|
|
|
|
self->area = *area;
|
|
|
|
self->font = g_object_ref (font);
|
|
|
|
self->glyph = glyph;
|
|
|
|
self->origin = *origin;
|
|
|
|
}
|