vulkan: add a command pool object for tracking

This commit is contained in:
Matthew Waters 2019-04-10 14:27:26 +10:00
parent 2d0cab20c0
commit 32d217a9df
13 changed files with 240 additions and 62 deletions

View file

@ -285,6 +285,10 @@ _vulkan_swapper_retrieve_surface_properties (GstVulkanSwapper * swapper,
if (data.graphics_queue) if (data.graphics_queue)
gst_object_unref (data.graphics_queue); gst_object_unref (data.graphics_queue);
if (!(swapper->cmd_pool =
gst_vulkan_queue_create_command_pool (swapper->queue, error)))
return FALSE;
err = err =
swapper->GetPhysicalDeviceSurfaceCapabilitiesKHR (gpu, swapper->surface, swapper->GetPhysicalDeviceSurfaceCapabilitiesKHR (gpu, swapper->surface,
&swapper->surf_props); &swapper->surf_props);
@ -359,6 +363,10 @@ gst_vulkan_swapper_finalize (GObject * object)
NULL); NULL);
swapper->swap_chain = VK_NULL_HANDLE; swapper->swap_chain = VK_NULL_HANDLE;
if (swapper->cmd_pool)
gst_object_unref (swapper->cmd_pool);
swapper->cmd_pool = NULL;
if (swapper->queue) if (swapper->queue)
gst_object_unref (swapper->queue); gst_object_unref (swapper->queue);
swapper->queue = NULL; swapper->queue = NULL;
@ -505,7 +513,7 @@ _swapper_set_image_layout (GstVulkanSwapper * swapper,
GstVulkanFence *fence = NULL; GstVulkanFence *fence = NULL;
VkResult err; VkResult err;
if (!gst_vulkan_device_create_cmd_buffer (swapper->device, &cmd, error)) if (!(cmd = gst_vulkan_command_pool_create (swapper->cmd_pool, error)))
goto error; goto error;
fence = gst_vulkan_fence_new (swapper->device, 0, error); fence = gst_vulkan_fence_new (swapper->device, 0, error);
@ -565,7 +573,7 @@ _swapper_set_image_layout (GstVulkanSwapper * swapper,
} }
swapper->priv->trash_list = g_list_prepend (swapper->priv->trash_list, swapper->priv->trash_list = g_list_prepend (swapper->priv->trash_list,
gst_vulkan_trash_new_free_command_buffer (fence, cmd)); gst_vulkan_trash_new_free_command_buffer (fence, swapper->cmd_pool, cmd));
fence = NULL; fence = NULL;
return TRUE; return TRUE;
@ -641,8 +649,8 @@ _allocate_swapchain (GstVulkanSwapper * swapper, GstCaps * caps,
n_images_wanted = swapper->surf_props.maxImageCount; n_images_wanted = swapper->surf_props.maxImageCount;
} }
if (swapper->surf_props. if (swapper->
supportedTransforms & VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR) { surf_props.supportedTransforms & VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR) {
preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR; preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
} else { } else {
preTransform = swapper->surf_props.currentTransform; preTransform = swapper->surf_props.currentTransform;
@ -673,8 +681,8 @@ _allocate_swapchain (GstVulkanSwapper * swapper, GstCaps * caps,
"Incorrect usage flags available for the swap images"); "Incorrect usage flags available for the swap images");
return FALSE; return FALSE;
} }
if ((swapper-> if ((swapper->surf_props.
surf_props.supportedUsageFlags & VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT) supportedUsageFlags & VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT)
!= 0) { != 0) {
usage |= VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT; usage |= VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
} else { } else {
@ -804,7 +812,7 @@ _build_render_buffer_cmd (GstVulkanSwapper * swapper, guint32 swap_idx,
g_return_val_if_fail (swap_idx < swapper->n_swap_chain_images, FALSE); g_return_val_if_fail (swap_idx < swapper->n_swap_chain_images, FALSE);
swap_mem = swapper->swap_chain_images[swap_idx]; swap_mem = swapper->swap_chain_images[swap_idx];
if (!gst_vulkan_device_create_cmd_buffer (swapper->device, &cmd, error)) if (!(cmd = gst_vulkan_command_pool_create (swapper->cmd_pool, error)))
return FALSE; return FALSE;
buf_mem = (GstVulkanBufferMemory *) gst_buffer_peek_memory (buffer, 0); buf_mem = (GstVulkanBufferMemory *) gst_buffer_peek_memory (buffer, 0);
@ -967,7 +975,7 @@ reacquire:
swapper->priv->trash_list = g_list_prepend (swapper->priv->trash_list, swapper->priv->trash_list = g_list_prepend (swapper->priv->trash_list,
gst_vulkan_trash_new_free_command_buffer (gst_vulkan_fence_ref (fence), gst_vulkan_trash_new_free_command_buffer (gst_vulkan_fence_ref (fence),
cmd)); swapper->cmd_pool, cmd));
swapper->priv->trash_list = g_list_prepend (swapper->priv->trash_list, swapper->priv->trash_list = g_list_prepend (swapper->priv->trash_list,
gst_vulkan_trash_new_free_semaphore (fence, acquire_semaphore)); gst_vulkan_trash_new_free_semaphore (fence, acquire_semaphore));
@ -1027,7 +1035,7 @@ error:
if (present_semaphore) if (present_semaphore)
vkDestroySemaphore (swapper->device->device, present_semaphore, NULL); vkDestroySemaphore (swapper->device->device, present_semaphore, NULL);
if (cmd) if (cmd)
vkFreeCommandBuffers (swapper->device->device, swapper->device->cmd_pool, vkFreeCommandBuffers (swapper->device->device, swapper->cmd_pool->pool,
1, &cmd); 1, &cmd);
return FALSE; return FALSE;
} }
@ -1066,6 +1074,7 @@ _on_window_draw (GstVulkanWindow * window, GstVulkanSwapper * swapper)
RENDER_LOCK (swapper); RENDER_LOCK (swapper);
if (!swapper->current_buffer) { if (!swapper->current_buffer) {
GST_DEBUG_OBJECT (swapper, "No buffer to render");
RENDER_UNLOCK (swapper); RENDER_UNLOCK (swapper);
return; return;
} }

View file

@ -48,6 +48,7 @@ struct _GstVulkanSwapper
GstVulkanDevice *device; GstVulkanDevice *device;
GstVulkanWindow *window; GstVulkanWindow *window;
GstVulkanQueue *queue; GstVulkanQueue *queue;
GstVulkanCommandPool *cmd_pool;
VkSurfaceKHR surface; VkSurfaceKHR surface;

View file

@ -138,26 +138,34 @@ gst_vulkan_trash_list_wait (GList * trash_list, guint64 timeout)
return err == VK_SUCCESS; return err == VK_SUCCESS;
} }
static void struct cmd_buffer_free_info
_free_command_buffer (GstVulkanDevice * device, VkCommandBuffer * cmd)
{ {
g_assert (cmd); GstVulkanCommandPool *pool;
vkFreeCommandBuffers (device->device, device->cmd_pool, 1, cmd); VkCommandBuffer cmd;
};
g_free (cmd); static void
_free_command_buffer (GstVulkanDevice * device,
struct cmd_buffer_free_info *info)
{
vkFreeCommandBuffers (device->device, info->pool->pool, 1, &info->cmd);
gst_object_unref (info->pool);
g_free (info);
} }
GstVulkanTrash * GstVulkanTrash *
gst_vulkan_trash_new_free_command_buffer (GstVulkanFence * fence, gst_vulkan_trash_new_free_command_buffer (GstVulkanFence * fence,
VkCommandBuffer cmd) GstVulkanCommandPool * pool, VkCommandBuffer cmd)
{ {
VkCommandBuffer *data; struct cmd_buffer_free_info *info;
GstVulkanTrash *trash; GstVulkanTrash *trash;
data = g_new0 (VkCommandBuffer, 1); info = g_new0 (struct cmd_buffer_free_info, 1);
*data = cmd; info->pool = gst_object_ref (pool);
info->cmd = cmd;
trash = gst_vulkan_trash_new (fence, trash = gst_vulkan_trash_new (fence,
(GstVulkanTrashNotify) _free_command_buffer, data); (GstVulkanTrashNotify) _free_command_buffer, info);
return trash; return trash;
} }

