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

Arc it's used because later, i can use on multi-threaded programs
This commit is contained in:
Florian RICHER 2024-11-15 22:40:36 +01:00
parent 1cb9309a56
commit 174e12591c
Signed by: florian.richer
GPG key ID: C73D37CBED7BFC77
5 changed files with 290 additions and 320 deletions

View file

@ -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());
} }
} }
} }

View file

@ -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");
} }
} }

View file

@ -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 = Arc::new(VkSwapchain::new(
let swapchain = surface.create_swapchain(
&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(())
} }

View file

@ -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);
} }
} }

View file

@ -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()
} }
} }