Use Arc to store reference and store dependencies for each vulkan types
Some checks failed
Build legacy Nix package on Ubuntu / build (push) Failing after 0s
Some checks failed
Build legacy Nix package on Ubuntu / build (push) Failing after 0s
Arc it's used because later, i can use on multi-threaded programs
This commit is contained in:
parent
1cb9309a56
commit
174e12591c
5 changed files with 290 additions and 320 deletions
|
@ -1,15 +1,18 @@
|
||||||
use crate::vulkan::{VkInstance, VkPhysicalDevice};
|
use std::sync::Arc;
|
||||||
|
use crate::vulkan::{VkInstance, VkPhysicalDevice, LOG_TARGET};
|
||||||
use ash::prelude::VkResult;
|
use ash::prelude::VkResult;
|
||||||
use ash::vk;
|
use ash::vk;
|
||||||
|
|
||||||
pub struct VkDevice {
|
pub struct VkDevice {
|
||||||
|
instance: Arc<VkInstance>,
|
||||||
pub(super) handle: ash::Device,
|
pub(super) handle: ash::Device,
|
||||||
|
pub(super) swapchain_loader: ash::khr::swapchain::Device,
|
||||||
queue_family_index: u32,
|
queue_family_index: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VkDevice {
|
impl VkDevice {
|
||||||
pub(super) fn new_graphics_device(
|
pub(super) fn new_graphics_device(
|
||||||
instance: &VkInstance,
|
instance: Arc<VkInstance>,
|
||||||
physical_device: &VkPhysicalDevice,
|
physical_device: &VkPhysicalDevice,
|
||||||
queue_family_index: u32,
|
queue_family_index: u32,
|
||||||
) -> anyhow::Result<Self> {
|
) -> anyhow::Result<Self> {
|
||||||
|
@ -33,11 +36,17 @@ impl VkDevice {
|
||||||
.enabled_extension_names(&device_extension_names_raw)
|
.enabled_extension_names(&device_extension_names_raw)
|
||||||
.enabled_features(&features);
|
.enabled_features(&features);
|
||||||
|
|
||||||
let device = instance
|
let device = unsafe {
|
||||||
.create_device(physical_device, &device_create_info, None)?;
|
instance.handle.create_device(physical_device.handle, &device_create_info, None)?
|
||||||
|
};
|
||||||
|
log::debug!(target: LOG_TARGET, "Device created ({:?})", device.handle());
|
||||||
|
|
||||||
|
let swapchain_loader = ash::khr::swapchain::Device::new(&instance.handle, &device);
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
instance,
|
||||||
handle: device,
|
handle: device,
|
||||||
|
swapchain_loader,
|
||||||
queue_family_index,
|
queue_family_index,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -93,6 +102,7 @@ impl Drop for VkDevice {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe {
|
||||||
self.handle.destroy_device(None);
|
self.handle.destroy_device(None);
|
||||||
|
log::debug!(target: LOG_TARGET, "Device destroyed ({:?})", self.handle.handle());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -13,6 +13,7 @@ use winit::raw_window_handle::{HasDisplayHandle, HasWindowHandle};
|
||||||
pub struct VkInstance {
|
pub struct VkInstance {
|
||||||
pub(super) entry: Entry,
|
pub(super) entry: Entry,
|
||||||
pub(super) handle: Instance,
|
pub(super) handle: Instance,
|
||||||
|
pub(super) surface_loader: surface::Instance,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VkInstance {
|
impl VkInstance {
|
||||||
|
@ -79,11 +80,14 @@ impl VkInstance {
|
||||||
.expect("Instance creation error")
|
.expect("Instance creation error")
|
||||||
};
|
};
|
||||||
|
|
||||||
log::debug!(target: LOG_TARGET, "Vulkan instance created");
|
let surface_loader = surface::Instance::new(&entry, &instance);
|
||||||
|
|
||||||
|
log::debug!(target: LOG_TARGET, "Vulkan instance created ({:?})", instance.handle());
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
entry,
|
entry,
|
||||||
handle: instance,
|
handle: instance,
|
||||||
|
surface_loader
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -94,52 +98,12 @@ impl VkInstance {
|
||||||
.iter().map(|physical_device| VkPhysicalDevice::new(&self.handle, *physical_device))
|
.iter().map(|physical_device| VkPhysicalDevice::new(&self.handle, *physical_device))
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn create_surface(
|
|
||||||
&self,
|
|
||||||
window: &crate::display::Window,
|
|
||||||
) -> anyhow::Result<VkSurface> {
|
|
||||||
let window_handle = window.handle()
|
|
||||||
.ok_or_else(|| anyhow::anyhow!("Window handle is not available."))?;
|
|
||||||
|
|
||||||
let surface_loader = surface::Instance::new(&self.entry, &self.handle);
|
|
||||||
|
|
||||||
let surface = unsafe {
|
|
||||||
ash_window::create_surface(
|
|
||||||
&self.entry,
|
|
||||||
&self.handle,
|
|
||||||
window_handle.display_handle()?.as_raw(),
|
|
||||||
window_handle.window_handle()?.as_raw(),
|
|
||||||
None,
|
|
||||||
)?
|
|
||||||
};
|
|
||||||
|
|
||||||
log::debug!(target: LOG_TARGET, "Surface created");
|
|
||||||
|
|
||||||
Ok(VkSurface::new(
|
|
||||||
surface_loader,
|
|
||||||
surface,
|
|
||||||
))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn create_device(
|
|
||||||
&self,
|
|
||||||
physical_device: &VkPhysicalDevice,
|
|
||||||
create_info: &vk::DeviceCreateInfo,
|
|
||||||
allocation_callbacks: Option<&vk::AllocationCallbacks>,
|
|
||||||
) -> VkResult<ash::Device> {
|
|
||||||
unsafe {
|
|
||||||
self.handle.create_device(physical_device.handle, &create_info, allocation_callbacks)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for VkInstance {
|
impl Drop for VkInstance {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe { self.handle.destroy_instance(None); }
|
||||||
self.handle.destroy_instance(None);
|
log::debug!(target: LOG_TARGET, "Vulkan instance destroyed ({:?})", self.handle.handle());
|
||||||
}
|
|
||||||
log::debug!(target: LOG_TARGET, "Vulkan instance destroyed");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,28 +1,26 @@
|
||||||
|
use std::sync::Arc;
|
||||||
use crate::vulkan::{VkDevice, VkInstance, VkPhysicalDevice, VkSurface, VkSwapchain};
|
use crate::vulkan::{VkDevice, VkInstance, VkPhysicalDevice, VkSurface, VkSwapchain};
|
||||||
use ash::vk;
|
use ash::vk;
|
||||||
use ash::vk::QueueFlags;
|
|
||||||
|
|
||||||
pub struct VkRenderContext {
|
pub struct VkRenderContext {
|
||||||
instance: VkInstance,
|
instance: Arc<VkInstance>,
|
||||||
surface: VkSurface,
|
surface: Arc<VkSurface>,
|
||||||
device: VkDevice,
|
device: Arc<VkDevice>,
|
||||||
|
|
||||||
swapchain: VkSwapchain,
|
swapchain: Arc<VkSwapchain>,
|
||||||
present_images: Vec<vk::Image>,
|
|
||||||
present_image_views: Vec<vk::ImageView>,
|
|
||||||
|
|
||||||
present_queue: vk::Queue,
|
// present_queue: vk::Queue,
|
||||||
|
//
|
||||||
pool: vk::CommandPool,
|
// pool: vk::CommandPool,
|
||||||
|
//
|
||||||
setup_command_buffer: vk::CommandBuffer,
|
// setup_command_buffer: vk::CommandBuffer,
|
||||||
draw_command_buffer: vk::CommandBuffer,
|
// draw_command_buffer: vk::CommandBuffer,
|
||||||
|
//
|
||||||
draw_commands_reuse_fence: vk::Fence,
|
// draw_commands_reuse_fence: vk::Fence,
|
||||||
setup_commands_reuse_fence: vk::Fence,
|
// setup_commands_reuse_fence: vk::Fence,
|
||||||
|
//
|
||||||
present_complete_semaphore: vk::Semaphore,
|
// present_complete_semaphore: vk::Semaphore,
|
||||||
rendering_complete_semaphore: vk::Semaphore,
|
// rendering_complete_semaphore: vk::Semaphore,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VkRenderContext {
|
impl VkRenderContext {
|
||||||
|
@ -30,75 +28,67 @@ impl VkRenderContext {
|
||||||
let required_extensions = window
|
let required_extensions = window
|
||||||
.required_extensions()?;
|
.required_extensions()?;
|
||||||
|
|
||||||
let instance = VkInstance::new(&required_extensions);
|
let instance = Arc::new(VkInstance::new(&required_extensions));
|
||||||
let surface = instance.create_surface(&window)?;
|
let surface = Arc::new(VkSurface::new(
|
||||||
|
&window,
|
||||||
|
instance.clone()
|
||||||
|
)?);
|
||||||
|
|
||||||
let mut physical_devices = instance.get_physical_devices();
|
let mut physical_devices = instance.get_physical_devices();
|
||||||
physical_devices.sort_by(|a, b| b.priority().cmp(&a.priority()));
|
physical_devices.sort_by(|a, b| b.priority().cmp(&a.priority()));
|
||||||
|
|
||||||
let (physical_device, queue_family_index, _) = VkPhysicalDevice::pick_physical_device_and_queue_by(
|
let (physical_device, queue_family_index, _) = VkPhysicalDevice::pick_physical_device_and_queue_by(
|
||||||
&physical_devices,
|
&physical_devices,
|
||||||
Some(QueueFlags::GRAPHICS),
|
Some(vk::QueueFlags::GRAPHICS),
|
||||||
Some(&surface),
|
Some(&surface),
|
||||||
).expect("Unable to find physical device");
|
).ok_or_else(|| anyhow::anyhow!("Unable to find physical device"))?;
|
||||||
|
|
||||||
let device = VkDevice::new_graphics_device(&instance, &physical_device, queue_family_index)
|
let device = Arc::new(VkDevice::new_graphics_device(instance.clone(), &physical_device, queue_family_index)?);
|
||||||
.expect("Unable to create device");
|
|
||||||
|
|
||||||
let swapchain = surface.create_swapchain(
|
let swapchain = Arc::new(VkSwapchain::new(
|
||||||
&window,
|
&window,
|
||||||
&instance,
|
surface.clone(),
|
||||||
&device,
|
device.clone(),
|
||||||
&physical_device,
|
&physical_device
|
||||||
).expect("Unable to create swapchain");
|
)?);
|
||||||
|
|
||||||
let present_images = swapchain.get_swapchain_images()
|
// let present_queue = device.get_device_queue(0);
|
||||||
.expect("Failed to get present images");
|
//
|
||||||
let present_image_views = present_images
|
// let pool_create_info = vk::CommandPoolCreateInfo::default()
|
||||||
.iter()
|
// .flags(vk::CommandPoolCreateFlags::RESET_COMMAND_BUFFER);
|
||||||
.map(|i| {
|
//
|
||||||
device.create_image_view(*i, swapchain.surface_format)
|
// let pool = device.create_command_pool(&pool_create_info)
|
||||||
.expect("Failed to create image view")
|
// .expect("Failed to create command pool");
|
||||||
})
|
//
|
||||||
.collect::<Vec<_>>();
|
// let command_buffer_allocate_info = vk::CommandBufferAllocateInfo::default()
|
||||||
|
// .command_buffer_count(2)
|
||||||
let present_queue = device.get_device_queue(0);
|
// .command_pool(pool)
|
||||||
|
// .level(vk::CommandBufferLevel::PRIMARY);
|
||||||
let pool_create_info = vk::CommandPoolCreateInfo::default()
|
//
|
||||||
.flags(vk::CommandPoolCreateFlags::RESET_COMMAND_BUFFER);
|
// let command_buffers = device
|
||||||
|
// .allocate_command_buffers(&command_buffer_allocate_info)
|
||||||
let pool = device.create_command_pool(&pool_create_info)
|
// .expect("Failed to create command buffers");
|
||||||
.expect("Failed to create command pool");
|
// let setup_command_buffer = command_buffers[0];
|
||||||
|
// let draw_command_buffer = command_buffers[1];
|
||||||
let command_buffer_allocate_info = vk::CommandBufferAllocateInfo::default()
|
//
|
||||||
.command_buffer_count(2)
|
// let fence_create_info =
|
||||||
.command_pool(pool)
|
// vk::FenceCreateInfo::default().flags(vk::FenceCreateFlags::SIGNALED);
|
||||||
.level(vk::CommandBufferLevel::PRIMARY);
|
//
|
||||||
|
// let draw_commands_reuse_fence = device
|
||||||
let command_buffers = device
|
// .create_fence(&fence_create_info)
|
||||||
.allocate_command_buffers(&command_buffer_allocate_info)
|
// .expect("Failed to create draw commands fence");
|
||||||
.expect("Failed to create command buffers");
|
// let setup_commands_reuse_fence = device
|
||||||
let setup_command_buffer = command_buffers[0];
|
// .create_fence(&fence_create_info)
|
||||||
let draw_command_buffer = command_buffers[1];
|
// .expect("Failed to create setup commands fence");
|
||||||
|
//
|
||||||
let fence_create_info =
|
// let semaphore_create_info = vk::SemaphoreCreateInfo::default();
|
||||||
vk::FenceCreateInfo::default().flags(vk::FenceCreateFlags::SIGNALED);
|
//
|
||||||
|
// let present_complete_semaphore = device
|
||||||
let draw_commands_reuse_fence = device
|
// .create_semaphore(&semaphore_create_info)
|
||||||
.create_fence(&fence_create_info)
|
// .expect("Failed to create present complete semaphore");
|
||||||
.expect("Failed to create draw commands fence");
|
// let rendering_complete_semaphore = device
|
||||||
let setup_commands_reuse_fence = device
|
// .create_semaphore(&semaphore_create_info)
|
||||||
.create_fence(&fence_create_info)
|
// .expect("Failed to create rendering complete semaphore");
|
||||||
.expect("Failed to create setup commands fence");
|
|
||||||
|
|
||||||
let semaphore_create_info = vk::SemaphoreCreateInfo::default();
|
|
||||||
|
|
||||||
let present_complete_semaphore = device
|
|
||||||
.create_semaphore(&semaphore_create_info)
|
|
||||||
.expect("Failed to create present complete semaphore");
|
|
||||||
let rendering_complete_semaphore = device
|
|
||||||
.create_semaphore(&semaphore_create_info)
|
|
||||||
.expect("Failed to create rendering complete semaphore");
|
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
instance,
|
instance,
|
||||||
|
@ -106,72 +96,72 @@ impl VkRenderContext {
|
||||||
device,
|
device,
|
||||||
|
|
||||||
swapchain,
|
swapchain,
|
||||||
present_images,
|
|
||||||
present_image_views,
|
|
||||||
|
|
||||||
present_queue,
|
// present_queue,
|
||||||
|
//
|
||||||
pool,
|
// pool,
|
||||||
|
//
|
||||||
setup_command_buffer,
|
// setup_command_buffer,
|
||||||
draw_command_buffer,
|
// draw_command_buffer,
|
||||||
|
//
|
||||||
present_complete_semaphore,
|
// present_complete_semaphore,
|
||||||
rendering_complete_semaphore,
|
// rendering_complete_semaphore,
|
||||||
|
//
|
||||||
draw_commands_reuse_fence,
|
// draw_commands_reuse_fence,
|
||||||
setup_commands_reuse_fence,
|
// setup_commands_reuse_fence,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn render(&mut self) -> anyhow::Result<()> {
|
pub fn render(&mut self) -> anyhow::Result<()> {
|
||||||
unsafe {
|
// unsafe {
|
||||||
self.device.handle
|
// self.device.handle
|
||||||
.wait_for_fences(&[self.draw_commands_reuse_fence], true, u64::MAX)
|
// .wait_for_fences(&[self.draw_commands_reuse_fence], true, u64::MAX)
|
||||||
.expect("Wait for fence failed.");
|
// .expect("Wait for fence failed.");
|
||||||
|
//
|
||||||
self.device.handle
|
// self.device.handle
|
||||||
.reset_fences(&[self.draw_commands_reuse_fence])
|
// .reset_fences(&[self.draw_commands_reuse_fence])
|
||||||
.expect("Reset fences failed.");
|
// .expect("Reset fences failed.");
|
||||||
|
//
|
||||||
self.device.handle
|
// self.device.handle
|
||||||
.reset_command_buffer(
|
// .reset_command_buffer(
|
||||||
self.draw_command_buffer,
|
// self.draw_command_buffer,
|
||||||
vk::CommandBufferResetFlags::RELEASE_RESOURCES,
|
// vk::CommandBufferResetFlags::RELEASE_RESOURCES,
|
||||||
)
|
// )
|
||||||
.expect("Reset command buffer failed.");
|
// .expect("Reset command buffer failed.");
|
||||||
|
//
|
||||||
let command_buffer_begin_info = vk::CommandBufferBeginInfo::default()
|
// let command_buffer_begin_info = vk::CommandBufferBeginInfo::default()
|
||||||
.flags(vk::CommandBufferUsageFlags::ONE_TIME_SUBMIT);
|
// .flags(vk::CommandBufferUsageFlags::ONE_TIME_SUBMIT);
|
||||||
|
//
|
||||||
self.device.handle
|
// self.device.handle
|
||||||
.begin_command_buffer(self.draw_command_buffer, &command_buffer_begin_info)
|
// .begin_command_buffer(self.draw_command_buffer, &command_buffer_begin_info)
|
||||||
.expect("Begin commandbuffer");
|
// .expect("Begin commandbuffer");
|
||||||
|
//
|
||||||
self.device.handle
|
// self.device.handle
|
||||||
.end_command_buffer(self.draw_command_buffer)
|
// .end_command_buffer(self.draw_command_buffer)
|
||||||
.expect("End commandbuffer");
|
// .expect("End commandbuffer");
|
||||||
|
//
|
||||||
let command_buffers = vec![self.draw_command_buffer];
|
// let command_buffers = vec![self.draw_command_buffer];
|
||||||
let semaphores = vec![self.rendering_complete_semaphore];
|
// let semaphores = vec![self.rendering_complete_semaphore];
|
||||||
let wait_mask = vec![vk::PipelineStageFlags::default()];
|
// let wait_mask = vec![vk::PipelineStageFlags::default()];
|
||||||
|
//
|
||||||
let submit_info = vk::SubmitInfo::default()
|
// let submit_info = vk::SubmitInfo::default()
|
||||||
.wait_semaphores(&semaphores)
|
// .wait_semaphores(&semaphores)
|
||||||
.wait_dst_stage_mask(&wait_mask)
|
// .wait_dst_stage_mask(&wait_mask)
|
||||||
.command_buffers(&command_buffers)
|
// .command_buffers(&command_buffers)
|
||||||
.signal_semaphores(&semaphores);
|
// .signal_semaphores(&semaphores);
|
||||||
|
//
|
||||||
self.device.handle
|
// self.device.handle
|
||||||
.queue_submit(self.present_queue, &[submit_info], self.draw_commands_reuse_fence)
|
// .queue_submit(self.present_queue, &[submit_info], self.draw_commands_reuse_fence)
|
||||||
.expect("queue submit failed.");
|
// .expect("queue submit failed.");
|
||||||
}
|
// }
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn update_resolution(&mut self, width: u32, height: u32) -> anyhow::Result<()> {
|
pub fn update_resolution(&mut self, width: u32, height: u32) -> anyhow::Result<()> {
|
||||||
self.swapchain.update_resolution(&self.surface, width, height)?;
|
if let Some(swapchain) = Arc::get_mut(&mut self.swapchain) {
|
||||||
|
swapchain.update_resolution(width, height)?;
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,27 +1,43 @@
|
||||||
use crate::display::Window;
|
use std::sync::Arc;
|
||||||
use crate::vulkan::{VkDevice, VkInstance, VkPhysicalDevice, VkSwapchain, LOG_TARGET};
|
use crate::vulkan::{VkInstance, VkPhysicalDevice, LOG_TARGET};
|
||||||
use ash::prelude::VkResult;
|
use ash::prelude::VkResult;
|
||||||
use ash::vk;
|
use ash::vk;
|
||||||
|
use winit::raw_window_handle::{HasDisplayHandle, HasWindowHandle};
|
||||||
|
|
||||||
pub struct VkSurface {
|
pub struct VkSurface {
|
||||||
surface_loader: ash::khr::surface::Instance,
|
instance: Arc<VkInstance>,
|
||||||
pub(super) surface: vk::SurfaceKHR,
|
pub(super) surface: vk::SurfaceKHR,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VkSurface {
|
impl VkSurface {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
surface_loader: ash::khr::surface::Instance,
|
window: &crate::display::Window,
|
||||||
surface: vk::SurfaceKHR,
|
instance: Arc<VkInstance>,
|
||||||
) -> Self {
|
) -> anyhow::Result<Self> {
|
||||||
Self {
|
let window_handle = window.handle()
|
||||||
surface_loader,
|
.ok_or_else(|| anyhow::anyhow!("Window handle is not available."))?;
|
||||||
|
|
||||||
|
let surface = unsafe {
|
||||||
|
ash_window::create_surface(
|
||||||
|
&instance.entry,
|
||||||
|
&instance.handle,
|
||||||
|
window_handle.display_handle()?.as_raw(),
|
||||||
|
window_handle.window_handle()?.as_raw(),
|
||||||
|
None,
|
||||||
|
)?
|
||||||
|
};
|
||||||
|
|
||||||
|
log::debug!(target: LOG_TARGET, "Surface created ({:?})", surface);
|
||||||
|
|
||||||
|
Ok(Self {
|
||||||
|
instance,
|
||||||
surface,
|
surface,
|
||||||
}
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn physical_device_queue_supported(&self, physical_device: &VkPhysicalDevice, queue_index: u32) -> VkResult<bool> {
|
pub fn physical_device_queue_supported(&self, physical_device: &VkPhysicalDevice, queue_index: u32) -> VkResult<bool> {
|
||||||
unsafe {
|
unsafe {
|
||||||
self.surface_loader.get_physical_device_surface_support(
|
self.instance.surface_loader.get_physical_device_surface_support(
|
||||||
physical_device.handle,
|
physical_device.handle,
|
||||||
queue_index,
|
queue_index,
|
||||||
self.surface,
|
self.surface,
|
||||||
|
@ -29,115 +45,37 @@ impl VkSurface {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_physical_device_surface_formats(&self, physical_device: &VkPhysicalDevice) -> VkResult<Vec<vk::SurfaceFormatKHR>> {
|
pub fn get_physical_device_surface_infos(&self, physical_device: &VkPhysicalDevice) -> VkResult<(
|
||||||
|
Vec<vk::SurfaceFormatKHR>,
|
||||||
|
vk::SurfaceCapabilitiesKHR,
|
||||||
|
Vec<vk::PresentModeKHR>
|
||||||
|
)> {
|
||||||
unsafe {
|
unsafe {
|
||||||
self.surface_loader.get_physical_device_surface_formats(
|
let formats = self.instance.surface_loader.get_physical_device_surface_formats(
|
||||||
physical_device.handle,
|
physical_device.handle,
|
||||||
self.surface,
|
self.surface,
|
||||||
)
|
)?;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_physical_device_surface_capabilities(&self, physical_device: &VkPhysicalDevice) -> VkResult<vk::SurfaceCapabilitiesKHR> {
|
let capabilities = self.instance.surface_loader.get_physical_device_surface_capabilities(
|
||||||
unsafe {
|
|
||||||
self.surface_loader.get_physical_device_surface_capabilities(
|
|
||||||
physical_device.handle,
|
physical_device.handle,
|
||||||
self.surface,
|
self.surface,
|
||||||
)
|
)?;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_physical_device_surface_present_modes(&self, physical_device: &VkPhysicalDevice) -> VkResult<Vec<vk::PresentModeKHR>> {
|
let present_modes = self.instance.surface_loader.get_physical_device_surface_present_modes(
|
||||||
unsafe {
|
|
||||||
self.surface_loader.get_physical_device_surface_present_modes(
|
|
||||||
physical_device.handle,
|
physical_device.handle,
|
||||||
self.surface,
|
self.surface,
|
||||||
)
|
)?;
|
||||||
|
|
||||||
|
Ok((formats, capabilities, present_modes))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn create_swapchain(
|
|
||||||
&self,
|
|
||||||
window: &Window,
|
|
||||||
instance: &VkInstance,
|
|
||||||
device: &VkDevice,
|
|
||||||
physical_device: &VkPhysicalDevice,
|
|
||||||
) -> anyhow::Result<VkSwapchain> {
|
|
||||||
log::debug!(target: LOG_TARGET, "Creating swapchain");
|
|
||||||
|
|
||||||
let surface_formats = self.get_physical_device_surface_formats(physical_device)?;
|
|
||||||
log::debug!(target: LOG_TARGET, "Supported surface formats by physical device: {surface_formats:#?}");
|
|
||||||
|
|
||||||
let surface_format = surface_formats
|
|
||||||
.first()
|
|
||||||
.and_then(|f| Some(*f))
|
|
||||||
.ok_or_else(|| anyhow::anyhow!("No available surface formats"))?;
|
|
||||||
log::debug!(target: LOG_TARGET, "Selected surface format: {surface_format:?}");
|
|
||||||
|
|
||||||
let surface_capabilities = self.get_physical_device_surface_capabilities(physical_device)?;
|
|
||||||
log::debug!(target: LOG_TARGET, "Surface capabilities: {surface_capabilities:#?}");
|
|
||||||
|
|
||||||
let mut desired_image_count = surface_capabilities.min_image_count + 1;
|
|
||||||
if surface_capabilities.max_image_count > 0
|
|
||||||
&& desired_image_count > surface_capabilities.max_image_count
|
|
||||||
{
|
|
||||||
desired_image_count = surface_capabilities.max_image_count;
|
|
||||||
}
|
|
||||||
log::debug!(target: LOG_TARGET, "Selected surface image count: {desired_image_count}");
|
|
||||||
|
|
||||||
let window_size = window.size()
|
|
||||||
.ok_or_else(|| anyhow::anyhow!("Window size is not valid"))?
|
|
||||||
.to_physical::<u32>(1.0);
|
|
||||||
log::debug!(target: LOG_TARGET, "Window size: {window_size:?}");
|
|
||||||
|
|
||||||
let surface_resolution = match surface_capabilities.current_extent.width {
|
|
||||||
u32::MAX => vk::Extent2D {
|
|
||||||
width: window_size.width,
|
|
||||||
height: window_size.height,
|
|
||||||
},
|
|
||||||
_ => surface_capabilities.current_extent,
|
|
||||||
};
|
|
||||||
log::debug!(target: LOG_TARGET, "Surface resolution: {surface_resolution:?}");
|
|
||||||
|
|
||||||
let pre_transform = if surface_capabilities
|
|
||||||
.supported_transforms
|
|
||||||
.contains(vk::SurfaceTransformFlagsKHR::IDENTITY)
|
|
||||||
{
|
|
||||||
vk::SurfaceTransformFlagsKHR::IDENTITY
|
|
||||||
} else {
|
|
||||||
surface_capabilities.current_transform
|
|
||||||
};
|
|
||||||
|
|
||||||
let present_modes = self
|
|
||||||
.get_physical_device_surface_present_modes(physical_device)?;
|
|
||||||
|
|
||||||
let present_mode = present_modes
|
|
||||||
.iter()
|
|
||||||
.cloned()
|
|
||||||
.find(|&mode| mode == vk::PresentModeKHR::MAILBOX)
|
|
||||||
.unwrap_or(vk::PresentModeKHR::FIFO);
|
|
||||||
let swapchain_loader = ash::khr::swapchain::Device::new(&instance.handle, &device.handle);
|
|
||||||
|
|
||||||
let mut swapchain = VkSwapchain::new(
|
|
||||||
swapchain_loader,
|
|
||||||
desired_image_count,
|
|
||||||
surface_format,
|
|
||||||
surface_resolution,
|
|
||||||
present_mode,
|
|
||||||
pre_transform,
|
|
||||||
);
|
|
||||||
|
|
||||||
swapchain.create_swapchain(&self)?;
|
|
||||||
|
|
||||||
Ok(swapchain)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for VkSurface {
|
impl Drop for VkSurface {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe {
|
||||||
self.surface_loader.destroy_surface(self.surface, None);
|
self.instance.surface_loader.destroy_surface(self.surface, None);
|
||||||
}
|
}
|
||||||
log::debug!(target: LOG_TARGET, "Surface destroyed");
|
log::debug!(target: LOG_TARGET, "Surface destroyed ({:?})", self.surface);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,9 +1,12 @@
|
||||||
use crate::vulkan::{VkSurface, LOG_TARGET};
|
use std::sync::Arc;
|
||||||
|
use crate::vulkan::{VkDevice, VkInstance, VkPhysicalDevice, VkSurface, LOG_TARGET};
|
||||||
use ash::prelude::VkResult;
|
use ash::prelude::VkResult;
|
||||||
use ash::vk;
|
use ash::vk;
|
||||||
|
use crate::display::Window;
|
||||||
|
|
||||||
pub struct VkSwapchain {
|
pub struct VkSwapchain {
|
||||||
pub(super) swapchain_loader: ash::khr::swapchain::Device,
|
surface: Arc<VkSurface>,
|
||||||
|
device: Arc<VkDevice>,
|
||||||
swapchain: Option<vk::SwapchainKHR>,
|
swapchain: Option<vk::SwapchainKHR>,
|
||||||
|
|
||||||
pub(super) desired_image_count: u32,
|
pub(super) desired_image_count: u32,
|
||||||
|
@ -11,70 +14,136 @@ pub struct VkSwapchain {
|
||||||
pub(super) surface_resolution: vk::Extent2D,
|
pub(super) surface_resolution: vk::Extent2D,
|
||||||
pub(super) present_mode: vk::PresentModeKHR,
|
pub(super) present_mode: vk::PresentModeKHR,
|
||||||
pub(super) pre_transform: vk::SurfaceTransformFlagsKHR,
|
pub(super) pre_transform: vk::SurfaceTransformFlagsKHR,
|
||||||
|
|
||||||
|
pub(super) present_images: Option<Vec<vk::Image>>,
|
||||||
|
pub(super) present_image_views: Option<Vec<vk::ImageView>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VkSwapchain {
|
impl VkSwapchain {
|
||||||
pub(super) fn new(
|
pub(super) fn new<'a, 'b>(
|
||||||
swapchain_loader: ash::khr::swapchain::Device,
|
window: &Window,
|
||||||
desired_image_count: u32,
|
surface: Arc<VkSurface>,
|
||||||
surface_format: vk::SurfaceFormatKHR,
|
device: Arc<VkDevice>,
|
||||||
surface_resolution: vk::Extent2D,
|
physical_device: &VkPhysicalDevice,
|
||||||
present_mode: vk::PresentModeKHR,
|
) -> anyhow::Result<Self> {
|
||||||
pre_transform: vk::SurfaceTransformFlagsKHR,
|
log::debug!(target: LOG_TARGET, "Creating swapchain");
|
||||||
) -> Self {
|
|
||||||
Self {
|
let (
|
||||||
swapchain_loader,
|
surface_formats,
|
||||||
|
surface_capabilities,
|
||||||
|
present_modes
|
||||||
|
) = surface.get_physical_device_surface_infos(physical_device)?;
|
||||||
|
log::debug!(target: LOG_TARGET, "Supported surface formats by physical device: {surface_formats:#?}");
|
||||||
|
log::debug!(target: LOG_TARGET, "Surface capabilities: {surface_capabilities:#?}");
|
||||||
|
|
||||||
|
let surface_format = surface_formats
|
||||||
|
.first()
|
||||||
|
.and_then(|f| Some(*f))
|
||||||
|
.ok_or_else(|| anyhow::anyhow!("No available surface formats"))?;
|
||||||
|
log::debug!(target: LOG_TARGET, "Selected surface format: {surface_format:?}");
|
||||||
|
|
||||||
|
let mut desired_image_count = surface_capabilities.min_image_count + 1;
|
||||||
|
if surface_capabilities.max_image_count > 0
|
||||||
|
&& desired_image_count > surface_capabilities.max_image_count
|
||||||
|
{
|
||||||
|
desired_image_count = surface_capabilities.max_image_count;
|
||||||
|
}
|
||||||
|
log::debug!(target: LOG_TARGET, "Selected surface image count: {desired_image_count}");
|
||||||
|
|
||||||
|
let window_size = window.size()
|
||||||
|
.ok_or_else(|| anyhow::anyhow!("Window size is not valid"))?
|
||||||
|
.to_physical::<u32>(1.0);
|
||||||
|
log::debug!(target: LOG_TARGET, "Window size: {window_size:?}");
|
||||||
|
|
||||||
|
let surface_resolution = match surface_capabilities.current_extent.width {
|
||||||
|
u32::MAX => vk::Extent2D {
|
||||||
|
width: window_size.width,
|
||||||
|
height: window_size.height,
|
||||||
|
},
|
||||||
|
_ => surface_capabilities.current_extent,
|
||||||
|
};
|
||||||
|
log::debug!(target: LOG_TARGET, "Surface resolution: {surface_resolution:?}");
|
||||||
|
|
||||||
|
let pre_transform = if surface_capabilities
|
||||||
|
.supported_transforms
|
||||||
|
.contains(vk::SurfaceTransformFlagsKHR::IDENTITY)
|
||||||
|
{
|
||||||
|
vk::SurfaceTransformFlagsKHR::IDENTITY
|
||||||
|
} else {
|
||||||
|
surface_capabilities.current_transform
|
||||||
|
};
|
||||||
|
|
||||||
|
let present_mode = present_modes
|
||||||
|
.iter()
|
||||||
|
.cloned()
|
||||||
|
.find(|&mode| mode == vk::PresentModeKHR::MAILBOX)
|
||||||
|
.unwrap_or(vk::PresentModeKHR::FIFO);
|
||||||
|
|
||||||
|
let mut swapchain = Self {
|
||||||
|
surface,
|
||||||
|
device,
|
||||||
desired_image_count,
|
desired_image_count,
|
||||||
surface_format,
|
surface_format,
|
||||||
surface_resolution,
|
surface_resolution,
|
||||||
present_mode,
|
present_mode,
|
||||||
pre_transform,
|
pre_transform,
|
||||||
swapchain: None,
|
swapchain: None,
|
||||||
}
|
present_images: None,
|
||||||
|
present_image_views: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
swapchain.create_swapchain()?;
|
||||||
|
|
||||||
|
Ok(swapchain)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn create_swapchain(&mut self, surface: &VkSurface) -> VkResult<()> {
|
pub(super) fn create_swapchain(&mut self) -> VkResult<()> {
|
||||||
let mut swapchain_create_info = self.create_swapchain_info(surface);
|
let mut swapchain_create_info = self.create_swapchain_info(&self.surface);
|
||||||
|
|
||||||
if let Some(old_swapchain) = self.swapchain {
|
if let Some(old_swapchain) = self.swapchain {
|
||||||
swapchain_create_info.old_swapchain = old_swapchain;
|
swapchain_create_info.old_swapchain = old_swapchain;
|
||||||
}
|
}
|
||||||
|
|
||||||
let swapchain = unsafe {
|
let swapchain = unsafe {
|
||||||
self.swapchain_loader.create_swapchain(&swapchain_create_info, None)?
|
self.device.swapchain_loader.create_swapchain(&swapchain_create_info, None)?
|
||||||
};
|
};
|
||||||
|
|
||||||
match self.swapchain {
|
let present_images = unsafe { self.device.swapchain_loader.get_swapchain_images(swapchain)? };
|
||||||
Some(_) => log::debug!(target: LOG_TARGET, "Swapchain created : {swapchain_create_info:#?}"),
|
let present_images_view = present_images
|
||||||
None => log::debug!(target: LOG_TARGET, "Swapchain updated : {swapchain_create_info:#?}")
|
.iter()
|
||||||
|
.map(|i| {
|
||||||
|
self.device.create_image_view(*i, self.surface_format)
|
||||||
|
.expect("Failed to create image view")
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
if log::log_enabled!(target: LOG_TARGET, log::Level::Debug) {
|
||||||
|
let label = match self.swapchain {
|
||||||
|
None => "Swapchain created",
|
||||||
|
Some(_) => "Swapchain updated"
|
||||||
|
};
|
||||||
|
log::debug!(target: LOG_TARGET, "{label} ({swapchain:?}) : {swapchain_create_info:#?}");
|
||||||
}
|
}
|
||||||
|
|
||||||
self.swapchain = Some(swapchain);
|
self.swapchain = Some(swapchain);
|
||||||
|
self.present_image_views = Some(present_images_view);
|
||||||
|
self.present_images = Some(present_images);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn update_resolution(&mut self, surface: &VkSurface, width: u32, height: u32) -> VkResult<()> {
|
pub(super) fn update_resolution(&mut self, width: u32, height: u32) -> VkResult<()> {
|
||||||
log::debug!(target: LOG_TARGET, "New resolution requested for swapchain {width}x{height}");
|
log::debug!(target: LOG_TARGET, "New resolution requested for swapchain {width}x{height}");
|
||||||
self.surface_resolution = vk::Extent2D {
|
self.surface_resolution = vk::Extent2D {
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
};
|
};
|
||||||
|
|
||||||
self.create_swapchain(surface)?;
|
self.create_swapchain()?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn get_swapchain_images(&self) -> anyhow::Result<Vec<vk::Image>> {
|
|
||||||
let swapchain = self.swapchain
|
|
||||||
.ok_or_else(|| anyhow::anyhow!("Can't get swapchain images : Swapchain is not set"))?;
|
|
||||||
|
|
||||||
let images = unsafe { self.swapchain_loader.get_swapchain_images(swapchain)? };
|
|
||||||
|
|
||||||
Ok(images)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create_swapchain_info(&self, surface: &VkSurface) -> vk::SwapchainCreateInfoKHR {
|
fn create_swapchain_info(&self, surface: &VkSurface) -> vk::SwapchainCreateInfoKHR {
|
||||||
vk::SwapchainCreateInfoKHR::default()
|
vk::SwapchainCreateInfoKHR::default()
|
||||||
.surface(surface.surface)
|
.surface(surface.surface)
|
||||||
|
@ -91,18 +160,17 @@ impl VkSwapchain {
|
||||||
.image_array_layers(1)
|
.image_array_layers(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn drop_current_swapchain(&mut self) {
|
fn drop_swapchain(&mut self) {
|
||||||
if let Some(swapchain) = self.swapchain {
|
if let Some(swapchain) = self.swapchain {
|
||||||
unsafe {
|
unsafe { self.device.swapchain_loader.destroy_swapchain(swapchain, None); }
|
||||||
self.swapchain_loader.destroy_swapchain(swapchain, None);
|
self.swapchain = None;
|
||||||
self.swapchain = None;
|
log::debug!(target: LOG_TARGET, "Swapchain destroyed ({swapchain:?})");
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for VkSwapchain {
|
impl Drop for VkSwapchain {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
self.drop_current_swapchain()
|
self.drop_swapchain()
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Add table
Reference in a new issue