mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-04 14:38:48 +00:00
2d31aba78d
Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/merge_requests/1506>
1491 lines
46 KiB
C
1491 lines
46 KiB
C
/*
|
|
* 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.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <string.h>
|
|
|
|
#include "gstvkswapper.h"
|
|
|
|
/**
|
|
* SECTION:vkswapper
|
|
* @title: GstVulkanSwapper
|
|
* @short_description: Vulkan helper object for rendering to a surface
|
|
* @see_also: #GstVulkanWindow, #GstVulkanQueue
|
|
*
|
|
* #GstVulkanSwapper is a helper object for rendering to a surface exposed by
|
|
* #GstVulkanWindow.
|
|
*/
|
|
|
|
#define GST_CAT_DEFAULT gst_vulkan_swapper_debug
|
|
GST_DEBUG_CATEGORY (GST_CAT_DEFAULT);
|
|
|
|
struct _GstVulkanSwapperPrivate
|
|
{
|
|
VkSurfaceKHR surface;
|
|
|
|
VkSurfaceCapabilitiesKHR surf_props;
|
|
VkSurfaceFormatKHR *surf_formats;
|
|
guint32 n_surf_formats;
|
|
VkPresentModeKHR *surf_present_modes;
|
|
guint32 n_surf_present_modes;
|
|
|
|
VkSwapchainKHR swap_chain;
|
|
GstVulkanImageMemory **swap_chain_images;
|
|
guint32 n_swap_chain_images;
|
|
|
|
GstCaps *caps;
|
|
GstVideoInfo v_info;
|
|
|
|
PFN_vkGetPhysicalDeviceSurfaceSupportKHR GetPhysicalDeviceSurfaceSupportKHR;
|
|
PFN_vkGetPhysicalDeviceSurfaceCapabilitiesKHR
|
|
GetPhysicalDeviceSurfaceCapabilitiesKHR;
|
|
PFN_vkGetPhysicalDeviceSurfaceFormatsKHR GetPhysicalDeviceSurfaceFormatsKHR;
|
|
PFN_vkGetPhysicalDeviceSurfacePresentModesKHR
|
|
GetPhysicalDeviceSurfacePresentModesKHR;
|
|
PFN_vkCreateSwapchainKHR CreateSwapchainKHR;
|
|
PFN_vkDestroySwapchainKHR DestroySwapchainKHR;
|
|
PFN_vkGetSwapchainImagesKHR GetSwapchainImagesKHR;
|
|
PFN_vkAcquireNextImageKHR AcquireNextImageKHR;
|
|
PFN_vkQueuePresentKHR QueuePresentKHR;
|
|
PFN_vkDestroySurfaceKHR DestroySurfaceKHR;
|
|
|
|
/* <private> */
|
|
/* runtime variables */
|
|
gint to_quit;
|
|
GstBuffer *current_buffer;
|
|
gboolean any_current_extent;
|
|
|
|
/* signal handlers */
|
|
gulong close_id;
|
|
gulong draw_id;
|
|
gulong resize_id;
|
|
|
|
/* properties */
|
|
gboolean force_aspect_ratio;
|
|
gint par_n;
|
|
gint par_d;
|
|
|
|
GMutex render_lock;
|
|
|
|
GstVulkanTrashList *trash_list;
|
|
|
|
/* source sizes accounting for all aspect ratios */
|
|
guint dar_width;
|
|
guint dar_height;
|
|
|
|
GstVideoRectangle surface_location;
|
|
GstVideoRectangle display_rect;
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_FORCE_ASPECT_RATIO,
|
|
PROP_PIXEL_ASPECT_RATIO,
|
|
};
|
|
|
|
#define DEFAULT_FORCE_ASPECT_RATIO TRUE
|
|
#define DEFAULT_PIXEL_ASPECT_RATIO_N 0
|
|
#define DEFAULT_PIXEL_ASPECT_RATIO_D 1
|
|
|
|
#define GET_PRIV(swapper) gst_vulkan_swapper_get_instance_private (swapper)
|
|
|
|
#define gst_vulkan_swapper_parent_class parent_class
|
|
G_DEFINE_TYPE_WITH_CODE (GstVulkanSwapper, gst_vulkan_swapper,
|
|
GST_TYPE_OBJECT, G_ADD_PRIVATE (GstVulkanSwapper)
|
|
GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT,
|
|
"vulkanswapper", 0, "Vulkan Swapper"));
|
|
|
|
static void _on_window_draw (GstVulkanWindow * window,
|
|
GstVulkanSwapper * swapper);
|
|
static void _on_window_resize (GstVulkanWindow * window,
|
|
guint width, guint height, GstVulkanSwapper * swapper);
|
|
|
|
static inline GMutex *
|
|
render_get_lock (gpointer swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
return &priv->render_lock;
|
|
}
|
|
|
|
#define RENDER_LOCK(o) g_mutex_lock (render_get_lock(o));
|
|
#define RENDER_UNLOCK(o) g_mutex_unlock (render_get_lock(o));
|
|
|
|
static gboolean
|
|
_get_function_table (GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
GstVulkanDevice *device = swapper->device;
|
|
GstVulkanInstance *instance = gst_vulkan_device_get_instance (device);
|
|
|
|
if (!instance) {
|
|
GST_ERROR_OBJECT (swapper, "Failed to get instance from the device");
|
|
return FALSE;
|
|
}
|
|
|
|
if (!gst_vulkan_device_is_extension_enabled (device,
|
|
VK_KHR_SWAPCHAIN_EXTENSION_NAME)) {
|
|
GST_ERROR_OBJECT (swapper, "Required extension \'%s\' is not enabled on "
|
|
"device %" GST_PTR_FORMAT, VK_KHR_SWAPCHAIN_EXTENSION_NAME, device);
|
|
return FALSE;
|
|
}
|
|
#define GET_PROC_ADDRESS_REQUIRED(type, name) \
|
|
G_STMT_START { \
|
|
priv->G_PASTE (, name) = G_PASTE(G_PASTE(gst_vulkan_, type), _get_proc_address) (type, "vk" G_STRINGIFY(name)); \
|
|
if (!priv->G_PASTE(, name)) { \
|
|
GST_ERROR_OBJECT (swapper, "Failed to find required function vk" G_STRINGIFY(name)); \
|
|
gst_object_unref (instance); \
|
|
return FALSE; \
|
|
} \
|
|
} G_STMT_END
|
|
|
|
GET_PROC_ADDRESS_REQUIRED (instance, GetPhysicalDeviceSurfaceSupportKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (instance, GetPhysicalDeviceSurfaceCapabilitiesKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (instance, GetPhysicalDeviceSurfaceFormatsKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (instance, GetPhysicalDeviceSurfacePresentModesKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (instance, DestroySurfaceKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (device, CreateSwapchainKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (device, DestroySwapchainKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (device, GetSwapchainImagesKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (device, AcquireNextImageKHR);
|
|
GET_PROC_ADDRESS_REQUIRED (device, QueuePresentKHR);
|
|
|
|
gst_object_unref (instance);
|
|
|
|
return TRUE;
|
|
|
|
#undef GET_PROC_ADDRESS_REQUIRED
|
|
}
|
|
|
|
static GstVideoFormat
|
|
_vk_format_to_video_format (VkFormat format)
|
|
{
|
|
switch (format) {
|
|
/* double check endianness */
|
|
case VK_FORMAT_R8G8B8A8_UNORM:
|
|
return GST_VIDEO_FORMAT_RGBA;
|
|
case VK_FORMAT_R8G8B8_UNORM:
|
|
return GST_VIDEO_FORMAT_RGB;
|
|
case VK_FORMAT_B8G8R8A8_UNORM:
|
|
return GST_VIDEO_FORMAT_BGRA;
|
|
case VK_FORMAT_B8G8R8_UNORM:
|
|
return GST_VIDEO_FORMAT_BGR;
|
|
default:
|
|
return GST_VIDEO_FORMAT_UNKNOWN;
|
|
}
|
|
}
|
|
|
|
static VkColorSpaceKHR
|
|
_vk_color_space_from_video_info (GstVideoInfo * v_info)
|
|
{
|
|
return VK_COLORSPACE_SRGB_NONLINEAR_KHR;
|
|
}
|
|
|
|
static void
|
|
_add_vk_format_to_list (GValue * list, VkFormat format)
|
|
{
|
|
GstVideoFormat v_format;
|
|
|
|
v_format = _vk_format_to_video_format (format);
|
|
if (v_format) {
|
|
const gchar *format_str = gst_video_format_to_string (v_format);
|
|
GValue item = G_VALUE_INIT;
|
|
GValue new_list = G_VALUE_INIT;
|
|
|
|
g_value_init (&item, G_TYPE_STRING);
|
|
g_value_set_string (&item, format_str);
|
|
gst_value_list_merge (&new_list, list, &item);
|
|
g_value_unset (&item);
|
|
|
|
g_value_unset (list);
|
|
*list = new_list;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
_vulkan_swapper_ensure_surface (GstVulkanSwapper * swapper, GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
if (!priv->surface) {
|
|
if (!(priv->surface =
|
|
gst_vulkan_window_get_surface (swapper->window, error))) {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
struct choose_data
|
|
{
|
|
GstVulkanSwapper *swapper;
|
|
GstVulkanQueue *graphics_queue;
|
|
GstVulkanQueue *present_queue;
|
|
};
|
|
|
|
static gboolean
|
|
_choose_queue (GstVulkanDevice * device, GstVulkanQueue * queue,
|
|
struct choose_data *data)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (data->swapper);
|
|
guint flags =
|
|
device->physical_device->queue_family_props[queue->family].queueFlags;
|
|
VkPhysicalDevice gpu;
|
|
gboolean supports_present;
|
|
|
|
gpu = gst_vulkan_device_get_physical_device (data->swapper->device);
|
|
|
|
{
|
|
VkResult err;
|
|
GError *error = NULL;
|
|
VkBool32 physical_device_supported;
|
|
|
|
err =
|
|
priv->GetPhysicalDeviceSurfaceSupportKHR (gpu,
|
|
queue->index, priv->surface, &physical_device_supported);
|
|
if (gst_vulkan_error_to_g_error (err, &error,
|
|
"GetPhysicalDeviceSurfaceSupport") < 0) {
|
|
GST_DEBUG_OBJECT (data->swapper,
|
|
"surface not supported by the physical device: %s", error->message);
|
|
g_clear_error (&error);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
supports_present =
|
|
gst_vulkan_window_get_presentation_support (data->swapper->window,
|
|
device, queue->index);
|
|
|
|
if ((flags & VK_QUEUE_GRAPHICS_BIT) != 0) {
|
|
if (supports_present) {
|
|
/* found one that supports both */
|
|
if (data->graphics_queue)
|
|
gst_object_unref (data->graphics_queue);
|
|
data->graphics_queue = gst_object_ref (queue);
|
|
if (data->present_queue)
|
|
gst_object_unref (data->present_queue);
|
|
data->present_queue = gst_object_ref (queue);
|
|
return FALSE;
|
|
}
|
|
if (!data->graphics_queue)
|
|
data->present_queue = gst_object_ref (queue);
|
|
} else if (supports_present) {
|
|
if (!data->present_queue)
|
|
data->present_queue = gst_object_ref (queue);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* gst_vulkan_swapper_choose_queue:
|
|
* @swapper: a #GstVulkanSwapper
|
|
* @available_queue: (transfer none): a #GstVulkanQueue chosen elsewhere
|
|
* @error: a #GError
|
|
*/
|
|
gboolean
|
|
gst_vulkan_swapper_choose_queue (GstVulkanSwapper * swapper,
|
|
GstVulkanQueue * available_queue, GError ** error)
|
|
{
|
|
if (!_vulkan_swapper_ensure_surface (swapper, error))
|
|
return FALSE;
|
|
|
|
if (swapper->queue)
|
|
return TRUE;
|
|
|
|
if (available_queue) {
|
|
guint flags =
|
|
swapper->device->physical_device->
|
|
queue_family_props[available_queue->family].queueFlags;
|
|
gboolean supports_present;
|
|
|
|
supports_present =
|
|
gst_vulkan_window_get_presentation_support (swapper->window,
|
|
swapper->device, available_queue->index);
|
|
if (supports_present && flags & VK_QUEUE_GRAPHICS_BIT)
|
|
swapper->queue = gst_object_ref (available_queue);
|
|
}
|
|
|
|
if (!swapper->queue) {
|
|
struct choose_data data;
|
|
|
|
data.swapper = swapper;
|
|
data.present_queue = NULL;
|
|
data.graphics_queue = NULL;
|
|
|
|
gst_vulkan_device_foreach_queue (swapper->device,
|
|
(GstVulkanDeviceForEachQueueFunc) _choose_queue, &data);
|
|
|
|
if (data.graphics_queue != data.present_queue) {
|
|
/* FIXME: add support for separate graphics/present queues */
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED,
|
|
"Failed to find a compatible present/graphics queue");
|
|
if (data.present_queue)
|
|
gst_object_unref (data.present_queue);
|
|
if (data.graphics_queue)
|
|
gst_object_unref (data.graphics_queue);
|
|
return FALSE;
|
|
}
|
|
|
|
swapper->queue = gst_object_ref (data.present_queue);
|
|
if (data.present_queue)
|
|
gst_object_unref (data.present_queue);
|
|
if (data.graphics_queue)
|
|
gst_object_unref (data.graphics_queue);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
dump_surface_properties (GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
GST_TRACE_OBJECT (swapper, "surface %"
|
|
GST_VULKAN_NON_DISPATCHABLE_HANDLE_FORMAT ", n images [%"
|
|
G_GUINT32_FORMAT ", %" G_GUINT32_FORMAT "], extent [%"
|
|
GST_VULKAN_EXTENT2D_FORMAT ", %" GST_VULKAN_EXTENT2D_FORMAT
|
|
"], max layers %" G_GUINT32_FORMAT " transforms supported 0x%x "
|
|
"current transform 0x%x, alpha flags 0x%x, "
|
|
"supported image usage flags 0x%x", priv->surface,
|
|
priv->surf_props.minImageCount,
|
|
priv->surf_props.maxImageCount,
|
|
GST_VULKAN_EXTENT2D_ARGS (priv->surf_props.minImageExtent),
|
|
GST_VULKAN_EXTENT2D_ARGS (priv->surf_props.maxImageExtent),
|
|
priv->surf_props.maxImageArrayLayers,
|
|
priv->surf_props.supportedTransforms,
|
|
priv->surf_props.currentTransform,
|
|
priv->surf_props.supportedCompositeAlpha,
|
|
priv->surf_props.supportedUsageFlags);
|
|
}
|
|
|
|
static void
|
|
dump_surface_formats (GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
int i;
|
|
|
|
for (i = 0; i < priv->n_surf_formats; i++) {
|
|
GST_DEBUG_OBJECT (swapper, "surface %"
|
|
GST_VULKAN_NON_DISPATCHABLE_HANDLE_FORMAT
|
|
" format 0x%x colorspace 0x%x", priv->surface,
|
|
priv->surf_formats[i].format, priv->surf_formats[i].colorSpace);
|
|
}
|
|
}
|
|
|
|
static void
|
|
dump_surface_present_modes (GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
int i;
|
|
|
|
for (i = 0; i < priv->n_surf_present_modes; i++) {
|
|
GST_DEBUG_OBJECT (swapper, "surface %"
|
|
GST_VULKAN_NON_DISPATCHABLE_HANDLE_FORMAT " present modes 0x%x",
|
|
priv->surface, priv->surf_present_modes[i]);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
_vulkan_swapper_retrieve_surface_properties (GstVulkanSwapper * swapper,
|
|
GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
VkPhysicalDevice gpu;
|
|
VkResult err;
|
|
|
|
if (priv->surf_formats)
|
|
return TRUE;
|
|
|
|
gpu = gst_vulkan_device_get_physical_device (swapper->device);
|
|
|
|
if (!gst_vulkan_swapper_choose_queue (swapper, NULL, error))
|
|
return FALSE;
|
|
|
|
if (!(swapper->cmd_pool =
|
|
gst_vulkan_queue_create_command_pool (swapper->queue, error)))
|
|
return FALSE;
|
|
|
|
err =
|
|
priv->GetPhysicalDeviceSurfaceCapabilitiesKHR (gpu,
|
|
priv->surface, &priv->surf_props);
|
|
if (gst_vulkan_error_to_g_error (err, error,
|
|
"GetPhysicalDeviceSurfaceCapabilitiesKHR") < 0)
|
|
return FALSE;
|
|
|
|
dump_surface_properties (swapper);
|
|
|
|
err =
|
|
priv->GetPhysicalDeviceSurfaceFormatsKHR (gpu,
|
|
priv->surface, &priv->n_surf_formats, NULL);
|
|
if (gst_vulkan_error_to_g_error (err, error,
|
|
"GetPhysicalDeviceSurfaceFormatsKHR") < 0)
|
|
return FALSE;
|
|
|
|
priv->surf_formats = g_new0 (VkSurfaceFormatKHR, priv->n_surf_formats);
|
|
err =
|
|
priv->GetPhysicalDeviceSurfaceFormatsKHR (gpu,
|
|
priv->surface, &priv->n_surf_formats, priv->surf_formats);
|
|
if (gst_vulkan_error_to_g_error (err, error,
|
|
"GetPhysicalDeviceSurfaceFormatsKHR") < 0)
|
|
return FALSE;
|
|
|
|
dump_surface_formats (swapper);
|
|
|
|
err =
|
|
priv->GetPhysicalDeviceSurfacePresentModesKHR (gpu,
|
|
priv->surface, &priv->n_surf_present_modes, NULL);
|
|
if (gst_vulkan_error_to_g_error (err, error,
|
|
"GetPhysicalDeviceSurfacePresentModesKHR") < 0)
|
|
return FALSE;
|
|
|
|
priv->surf_present_modes =
|
|
g_new0 (VkPresentModeKHR, priv->n_surf_present_modes);
|
|
err =
|
|
priv->GetPhysicalDeviceSurfacePresentModesKHR (gpu,
|
|
priv->surface, &priv->n_surf_present_modes, priv->surf_present_modes);
|
|
if (gst_vulkan_error_to_g_error (err, error,
|
|
"GetPhysicalDeviceSurfacePresentModesKHR") < 0)
|
|
return FALSE;
|
|
|
|
dump_surface_present_modes (swapper);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_on_window_close (GstVulkanWindow * window, GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
g_atomic_int_set (&priv->to_quit, 1);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gst_vulkan_swapper_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstVulkanSwapper *swapper = GST_VULKAN_SWAPPER (object);
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
switch (prop_id) {
|
|
case PROP_FORCE_ASPECT_RATIO:
|
|
priv->force_aspect_ratio = g_value_get_boolean (value);
|
|
break;
|
|
case PROP_PIXEL_ASPECT_RATIO:
|
|
priv->par_n = gst_value_get_fraction_numerator (value);
|
|
priv->par_d = gst_value_get_fraction_denominator (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_vulkan_swapper_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstVulkanSwapper *swapper = GST_VULKAN_SWAPPER (object);
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
switch (prop_id) {
|
|
case PROP_FORCE_ASPECT_RATIO:
|
|
g_value_set_boolean (value, priv->force_aspect_ratio);
|
|
break;
|
|
case PROP_PIXEL_ASPECT_RATIO:
|
|
gst_value_set_fraction (value, priv->par_n, priv->par_d);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_vulkan_swapper_finalize (GObject * object)
|
|
{
|
|
GstVulkanSwapper *swapper = GST_VULKAN_SWAPPER (object);
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
GstVulkanInstance *instance =
|
|
gst_vulkan_device_get_instance (swapper->device);
|
|
int i;
|
|
|
|
g_signal_handler_disconnect (swapper->window, priv->draw_id);
|
|
priv->draw_id = 0;
|
|
|
|
g_signal_handler_disconnect (swapper->window, priv->close_id);
|
|
priv->close_id = 0;
|
|
|
|
g_signal_handler_disconnect (swapper->window, priv->resize_id);
|
|
priv->resize_id = 0;
|
|
|
|
if (!gst_vulkan_trash_list_wait (priv->trash_list, -1))
|
|
GST_WARNING_OBJECT (swapper, "Failed to wait for all fences to complete "
|
|
"before shutting down");
|
|
gst_object_unref (priv->trash_list);
|
|
priv->trash_list = NULL;
|
|
|
|
if (priv->swap_chain_images) {
|
|
for (i = 0; i < priv->n_swap_chain_images; i++) {
|
|
gst_memory_unref ((GstMemory *) priv->swap_chain_images[i]);
|
|
priv->swap_chain_images[i] = NULL;
|
|
}
|
|
g_free (priv->swap_chain_images);
|
|
}
|
|
priv->swap_chain_images = NULL;
|
|
|
|
if (priv->swap_chain)
|
|
priv->DestroySwapchainKHR (swapper->device->device, priv->swap_chain, NULL);
|
|
priv->swap_chain = VK_NULL_HANDLE;
|
|
|
|
if (priv->surface) {
|
|
priv->DestroySurfaceKHR (instance->instance, priv->surface, NULL);
|
|
}
|
|
priv->surface = VK_NULL_HANDLE;
|
|
|
|
g_free (priv->surf_present_modes);
|
|
priv->surf_present_modes = NULL;
|
|
|
|
g_free (priv->surf_formats);
|
|
priv->surf_formats = NULL;
|
|
|
|
gst_buffer_replace (&priv->current_buffer, NULL);
|
|
gst_caps_replace (&priv->caps, NULL);
|
|
|
|
g_mutex_clear (&priv->render_lock);
|
|
|
|
if (swapper->cmd_pool)
|
|
gst_object_unref (swapper->cmd_pool);
|
|
swapper->cmd_pool = NULL;
|
|
|
|
if (swapper->queue)
|
|
gst_object_unref (swapper->queue);
|
|
swapper->queue = NULL;
|
|
|
|
if (swapper->device)
|
|
gst_object_unref (swapper->device);
|
|
swapper->device = NULL;
|
|
|
|
if (swapper->window)
|
|
gst_object_unref (swapper->window);
|
|
swapper->window = NULL;
|
|
|
|
gst_object_unref (instance);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gst_vulkan_swapper_init (GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
g_mutex_init (&priv->render_lock);
|
|
|
|
priv->force_aspect_ratio = DEFAULT_FORCE_ASPECT_RATIO;
|
|
priv->par_n = DEFAULT_PIXEL_ASPECT_RATIO_N;
|
|
priv->par_d = DEFAULT_PIXEL_ASPECT_RATIO_D;
|
|
|
|
priv->trash_list = gst_vulkan_trash_fence_list_new ();
|
|
}
|
|
|
|
static void
|
|
gst_vulkan_swapper_class_init (GstVulkanSwapperClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
|
|
|
gobject_class->set_property = gst_vulkan_swapper_set_property;
|
|
gobject_class->get_property = gst_vulkan_swapper_get_property;
|
|
gobject_class->finalize = gst_vulkan_swapper_finalize;
|
|
|
|
g_object_class_install_property (gobject_class, PROP_FORCE_ASPECT_RATIO,
|
|
g_param_spec_boolean ("force-aspect-ratio", "Force aspect ratio",
|
|
"When enabled, scaling will respect original aspect ratio",
|
|
DEFAULT_FORCE_ASPECT_RATIO,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_PIXEL_ASPECT_RATIO,
|
|
gst_param_spec_fraction ("pixel-aspect-ratio", "Pixel Aspect Ratio",
|
|
"The pixel aspect ratio of the device", 0, 1, G_MAXINT, 1, 1, 1,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
}
|
|
|
|
GstVulkanSwapper *
|
|
gst_vulkan_swapper_new (GstVulkanDevice * device, GstVulkanWindow * window)
|
|
{
|
|
GstVulkanSwapper *swapper;
|
|
GstVulkanSwapperPrivate *priv;
|
|
|
|
swapper = g_object_new (GST_TYPE_VULKAN_SWAPPER, NULL);
|
|
gst_object_ref_sink (swapper);
|
|
swapper->device = gst_object_ref (device);
|
|
swapper->window = gst_object_ref (window);
|
|
|
|
if (!_get_function_table (swapper)) {
|
|
gst_object_unref (swapper);
|
|
return NULL;
|
|
}
|
|
priv = GET_PRIV (swapper);
|
|
|
|
priv->close_id = g_signal_connect (swapper->window, "close",
|
|
(GCallback) _on_window_close, swapper);
|
|
priv->draw_id = g_signal_connect (swapper->window, "draw",
|
|
(GCallback) _on_window_draw, swapper);
|
|
priv->resize_id = g_signal_connect (swapper->window, "resize",
|
|
(GCallback) _on_window_resize, swapper);
|
|
|
|
return swapper;
|
|
}
|
|
|
|
GstCaps *
|
|
gst_vulkan_swapper_get_supported_caps (GstVulkanSwapper * swapper,
|
|
GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
GstStructure *s;
|
|
GstCaps *caps;
|
|
|
|
g_return_val_if_fail (GST_IS_VULKAN_SWAPPER (swapper), NULL);
|
|
|
|
if (!_vulkan_swapper_retrieve_surface_properties (swapper, error))
|
|
return NULL;
|
|
|
|
caps = gst_caps_new_empty_simple ("video/x-raw");
|
|
gst_caps_set_features (caps, 0,
|
|
gst_caps_features_from_string (GST_CAPS_FEATURE_MEMORY_VULKAN_IMAGE));
|
|
s = gst_caps_get_structure (caps, 0);
|
|
|
|
{
|
|
int i;
|
|
GValue list = G_VALUE_INIT;
|
|
|
|
g_value_init (&list, GST_TYPE_LIST);
|
|
|
|
if (priv->n_surf_formats
|
|
&& priv->surf_formats[0].format == VK_FORMAT_UNDEFINED) {
|
|
_add_vk_format_to_list (&list, VK_FORMAT_B8G8R8A8_UNORM);
|
|
} else {
|
|
for (i = 0; i < priv->n_surf_formats; i++) {
|
|
_add_vk_format_to_list (&list, priv->surf_formats[i].format);
|
|
}
|
|
}
|
|
|
|
gst_structure_set_value (s, "format", &list);
|
|
g_value_unset (&list);
|
|
}
|
|
|
|
{
|
|
guint32 max_dim =
|
|
swapper->device->physical_device->properties.limits.maxImageDimension2D;
|
|
|
|
gst_structure_set (s, "width", GST_TYPE_INT_RANGE, 1, (gint) max_dim,
|
|
"height", GST_TYPE_INT_RANGE, 1, (gint) max_dim, "pixel-aspect-ratio",
|
|
GST_TYPE_FRACTION, 1, 1, "framerate", GST_TYPE_FRACTION_RANGE, 0, 1,
|
|
G_MAXINT, 1, NULL);
|
|
}
|
|
|
|
GST_INFO_OBJECT (swapper, "Probed the following caps %" GST_PTR_FORMAT, caps);
|
|
|
|
return caps;
|
|
}
|
|
|
|
static gboolean
|
|
_allocate_swapchain (GstVulkanSwapper * swapper, GstCaps * caps,
|
|
GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
VkSurfaceTransformFlagsKHR preTransform;
|
|
VkCompositeAlphaFlagsKHR alpha_flags;
|
|
VkPresentModeKHR present_mode;
|
|
VkImageUsageFlags usage = 0;
|
|
VkColorSpaceKHR color_space;
|
|
VkImage *swap_chain_images;
|
|
VkExtent2D swapchain_dims;
|
|
guint32 n_images_wanted;
|
|
VkPhysicalDevice gpu;
|
|
VkFormat format;
|
|
VkResult err;
|
|
guint32 i;
|
|
|
|
if (!_vulkan_swapper_ensure_surface (swapper, error))
|
|
return FALSE;
|
|
|
|
gpu = gst_vulkan_device_get_physical_device (swapper->device);
|
|
err =
|
|
priv->GetPhysicalDeviceSurfaceCapabilitiesKHR (gpu,
|
|
priv->surface, &priv->surf_props);
|
|
if (gst_vulkan_error_to_g_error (err, error,
|
|
"GetPhysicalDeviceSurfaceCapabilitiesKHR") < 0)
|
|
return FALSE;
|
|
|
|
/* width and height are either both -1, or both not -1. */
|
|
if (priv->surf_props.currentExtent.width == -1) {
|
|
/* If the surface size is undefined, the size is set to
|
|
* the size of the images requested. */
|
|
guint width, height;
|
|
gst_vulkan_window_get_surface_dimensions (swapper->window, &width, &height);
|
|
swapchain_dims.width = width;
|
|
swapchain_dims.height = height;
|
|
priv->any_current_extent = TRUE;
|
|
} else {
|
|
/* If the surface size is defined, the swap chain size must match */
|
|
swapchain_dims = priv->surf_props.currentExtent;
|
|
priv->any_current_extent = FALSE;
|
|
}
|
|
priv->surface_location.w = swapchain_dims.width;
|
|
priv->surface_location.h = swapchain_dims.height;
|
|
|
|
/* If mailbox mode is available, use it, as is the lowest-latency non-
|
|
* tearing mode. If not, try IMMEDIATE which will usually be available,
|
|
* and is fastest (though it tears). If not, fall back to FIFO which is
|
|
* always available. */
|
|
present_mode = VK_PRESENT_MODE_FIFO_KHR;
|
|
for (i = 0; i < priv->n_surf_present_modes; i++) {
|
|
if (priv->surf_present_modes[i] == VK_PRESENT_MODE_MAILBOX_KHR) {
|
|
present_mode = VK_PRESENT_MODE_MAILBOX_KHR;
|
|
break;
|
|
}
|
|
if ((present_mode != VK_PRESENT_MODE_MAILBOX_KHR) &&
|
|
(priv->surf_present_modes[i] == VK_PRESENT_MODE_IMMEDIATE_KHR)) {
|
|
present_mode = VK_PRESENT_MODE_IMMEDIATE_KHR;
|
|
}
|
|
}
|
|
|
|
/* Determine the number of VkImage's to use in the swap chain (we desire to
|
|
* own only 1 image at a time, besides the images being displayed and
|
|
* queued for display): */
|
|
n_images_wanted = priv->surf_props.minImageCount + 1;
|
|
if ((priv->surf_props.maxImageCount > 0) &&
|
|
(n_images_wanted > priv->surf_props.maxImageCount)) {
|
|
/* Application must settle for fewer images than desired: */
|
|
n_images_wanted = priv->surf_props.maxImageCount;
|
|
}
|
|
|
|
if (priv->surf_props.
|
|
supportedTransforms & VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR) {
|
|
preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
|
|
} else {
|
|
preTransform = priv->surf_props.currentTransform;
|
|
}
|
|
|
|
format = gst_vulkan_format_from_video_info (&priv->v_info, 0);
|
|
color_space = _vk_color_space_from_video_info (&priv->v_info);
|
|
|
|
if ((priv->surf_props.supportedCompositeAlpha &
|
|
VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR) != 0) {
|
|
alpha_flags = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
|
|
} else if ((priv->surf_props.supportedCompositeAlpha &
|
|
VK_COMPOSITE_ALPHA_PRE_MULTIPLIED_BIT_KHR) != 0) {
|
|
alpha_flags = VK_COMPOSITE_ALPHA_PRE_MULTIPLIED_BIT_KHR;
|
|
} else if ((priv->surf_props.supportedCompositeAlpha &
|
|
VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR) != 0) {
|
|
alpha_flags = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
|
|
} else {
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED,
|
|
"Incorrect alpha flags (0x%x) available for the swap images",
|
|
priv->surf_props.supportedCompositeAlpha);
|
|
return FALSE;
|
|
}
|
|
|
|
if ((priv->surf_props.supportedUsageFlags &
|
|
VK_IMAGE_USAGE_TRANSFER_DST_BIT) != 0) {
|
|
usage |= VK_IMAGE_USAGE_TRANSFER_DST_BIT;
|
|
} else {
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED,
|
|
"Incorrect usage flags (0x%x) available for the swap images",
|
|
priv->surf_props.supportedUsageFlags);
|
|
return FALSE;
|
|
}
|
|
if ((priv->
|
|
surf_props.supportedUsageFlags & VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT)
|
|
!= 0) {
|
|
usage |= VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
|
|
} else {
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED,
|
|
"Incorrect usage flags (0x%x) available for the swap images",
|
|
priv->surf_props.supportedUsageFlags);
|
|
return FALSE;
|
|
}
|
|
|
|
{
|
|
VkSwapchainCreateInfoKHR swap_chain_info = { 0, };
|
|
VkSwapchainKHR old_swap_chain;
|
|
|
|
old_swap_chain = priv->swap_chain;
|
|
|
|
/* *INDENT-OFF* */
|
|
swap_chain_info = (VkSwapchainCreateInfoKHR) {
|
|
.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR,
|
|
.pNext = NULL,
|
|
.surface = priv->surface,
|
|
.minImageCount = n_images_wanted,
|
|
.imageFormat = format,
|
|
.imageColorSpace = color_space,
|
|
.imageExtent = swapchain_dims,
|
|
.imageArrayLayers = 1,
|
|
.imageUsage = usage,
|
|
.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE,
|
|
.queueFamilyIndexCount = 0,
|
|
.pQueueFamilyIndices = NULL,
|
|
.preTransform = preTransform,
|
|
.presentMode = present_mode,
|
|
.compositeAlpha = alpha_flags,
|
|
.clipped = TRUE,
|
|
.oldSwapchain = old_swap_chain
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
err =
|
|
priv->CreateSwapchainKHR (swapper->device->device,
|
|
&swap_chain_info, NULL, &priv->swap_chain);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkCreateSwapchainKHR") < 0)
|
|
return FALSE;
|
|
|
|
if (old_swap_chain != VK_NULL_HANDLE) {
|
|
priv->DestroySwapchainKHR (swapper->device->device, old_swap_chain, NULL);
|
|
}
|
|
}
|
|
|
|
err =
|
|
priv->GetSwapchainImagesKHR (swapper->device->device,
|
|
priv->swap_chain, &priv->n_swap_chain_images, NULL);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkGetSwapchainImagesKHR") < 0)
|
|
return FALSE;
|
|
|
|
swap_chain_images = g_new0 (VkImage, priv->n_swap_chain_images);
|
|
err =
|
|
priv->GetSwapchainImagesKHR (swapper->device->device,
|
|
priv->swap_chain, &priv->n_swap_chain_images, swap_chain_images);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkGetSwapchainImagesKHR") < 0) {
|
|
g_free (swap_chain_images);
|
|
return FALSE;
|
|
}
|
|
|
|
priv->swap_chain_images =
|
|
g_new0 (GstVulkanImageMemory *, priv->n_swap_chain_images);
|
|
for (i = 0; i < priv->n_swap_chain_images; i++) {
|
|
priv->swap_chain_images[i] = (GstVulkanImageMemory *)
|
|
gst_vulkan_image_memory_wrapped (swapper->device, swap_chain_images[i],
|
|
format, swapchain_dims.width, swapchain_dims.height,
|
|
VK_IMAGE_TILING_OPTIMAL, usage, NULL, NULL);
|
|
|
|
priv->swap_chain_images[i]->barrier.parent.pipeline_stages =
|
|
VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
|
|
priv->swap_chain_images[i]->barrier.parent.access_flags =
|
|
VK_ACCESS_MEMORY_READ_BIT;
|
|
priv->swap_chain_images[i]->barrier.image_layout =
|
|
VK_IMAGE_LAYOUT_UNDEFINED;
|
|
}
|
|
|
|
g_free (swap_chain_images);
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_swapchain_resize (GstVulkanSwapper * swapper, GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
int i;
|
|
|
|
if (!swapper->queue) {
|
|
if (!_vulkan_swapper_retrieve_surface_properties (swapper, error)) {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
if (priv->swap_chain_images) {
|
|
for (i = 0; i < priv->n_swap_chain_images; i++) {
|
|
if (priv->swap_chain_images[i])
|
|
gst_memory_unref ((GstMemory *) priv->swap_chain_images[i]);
|
|
}
|
|
g_free (priv->swap_chain_images);
|
|
priv->swap_chain_images = NULL;
|
|
}
|
|
|
|
return _allocate_swapchain (swapper, priv->caps, error);
|
|
}
|
|
|
|
|
|
static gboolean
|
|
configure_display_from_info (GstVulkanSwapper * swapper, GstVideoInfo * vinfo)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
gint width;
|
|
gint height;
|
|
gboolean ok;
|
|
gint par_n, par_d;
|
|
gint display_par_n, display_par_d;
|
|
guint display_ratio_num, display_ratio_den;
|
|
|
|
width = GST_VIDEO_INFO_WIDTH (vinfo);
|
|
height = GST_VIDEO_INFO_HEIGHT (vinfo);
|
|
|
|
par_n = GST_VIDEO_INFO_PAR_N (vinfo);
|
|
par_d = GST_VIDEO_INFO_PAR_D (vinfo);
|
|
|
|
if (!par_n)
|
|
par_n = 1;
|
|
|
|
/* get display's PAR */
|
|
if (priv->par_n != 0 && priv->par_d != 0) {
|
|
display_par_n = priv->par_n;
|
|
display_par_d = priv->par_d;
|
|
} else {
|
|
display_par_n = 1;
|
|
display_par_d = 1;
|
|
}
|
|
|
|
ok = gst_video_calculate_display_ratio (&display_ratio_num,
|
|
&display_ratio_den, width, height, par_n, par_d, display_par_n,
|
|
display_par_d);
|
|
|
|
if (!ok)
|
|
return FALSE;
|
|
|
|
GST_TRACE_OBJECT (swapper, "PAR: %u/%u DAR:%u/%u", par_n, par_d,
|
|
display_par_n, display_par_d);
|
|
|
|
if (height % display_ratio_den == 0) {
|
|
GST_DEBUG_OBJECT (swapper, "keeping video height");
|
|
priv->dar_width = (guint)
|
|
gst_util_uint64_scale_int (height, display_ratio_num,
|
|
display_ratio_den);
|
|
priv->dar_height = height;
|
|
} else if (width % display_ratio_num == 0) {
|
|
GST_DEBUG_OBJECT (swapper, "keeping video width");
|
|
priv->dar_width = width;
|
|
priv->dar_height = (guint)
|
|
gst_util_uint64_scale_int (width, display_ratio_den, display_ratio_num);
|
|
} else {
|
|
GST_DEBUG_OBJECT (swapper, "approximating while keeping video height");
|
|
priv->dar_width = (guint)
|
|
gst_util_uint64_scale_int (height, display_ratio_num,
|
|
display_ratio_den);
|
|
priv->dar_height = height;
|
|
}
|
|
GST_DEBUG_OBJECT (swapper, "scaling to %dx%d", priv->dar_width,
|
|
priv->dar_height);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_vulkan_swapper_set_caps (GstVulkanSwapper * swapper, GstCaps * caps,
|
|
GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
|
|
if (!gst_video_info_from_caps (&priv->v_info, caps)) {
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED, "Failed to get GstVideoInfo from caps");
|
|
return FALSE;
|
|
}
|
|
|
|
if (!configure_display_from_info (swapper, &priv->v_info)) {
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED, "Failed to configure display sizes");
|
|
return FALSE;
|
|
}
|
|
|
|
gst_caps_replace (&priv->caps, caps);
|
|
|
|
return _swapchain_resize (swapper, error);
|
|
}
|
|
|
|
static gboolean
|
|
_build_render_buffer_cmd (GstVulkanSwapper * swapper, guint32 swap_idx,
|
|
GstBuffer * buffer, GstVulkanCommandBuffer ** cmd_ret, GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
GstMemory *in_mem;
|
|
GstVulkanImageMemory *swap_img;
|
|
GstVulkanCommandBuffer *cmd_buf;
|
|
GstVideoRectangle src;
|
|
VkResult err;
|
|
|
|
g_return_val_if_fail (swap_idx < priv->n_swap_chain_images, FALSE);
|
|
swap_img = priv->swap_chain_images[swap_idx];
|
|
|
|
if (!(cmd_buf = gst_vulkan_command_pool_create (swapper->cmd_pool, error)))
|
|
return FALSE;
|
|
|
|
{
|
|
/* *INDENT-OFF* */
|
|
VkCommandBufferBeginInfo cmd_buf_info = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
|
|
.pNext = NULL,
|
|
.flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT,
|
|
.pInheritanceInfo = NULL
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
gst_vulkan_command_buffer_lock (cmd_buf);
|
|
err = vkBeginCommandBuffer (cmd_buf->cmd, &cmd_buf_info);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkBeginCommandBuffer") < 0)
|
|
goto unlock_error;
|
|
}
|
|
|
|
{
|
|
/* *INDENT-OFF* */
|
|
VkImageMemoryBarrier image_memory_barrier = {
|
|
.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER,
|
|
.pNext = NULL,
|
|
.srcAccessMask = swap_img->barrier.parent.access_flags,
|
|
.dstAccessMask = VK_ACCESS_TRANSFER_WRITE_BIT,
|
|
.oldLayout = swap_img->barrier.image_layout,
|
|
.newLayout = VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
|
|
/* FIXME: implement exclusive transfers */
|
|
.srcQueueFamilyIndex = 0,
|
|
.dstQueueFamilyIndex = 0,
|
|
.image = swap_img->image,
|
|
.subresourceRange = swap_img->barrier.subresource_range
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
vkCmdPipelineBarrier (cmd_buf->cmd,
|
|
swap_img->barrier.parent.pipeline_stages,
|
|
VK_PIPELINE_STAGE_TRANSFER_BIT, 0, 0, NULL, 0, NULL, 1,
|
|
&image_memory_barrier);
|
|
|
|
swap_img->barrier.parent.pipeline_stages = VK_PIPELINE_STAGE_TRANSFER_BIT;
|
|
swap_img->barrier.parent.access_flags = image_memory_barrier.dstAccessMask;
|
|
swap_img->barrier.image_layout = image_memory_barrier.newLayout;
|
|
}
|
|
|
|
src.x = src.y = 0;
|
|
src.w = priv->dar_width;
|
|
src.h = priv->dar_height;
|
|
|
|
g_assert (priv->surface_location.w ==
|
|
gst_vulkan_image_memory_get_width (swap_img));
|
|
g_assert (priv->surface_location.h ==
|
|
gst_vulkan_image_memory_get_height (swap_img));
|
|
|
|
gst_video_sink_center_rect (src, priv->surface_location, &priv->display_rect,
|
|
priv->force_aspect_ratio);
|
|
|
|
GST_TRACE_OBJECT (swapper, "rendering into result rectangle %ux%u+%u,%u "
|
|
"src %ux%u dst %ux%u", priv->display_rect.w, priv->display_rect.h,
|
|
priv->display_rect.x, priv->display_rect.y, src.w, src.h,
|
|
priv->surface_location.w, priv->surface_location.h);
|
|
|
|
in_mem = gst_buffer_peek_memory (buffer, 0);
|
|
{
|
|
GstVulkanImageMemory *img_mem = (GstVulkanImageMemory *) in_mem;
|
|
/* *INDENT-OFF* */
|
|
VkImageBlit blit = {
|
|
.srcSubresource = {
|
|
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
|
|
.mipLevel = 0,
|
|
.baseArrayLayer = 0,
|
|
.layerCount = 1,
|
|
},
|
|
.srcOffsets = {
|
|
{0, 0, 0},
|
|
{GST_VIDEO_INFO_WIDTH (&priv->v_info), GST_VIDEO_INFO_HEIGHT (&priv->v_info), 1},
|
|
},
|
|
.dstSubresource = {
|
|
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
|
|
.mipLevel = 0,
|
|
.baseArrayLayer = 0,
|
|
.layerCount = 1,
|
|
},
|
|
.dstOffsets = {
|
|
{priv->display_rect.x, priv->display_rect.y, 0},
|
|
{priv->display_rect.x + priv->display_rect.w, priv->display_rect.y + priv->display_rect.h, 1},
|
|
},
|
|
};
|
|
VkImageMemoryBarrier image_memory_barrier = {
|
|
.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER,
|
|
.pNext = NULL,
|
|
.srcAccessMask = img_mem->barrier.parent.access_flags,
|
|
.dstAccessMask = VK_ACCESS_TRANSFER_READ_BIT,
|
|
.oldLayout = img_mem->barrier.image_layout,
|
|
.newLayout = VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
|
|
/* FIXME: implement exclusive transfers */
|
|
.srcQueueFamilyIndex = 0,
|
|
.dstQueueFamilyIndex = 0,
|
|
.image = img_mem->image,
|
|
.subresourceRange = img_mem->barrier.subresource_range
|
|
};
|
|
VkClearColorValue clear = {{0.0, 0.0, 0.0, 1.0}};
|
|
VkImageSubresourceRange clear_range = {
|
|
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
|
|
.baseMipLevel = 0,
|
|
.levelCount = 1,
|
|
.baseArrayLayer = 0,
|
|
.layerCount = 1,
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
vkCmdPipelineBarrier (cmd_buf->cmd, img_mem->barrier.parent.pipeline_stages,
|
|
VK_PIPELINE_STAGE_TRANSFER_BIT, 0, 0, NULL, 0, NULL, 1,
|
|
&image_memory_barrier);
|
|
|
|
img_mem->barrier.parent.pipeline_stages = VK_PIPELINE_STAGE_TRANSFER_BIT;
|
|
img_mem->barrier.parent.access_flags = image_memory_barrier.dstAccessMask;
|
|
img_mem->barrier.image_layout = image_memory_barrier.newLayout;
|
|
|
|
vkCmdClearColorImage (cmd_buf->cmd, swap_img->image,
|
|
swap_img->barrier.image_layout, &clear, 1, &clear_range);
|
|
vkCmdBlitImage (cmd_buf->cmd, img_mem->image, img_mem->barrier.image_layout,
|
|
swap_img->image, swap_img->barrier.image_layout, 1, &blit,
|
|
VK_FILTER_LINEAR);
|
|
}
|
|
{
|
|
/* *INDENT-OFF* */
|
|
VkImageMemoryBarrier image_memory_barrier = {
|
|
.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER,
|
|
.pNext = NULL,
|
|
.srcAccessMask = swap_img->barrier.parent.access_flags,
|
|
.dstAccessMask = VK_ACCESS_MEMORY_READ_BIT,
|
|
.oldLayout = swap_img->barrier.image_layout,
|
|
.newLayout = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
|
|
/* FIXME: implement exclusive transfers */
|
|
.srcQueueFamilyIndex = 0,
|
|
.dstQueueFamilyIndex = 0,
|
|
.image = swap_img->image,
|
|
.subresourceRange = swap_img->barrier.subresource_range
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
vkCmdPipelineBarrier (cmd_buf->cmd,
|
|
swap_img->barrier.parent.pipeline_stages,
|
|
VK_PIPELINE_STAGE_TRANSFER_BIT, 0, 0, NULL, 0, NULL, 1,
|
|
&image_memory_barrier);
|
|
|
|
swap_img->barrier.parent.pipeline_stages = VK_PIPELINE_STAGE_TRANSFER_BIT;
|
|
swap_img->barrier.parent.access_flags = image_memory_barrier.dstAccessMask;
|
|
swap_img->barrier.image_layout = image_memory_barrier.newLayout;
|
|
}
|
|
|
|
err = vkEndCommandBuffer (cmd_buf->cmd);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkEndCommandBuffer") < 0)
|
|
goto unlock_error;
|
|
gst_vulkan_command_buffer_unlock (cmd_buf);
|
|
|
|
*cmd_ret = cmd_buf;
|
|
|
|
return TRUE;
|
|
|
|
unlock_error:
|
|
gst_vulkan_command_buffer_unlock (cmd_buf);
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
_render_buffer_unlocked (GstVulkanSwapper * swapper,
|
|
GstBuffer * buffer, GError ** error)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
VkSemaphore acquire_semaphore = { 0, };
|
|
VkSemaphore present_semaphore = { 0, };
|
|
VkSemaphoreCreateInfo semaphore_info = { 0, };
|
|
GstVulkanFence *fence = NULL;
|
|
VkPresentInfoKHR present;
|
|
GstVulkanCommandBuffer *cmd_buf = NULL;
|
|
guint32 swap_idx;
|
|
VkResult err, present_err = VK_SUCCESS;
|
|
|
|
gst_vulkan_trash_list_gc (priv->trash_list);
|
|
|
|
if (!buffer) {
|
|
g_set_error (error, GST_VULKAN_ERROR,
|
|
VK_ERROR_INITIALIZATION_FAILED, "Invalid buffer");
|
|
goto error;
|
|
}
|
|
|
|
if (g_atomic_int_get (&priv->to_quit)) {
|
|
g_set_error (error, GST_VULKAN_ERROR, VK_ERROR_SURFACE_LOST_KHR,
|
|
"Output window was closed");
|
|
goto error;
|
|
}
|
|
|
|
gst_buffer_replace (&priv->current_buffer, buffer);
|
|
|
|
/* *INDENT-OFF* */
|
|
semaphore_info = (VkSemaphoreCreateInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO,
|
|
.pNext = NULL,
|
|
.flags = 0,
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
reacquire:
|
|
err = vkCreateSemaphore (swapper->device->device, &semaphore_info,
|
|
NULL, &acquire_semaphore);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkCreateSemaphore") < 0)
|
|
goto error;
|
|
|
|
err =
|
|
priv->AcquireNextImageKHR (swapper->device->device,
|
|
priv->swap_chain, -1, acquire_semaphore, VK_NULL_HANDLE, &swap_idx);
|
|
/* TODO: Deal with the VK_SUBOPTIMAL_KHR and VK_ERROR_OUT_OF_DATE_KHR */
|
|
if (err == VK_ERROR_OUT_OF_DATE_KHR) {
|
|
GST_DEBUG_OBJECT (swapper, "out of date frame acquired");
|
|
|
|
vkDestroySemaphore (swapper->device->device, acquire_semaphore, NULL);
|
|
acquire_semaphore = VK_NULL_HANDLE;
|
|
if (!_swapchain_resize (swapper, error))
|
|
goto error;
|
|
goto reacquire;
|
|
} else if (gst_vulkan_error_to_g_error (err, error,
|
|
"vkAcquireNextImageKHR") < 0) {
|
|
goto error;
|
|
}
|
|
|
|
if (!_build_render_buffer_cmd (swapper, swap_idx, buffer, &cmd_buf, error))
|
|
goto error;
|
|
|
|
err = vkCreateSemaphore (swapper->device->device, &semaphore_info,
|
|
NULL, &present_semaphore);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkCreateSemaphore") < 0)
|
|
goto error;
|
|
|
|
{
|
|
VkPipelineStageFlags stages = VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
|
|
VkSubmitInfo submit_info = { 0, };
|
|
|
|
/* *INDENT-OFF* */
|
|
submit_info = (VkSubmitInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO,
|
|
.pNext = NULL,
|
|
.waitSemaphoreCount = 1,
|
|
.pWaitSemaphores = &acquire_semaphore,
|
|
.pWaitDstStageMask = &stages,
|
|
.commandBufferCount = 1,
|
|
.pCommandBuffers = &cmd_buf->cmd,
|
|
.signalSemaphoreCount = 1,
|
|
.pSignalSemaphores = &present_semaphore,
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
fence = gst_vulkan_device_create_fence (swapper->device, error);
|
|
if (!fence)
|
|
goto error;
|
|
|
|
gst_vulkan_queue_submit_lock (swapper->queue);
|
|
err =
|
|
vkQueueSubmit (swapper->queue->queue, 1, &submit_info,
|
|
GST_VULKAN_FENCE_FENCE (fence));
|
|
gst_vulkan_queue_submit_unlock (swapper->queue);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkQueueSubmit") < 0)
|
|
goto error;
|
|
|
|
gst_vulkan_trash_list_add (priv->trash_list,
|
|
gst_vulkan_trash_new_mini_object_unref (fence,
|
|
GST_MINI_OBJECT_CAST (cmd_buf)));
|
|
gst_vulkan_trash_list_add (priv->trash_list,
|
|
gst_vulkan_trash_new_free_semaphore (fence, acquire_semaphore));
|
|
acquire_semaphore = VK_NULL_HANDLE;
|
|
|
|
gst_vulkan_command_buffer_unlock (cmd_buf);
|
|
cmd_buf = NULL;
|
|
gst_vulkan_fence_unref (fence);
|
|
fence = NULL;
|
|
}
|
|
|
|
/* *INDENT-OFF* */
|
|
present = (VkPresentInfoKHR) {
|
|
.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR,
|
|
.pNext = NULL,
|
|
.waitSemaphoreCount = 1,
|
|
.pWaitSemaphores = &present_semaphore,
|
|
.swapchainCount = 1,
|
|
.pSwapchains = &priv->swap_chain,
|
|
.pImageIndices = &swap_idx,
|
|
.pResults = &present_err,
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
err = priv->QueuePresentKHR (swapper->queue->queue, &present);
|
|
|
|
if (present_err == VK_ERROR_OUT_OF_DATE_KHR) {
|
|
GST_DEBUG_OBJECT (swapper, "out of date frame submitted");
|
|
|
|
if (!_swapchain_resize (swapper, error))
|
|
goto error;
|
|
} else if (gst_vulkan_error_to_g_error (err, error, "vkQueuePresentKHR") < 0)
|
|
goto error;
|
|
|
|
{
|
|
VkSubmitInfo submit_info = { 0, };
|
|
VkPipelineStageFlags stages = VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
|
|
|
|
/* *INDENT-OFF* */
|
|
submit_info = (VkSubmitInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO,
|
|
.pWaitDstStageMask = &stages,
|
|
0,
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
fence = gst_vulkan_device_create_fence (swapper->device, error);
|
|
if (!fence)
|
|
goto error;
|
|
|
|
gst_vulkan_queue_submit_lock (swapper->queue);
|
|
err =
|
|
vkQueueSubmit (swapper->queue->queue, 1, &submit_info,
|
|
GST_VULKAN_FENCE_FENCE (fence));
|
|
gst_vulkan_queue_submit_unlock (swapper->queue);
|
|
if (gst_vulkan_error_to_g_error (err, error, "vkQueueSubmit") < 0)
|
|
goto error;
|
|
|
|
gst_vulkan_trash_list_add (priv->trash_list,
|
|
gst_vulkan_trash_new_free_semaphore (fence, present_semaphore));
|
|
gst_vulkan_trash_list_add (priv->trash_list,
|
|
gst_vulkan_trash_new_mini_object_unref (fence,
|
|
(GstMiniObject *) gst_buffer_ref (buffer)));
|
|
gst_vulkan_fence_unref (fence);
|
|
fence = NULL;
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
error:
|
|
{
|
|
if (acquire_semaphore)
|
|
vkDestroySemaphore (swapper->device->device, acquire_semaphore, NULL);
|
|
if (present_semaphore)
|
|
vkDestroySemaphore (swapper->device->device, present_semaphore, NULL);
|
|
if (cmd_buf) {
|
|
gst_vulkan_command_buffer_unlock (cmd_buf);
|
|
gst_vulkan_command_buffer_unref (cmd_buf);
|
|
}
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gst_vulkan_swapper_render_buffer (GstVulkanSwapper * swapper,
|
|
GstBuffer * buffer, GError ** error)
|
|
{
|
|
GstMemory *mem;
|
|
gboolean ret;
|
|
|
|
mem = gst_buffer_peek_memory (buffer, 0);
|
|
if (!mem) {
|
|
g_set_error_literal (error, GST_VULKAN_ERROR, VK_ERROR_FORMAT_NOT_SUPPORTED,
|
|
"Buffer has no memory");
|
|
return FALSE;
|
|
}
|
|
if (!gst_is_vulkan_image_memory (mem)) {
|
|
g_set_error_literal (error, GST_VULKAN_ERROR, VK_ERROR_FORMAT_NOT_SUPPORTED,
|
|
"Incorrect memory type");
|
|
return FALSE;
|
|
}
|
|
|
|
RENDER_LOCK (swapper);
|
|
ret = _render_buffer_unlocked (swapper, buffer, error);
|
|
RENDER_UNLOCK (swapper);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
_on_window_draw (GstVulkanWindow * window, GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
GError *error = NULL;
|
|
|
|
RENDER_LOCK (swapper);
|
|
if (!priv->current_buffer) {
|
|
GST_DEBUG_OBJECT (swapper, "No buffer to render");
|
|
RENDER_UNLOCK (swapper);
|
|
return;
|
|
}
|
|
|
|
/* TODO: perform some rate limiting of the number of redraw events */
|
|
if (!_render_buffer_unlocked (swapper, priv->current_buffer, &error))
|
|
GST_ERROR_OBJECT (swapper, "Failed to redraw buffer %p %s",
|
|
priv->current_buffer, error->message);
|
|
g_clear_error (&error);
|
|
RENDER_UNLOCK (swapper);
|
|
}
|
|
|
|
static void
|
|
_on_window_resize (GstVulkanWindow * window, guint width, guint height,
|
|
GstVulkanSwapper * swapper)
|
|
{
|
|
GstVulkanSwapperPrivate *priv = GET_PRIV (swapper);
|
|
GError *error = NULL;
|
|
|
|
RENDER_LOCK (swapper);
|
|
if (priv->any_current_extent) {
|
|
if (!_swapchain_resize (swapper, &error))
|
|
GST_ERROR_OBJECT (swapper, "Failed to resize swapchain: %s",
|
|
error->message);
|
|
g_clear_error (&error);
|
|
}
|
|
RENDER_UNLOCK (swapper);
|
|
}
|
|
|
|
/**
|
|
* gst_vulkan_swapper_get_surface_rectangles:
|
|
* @swapper: a #GstVulkanSwapper
|
|
* @input_image: (out) (nullable): The #GstVideoRectangle for the configured
|
|
* caps modified for DAR.
|
|
* @surface_location: (out) (nullable): The #GstVideoRectangle for where the
|
|
* output surface is located relative to its parent
|
|
* @display_rect: (out) (nullable): The #GstVideoRectangle for where the input
|
|
* images are placed inside @surface_location
|
|
*
|
|
* Since: 1.18
|
|
*/
|
|
void
|
|
gst_vulkan_swapper_get_surface_rectangles (GstVulkanSwapper * swapper,
|
|
GstVideoRectangle * input_image, GstVideoRectangle * surface_location,
|
|
GstVideoRectangle * display_rect)
|
|
{
|
|
GstVulkanSwapperPrivate *priv;
|
|
|
|
g_return_if_fail (GST_IS_VULKAN_SWAPPER (swapper));
|
|
|
|
priv = GET_PRIV (swapper);
|
|
|
|
RENDER_LOCK (swapper);
|
|
if (input_image) {
|
|
input_image->x = input_image->y = 0;
|
|
input_image->w = priv->dar_width;
|
|
input_image->h = priv->dar_height;
|
|
}
|
|
|
|
if (surface_location) {
|
|
*display_rect = priv->surface_location;
|
|
}
|
|
|
|
if (display_rect) {
|
|
*display_rect = priv->display_rect;
|
|
}
|
|
RENDER_UNLOCK (swapper);
|
|
}
|