Initial implementation of GPUQueue

Added WebIDL bindings for `GPUQueue`.
Implemented the `submit` function of `GPUQueue` and `defaultQueue` function of `GPUDevice`.
This commit is contained in:
Zakor 2020-02-03 17:25:01 +01:00 committed by Istvan Miklos
parent 0790c856d5
commit a3c6810b18
11 changed files with 181 additions and 16 deletions

View file

@ -154,7 +154,7 @@ use uuid::Uuid;
use webgpu::{ use webgpu::{
WebGPU, WebGPUAdapter, WebGPUBindGroup, WebGPUBindGroupLayout, WebGPUBuffer, WebGPU, WebGPUAdapter, WebGPUBindGroup, WebGPUBindGroupLayout, WebGPUBuffer,
WebGPUCommandBuffer, WebGPUCommandEncoder, WebGPUComputePipeline, WebGPUDevice, WebGPUCommandBuffer, WebGPUCommandEncoder, WebGPUComputePipeline, WebGPUDevice,
WebGPUPipelineLayout, WebGPUShaderModule, WebGPUPipelineLayout, WebGPUQueue, WebGPUShaderModule,
}; };
use webrender_api::{DocumentId, ImageKey}; use webrender_api::{DocumentId, ImageKey};
use webvr_traits::{WebVRGamepadData, WebVRGamepadHand, WebVRGamepadState}; use webvr_traits::{WebVRGamepadData, WebVRGamepadHand, WebVRGamepadState};
@ -537,6 +537,7 @@ unsafe_no_jsmanaged_fields!(WebGPUBindGroup);
unsafe_no_jsmanaged_fields!(WebGPUBindGroupLayout); unsafe_no_jsmanaged_fields!(WebGPUBindGroupLayout);
unsafe_no_jsmanaged_fields!(WebGPUComputePipeline); unsafe_no_jsmanaged_fields!(WebGPUComputePipeline);
unsafe_no_jsmanaged_fields!(WebGPUPipelineLayout); unsafe_no_jsmanaged_fields!(WebGPUPipelineLayout);
unsafe_no_jsmanaged_fields!(WebGPUQueue);
unsafe_no_jsmanaged_fields!(WebGPUShaderModule); unsafe_no_jsmanaged_fields!(WebGPUShaderModule);
unsafe_no_jsmanaged_fields!(WebGPUCommandBuffer); unsafe_no_jsmanaged_fields!(WebGPUCommandBuffer);
unsafe_no_jsmanaged_fields!(WebGPUCommandEncoder); unsafe_no_jsmanaged_fields!(WebGPUCommandEncoder);

View file

