ash/extensions/khr/
swapchain.rs

1#[cfg(doc)]
2use super::DeviceGroup;
3use crate::prelude::*;
4use crate::vk;
5use crate::RawPtr;
6use crate::{Device, Entry, Instance};
7use std::ffi::CStr;
8use std::mem;
9
10/// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_KHR_swapchain.html>
11#[derive(Clone)]
12pub struct Swapchain {
13    handle: vk::Device,
14    fp: vk::KhrSwapchainFn,
15}
16
17impl Swapchain {
18    /// # Warning
19    /// [`Instance`] functions cannot be loaded from a [`Device`] and will always panic when called:
20    /// - [`Self::get_physical_device_present_rectangles()`]
21    ///
22    /// Load this struct using an [`Instance`] instead via [`Self::new_from_instance()`] if the
23    /// above [`Instance`] function is called. This will be solved in the next breaking `ash`
24    /// release: <https://github.com/ash-rs/ash/issues/727>.
25    pub fn new(instance: &Instance, device: &Device) -> Self {
26        let handle = device.handle();
27        let fp = vk::KhrSwapchainFn::load(|name| unsafe {
28            mem::transmute(instance.get_device_proc_addr(handle, name.as_ptr()))
29        });
30        Self { handle, fp }
31    }
32
33    /// Loads all functions on the [`Instance`] instead of [`Device`]. This incurs an extra
34    /// dispatch table for [`Device`] functions but also allows the [`Instance`] function to be
35    /// loaded instead of always panicking. See also [`Self::new()`] for more details.
36    ///
37    /// It is okay to pass [`vk::Device::null()`] when this struct is only used to call the
38    /// [`Instance`] function.
39    pub fn new_from_instance(entry: &Entry, instance: &Instance, device: vk::Device) -> Self {
40        let fp = vk::KhrSwapchainFn::load(|name| unsafe {
41            mem::transmute(entry.get_instance_proc_addr(instance.handle(), name.as_ptr()))
42        });
43        Self { handle: device, fp }
44    }
45
46    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkCreateSwapchainKHR.html>
47    #[inline]
48    pub unsafe fn create_swapchain(
49        &self,
50        create_info: &vk::SwapchainCreateInfoKHR,
51        allocation_callbacks: Option<&vk::AllocationCallbacks>,
52    ) -> VkResult<vk::SwapchainKHR> {
53        let mut swapchain = mem::zeroed();
54        (self.fp.create_swapchain_khr)(
55            self.handle,
56            create_info,
57            allocation_callbacks.as_raw_ptr(),
58            &mut swapchain,
59        )
60        .result_with_success(swapchain)
61    }
62
63    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkDestroySwapchainKHR.html>
64    #[inline]
65    pub unsafe fn destroy_swapchain(
66        &self,
67        swapchain: vk::SwapchainKHR,
68        allocation_callbacks: Option<&vk::AllocationCallbacks>,
69    ) {
70        (self.fp.destroy_swapchain_khr)(self.handle, swapchain, allocation_callbacks.as_raw_ptr());
71    }
72
73    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkGetSwapchainImagesKHR.html>
74    #[inline]
75    pub unsafe fn get_swapchain_images(
76        &self,
77        swapchain: vk::SwapchainKHR,
78    ) -> VkResult<Vec<vk::Image>> {
79        read_into_uninitialized_vector(|count, data| {
80            (self.fp.get_swapchain_images_khr)(self.handle, swapchain, count, data)
81        })
82    }
83
84    /// On success, returns the next image's index and whether the swapchain is suboptimal for the surface.
85    ///
86    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkAcquireNextImageKHR.html>
87    #[inline]
88    pub unsafe fn acquire_next_image(
89        &self,
90        swapchain: vk::SwapchainKHR,
91        timeout: u64,
92        semaphore: vk::Semaphore,
93        fence: vk::Fence,
94    ) -> VkResult<(u32, bool)> {
95        let mut index = 0;
96        let err_code = (self.fp.acquire_next_image_khr)(
97            self.handle,
98            swapchain,
99            timeout,
100            semaphore,
101            fence,
102            &mut index,
103        );
104        match err_code {
105            vk::Result::SUCCESS => Ok((index, false)),
106            vk::Result::SUBOPTIMAL_KHR => Ok((index, true)),
107            _ => Err(err_code),
108        }
109    }
110
111    /// On success, returns whether the swapchain is suboptimal for the surface.
112    ///
113    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkQueuePresentKHR.html>
114    #[inline]
115    pub unsafe fn queue_present(
116        &self,
117        queue: vk::Queue,
118        present_info: &vk::PresentInfoKHR,
119    ) -> VkResult<bool> {
120        let err_code = (self.fp.queue_present_khr)(queue, present_info);
121        match err_code {
122            vk::Result::SUCCESS => Ok(false),
123            vk::Result::SUBOPTIMAL_KHR => Ok(true),
124            _ => Err(err_code),
125        }
126    }
127
128    /// Only available since [Vulkan 1.1].
129    ///
130    /// Also available as [`DeviceGroup::get_device_group_present_capabilities()`]
131    /// when [`VK_KHR_surface`] is enabled.
132    ///
133    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkGetDeviceGroupPresentCapabilitiesKHR.html>
134    ///
135    /// [Vulkan 1.1]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_VERSION_1_1.html
136    /// [`VK_KHR_surface`]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_KHR_surface.html
137    #[inline]
138    pub unsafe fn get_device_group_present_capabilities(
139        &self,
140        device_group_present_capabilities: &mut vk::DeviceGroupPresentCapabilitiesKHR,
141    ) -> VkResult<()> {
142        (self.fp.get_device_group_present_capabilities_khr)(
143            self.handle,
144            device_group_present_capabilities,
145        )
146        .result()
147    }
148
149    /// Only available since [Vulkan 1.1].
150    ///
151    /// Also available as [`DeviceGroup::get_device_group_surface_present_modes()`]
152    /// when [`VK_KHR_surface`] is enabled.
153    ///
154    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkGetDeviceGroupSurfacePresentModesKHR.html>
155    ///
156    /// [Vulkan 1.1]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_VERSION_1_1.html
157    /// [`VK_KHR_surface`]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_KHR_surface.html
158    #[inline]
159    pub unsafe fn get_device_group_surface_present_modes(
160        &self,
161        surface: vk::SurfaceKHR,
162    ) -> VkResult<vk::DeviceGroupPresentModeFlagsKHR> {
163        let mut modes = mem::zeroed();
164        (self.fp.get_device_group_surface_present_modes_khr)(self.handle, surface, &mut modes)
165            .result_with_success(modes)
166    }
167
168    /// Only available since [Vulkan 1.1].
169    ///
170    /// Also available as [`DeviceGroup::get_physical_device_present_rectangles()`]
171    /// when [`VK_KHR_surface`] is enabled.
172    ///
173    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkGetPhysicalDevicePresentRectanglesKHR.html>
174    ///
175    /// [Vulkan 1.1]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_VERSION_1_1.html
176    /// [`VK_KHR_surface`]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_KHR_surface.html
177    ///
178    /// # Warning
179    ///
180    /// Function will always panic unless this struct is loaded via [`Self::new_from_instance()`].
181    #[inline]
182    pub unsafe fn get_physical_device_present_rectangles(
183        &self,
184        physical_device: vk::PhysicalDevice,
185        surface: vk::SurfaceKHR,
186    ) -> VkResult<Vec<vk::Rect2D>> {
187        read_into_uninitialized_vector(|count, data| {
188            (self.fp.get_physical_device_present_rectangles_khr)(
189                physical_device,
190                surface,
191                count,
192                data,
193            )
194        })
195    }
196
197    /// On success, returns the next image's index and whether the swapchain is suboptimal for the surface.
198    ///
199    /// Only available since [Vulkan 1.1].
200    ///
201    /// Also available as [`DeviceGroup::acquire_next_image2()`]
202    /// when [`VK_KHR_swapchain`] is enabled.
203    ///
204    /// <https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/vkAcquireNextImage2KHR.html>
205    ///
206    /// [Vulkan 1.1]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_VERSION_1_1.html
207    /// [`VK_KHR_swapchain`]: https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html/VK_KHR_swapchain.html
208    #[inline]
209    pub unsafe fn acquire_next_image2(
210        &self,
211        acquire_info: &vk::AcquireNextImageInfoKHR,
212    ) -> VkResult<(u32, bool)> {
213        let mut index = 0;
214        let err_code = (self.fp.acquire_next_image2_khr)(self.handle, acquire_info, &mut index);
215        match err_code {
216            vk::Result::SUCCESS => Ok((index, false)),
217            vk::Result::SUBOPTIMAL_KHR => Ok((index, true)),
218            _ => Err(err_code),
219        }
220    }
221
222    #[inline]
223    pub const fn name() -> &'static CStr {
224        vk::KhrSwapchainFn::name()
225    }
226
227    #[inline]
228    pub fn fp(&self) -> &vk::KhrSwapchainFn {
229        &self.fp
230    }
231
232    #[inline]
233    pub fn device(&self) -> vk::Device {
234        self.handle
235    }
236}