With a surface provided, we can now move to one of the more daunting components of Vulkan: the swap chain. After that, we need to take care of image views so that the swap chain has something to actually swap in. This will probably be the most challenging bit so far. Probably. All of the code can be found at this repo.
We first start with yet another clunky means of figuring out whether or not we support a feature:
const DEVICE_EXTENSIONS: &[*const i8] = &[unsafe {
    CStr::from_bytes_with_nul_unchecked("VK_KHR_swapchain\0".as_bytes()).as_ptr()
}];
We need to handle this at the device level, which means creating yet another function:
    fn is_device_suitable(
        entry: &ash::Entry,
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
        surface: &vk::SurfaceKHR,
    ) -> bool {
        let (graphics_queue_index, present_queue_index) =
            VulkanDetails::find_queue_familes(entry, instance, device, surface);
        graphics_queue_index.is_some()
            && present_queue_index.is_some()
            && VulkanDetails::check_device_extension_support(instance, device)
    }
    fn check_device_extension_support(
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
    ) -> bool {
        let extension_properties = unsafe {
            instance
                .enumerate_device_extension_properties(*device)
                .unwrap()
        };
        for device_extension in DEVICE_EXTENSIONS {
            if extension_properties
                .iter()
                .find(|extension_property| unsafe {
                    &CStr::from_ptr(extension_property.extension_name.as_ptr())
                        .to_str()
                        .unwrap()
                        == &CStr::from_ptr(*device_extension).to_str().unwrap()
                })
                .is_none()
            {
                return false;
            }
        }
        true
    }
Of course, this just tells us that the extension is supported. We still need to actually enable it. That, at least, is simple: we just need to update our logical device creation.
        let device_create_info = vk::DeviceCreateInfo {
            s_type: vk::StructureType::DEVICE_CREATE_INFO,
            queue_create_info_count: device_queue_create_infos.len() as u32,
            p_queue_create_infos: device_queue_create_infos.as_ptr(),
            p_enabled_features: &device_features,
            enabled_layer_count: VALIDATION_LAYERS.len() as u32,
            pp_enabled_layer_names: VALIDATION_LAYERS.as_ptr(),
            enabled_extension_count: DEVICE_EXTENSIONS.len() as u32,
            pp_enabled_extension_names: DEVICE_EXTENSIONS.as_ptr(),
            ..Default::default()
        };
That’s all nice and straightforward, but there’s still a ton of properties beyond this we need to verify for our swap chain. We’ll keep the details tightly bound in a struct:
struct swap_chain_support_details {
    capabilities: vk::SurfaceCapabilitiesKHR,
    formats: Vec<vk::SurfaceFormatKHR>,
    present_modes: Vec<vk::PresentModeKHR>,
}
Since we’re trying to at least be slightly idomatic here, we can just handle our populating this struct with new:
impl swap_chain_support_details {
    pub fn new(entry: &ash::Entry, instance: &ash::Instance, device: &vk::PhysicalDevice, surface: &vk::SurfaceKHR) -> Self {
        let surface_interface = Surface::new(entry, instance);
        Self {
            capabilities: unsafe { surface_interface.get_physical_device_surface_capabilities(*device, *surface).unwrap() },
            formats: unsafe { surface_interface.get_physical_device_surface_formats(*device, *surface).unwrap() },
            present_modes: unsafe { surface_interface.get_physical_device_surface_present_modes(*device, *surface).unwrap() },
        }
    }
}
And with that, our is_device_suitable grows yet again:
    fn is_device_suitable(
        entry: &ash::Entry,
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
        surface: &vk::SurfaceKHR,
    ) -> bool {
        let (graphics_queue_index, present_queue_index) =
            VulkanDetails::find_queue_familes(entry, instance, device, surface);
        let swap_chain_support = swap_chain_support_details::new(entry, instance, device, surface);
        graphics_queue_index.is_some()
            && present_queue_index.is_some()
            && VulkanDetails::check_device_extension_support(instance, device)
            && !swap_chain_support.formats.is_empty()
            && !swap_chain_support.present_modes.is_empty()
    }