@ -107,7 +107,7 @@ impl GPUAdapterMethods for GPUAdapter {
impl AsyncWGPUListener for GPUAdapter { impl AsyncWGPUListener for GPUAdapter {
fn handle_response(&self, response: WebGPUResponse, promise: &Rc<Promise>) { fn handle_response(&self, response: WebGPUResponse, promise: &Rc<Promise>) {
match response { match response {
WebGPUResponse::RequestDevice(device_id, _descriptor) => { WebGPUResponse::RequestDevice(device_id, queue_id, _descriptor) => {
let device = GPUDevice::new( let device = GPUDevice::new(
&self.global(), &self.global(),
self.channel.clone(), self.channel.clone(),
@ -115,6 +115,7 @@ impl AsyncWGPUListener for GPUAdapter {
Heap::default(), Heap::default(),
Heap::default(), Heap::default(),
device_id, device_id,
queue_id,
); );
promise.resolve_native(&device); promise.resolve_native(&device);
}, },

View file

@ -2,7 +2,7 @@
* License, v. 2.0. If a copy of the MPL was not distributed with this * License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */ * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use crate::dom::bindings::cell::DomRefCell; use crate::dom::bindings::cell::{DomRefCell, Ref};
use crate::dom::bindings::codegen::Bindings::GPUBufferBinding::{ use crate::dom::bindings::codegen::Bindings::GPUBufferBinding::{
self, GPUBufferMethods, GPUBufferSize, self, GPUBufferMethods, GPUBufferSize,
}; };
@ -91,6 +91,10 @@ impl GPUBuffer {
pub fn usage(&self) -> u32 { pub fn usage(&self) -> u32 {
self.usage self.usage
} }
pub fn state(&self) -> Ref<GPUBufferState> {
self.state.borrow()
}
} }
impl Drop for GPUBuffer { impl Drop for GPUBuffer {
@ -106,6 +110,7 @@ impl GPUBufferMethods for GPUBuffer {
.0 .0
.send(WebGPURequest::UnmapBuffer(self.buffer)) .send(WebGPURequest::UnmapBuffer(self.buffer))
.unwrap(); .unwrap();
*self.state.borrow_mut() = GPUBufferState::Unmapped;
} }
/// https://gpuweb.github.io/gpuweb/#dom-gpubuffer-destroy /// https://gpuweb.github.io/gpuweb/#dom-gpubuffer-destroy

View file

@ -2,17 +2,28 @@
* License, v. 2.0. If a copy of the MPL was not distributed with this * License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */ * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use crate::dom::bindings::cell::DomRefCell; use crate::dom::bindings::cell::{DomRefCell, Ref};
use crate::dom::bindings::codegen::Bindings::GPUCommandBufferBinding::{ use crate::dom::bindings::codegen::Bindings::GPUCommandBufferBinding::{
self, GPUCommandBufferMethods, self, GPUCommandBufferMethods,
}; };
use crate::dom::bindings::reflector::{reflect_dom_object, Reflector}; use crate::dom::bindings::reflector::{reflect_dom_object, Reflector};
use crate::dom::bindings::root::Dom;
use crate::dom::bindings::root::DomRoot; use crate::dom::bindings::root::DomRoot;
use crate::dom::bindings::str::DOMString; use crate::dom::bindings::str::DOMString;
use crate::dom::globalscope::GlobalScope; use crate::dom::globalscope::GlobalScope;
use crate::dom::gpubuffer::GPUBuffer;
use dom_struct::dom_struct; use dom_struct::dom_struct;
use std::collections::HashSet;
use std::hash::{Hash, Hasher};
use webgpu::{WebGPU, WebGPUCommandBuffer}; use webgpu::{WebGPU, WebGPUCommandBuffer};
impl Eq for DomRoot<GPUBuffer> {}
impl Hash for DomRoot<GPUBuffer> {
fn hash<H: Hasher>(&self, state: &mut H) {
self.id().hash(state);
}
}
#[dom_struct] #[dom_struct]
pub struct GPUCommandBuffer { pub struct GPUCommandBuffer {
reflector_: Reflector, reflector_: Reflector,
@ -20,15 +31,21 @@ pub struct GPUCommandBuffer {
channel: WebGPU, channel: WebGPU,
label: DomRefCell<Option<DOMString>>, label: DomRefCell<Option<DOMString>>,
command_buffer: WebGPUCommandBuffer, command_buffer: WebGPUCommandBuffer,
buffers: DomRefCell<HashSet<Dom<GPUBuffer>>>,
} }
impl GPUCommandBuffer { impl GPUCommandBuffer {
pub fn new_inherited(channel: WebGPU, command_buffer: WebGPUCommandBuffer) -> GPUCommandBuffer { fn new_inherited(
channel: WebGPU,
command_buffer: WebGPUCommandBuffer,
buffers: HashSet<DomRoot<GPUBuffer>>,
) -> GPUCommandBuffer {
GPUCommandBuffer { GPUCommandBuffer {
channel, channel,
reflector_: Reflector::new(), reflector_: Reflector::new(),
label: DomRefCell::new(None), label: DomRefCell::new(None),
command_buffer, command_buffer,
buffers: DomRefCell::new(buffers.into_iter().map(|b| Dom::from_ref(&*b)).collect()),
} }
} }
@ -36,15 +53,30 @@ impl GPUCommandBuffer {
global: &GlobalScope, global: &GlobalScope,
channel: WebGPU, channel: WebGPU,
command_buffer: WebGPUCommandBuffer, command_buffer: WebGPUCommandBuffer,
buffers: HashSet<DomRoot<GPUBuffer>>,
) -> DomRoot<GPUCommandBuffer> { ) -> DomRoot<GPUCommandBuffer> {
reflect_dom_object( reflect_dom_object(
Box::new(GPUCommandBuffer::new_inherited(channel, command_buffer)), Box::new(GPUCommandBuffer::new_inherited(
channel,
command_buffer,
buffers,
)),
global, global,
GPUCommandBufferBinding::Wrap, GPUCommandBufferBinding::Wrap,
) )
} }
} }
impl GPUCommandBuffer {
pub fn id(&self) -> WebGPUCommandBuffer {
self.command_buffer
}
pub fn buffers(&self) -> Ref<HashSet<Dom<GPUBuffer>>> {
self.buffers.borrow()
}
}
impl GPUCommandBufferMethods for GPUCommandBuffer { impl GPUCommandBufferMethods for GPUCommandBuffer {
/// https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label /// https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label
fn GetLabel(&self) -> Option<DOMString> { fn GetLabel(&self) -> Option<DOMString> {

View file

@ -16,6 +16,7 @@ use crate::dom::gpucommandbuffer::GPUCommandBuffer;
use crate::dom::gpucomputepassencoder::GPUComputePassEncoder; use crate::dom::gpucomputepassencoder::GPUComputePassEncoder;
use dom_struct::dom_struct; use dom_struct::dom_struct;
use ipc_channel::ipc; use ipc_channel::ipc;
use std::collections::HashSet;
use webgpu::{wgpu::command::RawPass, WebGPU, WebGPUCommandEncoder, WebGPURequest}; use webgpu::{wgpu::command::RawPass, WebGPU, WebGPUCommandEncoder, WebGPURequest};
#[dom_struct] #[dom_struct]
@ -25,6 +26,7 @@ pub struct GPUCommandEncoder {
channel: WebGPU, channel: WebGPU,
label: DomRefCell<Option<DOMString>>, label: DomRefCell<Option<DOMString>>,
encoder: WebGPUCommandEncoder, encoder: WebGPUCommandEncoder,
buffers: DomRefCell<HashSet<DomRoot<GPUBuffer>>>,
} }
impl GPUCommandEncoder { impl GPUCommandEncoder {
@ -34,6 +36,7 @@ impl GPUCommandEncoder {
reflector_: Reflector::new(), reflector_: Reflector::new(),
label: DomRefCell::new(None), label: DomRefCell::new(None),
encoder, encoder,
buffers: DomRefCell::new(HashSet::new()),
} }
} }
@ -82,6 +85,10 @@ impl GPUCommandEncoderMethods for GPUCommandEncoder {
destination_offset: u64, destination_offset: u64,
size: u64, size: u64,
) { ) {
self.buffers.borrow_mut().insert(DomRoot::from_ref(source));
self.buffers
.borrow_mut()
.insert(DomRoot::from_ref(destination));
self.channel self.channel
.0 .0
.send(WebGPURequest::CopyBufferToBuffer( .send(WebGPURequest::CopyBufferToBuffer(
@ -106,9 +113,14 @@ impl GPUCommandEncoderMethods for GPUCommandEncoder {
// TODO(zakorgy): We should use `_descriptor` here after it's not empty // TODO(zakorgy): We should use `_descriptor` here after it's not empty
// and the underlying wgpu-core struct is serializable // and the underlying wgpu-core struct is serializable
)) ))
.expect("Failed to send CopyBufferToBuffer"); .expect("Failed to send Finish");
let buffer = receiver.recv().unwrap(); let buffer = receiver.recv().unwrap();
GPUCommandBuffer::new(&self.global(), self.channel.clone(), buffer) GPUCommandBuffer::new(
&self.global(),
self.channel.clone(),
buffer,
self.buffers.borrow_mut().drain().collect(),
)
} }
} }

View file

@ -31,6 +31,7 @@ use crate::dom::gpubuffer::{GPUBuffer, GPUBufferState};
use crate::dom::gpucommandencoder::GPUCommandEncoder; use crate::dom::gpucommandencoder::GPUCommandEncoder;
use crate::dom::gpucomputepipeline::GPUComputePipeline; use crate::dom::gpucomputepipeline::GPUComputePipeline;
use crate::dom::gpupipelinelayout::GPUPipelineLayout; use crate::dom::gpupipelinelayout::GPUPipelineLayout;
use crate::dom::gpuqueue::GPUQueue;
use crate::dom::gpushadermodule::GPUShaderModule; use crate::dom::gpushadermodule::GPUShaderModule;
use crate::script_runtime::JSContext as SafeJSContext; use crate::script_runtime::JSContext as SafeJSContext;
use dom_struct::dom_struct; use dom_struct::dom_struct;
@ -45,7 +46,7 @@ use webgpu::wgpu::binding_model::{
ShaderStage, ShaderStage,
}; };
use webgpu::wgpu::resource::{BufferDescriptor, BufferUsage}; use webgpu::wgpu::resource::{BufferDescriptor, BufferUsage};
use webgpu::{WebGPU, WebGPUBuffer, WebGPUDevice, WebGPURequest}; use webgpu::{WebGPU, WebGPUBuffer, WebGPUDevice, WebGPUQueue, WebGPURequest};
#[dom_struct] #[dom_struct]
pub struct GPUDevice { pub struct GPUDevice {
@ -59,6 +60,7 @@ pub struct GPUDevice {
limits: Heap<*mut JSObject>, limits: Heap<*mut JSObject>,
label: DomRefCell<Option<DOMString>>, label: DomRefCell<Option<DOMString>>,
device: WebGPUDevice, device: WebGPUDevice,
default_queue: Dom<GPUQueue>,
} }
impl GPUDevice { impl GPUDevice {
@ -68,6 +70,7 @@ impl GPUDevice {
extensions: Heap<*mut JSObject>, extensions: Heap<*mut JSObject>,
limits: Heap<*mut JSObject>, limits: Heap<*mut JSObject>,
device: WebGPUDevice, device: WebGPUDevice,
queue: &GPUQueue,
) -> GPUDevice { ) -> GPUDevice {
Self { Self {
eventtarget: EventTarget::new_inherited(), eventtarget: EventTarget::new_inherited(),
@ -77,6 +80,7 @@ impl GPUDevice {
limits, limits,
label: DomRefCell::new(None), label: DomRefCell::new(None),
device, device,
default_queue: Dom::from_ref(queue),
} }
} }
@ -88,10 +92,12 @@ impl GPUDevice {
extensions: Heap<*mut JSObject>, extensions: Heap<*mut JSObject>,
limits: Heap<*mut JSObject>, limits: Heap<*mut JSObject>,
device: WebGPUDevice, device: WebGPUDevice,
queue: WebGPUQueue,
) -> DomRoot<GPUDevice> { ) -> DomRoot<GPUDevice> {
let queue = GPUQueue::new(global, channel.clone(), queue);
reflect_dom_object( reflect_dom_object(
Box::new(GPUDevice::new_inherited( Box::new(GPUDevice::new_inherited(
channel, adapter, extensions, limits, device, channel, adapter, extensions, limits, device, &queue,
)), )),
global, global,
GPUDeviceBinding::Wrap, GPUDeviceBinding::Wrap,
@ -176,6 +182,11 @@ impl GPUDeviceMethods for GPUDevice {
NonNull::new(self.extensions.get()).unwrap() NonNull::new(self.extensions.get()).unwrap()
} }
/// https://gpuweb.github.io/gpuweb/#dom-gpudevice-defaultqueue
fn DefaultQueue(&self) -> DomRoot<GPUQueue> {
DomRoot::from_ref(&self.default_queue)
}
/// https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label /// https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label
fn GetLabel(&self) -> Option<DOMString> { fn GetLabel(&self) -> Option<DOMString> {
self.label.borrow().clone() self.label.borrow().clone()

View file

@ -0,0 +1,73 @@
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use crate::dom::bindings::cell::DomRefCell;
use crate::dom::bindings::codegen::Bindings::GPUQueueBinding::{self, GPUQueueMethods};
use crate::dom::bindings::reflector::{reflect_dom_object, Reflector};
use crate::dom::bindings::root::DomRoot;
use crate::dom::bindings::str::DOMString;
use crate::dom::globalscope::GlobalScope;
use crate::dom::gpubuffer::GPUBufferState;
use crate::dom::gpucommandbuffer::GPUCommandBuffer;
use dom_struct::dom_struct;
use webgpu::{WebGPU, WebGPUQueue, WebGPURequest};
#[dom_struct]
pub struct GPUQueue {
reflector_: Reflector,
#[ignore_malloc_size_of = "defined in webgpu"]
channel: WebGPU,
label: DomRefCell<Option<DOMString>>,
queue: WebGPUQueue,
}
impl GPUQueue {
fn new_inherited(channel: WebGPU, queue: WebGPUQueue) -> GPUQueue {
GPUQueue {
channel,
reflector_: Reflector::new(),
label: DomRefCell::new(None),
queue,
}
}
pub fn new(global: &GlobalScope, channel: WebGPU, queue: WebGPUQueue) -> DomRoot<GPUQueue> {
reflect_dom_object(
Box::new(GPUQueue::new_inherited(channel, queue)),
global,
GPUQueueBinding::Wrap,
)
}
}
impl GPUQueueMethods for GPUQueue {
/// https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label
fn GetLabel(&self) -> Option<DOMString> {
self.label.borrow().clone()
}
/// https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label
fn SetLabel(&self, value: Option<DOMString>) {
*self.label.borrow_mut() = value;
}
/// https://gpuweb.github.io/gpuweb/#dom-gpuqueue-submit
fn Submit(&self, command_buffers: Vec<DomRoot<GPUCommandBuffer>>) {
let valid = command_buffers.iter().all(|cb| {
cb.buffers().iter().all(|b| match *b.state() {
GPUBufferState::Unmapped => true,
_ => false,
})
});
if !valid {
// TODO: Generate error to the ErrorScope
return;
}
let buffer_ids = command_buffers.iter().map(|cb| cb.id().0).collect();
self.channel
.0
.send(WebGPURequest::Submit(self.queue.0, buffer_ids))
.unwrap();
}
}

View file

@ -328,6 +328,7 @@ pub mod gpucomputepassencoder;
pub mod gpucomputepipeline; pub mod gpucomputepipeline;
pub mod gpudevice; pub mod gpudevice;
pub mod gpupipelinelayout; pub mod gpupipelinelayout;
pub mod gpuqueue;
pub mod gpushadermodule; pub mod gpushadermodule;
pub mod gpushaderstage; pub mod gpushaderstage;
pub mod hashchangeevent; pub mod hashchangeevent;

View file

@ -5,13 +5,14 @@
// https://gpuweb.github.io/gpuweb/#gpudevice // https://gpuweb.github.io/gpuweb/#gpudevice
[Exposed=(Window, DedicatedWorker)/*, Serializable */, Pref="dom.webgpu.enabled"] [Exposed=(Window, DedicatedWorker)/*, Serializable */, Pref="dom.webgpu.enabled"]
interface GPUDevice : EventTarget { interface GPUDevice : EventTarget {
readonly attribute GPUAdapter adapter; /*[SameObject]*/ readonly attribute GPUAdapter adapter;
readonly attribute object extensions; readonly attribute object extensions;
readonly attribute object limits; readonly attribute object limits;
[SameObject] readonly attribute GPUQueue defaultQueue;
GPUBuffer createBuffer(GPUBufferDescriptor descriptor); GPUBuffer createBuffer(GPUBufferDescriptor descriptor);
GPUMappedBuffer createBufferMapped(GPUBufferDescriptor descriptor); GPUMappedBuffer createBufferMapped(GPUBufferDescriptor descriptor);
// Promise<GPUMappedBuffer> createBufferMappedAsync(GPUBufferDescriptor descriptor);
// GPUTexture createTexture(GPUTextureDescriptor descriptor); // GPUTexture createTexture(GPUTextureDescriptor descriptor);
// GPUSampler createSampler(optional GPUSamplerDescriptor descriptor = {}); // GPUSampler createSampler(optional GPUSamplerDescriptor descriptor = {});
@ -25,8 +26,6 @@ interface GPUDevice : EventTarget {
GPUCommandEncoder createCommandEncoder(optional GPUCommandEncoderDescriptor descriptor = {}); GPUCommandEncoder createCommandEncoder(optional GPUCommandEncoderDescriptor descriptor = {});
// GPURenderBundleEncoder createRenderBundleEncoder(GPURenderBundleEncoderDescriptor descriptor); // GPURenderBundleEncoder createRenderBundleEncoder(GPURenderBundleEncoderDescriptor descriptor);
// GPUQueue getQueue();
}; };
GPUDevice includes GPUObjectBase; GPUDevice includes GPUObjectBase;

View file

@ -0,0 +1,18 @@
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
// https://gpuweb.github.io/gpuweb/#gpuqueue
[Exposed=(Window, DedicatedWorker), Serializable, Pref="dom.webgpu.enabled"]
interface GPUQueue {
void submit(sequence<GPUCommandBuffer> commandBuffers);
// GPUFence createFence(optional GPUFenceDescriptor descriptor = {});
// void signal(GPUFence fence, unsigned long long signalValue);
// void copyImageBitmapToTexture(
// GPUImageBitmapCopyView source,
// GPUTextureCopyView destination,
// GPUExtent3D copySize);
};
GPUQueue includes GPUObjectBase;

View file

@ -15,7 +15,7 @@ use smallvec::SmallVec;
#[derive(Debug, Deserialize, Serialize)] #[derive(Debug, Deserialize, Serialize)]
pub enum WebGPUResponse { pub enum WebGPUResponse {
RequestAdapter(String, WebGPUAdapter, WebGPU), RequestAdapter(String, WebGPUAdapter, WebGPU),
RequestDevice(WebGPUDevice, wgpu::instance::DeviceDescriptor), RequestDevice(WebGPUDevice, WebGPUQueue, wgpu::instance::DeviceDescriptor),
} }
pub type WebGPUResponseResult = Result<WebGPUResponse, String>; pub type WebGPUResponseResult = Result<WebGPUResponse, String>;
@ -102,6 +102,7 @@ pub enum WebGPURequest {
// TODO(zakorgy): Serialize CommandBufferDescriptor in wgpu-core // TODO(zakorgy): Serialize CommandBufferDescriptor in wgpu-core
// wgpu::command::CommandBufferDescriptor, // wgpu::command::CommandBufferDescriptor,
), ),
Submit(wgpu::id::QueueId, Vec<wgpu::id::CommandBufferId>),
} }
#[derive(Clone, Debug, Deserialize, Serialize)] #[derive(Clone, Debug, Deserialize, Serialize)]
@ -222,9 +223,12 @@ impl WGPU {
id id
)); ));
let device = WebGPUDevice(id); let device = WebGPUDevice(id);
// Note: (zakorgy) Note sure if sending the queue is needed at all,
// since wgpu-core uses the same id for the device and the queue
let queue = WebGPUQueue(id);
self.devices.push(device); self.devices.push(device);
if let Err(e) = if let Err(e) =
sender.send(Ok(WebGPUResponse::RequestDevice(device, descriptor))) sender.send(Ok(WebGPUResponse::RequestDevice(device, queue, descriptor)))
{ {
warn!( warn!(
"Failed to send response to WebGPURequest::RequestDevice ({})", "Failed to send response to WebGPURequest::RequestDevice ({})",
@ -407,6 +411,13 @@ impl WGPU {
) )
} }
}, },
WebGPURequest::Submit(queue_id, command_buffer_ids) => {
let global = &self.global;
let _ = gfx_select!(queue_id => global.queue_submit(
queue_id,
&command_buffer_ids
));
},
WebGPURequest::Exit(sender) => { WebGPURequest::Exit(sender) => {
self.deinit(); self.deinit();
if let Err(e) = sender.send(()) { if let Err(e) = sender.send(()) {
@ -444,3 +455,4 @@ webgpu_resource!(WebGPUPipelineLayout, wgpu::id::PipelineLayoutId);
webgpu_resource!(WebGPUShaderModule, wgpu::id::ShaderModuleId); webgpu_resource!(WebGPUShaderModule, wgpu::id::ShaderModuleId);
webgpu_resource!(WebGPUCommandEncoder, wgpu::id::CommandEncoderId); webgpu_resource!(WebGPUCommandEncoder, wgpu::id::CommandEncoderId);
webgpu_resource!(WebGPUCommandBuffer, wgpu::id::CommandBufferId); webgpu_resource!(WebGPUCommandBuffer, wgpu::id::CommandBufferId);
webgpu_resource!(WebGPUQueue, wgpu::id::QueueId);