use std::ffi::CStr; use crate::hello_triangle::surface::VulkanSurface; use crate::hello_triangle::vulkan_base::VulkanBase; use crate::util::find_vulkan_memory_type; use crate::vulkan_primitives::VulkanImageMemory; use ash::prelude::VkResult; use ash::vk; use ash::vk::{Extent3D, Handle}; pub struct VulkanSwapchain { pub swapchain: vk::SwapchainKHR, pub loader: ash::extensions::khr::Swapchain, pub format: vk::SurfaceFormatKHR, pub extent: vk::Extent2D, mode: vk::PresentModeKHR, pub swapchain_images: Vec, pub swapchain_image_views: Vec, pub render_images: Vec<(vk::Image, VulkanImageMemory)>, pub render_image_views: Vec, device: ash::Device, } impl VulkanSwapchain { pub fn new( base: &VulkanBase, surface: &VulkanSurface, width: u32, height: u32, ) -> VkResult { let format = surface.choose_format(base)?; let mode = surface.choose_present_mode(base)?; let extent = surface.choose_swapchain_extent(base, width, height)?; let capabilities = surface.get_capabilities(base)?; let image_count = capabilities.min_image_count + 1; let image_count = if capabilities.max_image_count > 0 { image_count.min(capabilities.max_image_count) } else { image_count }; if base.graphics_queue != surface.present_queue { panic!("exclusive mode only") } let create_info = vk::SwapchainCreateInfoKHR::builder() .surface(surface.surface) .present_mode(mode) .min_image_count(image_count) .image_color_space(format.color_space) .image_format(format.format) .image_extent(extent) .image_sharing_mode(vk::SharingMode::EXCLUSIVE) .pre_transform(capabilities.current_transform) .composite_alpha(vk::CompositeAlphaFlagsKHR::OPAQUE) .clipped(true) .image_array_layers(1) // todo: switch to IMAGE_USAGE_TRANSFER_DST .image_usage(vk::ImageUsageFlags::COLOR_ATTACHMENT) .build(); let loader = ash::extensions::khr::Swapchain::new(&base.instance, &base.device); let swapchain = unsafe { loader.create_swapchain(&create_info, None)? }; let swapchain_images = unsafe { loader.get_swapchain_images(swapchain)? }; let mut render_images = vec![]; // create render imaghes for _ in 0..swapchain_images.len() { let create_info = vk::ImageCreateInfo::builder() .extent(Extent3D { width, height, depth: 1, }) .format(format.format) .image_type(vk::ImageType::TYPE_2D) .sharing_mode(vk::SharingMode::EXCLUSIVE) .samples(vk::SampleCountFlags::TYPE_1) .tiling(vk::ImageTiling::OPTIMAL) .array_layers(1) .mip_levels(1) .usage(vk::ImageUsageFlags::SAMPLED | vk::ImageUsageFlags::COLOR_ATTACHMENT | vk::ImageUsageFlags::TRANSFER_SRC) .initial_layout(vk::ImageLayout::UNDEFINED); unsafe { let image = base.device.create_image(&create_info, None)?; let mem_reqs = unsafe { base.device.get_image_memory_requirements(image.clone()) }; base.debug.loader.set_debug_utils_object_name(base.device.handle(), &vk::DebugUtilsObjectNameInfoEXT::builder() .object_handle(image.as_raw()) .object_name(CStr::from_bytes_with_nul_unchecked(b"RenderImage\0")) .object_type(vk::ObjectType::IMAGE) .build()) .expect("could not set object name"); let alloc_info = vk::MemoryAllocateInfo::builder() .allocation_size(mem_reqs.size) .memory_type_index(find_vulkan_memory_type( &base.mem_props, mem_reqs.memory_type_bits, vk::MemoryPropertyFlags::DEVICE_LOCAL, )) .build(); // todo: optimize by reusing existing memory. let memory = VulkanImageMemory::new(&base.device, &alloc_info).unwrap(); memory.bind(&image).unwrap(); render_images.push((image, memory)) } } let swapchain_image_views: VkResult> = swapchain_images .iter() .map(|image| { let create_info = vk::ImageViewCreateInfo::builder() .view_type(vk::ImageViewType::TYPE_2D) .format(format.format) .components(vk::ComponentMapping { r: vk::ComponentSwizzle::IDENTITY, g: vk::ComponentSwizzle::IDENTITY, b: vk::ComponentSwizzle::IDENTITY, a: vk::ComponentSwizzle::IDENTITY, }) .subresource_range(vk::ImageSubresourceRange { aspect_mask: vk::ImageAspectFlags::COLOR, base_mip_level: 0, level_count: 1, base_array_layer: 0, layer_count: 1, }) .image(*image) .build(); let view = unsafe { base.device.create_image_view(&create_info, None)? }; unsafe { base.debug.loader.set_debug_utils_object_name(base.device.handle(), &vk::DebugUtilsObjectNameInfoEXT::builder() .object_handle(image.as_raw()) .object_name(CStr::from_bytes_with_nul_unchecked(b"SwapchainImage\0")) .object_type(vk::ObjectType::IMAGE) .build()) .expect("could not set object name"); base.debug.loader.set_debug_utils_object_name(base.device.handle(), &vk::DebugUtilsObjectNameInfoEXT::builder() .object_handle(view.as_raw()) .object_name(CStr::from_bytes_with_nul_unchecked(b"SwapchainImageView\0")) .object_type(vk::ObjectType::IMAGE_VIEW) .build()) .expect("could not set object name"); } Ok(view) }) .collect(); let render_image_views: VkResult> = render_images .iter() .map(|(image, _)| { let create_info = vk::ImageViewCreateInfo::builder() .view_type(vk::ImageViewType::TYPE_2D) .format(format.format) .components(vk::ComponentMapping { r: vk::ComponentSwizzle::IDENTITY, g: vk::ComponentSwizzle::IDENTITY, b: vk::ComponentSwizzle::IDENTITY, a: vk::ComponentSwizzle::IDENTITY, }) .subresource_range(vk::ImageSubresourceRange { aspect_mask: vk::ImageAspectFlags::COLOR, base_mip_level: 0, level_count: 1, base_array_layer: 0, layer_count: 1, }) .image(*image) .build(); let view = unsafe { base.device.create_image_view(&create_info, None)? }; unsafe { base.debug.loader.set_debug_utils_object_name(base.device.handle(), &vk::DebugUtilsObjectNameInfoEXT::builder() .object_handle(view.as_raw()) .object_name(CStr::from_bytes_with_nul_unchecked(b"RenderImageView\0")) .object_type(vk::ObjectType::IMAGE_VIEW) .build()) .expect("could not set object name"); } Ok(view) }) .collect(); Ok(VulkanSwapchain { swapchain, loader, format, extent, mode, swapchain_images, render_images, swapchain_image_views: swapchain_image_views?, render_image_views: render_image_views?, device: base.device.clone(), }) } } impl Drop for VulkanSwapchain { fn drop(&mut self) { unsafe { for view in &self.render_image_views { self.device.destroy_image_view(*view, None) } for (view, _memory) in &self.render_images { self.device.destroy_image(*view, None); } self.loader.destroy_swapchain(self.swapchain, None) } } }