[gtk+/wip/otte/vulkan: 48/59] vulkan: More work on GdkVulkanContext
- From: Benjamin Otte <otte src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gtk+/wip/otte/vulkan: 48/59] vulkan: More work on GdkVulkanContext
- Date: Mon, 5 Dec 2016 12:31:31 +0000 (UTC)
commit a6ba365867e96e8af62689d8b1313c57ca49910a
Author: Benjamin Otte <otte redhat com>
Date: Tue Nov 29 03:20:31 2016 +0100
vulkan: More work on GdkVulkanContext
Code has now arrived at creation of swapchains.
gdk/gdkdisplayprivate.h | 1 +
gdk/gdkvulkancontext.c | 282 ++++++++++++++++++++++++++++++++++++++++-
gdk/gdkvulkancontext.h | 13 ++
gdk/gdkvulkancontextprivate.h | 2 +-
4 files changed, 291 insertions(+), 7 deletions(-)
---
diff --git a/gdk/gdkdisplayprivate.h b/gdk/gdkdisplayprivate.h
index 4164b9c..9f28db7 100644
--- a/gdk/gdkdisplayprivate.h
+++ b/gdk/gdkdisplayprivate.h
@@ -129,6 +129,7 @@ struct _GdkDisplay
VkPhysicalDevice vk_physical_device;
VkDevice vk_device;
VkQueue vk_queue;
+ uint32_t vk_queue_family_index;
guint vulkan_refcount;
#endif /* GDK_WINDOWING_VULKAN */
diff --git a/gdk/gdkvulkancontext.c b/gdk/gdkvulkancontext.c
index 3ae9b79..577d217 100644
--- a/gdk/gdkvulkancontext.c
+++ b/gdk/gdkvulkancontext.c
@@ -32,6 +32,10 @@ typedef struct _GdkVulkanContextPrivate GdkVulkanContextPrivate;
struct _GdkVulkanContextPrivate {
VkSurfaceKHR surface;
+ VkSurfaceFormatKHR image_format;
+
+ int swapchain_width, swapchain_height;
+ VkSwapchainKHR swapchain;
};
G_DEFINE_QUARK (gdk-vulkan-error-quark, gdk_vulkan_error)
@@ -42,6 +46,64 @@ G_DEFINE_ABSTRACT_TYPE_WITH_CODE (GdkVulkanContext, gdk_vulkan_context, GDK_TYPE
G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE, gdk_vulkan_context_initable_init)
G_ADD_PRIVATE (GdkVulkanContext))
+#ifdef GDK_WINDOWING_VULKAN
+
+const char *
+gdk_vulkan_strerror (VkResult result)
+{
+ switch (result)
+ {
+ case VK_SUCCESS:
+ return "Command successfully completed.";
+ case VK_NOT_READY:
+ return "A fence or query has not yet completed.";
+ case VK_TIMEOUT:
+ return "A wait operation has not completed in the specified time.";
+ case VK_EVENT_SET:
+ return "An event is signaled.";
+ case VK_EVENT_RESET:
+ return "An event is unsignaled.";
+ case VK_INCOMPLETE:
+ return "A return array was too small for the result.";
+ case VK_SUBOPTIMAL_KHR:
+ return "A swapchain no longer matches the surface properties exactly, but can still be used to present
to the surface successfully.";
+ case VK_ERROR_OUT_OF_HOST_MEMORY:
+ return "A host memory allocation has failed.";
+ case VK_ERROR_OUT_OF_DEVICE_MEMORY:
+ return "A device memory allocation has failed.";
+ case VK_ERROR_INITIALIZATION_FAILED:
+ return "Initialization of an object could not be completed for implementation-specific reasons.";
+ case VK_ERROR_DEVICE_LOST:
+ return "The logical or physical device has been lost.";
+ case VK_ERROR_MEMORY_MAP_FAILED:
+ return "Mapping of a memory object has failed.";
+ case VK_ERROR_LAYER_NOT_PRESENT:
+ return "A requested layer is not present or could not be loaded.";
+ case VK_ERROR_EXTENSION_NOT_PRESENT:
+ return "A requested extension is not supported.";
+ case VK_ERROR_FEATURE_NOT_PRESENT:
+ return "A requested feature is not supported.";
+ case VK_ERROR_INCOMPATIBLE_DRIVER:
+ return "The requested version of Vulkan is not supported by the driver or is otherwise incompatible
for implementation-specific reasons.";
+ case VK_ERROR_TOO_MANY_OBJECTS:
+ return "Too many objects of the type have already been created.";
+ case VK_ERROR_FORMAT_NOT_SUPPORTED:
+ return "A requested format is not supported on this device.";
+ case VK_ERROR_FRAGMENTED_POOL:
+ return "A requested pool allocation has failed due to fragmentation of the pool’s memory.";
+ case VK_ERROR_SURFACE_LOST_KHR:
+ return "A surface is no longer available.";
+ case VK_ERROR_NATIVE_WINDOW_IN_USE_KHR:
+ return "The requested window is already in use by Vulkan or another API in a manner which prevents it
from being used again.";
+ case VK_ERROR_OUT_OF_DATE_KHR:
+ return "A surface has changed in such a way that it is no longer compatible with the swapchain.";
+ case VK_ERROR_INCOMPATIBLE_DISPLAY_KHR:
+ return "The display used by a swapchain does not use the same presentable image layout, or is
incompatible in a way that prevents sharing an image.";
+ default:
+ return "Unknown Vulkan error.";
+ }
+}
+
static void
gdk_vulkan_context_dispose (GObject *gobject)
{
@@ -49,6 +111,14 @@ gdk_vulkan_context_dispose (GObject *gobject)
GdkVulkanContextPrivate *priv = gdk_vulkan_context_get_instance_private (context);
GdkDisplay *display;
+ if (priv->swapchain != VK_NULL_HANDLE)
+ {
+ vkDestroySwapchainKHR (gdk_vulkan_context_get_device (context),
+ priv->swapchain,
+ NULL);
+ priv->swapchain = VK_NULL_HANDLE;
+ }
+
if (priv->surface != VK_NULL_HANDLE)
{
vkDestroySurfaceKHR (gdk_vulkan_context_get_instance (context),
@@ -79,24 +149,166 @@ gdk_vulkan_context_init (GdkVulkanContext *self)
}
static gboolean
+gdk_vulkan_context_check_swapchain (GdkVulkanContext *context,
+ GError **error)
+{
+ GdkVulkanContextPrivate *priv = gdk_vulkan_context_get_instance_private (context);
+ GdkWindow *window = gdk_draw_context_get_window (GDK_DRAW_CONTEXT (context));
+ VkSurfaceCapabilitiesKHR capabilities;
+ VkCompositeAlphaFlagBitsKHR composite_alpha;
+ VkSwapchainKHR new_swapchain;
+ VkResult res;
+
+ if (gdk_window_get_width (window) == priv->swapchain_width &&
+ gdk_window_get_height (window) == priv->swapchain_height)
+ return TRUE;
+
+ res = GDK_VK_CHECK (vkGetPhysicalDeviceSurfaceCapabilitiesKHR, gdk_vulkan_context_get_physical_device
(context),
+ priv->surface,
+ &capabilities);
+ if (res != VK_SUCCESS)
+ {
+ g_set_error (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
+ "Could not query surface capabilities: %s", gdk_vulkan_strerror (res));
+ return FALSE;
+ }
+
+ if (capabilities.supportedCompositeAlpha & VK_COMPOSITE_ALPHA_PRE_MULTIPLIED_BIT_KHR)
+ composite_alpha = VK_COMPOSITE_ALPHA_PRE_MULTIPLIED_BIT_KHR;
+ else if (capabilities.supportedCompositeAlpha & VK_COMPOSITE_ALPHA_POST_MULTIPLIED_BIT_KHR)
+ composite_alpha = VK_COMPOSITE_ALPHA_POST_MULTIPLIED_BIT_KHR;
+ else if (capabilities.supportedCompositeAlpha & VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR)
+ {
+ /* let's hope the backend knows what it's doing */
+ composite_alpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
+ }
+ else
+ {
+ GDK_NOTE (VULKAN, g_warning ("Vulkan swapchain doesn't do transparency. Using opaque swapchain
instead."));
+ composite_alpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
+ }
+
+ res = GDK_VK_CHECK (vkCreateSwapchainKHR, gdk_vulkan_context_get_device (context),
+ &(VkSwapchainCreateInfoKHR) {
+ .sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR,
+ .pNext = NULL,
+ .flags = 0,
+ .surface = priv->surface,
+ .minImageCount = 2,
+ .imageFormat = priv->image_format.format,
+ .imageColorSpace = priv->image_format.colorSpace,
+ .imageExtent = capabilities.currentExtent,
+ .imageArrayLayers = 1,
+ .imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
+ .imageSharingMode = VK_SHARING_MODE_EXCLUSIVE,
+ .queueFamilyIndexCount = 1,
+ .pQueueFamilyIndices = (uint32_t[1]) {
+ gdk_draw_context_get_display (GDK_DRAW_CONTEXT
(context))->vk_queue_family_index
+ },
+ .preTransform = capabilities.currentTransform,
+ .compositeAlpha = composite_alpha,
+ .presentMode = VK_PRESENT_MODE_FIFO_KHR,
+ .clipped = VK_FALSE,
+ .oldSwapchain = priv->swapchain
+ },
+ NULL,
+ &new_swapchain);
+
+ if (priv->swapchain != VK_NULL_HANDLE)
+ vkDestroySwapchainKHR (gdk_vulkan_context_get_device (context),
+ priv->swapchain,
+ NULL);
+
+ if (res == VK_SUCCESS)
+ {
+ priv->swapchain_width = gdk_window_get_width (window);
+ priv->swapchain_height = gdk_window_get_height (window);
+ priv->swapchain = new_swapchain;
+ }
+ else
+ {
+ g_set_error (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
+ "Could not create swapchain for this window: %s", gdk_vulkan_strerror (res));
+ priv->swapchain = VK_NULL_HANDLE;
+ priv->swapchain_width = 0;
+ priv->swapchain_height = 0;
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+static gboolean
gdk_vulkan_context_real_init (GInitable *initable,
GCancellable *cancellable,
GError **error)
{
GdkVulkanContext *context = GDK_VULKAN_CONTEXT (initable);
GdkVulkanContextPrivate *priv = gdk_vulkan_context_get_instance_private (context);
+ VkResult res;
+ VkBool32 supported;
+ uint32_t i;
if (!gdk_display_ref_vulkan (gdk_draw_context_get_display (GDK_DRAW_CONTEXT (context)), error))
return FALSE;
- if (GDK_VULKAN_CONTEXT_GET_CLASS (context)->create_surface (context, &priv->surface) != VK_SUCCESS)
+ res = GDK_VULKAN_CONTEXT_GET_CLASS (context)->create_surface (context, &priv->surface);
+ if (res != VK_SUCCESS)
{
- g_set_error_literal (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
- "Vulkan support not available for this window.");
+ g_set_error (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
+ "Could not create surface for this window: %s", gdk_vulkan_strerror (res));
return FALSE;
}
- return TRUE;
+ res = GDK_VK_CHECK (vkGetPhysicalDeviceSurfaceSupportKHR, gdk_vulkan_context_get_physical_device (context),
+ gdk_vulkan_context_get_queue_family_index
(context),
+ priv->surface,
+ &supported);
+ if (res != VK_SUCCESS)
+ {
+ g_set_error (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
+ "Could not check if queue family supports this window: %s", gdk_vulkan_strerror (res));
+ }
+ else if (!supported)
+ {
+ g_set_error (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
+ "FIXME: Queue family does not support surface. Write code to try different queue
family.");
+ }
+ else
+ {
+ uint32_t n_formats;
+ GDK_VK_CHECK (vkGetPhysicalDeviceSurfaceFormatsKHR, gdk_vulkan_context_get_physical_device (context),
+ priv->surface,
+ &n_formats, NULL);
+ VkSurfaceFormatKHR formats[n_formats];
+ GDK_VK_CHECK (vkGetPhysicalDeviceSurfaceFormatsKHR, gdk_vulkan_context_get_physical_device (context),
+ priv->surface,
+ &n_formats, formats);
+ for (i = 0; i < n_formats; i++)
+ {
+ if (formats[i].format == VK_FORMAT_B8G8R8A8_SRGB)
+ break;
+ }
+ if (i == n_formats)
+ {
+ g_set_error_literal (error, GDK_VULKAN_ERROR, GDK_VULKAN_ERROR_NOT_AVAILABLE,
+ "No supported image format found.");
+ goto out_surface;
+ }
+ priv->image_format = formats[i];
+
+ if (!gdk_vulkan_context_check_swapchain (context, error))
+ goto out_surface;
+
+ return TRUE;
+ }
+
+out_surface:
+ vkDestroySurfaceKHR (gdk_vulkan_context_get_instance (context),
+ priv->surface,
+ NULL);
+ priv->surface = VK_NULL_HANDLE;
+ return FALSE;
}
static void
@@ -105,8 +317,6 @@ gdk_vulkan_context_initable_init (GInitableIface *iface)
iface->init = gdk_vulkan_context_real_init;
}
-#ifdef GDK_WINDOWING_VULKAN
-
VkInstance
gdk_vulkan_context_get_instance (GdkVulkanContext *context)
{
@@ -115,6 +325,48 @@ gdk_vulkan_context_get_instance (GdkVulkanContext *context)
return gdk_draw_context_get_display (GDK_DRAW_CONTEXT (context))->vk_instance;
}
+VkPhysicalDevice
+gdk_vulkan_context_get_physical_device (GdkVulkanContext *context)
+{
+ g_return_val_if_fail (GDK_IS_VULKAN_CONTEXT (context), NULL);
+
+ return gdk_draw_context_get_display (GDK_DRAW_CONTEXT (context))->vk_physical_device;
+}
+
+VkDevice
+gdk_vulkan_context_get_device (GdkVulkanContext *context)
+{
+ g_return_val_if_fail (GDK_IS_VULKAN_CONTEXT (context), NULL);
+
+ return gdk_draw_context_get_display (GDK_DRAW_CONTEXT (context))->vk_device;
+}
+
+VkQueue
+gdk_vulkan_context_get_queue (GdkVulkanContext *context)
+{
+ g_return_val_if_fail (GDK_IS_VULKAN_CONTEXT (context), NULL);
+
+ return gdk_draw_context_get_display (GDK_DRAW_CONTEXT (context))->vk_queue;
+}
+
+uint32_t
+gdk_vulkan_context_get_queue_family_index (GdkVulkanContext *context)
+{
+ g_return_val_if_fail (GDK_IS_VULKAN_CONTEXT (context), 0);
+
+ return gdk_draw_context_get_display (GDK_DRAW_CONTEXT (context))->vk_queue_family_index;
+}
+
+VkFormat
+gdk_vulkan_context_get_image_format (GdkVulkanContext *context)
+{
+ GdkVulkanContextPrivate *priv = gdk_vulkan_context_get_instance_private (context);
+
+ g_return_val_if_fail (GDK_IS_VULKAN_CONTEXT (context), VK_FORMAT_UNDEFINED);
+
+ return priv->image_format.format;
+}
+
static gboolean
gdk_display_create_vulkan_device (GdkDisplay *display,
GError **error)
@@ -189,6 +441,7 @@ gdk_display_create_vulkan_device (GdkDisplay *display,
display->vk_physical_device = devices[i];
vkGetDeviceQueue(display->vk_device, j, 0, &display->vk_queue);
+ display->vk_queue_family_index = j;
return TRUE;
}
}
@@ -301,4 +554,21 @@ gdk_display_unref_vulkan (GdkDisplay *display)
vkDestroyInstance (display->vk_instance, NULL);
}
+#else /* GDK_WINDOWING_VULKAN */
+
+static void
+gdk_vulkan_context_class_init (GdkVulkanContextClass *klass)
+{
+}
+
+static void
+gdk_vulkan_context_init (GdkVulkanContext *self)
+{
+}
+
+static void
+gdk_vulkan_context_initable_init (GInitableIface *iface)
+{
+}
+
#endif /* GDK_WINDOWING_VULKAN */
diff --git a/gdk/gdkvulkancontext.h b/gdk/gdkvulkancontext.h
index 9441d5f..d4b0789 100644
--- a/gdk/gdkvulkancontext.h
+++ b/gdk/gdkvulkancontext.h
@@ -49,7 +49,20 @@ GType gdk_vulkan_context_get_type (void) G_GNUC_CONST;
#ifdef GDK_WINDOWING_VULKAN
GDK_AVAILABLE_IN_3_90
+const char * gdk_vulkan_strerror (VkResult result);
+
+GDK_AVAILABLE_IN_3_90
VkInstance gdk_vulkan_context_get_instance (GdkVulkanContext *context);
+GDK_AVAILABLE_IN_3_90
+VkPhysicalDevice gdk_vulkan_context_get_physical_device (GdkVulkanContext *context);
+GDK_AVAILABLE_IN_3_90
+VkDevice gdk_vulkan_context_get_device (GdkVulkanContext *context);
+GDK_AVAILABLE_IN_3_90
+VkQueue gdk_vulkan_context_get_queue (GdkVulkanContext *context);
+GDK_AVAILABLE_IN_3_90
+uint32_t gdk_vulkan_context_get_queue_family_index (GdkVulkanContext *context);
+GDK_AVAILABLE_IN_3_90
+VkFormat gdk_vulkan_context_get_image_format (GdkVulkanContext *context);
#endif /* GDK_WINDOWING_VULKAN */
diff --git a/gdk/gdkvulkancontextprivate.h b/gdk/gdkvulkancontextprivate.h
index 696295c..1d56de8 100644
--- a/gdk/gdkvulkancontextprivate.h
+++ b/gdk/gdkvulkancontextprivate.h
@@ -57,7 +57,7 @@ gdk_vulkan_handle_result (VkResult res,
{
if (res != VK_SUCCESS)
{
- GDK_NOTE (VULKAN,g_printerr ("%s(): %d\n", called_function, res));
+ GDK_NOTE (VULKAN,g_printerr ("%s(): %s (%d)\n", called_function, gdk_vulkan_strerror (res), res));
}
return res;
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]