Resize swapchain

This commit is contained in:
David Gonzalez Martin 2024-11-26 06:55:39 -06:00 committed by David
parent e7c90645c0
commit b76e9d7ff3
4 changed files with 231 additions and 92 deletions

View File

@ -22,6 +22,7 @@ EXPORT GraphicsWindow* graphics_window_create(GraphicsWindowCreate create);
EXPORT u8 graphics_window_should_close(GraphicsWindow* window);
EXPORT void graphics_poll_events();
EXPORT GraphicsWindowSize graphics_window_size_get(GraphicsWindow* window);
EXPORT void graphics_window_consume_resize(GraphicsWindow* window);
#ifdef __linux__
typedef unsigned long XID;

View File

@ -32,7 +32,7 @@ fn void run(Arena* arena, char** envp, String compiler_path, CompilerBackend com
{
#if _WIN32
args = (CStringSlice) array_to_slice(((char*[]){
"C:\\Users\\David\\Downloads\\remedybg_0_4_0_7\\remedybg.exe",
"C:\\Users\\David\\Downloads\\remedybg_0_4_0_8\\remedybg.exe",
"-g",
common_compile_and_run_args
}));

View File

@ -6,15 +6,16 @@
global_variable u8 use_x11 = 0;
fn GraphicsWindow* graphics_window_from_glfw(GLFWwindow* window)
STRUCT(GraphicsWindow)
{
return (GraphicsWindow*)window;
}
fn GLFWwindow* glfw_window_from_graphics(GraphicsWindow* window)
{
return (GLFWwindow*)window;
}
GLFWwindow* handle;
u8 resized:1;
u32 width;
u32 height;
};
#define MAX_WINDOW_COUNT (32)
global_variable GraphicsWindow windows[MAX_WINDOW_COUNT];
global_variable u32 window_count = 0;
void graphics_init(u8 should_use_x11)
{
@ -30,17 +31,41 @@ void graphics_init(u8 should_use_x11)
}
}
fn void framebuffer_size_callback(GLFWwindow* w, int width, int height)
{
GraphicsWindow* window = glfwGetWindowUserPointer(w);
assert(window->handle == w);
window->width = width;
window->height = height;
window->resized = 1;
}
typedef void (* GLFWframebuffersizefun)(GLFWwindow* window, int width, int height);
GraphicsWindow* graphics_window_create(GraphicsWindowCreate create)
{
glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
GLFWwindow* window = glfwCreateWindow(create.size.width, create.size.height, string_to_c(create.name), 0, 0);
GraphicsWindow* window = &windows[window_count];
*window = (GraphicsWindow) {
.handle = glfwCreateWindow(create.size.width, create.size.height, string_to_c(create.name), 0, 0),
};
glfwSetWindowUserPointer(window->handle, window);
glfwSetFramebufferSizeCallback(window->handle, &framebuffer_size_callback);
window->width = create.size.width;
window->height = create.size.height;
return graphics_window_from_glfw(window);
return window;
}
void graphics_window_consume_resize(GraphicsWindow* window)
{
assert(window->resized);
window->resized = 0;
}
u8 graphics_window_should_close(GraphicsWindow* window)
{
return glfwWindowShouldClose(glfw_window_from_graphics(window));
return glfwWindowShouldClose(window->handle);
}
void graphics_poll_events()
@ -50,9 +75,8 @@ void graphics_poll_events()
GraphicsWindowSize graphics_window_size_get(GraphicsWindow* window)
{
GLFWwindow* w = glfw_window_from_graphics(window);
GraphicsWindowSize result;
glfwGetWindowSize(w, (int*)&result.width, (int*)&result.height);
glfwGetWindowSize(window->handle, (int*)&result.width, (int*)&result.height);
return result;
}
@ -60,7 +84,7 @@ GraphicsWindowSize graphics_window_size_get(GraphicsWindow* window)
#ifdef _WIN32
HANDLE graphics_win32_window_get(GraphicsWindow* window)
{
return glfwGetWin32Window(glfw_window_from_graphics(window));
return glfwGetWin32Window(window->handle);
}
#endif
@ -72,6 +96,6 @@ Display* graphics_x11_display_get()
Window graphics_x11_window_get(GraphicsWindow* window)
{
return glfwGetX11Window(glfw_window_from_graphics(window));
return glfwGetX11Window(window->handle);
}
#endif

