From 48078a800a2f1de70772641feee100b0506aaeaf Mon Sep 17 00:00:00 2001 From: Teodor Tanasoaia <28601907+teoxoy@users.noreply.github.com> Date: Mon, 31 Jul 2023 15:30:47 +0200 Subject: [PATCH] Derive storage bindings via `naga::StorageAccess` instead of `naga::GlobalUse` (#3985) Removes the feature check for read-only and read-write storage textures as it's later checked by `create_bind_group_layout`. https://github.com/gfx-rs/wgpu/blob/335f40f85a492a5d9a69eda6835a30173f2e6b03/wgpu-core/src/device/resource.rs#L1505-L1518 Also removes the `GlobalUse` checks from `check_binding_use` as naga is already checking those. https://github.com/gfx-rs/naga/blob/535701f6b2f38f7e0088701ee4cf6d8024f74a21/src/valid/interface.rs#L639-L669 --- CHANGELOG.md | 4 + wgpu-core/src/device/resource.rs | 3 +- wgpu-core/src/validation.rs | 172 ++++++++++--------------------- 3 files changed, 61 insertions(+), 118 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 72f0c83fcf..72aa2022c2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -46,6 +46,10 @@ Bottom level categories: ### Bug Fixes +#### General + +- Derive storage bindings via `naga::StorageAccess` instead of `naga::GlobalUse`. By @teoxoy in [#3985](https://github.com/gfx-rs/wgpu/pull/3985). + #### Vulkan - Fix enabling `wgpu::Features::PARTIALLY_BOUND_BINDING_ARRAY` not being actually enabled in vulkan backend. By @39ali in[#3772](https://github.com/gfx-rs/wgpu/pull/3772). diff --git a/wgpu-core/src/device/resource.rs b/wgpu-core/src/device/resource.rs index 20e057a934..074170e562 100644 --- a/wgpu-core/src/device/resource.rs +++ b/wgpu-core/src/device/resource.rs @@ -1285,8 +1285,7 @@ impl Device { inner: Box::new(inner), }) })?; - let interface = - validation::Interface::new(&module, &info, self.features, self.limits.clone()); + let interface = validation::Interface::new(&module, &info, self.limits.clone()); let hal_shader = hal::ShaderInput::Naga(hal::NagaShader { module, info }); let hal_desc = hal::ShaderModuleDescriptor { diff --git a/wgpu-core/src/validation.rs b/wgpu-core/src/validation.rs index 83cb90530a..84e1e71691 100644 --- a/wgpu-core/src/validation.rs +++ b/wgpu-core/src/validation.rs @@ -1,5 +1,4 @@ use crate::{binding_model::BindEntryMap, FastHashMap, FastHashSet}; -use naga::valid::GlobalUse; use std::{collections::hash_map::Entry, fmt}; use thiserror::Error; use wgt::{BindGroupLayoutEntry, BindingType}; @@ -112,7 +111,7 @@ struct SpecializationConstant { struct EntryPoint { inputs: Vec, outputs: Vec, - resources: Vec<(naga::Handle, GlobalUse)>, + resources: Vec>, #[allow(unused)] spec_constants: Vec, sampling_pairs: FastHashSet<(naga::Handle, naga::Handle)>, @@ -121,7 +120,6 @@ struct EntryPoint { #[derive(Debug)] pub struct Interface { - features: wgt::Features, limits: wgt::Limits, resources: naga::Arena, entry_points: FastHashMap<(naga::ShaderStage, String), EntryPoint>, @@ -174,11 +172,6 @@ pub enum BindingError { Missing, #[error("Visibility flags don't include the shader stage")] Invisible, - #[error("The shader requires the load/store access flags {required:?} but only {allowed:?} is allowed")] - WrongUsage { - required: GlobalUse, - allowed: GlobalUse, - }, #[error("Type on the shader side does not match the pipeline binding")] WrongType, #[error("Storage class {binding:?} doesn't match the shader {shader:?}")] @@ -206,9 +199,9 @@ pub enum BindingError { #[error("Texture format {0:?} is not supported for storage use")] BadStorageFormat(wgt::TextureFormat), #[error( - "Storage texture usage {0:?} doesn't have a matching supported `StorageTextureAccess`" + "Storage texture with access {0:?} doesn't have a matching supported `StorageTextureAccess`" )] - UnsupportedTextureStorageAccess(GlobalUse), + UnsupportedTextureStorageAccess(naga::StorageAccess), } #[derive(Clone, Debug, Error)] @@ -379,34 +372,23 @@ fn map_storage_format_from_naga(format: naga::StorageFormat) -> wgt::TextureForm } impl Resource { - fn check_binding_use( - &self, - entry: &BindGroupLayoutEntry, - shader_usage: GlobalUse, - ) -> Result<(), BindingError> { - let allowed_usage = match self.ty { + fn check_binding_use(&self, entry: &BindGroupLayoutEntry) -> Result<(), BindingError> { + match self.ty { ResourceType::Buffer { size } => { - let (allowed_usage, min_size) = match entry.ty { + let min_size = match entry.ty { BindingType::Buffer { ty, has_dynamic_offset: _, min_binding_size, } => { - let (class, global_use) = match ty { - wgt::BufferBindingType::Uniform => { - (naga::AddressSpace::Uniform, GlobalUse::READ) - } + let class = match ty { + wgt::BufferBindingType::Uniform => naga::AddressSpace::Uniform, wgt::BufferBindingType::Storage { read_only } => { - let mut global_use = GlobalUse::READ | GlobalUse::QUERY; - global_use.set(GlobalUse::WRITE, !read_only); let mut naga_access = naga::StorageAccess::LOAD; naga_access.set(naga::StorageAccess::STORE, !read_only); - ( - naga::AddressSpace::Storage { - access: naga_access, - }, - global_use, - ) + naga::AddressSpace::Storage { + access: naga_access, + } } }; if self.class != class { @@ -415,7 +397,7 @@ impl Resource { shader: self.class, }); } - (global_use, min_binding_size) + min_binding_size } _ => return Err(BindingError::WrongType), }; @@ -425,13 +407,10 @@ impl Resource { } _ => (), } - allowed_usage } ResourceType::Sampler { comparison } => match entry.ty { BindingType::Sampler(ty) => { - if (ty == wgt::SamplerBindingType::Comparison) == comparison { - GlobalUse::READ - } else { + if (ty == wgt::SamplerBindingType::Comparison) != comparison { return Err(BindingError::WrongSamplerComparison); } } @@ -480,29 +459,26 @@ impl Resource { } } } - let (expected_class, usage) = match entry.ty { + let expected_class = match entry.ty { BindingType::Texture { sample_type, view_dimension: _, multisampled: multi, - } => { - let class = match sample_type { - wgt::TextureSampleType::Float { .. } => naga::ImageClass::Sampled { - kind: naga::ScalarKind::Float, - multi, - }, - wgt::TextureSampleType::Sint => naga::ImageClass::Sampled { - kind: naga::ScalarKind::Sint, - multi, - }, - wgt::TextureSampleType::Uint => naga::ImageClass::Sampled { - kind: naga::ScalarKind::Uint, - multi, - }, - wgt::TextureSampleType::Depth => naga::ImageClass::Depth { multi }, - }; - (class, GlobalUse::READ | GlobalUse::QUERY) - } + } => match sample_type { + wgt::TextureSampleType::Float { .. } => naga::ImageClass::Sampled { + kind: naga::ScalarKind::Float, + multi, + }, + wgt::TextureSampleType::Sint => naga::ImageClass::Sampled { + kind: naga::ScalarKind::Sint, + multi, + }, + wgt::TextureSampleType::Uint => naga::ImageClass::Sampled { + kind: naga::ScalarKind::Uint, + multi, + }, + wgt::TextureSampleType::Depth => naga::ImageClass::Depth { multi }, + }, BindingType::StorageTexture { access, format, @@ -510,26 +486,15 @@ impl Resource { } => { let naga_format = map_storage_format_to_naga(format) .ok_or(BindingError::BadStorageFormat(format))?; - let (naga_access, usage) = match access { - wgt::StorageTextureAccess::ReadOnly => ( - naga::StorageAccess::LOAD, - GlobalUse::READ | GlobalUse::QUERY, - ), - wgt::StorageTextureAccess::WriteOnly => ( - naga::StorageAccess::STORE, - GlobalUse::WRITE | GlobalUse::QUERY, - ), - wgt::StorageTextureAccess::ReadWrite => { - (naga::StorageAccess::all(), GlobalUse::all()) - } + let naga_access = match access { + wgt::StorageTextureAccess::ReadOnly => naga::StorageAccess::LOAD, + wgt::StorageTextureAccess::WriteOnly => naga::StorageAccess::STORE, + wgt::StorageTextureAccess::ReadWrite => naga::StorageAccess::all(), }; - ( - naga::ImageClass::Storage { - format: naga_format, - access: naga_access, - }, - usage, - ) + naga::ImageClass::Storage { + format: naga_format, + access: naga_access, + } } _ => return Err(BindingError::WrongType), }; @@ -539,31 +504,19 @@ impl Resource { shader: class, }); } - usage } }; - if allowed_usage.contains(shader_usage) { - Ok(()) - } else { - Err(BindingError::WrongUsage { - required: shader_usage, - allowed: allowed_usage, - }) - } + Ok(()) } - fn derive_binding_type( - &self, - shader_usage: GlobalUse, - features: wgt::Features, - ) -> Result { + fn derive_binding_type(&self) -> Result { Ok(match self.ty { ResourceType::Buffer { size } => BindingType::Buffer { ty: match self.class { naga::AddressSpace::Uniform => wgt::BufferBindingType::Uniform, - naga::AddressSpace::Storage { .. } => wgt::BufferBindingType::Storage { - read_only: !shader_usage.contains(GlobalUse::WRITE), + naga::AddressSpace::Storage { access } => wgt::BufferBindingType::Storage { + read_only: access == naga::StorageAccess::LOAD, }, _ => return Err(BindingError::WrongType), }, @@ -606,19 +559,15 @@ impl Resource { view_dimension, multisampled: multi, }, - naga::ImageClass::Storage { format, .. } => BindingType::StorageTexture { - access: if !shader_usage.contains(GlobalUse::READ) { - wgt::StorageTextureAccess::WriteOnly - } else if !features - .contains(wgt::Features::TEXTURE_ADAPTER_SPECIFIC_FORMAT_FEATURES) - { - return Err(BindingError::UnsupportedTextureStorageAccess( - shader_usage, - )); - } else if shader_usage.contains(GlobalUse::WRITE) { - wgt::StorageTextureAccess::ReadWrite - } else { - wgt::StorageTextureAccess::ReadOnly + naga::ImageClass::Storage { format, access } => BindingType::StorageTexture { + access: { + const LOAD_STORE: naga::StorageAccess = naga::StorageAccess::all(); + match access { + naga::StorageAccess::LOAD => wgt::StorageTextureAccess::ReadOnly, + naga::StorageAccess::STORE => wgt::StorageTextureAccess::WriteOnly, + LOAD_STORE => wgt::StorageTextureAccess::ReadWrite, + _ => unreachable!(), + } }, view_dimension, format: { @@ -880,12 +829,7 @@ impl Interface { list.push(varying); } - pub fn new( - module: &naga::Module, - info: &naga::valid::ModuleInfo, - features: wgt::Features, - limits: wgt::Limits, - ) -> Self { + pub fn new(module: &naga::Module, info: &naga::valid::ModuleInfo, limits: wgt::Limits) -> Self { let mut resources = naga::Arena::new(); let mut resource_mapping = FastHashMap::default(); for (var_handle, var) in module.global_variables.iter() { @@ -949,11 +893,8 @@ impl Interface { for (var_handle, var) in module.global_variables.iter() { let usage = info[var_handle]; - if usage.is_empty() { - continue; - } - if var.binding.is_some() { - ep.resources.push((resource_mapping[&var_handle], usage)); + if !usage.is_empty() && var.binding.is_some() { + ep.resources.push(resource_mapping[&var_handle]); } } @@ -968,7 +909,6 @@ impl Interface { } Self { - features, limits, resources, entry_points, @@ -1000,7 +940,7 @@ impl Interface { .ok_or(StageError::MissingEntryPoint(pair.1))?; // check resources visibility - for &(handle, usage) in entry_point.resources.iter() { + for &handle in entry_point.resources.iter() { let res = &self.resources[handle]; let result = match given_layouts { Some(layouts) => { @@ -1026,13 +966,13 @@ impl Interface { Err(BindingError::Invisible) } }) - .and_then(|entry| res.check_binding_use(entry, usage)) + .and_then(|entry| res.check_binding_use(entry)) } None => derived_layouts .get_mut(res.bind.group as usize) .ok_or(BindingError::Missing) .and_then(|set| { - let ty = res.derive_binding_type(usage, self.features)?; + let ty = res.derive_binding_type()?; match set.entry(res.bind.binding) { Entry::Occupied(e) if e.get().ty != ty => { return Err(BindingError::InconsistentlyDerivedType)