From 0175bcd2a690532ab21c498aaf2a26519b5df452 Mon Sep 17 00:00:00 2001 From: Rua Date: Sat, 7 Oct 2023 15:13:58 +0200 Subject: [PATCH] Add `DeviceAddress`, handle pointers as SPIR-V struct members (#2351) * Add `DeviceAddress`, handle pointers as SPIR-V struct members * Add pointers to vulkano-shaders --- vulkano-shaders/src/structs.rs | 28 +++++++++++++++++++++++++++ vulkano/src/acceleration_structure.rs | 10 +++++----- vulkano/src/buffer/mod.rs | 10 +++++----- vulkano/src/buffer/subbuffer.rs | 10 +++++----- vulkano/src/lib.rs | 6 ++++++ 5 files changed, 49 insertions(+), 15 deletions(-) diff --git a/vulkano-shaders/src/structs.rs b/vulkano-shaders/src/structs.rs index 4e71028831..0604885dea 100644 --- a/vulkano-shaders/src/structs.rs +++ b/vulkano-shaders/src/structs.rs @@ -220,6 +220,7 @@ fn is_aligned(offset: usize, alignment: Alignment) -> bool { #[derive(Clone, Debug, PartialEq, Eq)] enum Type { Scalar(TypeScalar), + Pointer(TypePointer), Vector(TypeVector), Matrix(TypeMatrix), Array(TypeArray), @@ -238,6 +239,7 @@ impl Type { Instruction::TypeFloat { width, .. } => { Type::Scalar(TypeScalar::Float(TypeFloat::new(shader, width)?)) } + Instruction::TypePointer { .. } => Type::Pointer(TypePointer::new(shader)?), Instruction::TypeVector { component_type, component_count, @@ -268,6 +270,7 @@ impl Type { fn size(&self) -> Option { match self { Self::Scalar(ty) => Some(ty.size()), + Self::Pointer(ty) => Some(ty.size()), Self::Vector(ty) => Some(ty.size()), Self::Matrix(ty) => Some(ty.size()), Self::Array(ty) => ty.size(), @@ -278,6 +281,7 @@ impl Type { fn scalar_alignment(&self) -> Alignment { match self { Self::Scalar(ty) => ty.alignment(), + Self::Pointer(ty) => ty.alignment(), Self::Vector(ty) => ty.component_type.alignment(), Self::Matrix(ty) => ty.component_type.alignment(), Self::Array(ty) => ty.scalar_alignment(), @@ -439,6 +443,29 @@ impl ToTokens for TypeFloat { } } +#[derive(Clone, Debug, PartialEq, Eq)] +struct TypePointer; + +impl TypePointer { + fn new(_shader: &Shader) -> Result { + Ok(TypePointer) + } + + fn size(&self) -> usize { + 8 + } + + fn alignment(&self) -> Alignment { + Alignment::A8 + } +} + +impl ToTokens for TypePointer { + fn to_tokens(&self, tokens: &mut TokenStream) { + tokens.extend(quote! { ::vulkano::DeviceAddress }); + } +} + #[derive(Clone, Debug, PartialEq, Eq)] struct TypeVector { component_type: TypeScalar, @@ -863,6 +890,7 @@ impl ToTokens for Serializer<'_, Type> { fn to_tokens(&self, tokens: &mut TokenStream) { match &self.0 { Type::Scalar(ty) => ty.to_tokens(tokens), + Type::Pointer(ty) => ty.to_tokens(tokens), Type::Vector(ty) => Serializer(ty, self.1).to_tokens(tokens), Type::Matrix(ty) => Serializer(ty, self.1).to_tokens(tokens), Type::Array(ty) => Serializer(ty, self.1).to_tokens(tokens), diff --git a/vulkano/src/acceleration_structure.rs b/vulkano/src/acceleration_structure.rs index 6bd6870d1f..2aa52f95bc 100644 --- a/vulkano/src/acceleration_structure.rs +++ b/vulkano/src/acceleration_structure.rs @@ -97,8 +97,8 @@ use crate::{ format::{Format, FormatFeatures}, instance::InstanceOwnedDebugWrapper, macros::{impl_id_counter, vulkan_bitflags, vulkan_enum}, - DeviceSize, NonZeroDeviceSize, Packed24_8, Requires, RequiresAllOf, RequiresOneOf, Validated, - ValidationError, VulkanError, VulkanObject, + DeviceAddress, DeviceSize, NonNullDeviceAddress, Packed24_8, Requires, RequiresAllOf, + RequiresOneOf, Validated, ValidationError, VulkanError, VulkanObject, }; use bytemuck::{Pod, Zeroable}; use std::{fmt::Debug, hash::Hash, mem::MaybeUninit, num::NonZeroU64, ptr, sync::Arc}; @@ -264,7 +264,7 @@ impl AccelerationStructure { /// /// The device address of the acceleration structure may be different from the device address /// of the underlying buffer. - pub fn device_address(&self) -> NonZeroDeviceSize { + pub fn device_address(&self) -> NonNullDeviceAddress { let info_vk = ash::vk::AccelerationStructureDeviceAddressInfoKHR { acceleration_structure: self.handle, ..Default::default() @@ -277,7 +277,7 @@ impl AccelerationStructure { ) }; - NonZeroDeviceSize::new(ptr).unwrap() + NonNullDeviceAddress::new(ptr).unwrap() } } @@ -1277,7 +1277,7 @@ pub struct AccelerationStructureInstance { /// The device address of the bottom-level acceleration structure in this instance. /// /// The default value is 0 (null). - pub acceleration_structure_reference: DeviceSize, + pub acceleration_structure_reference: DeviceAddress, } impl Default for AccelerationStructureInstance { diff --git a/vulkano/src/buffer/mod.rs b/vulkano/src/buffer/mod.rs index 3482ac2950..0f8e0f0425 100644 --- a/vulkano/src/buffer/mod.rs +++ b/vulkano/src/buffer/mod.rs @@ -93,8 +93,8 @@ use crate::{ }, range_map::RangeMap, sync::{future::AccessError, AccessConflict, CurrentAccess, Sharing}, - DeviceSize, NonZeroDeviceSize, Requires, RequiresAllOf, RequiresOneOf, Validated, - ValidationError, Version, VulkanError, VulkanObject, + DeviceSize, NonNullDeviceAddress, NonZeroDeviceSize, Requires, RequiresAllOf, RequiresOneOf, + Validated, ValidationError, Version, VulkanError, VulkanObject, }; use parking_lot::{Mutex, MutexGuard}; use smallvec::SmallVec; @@ -476,7 +476,7 @@ impl Buffer { /// Returns the device address for this buffer. // TODO: Caching? - pub fn device_address(&self) -> Result> { + pub fn device_address(&self) -> Result> { self.validate_device_address()?; unsafe { Ok(self.device_address_unchecked()) } @@ -508,7 +508,7 @@ impl Buffer { } #[cfg_attr(not(feature = "document_unchecked"), doc(hidden))] - pub unsafe fn device_address_unchecked(&self) -> NonZeroDeviceSize { + pub unsafe fn device_address_unchecked(&self) -> NonNullDeviceAddress { let device = self.device(); let info_vk = ash::vk::BufferDeviceAddressInfo { @@ -528,7 +528,7 @@ impl Buffer { f(device.handle(), &info_vk) }; - NonZeroDeviceSize::new(ptr).unwrap() + NonNullDeviceAddress::new(ptr).unwrap() } pub(crate) fn state(&self) -> MutexGuard<'_, BufferState> { diff --git a/vulkano/src/buffer/subbuffer.rs b/vulkano/src/buffer/subbuffer.rs index 09c87e1baf..baabbb71f7 100644 --- a/vulkano/src/buffer/subbuffer.rs +++ b/vulkano/src/buffer/subbuffer.rs @@ -19,7 +19,7 @@ use crate::{ is_aligned, DeviceAlignment, MappedMemoryRange, }, sync::HostAccessError, - DeviceSize, NonZeroDeviceSize, ValidationError, + DeviceSize, NonNullDeviceAddress, NonZeroDeviceSize, ValidationError, }; use bytemuck::AnyBitPattern; use std::{ @@ -138,21 +138,21 @@ impl Subbuffer { } /// Returns the device address for this subbuffer. - pub fn device_address(&self) -> Result> { + pub fn device_address(&self) -> Result> { self.buffer().device_address().map(|ptr| { // SAFETY: The original address came from the Vulkan implementation, and allocation // sizes are guaranteed to not exceed `DeviceLayout::MAX_SIZE`, so the offset better be // in range. - unsafe { NonZeroDeviceSize::new_unchecked(ptr.get() + self.offset) } + unsafe { NonNullDeviceAddress::new_unchecked(ptr.get() + self.offset) } }) } #[cfg_attr(not(feature = "document_unchecked"), doc(hidden))] - pub unsafe fn device_address_unchecked(&self) -> NonZeroDeviceSize { + pub unsafe fn device_address_unchecked(&self) -> NonNullDeviceAddress { // SAFETY: The original address came from the Vulkan implementation, and allocation // sizes are guaranteed to not exceed `DeviceLayout::MAX_SIZE`, so the offset better be // in range. - NonZeroDeviceSize::new_unchecked( + NonNullDeviceAddress::new_unchecked( self.buffer().device_address_unchecked().get() + self.offset, ) } diff --git a/vulkano/src/lib.rs b/vulkano/src/lib.rs index 328392cca8..3444529618 100644 --- a/vulkano/src/lib.rs +++ b/vulkano/src/lib.rs @@ -206,6 +206,12 @@ pub use ash::vk::DeviceSize; /// A [`DeviceSize`] that is known not to equal zero. pub type NonZeroDeviceSize = NonZeroU64; +/// Represents an address (pointer) on a Vulkan device. +pub use ash::vk::DeviceAddress; + +/// A [`DeviceAddress`] that is known not to equal zero. +pub type NonNullDeviceAddress = NonZeroU64; + /// Holds 24 bits in the least significant bits of memory, /// and 8 bytes in the most significant bits of that memory, /// occupying a single [`u32`] in total.