View file

@ -41,6 +41,7 @@ GstVulkanTrash * gst_vulkan_trash_new (GstVulkanFence
GstVulkanTrashNotify notify, GstVulkanTrashNotify notify,
gpointer user_data); gpointer user_data);
GstVulkanTrash * gst_vulkan_trash_new_free_command_buffer (GstVulkanFence * fence, GstVulkanTrash * gst_vulkan_trash_new_free_command_buffer (GstVulkanFence * fence,
GstVulkanCommandPool * pool,
VkCommandBuffer cmd); VkCommandBuffer cmd);
GstVulkanTrash * gst_vulkan_trash_new_free_semaphore (GstVulkanFence * fence, GstVulkanTrash * gst_vulkan_trash_new_free_semaphore (GstVulkanFence * fence,
VkSemaphore cmd); VkSemaphore cmd);

View file

@ -0,0 +1,96 @@
/*
* GStreamer
* Copyright (C) 2019 Matthew Waters <matthew@centricular.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "gstvkcommandpool.h"
#define GST_CAT_DEFAULT gst_vulkan_command_pool_debug
GST_DEBUG_CATEGORY (GST_CAT_DEFAULT);
#define parent_class gst_vulkan_command_pool_parent_class
G_DEFINE_TYPE_WITH_CODE (GstVulkanCommandPool, gst_vulkan_command_pool,
GST_TYPE_OBJECT, GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT,
"vulkancommandpool", 0, "Vulkan Command Pool"));
static void gst_vulkan_command_pool_dispose (GObject * object);
static void
gst_vulkan_command_pool_init (GstVulkanCommandPool * device)
{
}
static void
gst_vulkan_command_pool_class_init (GstVulkanCommandPoolClass * device_class)
{
GObjectClass *gobject_class = (GObjectClass *) device_class;
gobject_class->dispose = gst_vulkan_command_pool_dispose;
}
static void
gst_vulkan_command_pool_dispose (GObject * object)
{
GstVulkanCommandPool *pool = GST_VULKAN_COMMAND_POOL (object);
if (pool->pool)
vkDestroyCommandPool (pool->queue->device->device, pool->pool, NULL);
pool->pool = VK_NULL_HANDLE;
if (pool->queue)
gst_object_unref (pool->queue);
pool->queue = NULL;
G_OBJECT_CLASS (parent_class)->dispose (object);
}
GstVulkanQueue *
gst_vulkan_command_pool_get_queue (GstVulkanCommandPool * pool)
{
g_return_val_if_fail (GST_IS_VULKAN_COMMAND_POOL (pool), NULL);
return pool->queue ? gst_object_ref (pool->queue) : NULL;
}
VkCommandBuffer
gst_vulkan_command_pool_create (GstVulkanCommandPool * pool, GError ** error)
{
VkResult err;
VkCommandBufferAllocateInfo cmd_info = { 0, };
VkCommandBuffer cmd;
g_return_val_if_fail (GST_IS_VULKAN_COMMAND_POOL (pool), NULL);
cmd_info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;
cmd_info.pNext = NULL;
cmd_info.commandPool = pool->pool;
cmd_info.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;
cmd_info.commandBufferCount = 1;
err = vkAllocateCommandBuffers (pool->queue->device->device, &cmd_info, &cmd);
if (gst_vulkan_error_to_g_error (err, error, "vkCreateCommandBuffer") < 0)
return NULL;
GST_LOG_OBJECT (pool, "created cmd buffer %p", cmd);
return cmd;
}

View file

@ -0,0 +1,56 @@
/*
* GStreamer
* Copyright (C) 2015 Matthew Waters <matthew@centricular.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifndef __GST_VULKAN_COMMAND_POOL_H__
#define __GST_VULKAN_COMMAND_POOL_H__
#include <gst/vulkan/vulkan.h>
#define GST_TYPE_VULKAN_COMMAND_POOL (gst_vulkan_command_pool_get_type())
#define GST_VULKAN_COMMAND_POOL(o) (G_TYPE_CHECK_INSTANCE_CAST((o), GST_TYPE_VULKAN_COMMAND_POOL, GstVulkanCommandPool))
#define GST_VULKAN_COMMAND_POOL_CLASS(k) (G_TYPE_CHECK_CLASS_CAST((k), GST_TYPE_VULKAN_COMMAND_POOL, GstVulkanCommandPoolClass))
#define GST_IS_VULKAN_COMMAND_POOL(o) (G_TYPE_CHECK_INSTANCE_TYPE((o), GST_TYPE_VULKAN_COMMAND_POOL))
#define GST_IS_VULKAN_COMMAND_POOL_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE((k), GST_TYPE_VULKAN_COMMAND_POOL))
#define GST_VULKAN_COMMAND_POOL_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS((o), GST_TYPE_VULKAN_COMMAND_POOL, GstVulkanCommandPoolClass))
GST_VULKAN_API
GType gst_vulkan_command_pool_get_type (void);
struct _GstVulkanCommandPool
{
GstObject parent;
GstVulkanQueue *queue;
VkCommandPool pool; /* hides a pointer */
};
struct _GstVulkanCommandPoolClass
{
GstObjectClass parent_class;
};
GST_VULKAN_API
GstVulkanQueue * gst_vulkan_command_pool_get_queue (GstVulkanCommandPool * pool);
GST_VULKAN_API
VkCommandBuffer gst_vulkan_command_pool_create (GstVulkanCommandPool * pool,
GError ** error);
#endif /* __GST_VULKAN_COMMAND_POOL_H__ */