That gives us the formats and the modes, but we still need to figure out the right settings for the swap chain such as the surface format, presentation mode, and swap extent. This means, of course, a new member to our vulkan_details:
pub struct VulkanDetails {
    entry: ash::Entry,
    instance: ash::Instance,
    debug_messenger: vk::DebugUtilsMessengerEXT,
    surface: vk::SurfaceKHR,
    physical_device: vk::PhysicalDevice,
    device: ash::Device,
    graphics_queue: vk::Queue,
    present_queue: vk::Queue,
    swap_chain: vk::SwapchainKHR,
}
Now, we need functions to evaluate the best of what we can get out of our swap chain support details. Here we strictly match the tutorial:
    fn choose_swap_surface_format(formats: Vec<vk::SurfaceFormatKHR>) -> vk::SurfaceFormatKHR {
        for available_format in &formats {
            if available_format.format == vk::Format::B8G8R8A8_SRGB && available_format.color_space == vk::ColorSpaceKHR::SRGB_NONLINEAR {
                return *available_format;
            }
        }
        formats[0]
    }
    fn choose_swap_present_mode(present_modes: Vec<vk::PresentModeKHR>) -> vk::PresentModeKHR {
        for available_present_mode in present_modes {
            if available_present_mode == vk::PresentModeKHR::MAILBOX {
                return available_present_mode;
            }
        }
        vk::PresentModeKHR::FIFO
    }
    fn choose_swap_extent(window: &winit::window::Window, capabilities: vk::SurfaceCapabilitiesKHR) -> vk::Extent2D {
        if capabilities.current_extent.width != u32::MAX {
            capabilities.current_extent
        }
        else {
            let window_size = window.inner_size();
            vk::Extent2D {
                width: window_size.width.clamp(capabilities.min_image_extent.width, capabilities.max_image_extent.width),
                height: window_size.height.clamp(capabilities.min_image_extent.height, capabilities.max_image_extent.height),
            }
        }
    }
Similarly for building our swapchain:
    fn create_swap_chain(
        window: &winit::window::Window,
        entry: &ash::Entry,
        instance: &ash::Instance,
        physical_device: &vk::PhysicalDevice,
        device: &ash::Device,
        surface: &vk::SurfaceKHR,
    ) -> vk::SwapchainKHR {
        let swap_chain_support =
            SwapchainSupportDetails::new(entry, instance, physical_device, surface);
        let format = VulkanDetails::choose_swap_surface_format(swap_chain_support.formats);
        let present_mode =
            VulkanDetails::choose_swap_present_mode(swap_chain_support.present_modes);
        let image_count = {
            if swap_chain_support.capabilities.max_image_count > 0
                && swap_chain_support.capabilities.min_image_count
                    > swap_chain_support.capabilities.max_image_count
            {
                swap_chain_support.capabilities.max_image_count
            } else {
                swap_chain_support.capabilities.min_image_count + 1
            }
        };
        let extent = VulkanDetails::choose_swap_extent(window, &swap_chain_support.capabilities);
        let (graphics_queue_index, present_mode_index) =
            VulkanDetails::find_queue_familes(entry, instance, physical_device, surface);
        let queue_index_equivalent = graphics_queue_index.unwrap() == present_mode_index.unwrap();
        let queue_family_indices = vec![graphics_queue_index.unwrap(), present_mode_index.unwrap()];
        let create_info = vk::SwapchainCreateInfoKHR {
            s_type: vk::StructureType::SWAPCHAIN_CREATE_INFO_KHR,
            surface: *surface,
            min_image_count: image_count,
            image_format: format.format,
            image_color_space: format.color_space,
            image_extent: extent,
            image_array_layers: 1,
            image_usage: vk::ImageUsageFlags::COLOR_ATTACHMENT,
            image_sharing_mode: if queue_index_equivalent {
                vk::SharingMode::EXCLUSIVE
            } else {
                vk::SharingMode::CONCURRENT
            },
            queue_family_index_count: if queue_index_equivalent { 0 } else { 2 },
            p_queue_family_indices: if queue_index_equivalent {
                ptr::null()
            } else {
                queue_family_indices.as_ptr() as *const u32
            },
            pre_transform: swap_chain_support.capabilities.current_transform,
            composite_alpha: vk::CompositeAlphaFlagsKHR::OPAQUE,
            present_mode: present_mode,
            clipped: vk::TRUE,
            old_swapchain: vk::SwapchainKHR::null(),
            ..Default::default()
        };
        unsafe {
            Swapchain::new(instance, device)
                .create_swapchain(&create_info, None)
                .unwrap()
        }
    }
    fn cleanup(&mut self) {
        unsafe {
            Swapchain::new(&self.instance, &self.device).destroy_swapchain(self.swap_chain, None);
            self.device.destroy_device(None);
            DebugUtils::new(&self.entry, &self.instance)
                .destroy_debug_utils_messenger(self.debug_messenger, None);
            Surface::new(&self.entry, &self.instance).destroy_surface(self.surface, None);
            self.instance.destroy_instance(None);
        }
    }
