vulkano 0.35.2

Safe wrapper for the Vulkan graphics API
Documentation
//! Low-level descriptor set.

use super::{
    allocator::{DescriptorSetAlloc, DescriptorSetAllocator},
    pool::DescriptorPool,
    CopyDescriptorSet,
};
use crate::{
    descriptor_set::{layout::DescriptorSetLayout, update::WriteDescriptorSet},
    device::{Device, DeviceOwned},
    Validated, ValidationError, VulkanError, VulkanObject,
};
use smallvec::SmallVec;
use std::{
    fmt::Debug,
    hash::{Hash, Hasher},
    mem::ManuallyDrop,
    sync::Arc,
};

/// A raw descriptor set corresponding directly to a `VkDescriptorSet`.
///
/// This descriptor set does not keep track of synchronization, nor does it store any information
/// on what resources have been written to each descriptor.
#[derive(Debug)]
pub struct RawDescriptorSet {
    allocation: ManuallyDrop<DescriptorSetAlloc>,
    allocator: Arc<dyn DescriptorSetAllocator>,
}

impl RawDescriptorSet {
    /// Allocates a new descriptor set and returns it.
    #[inline]
    pub fn new(
        allocator: Arc<dyn DescriptorSetAllocator>,
        layout: &Arc<DescriptorSetLayout>,
        variable_descriptor_count: u32,
    ) -> Result<RawDescriptorSet, Validated<VulkanError>> {
        let allocation = allocator.allocate(layout, variable_descriptor_count)?;

        Ok(RawDescriptorSet {
            allocation: ManuallyDrop::new(allocation),
            allocator,
        })
    }

    /// Returns the allocation of this descriptor set.
    #[inline]
    pub fn alloc(&self) -> &DescriptorSetAlloc {
        &self.allocation
    }

    /// Returns the descriptor pool that the descriptor set was allocated from.
    #[inline]
    pub fn pool(&self) -> &DescriptorPool {
        &self.allocation.pool
    }

    /// Returns the layout of this descriptor set.
    #[inline]
    pub fn layout(&self) -> &Arc<DescriptorSetLayout> {
        self.allocation.inner.layout()
    }

    /// Returns the variable descriptor count that this descriptor set was allocated with.
    #[inline]
    pub fn variable_descriptor_count(&self) -> u32 {
        self.allocation.inner.variable_descriptor_count()
    }

    /// Updates the descriptor set with new values.
    ///
    /// # Safety
    ///
    /// - The resources in `descriptor_writes` and `descriptor_copies` must be kept alive for as
    ///   long as `self` is in use.
    /// - The descriptor set must not be in use by the device, or be recorded to a command buffer
    ///   as part of a bind command.
    /// - Host access to the descriptor set must be externally synchronized.
    #[inline]
    pub unsafe fn update(
        &self,
        descriptor_writes: &[WriteDescriptorSet],
        descriptor_copies: &[CopyDescriptorSet],
    ) -> Result<(), Box<ValidationError>> {
        if descriptor_writes.is_empty() && descriptor_copies.is_empty() {
            return Ok(());
        }

        self.validate_update(descriptor_writes, descriptor_copies)?;

        unsafe { self.update_unchecked(descriptor_writes, descriptor_copies) };
        Ok(())
    }

    pub(super) fn validate_update(
        &self,
        descriptor_writes: &[WriteDescriptorSet],
        descriptor_copies: &[CopyDescriptorSet],
    ) -> Result<(), Box<ValidationError>> {
        for (index, write) in descriptor_writes.iter().enumerate() {
            write
                .validate(self.layout(), self.variable_descriptor_count())
                .map_err(|err| err.add_context(format!("descriptor_writes[{}]", index)))?;
        }

        for (index, copy) in descriptor_copies.iter().enumerate() {
            copy.validate(self)
                .map_err(|err| err.add_context(format!("descriptor_copies[{}]", index)))?;
        }

        Ok(())
    }

    #[cfg_attr(not(feature = "document_unchecked"), doc(hidden))]
    pub unsafe fn update_unchecked(
        &self,
        descriptor_writes: &[WriteDescriptorSet],
        descriptor_copies: &[CopyDescriptorSet],
    ) {
        if descriptor_writes.is_empty() && descriptor_copies.is_empty() {
            return;
        }

        let set_layout_bindings = self.layout().bindings();
        let writes_fields1_vk: SmallVec<[_; 8]> = descriptor_writes
            .iter()
            .map(|write| {
                let default_image_layout = set_layout_bindings[&write.binding()]
                    .descriptor_type
                    .default_image_layout();
                write.to_vk_fields1(default_image_layout)
            })
            .collect();
        let mut write_extensions_vk: SmallVec<[_; 8]> = descriptor_writes
            .iter()
            .zip(&writes_fields1_vk)
            .map(|(write, fields1_vk)| write.to_vk_extensions(fields1_vk))
            .collect();
        let writes_vk: SmallVec<[_; 8]> = descriptor_writes
            .iter()
            .zip(&writes_fields1_vk)
            .zip(&mut write_extensions_vk)
            .map(|((write, write_info_vk), write_extension_vk)| {
                write.to_vk(
                    self.handle(),
                    set_layout_bindings[&write.binding()].descriptor_type,
                    write_info_vk,
                    write_extension_vk,
                )
            })
            .collect();

        let copies_vk: SmallVec<[_; 8]> = descriptor_copies
            .iter()
            .map(|copy| copy.to_vk(self.handle()))
            .collect();

        let fns = self.device().fns();
        unsafe {
            (fns.v1_0.update_descriptor_sets)(
                self.device().handle(),
                writes_vk.len() as u32,
                writes_vk.as_ptr(),
                copies_vk.len() as u32,
                copies_vk.as_ptr(),
            )
        };
    }
}

impl Drop for RawDescriptorSet {
    #[inline]
    fn drop(&mut self) {
        let allocation = unsafe { ManuallyDrop::take(&mut self.allocation) };
        unsafe { self.allocator.deallocate(allocation) };
    }
}

unsafe impl VulkanObject for RawDescriptorSet {
    type Handle = ash::vk::DescriptorSet;

    #[inline]
    fn handle(&self) -> Self::Handle {
        self.allocation.inner.handle()
    }
}

unsafe impl DeviceOwned for RawDescriptorSet {
    #[inline]
    fn device(&self) -> &Arc<Device> {
        self.allocation.inner.device()
    }
}

impl PartialEq for RawDescriptorSet {
    #[inline]
    fn eq(&self, other: &Self) -> bool {
        self.allocation.inner == other.allocation.inner
    }
}

impl Eq for RawDescriptorSet {}

impl Hash for RawDescriptorSet {
    #[inline]
    fn hash<H: Hasher>(&self, state: &mut H) {
        self.allocation.inner.hash(state);
    }
}