View file

@ -80,10 +80,6 @@ gst_vulkan_device_finalize (GObject * object)
g_free (device->queue_family_props); g_free (device->queue_family_props);
device->queue_family_props = NULL; device->queue_family_props = NULL;
if (device->cmd_pool)
vkDestroyCommandPool (device->device, device->cmd_pool, NULL);
device->cmd_pool = VK_NULL_HANDLE;
if (device->device) { if (device->device) {
vkDeviceWaitIdle (device->device); vkDeviceWaitIdle (device->device);
vkDestroyDevice (device->device, NULL); vkDestroyDevice (device->device, NULL);
@ -284,21 +280,6 @@ gst_vulkan_device_open (GstVulkanDevice * device, GError ** error)
} }
} }
{
VkCommandPoolCreateInfo cmd_pool_info = { 0, };
cmd_pool_info.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO;
cmd_pool_info.pNext = NULL;
cmd_pool_info.queueFamilyIndex = device->queue_family_id;
cmd_pool_info.flags = 0;
err =
vkCreateCommandPool (device->device, &cmd_pool_info, NULL,
&device->cmd_pool);
if (gst_vulkan_error_to_g_error (err, error, "vkCreateCommandPool") < 0)
goto error;
}
GST_OBJECT_UNLOCK (device); GST_OBJECT_UNLOCK (device);
return TRUE; return TRUE;
@ -387,28 +368,6 @@ gst_vulkan_device_get_physical_device (GstVulkanDevice * device)
return device->instance->physical_devices[device->device_index]; return device->instance->physical_devices[device->device_index];
} }
gboolean
gst_vulkan_device_create_cmd_buffer (GstVulkanDevice * device,
VkCommandBuffer * cmd, GError ** error)
{
VkResult err;
VkCommandBufferAllocateInfo cmd_info = { 0, };
cmd_info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;
cmd_info.pNext = NULL;
cmd_info.commandPool = device->cmd_pool;
cmd_info.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;
cmd_info.commandBufferCount = 1;
err = vkAllocateCommandBuffers (device->device, &cmd_info, cmd);
if (gst_vulkan_error_to_g_error (err, error, "vkCreateCommandBuffer") < 0)
return FALSE;
GST_LOG_OBJECT (device, "created cmd buffer %p", cmd);
return TRUE;
}
/** /**
* gst_context_set_vulkan_device: * gst_context_set_vulkan_device:
* @context: a #GstContext * @context: a #GstContext

View file

@ -58,8 +58,6 @@ struct _GstVulkanDevice
guint32 queue_family_id; guint32 queue_family_id;
guint32 n_queues; guint32 n_queues;
VkCommandPool cmd_pool;
GstVulkanDevicePrivate *priv; GstVulkanDevicePrivate *priv;
}; };

View file

@ -28,6 +28,7 @@
GST_DEBUG_CATEGORY (GST_CAT_DEFAULT); GST_DEBUG_CATEGORY (GST_CAT_DEFAULT);
GST_DEBUG_CATEGORY_STATIC (GST_CAT_CONTEXT); GST_DEBUG_CATEGORY_STATIC (GST_CAT_CONTEXT);
#define parent_class gst_vulkan_queue_parent_class
G_DEFINE_TYPE_WITH_CODE (GstVulkanQueue, gst_vulkan_queue, GST_TYPE_OBJECT, G_DEFINE_TYPE_WITH_CODE (GstVulkanQueue, gst_vulkan_queue, GST_TYPE_OBJECT,
GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT, "vulkanqueue", 0, GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT, "vulkanqueue", 0,
"Vulkan Queue"); "Vulkan Queue");
@ -56,6 +57,8 @@ gst_vulkan_queue_dispose (GObject * object)
if (queue->device) if (queue->device)
gst_object_unref (queue->device); gst_object_unref (queue->device);
queue->device = NULL; queue->device = NULL;
G_OBJECT_CLASS (parent_class)->dispose (object);
} }
GstVulkanDevice * GstVulkanDevice *
@ -66,6 +69,42 @@ gst_vulkan_queue_get_device (GstVulkanQueue * queue)
return queue->device ? gst_object_ref (queue->device) : NULL; return queue->device ? gst_object_ref (queue->device) : NULL;
} }
GstVulkanCommandPool *
gst_vulkan_queue_create_command_pool (GstVulkanQueue * queue, GError ** error)
{
GstVulkanCommandPool *pool;
VkCommandPoolCreateInfo cmd_pool_info = { 0, };
VkCommandPool vk_pool;
VkResult err;
g_return_val_if_fail (GST_IS_VULKAN_QUEUE (queue), NULL);
cmd_pool_info.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO;
cmd_pool_info.pNext = NULL;
cmd_pool_info.queueFamilyIndex = queue->family;
cmd_pool_info.flags = 0;
GST_OBJECT_LOCK (queue->device);
err =
vkCreateCommandPool (queue->device->device, &cmd_pool_info, NULL,
&vk_pool);
if (gst_vulkan_error_to_g_error (err, error, "vkCreateCommandPool") < 0) {
GST_OBJECT_LOCK (queue->device);
goto error;
}
GST_OBJECT_UNLOCK (queue->device);
pool = g_object_new (GST_TYPE_VULKAN_COMMAND_POOL, NULL);
gst_object_ref_sink (pool);
pool->queue = gst_object_ref (queue);
pool->pool = vk_pool;
return pool;
error:
return NULL;
}
/** /**
* gst_context_set_vulkan_queue: * gst_context_set_vulkan_queue:
* @context: a #GstContext * @context: a #GstContext

View file

@ -53,6 +53,10 @@ struct _GstVulkanQueueClass
GST_VULKAN_API GST_VULKAN_API
GstVulkanDevice * gst_vulkan_queue_get_device (GstVulkanQueue * queue); GstVulkanDevice * gst_vulkan_queue_get_device (GstVulkanQueue * queue);
GST_VULKAN_API
GstVulkanCommandPool * gst_vulkan_queue_create_command_pool (GstVulkanQueue * queue,
GError ** error);
GST_VULKAN_API GST_VULKAN_API
void gst_context_set_vulkan_queue (GstContext * context, void gst_context_set_vulkan_queue (GstContext * context,
GstVulkanQueue * queue); GstVulkanQueue * queue);

View file

@ -6,6 +6,7 @@ endif
vulkan_sources = [ vulkan_sources = [
'gstvkbuffermemory.c', 'gstvkbuffermemory.c',
'gstvkbufferpool.c', 'gstvkbufferpool.c',
'gstvkcommandpool.c',
'gstvkdevice.c', 'gstvkdevice.c',
'gstvkdisplay.c', 'gstvkdisplay.c',
'gstvkerror.c', 'gstvkerror.c',
@ -21,6 +22,7 @@ vulkan_sources = [
vulkan_headers = [ vulkan_headers = [
'gstvkbuffermemory.h', 'gstvkbuffermemory.h',
'gstvkbufferpool.h', 'gstvkbufferpool.h',
'gstvkcommandpool.h',
'gstvkdevice.h', 'gstvkdevice.h',
'gstvkdisplay.h', 'gstvkdisplay.h',
'gstvkerror.h', 'gstvkerror.h',

View file

@ -38,5 +38,6 @@
#include <gst/vulkan/gstvkimagememory.h> #include <gst/vulkan/gstvkimagememory.h>
#include <gst/vulkan/gstvkbufferpool.h> #include <gst/vulkan/gstvkbufferpool.h>
#include <gst/vulkan/gstvkutils.h> #include <gst/vulkan/gstvkutils.h>
#include <gst/vulkan/gstvkcommandpool.h>
#endif /* __GST_VULKAN_H__ */ #endif /* __GST_VULKAN_H__ */

View file

@ -36,6 +36,10 @@ typedef struct _GstVulkanDevicePrivate GstVulkanDevicePrivate;
typedef struct _GstVulkanQueue GstVulkanQueue; typedef struct _GstVulkanQueue GstVulkanQueue;
typedef struct _GstVulkanQueueClass GstVulkanQueueClass; typedef struct _GstVulkanQueueClass GstVulkanQueueClass;
typedef struct _GstVulkanCommandPool GstVulkanCommandPool;
typedef struct _GstVulkanCommandPoolClass GstVulkanCommandPoolClass;
typedef struct _GstVulkanCommandPoolPrivate GstVulkanCommandPoolPrivate;
typedef enum _GstVulkanDisplayType GstVulkanDisplayType; typedef enum _GstVulkanDisplayType GstVulkanDisplayType;
typedef struct _GstVulkanDisplay GstVulkanDisplay; typedef struct _GstVulkanDisplay GstVulkanDisplay;