That handles the swapchain! Now we need to set up actually pulling images out of it, and setting up some additional data entries in our overall struct, leaving us with this:
use ash::extensions::ext::DebugUtils;
use ash::extensions::khr::Surface;
use ash::extensions::khr::Swapchain;
use ash::extensions::khr::{AndroidSurface, WaylandSurface, Win32Surface, XcbSurface, XlibSurface};
use ash::prelude::*;
use ash::{vk, Entry};
use raw_window_handle::HasRawWindowHandle;
use std::collections::HashSet;
use std::ffi::c_void;
use std::ffi::CStr;
use std::ptr;
use std::vec::Vec;
use winit::{
    dpi::PhysicalSize,
    event::{Event, WindowEvent},
    event_loop::{ControlFlow, EventLoop},
    window::WindowBuilder,
};
const WIDTH: u32 = 800;
const HEIGHT: u32 = 600;
const VALIDATION_LAYERS: &[*const i8] = &[unsafe {
    CStr::from_bytes_with_nul_unchecked("VK_LAYER_KHRONOS_validation\0".as_bytes()).as_ptr()
}];
const DEVICE_EXTENSIONS: &[*const i8] =
    &[unsafe { CStr::from_bytes_with_nul_unchecked("VK_KHR_swapchain\0".as_bytes()).as_ptr() }];
