2016-12-07 13:50:52 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2017-04-18 14:14:35 +00:00
|
|
|
#include "gskprivate.h"
|
|
|
|
|
2016-12-07 13:50:52 +00:00
|
|
|
#include "gskvulkanrenderprivate.h"
|
|
|
|
|
2016-12-08 12:26:36 +00:00
|
|
|
#include "gskrendererprivate.h"
|
|
|
|
#include "gskvulkanbufferprivate.h"
|
2016-12-16 05:10:24 +00:00
|
|
|
#include "gskvulkancommandpoolprivate.h"
|
2023-07-03 07:15:04 +00:00
|
|
|
#include "gskvulkanglyphcacheprivate.h"
|
2023-07-03 13:57:30 +00:00
|
|
|
#include "gskvulkanprivate.h"
|
2023-07-03 13:47:53 +00:00
|
|
|
#include "gskvulkanpushconstantsopprivate.h"
|
2023-07-03 07:15:04 +00:00
|
|
|
#include "gskvulkanrendererprivate.h"
|
2016-12-08 12:26:36 +00:00
|
|
|
#include "gskvulkanrenderpassprivate.h"
|
|
|
|
|
2023-06-29 04:36:17 +00:00
|
|
|
#include "gdk/gdkvulkancontextprivate.h"
|
|
|
|
|
2023-05-20 00:56:02 +00:00
|
|
|
#define DESCRIPTOR_POOL_MAXITEMS 50000
|
2023-07-08 07:04:12 +00:00
|
|
|
#define VERTEX_BUFFER_SIZE_STEP 128 * 1024 /* 128kB */
|
2023-05-20 00:56:02 +00:00
|
|
|
|
|
|
|
#define GDK_ARRAY_NAME gsk_descriptor_image_infos
|
|
|
|
#define GDK_ARRAY_TYPE_NAME GskDescriptorImageInfos
|
|
|
|
#define GDK_ARRAY_ELEMENT_TYPE VkDescriptorImageInfo
|
|
|
|
#define GDK_ARRAY_BY_VALUE 1
|
|
|
|
#define GDK_ARRAY_PREALLOC 1024
|
|
|
|
#define GDK_ARRAY_NO_MEMSET 1
|
|
|
|
#include "gdk/gdkarrayimpl.c"
|
2016-12-09 01:55:47 +00:00
|
|
|
|
2023-05-25 20:21:23 +00:00
|
|
|
#define GDK_ARRAY_NAME gsk_descriptor_buffer_infos
|
|
|
|
#define GDK_ARRAY_TYPE_NAME GskDescriptorBufferInfos
|
|
|
|
#define GDK_ARRAY_ELEMENT_TYPE VkDescriptorBufferInfo
|
|
|
|
#define GDK_ARRAY_BY_VALUE 1
|
|
|
|
#define GDK_ARRAY_PREALLOC 1024
|
|
|
|
#define GDK_ARRAY_NO_MEMSET 1
|
|
|
|
#include "gdk/gdkarrayimpl.c"
|
|
|
|
|
2023-06-28 18:50:43 +00:00
|
|
|
#define N_DESCRIPTOR_SETS 2
|
2023-06-20 13:39:29 +00:00
|
|
|
|
2016-12-08 16:55:32 +00:00
|
|
|
struct _GskVulkanRender
|
|
|
|
{
|
|
|
|
GskRenderer *renderer;
|
|
|
|
GdkVulkanContext *vulkan;
|
|
|
|
|
2023-04-03 13:39:25 +00:00
|
|
|
double scale;
|
2017-09-30 14:42:32 +00:00
|
|
|
graphene_rect_t viewport;
|
2016-12-26 23:18:52 +00:00
|
|
|
cairo_region_t *clip;
|
2016-12-08 16:55:32 +00:00
|
|
|
|
2016-12-16 05:10:24 +00:00
|
|
|
GskVulkanCommandPool *command_pool;
|
2016-12-08 17:03:05 +00:00
|
|
|
VkFence fence;
|
2023-06-20 13:39:29 +00:00
|
|
|
VkDescriptorSetLayout descriptor_set_layouts[N_DESCRIPTOR_SETS];
|
2023-05-12 18:49:31 +00:00
|
|
|
VkPipelineLayout pipeline_layout;
|
2023-07-05 05:16:58 +00:00
|
|
|
|
2016-12-17 03:22:44 +00:00
|
|
|
GskVulkanUploader *uploader;
|
2023-07-05 05:16:58 +00:00
|
|
|
GskVulkanRenderPass *render_pass;
|
2016-12-09 01:55:47 +00:00
|
|
|
|
2023-05-23 03:08:13 +00:00
|
|
|
GskDescriptorImageInfos descriptor_images;
|
2023-05-25 20:21:23 +00:00
|
|
|
GskDescriptorBufferInfos descriptor_buffers;
|
2016-12-08 23:06:59 +00:00
|
|
|
VkDescriptorPool descriptor_pool;
|
2023-06-20 13:39:29 +00:00
|
|
|
VkDescriptorSet descriptor_sets[N_DESCRIPTOR_SETS];
|
2023-06-29 04:36:17 +00:00
|
|
|
GHashTable *pipeline_cache;
|
2023-07-08 18:40:48 +00:00
|
|
|
GHashTable *render_pass_cache;
|
2016-12-08 17:03:05 +00:00
|
|
|
|
2016-12-08 21:35:16 +00:00
|
|
|
GskVulkanImage *target;
|
|
|
|
|
2023-07-08 07:04:12 +00:00
|
|
|
GskVulkanBuffer *vertex_buffer;
|
2023-05-23 03:26:08 +00:00
|
|
|
VkSampler samplers[3];
|
2023-05-25 20:21:23 +00:00
|
|
|
GskVulkanBuffer *storage_buffer;
|
|
|
|
guchar *storage_buffer_memory;
|
|
|
|
gsize storage_buffer_used;
|
2017-09-30 02:38:59 +00:00
|
|
|
|
2017-09-28 12:29:14 +00:00
|
|
|
GQuark render_pass_counter;
|
2017-09-28 14:27:58 +00:00
|
|
|
GQuark gpu_time_timer;
|
2016-12-08 16:55:32 +00:00
|
|
|
};
|
|
|
|
|
2023-06-29 04:36:17 +00:00
|
|
|
typedef struct _PipelineCacheKey PipelineCacheKey;
|
2023-07-08 18:40:48 +00:00
|
|
|
typedef struct _RenderPassCacheKey RenderPassCacheKey;
|
2023-06-29 04:36:17 +00:00
|
|
|
|
|
|
|
struct _PipelineCacheKey
|
|
|
|
{
|
2023-07-07 00:38:22 +00:00
|
|
|
const GskVulkanOpClass *op_class;
|
2023-06-29 04:36:17 +00:00
|
|
|
const /* interned */ char *clip_type;
|
|
|
|
VkFormat format;
|
|
|
|
};
|
|
|
|
|
2023-07-08 18:40:48 +00:00
|
|
|
struct _RenderPassCacheKey
|
|
|
|
{
|
|
|
|
VkFormat format;
|
|
|
|
VkImageLayout from_layout;
|
|
|
|
VkImageLayout to_layout;
|
|
|
|
};
|
|
|
|
|
2023-06-29 04:36:17 +00:00
|
|
|
static guint
|
|
|
|
pipeline_cache_key_hash (gconstpointer data)
|
|
|
|
{
|
|
|
|
const PipelineCacheKey *key = data;
|
|
|
|
|
2023-07-07 00:38:22 +00:00
|
|
|
return GPOINTER_TO_UINT (key->op_class) ^
|
2023-06-29 04:36:17 +00:00
|
|
|
GPOINTER_TO_UINT (key->clip_type) ^
|
|
|
|
key->format;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
pipeline_cache_key_equal (gconstpointer a,
|
|
|
|
gconstpointer b)
|
|
|
|
{
|
|
|
|
const PipelineCacheKey *keya = a;
|
|
|
|
const PipelineCacheKey *keyb = b;
|
|
|
|
|
2023-07-07 00:38:22 +00:00
|
|
|
return keya->op_class == keyb->op_class &&
|
2023-06-29 04:36:17 +00:00
|
|
|
keya->clip_type == keyb->clip_type &&
|
|
|
|
keya->format == keyb->format;
|
|
|
|
}
|
|
|
|
|
2023-07-08 18:40:48 +00:00
|
|
|
static guint
|
|
|
|
render_pass_cache_key_hash (gconstpointer data)
|
|
|
|
{
|
|
|
|
const RenderPassCacheKey *key = data;
|
|
|
|
|
|
|
|
return (key->from_layout << 20) ^
|
|
|
|
(key->to_layout << 16) ^
|
|
|
|
(key->format);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
render_pass_cache_key_equal (gconstpointer a,
|
|
|
|
gconstpointer b)
|
|
|
|
{
|
|
|
|
const RenderPassCacheKey *keya = a;
|
|
|
|
const RenderPassCacheKey *keyb = b;
|
|
|
|
|
|
|
|
return keya->from_layout == keyb->from_layout &&
|
|
|
|
keya->to_layout == keyb->to_layout &&
|
|
|
|
keya->format == keyb->format;
|
|
|
|
}
|
|
|
|
|
2023-07-05 04:32:52 +00:00
|
|
|
static void
|
|
|
|
gsk_vulkan_render_verbose_print (GskVulkanRender *self,
|
|
|
|
const char *heading)
|
|
|
|
{
|
|
|
|
#ifdef G_ENABLE_DEBUG
|
|
|
|
if (GSK_RENDERER_DEBUG_CHECK (self->renderer, VERBOSE))
|
|
|
|
{
|
|
|
|
GString *string = g_string_new (heading);
|
|
|
|
g_string_append (string, ":\n");
|
2023-07-05 05:16:58 +00:00
|
|
|
gsk_vulkan_render_pass_print (self->render_pass, string, 1);
|
2023-07-05 04:32:52 +00:00
|
|
|
g_print ("%s\n", string->str);
|
|
|
|
g_string_free (string, TRUE);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-12-08 12:26:36 +00:00
|
|
|
static void
|
2016-12-26 23:18:52 +00:00
|
|
|
gsk_vulkan_render_setup (GskVulkanRender *self,
|
|
|
|
GskVulkanImage *target,
|
2018-03-28 13:01:34 +00:00
|
|
|
const graphene_rect_t *rect,
|
|
|
|
const cairo_region_t *clip)
|
2016-12-08 12:26:36 +00:00
|
|
|
{
|
2023-04-03 11:54:34 +00:00
|
|
|
GdkSurface *surface = gsk_renderer_get_surface (self->renderer);
|
2016-12-26 23:18:52 +00:00
|
|
|
|
|
|
|
self->target = g_object_ref (target);
|
2016-12-08 12:26:36 +00:00
|
|
|
|
2016-12-22 18:01:07 +00:00
|
|
|
if (rect)
|
|
|
|
{
|
2017-09-30 14:42:32 +00:00
|
|
|
self->viewport = *rect;
|
2023-04-03 13:39:25 +00:00
|
|
|
self->scale = 1.0;
|
2016-12-22 18:01:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-04-03 13:39:25 +00:00
|
|
|
self->scale = gdk_surface_get_scale (surface);
|
2017-09-30 14:42:32 +00:00
|
|
|
self->viewport = GRAPHENE_RECT_INIT (0, 0,
|
2023-04-03 13:39:25 +00:00
|
|
|
(int) ceil (gdk_surface_get_width (surface) * self->scale),
|
|
|
|
(int) ceil (gdk_surface_get_height (surface) * self->scale));
|
2018-03-28 13:01:34 +00:00
|
|
|
}
|
|
|
|
if (clip)
|
|
|
|
{
|
2023-06-28 23:56:41 +00:00
|
|
|
self->clip = cairo_region_reference ((cairo_region_t *) clip);
|
2018-03-28 13:01:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
self->clip = cairo_region_create_rectangle (&(cairo_rectangle_int_t) {
|
|
|
|
0, 0,
|
|
|
|
gsk_vulkan_image_get_width (target),
|
|
|
|
gsk_vulkan_image_get_height (target)
|
|
|
|
});
|
2016-12-22 18:01:07 +00:00
|
|
|
}
|
2016-12-08 12:26:36 +00:00
|
|
|
}
|
2016-12-07 13:50:52 +00:00
|
|
|
|
2016-12-08 16:46:08 +00:00
|
|
|
GskVulkanRender *
|
|
|
|
gsk_vulkan_render_new (GskRenderer *renderer,
|
2016-12-08 23:06:59 +00:00
|
|
|
GdkVulkanContext *context)
|
2016-12-07 13:50:52 +00:00
|
|
|
{
|
2016-12-08 16:46:08 +00:00
|
|
|
GskVulkanRender *self;
|
2016-12-08 17:03:05 +00:00
|
|
|
VkDevice device;
|
2016-12-08 16:46:08 +00:00
|
|
|
|
2023-03-03 11:48:49 +00:00
|
|
|
self = g_new0 (GskVulkanRender, 1);
|
2016-12-08 16:46:08 +00:00
|
|
|
|
2016-12-07 13:50:52 +00:00
|
|
|
self->vulkan = context;
|
2016-12-08 12:26:36 +00:00
|
|
|
self->renderer = renderer;
|
2023-05-23 03:08:13 +00:00
|
|
|
gsk_descriptor_image_infos_init (&self->descriptor_images);
|
2023-05-25 20:21:23 +00:00
|
|
|
gsk_descriptor_buffer_infos_init (&self->descriptor_buffers);
|
2016-12-07 13:50:52 +00:00
|
|
|
|
2016-12-08 17:03:05 +00:00
|
|
|
device = gdk_vulkan_context_get_device (self->vulkan);
|
|
|
|
|
2016-12-16 05:10:24 +00:00
|
|
|
self->command_pool = gsk_vulkan_command_pool_new (self->vulkan);
|
2016-12-08 17:03:05 +00:00
|
|
|
GSK_VK_CHECK (vkCreateFence, device,
|
|
|
|
&(VkFenceCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO,
|
2016-12-08 17:40:35 +00:00
|
|
|
.flags = VK_FENCE_CREATE_SIGNALED_BIT
|
2016-12-08 17:03:05 +00:00
|
|
|
},
|
|
|
|
NULL,
|
|
|
|
&self->fence);
|
|
|
|
|
2016-12-08 23:06:59 +00:00
|
|
|
GSK_VK_CHECK (vkCreateDescriptorPool, device,
|
|
|
|
&(VkDescriptorPoolCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO,
|
2023-05-20 00:56:02 +00:00
|
|
|
.flags = VK_DESCRIPTOR_POOL_CREATE_UPDATE_AFTER_BIND_BIT,
|
2023-06-20 13:39:29 +00:00
|
|
|
.maxSets = N_DESCRIPTOR_SETS,
|
|
|
|
.poolSizeCount = N_DESCRIPTOR_SETS,
|
|
|
|
.pPoolSizes = (VkDescriptorPoolSize[N_DESCRIPTOR_SETS]) {
|
2016-12-08 23:06:59 +00:00
|
|
|
{
|
2023-06-28 18:50:43 +00:00
|
|
|
.type = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
|
2023-05-20 00:56:02 +00:00
|
|
|
.descriptorCount = DESCRIPTOR_POOL_MAXITEMS
|
2023-05-25 20:21:23 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER,
|
|
|
|
.descriptorCount = DESCRIPTOR_POOL_MAXITEMS
|
2016-12-08 23:06:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
NULL,
|
|
|
|
&self->descriptor_pool);
|
|
|
|
|
2023-06-20 13:39:29 +00:00
|
|
|
GSK_VK_CHECK (vkCreateDescriptorSetLayout, device,
|
|
|
|
&(VkDescriptorSetLayoutCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
|
|
|
|
.bindingCount = 1,
|
|
|
|
.flags = VK_DESCRIPTOR_SET_LAYOUT_CREATE_UPDATE_AFTER_BIND_POOL_BIT,
|
|
|
|
.pBindings = (VkDescriptorSetLayoutBinding[1]) {
|
2023-05-23 03:08:13 +00:00
|
|
|
{
|
2023-06-20 13:39:29 +00:00
|
|
|
.binding = 0,
|
2023-06-28 18:50:43 +00:00
|
|
|
.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
|
2023-05-20 00:56:02 +00:00
|
|
|
.descriptorCount = DESCRIPTOR_POOL_MAXITEMS,
|
2017-09-22 14:02:27 +00:00
|
|
|
.stageFlags = VK_SHADER_STAGE_FRAGMENT_BIT
|
2023-06-20 13:39:29 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
.pNext = &(VkDescriptorSetLayoutBindingFlagsCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_BINDING_FLAGS_CREATE_INFO,
|
|
|
|
.bindingCount = 1,
|
|
|
|
.pBindingFlags = (VkDescriptorBindingFlags[1]) {
|
|
|
|
VK_DESCRIPTOR_BINDING_PARTIALLY_BOUND_BIT
|
|
|
|
| VK_DESCRIPTOR_BINDING_VARIABLE_DESCRIPTOR_COUNT_BIT
|
|
|
|
| VK_DESCRIPTOR_BINDING_UPDATE_AFTER_BIND_BIT,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
NULL,
|
2023-06-28 18:50:43 +00:00
|
|
|
&self->descriptor_set_layouts[0]);
|
2023-06-20 13:39:29 +00:00
|
|
|
|
|
|
|
GSK_VK_CHECK (vkCreateDescriptorSetLayout, device,
|
|
|
|
&(VkDescriptorSetLayoutCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
|
|
|
|
.bindingCount = 1,
|
|
|
|
.flags = VK_DESCRIPTOR_SET_LAYOUT_CREATE_UPDATE_AFTER_BIND_POOL_BIT,
|
|
|
|
.pBindings = (VkDescriptorSetLayoutBinding[1]) {
|
2023-05-25 20:21:23 +00:00
|
|
|
{
|
2023-06-20 13:39:29 +00:00
|
|
|
.binding = 0,
|
2023-05-25 20:21:23 +00:00
|
|
|
.descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER,
|
|
|
|
.descriptorCount = DESCRIPTOR_POOL_MAXITEMS,
|
|
|
|
.stageFlags = VK_SHADER_STAGE_FRAGMENT_BIT
|
|
|
|
},
|
2023-05-20 00:56:02 +00:00
|
|
|
},
|
|
|
|
.pNext = &(VkDescriptorSetLayoutBindingFlagsCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_BINDING_FLAGS_CREATE_INFO,
|
2023-06-20 13:39:29 +00:00
|
|
|
.bindingCount = 1,
|
|
|
|
.pBindingFlags = (VkDescriptorBindingFlags[1]) {
|
2023-05-20 00:56:02 +00:00
|
|
|
VK_DESCRIPTOR_BINDING_PARTIALLY_BOUND_BIT
|
|
|
|
| VK_DESCRIPTOR_BINDING_VARIABLE_DESCRIPTOR_COUNT_BIT
|
|
|
|
| VK_DESCRIPTOR_BINDING_UPDATE_AFTER_BIND_BIT,
|
|
|
|
},
|
2017-09-22 14:02:27 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
NULL,
|
2023-06-28 18:50:43 +00:00
|
|
|
&self->descriptor_set_layouts[1]);
|
2017-09-22 14:02:27 +00:00
|
|
|
|
2023-05-12 18:49:31 +00:00
|
|
|
GSK_VK_CHECK (vkCreatePipelineLayout, device,
|
|
|
|
&(VkPipelineLayoutCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
|
2023-06-20 13:39:29 +00:00
|
|
|
.setLayoutCount = G_N_ELEMENTS (self->descriptor_set_layouts),
|
|
|
|
.pSetLayouts = self->descriptor_set_layouts,
|
2023-05-12 18:49:31 +00:00
|
|
|
.pushConstantRangeCount = gsk_vulkan_push_constants_get_range_count (),
|
|
|
|
.pPushConstantRanges = gsk_vulkan_push_constants_get_ranges ()
|
|
|
|
},
|
|
|
|
NULL,
|
|
|
|
&self->pipeline_layout);
|
2016-12-14 07:00:58 +00:00
|
|
|
|
2017-09-30 02:38:59 +00:00
|
|
|
GSK_VK_CHECK (vkCreateSampler, device,
|
|
|
|
&(VkSamplerCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
|
|
|
|
.magFilter = VK_FILTER_LINEAR,
|
|
|
|
.minFilter = VK_FILTER_LINEAR,
|
2023-05-13 04:02:51 +00:00
|
|
|
.addressModeU = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
|
|
|
|
.addressModeV = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
|
2017-09-30 02:38:59 +00:00
|
|
|
.addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT,
|
|
|
|
.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK,
|
|
|
|
.unnormalizedCoordinates = VK_FALSE,
|
|
|
|
.maxAnisotropy = 1.0,
|
|
|
|
},
|
|
|
|
NULL,
|
2023-05-23 03:08:13 +00:00
|
|
|
&self->samplers[GSK_VULKAN_SAMPLER_DEFAULT]);
|
2017-09-30 02:38:59 +00:00
|
|
|
|
2017-09-30 03:01:34 +00:00
|
|
|
GSK_VK_CHECK (vkCreateSampler, device,
|
|
|
|
&(VkSamplerCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
|
|
|
|
.magFilter = VK_FILTER_LINEAR,
|
|
|
|
.minFilter = VK_FILTER_LINEAR,
|
|
|
|
.addressModeU = VK_SAMPLER_ADDRESS_MODE_REPEAT,
|
|
|
|
.addressModeV = VK_SAMPLER_ADDRESS_MODE_REPEAT,
|
|
|
|
.addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT,
|
|
|
|
.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK,
|
|
|
|
.unnormalizedCoordinates = VK_FALSE,
|
|
|
|
.maxAnisotropy = 1.0,
|
|
|
|
},
|
|
|
|
NULL,
|
2023-05-23 03:08:13 +00:00
|
|
|
&self->samplers[GSK_VULKAN_SAMPLER_REPEAT]);
|
|
|
|
|
2023-05-23 03:26:08 +00:00
|
|
|
GSK_VK_CHECK (vkCreateSampler, device,
|
|
|
|
&(VkSamplerCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
|
|
|
|
.magFilter = VK_FILTER_NEAREST,
|
|
|
|
.minFilter = VK_FILTER_NEAREST,
|
|
|
|
.addressModeU = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
|
|
|
|
.addressModeV = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
|
|
|
|
.addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT,
|
|
|
|
.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK,
|
|
|
|
.unnormalizedCoordinates = VK_FALSE,
|
|
|
|
.maxAnisotropy = 1.0,
|
|
|
|
},
|
|
|
|
NULL,
|
|
|
|
&self->samplers[GSK_VULKAN_SAMPLER_NEAREST]);
|
|
|
|
|
2017-09-30 03:01:34 +00:00
|
|
|
|
2016-12-17 03:22:44 +00:00
|
|
|
self->uploader = gsk_vulkan_uploader_new (self->vulkan, self->command_pool);
|
2023-06-29 04:36:17 +00:00
|
|
|
self->pipeline_cache = g_hash_table_new (pipeline_cache_key_hash, pipeline_cache_key_equal);
|
2023-07-08 18:40:48 +00:00
|
|
|
self->render_pass_cache = g_hash_table_new (render_pass_cache_key_hash, render_pass_cache_key_equal);
|
2016-12-17 03:22:44 +00:00
|
|
|
|
2017-09-28 12:29:14 +00:00
|
|
|
#ifdef G_ENABLE_DEBUG
|
|
|
|
self->render_pass_counter = g_quark_from_static_string ("render-passes");
|
2017-09-28 14:27:58 +00:00
|
|
|
self->gpu_time_timer = g_quark_from_static_string ("gpu-time");
|
2017-09-28 12:29:14 +00:00
|
|
|
#endif
|
|
|
|
|
2016-12-08 16:46:08 +00:00
|
|
|
return self;
|
2016-12-07 13:50:52 +00:00
|
|
|
}
|
|
|
|
|
2023-05-24 14:55:18 +00:00
|
|
|
VkFence
|
|
|
|
gsk_vulkan_render_get_fence (GskVulkanRender *self)
|
|
|
|
{
|
|
|
|
return self->fence;
|
|
|
|
}
|
|
|
|
|
2023-07-05 05:06:17 +00:00
|
|
|
static void
|
2016-12-08 12:26:36 +00:00
|
|
|
gsk_vulkan_render_add_node (GskVulkanRender *self,
|
|
|
|
GskRenderNode *node)
|
|
|
|
{
|
2023-05-10 00:46:57 +00:00
|
|
|
graphene_vec2_t scale;
|
|
|
|
|
|
|
|
graphene_vec2_init (&scale, self->scale, self->scale);
|
2017-09-28 00:56:01 +00:00
|
|
|
|
2023-07-05 05:16:58 +00:00
|
|
|
self->render_pass = gsk_vulkan_render_pass_new (self->vulkan,
|
|
|
|
self,
|
|
|
|
self->target,
|
|
|
|
&scale,
|
|
|
|
&self->viewport,
|
|
|
|
self->clip,
|
|
|
|
node,
|
2023-07-05 07:10:08 +00:00
|
|
|
TRUE);
|
2017-09-28 12:29:14 +00:00
|
|
|
|
2023-07-05 04:32:52 +00:00
|
|
|
gsk_vulkan_render_verbose_print (self, "start of frame");
|
2016-12-08 12:26:36 +00:00
|
|
|
}
|
|
|
|
|
2017-09-28 00:56:01 +00:00
|
|
|
void
|
|
|
|
gsk_vulkan_render_upload (GskVulkanRender *self)
|
2016-12-08 12:26:36 +00:00
|
|
|
{
|
2023-07-03 07:15:04 +00:00
|
|
|
gsk_vulkan_glyph_cache_upload (gsk_vulkan_renderer_get_glyph_cache (GSK_VULKAN_RENDERER (self->renderer)),
|
|
|
|
self->uploader);
|
|
|
|
|
2023-07-08 19:28:56 +00:00
|
|
|
gsk_vulkan_render_pass_upload (self->render_pass, self->uploader);
|
2016-12-08 12:26:36 +00:00
|
|
|
|
2017-09-28 00:56:01 +00:00
|
|
|
gsk_vulkan_uploader_upload (self->uploader);
|
2016-12-08 12:26:36 +00:00
|
|
|
}
|
|
|
|
|
2023-06-29 04:36:17 +00:00
|
|
|
VkPipeline
|
2023-07-08 18:16:39 +00:00
|
|
|
gsk_vulkan_render_get_pipeline (GskVulkanRender *self,
|
|
|
|
const GskVulkanOpClass *op_class,
|
|
|
|
const char *clip_type,
|
|
|
|
VkFormat format,
|
|
|
|
VkRenderPass render_pass)
|
2023-06-29 04:36:17 +00:00
|
|
|
{
|
|
|
|
PipelineCacheKey cache_key;
|
|
|
|
VkPipeline pipeline;
|
|
|
|
GdkDisplay *display;
|
|
|
|
char *vertex_shader_name, *fragment_shader_name;
|
|
|
|
|
|
|
|
cache_key = (PipelineCacheKey) {
|
2023-07-07 00:38:22 +00:00
|
|
|
.op_class = op_class,
|
|
|
|
.clip_type = clip_type,
|
2023-06-29 04:36:17 +00:00
|
|
|
.format = format,
|
|
|
|
};
|
|
|
|
pipeline = g_hash_table_lookup (self->pipeline_cache, &cache_key);
|
|
|
|
if (pipeline)
|
|
|
|
return pipeline;
|
|
|
|
|
|
|
|
display = gdk_draw_context_get_display (GDK_DRAW_CONTEXT (self->vulkan));
|
2023-07-07 00:38:22 +00:00
|
|
|
vertex_shader_name = g_strconcat ("/org/gtk/libgsk/vulkan/", op_class->shader_name, clip_type, ".vert.spv", NULL);
|
|
|
|
fragment_shader_name = g_strconcat ("/org/gtk/libgsk/vulkan/", op_class->shader_name, clip_type, ".frag.spv", NULL);
|
2023-06-29 04:36:17 +00:00
|
|
|
|
|
|
|
GSK_VK_CHECK (vkCreateGraphicsPipelines, gdk_vulkan_context_get_device (self->vulkan),
|
|
|
|
gdk_vulkan_context_get_pipeline_cache (self->vulkan),
|
|
|
|
1,
|
|
|
|
&(VkGraphicsPipelineCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO,
|
|
|
|
.stageCount = 2,
|
|
|
|
.pStages = (VkPipelineShaderStageCreateInfo[2]) {
|
|
|
|
{
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO,
|
|
|
|
.stage = VK_SHADER_STAGE_VERTEX_BIT,
|
|
|
|
.module = gdk_display_get_vk_shader_module (display, vertex_shader_name),
|
|
|
|
.pName = "main",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO,
|
|
|
|
.stage = VK_SHADER_STAGE_FRAGMENT_BIT,
|
|
|
|
.module = gdk_display_get_vk_shader_module (display, fragment_shader_name),
|
|
|
|
.pName = "main",
|
|
|
|
},
|
|
|
|
},
|
2023-07-07 00:38:22 +00:00
|
|
|
.pVertexInputState = op_class->vertex_input_state,
|
2023-06-29 04:36:17 +00:00
|
|
|
.pInputAssemblyState = &(VkPipelineInputAssemblyStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO,
|
|
|
|
.topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST,
|
|
|
|
.primitiveRestartEnable = VK_FALSE,
|
|
|
|
},
|
|
|
|
.pTessellationState = NULL,
|
|
|
|
.pViewportState = &(VkPipelineViewportStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO,
|
|
|
|
.viewportCount = 1,
|
|
|
|
.scissorCount = 1
|
|
|
|
},
|
|
|
|
.pRasterizationState = &(VkPipelineRasterizationStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO,
|
|
|
|
.depthClampEnable = VK_FALSE,
|
|
|
|
.rasterizerDiscardEnable = VK_FALSE,
|
|
|
|
.polygonMode = VK_POLYGON_MODE_FILL,
|
|
|
|
.cullMode = VK_CULL_MODE_NONE,
|
|
|
|
.frontFace = VK_FRONT_FACE_CLOCKWISE,
|
|
|
|
.lineWidth = 1.0f,
|
|
|
|
},
|
|
|
|
.pMultisampleState = &(VkPipelineMultisampleStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO,
|
|
|
|
.rasterizationSamples = 1,
|
|
|
|
},
|
|
|
|
.pDepthStencilState = &(VkPipelineDepthStencilStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_DEPTH_STENCIL_STATE_CREATE_INFO
|
|
|
|
},
|
|
|
|
.pColorBlendState = &(VkPipelineColorBlendStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO,
|
|
|
|
.attachmentCount = 1,
|
|
|
|
.pAttachments = (VkPipelineColorBlendAttachmentState []) {
|
|
|
|
{
|
|
|
|
.blendEnable = VK_TRUE,
|
|
|
|
.colorBlendOp = VK_BLEND_OP_ADD,
|
|
|
|
.srcColorBlendFactor = VK_BLEND_FACTOR_ONE,
|
|
|
|
.dstColorBlendFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA,
|
|
|
|
.alphaBlendOp = VK_BLEND_OP_ADD,
|
|
|
|
.srcAlphaBlendFactor = VK_BLEND_FACTOR_ONE,
|
|
|
|
.dstAlphaBlendFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA,
|
|
|
|
.colorWriteMask = VK_COLOR_COMPONENT_A_BIT
|
|
|
|
| VK_COLOR_COMPONENT_R_BIT
|
|
|
|
| VK_COLOR_COMPONENT_G_BIT
|
|
|
|
| VK_COLOR_COMPONENT_B_BIT
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.pDynamicState = &(VkPipelineDynamicStateCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_DYNAMIC_STATE_CREATE_INFO,
|
|
|
|
.dynamicStateCount = 2,
|
|
|
|
.pDynamicStates = (VkDynamicState[2]) {
|
|
|
|
VK_DYNAMIC_STATE_VIEWPORT,
|
|
|
|
VK_DYNAMIC_STATE_SCISSOR
|
|
|
|
},
|
|
|
|
},
|
|
|
|
.layout = self->pipeline_layout,
|
|
|
|
.renderPass = render_pass,
|
|
|
|
.subpass = 0,
|
|
|
|
.basePipelineHandle = VK_NULL_HANDLE,
|
|
|
|
.basePipelineIndex = -1,
|
|
|
|
},
|
|
|
|
NULL,
|
|
|
|
&pipeline);
|
|
|
|
|
|
|
|
g_free (fragment_shader_name);
|
|
|
|
g_free (vertex_shader_name);
|
|
|
|
|
|
|
|
g_hash_table_insert (self->pipeline_cache, g_memdup (&cache_key, sizeof (PipelineCacheKey)), pipeline);
|
|
|
|
gdk_vulkan_context_pipeline_cache_updated (self->vulkan);
|
|
|
|
|
|
|
|
return pipeline;
|
|
|
|
}
|
|
|
|
|
2023-07-08 18:40:48 +00:00
|
|
|
VkRenderPass
|
|
|
|
gsk_vulkan_render_get_render_pass (GskVulkanRender *self,
|
|
|
|
VkFormat format,
|
|
|
|
VkImageLayout from_layout,
|
|
|
|
VkImageLayout to_layout)
|
|
|
|
{
|
|
|
|
RenderPassCacheKey cache_key;
|
|
|
|
VkRenderPass render_pass;
|
|
|
|
|
|
|
|
cache_key = (RenderPassCacheKey) {
|
|
|
|
.format = format,
|
|
|
|
.from_layout = from_layout,
|
|
|
|
.to_layout = to_layout,
|
|
|
|
};
|
|
|
|
render_pass = g_hash_table_lookup (self->render_pass_cache, &cache_key);
|
|
|
|
if (render_pass)
|
|
|
|
return render_pass;
|
|
|
|
|
|
|
|
GSK_VK_CHECK (vkCreateRenderPass, gdk_vulkan_context_get_device (self->vulkan),
|
|
|
|
&(VkRenderPassCreateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO,
|
|
|
|
.attachmentCount = 1,
|
|
|
|
.pAttachments = (VkAttachmentDescription[]) {
|
|
|
|
{
|
|
|
|
.format = format,
|
|
|
|
.samples = VK_SAMPLE_COUNT_1_BIT,
|
|
|
|
.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR,
|
|
|
|
.storeOp = VK_ATTACHMENT_STORE_OP_STORE,
|
|
|
|
.initialLayout = from_layout,
|
|
|
|
.finalLayout = to_layout
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.subpassCount = 1,
|
|
|
|
.pSubpasses = (VkSubpassDescription []) {
|
|
|
|
{
|
|
|
|
.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS,
|
|
|
|
.inputAttachmentCount = 0,
|
|
|
|
.colorAttachmentCount = 1,
|
|
|
|
.pColorAttachments = (VkAttachmentReference []) {
|
|
|
|
{
|
|
|
|
.attachment = 0,
|
|
|
|
.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.pResolveAttachments = (VkAttachmentReference []) {
|
|
|
|
{
|
|
|
|
.attachment = VK_ATTACHMENT_UNUSED,
|
|
|
|
.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.pDepthStencilAttachment = NULL,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.dependencyCount = 0
|
|
|
|
},
|
|
|
|
NULL,
|
|
|
|
&render_pass);
|
|
|
|
|
|
|
|
g_hash_table_insert (self->render_pass_cache, g_memdup (&cache_key, sizeof (RenderPassCacheKey)), render_pass);
|
|
|
|
|
|
|
|
return render_pass;
|
|
|
|
}
|
|
|
|
|
2023-06-20 13:39:29 +00:00
|
|
|
void
|
|
|
|
gsk_vulkan_render_bind_descriptor_sets (GskVulkanRender *self,
|
|
|
|
VkCommandBuffer command_buffer)
|
2016-12-09 01:55:47 +00:00
|
|
|
{
|
2023-06-20 13:39:29 +00:00
|
|
|
vkCmdBindDescriptorSets (command_buffer,
|
|
|
|
VK_PIPELINE_BIND_POINT_GRAPHICS,
|
|
|
|
self->pipeline_layout,
|
|
|
|
0,
|
2023-06-28 18:50:43 +00:00
|
|
|
N_DESCRIPTOR_SETS,
|
2023-06-20 13:39:29 +00:00
|
|
|
self->descriptor_sets,
|
|
|
|
0,
|
|
|
|
NULL);
|
2017-09-30 02:38:59 +00:00
|
|
|
}
|
|
|
|
|
2016-12-09 01:55:47 +00:00
|
|
|
gsize
|
2023-06-28 18:50:43 +00:00
|
|
|
gsk_vulkan_render_get_image_descriptor (GskVulkanRender *self,
|
|
|
|
GskVulkanImage *image,
|
|
|
|
GskVulkanRenderSampler render_sampler)
|
2016-12-09 01:55:47 +00:00
|
|
|
{
|
2023-05-20 00:56:02 +00:00
|
|
|
gsize result;
|
2016-12-09 01:55:47 +00:00
|
|
|
|
2023-05-23 03:08:13 +00:00
|
|
|
result = gsk_descriptor_image_infos_get_size (&self->descriptor_images);
|
|
|
|
gsk_descriptor_image_infos_append (&self->descriptor_images,
|
|
|
|
&(VkDescriptorImageInfo) {
|
2023-06-28 18:50:43 +00:00
|
|
|
.sampler = self->samplers[render_sampler],
|
2023-05-23 03:08:13 +00:00
|
|
|
.imageView = gsk_vulkan_image_get_image_view (image),
|
|
|
|
.imageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL
|
|
|
|
});
|
2017-09-30 02:38:59 +00:00
|
|
|
|
2023-05-20 00:56:02 +00:00
|
|
|
g_assert (result < DESCRIPTOR_POOL_MAXITEMS);
|
2017-09-30 02:38:59 +00:00
|
|
|
|
2023-05-20 00:56:02 +00:00
|
|
|
return result;
|
2016-12-09 01:55:47 +00:00
|
|
|
}
|
|
|
|
|
2023-05-25 20:21:23 +00:00
|
|
|
static void
|
|
|
|
gsk_vulkan_render_ensure_storage_buffer (GskVulkanRender *self)
|
|
|
|
{
|
|
|
|
if (self->storage_buffer_memory != NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (self->storage_buffer == NULL)
|
|
|
|
{
|
|
|
|
self->storage_buffer = gsk_vulkan_buffer_new_storage (self->vulkan,
|
|
|
|
/* random */
|
|
|
|
sizeof (float) * 1024 * 1024);
|
|
|
|
}
|
|
|
|
|
|
|
|
self->storage_buffer_memory = gsk_vulkan_buffer_map (self->storage_buffer);
|
|
|
|
|
|
|
|
if (gsk_vulkan_render_get_buffer_descriptor (self, self->storage_buffer) != 0)
|
|
|
|
{
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gsize
|
|
|
|
gsk_vulkan_render_get_buffer_descriptor (GskVulkanRender *self,
|
|
|
|
GskVulkanBuffer *buffer)
|
|
|
|
{
|
|
|
|
gsize result;
|
|
|
|
|
|
|
|
gsk_vulkan_render_ensure_storage_buffer (self);
|
|
|
|
|
|
|
|
result = gsk_descriptor_buffer_infos_get_size (&self->descriptor_buffers);
|
|
|
|
gsk_descriptor_buffer_infos_append (&self->descriptor_buffers,
|
|
|
|
&(VkDescriptorBufferInfo) {
|
|
|
|
.buffer = gsk_vulkan_buffer_get_buffer (buffer),
|
|
|
|
.offset = 0,
|
|
|
|
.range = VK_WHOLE_SIZE
|
|
|
|
});
|
|
|
|
|
|
|
|
g_assert (result < DESCRIPTOR_POOL_MAXITEMS);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline gsize
|
|
|
|
round_up (gsize number, gsize divisor)
|
|
|
|
{
|
|
|
|
return (number + divisor - 1) / divisor * divisor;
|
|
|
|
}
|
|
|
|
|
|
|
|
guchar *
|
|
|
|
gsk_vulkan_render_get_buffer_memory (GskVulkanRender *self,
|
|
|
|
gsize size,
|
|
|
|
gsize alignment,
|
|
|
|
gsize *out_offset)
|
|
|
|
{
|
|
|
|
guchar *result;
|
|
|
|
|
|
|
|
g_assert (alignment >= sizeof (float));
|
|
|
|
|
|
|
|
gsk_vulkan_render_ensure_storage_buffer (self);
|
|
|
|
|
|
|
|
self->storage_buffer_used = round_up (self->storage_buffer_used, alignment);
|
|
|
|
result = self->storage_buffer_memory + self->storage_buffer_used;
|
|
|
|
*out_offset = self->storage_buffer_used / sizeof (float);
|
|
|
|
|
|
|
|
self->storage_buffer_used += size;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-12-09 01:55:47 +00:00
|
|
|
static void
|
2017-09-30 02:38:59 +00:00
|
|
|
gsk_vulkan_render_prepare_descriptor_sets (GskVulkanRender *self)
|
2016-12-09 01:55:47 +00:00
|
|
|
{
|
|
|
|
VkDevice device;
|
2023-06-28 18:50:43 +00:00
|
|
|
VkWriteDescriptorSet descriptor_sets[N_DESCRIPTOR_SETS];
|
2023-06-08 04:55:40 +00:00
|
|
|
gsize n_descriptor_sets;
|
2016-12-09 01:55:47 +00:00
|
|
|
|
|
|
|
device = gdk_vulkan_context_get_device (self->vulkan);
|
|
|
|
|
2023-07-05 05:16:58 +00:00
|
|
|
gsk_vulkan_render_pass_reserve_descriptor_sets (self->render_pass, self);
|
2016-12-09 01:55:47 +00:00
|
|
|
|
2023-05-25 20:21:23 +00:00
|
|
|
if (self->storage_buffer_memory)
|
|
|
|
{
|
|
|
|
gsk_vulkan_buffer_unmap (self->storage_buffer);
|
|
|
|
self->storage_buffer_memory = NULL;
|
2023-06-18 16:38:48 +00:00
|
|
|
self->storage_buffer_used = 0;
|
2023-05-25 20:21:23 +00:00
|
|
|
}
|
|
|
|
|
2017-01-06 17:10:12 +00:00
|
|
|
GSK_VK_CHECK (vkAllocateDescriptorSets, device,
|
|
|
|
&(VkDescriptorSetAllocateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
|
|
|
|
.descriptorPool = self->descriptor_pool,
|
2023-06-20 13:39:29 +00:00
|
|
|
.descriptorSetCount = N_DESCRIPTOR_SETS,
|
|
|
|
.pSetLayouts = self->descriptor_set_layouts,
|
2023-05-20 00:56:02 +00:00
|
|
|
.pNext = &(VkDescriptorSetVariableDescriptorCountAllocateInfo) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_VARIABLE_DESCRIPTOR_COUNT_ALLOCATE_INFO,
|
2023-06-20 13:39:29 +00:00
|
|
|
.descriptorSetCount = N_DESCRIPTOR_SETS,
|
|
|
|
.pDescriptorCounts = (uint32_t[N_DESCRIPTOR_SETS]) {
|
|
|
|
MAX (1, gsk_descriptor_image_infos_get_size (&self->descriptor_images)),
|
|
|
|
MAX (1, gsk_descriptor_buffer_infos_get_size (&self->descriptor_buffers))
|
2023-05-20 00:56:02 +00:00
|
|
|
}
|
|
|
|
}
|
2017-01-06 17:10:12 +00:00
|
|
|
},
|
2023-06-20 13:39:29 +00:00
|
|
|
self->descriptor_sets);
|
2023-05-20 00:56:02 +00:00
|
|
|
|
2023-06-08 04:55:40 +00:00
|
|
|
n_descriptor_sets = 0;
|
|
|
|
if (gsk_descriptor_image_infos_get_size (&self->descriptor_images) > 0)
|
|
|
|
{
|
|
|
|
descriptor_sets[n_descriptor_sets++] = (VkWriteDescriptorSet) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
|
2023-06-20 13:39:29 +00:00
|
|
|
.dstSet = self->descriptor_sets[0],
|
2023-06-08 04:55:40 +00:00
|
|
|
.dstBinding = 0,
|
|
|
|
.dstArrayElement = 0,
|
|
|
|
.descriptorCount = gsk_descriptor_image_infos_get_size (&self->descriptor_images),
|
2023-06-28 18:50:43 +00:00
|
|
|
.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
|
2023-06-08 04:55:40 +00:00
|
|
|
.pImageInfo = gsk_descriptor_image_infos_get_data (&self->descriptor_images)
|
|
|
|
};
|
|
|
|
}
|
2023-05-25 20:21:23 +00:00
|
|
|
if (gsk_descriptor_buffer_infos_get_size (&self->descriptor_buffers) > 0)
|
|
|
|
{
|
|
|
|
descriptor_sets[n_descriptor_sets++] = (VkWriteDescriptorSet) {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
|
2023-06-28 18:50:43 +00:00
|
|
|
.dstSet = self->descriptor_sets[1],
|
2023-06-20 13:39:29 +00:00
|
|
|
.dstBinding = 0,
|
2023-05-25 20:21:23 +00:00
|
|
|
.dstArrayElement = 0,
|
|
|
|
.descriptorCount = gsk_descriptor_buffer_infos_get_size (&self->descriptor_buffers),
|
|
|
|
.descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER,
|
|
|
|
.pBufferInfo = gsk_descriptor_buffer_infos_get_data (&self->descriptor_buffers)
|
|
|
|
};
|
|
|
|
}
|
2023-06-08 04:55:40 +00:00
|
|
|
|
2023-05-20 00:56:02 +00:00
|
|
|
vkUpdateDescriptorSets (device,
|
2023-06-08 04:55:40 +00:00
|
|
|
n_descriptor_sets,
|
|
|
|
descriptor_sets,
|
2023-05-20 00:56:02 +00:00
|
|
|
0, NULL);
|
2016-12-09 01:55:47 +00:00
|
|
|
}
|
|
|
|
|
2023-07-08 07:04:12 +00:00
|
|
|
static void
|
|
|
|
gsk_vulkan_render_collect_vertex_buffer (GskVulkanRender *self)
|
|
|
|
{
|
|
|
|
gsize n_bytes;
|
|
|
|
guchar *data;
|
|
|
|
|
|
|
|
n_bytes = gsk_vulkan_render_pass_count_vertex_data (self->render_pass, 0);
|
|
|
|
if (n_bytes == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (self->vertex_buffer && gsk_vulkan_buffer_get_size (self->vertex_buffer) < n_bytes)
|
|
|
|
g_clear_pointer (&self->vertex_buffer, gsk_vulkan_buffer_free);
|
|
|
|
|
|
|
|
if (self->vertex_buffer == NULL)
|
|
|
|
self->vertex_buffer = gsk_vulkan_buffer_new (self->vulkan, round_up (n_bytes, VERTEX_BUFFER_SIZE_STEP));
|
|
|
|
|
|
|
|
data = gsk_vulkan_buffer_map (self->vertex_buffer);
|
2023-07-08 19:28:56 +00:00
|
|
|
gsk_vulkan_render_pass_collect_vertex_data (self->render_pass, data);
|
2023-07-08 07:04:12 +00:00
|
|
|
gsk_vulkan_buffer_unmap (self->vertex_buffer);
|
|
|
|
}
|
|
|
|
|
2023-06-25 03:02:48 +00:00
|
|
|
void
|
|
|
|
gsk_vulkan_render_draw_pass (GskVulkanRender *self,
|
|
|
|
GskVulkanRenderPass *pass,
|
|
|
|
VkFence fence)
|
|
|
|
{
|
|
|
|
VkCommandBuffer command_buffer;
|
|
|
|
|
|
|
|
command_buffer = gsk_vulkan_command_pool_get_buffer (self->command_pool);
|
|
|
|
|
2023-07-08 07:04:12 +00:00
|
|
|
if (self->vertex_buffer)
|
|
|
|
vkCmdBindVertexBuffers (command_buffer,
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
(VkBuffer[1]) {
|
|
|
|
gsk_vulkan_buffer_get_buffer (self->vertex_buffer)
|
|
|
|
},
|
|
|
|
(VkDeviceSize[1]) { 0 });
|
|
|
|
|
2023-06-25 03:02:48 +00:00
|
|
|
gsk_vulkan_render_pass_draw (pass, self, self->pipeline_layout, command_buffer);
|
|
|
|
|
|
|
|
gsk_vulkan_command_pool_submit_buffer (self->command_pool,
|
|
|
|
command_buffer,
|
2023-07-05 07:10:08 +00:00
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL,
|
2023-06-25 03:02:48 +00:00
|
|
|
fence);
|
|
|
|
}
|
|
|
|
|
2016-12-08 12:26:36 +00:00
|
|
|
void
|
2017-09-30 02:38:59 +00:00
|
|
|
gsk_vulkan_render_draw (GskVulkanRender *self)
|
2016-12-08 12:26:36 +00:00
|
|
|
{
|
2017-09-28 14:27:58 +00:00
|
|
|
#ifdef G_ENABLE_DEBUG
|
2018-01-14 20:11:33 +00:00
|
|
|
if (GSK_RENDERER_DEBUG_CHECK (self->renderer, SYNC))
|
2017-09-28 14:27:58 +00:00
|
|
|
gsk_profiler_timer_begin (gsk_renderer_get_profiler (self->renderer), self->gpu_time_timer);
|
|
|
|
#endif
|
|
|
|
|
2017-09-30 02:38:59 +00:00
|
|
|
gsk_vulkan_render_prepare_descriptor_sets (self);
|
2016-12-08 12:26:36 +00:00
|
|
|
|
2023-07-08 07:04:12 +00:00
|
|
|
gsk_vulkan_render_collect_vertex_buffer (self);
|
|
|
|
|
2023-07-05 05:16:58 +00:00
|
|
|
gsk_vulkan_render_draw_pass (self,
|
|
|
|
self->render_pass,
|
|
|
|
self->fence);
|
2016-12-08 12:26:36 +00:00
|
|
|
|
2018-01-14 20:11:33 +00:00
|
|
|
#ifdef G_ENABLE_DEBUG
|
|
|
|
if (GSK_RENDERER_DEBUG_CHECK (self->renderer, SYNC))
|
2016-12-08 18:56:56 +00:00
|
|
|
{
|
2017-09-28 14:27:58 +00:00
|
|
|
GskProfiler *profiler;
|
|
|
|
gint64 gpu_time;
|
|
|
|
|
2016-12-08 18:56:56 +00:00
|
|
|
GSK_VK_CHECK (vkWaitForFences, gdk_vulkan_context_get_device (self->vulkan),
|
|
|
|
1,
|
|
|
|
&self->fence,
|
|
|
|
VK_TRUE,
|
|
|
|
INT64_MAX);
|
2017-09-28 14:27:58 +00:00
|
|
|
|
|
|
|
profiler = gsk_renderer_get_profiler (self->renderer);
|
|
|
|
gpu_time = gsk_profiler_timer_end (profiler, self->gpu_time_timer);
|
|
|
|
gsk_profiler_timer_set (profiler, self->gpu_time_timer, gpu_time);
|
2016-12-08 18:56:56 +00:00
|
|
|
}
|
2018-01-14 20:11:33 +00:00
|
|
|
#endif
|
2016-12-07 13:50:52 +00:00
|
|
|
}
|
|
|
|
|
2017-11-02 20:39:00 +00:00
|
|
|
GdkTexture *
|
2016-12-22 18:01:07 +00:00
|
|
|
gsk_vulkan_render_download_target (GskVulkanRender *self)
|
|
|
|
{
|
2023-04-14 19:27:42 +00:00
|
|
|
GdkTexture *texture;
|
|
|
|
|
|
|
|
texture = gsk_vulkan_image_download (self->target, self->uploader);
|
2016-12-22 18:01:07 +00:00
|
|
|
gsk_vulkan_uploader_reset (self->uploader);
|
|
|
|
|
2023-04-14 19:27:42 +00:00
|
|
|
return texture;
|
2016-12-22 18:01:07 +00:00
|
|
|
}
|
|
|
|
|
2016-12-08 17:40:35 +00:00
|
|
|
static void
|
|
|
|
gsk_vulkan_render_cleanup (GskVulkanRender *self)
|
2016-12-07 13:50:52 +00:00
|
|
|
{
|
2016-12-08 17:03:05 +00:00
|
|
|
VkDevice device = gdk_vulkan_context_get_device (self->vulkan);
|
|
|
|
|
2016-12-08 17:40:35 +00:00
|
|
|
/* XXX: Wait for fence here or just in reset()? */
|
|
|
|
GSK_VK_CHECK (vkWaitForFences, device,
|
|
|
|
1,
|
|
|
|
&self->fence,
|
|
|
|
VK_TRUE,
|
|
|
|
INT64_MAX);
|
|
|
|
|
|
|
|
GSK_VK_CHECK (vkResetFences, device,
|
|
|
|
1,
|
|
|
|
&self->fence);
|
2016-12-16 05:10:24 +00:00
|
|
|
|
2016-12-17 03:22:44 +00:00
|
|
|
gsk_vulkan_uploader_reset (self->uploader);
|
|
|
|
|
2016-12-16 05:10:24 +00:00
|
|
|
gsk_vulkan_command_pool_reset (self->command_pool);
|
2016-12-08 12:26:36 +00:00
|
|
|
|
2017-01-06 15:31:03 +00:00
|
|
|
GSK_VK_CHECK (vkResetDescriptorPool, device,
|
|
|
|
self->descriptor_pool,
|
|
|
|
0);
|
2023-05-23 03:08:13 +00:00
|
|
|
gsk_descriptor_image_infos_set_size (&self->descriptor_images, 0);
|
2023-05-25 20:21:23 +00:00
|
|
|
gsk_descriptor_buffer_infos_set_size (&self->descriptor_buffers, 0);
|
2016-12-09 01:55:47 +00:00
|
|
|
|
2023-07-05 05:16:58 +00:00
|
|
|
g_clear_pointer (&self->render_pass, gsk_vulkan_render_pass_free);
|
2016-12-26 23:18:52 +00:00
|
|
|
g_clear_pointer (&self->clip, cairo_region_destroy);
|
2016-12-08 21:35:16 +00:00
|
|
|
g_clear_object (&self->target);
|
2016-12-08 17:40:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gsk_vulkan_render_free (GskVulkanRender *self)
|
|
|
|
{
|
|
|
|
VkDevice device;
|
2023-06-29 04:36:17 +00:00
|
|
|
GHashTableIter iter;
|
|
|
|
gpointer key, value;
|
2016-12-14 08:40:15 +00:00
|
|
|
guint i;
|
2016-12-08 17:40:35 +00:00
|
|
|
|
|
|
|
gsk_vulkan_render_cleanup (self);
|
|
|
|
|
2023-07-08 07:04:12 +00:00
|
|
|
g_clear_pointer (&self->vertex_buffer, gsk_vulkan_buffer_free);
|
|
|
|
|
2016-12-08 17:40:35 +00:00
|
|
|
device = gdk_vulkan_context_get_device (self->vulkan);
|
2016-12-08 16:46:08 +00:00
|
|
|
|
2023-06-29 04:36:17 +00:00
|
|
|
g_hash_table_iter_init (&iter, self->pipeline_cache);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
|
|
|
{
|
|
|
|
g_free (key);
|
|
|
|
vkDestroyPipeline (device, value, NULL);
|
|
|
|
}
|
|
|
|
g_hash_table_unref (self->pipeline_cache);
|
|
|
|
|
2023-07-08 18:40:48 +00:00
|
|
|
g_hash_table_iter_init (&iter, self->render_pass_cache);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
|
|
|
{
|
|
|
|
g_free (key);
|
|
|
|
vkDestroyRenderPass (device, value, NULL);
|
|
|
|
}
|
|
|
|
g_hash_table_unref (self->render_pass_cache);
|
|
|
|
|
2016-12-17 03:22:44 +00:00
|
|
|
g_clear_pointer (&self->uploader, gsk_vulkan_uploader_free);
|
|
|
|
|
2023-05-20 00:56:02 +00:00
|
|
|
|
2023-05-12 18:49:31 +00:00
|
|
|
vkDestroyPipelineLayout (device,
|
|
|
|
self->pipeline_layout,
|
|
|
|
NULL);
|
2016-12-14 07:00:58 +00:00
|
|
|
|
2016-12-08 23:06:59 +00:00
|
|
|
vkDestroyDescriptorPool (device,
|
|
|
|
self->descriptor_pool,
|
|
|
|
NULL);
|
2023-05-23 03:08:13 +00:00
|
|
|
gsk_descriptor_image_infos_clear (&self->descriptor_images);
|
2023-05-25 20:21:23 +00:00
|
|
|
gsk_descriptor_buffer_infos_clear (&self->descriptor_buffers);
|
2016-12-08 23:06:59 +00:00
|
|
|
|
2023-06-20 13:39:29 +00:00
|
|
|
for (i = 0; i < N_DESCRIPTOR_SETS; i++)
|
|
|
|
vkDestroyDescriptorSetLayout (device,
|
|
|
|
self->descriptor_set_layouts[i],
|
|
|
|
NULL);
|
2017-09-22 14:02:27 +00:00
|
|
|
|
2016-12-08 17:03:05 +00:00
|
|
|
vkDestroyFence (device,
|
|
|
|
self->fence,
|
|
|
|
NULL);
|
2016-12-16 05:10:24 +00:00
|
|
|
|
2023-05-23 03:08:13 +00:00
|
|
|
for (i = 0; i < G_N_ELEMENTS (self->samplers); i++)
|
|
|
|
{
|
|
|
|
vkDestroySampler (device,
|
|
|
|
self->samplers[i],
|
|
|
|
NULL);
|
|
|
|
}
|
2017-09-30 03:01:34 +00:00
|
|
|
|
2016-12-16 05:10:24 +00:00
|
|
|
gsk_vulkan_command_pool_free (self->command_pool);
|
2016-12-08 17:03:05 +00:00
|
|
|
|
2023-03-03 11:48:49 +00:00
|
|
|
g_free (self);
|
2016-12-07 13:50:52 +00:00
|
|
|
}
|
2016-12-08 16:55:32 +00:00
|
|
|
|
2016-12-08 17:40:35 +00:00
|
|
|
gboolean
|
|
|
|
gsk_vulkan_render_is_busy (GskVulkanRender *self)
|
|
|
|
{
|
|
|
|
return vkGetFenceStatus (gdk_vulkan_context_get_device (self->vulkan), self->fence) != VK_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2016-12-22 18:01:07 +00:00
|
|
|
gsk_vulkan_render_reset (GskVulkanRender *self,
|
|
|
|
GskVulkanImage *target,
|
2018-03-28 13:01:34 +00:00
|
|
|
const graphene_rect_t *rect,
|
2023-07-05 05:06:17 +00:00
|
|
|
const cairo_region_t *clip,
|
|
|
|
GskRenderNode *node)
|
2016-12-08 17:40:35 +00:00
|
|
|
{
|
|
|
|
gsk_vulkan_render_cleanup (self);
|
|
|
|
|
2018-03-28 13:01:34 +00:00
|
|
|
gsk_vulkan_render_setup (self, target, rect, clip);
|
2023-07-05 05:06:17 +00:00
|
|
|
|
|
|
|
gsk_vulkan_render_add_node (self, node);
|
2016-12-08 17:40:35 +00:00
|
|
|
}
|
|
|
|
|
2016-12-08 16:55:32 +00:00
|
|
|
GskRenderer *
|
|
|
|
gsk_vulkan_render_get_renderer (GskVulkanRender *self)
|
|
|
|
{
|
|
|
|
return self->renderer;
|
|
|
|
}
|