mirror of https://github.com/snes9xgit/snes9x.git
Restrict max latency to 3 frames.
This commit is contained in:
parent
e4768f9b80
commit
23cf7ec056
|
@ -76,7 +76,7 @@ bool Context::init_wayland(wl_display *dpy, wl_surface *parent, int initial_widt
|
||||||
init_command_pool();
|
init_command_pool();
|
||||||
init_descriptor_pool();
|
init_descriptor_pool();
|
||||||
create_swapchain(initial_width, initial_height);
|
create_swapchain(initial_width, initial_height);
|
||||||
device.waitIdle();
|
wait_idle();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -90,9 +90,7 @@ bool Context::init(int preferred_device)
|
||||||
init_descriptor_pool();
|
init_descriptor_pool();
|
||||||
|
|
||||||
create_swapchain();
|
create_swapchain();
|
||||||
|
wait_idle();
|
||||||
device.waitIdle();
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -239,14 +239,14 @@ bool ShaderChain::load_shader_preset(std::string filename)
|
||||||
std::array<vk::DescriptorPoolSize, 2> descriptor_pool_sizes;
|
std::array<vk::DescriptorPoolSize, 2> descriptor_pool_sizes;
|
||||||
descriptor_pool_sizes[0]
|
descriptor_pool_sizes[0]
|
||||||
.setType(vk::DescriptorType::eUniformBuffer)
|
.setType(vk::DescriptorType::eUniformBuffer)
|
||||||
.setDescriptorCount(num_ubos * context->swapchain->get_num_frames());
|
.setDescriptorCount(num_ubos * queue_size);
|
||||||
descriptor_pool_sizes[1]
|
descriptor_pool_sizes[1]
|
||||||
.setType(vk::DescriptorType::eCombinedImageSampler)
|
.setType(vk::DescriptorType::eCombinedImageSampler)
|
||||||
.setDescriptorCount(num_samplers * context->swapchain->get_num_frames());
|
.setDescriptorCount(num_samplers * queue_size);
|
||||||
|
|
||||||
auto descriptor_pool_create_info = vk::DescriptorPoolCreateInfo{}
|
auto descriptor_pool_create_info = vk::DescriptorPoolCreateInfo{}
|
||||||
.setPoolSizes(descriptor_pool_sizes)
|
.setPoolSizes(descriptor_pool_sizes)
|
||||||
.setMaxSets(pipelines.size() * context->swapchain->get_num_frames())
|
.setMaxSets(pipelines.size() * queue_size)
|
||||||
.setFlags(vk::DescriptorPoolCreateFlagBits::eFreeDescriptorSet);
|
.setFlags(vk::DescriptorPoolCreateFlagBits::eFreeDescriptorSet);
|
||||||
|
|
||||||
descriptor_pool = context->device.createDescriptorPoolUnique(descriptor_pool_create_info);
|
descriptor_pool = context->device.createDescriptorPoolUnique(descriptor_pool_create_info);
|
||||||
|
@ -276,6 +276,8 @@ bool ShaderChain::load_shader_preset(std::string filename)
|
||||||
context->allocator.flushAllocation(vertex_buffer_allocation, 0, sizeof(vertex_data));
|
context->allocator.flushAllocation(vertex_buffer_allocation, 0, sizeof(vertex_data));
|
||||||
|
|
||||||
frame_count = 0;
|
frame_count = 0;
|
||||||
|
current_frame_index = 0;
|
||||||
|
last_frame_index = 2;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -393,7 +395,6 @@ void ShaderChain::do_frame(uint8_t *data, int width, int height, int stride, vk:
|
||||||
|
|
||||||
auto cmd = context->swapchain->get_cmd();
|
auto cmd = context->swapchain->get_cmd();
|
||||||
|
|
||||||
current_frame_index = context->swapchain->get_current_frame();
|
|
||||||
update_and_propagate_sizes(width, height, viewport_width, viewport_height);
|
update_and_propagate_sizes(width, height, viewport_width, viewport_height);
|
||||||
|
|
||||||
update_framebuffers(cmd, current_frame_index);
|
update_framebuffers(cmd, current_frame_index);
|
||||||
|
@ -508,6 +509,7 @@ void ShaderChain::do_frame(uint8_t *data, int width, int height, int stride, vk:
|
||||||
context->swapchain->end_frame();
|
context->swapchain->end_frame();
|
||||||
|
|
||||||
last_frame_index = current_frame_index;
|
last_frame_index = current_frame_index;
|
||||||
|
current_frame_index = (current_frame_index + 1) % queue_size;
|
||||||
frame_count++;
|
frame_count++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,8 @@ namespace Vulkan
|
||||||
class ShaderChain
|
class ShaderChain
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
const int queue_size = 3;
|
||||||
|
|
||||||
ShaderChain(Context *context_);
|
ShaderChain(Context *context_);
|
||||||
~ShaderChain();
|
~ShaderChain();
|
||||||
bool load_shader_preset(std::string filename);
|
bool load_shader_preset(std::string filename);
|
||||||
|
|
|
@ -34,7 +34,7 @@ class SlangPipeline
|
||||||
vk::UniqueDescriptorSet descriptor_set;
|
vk::UniqueDescriptorSet descriptor_set;
|
||||||
PipelineImage image;
|
PipelineImage image;
|
||||||
vk::UniqueFence fence;
|
vk::UniqueFence fence;
|
||||||
} frame[4];
|
} frame[3];
|
||||||
|
|
||||||
vk::Buffer uniform_buffer;
|
vk::Buffer uniform_buffer;
|
||||||
vma::Allocation uniform_buffer_allocation;
|
vma::Allocation uniform_buffer_allocation;
|
||||||
|
|
|
@ -23,7 +23,6 @@ bool Swapchain::set_vsync(bool new_setting)
|
||||||
if (new_setting == vsync)
|
if (new_setting == vsync)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
vsync = new_setting;
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -75,8 +74,7 @@ void Swapchain::create_render_pass()
|
||||||
bool Swapchain::recreate(int new_width, int new_height)
|
bool Swapchain::recreate(int new_width, int new_height)
|
||||||
{
|
{
|
||||||
device.waitIdle();
|
device.waitIdle();
|
||||||
|
return create(num_swapchain_images, new_width, new_height);
|
||||||
return create(num_frames, new_width, new_height);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
vk::Image Swapchain::get_image()
|
vk::Image Swapchain::get_image()
|
||||||
|
@ -84,17 +82,17 @@ vk::Image Swapchain::get_image()
|
||||||
return imageviewfbs[current_swapchain_image].image;
|
return imageviewfbs[current_swapchain_image].image;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Swapchain::create(unsigned int desired_num_frames, int new_width, int new_height)
|
bool Swapchain::create(unsigned int desired_num_swapchain_images, int new_width, int new_height)
|
||||||
{
|
{
|
||||||
frames.clear();
|
frames.clear();
|
||||||
imageviewfbs.clear();
|
imageviewfbs.clear();
|
||||||
|
|
||||||
auto surface_capabilities = physical_device.getSurfaceCapabilitiesKHR(surface);
|
auto surface_capabilities = physical_device.getSurfaceCapabilitiesKHR(surface);
|
||||||
|
|
||||||
if (surface_capabilities.minImageCount > desired_num_frames)
|
if (surface_capabilities.minImageCount > desired_num_swapchain_images)
|
||||||
num_frames = surface_capabilities.minImageCount;
|
num_swapchain_images = surface_capabilities.minImageCount;
|
||||||
else
|
else
|
||||||
num_frames = desired_num_frames;
|
num_swapchain_images = desired_num_swapchain_images;
|
||||||
|
|
||||||
extents = surface_capabilities.currentExtent;
|
extents = surface_capabilities.currentExtent;
|
||||||
|
|
||||||
|
@ -113,7 +111,7 @@ bool Swapchain::create(unsigned int desired_num_frames, int new_width, int new_h
|
||||||
}
|
}
|
||||||
|
|
||||||
auto swapchain_create_info = vk::SwapchainCreateInfoKHR{}
|
auto swapchain_create_info = vk::SwapchainCreateInfoKHR{}
|
||||||
.setMinImageCount(num_frames)
|
.setMinImageCount(num_swapchain_images)
|
||||||
.setImageFormat(vk::Format::eB8G8R8A8Unorm)
|
.setImageFormat(vk::Format::eB8G8R8A8Unorm)
|
||||||
.setImageExtent(extents)
|
.setImageExtent(extents)
|
||||||
.setImageColorSpace(vk::ColorSpaceKHR::eSrgbNonlinear)
|
.setImageColorSpace(vk::ColorSpaceKHR::eSrgbNonlinear)
|
||||||
|
@ -133,18 +131,18 @@ bool Swapchain::create(unsigned int desired_num_frames, int new_width, int new_h
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
auto swapchain_images = device.getSwapchainImagesKHR(swapchain_object.get());
|
auto swapchain_images = device.getSwapchainImagesKHR(swapchain_object.get());
|
||||||
vk::CommandBufferAllocateInfo command_buffer_allocate_info(command_pool, vk::CommandBufferLevel::ePrimary, num_frames);
|
vk::CommandBufferAllocateInfo command_buffer_allocate_info(command_pool, vk::CommandBufferLevel::ePrimary, max_latency);
|
||||||
auto command_buffers = device.allocateCommandBuffersUnique(command_buffer_allocate_info);
|
auto command_buffers = device.allocateCommandBuffersUnique(command_buffer_allocate_info);
|
||||||
|
|
||||||
if (imageviewfbs.size() > num_frames)
|
if (imageviewfbs.size() > num_swapchain_images)
|
||||||
num_frames = imageviewfbs.size();
|
num_swapchain_images = imageviewfbs.size();
|
||||||
|
|
||||||
frames.resize(num_frames);
|
frames.resize(max_latency);
|
||||||
imageviewfbs.resize(num_frames);
|
imageviewfbs.resize(num_swapchain_images);
|
||||||
|
|
||||||
vk::FenceCreateInfo fence_create_info(vk::FenceCreateFlagBits::eSignaled);
|
vk::FenceCreateInfo fence_create_info(vk::FenceCreateFlagBits::eSignaled);
|
||||||
|
|
||||||
for (unsigned int i = 0; i < num_frames; i++)
|
for (int i = 0; i < max_latency; i++)
|
||||||
{
|
{
|
||||||
// Create frame queue resources
|
// Create frame queue resources
|
||||||
auto &frame = frames[i];
|
auto &frame = frames[i];
|
||||||
|
@ -155,7 +153,7 @@ bool Swapchain::create(unsigned int desired_num_frames, int new_width, int new_h
|
||||||
}
|
}
|
||||||
current_frame = 0;
|
current_frame = 0;
|
||||||
|
|
||||||
for (unsigned int i = 0; i < num_frames; i++)
|
for (unsigned int i = 0; i < num_swapchain_images; i++)
|
||||||
{
|
{
|
||||||
// Create resources associated with swapchain images
|
// Create resources associated with swapchain images
|
||||||
auto &image = imageviewfbs[i];
|
auto &image = imageviewfbs[i];
|
||||||
|
@ -216,16 +214,17 @@ bool Swapchain::begin_frame()
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
device.resetFences(frame.fence.get());
|
|
||||||
current_swapchain_image = result_value.value;
|
current_swapchain_image = result_value.value;
|
||||||
|
|
||||||
|
device.resetFences(frame.fence.get());
|
||||||
|
|
||||||
vk::CommandBufferBeginInfo command_buffer_begin_info(vk::CommandBufferUsageFlags{vk::CommandBufferUsageFlagBits::eOneTimeSubmit});
|
vk::CommandBufferBeginInfo command_buffer_begin_info(vk::CommandBufferUsageFlags{vk::CommandBufferUsageFlagBits::eOneTimeSubmit});
|
||||||
frame.command_buffer->begin(command_buffer_begin_info);
|
frame.command_buffer->begin(command_buffer_begin_info);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Swapchain::end_frame(vk::Fence extra_fence)
|
bool Swapchain::end_frame()
|
||||||
{
|
{
|
||||||
auto &frame = frames[current_frame];
|
auto &frame = frames[current_frame];
|
||||||
frame.command_buffer->end();
|
frame.command_buffer->end();
|
||||||
|
@ -237,8 +236,6 @@ bool Swapchain::end_frame(vk::Fence extra_fence)
|
||||||
frame.command_buffer.get(),
|
frame.command_buffer.get(),
|
||||||
frame.complete.get());
|
frame.complete.get());
|
||||||
|
|
||||||
if (extra_fence)
|
|
||||||
queue.submit(submit_info, extra_fence);
|
|
||||||
queue.submit(submit_info, frame.fence.get());
|
queue.submit(submit_info, frame.fence.get());
|
||||||
|
|
||||||
auto present_info = vk::PresentInfoKHR{}
|
auto present_info = vk::PresentInfoKHR{}
|
||||||
|
@ -248,7 +245,7 @@ bool Swapchain::end_frame(vk::Fence extra_fence)
|
||||||
|
|
||||||
auto result = queue.presentKHR(present_info);
|
auto result = queue.presentKHR(present_info);
|
||||||
|
|
||||||
current_frame = (current_frame + 1) % num_frames;
|
current_frame = (current_frame + 1) % max_latency;
|
||||||
|
|
||||||
if (result != vk::Result::eSuccess)
|
if (result != vk::Result::eSuccess)
|
||||||
return false;
|
return false;
|
||||||
|
@ -308,7 +305,7 @@ vk::RenderPass &Swapchain::get_render_pass()
|
||||||
|
|
||||||
unsigned int Swapchain::get_num_frames()
|
unsigned int Swapchain::get_num_frames()
|
||||||
{
|
{
|
||||||
return num_frames;
|
return max_latency;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Vulkan
|
} // namespace Vulkan
|
|
@ -3,6 +3,7 @@
|
||||||
#include "vulkan/vulkan.hpp"
|
#include "vulkan/vulkan.hpp"
|
||||||
#include "vulkan/vulkan_handles.hpp"
|
#include "vulkan/vulkan_handles.hpp"
|
||||||
#include "vulkan/vulkan_structs.hpp"
|
#include "vulkan/vulkan_structs.hpp"
|
||||||
|
#include <deque>
|
||||||
|
|
||||||
namespace Vulkan
|
namespace Vulkan
|
||||||
{
|
{
|
||||||
|
@ -10,6 +11,8 @@ namespace Vulkan
|
||||||
class Swapchain
|
class Swapchain
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
const int max_latency = 3;
|
||||||
|
|
||||||
Swapchain(vk::Device device,
|
Swapchain(vk::Device device,
|
||||||
vk::PhysicalDevice physical_device,
|
vk::PhysicalDevice physical_device,
|
||||||
vk::Queue queue,
|
vk::Queue queue,
|
||||||
|
@ -22,7 +25,7 @@ class Swapchain
|
||||||
void begin_render_pass();
|
void begin_render_pass();
|
||||||
void end_render_pass();
|
void end_render_pass();
|
||||||
bool wait_on_frame(int frame_num);
|
bool wait_on_frame(int frame_num);
|
||||||
bool end_frame(vk::Fence extra_fence = nullptr);
|
bool end_frame();
|
||||||
// Returns true if vsync setting was changed, false if it was the same
|
// Returns true if vsync setting was changed, false if it was the same
|
||||||
bool set_vsync(bool on);
|
bool set_vsync(bool on);
|
||||||
|
|
||||||
|
@ -60,7 +63,7 @@ class Swapchain
|
||||||
|
|
||||||
unsigned int current_frame = 0;
|
unsigned int current_frame = 0;
|
||||||
unsigned int current_swapchain_image = 0;
|
unsigned int current_swapchain_image = 0;
|
||||||
unsigned int num_frames = 0;
|
unsigned int num_swapchain_images = 0;
|
||||||
bool vsync = true;
|
bool vsync = true;
|
||||||
std::vector<Frame> frames;
|
std::vector<Frame> frames;
|
||||||
std::vector<ImageViewFB> imageviewfbs;
|
std::vector<ImageViewFB> imageviewfbs;
|
||||||
|
|
Loading…
Reference in New Issue