const REQUIRED_EXTENSIONS: &[*const i8] = &[
    Surface::name().as_ptr(),
    Win32Surface::name().as_ptr(),
    DebugUtils::name().as_ptr(),
];
extern "system" fn debug_callback(
    _message_severity: vk::DebugUtilsMessageSeverityFlagsEXT,
    _message_type: vk::DebugUtilsMessageTypeFlagsEXT,
    callback_data: *const vk::DebugUtilsMessengerCallbackDataEXT,
    _: *mut c_void,
) -> vk::Bool32 {
    print!("validation layer: {}", unsafe {
        CStr::from_ptr((*callback_data).p_message).to_str().unwrap()
    });
    vk::FALSE
}
struct SwapchainSupportDetails {
    capabilities: vk::SurfaceCapabilitiesKHR,
    formats: Vec<vk::SurfaceFormatKHR>,
    present_modes: Vec<vk::PresentModeKHR>,
}
impl SwapchainSupportDetails {
    pub fn new(
        entry: &ash::Entry,
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
        surface: &vk::SurfaceKHR,
    ) -> Self {
        let surface_interface = Surface::new(entry, instance);
        Self {
            capabilities: unsafe {
                surface_interface
                    .get_physical_device_surface_capabilities(*device, *surface)
                    .unwrap()
            },
            formats: unsafe {
                surface_interface
                    .get_physical_device_surface_formats(*device, *surface)
                    .unwrap()
            },
            present_modes: unsafe {
                surface_interface
                    .get_physical_device_surface_present_modes(*device, *surface)
                    .unwrap()
            },
        }
    }
}
pub struct VulkanDetails {
    entry: ash::Entry,
    instance: ash::Instance,
    debug_messenger: vk::DebugUtilsMessengerEXT,
    surface: vk::SurfaceKHR,
    physical_device: vk::PhysicalDevice,
    device: ash::Device,
    graphics_queue: vk::Queue,
    present_queue: vk::Queue,
    swap_chain: vk::SwapchainKHR,
    swap_chain_images: Vec<vk::Image>,
    swap_chain_image_format: vk::Format,
    swap_chain_extent: vk::Extent2D,
}
pub struct HelloTriangleApplication {
    event_loop: winit::event_loop::EventLoop<()>,
    window: winit::window::Window,
    vulkan_details: VulkanDetails,
}
impl VulkanDetails {
    pub fn new(window: &winit::window::Window) -> Self {
        let (
            entry,
            instance,
            debug_messenger,
            surface,
            physical_device,
            device,
            graphics_queue,
            present_queue,
            swap_chain,
            swap_chain_images,
            swap_chain_image_format,
            swap_chain_extent,
        ) = VulkanDetails::init_vulkan(&window);
        Self {
            entry: entry,
            instance: instance,
            debug_messenger: debug_messenger,
            surface: surface,
            physical_device: physical_device,
            device: device,
            graphics_queue: graphics_queue,
            present_queue: present_queue,
            swap_chain: swap_chain,
            swap_chain_images: swap_chain_images,
            swap_chain_image_format: swap_chain_image_format,
            swap_chain_extent: swap_chain_extent,
        }
    }
    fn init_vulkan(
        window: &winit::window::Window,
    ) -> (
        ash::Entry,
        ash::Instance,
        vk::DebugUtilsMessengerEXT,
        vk::SurfaceKHR,
        vk::PhysicalDevice,
        ash::Device,
        vk::Queue,
        vk::Queue,
        vk::SwapchainKHR,
        Vec<vk::Image>,
        vk::Format,
        vk::Extent2D,
    ) {
        let entry = Entry::linked();
        let instance = VulkanDetails::create_instance(&entry).unwrap();
        let debug_messenger = VulkanDetails::create_debug_messenger(&entry, &instance);
        let surface = VulkanDetails::create_surface(&window, &entry, &instance).unwrap();
        let physical_device =
            VulkanDetails::pick_physical_device(&entry, &instance, &surface).unwrap();
        let device =
            VulkanDetails::create_logical_device(&entry, &instance, &physical_device, &surface);
        let (graphics_queue_index, present_queue_index) =
            VulkanDetails::find_queue_familes(&entry, &instance, &physical_device, &surface);
        let graphics_queue =
            unsafe { device.get_device_queue(graphics_queue_index.unwrap() as u32, 0) };
        let present_queue =
            unsafe { device.get_device_queue(present_queue_index.unwrap() as u32, 0) };
        let (swap_chain, swap_chain_images, swap_chain_format, swap_chain_extent) =
            VulkanDetails::create_swap_chain(
                window,
                &entry,
                &instance,
                &physical_device,
                &device,
                &surface,
            );
        (
            entry,
            instance,
            debug_messenger,
            surface,
            physical_device,
            device,
            graphics_queue,
            present_queue,
            swap_chain,
            swap_chain_images,
            swap_chain_format,
            swap_chain_extent,
        )
    }
    fn create_instance(entry: &ash::Entry) -> VkResult<ash::Instance> {
        if !VulkanDetails::check_validation_layer_support(&entry) {
            return Err(vk::Result::ERROR_INITIALIZATION_FAILED);
        }
        let app_info = vk::ApplicationInfo {
            s_type: vk::StructureType::APPLICATION_INFO,
            p_application_name: unsafe {
                CStr::from_bytes_with_nul_unchecked("Hello Triangle\0".as_bytes()).as_ptr()
            },
            application_version: vk::make_api_version(0, 1, 0, 0),
            p_engine_name: unsafe {
                CStr::from_bytes_with_nul_unchecked("No Engine\0".as_bytes()).as_ptr()
            },
            engine_version: vk::make_api_version(0, 1, 0, 0),
            api_version: vk::make_api_version(0, 1, 0, 0),
            ..Default::default()
        };
        let create_info = vk::InstanceCreateInfo {
            p_application_info: &app_info,
            enabled_layer_count: VALIDATION_LAYERS.len() as u32,
            pp_enabled_layer_names: VALIDATION_LAYERS.as_ptr(),
            enabled_extension_count: REQUIRED_EXTENSIONS.len() as u32,
            pp_enabled_extension_names: REQUIRED_EXTENSIONS.as_ptr(),
            p_next: &VulkanDetails::populate_debug_messenger_create_info() as *const _
                as *const c_void,
            ..Default::default()
        };
        unsafe { entry.create_instance(&create_info, None) }
    }
    fn check_validation_layer_support(entry: &ash::Entry) -> bool {
        let layer_properties = entry.enumerate_instance_layer_properties().unwrap();
        for layer in VALIDATION_LAYERS {
            if let None = layer_properties.iter().find(|l| {
                // This horrible construction is because Vulkan operates with C strings and Rust does not
                unsafe {
                    &CStr::from_ptr(l.layer_name.as_ptr()).to_str().unwrap()
                        == &CStr::from_ptr(*layer).to_str().unwrap()
                }
            }) {
                return false;
            }
        }
        true
    }
    fn create_debug_messenger(
        entry: &ash::Entry,
        instance: &ash::Instance,
    ) -> vk::DebugUtilsMessengerEXT {
        unsafe {
            DebugUtils::new(&entry, &instance)
                .create_debug_utils_messenger(
                    &VulkanDetails::populate_debug_messenger_create_info(),
                    None,
                )
                .unwrap()
        }
    }
    fn create_surface(
        window: &winit::window::Window,
        entry: &ash::Entry,
        instance: &ash::Instance,
    ) -> VkResult<vk::SurfaceKHR> {
        match window.raw_window_handle() {
            raw_window_handle::RawWindowHandle::AndroidNdk(handle) => {
                let surface_create_info = vk::AndroidSurfaceCreateInfoKHR {
                    s_type: vk::StructureType::ANDROID_SURFACE_CREATE_INFO_KHR,
                    window: handle.a_native_window,
                    ..Default::default()
                };
                let android_surface = AndroidSurface::new(&entry, &instance);
                unsafe { android_surface.create_android_surface(&surface_create_info, None) }
            }
            raw_window_handle::RawWindowHandle::Win32(handle) => {
                let surface_create_info = vk::Win32SurfaceCreateInfoKHR {
                    s_type: vk::StructureType::WIN32_SURFACE_CREATE_INFO_KHR,
                    hwnd: handle.hwnd,
                    hinstance: handle.hinstance,
                    ..Default::default()
                };
                let win32_surface = Win32Surface::new(&entry, &instance);
                unsafe { win32_surface.create_win32_surface(&surface_create_info, None) }
            }
            raw_window_handle::RawWindowHandle::Wayland(handle) => {
                let surface_create_info = vk::WaylandSurfaceCreateInfoKHR {
                    s_type: vk::StructureType::WAYLAND_SURFACE_CREATE_INFO_KHR,
                    display: handle.surface,
                    ..Default::default()
                };
                let wayland_surface = WaylandSurface::new(&entry, &instance);
                unsafe { wayland_surface.create_wayland_surface(&surface_create_info, None) }
            }
            raw_window_handle::RawWindowHandle::Xcb(handle) => {
                let surface_create_info = vk::XcbSurfaceCreateInfoKHR {
                    s_type: vk::StructureType::XCB_SURFACE_CREATE_INFO_KHR,
                    window: handle.window,
                    ..Default::default()
                };
                let xcb_surface = XcbSurface::new(&entry, &instance);
                unsafe { xcb_surface.create_xcb_surface(&surface_create_info, None) }
            }
            raw_window_handle::RawWindowHandle::Xlib(handle) => {
                let surface_create_info = vk::XlibSurfaceCreateInfoKHR {
                    s_type: vk::StructureType::XLIB_SURFACE_CREATE_INFO_KHR,
                    window: handle.window,
                    ..Default::default()
                };
                let xlib_surface = XlibSurface::new(&entry, &instance);
                unsafe { xlib_surface.create_xlib_surface(&surface_create_info, None) }
            }
            _ => Err(vk::Result::ERROR_INITIALIZATION_FAILED),
        }
    }
    fn populate_debug_messenger_create_info() -> vk::DebugUtilsMessengerCreateInfoEXT {
        vk::DebugUtilsMessengerCreateInfoEXT {
            s_type: vk::StructureType::DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT,
            message_severity: vk::DebugUtilsMessageSeverityFlagsEXT::VERBOSE
                | vk::DebugUtilsMessageSeverityFlagsEXT::WARNING
                | vk::DebugUtilsMessageSeverityFlagsEXT::ERROR,
            message_type: vk::DebugUtilsMessageTypeFlagsEXT::GENERAL
                | vk::DebugUtilsMessageTypeFlagsEXT::VALIDATION
                | vk::DebugUtilsMessageTypeFlagsEXT::PERFORMANCE,
            pfn_user_callback: Some(debug_callback),
            ..Default::default()
        }
    }
    fn pick_physical_device(
        entry: &ash::Entry,
        instance: &ash::Instance,
        surface: &vk::SurfaceKHR,
    ) -> VkResult<vk::PhysicalDevice> {
        let mut physical_device: Option<vk::PhysicalDevice> = None;
        let devices = unsafe { instance.enumerate_physical_devices().unwrap() };
        if devices.len() == 0 {
            return Err(vk::Result::ERROR_INITIALIZATION_FAILED);
        }
        for device in devices {
            if VulkanDetails::is_device_suitable(entry, instance, &device, surface) {
                physical_device = Some(device);
            }
        }
        physical_device.ok_or(vk::Result::ERROR_INITIALIZATION_FAILED)
    }
    fn is_device_suitable(
        entry: &ash::Entry,
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
        surface: &vk::SurfaceKHR,
    ) -> bool {
        let (graphics_queue_index, present_queue_index) =
            VulkanDetails::find_queue_familes(entry, instance, device, surface);
        let swap_chain_support = SwapchainSupportDetails::new(entry, instance, device, surface);
        graphics_queue_index.is_some()
            && present_queue_index.is_some()
            && VulkanDetails::check_device_extension_support(instance, device)
            && !swap_chain_support.formats.is_empty()
            && !swap_chain_support.present_modes.is_empty()
    }
    fn check_device_extension_support(
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
    ) -> bool {
        let extension_properties = unsafe {
            instance
                .enumerate_device_extension_properties(*device)
                .unwrap()
        };
        for device_extension in DEVICE_EXTENSIONS {
            if extension_properties
                .iter()
                .find(|extension_property| unsafe {
                    &CStr::from_ptr(extension_property.extension_name.as_ptr())
                        .to_str()
                        .unwrap()
                        == &CStr::from_ptr(*device_extension).to_str().unwrap()
                })
                .is_none()
            {
                return false;
            }
        }
        true
    }
    fn find_queue_familes(
        entry: &ash::Entry,
        instance: &ash::Instance,
        device: &vk::PhysicalDevice,
        surface: &vk::SurfaceKHR,
    ) -> (Option<usize>, Option<usize>) {
        let queue_family_properties =
            unsafe { instance.get_physical_device_queue_family_properties(*device) };
        let surface_details = Surface::new(entry, instance);
        (
            queue_family_properties.iter().position(|&queue_family| {
                queue_family.queue_flags & vk::QueueFlags::GRAPHICS == vk::QueueFlags::GRAPHICS
            }),
            queue_family_properties
                .iter()
                .enumerate()
                .position(|(index, _)| unsafe {
                    surface_details
                        .get_physical_device_surface_support(*device, index as u32, *surface)
                        .unwrap()
                }),
        )
    }
    fn create_logical_device(
        entry: &ash::Entry,
        instance: &ash::Instance,
        physical_device: &vk::PhysicalDevice,
        surface: &vk::SurfaceKHR,
    ) -> ash::Device {
        let (gq, pq) = VulkanDetails::find_queue_familes(entry, instance, physical_device, surface);
        let mut queues = HashSet::new();
        queues.insert(gq.unwrap() as u32);
        queues.insert(pq.unwrap() as u32);
        let mut device_queue_create_infos = Vec::new();
        for queue in queues {
            device_queue_create_infos.push(vk::DeviceQueueCreateInfo {
                s_type: vk::StructureType::DEVICE_QUEUE_CREATE_INFO,
                queue_family_index: queue,
                queue_count: 1,
                p_queue_priorities: &1.0,
                ..Default::default()
            })
        }
        let device_features = vk::PhysicalDeviceFeatures {
            ..Default::default()
        };
        let device_create_info = vk::DeviceCreateInfo {
            s_type: vk::StructureType::DEVICE_CREATE_INFO,
            queue_create_info_count: device_queue_create_infos.len() as u32,
            p_queue_create_infos: device_queue_create_infos.as_ptr(),
            p_enabled_features: &device_features,
            enabled_layer_count: VALIDATION_LAYERS.len() as u32,
            pp_enabled_layer_names: VALIDATION_LAYERS.as_ptr(),
            enabled_extension_count: DEVICE_EXTENSIONS.len() as u32,
            pp_enabled_extension_names: DEVICE_EXTENSIONS.as_ptr(),
            ..Default::default()
        };
        unsafe {
            instance
                .create_device(*physical_device, &device_create_info, None)
                .unwrap()
        }
    }
    fn create_swap_chain(
        window: &winit::window::Window,
        entry: &ash::Entry,
        instance: &ash::Instance,
        physical_device: &vk::PhysicalDevice,
        device: &ash::Device,
        surface: &vk::SurfaceKHR,
    ) -> (vk::SwapchainKHR, Vec<vk::Image>, vk::Format, vk::Extent2D) {
        let swap_chain_support =
            SwapchainSupportDetails::new(entry, instance, physical_device, surface);
        let format = VulkanDetails::choose_swap_surface_format(swap_chain_support.formats);
        let present_mode =
            VulkanDetails::choose_swap_present_mode(swap_chain_support.present_modes);
        let image_count = {
            if swap_chain_support.capabilities.max_image_count > 0
                && swap_chain_support.capabilities.min_image_count
                    > swap_chain_support.capabilities.max_image_count
            {
                swap_chain_support.capabilities.max_image_count
            } else {
                swap_chain_support.capabilities.min_image_count + 1
            }
        };
        let extent = VulkanDetails::choose_swap_extent(window, &swap_chain_support.capabilities);
        let (graphics_queue_index, present_mode_index) =
            VulkanDetails::find_queue_familes(entry, instance, physical_device, surface);
        let queue_index_equivalent = graphics_queue_index.unwrap() == present_mode_index.unwrap();
        let queue_family_indices = vec![graphics_queue_index.unwrap(), present_mode_index.unwrap()];
        let create_info = vk::SwapchainCreateInfoKHR {
            s_type: vk::StructureType::SWAPCHAIN_CREATE_INFO_KHR,
            surface: *surface,
            min_image_count: image_count,
            image_format: format.format,
            image_color_space: format.color_space,
            image_extent: extent,
            image_array_layers: 1,
            image_usage: vk::ImageUsageFlags::COLOR_ATTACHMENT,
            image_sharing_mode: if queue_index_equivalent {
                vk::SharingMode::EXCLUSIVE
            } else {
                vk::SharingMode::CONCURRENT
            },
            queue_family_index_count: if queue_index_equivalent { 0 } else { 2 },
            p_queue_family_indices: if queue_index_equivalent {
                ptr::null()
            } else {
                queue_family_indices.as_ptr() as *const u32
            },
            pre_transform: swap_chain_support.capabilities.current_transform,
            composite_alpha: vk::CompositeAlphaFlagsKHR::OPAQUE,
            present_mode: present_mode,
            clipped: vk::TRUE,
            old_swapchain: vk::SwapchainKHR::null(),
            ..Default::default()
        };
        let swap_chain_handle = Swapchain::new(instance, device);
        let swap_chain = unsafe {
            swap_chain_handle
                .create_swapchain(&create_info, None)
                .unwrap()
        };
        let swap_chain_images =
            unsafe { swap_chain_handle.get_swapchain_images(swap_chain).unwrap() };
        (swap_chain, swap_chain_images, format.format, extent)
    }
    fn choose_swap_surface_format(formats: Vec<vk::SurfaceFormatKHR>) -> vk::SurfaceFormatKHR {
        for available_format in &formats {
            if available_format.format == vk::Format::B8G8R8A8_SRGB
                && available_format.color_space == vk::ColorSpaceKHR::SRGB_NONLINEAR
            {
                return *available_format;
            }
        }
        formats[0]
    }
    fn choose_swap_present_mode(present_modes: Vec<vk::PresentModeKHR>) -> vk::PresentModeKHR {
        for available_present_mode in present_modes {
            if available_present_mode == vk::PresentModeKHR::MAILBOX {
                return available_present_mode;
            }
        }
        vk::PresentModeKHR::FIFO
    }
    fn choose_swap_extent(
        window: &winit::window::Window,
        capabilities: &vk::SurfaceCapabilitiesKHR,
    ) -> vk::Extent2D {
        if capabilities.current_extent.width != u32::MAX {
            capabilities.current_extent
        } else {
            let window_size = window.inner_size();
            vk::Extent2D {
                width: window_size.width.clamp(
                    capabilities.min_image_extent.width,
                    capabilities.max_image_extent.width,
                ),
                height: window_size.height.clamp(
                    capabilities.min_image_extent.height,
                    capabilities.max_image_extent.height,
                ),
            }
        }
    }
    fn cleanup(&mut self) {
        unsafe {
            Swapchain::new(&self.instance, &self.device).destroy_swapchain(self.swap_chain, None);
            self.device.destroy_device(None);
            DebugUtils::new(&self.entry, &self.instance)
                .destroy_debug_utils_messenger(self.debug_messenger, None);
            Surface::new(&self.entry, &self.instance).destroy_surface(self.surface, None);
            self.instance.destroy_instance(None);
        }
    }
}
impl HelloTriangleApplication {
    pub fn new() -> Self {
        let (event_loop, window) = HelloTriangleApplication::init_window().unwrap();
        let vulkan_details = VulkanDetails::new(&window);
        Self {
            event_loop: event_loop,
            window: window,
            vulkan_details: vulkan_details,
        }
    }
    pub fn run(mut self) -> ! {
        self.event_loop.run(move |event, _, control_flow| {
            *control_flow = ControlFlow::Wait;
            match event {
                Event::WindowEvent {
                    event: WindowEvent::CloseRequested,
                    window_id,
                } if window_id == self.window.id() => {
                    self.vulkan_details.cleanup();
                    *control_flow = ControlFlow::Exit
                }
                _ => (),
            }
        });
    }
    fn init_window(
    ) -> Result<(winit::event_loop::EventLoop<()>, winit::window::Window), winit::error::OsError>
    {
        let event_loop = EventLoop::new();
        let window = WindowBuilder::new()
            .with_resizable(false)
            .with_inner_size(PhysicalSize::new(WIDTH, HEIGHT))
            .build(&event_loop)?;
        Ok((event_loop, window))
    }
}
Things are getting quite sizable now! But, we’re not done yet: we need to add some image views. The struct grows once more:
pub struct VulkanDetails {
    entry: ash::Entry,
    instance: ash::Instance,
    debug_messenger: vk::DebugUtilsMessengerEXT,
    surface: vk::SurfaceKHR,
    physical_device: vk::PhysicalDevice,
    device: ash::Device,
    graphics_queue: vk::Queue,
    present_queue: vk::Queue,
    swap_chain: vk::SwapchainKHR,
    swap_chain_images: Vec<vk::Image>,
    swap_chain_image_format: vk::Format,
    swap_chain_extent: vk::Extent2D,
    swap_chain_image_views: Vec<vk::ImageView>,
}
Then we construct our image view, similarly to the tutorial though as always involving a lot more passing of data around:
    fn create_image_views(
        device: &ash::Device,
        swap_chain_images: &Vec<vk::Image>,
        swap_chain_image_format: &vk::Format,
    ) -> Vec<vk::ImageView> {
        let mut output_vec = Vec::new();
        for image in swap_chain_images {
            let create_info = vk::ImageViewCreateInfo {
                s_type: vk::StructureType::IMAGE_VIEW_CREATE_INFO,
                image: *image,
                view_type: vk::ImageViewType::TYPE_2D,
                format: *swap_chain_image_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,
                },
                ..Default::default()
            };
            unsafe { output_vec.push(device.create_image_view(&create_info, None).unwrap()) };
        }
        output_vec
    }
    fn cleanup(&mut self) {
        unsafe {
            for image_view in &self.swap_chain_image_views {
                self.device.destroy_image_view(*image_view, None);
            }
            Swapchain::new(&self.instance, &self.device).destroy_swapchain(self.swap_chain, None);
            self.device.destroy_device(None);
            DebugUtils::new(&self.entry, &self.instance)
                .destroy_debug_utils_messenger(self.debug_messenger, None);
            Surface::new(&self.entry, &self.instance).destroy_surface(self.surface, None);
            self.instance.destroy_instance(None);
        }
    }
Throw in the call in the expected place and everything lines up perfectly. Next up is the graphics pipeline, which will be quite the adventure.