From b938770d1345b18821ae136dd95aa8d7f9c6c182 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 19:06:07 +0100 Subject: [PATCH 01/13] Rename backends: web -> webgpu, direct -> wgpu_core --- wgpu/src/backend/mod.rs | 8 ++++---- wgpu/src/backend/{web.rs => webgpu.rs} | 0 wgpu/src/backend/{direct.rs => wgpu_core.rs} | 0 3 files changed, 4 insertions(+), 4 deletions(-) rename wgpu/src/backend/{web.rs => webgpu.rs} (100%) rename wgpu/src/backend/{direct.rs => wgpu_core.rs} (100%) diff --git a/wgpu/src/backend/mod.rs b/wgpu/src/backend/mod.rs index 9b8157e3ba..c20c87a06e 100644 --- a/wgpu/src/backend/mod.rs +++ b/wgpu/src/backend/mod.rs @@ -1,9 +1,9 @@ #[cfg(webgpu)] -mod web; +mod webgpu; #[cfg(webgpu)] -pub(crate) use web::Context; +pub(crate) use webgpu::Context; #[cfg(not(webgpu))] -mod direct; +mod wgpu_core; #[cfg(not(webgpu))] -pub(crate) use direct::Context; +pub(crate) use wgpu_core::Context; diff --git a/wgpu/src/backend/web.rs b/wgpu/src/backend/webgpu.rs similarity index 100% rename from wgpu/src/backend/web.rs rename to wgpu/src/backend/webgpu.rs diff --git a/wgpu/src/backend/direct.rs b/wgpu/src/backend/wgpu_core.rs similarity index 100% rename from wgpu/src/backend/direct.rs rename to wgpu/src/backend/wgpu_core.rs From 545dc0624e8a63c3cf8e3ba1c46d8e317ed4b09a Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 19:17:40 +0100 Subject: [PATCH 02/13] rename context objects for web & core --- wgpu/src/backend/mod.rs | 4 +-- wgpu/src/backend/webgpu.rs | 54 +++++++++++++++++++---------------- wgpu/src/backend/wgpu_core.rs | 14 +++++---- 3 files changed, 39 insertions(+), 33 deletions(-) diff --git a/wgpu/src/backend/mod.rs b/wgpu/src/backend/mod.rs index c20c87a06e..15288b3c93 100644 --- a/wgpu/src/backend/mod.rs +++ b/wgpu/src/backend/mod.rs @@ -1,9 +1,9 @@ #[cfg(webgpu)] mod webgpu; #[cfg(webgpu)] -pub(crate) use webgpu::Context; +pub(crate) type Context = webgpu::ContextWebGpu; #[cfg(not(webgpu))] mod wgpu_core; #[cfg(not(webgpu))] -pub(crate) use wgpu_core::Context; +pub(crate) type Context = wgpu_core::ContextWgpuCore; diff --git a/wgpu/src/backend/webgpu.rs b/wgpu/src/backend/webgpu.rs index 705d33dcca..3ac901caea 100644 --- a/wgpu/src/backend/webgpu.rs +++ b/wgpu/src/backend/webgpu.rs @@ -69,19 +69,21 @@ unsafe impl Send for Identified {} #[cfg(send_sync)] unsafe impl Sync for Identified {} -pub(crate) struct Context(web_sys::Gpu); +pub(crate) struct ContextWebGpu(web_sys::Gpu); #[cfg(send_sync)] -unsafe impl Send for Context {} +unsafe impl Send for ContextWebGpu {} #[cfg(send_sync)] -unsafe impl Sync for Context {} +unsafe impl Sync for ContextWebGpu {} #[cfg(send_sync)] unsafe impl Send for BufferMappedRange {} #[cfg(send_sync)] unsafe impl Sync for BufferMappedRange {} -impl fmt::Debug for Context { +impl fmt::Debug for ContextWebGpu { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - f.debug_struct("Context").field("type", &"Web").finish() + f.debug_struct("ContextWebGpu") + .field("type", &"Web") + .finish() } } @@ -540,7 +542,8 @@ fn map_texture_view_dimension( } fn map_buffer_copy_view(view: crate::ImageCopyBuffer<'_>) -> web_sys::GpuImageCopyBuffer { - let buffer: &::BufferData = downcast_ref(view.buffer.data.as_ref()); + let buffer: &::BufferData = + downcast_ref(view.buffer.data.as_ref()); let mut mapped = web_sys::GpuImageCopyBuffer::new(&buffer.0.buffer); if let Some(bytes_per_row) = view.layout.bytes_per_row { mapped.bytes_per_row(bytes_per_row); @@ -553,7 +556,7 @@ fn map_buffer_copy_view(view: crate::ImageCopyBuffer<'_>) -> web_sys::GpuImageCo } fn map_texture_copy_view(view: crate::ImageCopyTexture<'_>) -> web_sys::GpuImageCopyTexture { - let texture: &::TextureData = + let texture: &::TextureData = downcast_ref(view.texture.data.as_ref()); let mut mapped = web_sys::GpuImageCopyTexture::new(&texture.0); mapped.mip_level(view.mip_level); @@ -564,7 +567,7 @@ fn map_texture_copy_view(view: crate::ImageCopyTexture<'_>) -> web_sys::GpuImage fn map_tagged_texture_copy_view( view: crate::ImageCopyTextureTagged<'_>, ) -> web_sys::GpuImageCopyTextureTagged { - let texture: &::TextureData = + let texture: &::TextureData = downcast_ref(view.texture.data.as_ref()); let mut mapped = web_sys::GpuImageCopyTextureTagged::new(&texture.0); mapped.mip_level(view.mip_level); @@ -877,7 +880,7 @@ where *rc_callback.borrow_mut() = Some((closure_success, closure_rejected, callback)); } -impl Context { +impl ContextWebGpu { /// Common portion of the internal branches of the public `instance_create_surface` function. /// /// Note: Analogous code also exists in the WebGL2 backend at @@ -952,7 +955,7 @@ pub enum Canvas { Offscreen(web_sys::OffscreenCanvas), } -impl crate::context::Context for Context { +impl crate::context::Context for ContextWebGpu { type AdapterId = Identified; type AdapterData = Sendable; type DeviceId = Identified; @@ -1035,7 +1038,7 @@ impl crate::context::Context for Context { "Accessing the GPU is only supported on the main thread or from a dedicated worker" ); }; - Context(gpu) + ContextWebGpu(gpu) } unsafe fn instance_create_surface( @@ -1587,7 +1590,7 @@ impl crate::context::Context for Context { offset, size, }) => { - let buffer: &::BufferData = + let buffer: &::BufferData = downcast_ref(buffer.data.as_ref()); let mut mapped_buffer_binding = web_sys::GpuBufferBinding::new(&buffer.0.buffer); @@ -1601,7 +1604,7 @@ impl crate::context::Context for Context { panic!("Web backend does not support arrays of buffers") } crate::BindingResource::Sampler(sampler) => { - let sampler: &::SamplerData = + let sampler: &::SamplerData = downcast_ref(sampler.data.as_ref()); JsValue::from(&sampler.0) } @@ -1609,7 +1612,7 @@ impl crate::context::Context for Context { panic!("Web backend does not support arrays of samplers") } crate::BindingResource::TextureView(texture_view) => { - let texture_view: &::TextureViewData = + let texture_view: &::TextureViewData = downcast_ref(texture_view.data.as_ref()); JsValue::from(&texture_view.0) } @@ -1622,7 +1625,7 @@ impl crate::context::Context for Context { }) .collect::(); - let bgl: &::BindGroupLayoutData = + let bgl: &::BindGroupLayoutData = downcast_ref(desc.layout.data.as_ref()); let mut mapped_desc = web_sys::GpuBindGroupDescriptor::new(&mapped_entries, &bgl.0); if let Some(label) = desc.label { @@ -1641,7 +1644,7 @@ impl crate::context::Context for Context { .bind_group_layouts .iter() .map(|bgl| { - let bgl: &::BindGroupLayoutData = + let bgl: &::BindGroupLayoutData = downcast_ref(bgl.data.as_ref()); &bgl.0 }) @@ -1659,7 +1662,7 @@ impl crate::context::Context for Context { device_data: &Self::DeviceData, desc: &crate::RenderPipelineDescriptor<'_>, ) -> (Self::RenderPipelineId, Self::RenderPipelineData) { - let module: &::ShaderModuleData = + let module: &::ShaderModuleData = downcast_ref(desc.vertex.module.data.as_ref()); let mut mapped_vertex_state = web_sys::GpuVertexState::new(desc.vertex.entry_point, &module.0); @@ -1696,7 +1699,7 @@ impl crate::context::Context for Context { let mut mapped_desc = web_sys::GpuRenderPipelineDescriptor::new( &match desc.layout { Some(layout) => { - let layout: &::PipelineLayoutData = + let layout: &::PipelineLayoutData = downcast_ref(layout.data.as_ref()); JsValue::from(&layout.0) } @@ -1734,7 +1737,7 @@ impl crate::context::Context for Context { None => wasm_bindgen::JsValue::null(), }) .collect::(); - let module: &::ShaderModuleData = + let module: &::ShaderModuleData = downcast_ref(frag.module.data.as_ref()); let mapped_fragment_desc = web_sys::GpuFragmentState::new(frag.entry_point, &module.0, &targets); @@ -1759,7 +1762,7 @@ impl crate::context::Context for Context { device_data: &Self::DeviceData, desc: &crate::ComputePipelineDescriptor<'_>, ) -> (Self::ComputePipelineId, Self::ComputePipelineData) { - let shader_module: &::ShaderModuleData = + let shader_module: &::ShaderModuleData = downcast_ref(desc.module.data.as_ref()); let mapped_compute_stage = web_sys::GpuProgrammableStage::new(desc.entry_point, &shader_module.0); @@ -1767,7 +1770,7 @@ impl crate::context::Context for Context { let mut mapped_desc = web_sys::GpuComputePipelineDescriptor::new( &match desc.layout { Some(layout) => { - let layout: &::PipelineLayoutData = + let layout: &::PipelineLayoutData = downcast_ref(layout.data.as_ref()); JsValue::from(&layout.0) } @@ -2322,7 +2325,7 @@ impl crate::context::Context for Context { crate::LoadOp::Load => web_sys::GpuLoadOp::Load, }; - let view: &::TextureViewData = + let view: &::TextureViewData = downcast_ref(ca.view.data.as_ref()); let mut mapped_color_attachment = web_sys::GpuRenderPassColorAttachment::new( @@ -2334,7 +2337,7 @@ impl crate::context::Context for Context { mapped_color_attachment.clear_value(&cv); } if let Some(rt) = ca.resolve_target { - let resolve_target_view: &::TextureViewData = + let resolve_target_view: &::TextureViewData = downcast_ref(rt.data.as_ref()); mapped_color_attachment.resolve_target(&resolve_target_view.0); } @@ -2353,7 +2356,7 @@ impl crate::context::Context for Context { } if let Some(dsa) = &desc.depth_stencil_attachment { - let depth_stencil_attachment: &::TextureViewData = + let depth_stencil_attachment: &::TextureViewData = downcast_ref(dsa.view.data.as_ref()); let mut mapped_depth_stencil_attachment = web_sys::GpuRenderPassDepthStencilAttachment::new(&depth_stencil_attachment.0); @@ -2430,7 +2433,8 @@ impl crate::context::Context for Context { offset: wgt::BufferAddress, size: Option, ) { - let buffer: &::BufferData = downcast_ref(buffer.data.as_ref()); + let buffer: &::BufferData = + downcast_ref(buffer.data.as_ref()); match size { Some(size) => encoder_data.0.clear_buffer_with_f64_and_f64( &buffer.0.buffer, diff --git a/wgpu/src/backend/wgpu_core.rs b/wgpu/src/backend/wgpu_core.rs index 05a0db4263..2fec399cd3 100644 --- a/wgpu/src/backend/wgpu_core.rs +++ b/wgpu/src/backend/wgpu_core.rs @@ -31,21 +31,23 @@ use wgt::WasmNotSendSync; const LABEL: &str = "label"; -pub struct Context(wgc::global::Global); +pub struct ContextWgpuCore(wgc::global::Global); -impl Drop for Context { +impl Drop for ContextWgpuCore { fn drop(&mut self) { //nothing } } -impl fmt::Debug for Context { +impl fmt::Debug for ContextWgpuCore { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - f.debug_struct("Context").field("type", &"Native").finish() + f.debug_struct("ContextWgpuCore") + .field("type", &"Native") + .finish() } } -impl Context { +impl ContextWgpuCore { pub unsafe fn from_hal_instance(hal_instance: A::Instance) -> Self { Self(unsafe { wgc::global::Global::from_hal_instance::( @@ -445,7 +447,7 @@ pub struct CommandEncoder { open: bool, } -impl crate::Context for Context { +impl crate::Context for ContextWgpuCore { type AdapterId = wgc::id::AdapterId; type AdapterData = (); type DeviceId = wgc::id::DeviceId; From 28cdd0849d23846ea24e4ce76049adaebc0da025 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 22:06:52 +0100 Subject: [PATCH 03/13] allow webgpu & webgl features side-by-side --- wgpu/Cargo.toml | 6 +- wgpu/build.rs | 3 +- wgpu/src/backend/mod.rs | 8 +- wgpu/src/backend/wgpu_core.rs | 13 +++ wgpu/src/lib.rs | 211 ++++++++++++++++++++-------------- 5 files changed, 147 insertions(+), 94 deletions(-) diff --git a/wgpu/Cargo.toml b/wgpu/Cargo.toml index 2a83aa364c..8848371b9d 100644 --- a/wgpu/Cargo.toml +++ b/wgpu/Cargo.toml @@ -24,7 +24,7 @@ targets = [ [lib] [features] -default = ["wgsl", "dx12", "metal"] +default = ["wgsl", "dx12", "metal", "webgpu"] #! ### Backends # -------------------------------------------------------------------- @@ -44,10 +44,12 @@ angle = ["wgc?/gles"] ## Enables the Vulkan backend on macOS & iOS. vulkan-portability = ["wgc?/vulkan"] +## Enables the WebGPU backend on Wasm. Disabled when targeting `emscripten`. +webgpu = [] + ## Enables the GLES backend on Wasm ## ## * ⚠️ WIP: Currently will also enable GLES dependencies on any other targets. -## * ⚠️ WIP: This automatically disables use of WebGPU. See [#2804](https://github.com/gfx-rs/wgpu/issues/3514). webgl = ["hal", "wgc/gles"] #! ### Shading language support diff --git a/wgpu/build.rs b/wgpu/build.rs index b419deee24..bdf3568018 100644 --- a/wgpu/build.rs +++ b/wgpu/build.rs @@ -2,7 +2,8 @@ fn main() { cfg_aliases::cfg_aliases! { native: { not(target_arch = "wasm32") }, webgl: { all(target_arch = "wasm32", not(target_os = "emscripten"), feature = "webgl") }, - webgpu: { all(target_arch = "wasm32", not(target_os = "emscripten"), not(feature = "webgl")) }, + webgpu: { all(target_arch = "wasm32", not(target_os = "emscripten"), feature = "webgpu") }, + wgpu_core: { any(native, webgl) }, emscripten: { all(target_arch = "wasm32", target_os = "emscripten") }, send_sync: { any( not(target_arch = "wasm32"), diff --git a/wgpu/src/backend/mod.rs b/wgpu/src/backend/mod.rs index 15288b3c93..306b3387d6 100644 --- a/wgpu/src/backend/mod.rs +++ b/wgpu/src/backend/mod.rs @@ -1,9 +1,9 @@ #[cfg(webgpu)] mod webgpu; #[cfg(webgpu)] -pub(crate) type Context = webgpu::ContextWebGpu; +pub(crate) use webgpu::ContextWebGpu; -#[cfg(not(webgpu))] +#[cfg(wgpu_core)] mod wgpu_core; -#[cfg(not(webgpu))] -pub(crate) type Context = wgpu_core::ContextWgpuCore; +#[cfg(wgpu_core)] +pub(crate) use wgpu_core::ContextWgpuCore; diff --git a/wgpu/src/backend/wgpu_core.rs b/wgpu/src/backend/wgpu_core.rs index 2fec399cd3..560642502e 100644 --- a/wgpu/src/backend/wgpu_core.rs +++ b/wgpu/src/backend/wgpu_core.rs @@ -1539,6 +1539,19 @@ impl crate::Context for ContextWgpuCore { } } + #[cfg(webgl)] + fn buffer_get_mapped_range_as_array_buffer( + &self, + buffer: &Self::BufferId, + buffer_data: &Self::BufferData, + sub_range: Range, + ) -> js_sys::ArrayBuffer { + let mapped_range = self.buffer_get_mapped_range(buffer, buffer_data, sub_range); + let array_buffer = js_sys::Uint8Array::new_with_length(mapped_range.slice().len() as u32); + array_buffer.copy_from(mapped_range.slice()); + array_buffer.buffer() + } + fn buffer_unmap(&self, buffer: &Self::BufferId, buffer_data: &Self::BufferData) { let global = &self.0; match wgc::gfx_select!(buffer => global.buffer_unmap(*buffer)) { diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index 0c65a43100..57105b963d 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -16,11 +16,10 @@ //! - **`angle`** --- Enables the GLES backend via [ANGLE](https://github.com/google/angle) on macOS //! using. //! - **`vulkan-portability`** --- Enables the Vulkan backend on macOS & iOS. +//! - **`webgpu`** --- Enables the WebGPU backend on Wasm. Disabled when targeting `emscripten`. //! - **`webgl`** --- Enables the GLES backend on Wasm //! //! - ⚠️ WIP: Currently will also enable GLES dependencies on any other targets. -//! - ⚠️ WIP: This automatically disables use of WebGPU. See -//! [#2804](https://github.com/gfx-rs/wgpu/issues/3514). //! //! ### Shading language support //! @@ -1748,6 +1747,7 @@ impl Instance { /// /// If no backend feature for the active target platform is enabled, /// this method will panic, see [`Instance::any_backend_feature_enabled()`]. + #[allow(unreachable_code)] pub fn new(instance_desc: InstanceDescriptor) -> Self { if !Self::any_backend_feature_enabled() { panic!( @@ -1756,9 +1756,25 @@ impl Instance { ); } - Self { - context: Arc::from(crate::backend::Context::init(instance_desc)), + #[cfg(webgpu)] + { + // TODO: If both webgpu and webgl are enabled, check if browser supports WebGPU and fall through + // to wgpucore otherwise. + return Self { + context: Arc::from(crate::backend::ContextWebGpu::init(instance_desc)), + }; + } + + #[cfg(wgpu_core)] + { + return Self { + context: Arc::from(crate::backend::ContextWgpuCore::init(instance_desc)), + }; } + + unreachable!( + "Earlier check of `any_backend_feature_enabled` should have prevented getting here!" + ); } /// Create an new instance of wgpu from a wgpu-hal instance. @@ -1770,11 +1786,11 @@ impl Instance { /// # Safety /// /// Refer to the creation of wgpu-hal Instance for every backend. - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn from_hal(hal_instance: A::Instance) -> Self { Self { context: Arc::new(unsafe { - crate::backend::Context::from_hal_instance::(hal_instance) + crate::backend::ContextWgpuCore::from_hal_instance::(hal_instance) }), } } @@ -1789,15 +1805,13 @@ impl Instance { /// - The raw instance handle returned must not be manually destroyed. /// /// [`Instance`]: hal::Api::Instance - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn as_hal(&self) -> Option<&A::Instance> { - unsafe { - self.context - .as_any() - .downcast_ref::() - .unwrap() - .instance_as_hal::() - } + self.context + .as_any() + // If we don't have a wgpu-core instance, we don't have a hal instance either. + .downcast_ref::() + .and_then(|ctx| unsafe { ctx.instance_as_hal::() }) } /// Create an new instance of wgpu from a wgpu-core instance. @@ -1809,34 +1823,40 @@ impl Instance { /// # Safety /// /// Refer to the creation of wgpu-core Instance. - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn from_core(core_instance: wgc::instance::Instance) -> Self { Self { context: Arc::new(unsafe { - crate::backend::Context::from_core_instance(core_instance) + crate::backend::ContextWgpuCore::from_core_instance(core_instance) }), } } /// Retrieves all available [`Adapter`]s that match the given [`Backends`]. /// + /// Always returns an empty iterator if the instance decided upon creation to + /// target WebGPU since adapter creation is always async on WebGPU. + /// /// # Arguments /// /// - `backends` - Backends from which to enumerate adapters. - #[cfg(not(webgpu))] - pub fn enumerate_adapters(&self, backends: Backends) -> impl ExactSizeIterator { + #[cfg(wgpu_core)] + pub fn enumerate_adapters(&self, backends: Backends) -> Vec { let context = Arc::clone(&self.context); self.context .as_any() - .downcast_ref::() - .unwrap() - .enumerate_adapters(backends) - .into_iter() - .map(move |id| crate::Adapter { - context: Arc::clone(&context), - id: ObjectId::from(id), - data: Box::new(()), + .downcast_ref::() + .map(|ctx| { + ctx.enumerate_adapters(backends) + .into_iter() + .map(move |id| crate::Adapter { + context: Arc::clone(&context), + id: ObjectId::from(id), + data: Box::new(()), + }) + .collect() }) + .unwrap_or_default() } /// Retrieves an [`Adapter`] which matches the given [`RequestAdapterOptions`]. @@ -1862,7 +1882,7 @@ impl Instance { /// # Safety /// /// `hal_adapter` must be created from this instance internal handle. - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn create_adapter_from_hal( &self, hal_adapter: hal::ExposedAdapter, @@ -1871,7 +1891,7 @@ impl Instance { let id = unsafe { context .as_any() - .downcast_ref::() + .downcast_ref::() .unwrap() .create_adapter_from_hal(hal_adapter) .into() @@ -1999,13 +2019,15 @@ impl Instance { } /// Generates memory report. - #[cfg(not(webgpu))] - pub fn generate_report(&self) -> wgc::global::GlobalReport { + /// + /// Returns `None` if the feature is not supported by the backend + /// which happens only when WebGPU is pre-selected by the instance creation. + #[cfg(wgpu_core)] + pub fn generate_report(&self) -> Option { self.context .as_any() - .downcast_ref::() - .unwrap() - .generate_report() + .downcast_ref::() + .map(|ctx| ctx.generate_report()) } } @@ -2070,7 +2092,7 @@ impl Adapter { /// /// - `hal_device` must be created from this adapter internal handle. /// - `desc.features` must be a subset of `hal_device` features. - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn create_device_from_hal( &self, hal_device: hal::OpenDevice, @@ -2081,7 +2103,9 @@ impl Adapter { unsafe { self.context .as_any() - .downcast_ref::() + .downcast_ref::() + // Part of the safety requirements is that the device was generated from the same adapter. + // Therefore, unwrap is fine here since only WgpuCoreContext based adapters have the ability to create hal devices. .unwrap() .create_device_from_hal(&self.id.into(), hal_device, desc, trace_path) } @@ -2120,17 +2144,19 @@ impl Adapter { /// - The raw handle passed to the callback must not be manually destroyed. /// /// [`A::Adapter`]: hal::Api::Adapter - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn as_hal) -> R, R>( &self, hal_adapter_callback: F, ) -> R { - unsafe { - self.context - .as_any() - .downcast_ref::() - .unwrap() - .adapter_as_hal::(self.id.into(), hal_adapter_callback) + if let Some(ctx) = self + .context + .as_any() + .downcast_ref::() + { + unsafe { ctx.adapter_as_hal::(self.id.into(), hal_adapter_callback) } + } else { + hal_adapter_callback(None) } } @@ -2462,7 +2488,7 @@ impl Device { /// - `hal_texture` must be created from this device internal handle /// - `hal_texture` must be created respecting `desc` /// - `hal_texture` must be initialized - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn create_texture_from_hal( &self, hal_texture: A::Texture, @@ -2471,7 +2497,9 @@ impl Device { let texture = unsafe { self.context .as_any() - .downcast_ref::() + .downcast_ref::() + // Part of the safety requirements is that the texture was generated from the same hal device. + // Therefore, unwrap is fine here since only WgpuCoreContext has the ability to create hal textures. .unwrap() .create_texture_from_hal::( hal_texture, @@ -2499,7 +2527,7 @@ impl Device { /// - `hal_buffer` must be created from this device internal handle /// - `hal_buffer` must be created respecting `desc` /// - `hal_buffer` must be initialized - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn create_buffer_from_hal( &self, hal_buffer: A::Buffer, @@ -2513,7 +2541,9 @@ impl Device { let (id, buffer) = unsafe { self.context .as_any() - .downcast_ref::() + .downcast_ref::() + // Part of the safety requirements is that the buffer was generated from the same hal device. + // Therefore, unwrap is fine here since only WgpuCoreContext has the ability to create hal buffers. .unwrap() .create_buffer_from_hal::( hal_buffer, @@ -2603,21 +2633,20 @@ impl Device { /// - The raw handle passed to the callback must not be manually destroyed. /// /// [`A::Device`]: hal::Api::Device - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn as_hal) -> R, R>( &self, hal_device_callback: F, - ) -> R { - unsafe { - self.context - .as_any() - .downcast_ref::() - .unwrap() - .device_as_hal::( + ) -> Option { + self.context + .as_any() + .downcast_ref::() + .map(|ctx| unsafe { + ctx.device_as_hal::( self.data.as_ref().downcast_ref().unwrap(), hal_device_callback, ) - } + }) } /// Destroy this device. @@ -2656,8 +2685,8 @@ pub struct RequestDeviceError { enum RequestDeviceErrorKind { /// Error from [`wgpu_core`]. // must match dependency cfg - #[cfg(not(webgpu))] - Core(core::instance::RequestDeviceError), + #[cfg(wgpu_core)] + Core(wgc::instance::RequestDeviceError), /// Error from web API that was called by `wgpu` to request a device. /// @@ -2677,9 +2706,9 @@ static_assertions::assert_impl_all!(RequestDeviceError: Send, Sync); impl fmt::Display for RequestDeviceError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match &self.inner { - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] RequestDeviceErrorKind::Core(error) => error.fmt(f), - #[cfg(webgpu)] + #[cfg(all(target_arch = "wasm32", not(target_os = "emscripten")))] RequestDeviceErrorKind::Web(error_js_value) => { // wasm-bindgen provides a reasonable error stringification via `Debug` impl write!(f, "{error_js_value:?}") @@ -2691,7 +2720,7 @@ impl fmt::Display for RequestDeviceError { impl error::Error for RequestDeviceError { fn source(&self) -> Option<&(dyn error::Error + 'static)> { match &self.inner { - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] RequestDeviceErrorKind::Core(error) => error.source(), #[cfg(webgpu)] RequestDeviceErrorKind::Web(_) => None, @@ -2699,9 +2728,9 @@ impl error::Error for RequestDeviceError { } } -#[cfg(not(webgpu))] -impl From for RequestDeviceError { - fn from(error: core::instance::RequestDeviceError) -> Self { +#[cfg(wgpu_core)] +impl From for RequestDeviceError { + fn from(error: wgc::instance::RequestDeviceError) -> Self { Self { inner: RequestDeviceErrorKind::Core(error), } @@ -2717,7 +2746,7 @@ pub struct CreateSurfaceError { #[derive(Clone, Debug)] enum CreateSurfaceErrorKind { /// Error from [`wgpu_hal`]. - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] Hal(hal::InstanceError), /// Error from WebGPU surface creation. @@ -2733,7 +2762,7 @@ static_assertions::assert_impl_all!(CreateSurfaceError: Send, Sync); impl fmt::Display for CreateSurfaceError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match &self.inner { - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] CreateSurfaceErrorKind::Hal(e) => e.fmt(f), CreateSurfaceErrorKind::Web(e) => e.fmt(f), CreateSurfaceErrorKind::RawHandle(e) => e.fmt(f), @@ -2744,7 +2773,7 @@ impl fmt::Display for CreateSurfaceError { impl error::Error for CreateSurfaceError { fn source(&self) -> Option<&(dyn error::Error + 'static)> { match &self.inner { - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] CreateSurfaceErrorKind::Hal(e) => e.source(), CreateSurfaceErrorKind::Web(_) => None, CreateSurfaceErrorKind::RawHandle(e) => e.source(), @@ -2752,7 +2781,7 @@ impl error::Error for CreateSurfaceError { } } -#[cfg(not(webgpu))] +#[cfg(wgpu_core)] impl From for CreateSurfaceError { fn from(e: hal::InstanceError) -> Self { Self { @@ -2987,10 +3016,13 @@ impl<'a> BufferSlice<'a> { /// Synchronously and immediately map a buffer for reading. If the buffer is not immediately mappable /// through [`BufferDescriptor::mapped_at_creation`] or [`BufferSlice::map_async`], will panic. /// - /// This is useful in wasm builds when you want to pass mapped data directly to js. Unlike `get_mapped_range` - /// which unconditionally copies mapped data into the wasm heap, this function directly hands you the - /// ArrayBuffer that we mapped the data into in js. - #[cfg(webgpu)] + /// This is useful when targeting WebGPU and you want to pass mapped data directly to js. + /// Unlike `get_mapped_range` which unconditionally copies mapped data into the wasm heap, + /// this function directly hands you the ArrayBuffer that we mapped the data into in js. + /// + /// If you are not targeting WebGPU, this function will perform an additional copy from the wasm heap to + /// the returned array buffer. + #[cfg(all(not(native), not(target_os = "emscripten")))] pub fn get_mapped_range_as_array_buffer(&self) -> js_sys::ArrayBuffer { let end = self.buffer.map_context.lock().add(self.offset, self.size); DynContext::buffer_get_mapped_range_as_array_buffer( @@ -3034,18 +3066,21 @@ impl Texture { /// # Safety /// /// - The raw handle obtained from the hal Texture must not be manually destroyed - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn as_hal)>( &self, hal_texture_callback: F, ) { let texture = self.data.as_ref().downcast_ref().unwrap(); - unsafe { - self.context - .as_any() - .downcast_ref::() - .unwrap() - .texture_as_hal::(texture, hal_texture_callback) + + if let Some(ctx) = self + .context + .as_any() + .downcast_ref::() + { + unsafe { ctx.texture_as_hal::(texture, hal_texture_callback) } + } else { + hal_texture_callback(None) } } @@ -4782,18 +4817,20 @@ impl Surface<'_> { /// # Safety /// /// - The raw handle obtained from the hal Surface must not be manually destroyed - #[cfg(not(webgpu))] + #[cfg(wgpu_core)] pub unsafe fn as_hal) -> R, R>( &mut self, hal_surface_callback: F, - ) -> R { - unsafe { - self.context - .as_any() - .downcast_ref::() - .unwrap() - .surface_as_hal::(self.data.downcast_ref().unwrap(), hal_surface_callback) - } + ) -> Option { + self.context + .as_any() + .downcast_ref::() + .map(|ctx| unsafe { + ctx.surface_as_hal::( + self.data.downcast_ref().unwrap(), + hal_surface_callback, + ) + }) } } From 49a48b60123e986c678f4c603d365704c66f4c97 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 22:14:38 +0100 Subject: [PATCH 04/13] make sure webgl ci doesn't use webgpu --- .github/workflows/ci.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 81d79f25e7..a11d33ba87 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -354,7 +354,7 @@ jobs: - name: execute tests run: | cd wgpu - wasm-pack test --headless --chrome --features webgl --workspace + wasm-pack test --headless --chrome --no-default-features --features wgsl,webgl --workspace gpu-test: strategy: From 4a9948091906c5ef53d667063dbe297bca282b69 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 22:21:01 +0100 Subject: [PATCH 05/13] update any_backend_feature_enabled --- README.md | 2 +- wgpu/src/lib.rs | 5 +++-- xtask/src/run_wasm.rs | 2 +- 3 files changed, 5 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index ae3edf774b..4dbcd243ef 100644 --- a/README.md +++ b/README.md @@ -174,7 +174,7 @@ To run the test suite on WebGL (currently incomplete): ``` cd wgpu -wasm-pack test --headless --chrome --features webgl --workspace +wasm-pack test --headless --chrome --no-default-features --features webgl --workspace ``` This will automatically run the tests using a packaged browser. Remove `--headless` to run the tests with whatever browser you wish at `http://localhost:8000`. diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index 57105b963d..90eb4079e4 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -1720,8 +1720,6 @@ impl Instance { /// See /// * Windows: always enables Vulkan and GLES with no way to opt out /// * Linux: always enables Vulkan and GLES with no way to opt out - /// * Web: either targets WebGPU backend or, if `webgl` enabled, WebGL - /// * TODO: Support both WebGPU and WebGL at the same time, see pub const fn any_backend_feature_enabled() -> bool { // Method intentionally kept verbose to keep it a bit easier to follow! @@ -1731,6 +1729,9 @@ impl Instance { cfg!(feature = "metal") || cfg!(feature = "vulkan-portability") || cfg!(feature = "angle") + // On the web, either WebGPU or WebGL must be enabled. + } else if cfg!(target_arch = "wasm32") { + cfg!(feature = "webgpu") || cfg!(feature = "webgl") } else { true } diff --git a/xtask/src/run_wasm.rs b/xtask/src/run_wasm.rs index d53d33a2c1..280ef82775 100644 --- a/xtask/src/run_wasm.rs +++ b/xtask/src/run_wasm.rs @@ -61,7 +61,7 @@ pub(crate) fn run_wasm(mut args: Arguments) -> Result<(), anyhow::Error> { xshell::cmd!( shell, - "cargo build --target wasm32-unknown-unknown --bin wgpu-examples --features webgl {release_flag...}" + "cargo build --target wasm32-unknown-unknown --bin wgpu-examples --no-default-features --features wgsl,webgl {release_flag...}" ) .args(&cargo_args) .quiet() From a1a004b475de6fe928d91f64056cc02b01623a80 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 22:22:42 +0100 Subject: [PATCH 06/13] add panicing generate_report method for compatibility --- wgpu/src/lib.rs | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index 90eb4079e4..bbe8aee479 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -2019,12 +2019,21 @@ impl Instance { self.context.instance_poll_all_devices(force_wait) } + /// Generates memory report. + /// + /// Like `try_generate_report` but panics if the active backend is WebGPU. + // TODO: We should instead create an empty report that indicates that this is WebGPU. + #[cfg(wgpu_core)] + pub fn generate_report(&self) -> wgc::global::GlobalReport { + self.try_generate_report().unwrap() + } + /// Generates memory report. /// /// Returns `None` if the feature is not supported by the backend /// which happens only when WebGPU is pre-selected by the instance creation. #[cfg(wgpu_core)] - pub fn generate_report(&self) -> Option { + pub fn try_generate_report(&self) -> Option { self.context .as_any() .downcast_ref::() From 81cb332a987de64d17dfd48ae8125ac06fdd06fb Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 10 Jan 2024 22:38:10 +0100 Subject: [PATCH 07/13] RequestDeviceErrorKind::Web rename, fixup various cfg attributes --- wgpu/src/backend/webgpu.rs | 2 +- wgpu/src/context.rs | 6 +++--- wgpu/src/lib.rs | 8 ++++---- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/wgpu/src/backend/webgpu.rs b/wgpu/src/backend/webgpu.rs index 3ac901caea..ee7cce5a71 100644 --- a/wgpu/src/backend/webgpu.rs +++ b/wgpu/src/backend/webgpu.rs @@ -824,7 +824,7 @@ fn future_request_device( (device_id, device_data, queue_id, queue_data) }) .map_err(|error_value| crate::RequestDeviceError { - inner: crate::RequestDeviceErrorKind::Web(error_value), + inner: crate::RequestDeviceErrorKind::WebGpu(error_value), }) } diff --git a/wgpu/src/context.rs b/wgpu/src/context.rs index 2221921623..de92ea9de2 100644 --- a/wgpu/src/context.rs +++ b/wgpu/src/context.rs @@ -320,7 +320,7 @@ pub trait Context: Debug + WasmNotSendSync + Sized { buffer_data: &Self::BufferData, sub_range: Range, ) -> Box; - #[cfg(webgpu)] + #[cfg(any(webgpu, webgl))] fn buffer_get_mapped_range_as_array_buffer( &self, buffer: &Self::BufferId, @@ -1342,7 +1342,7 @@ pub(crate) trait DynContext: Debug + WasmNotSendSync { buffer_data: &crate::Data, sub_range: Range, ) -> Box; - #[cfg(webgpu)] + #[cfg(any(webgpu, webgl))] fn buffer_get_mapped_range_as_array_buffer( &self, buffer: &ObjectId, @@ -2461,7 +2461,7 @@ where Context::buffer_get_mapped_range(self, &buffer, buffer_data, sub_range) } - #[cfg(webgpu)] + #[cfg(any(webgpu, webgl))] fn buffer_get_mapped_range_as_array_buffer( &self, buffer: &ObjectId, diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index bbe8aee479..3caef59397 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -2702,7 +2702,7 @@ enum RequestDeviceErrorKind { /// /// (This is currently never used by the webgl backend, but it could be.) #[cfg(webgpu)] - Web(wasm_bindgen::JsValue), + WebGpu(wasm_bindgen::JsValue), } #[cfg(send_sync)] @@ -2718,8 +2718,8 @@ impl fmt::Display for RequestDeviceError { match &self.inner { #[cfg(wgpu_core)] RequestDeviceErrorKind::Core(error) => error.fmt(f), - #[cfg(all(target_arch = "wasm32", not(target_os = "emscripten")))] - RequestDeviceErrorKind::Web(error_js_value) => { + #[cfg(webgpu)] + RequestDeviceErrorKind::WebGpu(error_js_value) => { // wasm-bindgen provides a reasonable error stringification via `Debug` impl write!(f, "{error_js_value:?}") } @@ -2733,7 +2733,7 @@ impl error::Error for RequestDeviceError { #[cfg(wgpu_core)] RequestDeviceErrorKind::Core(error) => error.source(), #[cfg(webgpu)] - RequestDeviceErrorKind::Web(_) => None, + RequestDeviceErrorKind::WebGpu(_) => None, } } } From 36d0006a4561bdfca8dbe510b9fd9c763820866f Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Thu, 11 Jan 2024 14:47:13 +0100 Subject: [PATCH 08/13] automatic webgpu support detection --- tests/src/init.rs | 12 +++++++++++- wgpu-types/src/lib.rs | 7 ++++++- wgpu/src/backend/mod.rs | 2 +- wgpu/src/backend/webgpu.rs | 34 +++++++++++++++++++++++++--------- wgpu/src/lib.rs | 13 +++++++++++-- 5 files changed, 54 insertions(+), 14 deletions(-) diff --git a/tests/src/init.rs b/tests/src/init.rs index 6a24f5faec..9a21c98471 100644 --- a/tests/src/init.rs +++ b/tests/src/init.rs @@ -16,7 +16,17 @@ pub fn initialize_instance() -> Instance { // // We can potentially work support back into the test runner in the future, but as the adapters are matched up // based on adapter index, removing some backends messes up the indexes in annoying ways. - let backends = Backends::all(); + // + // WORKAROUND for https://github.com/rust-lang/cargo/issues/7160: + // `--no-default-features` is not passed through correctly to the test runner. + // We use it whenever we want to explicitly run with webgl instead of webgpu. + // To "disable" webgpu regardless, we do this by removing the webgpu backend whenever we see + // the webgl feature. + let backends = if cfg!(feature = "webgl") { + Backends::all() - Backends::BROWSER_WEBGPU + } else { + Backends::all() + }; let dx12_shader_compiler = wgpu::util::dx12_shader_compiler_from_env().unwrap_or_default(); let gles_minor_version = wgpu::util::gles_minor_version_from_env().unwrap_or_default(); Instance::new(wgpu::InstanceDescriptor { diff --git a/wgpu-types/src/lib.rs b/wgpu-types/src/lib.rs index 43c0f465ee..ef34d32444 100644 --- a/wgpu-types/src/lib.rs +++ b/wgpu-types/src/lib.rs @@ -155,7 +155,12 @@ bitflags::bitflags! { const METAL = 1 << Backend::Metal as u32; /// Supported on Windows 10 const DX12 = 1 << Backend::Dx12 as u32; - /// Supported when targeting the web through webassembly + /// Supported when targeting the web through webassembly with the `webgpu` feature enabled. + /// + /// The WebGPU backend is special in several ways: + /// It is not not implemented by `wgpu_core` and instead by the higher level `wgpu` crate. + /// Whether WebGPU is targeted is decided upon the creation of the `wgpu::Instance`, + /// *not* upon adapter creation. See `wgpu::Instance::new`. const BROWSER_WEBGPU = 1 << Backend::BrowserWebGpu as u32; /// All the apis that wgpu offers first tier of support for. /// diff --git a/wgpu/src/backend/mod.rs b/wgpu/src/backend/mod.rs index 306b3387d6..02d9632efb 100644 --- a/wgpu/src/backend/mod.rs +++ b/wgpu/src/backend/mod.rs @@ -1,7 +1,7 @@ #[cfg(webgpu)] mod webgpu; #[cfg(webgpu)] -pub(crate) use webgpu::ContextWebGpu; +pub(crate) use webgpu::{get_browser_gpu_property, ContextWebGpu}; #[cfg(wgpu_core)] mod wgpu_core; diff --git a/wgpu/src/backend/webgpu.rs b/wgpu/src/backend/webgpu.rs index ee7cce5a71..04f2e12329 100644 --- a/wgpu/src/backend/webgpu.rs +++ b/wgpu/src/backend/webgpu.rs @@ -955,6 +955,30 @@ pub enum Canvas { Offscreen(web_sys::OffscreenCanvas), } +/// Returns the browsers gpu object or `None` if the current context is neither the main thread nor a dedicated worker. +/// +/// If WebGPU is not supported, the Gpu property is `undefined` (but *not* necessarily `None`). +/// +/// See: +/// * +/// * +pub fn get_browser_gpu_property() -> Option { + let global: Global = js_sys::global().unchecked_into(); + + if !global.window().is_undefined() { + Some(global.unchecked_into::().navigator().gpu()) + } else if !global.worker().is_undefined() { + Some( + global + .unchecked_into::() + .navigator() + .gpu(), + ) + } else { + None + } +} + impl crate::context::Context for ContextWebGpu { type AdapterId = Identified; type AdapterData = Sendable; @@ -1025,15 +1049,7 @@ impl crate::context::Context for ContextWebGpu { MakeSendFuture Option>; fn init(_instance_desc: wgt::InstanceDescriptor) -> Self { - let global: Global = js_sys::global().unchecked_into(); - let gpu = if !global.window().is_undefined() { - global.unchecked_into::().navigator().gpu() - } else if !global.worker().is_undefined() { - global - .unchecked_into::() - .navigator() - .gpu() - } else { + let Some(gpu) = get_browser_gpu_property() else { panic!( "Accessing the GPU is only supported on the main thread or from a dedicated worker" ); diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index 3caef59397..70b6d3b17c 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -1744,6 +1744,15 @@ impl Instance { /// - `instance_desc` - Has fields for which [backends][Backends] wgpu will choose /// during instantiation, and which [DX12 shader compiler][Dx12Compiler] wgpu will use. /// + /// [`Backends::BROWSER_WEBGPU`] takes a special role: + /// If it is set and WebGPU support is detected, this instance will *only* be able to create + /// WebGPU adapters. If you instead want to force use of WebGL, either + /// disable the `webgpu` compile-time feature or do add the [`Backends::BROWSER_WEBGPU`] + /// flag to the the `instance_desc`'s `backends` field. + /// If it is set and WebGPU support is *not* detected, the instance will use wgpu-core + /// to create adapters. Meaning that if the `webgl` feature is enabled, it is able to create + /// a WebGL adapter. + /// /// # Panics /// /// If no backend feature for the active target platform is enabled, @@ -1758,9 +1767,9 @@ impl Instance { } #[cfg(webgpu)] + if instance_desc.backends.contains(Backends::BROWSER_WEBGPU) + && crate::backend::get_browser_gpu_property().map_or(false, |gpu| !gpu.is_undefined()) { - // TODO: If both webgpu and webgl are enabled, check if browser supports WebGPU and fall through - // to wgpucore otherwise. return Self { context: Arc::from(crate::backend::ContextWebGpu::init(instance_desc)), }; From 08e6bbaab7b42572fb70fedd3aea2348609ee128 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Thu, 11 Jan 2024 16:35:29 +0100 Subject: [PATCH 09/13] changelog entry --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index d7b360b5dd..46606fc272 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -67,6 +67,15 @@ Conversion to `wgpu::SurfaceTarget` is automatic for anything implementing `raw- meaning that you can continue to e.g. pass references to winit windows as before. By @wumpf in [#4984](https://github.com/gfx-rs/wgpu/pull/4984) +### WebGPU & WebGL in the same binary + +Enabling `webgl` no longer removes the `webgpu` backend. +Instead, there's a new (default enabled) `webgpu` feature that allows to explicitly opt-out of `webgpu` if so desired. +If both `webgl` & `webgpu` are enabled, `wgpu::Instance` decides upon creation whether to target wgpu-core/WebGL or WebGPU. +This means that adapter selection is not handled as with regular adapters, but still allows to decide at runtime whether +`webgpu` or the `webgl` backend should be used using a single wasm binary. +By @wumpf in [#5044](https://github.com/gfx-rs/wgpu/pull/5044) + ### New Features #### General From 8147cfe8ef88dea240123d246de7a75e838cb2e8 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Thu, 11 Jan 2024 16:37:14 +0100 Subject: [PATCH 10/13] fix emscripten --- wgpu/build.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wgpu/build.rs b/wgpu/build.rs index bdf3568018..a263c6559c 100644 --- a/wgpu/build.rs +++ b/wgpu/build.rs @@ -3,8 +3,8 @@ fn main() { native: { not(target_arch = "wasm32") }, webgl: { all(target_arch = "wasm32", not(target_os = "emscripten"), feature = "webgl") }, webgpu: { all(target_arch = "wasm32", not(target_os = "emscripten"), feature = "webgpu") }, - wgpu_core: { any(native, webgl) }, emscripten: { all(target_arch = "wasm32", target_os = "emscripten") }, + wgpu_core: { any(native, webgl, emscripten) }, send_sync: { any( not(target_arch = "wasm32"), all(feature = "fragile-send-sync-non-atomic-wasm", not(target_feature = "atomics")) From de14d1b18c0b453efd572f10a0831e23e5ee5d48 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Fri, 12 Jan 2024 09:24:31 +0100 Subject: [PATCH 11/13] fix weird cfg, fix comment typo --- wgpu/src/lib.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index 70b6d3b17c..ccd193571b 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -1844,7 +1844,7 @@ impl Instance { /// Retrieves all available [`Adapter`]s that match the given [`Backends`]. /// - /// Always returns an empty iterator if the instance decided upon creation to + /// Always returns an empty vector if the instance decided upon creation to /// target WebGPU since adapter creation is always async on WebGPU. /// /// # Arguments @@ -3041,7 +3041,7 @@ impl<'a> BufferSlice<'a> { /// /// If you are not targeting WebGPU, this function will perform an additional copy from the wasm heap to /// the returned array buffer. - #[cfg(all(not(native), not(target_os = "emscripten")))] + #[cfg(any(webgpu, webgl))] pub fn get_mapped_range_as_array_buffer(&self) -> js_sys::ArrayBuffer { let end = self.buffer.map_context.lock().add(self.offset, self.size); DynContext::buffer_get_mapped_range_as_array_buffer( From 11fc92c7199b713c1325d748bf26be2bc21f3686 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Fri, 12 Jan 2024 09:26:51 +0100 Subject: [PATCH 12/13] remove try_generate_report again --- tests/tests/device.rs | 4 ++-- tests/tests/mem_leaks.rs | 32 ++++++++++++++++---------------- wgpu/src/lib.rs | 11 +---------- 3 files changed, 19 insertions(+), 28 deletions(-) diff --git a/tests/tests/device.rs b/tests/tests/device.rs index 52438670e6..33d2d37484 100644 --- a/tests/tests/device.rs +++ b/tests/tests/device.rs @@ -50,11 +50,11 @@ fn device_lifetime_check() { instance.poll_all(false); - let pre_report = instance.generate_report(); + let pre_report = instance.generate_report().unwrap().unwrap(); drop(queue); drop(device); - let post_report = instance.generate_report(); + let post_report = instance.generate_report().unwrap().unwrap(); assert_ne!( pre_report, post_report, "Queue and Device has not been dropped as expected" diff --git a/tests/tests/mem_leaks.rs b/tests/tests/mem_leaks.rs index f8dc4ab9d4..fc81ce86ee 100644 --- a/tests/tests/mem_leaks.rs +++ b/tests/tests/mem_leaks.rs @@ -12,7 +12,7 @@ fn draw_test_with_reports( use wgpu::util::DeviceExt; - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.devices.num_allocated, 1); assert_eq!(report.queues.num_allocated, 1); @@ -21,7 +21,7 @@ fn draw_test_with_reports( .device .create_shader_module(wgpu::include_wgsl!("./vertex_indices/draw.vert.wgsl")); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.shader_modules.num_allocated, 1); @@ -41,7 +41,7 @@ fn draw_test_with_reports( }], }); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 0); assert_eq!(report.bind_groups.num_allocated, 0); @@ -54,7 +54,7 @@ fn draw_test_with_reports( mapped_at_creation: false, }); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); @@ -67,7 +67,7 @@ fn draw_test_with_reports( }], }); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); assert_eq!(report.bind_groups.num_allocated, 1); @@ -81,7 +81,7 @@ fn draw_test_with_reports( push_constant_ranges: &[], }); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); assert_eq!(report.pipeline_layouts.num_allocated, 1); @@ -113,7 +113,7 @@ fn draw_test_with_reports( multiview: None, }); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); assert_eq!(report.bind_groups.num_allocated, 1); @@ -125,7 +125,7 @@ fn draw_test_with_reports( drop(shader); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.shader_modules.num_allocated, 1); assert_eq!(report.shader_modules.num_kept_from_user, 0); @@ -153,7 +153,7 @@ fn draw_test_with_reports( ); let texture_view = texture.create_view(&wgpu::TextureViewDescriptor::default()); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); assert_eq!(report.texture_views.num_allocated, 1); @@ -161,7 +161,7 @@ fn draw_test_with_reports( drop(texture); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); assert_eq!(report.texture_views.num_allocated, 1); @@ -173,7 +173,7 @@ fn draw_test_with_reports( .device .create_command_encoder(&wgpu::CommandEncoderDescriptor::default()); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.command_buffers.num_allocated, 1); assert_eq!(report.buffers.num_allocated, 1); @@ -193,7 +193,7 @@ fn draw_test_with_reports( rpass.set_pipeline(&pipeline); rpass.set_bind_group(0, &bg, &[]); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.buffers.num_allocated, 1); assert_eq!(report.bind_groups.num_allocated, 1); @@ -216,7 +216,7 @@ fn draw_test_with_reports( drop(bg); drop(buffer); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.command_buffers.num_kept_from_user, 1); assert_eq!(report.render_pipelines.num_kept_from_user, 0); @@ -237,7 +237,7 @@ fn draw_test_with_reports( let submit_index = ctx.queue.submit(Some(encoder.finish())); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.command_buffers.num_allocated, 0); @@ -248,7 +248,7 @@ fn draw_test_with_reports( // Call poll twice to ensure all destroyable resources are destroyed. ctx.device.poll(wgpu::Maintain::Poll); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.render_pipelines.num_allocated, 0); @@ -263,7 +263,7 @@ fn draw_test_with_reports( drop(ctx.device); drop(ctx.adapter); - let global_report = ctx.instance.generate_report(); + let global_report = ctx.instance.generate_report().unwrap(); let report = global_report.hub_report(ctx.adapter_info.backend); assert_eq!(report.queues.num_kept_from_user, 0); diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index ccd193571b..e0922d7c76 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -2028,21 +2028,12 @@ impl Instance { self.context.instance_poll_all_devices(force_wait) } - /// Generates memory report. - /// - /// Like `try_generate_report` but panics if the active backend is WebGPU. - // TODO: We should instead create an empty report that indicates that this is WebGPU. - #[cfg(wgpu_core)] - pub fn generate_report(&self) -> wgc::global::GlobalReport { - self.try_generate_report().unwrap() - } - /// Generates memory report. /// /// Returns `None` if the feature is not supported by the backend /// which happens only when WebGPU is pre-selected by the instance creation. #[cfg(wgpu_core)] - pub fn try_generate_report(&self) -> Option { + pub fn generate_report(&self) -> Option { self.context .as_any() .downcast_ref::() From 6c105754e92da2997aeb15d461815aeaec94f2da Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Sat, 13 Jan 2024 11:22:29 +0100 Subject: [PATCH 13/13] Make get_mapped_range_as_array_buffer WebGPU only again --- wgpu/src/backend/webgpu.rs | 18 +++++++++--------- wgpu/src/backend/wgpu_core.rs | 13 ------------- wgpu/src/context.rs | 26 -------------------------- wgpu/src/lib.rs | 25 +++++++++++++------------ 4 files changed, 22 insertions(+), 60 deletions(-) diff --git a/wgpu/src/backend/webgpu.rs b/wgpu/src/backend/webgpu.rs index 04f2e12329..019e41482c 100644 --- a/wgpu/src/backend/webgpu.rs +++ b/wgpu/src/backend/webgpu.rs @@ -932,6 +932,15 @@ impl ContextWebGpu { Ok(create_identified((canvas, context))) } + + /// Get mapped buffer range directly as a `js_sys::ArrayBuffer`. + pub fn buffer_get_mapped_range_as_array_buffer( + &self, + buffer_data: &::BufferData, + sub_range: Range, + ) -> js_sys::ArrayBuffer { + buffer_data.0.get_mapped_array_buffer(sub_range) + } } // Represents the global object in the JavaScript context. @@ -2048,15 +2057,6 @@ impl crate::context::Context for ContextWebGpu { }) } - fn buffer_get_mapped_range_as_array_buffer( - &self, - _buffer: &Self::BufferId, - buffer_data: &Self::BufferData, - sub_range: Range, - ) -> js_sys::ArrayBuffer { - buffer_data.0.get_mapped_array_buffer(sub_range) - } - fn buffer_unmap(&self, _buffer: &Self::BufferId, buffer_data: &Self::BufferData) { buffer_data.0.buffer.unmap(); buffer_data.0.mapping.borrow_mut().mapped_buffer = None; diff --git a/wgpu/src/backend/wgpu_core.rs b/wgpu/src/backend/wgpu_core.rs index 560642502e..2fec399cd3 100644 --- a/wgpu/src/backend/wgpu_core.rs +++ b/wgpu/src/backend/wgpu_core.rs @@ -1539,19 +1539,6 @@ impl crate::Context for ContextWgpuCore { } } - #[cfg(webgl)] - fn buffer_get_mapped_range_as_array_buffer( - &self, - buffer: &Self::BufferId, - buffer_data: &Self::BufferData, - sub_range: Range, - ) -> js_sys::ArrayBuffer { - let mapped_range = self.buffer_get_mapped_range(buffer, buffer_data, sub_range); - let array_buffer = js_sys::Uint8Array::new_with_length(mapped_range.slice().len() as u32); - array_buffer.copy_from(mapped_range.slice()); - array_buffer.buffer() - } - fn buffer_unmap(&self, buffer: &Self::BufferId, buffer_data: &Self::BufferData) { let global = &self.0; match wgc::gfx_select!(buffer => global.buffer_unmap(*buffer)) { diff --git a/wgpu/src/context.rs b/wgpu/src/context.rs index de92ea9de2..12881fa12b 100644 --- a/wgpu/src/context.rs +++ b/wgpu/src/context.rs @@ -320,13 +320,6 @@ pub trait Context: Debug + WasmNotSendSync + Sized { buffer_data: &Self::BufferData, sub_range: Range, ) -> Box; - #[cfg(any(webgpu, webgl))] - fn buffer_get_mapped_range_as_array_buffer( - &self, - buffer: &Self::BufferId, - buffer_data: &Self::BufferData, - sub_range: Range, - ) -> js_sys::ArrayBuffer; fn buffer_unmap(&self, buffer: &Self::BufferId, buffer_data: &Self::BufferData); fn texture_create_view( &self, @@ -1342,13 +1335,6 @@ pub(crate) trait DynContext: Debug + WasmNotSendSync { buffer_data: &crate::Data, sub_range: Range, ) -> Box; - #[cfg(any(webgpu, webgl))] - fn buffer_get_mapped_range_as_array_buffer( - &self, - buffer: &ObjectId, - buffer_data: &crate::Data, - sub_range: Range, - ) -> js_sys::ArrayBuffer; fn buffer_unmap(&self, buffer: &ObjectId, buffer_data: &crate::Data); fn texture_create_view( &self, @@ -2461,18 +2447,6 @@ where Context::buffer_get_mapped_range(self, &buffer, buffer_data, sub_range) } - #[cfg(any(webgpu, webgl))] - fn buffer_get_mapped_range_as_array_buffer( - &self, - buffer: &ObjectId, - buffer_data: &crate::Data, - sub_range: Range, - ) -> js_sys::ArrayBuffer { - let buffer = ::from(*buffer); - let buffer_data = downcast_ref(buffer_data); - Context::buffer_get_mapped_range_as_array_buffer(self, &buffer, buffer_data, sub_range) - } - fn buffer_unmap(&self, buffer: &ObjectId, buffer_data: &crate::Data) { let buffer = ::from(*buffer); let buffer_data = downcast_ref(buffer_data); diff --git a/wgpu/src/lib.rs b/wgpu/src/lib.rs index e0922d7c76..c56a71295c 100644 --- a/wgpu/src/lib.rs +++ b/wgpu/src/lib.rs @@ -3024,23 +3024,24 @@ impl<'a> BufferSlice<'a> { } /// Synchronously and immediately map a buffer for reading. If the buffer is not immediately mappable - /// through [`BufferDescriptor::mapped_at_creation`] or [`BufferSlice::map_async`], will panic. + /// through [`BufferDescriptor::mapped_at_creation`] or [`BufferSlice::map_async`], will fail. /// /// This is useful when targeting WebGPU and you want to pass mapped data directly to js. /// Unlike `get_mapped_range` which unconditionally copies mapped data into the wasm heap, /// this function directly hands you the ArrayBuffer that we mapped the data into in js. /// - /// If you are not targeting WebGPU, this function will perform an additional copy from the wasm heap to - /// the returned array buffer. - #[cfg(any(webgpu, webgl))] - pub fn get_mapped_range_as_array_buffer(&self) -> js_sys::ArrayBuffer { - let end = self.buffer.map_context.lock().add(self.offset, self.size); - DynContext::buffer_get_mapped_range_as_array_buffer( - &*self.buffer.context, - &self.buffer.id, - self.buffer.data.as_ref(), - self.offset..end, - ) + /// This is only available on WebGPU, on any other backends this will return `None`. + #[cfg(webgpu)] + pub fn get_mapped_range_as_array_buffer(&self) -> Option { + self.buffer + .context + .as_any() + .downcast_ref::() + .map(|ctx| { + let buffer_data = crate::context::downcast_ref(self.buffer.data.as_ref()); + let end = self.buffer.map_context.lock().add(self.offset, self.size); + ctx.buffer_get_mapped_range_as_array_buffer(buffer_data, self.offset..end) + }) } /// Synchronously and immediately map a buffer for writing. If the buffer is not immediately mappable