/* 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 dom_struct::dom_struct; use ipc_channel::ipc::IpcSender; use webgpu::wgc::pipeline::RenderPipelineDescriptor; use webgpu::{WebGPU, WebGPUBindGroupLayout, WebGPURenderPipeline, WebGPURequest, WebGPUResponse}; use crate::dom::bindings::cell::DomRefCell; use crate::dom::bindings::codegen::Bindings::WebGPUBinding::GPURenderPipelineMethods; use crate::dom::bindings::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::gpubindgrouplayout::GPUBindGroupLayout; use crate::dom::gpudevice::{GPUDevice, PipelineLayout}; #[dom_struct] pub struct GPURenderPipeline { reflector_: Reflector, #[ignore_malloc_size_of = "channels are hard"] #[no_trace] channel: WebGPU, label: DomRefCell, #[no_trace] render_pipeline: WebGPURenderPipeline, device: Dom, } impl GPURenderPipeline { fn new_inherited( render_pipeline: WebGPURenderPipeline, label: USVString, device: &GPUDevice, ) -> Self { Self { reflector_: Reflector::new(), channel: device.channel(), label: DomRefCell::new(label), render_pipeline, device: Dom::from_ref(device), } } pub fn new( global: &GlobalScope, render_pipeline: WebGPURenderPipeline, label: USVString, device: &GPUDevice, ) -> DomRoot { reflect_dom_object( Box::new(GPURenderPipeline::new_inherited( render_pipeline, label, device, )), global, ) } } impl GPURenderPipeline { pub fn id(&self) -> WebGPURenderPipeline { self.render_pipeline } /// pub fn create( device: &GPUDevice, pipeline_layout: PipelineLayout, descriptor: RenderPipelineDescriptor<'static>, async_sender: Option>, ) -> Fallible { let render_pipeline_id = device.global().wgpu_id_hub().create_render_pipeline_id(); device .channel() .0 .send(WebGPURequest::CreateRenderPipeline { device_id: device.id().0, render_pipeline_id, descriptor, implicit_ids: pipeline_layout.implicit(), async_sender, }) .expect("Failed to create WebGPU render pipeline"); Ok(WebGPURenderPipeline(render_pipeline_id)) } } impl GPURenderPipelineMethods for GPURenderPipeline { /// fn Label(&self) -> USVString { self.label.borrow().clone() } /// fn SetLabel(&self, value: USVString) { *self.label.borrow_mut() = value; } /// fn GetBindGroupLayout(&self, index: u32) -> Fallible> { let id = self.global().wgpu_id_hub().create_bind_group_layout_id(); if let Err(e) = self .channel .0 .send(WebGPURequest::RenderGetBindGroupLayout { device_id: self.device.id().0, pipeline_id: self.render_pipeline.0, index, id, }) { warn!("Failed to send WebGPURequest::RenderGetBindGroupLayout {e:?}"); } Ok(GPUBindGroupLayout::new( &self.global(), self.channel.clone(), WebGPUBindGroupLayout(id), USVString::default(), )) } } impl Drop for GPURenderPipeline { fn drop(&mut self) { if let Err(e) = self .channel .0 .send(WebGPURequest::DropRenderPipeline(self.render_pipeline.0)) { warn!( "Failed to send WebGPURequest::DropRenderPipeline({:?}) ({})", self.render_pipeline.0, e ); }; } }