use std::sync::Arc;
use std::marker::PhantomData;
use smallvec::SmallVec;
use vks;
use ::{VdResult, Device, DescriptorSetLayoutHandle, Handle,
WriteDescriptorSet, CopyDescriptorSet, DescriptorSet,
DescriptorSetAllocateInfo, DescriptorSetHandle};
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
#[repr(C)]
pub struct DescriptorPoolHandle(pub(crate) vks::VkDescriptorPool);
impl DescriptorPoolHandle {
#[inline(always)]
pub fn to_raw(&self) -> vks::VkDescriptorPool {
self.0
}
}
unsafe impl Handle for DescriptorPoolHandle {
type Target = DescriptorPoolHandle;
#[inline(always)]
fn handle(&self) -> Self::Target {
*self
}
}
#[derive(Debug)]
struct Inner {
handle: DescriptorPoolHandle,
device: Device,
}
impl Drop for Inner {
fn drop(&mut self) {
unsafe {
self.device.destroy_descriptor_pool(self.handle, None);
}
}
}
#[derive(Debug, Clone)]
pub struct DescriptorPool {
inner: Arc<Inner>,
}
impl DescriptorPool {
pub fn builder<'b>() -> DescriptorPoolBuilder<'b> {
DescriptorPoolBuilder::new()
}
pub fn handle(&self) -> DescriptorPoolHandle {
self.inner.handle
}
pub fn device(&self) -> &Device {
&self.inner.device
}
pub fn allocate_descriptor_sets<Ds>(&self, layouts: &[Ds])
-> VdResult<SmallVec<[DescriptorSet; 8]>>
where Ds: Handle<Target=DescriptorSetLayoutHandle> {
let layouts: SmallVec<[DescriptorSetLayoutHandle; 8]> = layouts.iter().map(|ds|
ds.handle()).collect();
let alloc_info = DescriptorSetAllocateInfo::builder()
.descriptor_pool(self.handle())
.set_layouts(&layouts)
.build();
let descriptor_set_handles: SmallVec<[DescriptorSetHandle; 8]> = unsafe {
self.inner.device.allocate_descriptor_sets(&alloc_info)?
};
Ok(descriptor_set_handles.iter().map(|&dsh| DescriptorSet(dsh)).collect())
}
pub fn update_descriptor_sets(&self, descriptor_writes: &[WriteDescriptorSet],
descriptor_copies: &[CopyDescriptorSet]) {
self.inner.device.update_descriptor_sets(descriptor_writes, descriptor_copies)
}
}
unsafe impl<'d> Handle for &'d DescriptorPool {
type Target = DescriptorPoolHandle;
#[inline(always)]
fn handle(&self) -> Self::Target {
self.inner.handle
}
}
#[derive(Debug, Clone)]
pub struct DescriptorPoolBuilder<'b> {
create_info: ::DescriptorPoolCreateInfo<'b>,
_p: PhantomData<&'b ()>,
}
impl<'b> DescriptorPoolBuilder<'b> {
pub fn new() -> DescriptorPoolBuilder<'b> {
DescriptorPoolBuilder {
create_info: ::DescriptorPoolCreateInfo::default(),
_p: PhantomData,
}
}
pub fn flags<'s>(&'s mut self, flags: ::DescriptorPoolCreateFlags)
-> &'s mut DescriptorPoolBuilder<'b> {
self.create_info.set_flags(flags);
self
}
pub fn max_sets<'s>(&'s mut self, max_sets: u32)
-> &'s mut DescriptorPoolBuilder<'b> {
self.create_info.set_max_sets(max_sets);
self
}
pub fn pool_sizes<'s, 'p>(&'s mut self,
pool_sizes: &'p [::DescriptorPoolSize])
-> &'s mut DescriptorPoolBuilder<'b>
where 'p: 'b {
self.create_info.set_pool_sizes(pool_sizes);
self
}
pub fn build(&self, device: Device) -> VdResult<DescriptorPool> {
let handle = unsafe { device.create_descriptor_pool(&self.create_info, None)? };
Ok(DescriptorPool {
inner: Arc::new(Inner {
handle,
device,
})
})
}
}