View File

@ -22,12 +22,72 @@ STRUCT(VulkanImage)
VkFormat format;
};
fn String vulkan_result_to_string(VkResult result)
{
switch (result)
{
case_to_name(VK_, SUCCESS);
case_to_name(VK_, NOT_READY);
case_to_name(VK_, TIMEOUT);
case_to_name(VK_, EVENT_SET);
case_to_name(VK_, EVENT_RESET);
case_to_name(VK_, INCOMPLETE);
case_to_name(VK_, ERROR_OUT_OF_HOST_MEMORY);
case_to_name(VK_, ERROR_OUT_OF_DEVICE_MEMORY);
case_to_name(VK_, ERROR_INITIALIZATION_FAILED);
case_to_name(VK_, ERROR_DEVICE_LOST);
case_to_name(VK_, ERROR_MEMORY_MAP_FAILED);
case_to_name(VK_, ERROR_LAYER_NOT_PRESENT);
case_to_name(VK_, ERROR_EXTENSION_NOT_PRESENT);
case_to_name(VK_, ERROR_FEATURE_NOT_PRESENT);
case_to_name(VK_, ERROR_INCOMPATIBLE_DRIVER);
case_to_name(VK_, ERROR_TOO_MANY_OBJECTS);
case_to_name(VK_, ERROR_FORMAT_NOT_SUPPORTED);
case_to_name(VK_, ERROR_FRAGMENTED_POOL);
case_to_name(VK_, ERROR_UNKNOWN);
case_to_name(VK_, ERROR_OUT_OF_POOL_MEMORY);
case_to_name(VK_, ERROR_INVALID_EXTERNAL_HANDLE);
case_to_name(VK_, ERROR_FRAGMENTATION);
case_to_name(VK_, ERROR_INVALID_OPAQUE_CAPTURE_ADDRESS);
case_to_name(VK_, PIPELINE_COMPILE_REQUIRED);
case_to_name(VK_, ERROR_SURFACE_LOST_KHR);
case_to_name(VK_, ERROR_NATIVE_WINDOW_IN_USE_KHR);
case_to_name(VK_, SUBOPTIMAL_KHR);
case_to_name(VK_, ERROR_OUT_OF_DATE_KHR);
case_to_name(VK_, ERROR_INCOMPATIBLE_DISPLAY_KHR);
case_to_name(VK_, ERROR_VALIDATION_FAILED_EXT);
case_to_name(VK_, ERROR_INVALID_SHADER_NV);
case_to_name(VK_, ERROR_IMAGE_USAGE_NOT_SUPPORTED_KHR);
case_to_name(VK_, ERROR_VIDEO_PICTURE_LAYOUT_NOT_SUPPORTED_KHR);
case_to_name(VK_, ERROR_VIDEO_PROFILE_OPERATION_NOT_SUPPORTED_KHR);
case_to_name(VK_, ERROR_VIDEO_PROFILE_FORMAT_NOT_SUPPORTED_KHR);
case_to_name(VK_, ERROR_VIDEO_PROFILE_CODEC_NOT_SUPPORTED_KHR);
case_to_name(VK_, ERROR_VIDEO_STD_VERSION_NOT_SUPPORTED_KHR);
case_to_name(VK_, ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT);
case_to_name(VK_, ERROR_NOT_PERMITTED_KHR);
case_to_name(VK_, ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT);
case_to_name(VK_, THREAD_IDLE_KHR);
case_to_name(VK_, THREAD_DONE_KHR);
case_to_name(VK_, OPERATION_DEFERRED_KHR);
case_to_name(VK_, OPERATION_NOT_DEFERRED_KHR);
case_to_name(VK_, ERROR_INVALID_VIDEO_STD_PARAMETERS_KHR);
case_to_name(VK_, ERROR_COMPRESSION_EXHAUSTED_EXT);
case_to_name(VK_, INCOMPATIBLE_SHADER_BINARY_EXT);
case_to_name(VK_, PIPELINE_BINARY_MISSING_KHR);
case_to_name(VK_, ERROR_NOT_ENOUGH_SPACE_KHR);
case_to_name(VK_, RESULT_MAX_ENUM);
}
}
[[noreturn]] [[gnu::cold]] fn void wrong_vulkan_result(VkResult result, String call_string, String file, int line)
{
unused(result);
unused(call_string);
unused(file);
unused(line);
String result_name = vulkan_result_to_string(result);
print("Wrong Vulkan result {s} at \"{s}\" {s}:{u32}\n", result_name, call_string, file, line);
trap();
}
@ -38,7 +98,7 @@ STRUCT(VulkanImage)
#define vkok_swapchain(call) do {\
VkResult result = call; \
if (unlikely(result != VK_SUCCESS && result != VK_SUBOPTIMAL_KHR)) wrong_vulkan_result(result, strlit(#call), strlit(__FILE__), __LINE__); \
if (unlikely(result != VK_SUCCESS)) wrong_vulkan_result(result, strlit(#call), strlit(__FILE__), __LINE__); \
} while(0)
fn u8 vk_layer_is_supported(String layer_name)
@ -109,7 +169,7 @@ fn String message_type_to_string(VkDebugUtilsMessageTypeFlagBitsEXT message_type
}
}
fn VkBool32 VKAPI_CALL debug_callback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity, VkDebugUtilsMessageTypeFlagBitsEXT message_type, const VkDebugUtilsMessengerCallbackDataEXT* callback_data, void* user_data)
fn VkBool32 VKAPI_CALL debug_callback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity, VkDebugUtilsMessageTypeFlagsEXT message_type, const VkDebugUtilsMessengerCallbackDataEXT* callback_data, void* user_data)
{
unused(message_severity);
unused(message_type);
@ -401,7 +461,8 @@ STRUCT(RenderWindow)
{
GraphicsWindow* graphics_window;
VkSwapchainKHR swapchain;
VulkanImage render_image;
VkSurfaceKHR surface;
VkFormat swapchain_image_format;
u32 width;
u32 height;
u32 last_width;
@ -409,9 +470,9 @@ STRUCT(RenderWindow)
u32 frame_index;
u32 swapchain_image_index;
u32 swapchain_image_count;
VulkanImage render_image;
VkImage swapchain_images[MAX_SWAPCHAIN_IMAGE_COUNT];
VkImageView swapchain_image_views[MAX_SWAPCHAIN_IMAGE_COUNT];
VkFormat swapchain_image_format;
WindowFrame frames[MAX_FRAMES];
};
@ -523,6 +584,7 @@ Renderer* renderer_initialize()
{
u32 physical_device_count;
VkPhysicalDevice physical_devices[256];
vkok(vkEnumeratePhysicalDevices(result->instance, &physical_device_count, 0));
@ -542,6 +604,8 @@ Renderer* renderer_initialize()
result->physical_device = physical_devices[0];
}
vkGetPhysicalDeviceMemoryProperties(result->physical_device, &result->memory_properties);
u32 graphics_queue_family_index;
{
u32 present_queue_family_index;
@ -696,91 +760,86 @@ Renderer* renderer_initialize()
return result;
}
RenderWindow* renderer_window_initialize(Renderer* renderer, GraphicsWindow* window)
fn void destroy_image(Renderer* renderer, VulkanImage image)
{
RenderWindow* result = &renderer_window_memory;
result->graphics_window = window;
VkSurfaceKHR surface;
{
#ifdef VK_USE_PLATFORM_WIN32_KHR
VkWin32SurfaceCreateInfoKHR create_info = {
.sType = VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR,
.pNext = 0,
.flags = 0,
.hinstance = os_windows_get_module_handle(),
.hwnd = graphics_win32_window_get(window),
};
vkok(vkCreateWin32SurfaceKHR(renderer->instance, &create_info, renderer->allocator, &surface));
#endif
#ifdef VK_USE_PLATFORM_XLIB_KHR
VkXlibSurfaceCreateInfoKHR create_info = {
.sType = VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR,
.pNext = 0,
.flags = 0,
.dpy = graphics_x11_display_get(),
.window = graphics_x11_window_get(window),
};
vkok(vkCreateXlibSurfaceKHR(renderer->instance, &create_info, renderer->allocator, &surface));
#endif
#ifdef VK_USE_PLATFORM_WAYLAND_KHR
VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME,
#endif
#ifdef VK_USE_PLATFORM_MACOS_MVK
VK_KHR_COCOA_SURFACE_EXTENSION_NAME,
#endif
}
vkDestroyImageView(renderer->device, image.view, renderer->allocator);
vkDestroyImage(renderer->device, image.handle, renderer->allocator);
vkFreeMemory(renderer->device, image.memory, renderer->allocator);
}
VkSurfaceCapabilitiesKHR original_capabilities;
vkok(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(renderer->physical_device, surface, &original_capabilities));
fn void swapchain_recreate(Renderer* renderer, RenderWindow* window, VkSurfaceCapabilitiesKHR surface_capabilities)
{
VkSwapchainKHR old_swapchain = window->swapchain;
VkImageView old_swapchain_image_views[MAX_SWAPCHAIN_IMAGE_COUNT];
if (old_swapchain)
{
vkok(vkDeviceWaitIdle(renderer->device));
for (u32 i = 0; i < window->swapchain_image_count; i += 1)
{
old_swapchain_image_views[i] = window->swapchain_image_views[i];
}
}
u32 queue_family_indices[] = { renderer->graphics_queue_family_index };
VkImageUsageFlags swapchain_image_usage_flags = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT;
result->swapchain_image_format = VK_FORMAT_B8G8R8A8_UNORM;
result->width = original_capabilities.currentExtent.width;
result->height = original_capabilities.currentExtent.height;
result->last_width = result->width;
result->last_height = result->height;
window->swapchain_image_format = VK_FORMAT_B8G8R8A8_UNORM;
window->last_width = window->width;
window->last_height = window->height;
window->width = surface_capabilities.currentExtent.width;
window->height = surface_capabilities.currentExtent.height;
VkSwapchainCreateInfoKHR swapchain_create_info = {
.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR,
.pNext = 0,
.flags = 0,
.surface = surface,
.minImageCount = original_capabilities.minImageCount,
.imageFormat = result->swapchain_image_format,
.surface = window->surface,
.minImageCount = surface_capabilities.minImageCount,
.imageFormat = window->swapchain_image_format,
.imageColorSpace = VK_COLOR_SPACE_SRGB_NONLINEAR_KHR,
.imageExtent = original_capabilities.currentExtent,
.imageExtent = surface_capabilities.currentExtent,
.imageArrayLayers = 1,
.imageUsage = swapchain_image_usage_flags,
.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE,
.queueFamilyIndexCount = array_length(queue_family_indices),
.pQueueFamilyIndices = queue_family_indices,
.preTransform = original_capabilities.currentTransform,
.preTransform = surface_capabilities.currentTransform,
.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR,
.presentMode = VK_PRESENT_MODE_FIFO_KHR,
.clipped = 0,
.oldSwapchain = result->swapchain,
.oldSwapchain = window->swapchain,
};
vkok(vkCreateSwapchainKHR(renderer->device, &swapchain_create_info, renderer->allocator, &window->swapchain));
assert(window->swapchain != old_swapchain);
if (old_swapchain)
{
VkSwapchainKHR new_swapchain;
vkok(vkCreateSwapchainKHR(renderer->device, &swapchain_create_info, renderer->allocator, &new_swapchain));
result->swapchain = new_swapchain;
for (u32 i = 0; i < window->swapchain_image_count; i += 1)
{
vkDestroyImageView(renderer->device, old_swapchain_image_views[i], renderer->allocator);
}
vkDestroySwapchainKHR(renderer->device, old_swapchain, renderer->allocator);
destroy_image(renderer, window->render_image);
}
{
vkok(vkGetSwapchainImagesKHR(renderer->device, result->swapchain, &result->swapchain_image_count, 0));
vkok(vkGetSwapchainImagesKHR(renderer->device, window->swapchain, &window->swapchain_image_count, 0));
if (result->swapchain_image_count == 0)
if (window->swapchain_image_count == 0)
{
failed_execution();
}
if (result->swapchain_image_count > array_length(result->swapchain_images))
if (window->swapchain_image_count > array_length(window->swapchain_images))
{
failed_execution();
}
vkok(vkGetSwapchainImagesKHR(renderer->device, result->swapchain, &result->swapchain_image_count, result->swapchain_images));
vkok(vkGetSwapchainImagesKHR(renderer->device, window->swapchain, &window->swapchain_image_count, window->swapchain_images));
// VkImageViewUsageCreateInfo image_view_usage_create_info = {
// .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_USAGE_CREATE_INFO,
@ -788,13 +847,13 @@ RenderWindow* renderer_window_initialize(Renderer* renderer, GraphicsWindow* win
// .usage = swapchain_create_info.imageUsage,
// };
for (u32 i = 0; i < result->swapchain_image_count; i += 1)
for (u32 i = 0; i < window->swapchain_image_count; i += 1)
{
VkImageViewCreateInfo create_info = {
.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
// .pNext = &image_view_usage_create_info,
.flags = 0,
.image = result->swapchain_images[i],
.image = window->swapchain_images[i],
.viewType = VK_IMAGE_VIEW_TYPE_2D,
.format = swapchain_create_info.imageFormat,
.components = {
@ -812,21 +871,55 @@ RenderWindow* renderer_window_initialize(Renderer* renderer, GraphicsWindow* win
},
};
vkok(vkCreateImageView(renderer->device, &create_info, renderer->allocator, &result->swapchain_image_views[i]));
vkok(vkCreateImageView(renderer->device, &create_info, renderer->allocator, &window->swapchain_image_views[i]));
}
}
vkGetPhysicalDeviceMemoryProperties(renderer->physical_device, &renderer->memory_properties);
auto initial_window_size = graphics_window_size_get(window);
result->render_image = vk_image_create(renderer->device, renderer->allocator, renderer->memory_properties, (VulkanImageCreate) {
.width = initial_window_size.width,
.height = initial_window_size.height,
window->render_image = vk_image_create(renderer->device, renderer->allocator, renderer->memory_properties, (VulkanImageCreate) {
.width = surface_capabilities.currentExtent.width,
.height = surface_capabilities.currentExtent.height,
.mip_levels = 1,
.format = result->swapchain_image_format,
.format = window->swapchain_image_format,
.usage = VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_STORAGE_BIT | VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
});
}
RenderWindow* renderer_window_initialize(Renderer* renderer, GraphicsWindow* window)
{
RenderWindow* result = &renderer_window_memory;
result->graphics_window = window;
{
#ifdef VK_USE_PLATFORM_WIN32_KHR
VkWin32SurfaceCreateInfoKHR create_info = {
.sType = VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR,
.pNext = 0,
.flags = 0,
.hinstance = os_windows_get_module_handle(),
.hwnd = graphics_win32_window_get(window),
};
vkok(vkCreateWin32SurfaceKHR(renderer->instance, &create_info, renderer->allocator, &result->surface));
#endif
#ifdef VK_USE_PLATFORM_XLIB_KHR
VkXlibSurfaceCreateInfoKHR create_info = {
.sType = VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR,
.pNext = 0,
.flags = 0,
.dpy = graphics_x11_display_get(),
.window = graphics_x11_window_get(window),
};
vkok(vkCreateXlibSurfaceKHR(renderer->instance, &create_info, renderer->allocator, &result->surface));
#endif
#ifdef VK_USE_PLATFORM_WAYLAND_KHR
VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME,
#endif
#ifdef VK_USE_PLATFORM_MACOS_MVK
VK_KHR_COCOA_SURFACE_EXTENSION_NAME,
#endif
}
VkSurfaceCapabilitiesKHR surface_capabilities;
vkok(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(renderer->physical_device, result->surface, &surface_capabilities));
swapchain_recreate(renderer, result, surface_capabilities);
VkCommandPoolCreateInfo command_pool_create_info = {
.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
@ -1384,11 +1477,23 @@ fn void queue_present(Renderer* renderer, RenderWindow* window)
.pResults = results,
};
vkok_swapchain(vkQueuePresentKHR(renderer->graphics_queue, &present_info));
for (u32 i = 0; i < array_length(results); i += 1)
VkResult present_result = vkQueuePresentKHR(renderer->graphics_queue, &present_info);
if (present_result == VK_SUCCESS)
{
vkok_swapchain(results[i]);
for (u32 i = 0; i < array_length(results); i += 1)
{
vkok_swapchain(results[i]);
}
}
else if (present_result == VK_ERROR_OUT_OF_DATE_KHR || present_result == VK_SUBOPTIMAL_KHR)
{
VkSurfaceCapabilitiesKHR surface_capabilities;
vkok(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(renderer->physical_device, window->surface, &surface_capabilities));
swapchain_recreate(renderer, window, surface_capabilities);
}
else
{
vkok(present_result);
}
}
@ -1401,17 +1506,24 @@ GraphicsWindowSize renderer_window_frame_begin(Renderer* renderer, RenderWindow*
VkBool32 wait_all = 1;
vkok(vkWaitForFences(renderer->device, fence_count, &frame->render_fence, wait_all, timeout));
VkFence image_fence = 0;
vkok_swapchain(vkAcquireNextImageKHR(renderer->device, window->swapchain, timeout, frame->swapchain_semaphore, image_fence, &window->swapchain_image_index));
VkResult next_image_result = vkAcquireNextImageKHR(renderer->device, window->swapchain, timeout, frame->swapchain_semaphore, image_fence, &window->swapchain_image_index);
if (next_image_result == VK_ERROR_OUT_OF_DATE_KHR)
{
VkSurfaceCapabilitiesKHR surface_capabilities;
vkok(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(renderer->physical_device, window->surface, &surface_capabilities));
swapchain_recreate(renderer, window, surface_capabilities);
}
else if (next_image_result != VK_SUCCESS && next_image_result != VK_SUBOPTIMAL_KHR)
{
vkok(next_image_result);
}
vkok(vkResetFences(renderer->device, fence_count, &frame->render_fence));
VkCommandBufferResetFlags reset_flags = 0;
vkok(vkResetCommandBuffer(frame->command_buffer, reset_flags));
window->last_width = window->width;
window->last_height = window->height;
auto window_size = graphics_window_size_get(window->graphics_window);
window->width = window_size.width;
window->height = window_size.height;
return window_size;
}
@ -1896,3 +2008,5 @@ void window_draw_indexed(RenderWindow* window, u32 index_count, u32 instance_cou
auto* frame = window_frame(window);
vkCmdDrawIndexed(frame->command_buffer, index_count, instance_count, first_index, vertex_offset, first_instance);
}