diff options
Diffstat (limited to 'components/script/dom/webgpu/gpuqueue.rs')
-rw-r--r-- | components/script/dom/webgpu/gpuqueue.rs | 225 |
1 files changed, 225 insertions, 0 deletions
diff --git a/components/script/dom/webgpu/gpuqueue.rs b/components/script/dom/webgpu/gpuqueue.rs new file mode 100644 index 00000000000..403628737d3 --- /dev/null +++ b/components/script/dom/webgpu/gpuqueue.rs @@ -0,0 +1,225 @@ +/* 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 std::rc::Rc; + +use dom_struct::dom_struct; +use ipc_channel::ipc::IpcSharedMemory; +use webgpu::{wgt, WebGPU, WebGPUQueue, WebGPURequest, WebGPUResponse}; + +use super::gpu::{response_async, AsyncWGPUListener}; +use crate::dom::bindings::cell::DomRefCell; +use crate::dom::bindings::codegen::Bindings::WebGPUBinding::{ + GPUExtent3D, GPUImageCopyTexture, GPUImageDataLayout, GPUQueueMethods, GPUSize64, +}; +use crate::dom::bindings::codegen::UnionTypes::ArrayBufferViewOrArrayBuffer as BufferSource; +use crate::dom::bindings::error::{Error, Fallible}; +use crate::dom::bindings::reflector::{reflect_dom_object, DomObject, Reflector}; +use crate::dom::bindings::root::{Dom, DomRoot}; +use crate::dom::bindings::str::USVString; +use crate::dom::globalscope::GlobalScope; +use crate::dom::gpubuffer::GPUBuffer; +use crate::dom::gpucommandbuffer::GPUCommandBuffer; +use crate::dom::gpudevice::GPUDevice; +use crate::dom::promise::Promise; +use crate::script_runtime::CanGc; + +#[dom_struct] +pub struct GPUQueue { + reflector_: Reflector, + #[ignore_malloc_size_of = "defined in webgpu"] + #[no_trace] + channel: WebGPU, + device: DomRefCell<Option<Dom<GPUDevice>>>, + label: DomRefCell<USVString>, + #[no_trace] + queue: WebGPUQueue, +} + +impl GPUQueue { + fn new_inherited(channel: WebGPU, queue: WebGPUQueue) -> Self { + GPUQueue { + channel, + reflector_: Reflector::new(), + device: DomRefCell::new(None), + label: DomRefCell::new(USVString::default()), + queue, + } + } + + pub fn new(global: &GlobalScope, channel: WebGPU, queue: WebGPUQueue) -> DomRoot<Self> { + reflect_dom_object(Box::new(GPUQueue::new_inherited(channel, queue)), global) + } +} + +impl GPUQueue { + pub fn set_device(&self, device: &GPUDevice) { + *self.device.borrow_mut() = Some(Dom::from_ref(device)); + } + + pub fn id(&self) -> WebGPUQueue { + self.queue + } +} + +impl GPUQueueMethods<crate::DomTypeHolder> for GPUQueue { + /// <https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label> + fn Label(&self) -> USVString { + self.label.borrow().clone() + } + + /// <https://gpuweb.github.io/gpuweb/#dom-gpuobjectbase-label> + fn SetLabel(&self, value: USVString) { + *self.label.borrow_mut() = value; + } + + /// <https://gpuweb.github.io/gpuweb/#dom-gpuqueue-submit> + fn Submit(&self, command_buffers: Vec<DomRoot<GPUCommandBuffer>>) { + let command_buffers = command_buffers.iter().map(|cb| cb.id().0).collect(); + self.channel + .0 + .send(WebGPURequest::Submit { + device_id: self.device.borrow().as_ref().unwrap().id().0, + queue_id: self.queue.0, + command_buffers, + }) + .unwrap(); + } + + /// <https://gpuweb.github.io/gpuweb/#dom-gpuqueue-writebuffer> + #[allow(unsafe_code)] + fn WriteBuffer( + &self, + buffer: &GPUBuffer, + buffer_offset: GPUSize64, + data: BufferSource, + data_offset: GPUSize64, + size: Option<GPUSize64>, + ) -> Fallible<()> { + // Step 1 + let sizeof_element: usize = match data { + BufferSource::ArrayBufferView(ref d) => d.get_array_type().byte_size().unwrap_or(1), + BufferSource::ArrayBuffer(_) => 1, + }; + let data = match data { + BufferSource::ArrayBufferView(d) => d.to_vec(), + BufferSource::ArrayBuffer(d) => d.to_vec(), + }; + // Step 2 + let data_size: usize = data.len() / sizeof_element; + debug_assert_eq!(data.len() % sizeof_element, 0); + // Step 3 + let content_size = if let Some(s) = size { + s + } else { + (data_size as GPUSize64) + .checked_sub(data_offset) + .ok_or(Error::Operation)? + }; + + // Step 4 + let valid = data_offset + content_size <= data_size as u64 && + content_size * sizeof_element as u64 % wgt::COPY_BUFFER_ALIGNMENT == 0; + if !valid { + return Err(Error::Operation); + } + + // Step 5&6 + let contents = IpcSharedMemory::from_bytes( + &data[(data_offset as usize) * sizeof_element.. + ((data_offset + content_size) as usize) * sizeof_element], + ); + if let Err(e) = self.channel.0.send(WebGPURequest::WriteBuffer { + device_id: self.device.borrow().as_ref().unwrap().id().0, + queue_id: self.queue.0, + buffer_id: buffer.id().0, + buffer_offset, + data: contents, + }) { + warn!("Failed to send WriteBuffer({:?}) ({})", buffer.id(), e); + return Err(Error::Operation); + } + + Ok(()) + } + + /// <https://gpuweb.github.io/gpuweb/#dom-gpuqueue-writetexture> + fn WriteTexture( + &self, + destination: &GPUImageCopyTexture, + data: BufferSource, + data_layout: &GPUImageDataLayout, + size: GPUExtent3D, + ) -> Fallible<()> { + let (bytes, len) = match data { + BufferSource::ArrayBufferView(d) => (d.to_vec(), d.len() as u64), + BufferSource::ArrayBuffer(d) => (d.to_vec(), d.len() as u64), + }; + let valid = data_layout.offset <= len; + + if !valid { + return Err(Error::Operation); + } + + let texture_cv = destination.try_into()?; + let texture_layout = data_layout.into(); + let write_size = (&size).try_into()?; + let final_data = IpcSharedMemory::from_bytes(&bytes); + + if let Err(e) = self.channel.0.send(WebGPURequest::WriteTexture { + device_id: self.device.borrow().as_ref().unwrap().id().0, + queue_id: self.queue.0, + texture_cv, + data_layout: texture_layout, + size: write_size, + data: final_data, + }) { + warn!( + "Failed to send WriteTexture({:?}) ({})", + destination.texture.id().0, + e + ); + return Err(Error::Operation); + } + + Ok(()) + } + + /// <https://gpuweb.github.io/gpuweb/#dom-gpuqueue-onsubmittedworkdone> + fn OnSubmittedWorkDone(&self, can_gc: CanGc) -> Rc<Promise> { + let global = self.global(); + let promise = Promise::new(&global, can_gc); + let sender = response_async(&promise, self); + if let Err(e) = self + .channel + .0 + .send(WebGPURequest::QueueOnSubmittedWorkDone { + sender, + queue_id: self.queue.0, + }) + { + warn!("QueueOnSubmittedWorkDone failed with {e}") + } + promise + } +} + +impl AsyncWGPUListener for GPUQueue { + fn handle_response( + &self, + response: webgpu::WebGPUResponse, + promise: &Rc<Promise>, + _can_gc: CanGc, + ) { + match response { + WebGPUResponse::SubmittedWorkDone => { + promise.resolve_native(&()); + }, + _ => { + warn!("GPUQueue received wrong WebGPUResponse"); + promise.reject_error(Error::Operation); + }, + } + } +} |